1 /* kill -- send a signal to a process
2 Copyright (C) 2002-2023 Free Software Foundation, Inc.
4 This program is free software: you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation, either version 3 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program. If not, see <https://www.gnu.org/licenses/>. */
17 /* Written by Paul Eggert. */
20 #include <stdckdint.h>
23 #include <sys/types.h>
28 #include "operand2sig.h"
31 /* The official name of this program (e.g., no 'g' prefix). */
32 #define PROGRAM_NAME "kill"
34 #define AUTHORS proper_name ("Paul Eggert")
36 #if ! (HAVE_DECL_STRSIGNAL || defined strsignal)
37 # if ! (HAVE_DECL_SYS_SIGLIST || defined sys_siglist)
38 # if HAVE_DECL__SYS_SIGLIST || defined _sys_siglist
39 # define sys_siglist _sys_siglist
40 # elif HAVE_DECL___SYS_SIGLIST || defined __sys_siglist
41 # define sys_siglist __sys_siglist
44 # if HAVE_DECL_SYS_SIGLIST || defined sys_siglist
45 # define strsignal(signum) (0 <= (signum) && (signum) <= SIGNUM_BOUND \
46 ? sys_siglist[signum] \
50 # define strsignal(signum) 0
54 static char const short_options
[] =
55 "0::1::2::3::4::5::6::7::8::9::"
56 "A::B::C::D::E::F::G::H::I::J::K::M::"
57 "N::O::P::Q::R::S::T::U::V::W::X::Y::Z::"
60 static struct option
const long_options
[] =
62 {"list", no_argument
, nullptr, 'l'},
63 {"signal", required_argument
, nullptr, 's'},
64 {"table", no_argument
, nullptr, 't'},
65 {GETOPT_HELP_OPTION_DECL
},
66 {GETOPT_VERSION_OPTION_DECL
},
67 {nullptr, 0, nullptr, 0}
73 if (status
!= EXIT_SUCCESS
)
78 Usage: %s [-s SIGNAL | -SIGNAL] PID...\n\
79 or: %s -l [SIGNAL]...\n\
80 or: %s -t [SIGNAL]...\n\
82 program_name
, program_name
, program_name
);
84 Send signals to processes, or list signals.\n\
87 emit_mandatory_arg_note ();
90 -s, --signal=SIGNAL, -SIGNAL\n\
91 specify the name or number of the signal to be sent\n\
92 -l, --list list signal names, or convert signal names to/from numbers\n\
93 -t, --table print a table of signal information\n\
95 fputs (HELP_OPTION_DESCRIPTION
, stdout
);
96 fputs (VERSION_OPTION_DESCRIPTION
, stdout
);
98 SIGNAL may be a signal name like 'HUP', or a signal number like '1',\n\
99 or the exit status of a process terminated by a signal.\n\
100 PID is an integer; if negative it identifies a process group.\n\
102 printf (USAGE_BUILTIN_WARNING
, PROGRAM_NAME
);
103 emit_ancillary_info (PROGRAM_NAME
);
108 /* Print a row of 'kill -t' output. NUM_WIDTH is the maximum signal
109 number width, and SIGNUM is the signal number to print. The
110 maximum name width is NAME_WIDTH, and SIGNAME is the name to print. */
113 print_table_row (int num_width
, int signum
,
114 int name_width
, char const *signame
)
116 char const *description
= strsignal (signum
);
117 printf ("%*d %-*s %s\n", num_width
, signum
, name_width
, signame
,
118 description
? description
: "?");
121 /* Print a list of signal names. If TABLE, print a table.
122 Print the names specified by ARGV if nonzero; otherwise,
123 print all known names. Return a suitable exit status. */
126 list_signals (bool table
, char *const *argv
)
129 int status
= EXIT_SUCCESS
;
130 char signame
[SIG2STR_MAX
];
134 unsigned int name_width
= 0;
136 /* Compute the maximum width of a signal number. */
137 unsigned int num_width
= 1;
138 for (signum
= 1; signum
<= SIGNUM_BOUND
/ 10; signum
*= 10)
141 /* Compute the maximum width of a signal name. */
142 for (signum
= 1; signum
<= SIGNUM_BOUND
; signum
++)
143 if (sig2str (signum
, signame
) == 0)
145 size_t len
= strlen (signame
);
146 if (name_width
< len
)
151 for (; *argv
; argv
++)
153 signum
= operand2sig (*argv
, signame
);
155 status
= EXIT_FAILURE
;
157 print_table_row (num_width
, signum
, name_width
, signame
);
160 for (signum
= 1; signum
<= SIGNUM_BOUND
; signum
++)
161 if (sig2str (signum
, signame
) == 0)
162 print_table_row (num_width
, signum
, name_width
, signame
);
167 for (; *argv
; argv
++)
169 signum
= operand2sig (*argv
, signame
);
171 status
= EXIT_FAILURE
;
174 if (ISDIGIT (**argv
))
177 printf ("%d\n", signum
);
181 for (signum
= 1; signum
<= SIGNUM_BOUND
; signum
++)
182 if (sig2str (signum
, signame
) == 0)
189 /* Send signal SIGNUM to all the processes or process groups specified
190 by ARGV. Return a suitable exit status. */
193 send_signals (int signum
, char *const *argv
)
195 int status
= EXIT_SUCCESS
;
196 char const *arg
= *argv
;
201 intmax_t n
= (errno
= 0, strtoimax (arg
, &endp
, 10));
204 if (errno
== ERANGE
|| ckd_add (&pid
, n
, 0)
205 || arg
== endp
|| *endp
)
207 error (0, 0, _("%s: invalid process id"), quote (arg
));
208 status
= EXIT_FAILURE
;
210 else if (kill (pid
, signum
) != 0)
212 error (0, errno
, "%s", quote (arg
));
213 status
= EXIT_FAILURE
;
216 while ((arg
= *++argv
));
222 main (int argc
, char **argv
)
228 char signame
[SIG2STR_MAX
];
230 initialize_main (&argc
, &argv
);
231 set_program_name (argv
[0]);
232 setlocale (LC_ALL
, "");
233 bindtextdomain (PACKAGE
, LOCALEDIR
);
234 textdomain (PACKAGE
);
236 atexit (close_stdout
);
238 while ((optc
= getopt_long (argc
, argv
, short_options
, long_options
, nullptr))
242 case '0': case '1': case '2': case '3': case '4':
243 case '5': case '6': case '7': case '8': case '9':
246 /* This option is actually a process-id. */
248 goto no_more_options
;
251 case 'A': case 'B': case 'C': case 'D': case 'E':
252 case 'F': case 'G': case 'H': case 'I': case 'J':
253 case 'K': /*case 'L':*/ case 'M': case 'N': case 'O':
254 case 'P': case 'Q': case 'R': case 'S': case 'T':
255 case 'U': case 'V': case 'W': case 'X': case 'Y':
258 optarg
= argv
[optind
- 1] + strlen (argv
[optind
- 1]);
259 if (optarg
!= argv
[optind
- 1] + 2)
261 error (0, 0, _("invalid option -- %c"), optc
);
262 usage (EXIT_FAILURE
);
266 case 'n': /* -n is not documented, but is for Bash compatibility. */
270 error (0, 0, _("%s: multiple signals specified"), quote (optarg
));
271 usage (EXIT_FAILURE
);
273 signum
= operand2sig (optarg
, signame
);
275 usage (EXIT_FAILURE
);
278 case 'L': /* -L is not documented, but is for procps compatibility. */
285 error (0, 0, _("multiple -l or -t options specified"));
286 usage (EXIT_FAILURE
);
291 case_GETOPT_HELP_CHAR
;
292 case_GETOPT_VERSION_CHAR (PROGRAM_NAME
, AUTHORS
);
294 usage (EXIT_FAILURE
);
302 error (0, 0, _("cannot combine signal with -l or -t"));
303 usage (EXIT_FAILURE
);
306 if ( ! list
&& argc
<= optind
)
308 error (0, 0, _("no process ID specified"));
309 usage (EXIT_FAILURE
);
313 ? list_signals (table
, optind
< argc
? argv
+ optind
: nullptr)
314 : send_signals (signum
, argv
+ optind
));