Sync usage with man page.
[netbsd-mini2440.git] / external / cddl / osnet / dist / lib / libuutil / common / uu_misc.c
blob74ec177c11b79b57d7e380960e2d4f390b6277fc
1 /*
2 * CDDL HEADER START
4 * The contents of this file are subject to the terms of the
5 * Common Development and Distribution License (the "License").
6 * You may not use this file except in compliance with the License.
8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9 * or http://www.opensolaris.org/os/licensing.
10 * See the License for the specific language governing permissions
11 * and limitations under the License.
13 * When distributing Covered Code, include this CDDL HEADER in each
14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15 * If applicable, add the following below this CDDL HEADER, with the
16 * fields enclosed by brackets "[]" replaced with your own identifying
17 * information: Portions Copyright [yyyy] [name of copyright owner]
19 * CDDL HEADER END
23 * Copyright 2007 Sun Microsystems, Inc. All rights reserved.
24 * Use is subject to license terms.
27 #pragma ident "%Z%%M% %I% %E% SMI"
29 #include "libuutil_common.h"
31 #include <assert.h>
32 #include <errno.h>
33 #include <libintl.h>
34 #include <pthread.h>
35 #include <stdarg.h>
36 #include <stdio.h>
37 #include <stdlib.h>
38 #include <string.h>
39 #include <sys/debug.h>
40 #include <thread.h>
41 #include <unistd.h>
43 #if !defined(TEXT_DOMAIN)
44 #define TEXT_DOMAIN "SYS_TEST"
45 #endif
48 * All of the old code under !defined(PTHREAD_ONCE_KEY_NP)
49 * is here to enable the building of a native version of
50 * libuutil.so when the build machine has not yet been upgraded
51 * to a version of libc that provides pthread_key_create_once_np().
52 * It should all be deleted when solaris_nevada ships.
53 * The code is not MT-safe in a relaxed memory model.
56 #if defined(PTHREAD_ONCE_KEY_NP)
57 static pthread_key_t uu_error_key = PTHREAD_ONCE_KEY_NP;
58 #else /* PTHREAD_ONCE_KEY_NP */
59 static pthread_key_t uu_error_key = 0;
60 static pthread_mutex_t uu_key_lock = PTHREAD_MUTEX_INITIALIZER;
61 #endif /* PTHREAD_ONCE_KEY_NP */
63 static int uu_error_key_setup = 0;
65 static pthread_mutex_t uu_panic_lock = PTHREAD_MUTEX_INITIALIZER;
66 /* LINTED static unused */
67 static const char *uu_panic_format;
68 /* LINTED static unused */
69 static va_list uu_panic_args;
70 static pthread_t uu_panic_thread;
72 static uint32_t _uu_main_error;
74 void
75 uu_set_error(uint_t code)
77 if (thr_main() != 0) {
78 _uu_main_error = code;
79 return;
81 #if defined(PTHREAD_ONCE_KEY_NP)
82 if (pthread_key_create_once_np(&uu_error_key, NULL) != 0)
83 uu_error_key_setup = -1;
84 else
85 uu_error_key_setup = 1;
86 #else /* PTHREAD_ONCE_KEY_NP */
87 if (uu_error_key_setup == 0) {
88 (void) pthread_mutex_lock(&uu_key_lock);
89 if (uu_error_key_setup == 0) {
90 if (pthread_key_create(&uu_error_key, NULL) != 0)
91 uu_error_key_setup = -1;
92 else
93 uu_error_key_setup = 1;
95 (void) pthread_mutex_unlock(&uu_key_lock);
97 #endif /* PTHREAD_ONCE_KEY_NP */
98 if (uu_error_key_setup > 0)
99 (void) pthread_setspecific(uu_error_key,
100 (void *)(uintptr_t)code);
103 uint32_t
104 uu_error(void)
106 if (thr_main() != 0)
107 return (_uu_main_error);
109 if (uu_error_key_setup < 0) /* can't happen? */
110 return (UU_ERROR_UNKNOWN);
113 * Because UU_ERROR_NONE == 0, if uu_set_error() was
114 * never called, then this will return UU_ERROR_NONE:
116 return ((uint32_t)(uintptr_t)pthread_getspecific(uu_error_key));
119 const char *
120 uu_strerror(uint32_t code)
122 const char *str;
124 switch (code) {
125 case UU_ERROR_NONE:
126 str = dgettext(TEXT_DOMAIN, "No error");
127 break;
129 case UU_ERROR_INVALID_ARGUMENT:
130 str = dgettext(TEXT_DOMAIN, "Invalid argument");
131 break;
133 case UU_ERROR_UNKNOWN_FLAG:
134 str = dgettext(TEXT_DOMAIN, "Unknown flag passed");
135 break;
137 case UU_ERROR_NO_MEMORY:
138 str = dgettext(TEXT_DOMAIN, "Out of memory");
139 break;
141 case UU_ERROR_CALLBACK_FAILED:
142 str = dgettext(TEXT_DOMAIN, "Callback-initiated failure");
143 break;
145 case UU_ERROR_NOT_SUPPORTED:
146 str = dgettext(TEXT_DOMAIN, "Operation not supported");
147 break;
149 case UU_ERROR_EMPTY:
150 str = dgettext(TEXT_DOMAIN, "No value provided");
151 break;
153 case UU_ERROR_UNDERFLOW:
154 str = dgettext(TEXT_DOMAIN, "Value too small");
155 break;
157 case UU_ERROR_OVERFLOW:
158 str = dgettext(TEXT_DOMAIN, "Value too large");
159 break;
161 case UU_ERROR_INVALID_CHAR:
162 str = dgettext(TEXT_DOMAIN,
163 "Value contains unexpected character");
164 break;
166 case UU_ERROR_INVALID_DIGIT:
167 str = dgettext(TEXT_DOMAIN,
168 "Value contains digit not in base");
169 break;
171 case UU_ERROR_SYSTEM:
172 str = dgettext(TEXT_DOMAIN, "Underlying system error");
173 break;
175 case UU_ERROR_UNKNOWN:
176 str = dgettext(TEXT_DOMAIN, "Error status not known");
177 break;
179 default:
180 errno = ESRCH;
181 str = NULL;
182 break;
184 return (str);
187 void
188 uu_panic(const char *format, ...)
190 va_list args;
192 va_start(args, format);
194 (void) pthread_mutex_lock(&uu_panic_lock);
195 if (uu_panic_thread == 0) {
196 uu_panic_thread = pthread_self();
197 uu_panic_format = format;
198 va_copy(uu_panic_args, args);
200 (void) pthread_mutex_unlock(&uu_panic_lock);
202 (void) vfprintf(stderr, format, args);
204 if (uu_panic_thread == pthread_self())
205 abort();
206 else
207 for (;;)
208 (void) pause();
212 assfail(const char *astring, const char *file, int line)
214 __assert(astring, file, line);
215 /*NOTREACHED*/
216 return (0);
219 static void
220 uu_lockup(void)
222 (void) pthread_mutex_lock(&uu_panic_lock);
223 #if !defined(PTHREAD_ONCE_KEY_NP)
224 (void) pthread_mutex_lock(&uu_key_lock);
225 #endif
226 uu_avl_lockup();
227 uu_list_lockup();
230 static void
231 uu_release(void)
233 (void) pthread_mutex_unlock(&uu_panic_lock);
234 #if !defined(PTHREAD_ONCE_KEY_NP)
235 (void) pthread_mutex_unlock(&uu_key_lock);
236 #endif
237 uu_avl_release();
238 uu_list_release();
241 static void
242 uu_release_child(void)
244 uu_panic_format = NULL;
245 uu_panic_thread = 0;
247 uu_release();
250 #pragma init(uu_init)
251 static void
252 uu_init(void)
254 (void) pthread_atfork(uu_lockup, uu_release, uu_release_child);