kill[all[5]],pkill: more correct, and smaller, SIGRTMIN/MAX code
[oweals/busybox.git] / libbb / u_signal_names.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Signal name/number conversion routines.
4  *
5  * Copyright 2006 Rob Landley <rob@landley.net>
6  *
7  * Licensed under GPLv2 or later, see file LICENSE in this source tree.
8  */
9
10 //config:config FEATURE_RTMINMAX
11 //config:       bool "Support RTMIN[+n] and RTMAX[-n] signal names"
12 //config:       default y
13 //config:       help
14 //config:         Support RTMIN[+n] and RTMAX[-n] signal names
15 //config:         in kill, killall etc. This costs ~250 bytes.
16
17 #include "libbb.h"
18
19 /* Believe it or not, but some arches have more than 32 SIGs!
20  * HPPA: SIGSTKFLT == 36. */
21
22 static const char signals[][7] = {
23         // SUSv3 says kill must support these, and specifies the numerical values,
24         // http://www.opengroup.org/onlinepubs/009695399/utilities/kill.html
25         // {0, "EXIT"}, {1, "HUP"}, {2, "INT"}, {3, "QUIT"},
26         // {6, "ABRT"}, {9, "KILL"}, {14, "ALRM"}, {15, "TERM"}
27         // And Posix adds the following:
28         // {SIGILL, "ILL"}, {SIGTRAP, "TRAP"}, {SIGFPE, "FPE"}, {SIGUSR1, "USR1"},
29         // {SIGSEGV, "SEGV"}, {SIGUSR2, "USR2"}, {SIGPIPE, "PIPE"}, {SIGCHLD, "CHLD"},
30         // {SIGCONT, "CONT"}, {SIGSTOP, "STOP"}, {SIGTSTP, "TSTP"}, {SIGTTIN, "TTIN"},
31         // {SIGTTOU, "TTOU"}
32
33         [0] = "EXIT",
34 #ifdef SIGHUP
35         [SIGHUP   ] = "HUP",
36 #endif
37 #ifdef SIGINT
38         [SIGINT   ] = "INT",
39 #endif
40 #ifdef SIGQUIT
41         [SIGQUIT  ] = "QUIT",
42 #endif
43 #ifdef SIGILL
44         [SIGILL   ] = "ILL",
45 #endif
46 #ifdef SIGTRAP
47         [SIGTRAP  ] = "TRAP",
48 #endif
49 #ifdef SIGABRT
50         [SIGABRT  ] = "ABRT",
51 #endif
52 #ifdef SIGBUS
53         [SIGBUS   ] = "BUS",
54 #endif
55 #ifdef SIGFPE
56         [SIGFPE   ] = "FPE",
57 #endif
58 #ifdef SIGKILL
59         [SIGKILL  ] = "KILL",
60 #endif
61 #ifdef SIGUSR1
62         [SIGUSR1  ] = "USR1",
63 #endif
64 #ifdef SIGSEGV
65         [SIGSEGV  ] = "SEGV",
66 #endif
67 #ifdef SIGUSR2
68         [SIGUSR2  ] = "USR2",
69 #endif
70 #ifdef SIGPIPE
71         [SIGPIPE  ] = "PIPE",
72 #endif
73 #ifdef SIGALRM
74         [SIGALRM  ] = "ALRM",
75 #endif
76 #ifdef SIGTERM
77         [SIGTERM  ] = "TERM",
78 #endif
79 #ifdef SIGSTKFLT
80         [SIGSTKFLT] = "STKFLT",
81 #endif
82 #ifdef SIGCHLD
83         [SIGCHLD  ] = "CHLD",
84 #endif
85 #ifdef SIGCONT
86         [SIGCONT  ] = "CONT",
87 #endif
88 #ifdef SIGSTOP
89         [SIGSTOP  ] = "STOP",
90 #endif
91 #ifdef SIGTSTP
92         [SIGTSTP  ] = "TSTP",
93 #endif
94 #ifdef SIGTTIN
95         [SIGTTIN  ] = "TTIN",
96 #endif
97 #ifdef SIGTTOU
98         [SIGTTOU  ] = "TTOU",
99 #endif
100 #ifdef SIGURG
101         [SIGURG   ] = "URG",
102 #endif
103 #ifdef SIGXCPU
104         [SIGXCPU  ] = "XCPU",
105 #endif
106 #ifdef SIGXFSZ
107         [SIGXFSZ  ] = "XFSZ",
108 #endif
109 #ifdef SIGVTALRM
110         [SIGVTALRM] = "VTALRM",
111 #endif
112 #ifdef SIGPROF
113         [SIGPROF  ] = "PROF",
114 #endif
115 #ifdef SIGWINCH
116         [SIGWINCH ] = "WINCH",
117 #endif
118 #ifdef SIGPOLL
119         [SIGPOLL  ] = "POLL",
120 #endif
121 #ifdef SIGPWR
122         [SIGPWR   ] = "PWR",
123 #endif
124 #ifdef SIGSYS
125         [SIGSYS   ] = "SYS",
126 #endif
127 };
128
129 // Convert signal name to number.
130
131 int FAST_FUNC get_signum(const char *name)
132 {
133         unsigned i;
134
135         i = bb_strtou(name, NULL, 10);
136         if (!errno)
137                 return i;
138         if (strncasecmp(name, "SIG", 3) == 0)
139                 name += 3;
140         for (i = 0; i < ARRAY_SIZE(signals); i++)
141                 if (strcasecmp(name, signals[i]) == 0)
142                         return i;
143
144 #if ENABLE_DESKTOP
145 # if defined(SIGIOT) || defined(SIGIO)
146         /* SIGIO[T] are aliased to other names,
147          * thus cannot be stored in the signals[] array.
148          * Need special code to recognize them */
149         if ((name[0] | 0x20) == 'i' && (name[1] | 0x20) == 'o') {
150 #  ifdef SIGIO
151                 if (!name[2])
152                         return SIGIO;
153 #  endif
154 #  ifdef SIGIOT
155                 if ((name[2] | 0x20) == 't' && !name[3])
156                         return SIGIOT;
157 #  endif
158         }
159 # endif
160 #endif
161
162 #if ENABLE_FEATURE_RTMINMAX
163 # if defined(SIGRTMIN) && defined(SIGRTMAX)
164 /* libc may use some rt sigs for pthreads and therefore "remap" SIGRTMIN/MAX,
165  * but we want to use "raw" SIGRTMIN/MAX. Underscored names, if exist, provide
166  * them. If they don't exist, fall back to non-underscored ones: */
167 #  if !defined(__SIGRTMIN)
168 #   define __SIGRTMIN SIGRTMIN
169 #  endif
170 #  if !defined(__SIGRTMAX)
171 #   define __SIGRTMAX SIGRTMAX
172 #  endif
173         if (strncasecmp(name, "RTMIN", 5) == 0) {
174                 if (!name[5])
175                         return __SIGRTMIN;
176                 if (name[5] == '+') {
177                         i = bb_strtou(name + 6, NULL, 10);
178                         if (!errno && i <= __SIGRTMAX - __SIGRTMIN)
179                                 return __SIGRTMIN + i;
180                 }
181         }
182         else if (strncasecmp(name, "RTMAX", 5) == 0) {
183                 if (!name[5])
184                         return __SIGRTMAX;
185                 if (name[5] == '-') {
186                         i = bb_strtou(name + 6, NULL, 10);
187                         if (!errno && i <= __SIGRTMAX - __SIGRTMIN)
188                                 return __SIGRTMAX - i;
189                 }
190         }
191 # endif
192 #endif
193
194         return -1;
195 }
196
197 // Convert signal number to name
198
199 const char* FAST_FUNC get_signame(int number)
200 {
201         if ((unsigned)number < ARRAY_SIZE(signals)) {
202                 if (signals[number][0]) /* if it's not an empty str */
203                         return signals[number];
204         }
205
206         return itoa(number);
207 }
208
209
210 // Print the whole signal list
211
212 void FAST_FUNC print_signames(void)
213 {
214         unsigned signo;
215
216         for (signo = 1; signo < ARRAY_SIZE(signals); signo++) {
217                 const char *name = signals[signo];
218                 if (name[0])
219                         puts(name);
220         }
221 }