64da0d1ebc
1) Refactor the manpage with num-options, optimize wording to be more concise and to the point, pid also specifies process groups. 2) Make int sig const. 3) Remove prototypes. 4) /* not reached */ consistency. 5) Refactor usage() with eprintf. 6) Refactor arg-parser with a switch, use estrtonum 7) Use return instead of exit() in main() 8) argc-argv-correctness.
131 lines
2.3 KiB
C
131 lines
2.3 KiB
C
/* See LICENSE file for copyright and license details. */
|
|
#include <sys/wait.h>
|
|
|
|
#include <ctype.h>
|
|
#include <limits.h>
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#include "util.h"
|
|
|
|
struct {
|
|
const char *name;
|
|
const int sig;
|
|
} sigs[] = {
|
|
{ "0", 0 },
|
|
#define SIG(n) { #n, SIG##n }
|
|
SIG(ABRT), SIG(ALRM), SIG(BUS), SIG(CHLD), SIG(CONT), SIG(FPE), SIG(HUP),
|
|
SIG(ILL), SIG(INT), SIG(KILL), SIG(PIPE), SIG(QUIT), SIG(SEGV), SIG(STOP),
|
|
SIG(TERM), SIG(TSTP), SIG(TTIN), SIG(TTOU), SIG(USR1), SIG(USR2), SIG(URG),
|
|
#undef SIG
|
|
};
|
|
|
|
const char *
|
|
sig2name(const int sig)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < LEN(sigs); i++)
|
|
if (sigs[i].sig == sig)
|
|
return sigs[i].name;
|
|
eprintf("%d: bad signal number\n", sig);
|
|
|
|
return NULL; /* not reached */
|
|
}
|
|
|
|
const int
|
|
name2sig(const char *name)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; i < LEN(sigs); i++)
|
|
if (!strcasecmp(sigs[i].name, name))
|
|
return sigs[i].sig;
|
|
eprintf("%s: bad signal name\n", name);
|
|
|
|
return -1; /* not reached */
|
|
}
|
|
|
|
static void
|
|
usage(void)
|
|
{
|
|
eprintf("usage: %s [-s signame | -num | -signame] pid ...\n"
|
|
" %s -l [num]\n", argv0, argv0);
|
|
}
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
pid_t pid;
|
|
size_t i;
|
|
int ret = 0, sig = SIGTERM;
|
|
|
|
argv0 = argv[0], argc--, argv++;
|
|
if (!argc)
|
|
usage();
|
|
|
|
if ((*argv)[0] == '-') {
|
|
switch ((*argv)[1]) {
|
|
case 'l':
|
|
if ((*argv)[2])
|
|
goto longopt;
|
|
argc--, argv++;
|
|
if (!argc) {
|
|
for (i = 0; i < LEN(sigs); i++)
|
|
puts(sigs[i].name);
|
|
return 0;
|
|
} else if (argc == 1) {
|
|
sig = estrtonum(*argv, 0, INT_MAX);
|
|
if (sig > 128)
|
|
sig = WTERMSIG(sig);
|
|
puts(sig2name(sig));
|
|
return 0;
|
|
} else {
|
|
usage();
|
|
}
|
|
break;
|
|
case 's':
|
|
if ((*argv)[2])
|
|
goto longopt;
|
|
argc--, argv++;
|
|
if (!argc)
|
|
usage();
|
|
sig = name2sig(*argv);
|
|
argc--, argv++;
|
|
break;
|
|
case '-':
|
|
if ((*argv)[2])
|
|
goto longopt;
|
|
argc--, argv++;
|
|
break;
|
|
default:
|
|
longopt:
|
|
/* XSI-extensions -argnum and -argname*/
|
|
if (isdigit((*argv)[1])) {
|
|
sig = estrtonum((*argv) + 1, 0, INT_MAX);
|
|
sig2name(sig);
|
|
} else {
|
|
sig = name2sig((*argv) + 1);
|
|
}
|
|
argc--, argv++;
|
|
}
|
|
}
|
|
|
|
if (argc && !strcmp(*argv, "--"))
|
|
argc--, argv++;
|
|
|
|
if (!argc)
|
|
usage();
|
|
|
|
for (; *argv; argc--, argv++) {
|
|
pid = estrtonum(*argv, INT_MIN, INT_MAX);
|
|
if (kill(pid, sig) < 0) {
|
|
weprintf("kill %d:", pid);
|
|
ret = 1;
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|