]> git.saurik.com Git - apple/libc.git/blame - string/FreeBSD/strsignal.c
Libc-1439.40.11.tar.gz
[apple/libc.git] / string / FreeBSD / strsignal.c
CommitLineData
9385eb3d
A
1/*
2 * Copyright (c) 1988, 1993
e9ce8d39
A
3 * The Regents of the University of California. All rights reserved.
4 *
e9ce8d39
A
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
e9ce8d39
A
13 * 4. Neither the name of the University nor the names of its contributors
14 * may be used to endorse or promote products derived from this software
15 * without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 */
29
30#if defined(LIBC_SCCS) && !defined(lint)
9385eb3d 31static char sccsid[] = "@(#)strerror.c 8.1 (Berkeley) 6/4/93";
e9ce8d39 32#endif /* LIBC_SCCS and not lint */
9385eb3d 33#include <sys/cdefs.h>
1f2f436a 34__FBSDID("$FreeBSD: src/lib/libc/string/strsignal.c,v 1.9 2010/01/24 10:35:26 ume Exp $");
e9ce8d39 35
1f2f436a
A
36#include "namespace.h"
37#if defined(NLS)
38#include <nl_types.h>
39#endif
40#include <limits.h>
41#include <errno.h>
42#include <stdlib.h>
9385eb3d
A
43#include <string.h>
44#include <signal.h>
1f2f436a
A
45#include "reentrant.h"
46#include "un-namespace.h"
e9ce8d39 47
1f2f436a
A
48#define UPREFIX "Unknown signal"
49
a9aaacca
A
50/*
51 * Define a buffer size big enough to describe a 64-bit signed integer
52 * converted to ASCII decimal (19 bytes), with an optional leading sign
53 * (1 byte); delimiter (": ", 2 bytes); and a trailing NUL (1 byte).
54 */
55#define TMPSIZE (19 + 1 + 2 + 1)
56#define EBUFSIZE NL_TEXTMAX * sizeof(char)
57
1f2f436a
A
58static once_t sig_init_once = ONCE_INITIALIZER;
59static thread_key_t sig_key;
60static int sig_keycreated = 0;
61
62static void
63sig_keycreate(void)
64{
65 sig_keycreated = (thr_keycreate(&sig_key, free) == 0);
66}
67
68static char *
69sig_tlsalloc(void)
70{
71 char *ebuf = NULL;
72
ad3c9f2a
A
73 if (thr_once(&sig_init_once, sig_keycreate) != 0 ||
74 !sig_keycreated)
75 goto thr_err;
76 if ((ebuf = thr_getspecific(sig_key)) == NULL) {
a9aaacca 77 if ((ebuf = malloc(EBUFSIZE)) == NULL)
ad3c9f2a
A
78 goto thr_err;
79 if (thr_setspecific(sig_key, ebuf) != 0) {
80 free(ebuf);
81 ebuf = NULL;
1f2f436a 82 goto thr_err;
1f2f436a
A
83 }
84 }
85thr_err:
1f2f436a
A
86 return (ebuf);
87}
88
a9aaacca
A
89int
90strsignal_r(int num, char *strsignalbuf, size_t buflen)
e9ce8d39 91{
a9aaacca
A
92 int retval = 0;
93 char tmp[TMPSIZE] = { 0 };
1f2f436a
A
94 size_t n;
95 int signum;
96 char *t, *p;
9385eb3d 97
a9aaacca
A
98 signum = num;
99 if (num < 0) {
100 signum = -signum;
101 }
5b2abdfb 102
a9aaacca
A
103 t = tmp;
104 do {
105 *t++ = "0123456789"[signum % 10];
106 } while (signum /= 10);
107 if (num < 0) {
108 *t++ = '-';
ad3c9f2a 109 }
a9aaacca 110 int suffixlen = strlen(tmp) + 2;
1f2f436a 111
ad3c9f2a 112 if (num > 0 && num < NSIG) {
a9aaacca 113 n = strlcpy(strsignalbuf,
1f2f436a 114 sys_siglist[num],
a9aaacca
A
115 buflen);
116 if (n >= (buflen - suffixlen)) {
117 retval = ERANGE;
118 }
1f2f436a 119 } else {
a9aaacca 120 n = strlcpy(strsignalbuf,
1f2f436a 121 UPREFIX,
a9aaacca
A
122 buflen);
123 retval = EINVAL;
1f2f436a
A
124 }
125
a9aaacca
A
126 if (n < (buflen - suffixlen)) {
127 p = (strsignalbuf + n);
128 *p++ = ':';
129 *p++ = ' ';
1f2f436a 130
a9aaacca
A
131 for (;;) {
132 *p++ = *--t;
133 if (t <= tmp)
134 break;
135 }
136 *p = '\0';
137 }
138
139 return retval;
140}
1f2f436a 141
a9aaacca
A
142/* XXX: negative 'num' ? (REGR) */
143char *
144strsignal(int num)
145{
146 char *ebuf;
147
148 ebuf = sig_tlsalloc();
149 if (ebuf == NULL) {
150 errno = ENOMEM;
151 return NULL;
152 }
1f2f436a 153
a9aaacca
A
154 if (strsignal_r(num, ebuf, EBUFSIZE)) {
155 errno = EINVAL;
9385eb3d 156 }
1f2f436a 157
9385eb3d 158 return (ebuf);
e9ce8d39 159}