2000-02-08 19:58:47 +00:00
|
|
|
/* vi: set sw=4 ts=4: */
|
1999-10-20 22:08:37 +00:00
|
|
|
/*
|
2000-04-05 01:00:52 +00:00
|
|
|
* Mini kill/killall implementation for busybox
|
1999-10-20 22:08:37 +00:00
|
|
|
*
|
|
|
|
* Copyright (C) 1995, 1996 by Bruce Perens <bruce@pixar.com>.
|
2004-03-15 08:29:22 +00:00
|
|
|
* Copyright (C) 1999-2004 by Erik Andersen <andersen@codepoet.org>
|
1999-10-20 22:08:37 +00:00
|
|
|
*
|
2006-06-02 20:56:16 +00:00
|
|
|
* Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
|
1999-10-20 22:08:37 +00:00
|
|
|
*/
|
|
|
|
|
2006-06-02 20:56:16 +00:00
|
|
|
#include "busybox.h"
|
1999-10-05 16:24:54 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
2000-03-07 07:41:42 +00:00
|
|
|
#include <errno.h>
|
1999-10-05 16:24:54 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
#include <signal.h>
|
1999-10-23 05:42:08 +00:00
|
|
|
#include <ctype.h>
|
2001-01-27 08:24:39 +00:00
|
|
|
#include <string.h>
|
1999-12-08 23:19:36 +00:00
|
|
|
#include <unistd.h>
|
|
|
|
|
2006-03-06 20:47:33 +00:00
|
|
|
int kill_main(int argc, char **argv)
|
1999-10-05 16:24:54 +00:00
|
|
|
{
|
2006-07-12 19:17:55 +00:00
|
|
|
int killall, signo = SIGTERM, errors = 0, quiet=0;
|
|
|
|
|
|
|
|
killall = (ENABLE_KILLALL && bb_applet_name[4]=='a') ? 1 : 0;
|
2000-02-08 19:58:47 +00:00
|
|
|
|
|
|
|
/* Parse any options */
|
2003-07-26 07:41:56 +00:00
|
|
|
if (argc < 2)
|
2003-03-19 09:13:01 +00:00
|
|
|
bb_show_usage();
|
2000-02-08 19:58:47 +00:00
|
|
|
|
2003-07-26 07:41:56 +00:00
|
|
|
if(argv[1][0] != '-'){
|
|
|
|
argv++;
|
|
|
|
argc--;
|
|
|
|
goto do_it_now;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The -l option, which prints out signal names. */
|
|
|
|
if(argv[1][1]=='l' && argv[1][2]=='\0'){
|
|
|
|
if(argc==2) {
|
|
|
|
/* Print the whole signal list */
|
|
|
|
int col = 0;
|
2006-07-12 19:17:55 +00:00
|
|
|
|
|
|
|
for(signo = 0;;) {
|
|
|
|
char *name = get_signame(++signo);
|
|
|
|
if (isdigit(*name)) break;
|
|
|
|
|
2003-07-26 07:41:56 +00:00
|
|
|
if (col > 60) {
|
|
|
|
printf("\n");
|
|
|
|
col = 0;
|
|
|
|
}
|
2006-07-12 19:17:55 +00:00
|
|
|
col += printf("%2d) %-16s", signo, name);
|
2003-07-26 07:41:56 +00:00
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
} else {
|
|
|
|
for(argv++; *argv; argv++) {
|
2006-07-12 19:17:55 +00:00
|
|
|
char *name;
|
|
|
|
|
|
|
|
if (isdigit(**argv)) name = get_signame(atoi(*argv));
|
|
|
|
else {
|
|
|
|
int temp = get_signum(*argv);
|
|
|
|
if (temp<0)
|
|
|
|
bb_error_msg_and_die("unknown signal %s", *argv);
|
|
|
|
name = get_signame(temp);
|
|
|
|
}
|
|
|
|
puts(name);
|
2002-07-23 23:45:11 +00:00
|
|
|
}
|
1999-10-05 16:24:54 +00:00
|
|
|
}
|
2003-07-26 07:41:56 +00:00
|
|
|
/* If they specified -l, were all done */
|
|
|
|
return EXIT_SUCCESS;
|
1999-10-05 16:24:54 +00:00
|
|
|
}
|
1999-12-08 23:19:36 +00:00
|
|
|
|
2003-07-26 07:41:56 +00:00
|
|
|
/* The -q quiet option */
|
2006-07-12 19:17:55 +00:00
|
|
|
if(killall && argv[1][1]=='q' && argv[1][2]=='\0'){
|
2003-07-26 07:41:56 +00:00
|
|
|
quiet++;
|
|
|
|
argv++;
|
|
|
|
argc--;
|
2003-08-06 07:43:58 +00:00
|
|
|
if(argc<2 || argv[1][0] != '-'){
|
2003-07-26 07:41:56 +00:00
|
|
|
goto do_it_now;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-07-12 19:17:55 +00:00
|
|
|
if(0>(signo = get_signum(argv[1]+1)))
|
2003-07-26 07:41:56 +00:00
|
|
|
bb_error_msg_and_die( "bad signal name '%s'", argv[1]+1);
|
|
|
|
argv+=2;
|
|
|
|
argc-=2;
|
|
|
|
|
2002-07-23 23:45:11 +00:00
|
|
|
do_it_now:
|
1999-10-23 05:42:08 +00:00
|
|
|
|
2006-01-25 13:21:08 +00:00
|
|
|
/* Pid or name required */
|
|
|
|
if (argc <= 0)
|
|
|
|
bb_show_usage();
|
|
|
|
|
2006-07-12 19:17:55 +00:00
|
|
|
if (!killall) {
|
2000-03-07 07:41:42 +00:00
|
|
|
/* Looks like they want to do a kill. Do that */
|
|
|
|
while (--argc >= 0) {
|
|
|
|
int pid;
|
2000-02-08 19:58:47 +00:00
|
|
|
|
2006-01-04 00:13:22 +00:00
|
|
|
if (!isdigit(**argv) && **argv != '-')
|
2003-07-26 07:41:56 +00:00
|
|
|
bb_error_msg_and_die( "Bad PID '%s'", *argv);
|
2000-03-07 07:41:42 +00:00
|
|
|
pid = strtol(*argv, NULL, 0);
|
2003-07-26 07:41:56 +00:00
|
|
|
if (kill(pid, signo) != 0) {
|
2003-03-19 09:13:01 +00:00
|
|
|
bb_perror_msg( "Could not kill pid '%d'", pid);
|
2002-09-16 06:22:25 +00:00
|
|
|
errors++;
|
2002-07-23 23:45:11 +00:00
|
|
|
}
|
2000-03-07 07:41:42 +00:00
|
|
|
argv++;
|
2000-02-08 19:58:47 +00:00
|
|
|
}
|
2002-09-16 06:22:25 +00:00
|
|
|
|
2006-07-12 19:17:55 +00:00
|
|
|
} else {
|
2000-04-18 20:00:03 +00:00
|
|
|
pid_t myPid=getpid();
|
2006-07-12 19:17:55 +00:00
|
|
|
|
2000-03-07 07:41:42 +00:00
|
|
|
/* Looks like they want to do a killall. Do that */
|
|
|
|
while (--argc >= 0) {
|
2001-12-06 14:52:32 +00:00
|
|
|
long* pidList;
|
2000-04-07 06:00:07 +00:00
|
|
|
|
2002-09-16 06:22:25 +00:00
|
|
|
pidList = find_pid_by_name(*argv);
|
2003-07-26 07:41:56 +00:00
|
|
|
if (!pidList || *pidList<=0) {
|
2002-09-16 06:22:25 +00:00
|
|
|
errors++;
|
|
|
|
if (quiet==0)
|
2003-03-19 09:13:01 +00:00
|
|
|
bb_error_msg( "%s: no process killed", *argv);
|
2003-07-26 07:41:56 +00:00
|
|
|
} else {
|
|
|
|
long *pl;
|
|
|
|
|
|
|
|
for(pl = pidList; *pl !=0 ; pl++) {
|
|
|
|
if (*pl==myPid)
|
|
|
|
continue;
|
|
|
|
if (kill(*pl, signo) != 0) {
|
|
|
|
errors++;
|
|
|
|
if (quiet==0)
|
|
|
|
bb_perror_msg( "Could not kill pid '%ld'", *pl);
|
2002-07-23 23:45:11 +00:00
|
|
|
}
|
2003-07-26 07:41:56 +00:00
|
|
|
}
|
2000-03-07 07:41:42 +00:00
|
|
|
}
|
2002-10-22 12:21:15 +00:00
|
|
|
free(pidList);
|
2000-03-07 07:41:42 +00:00
|
|
|
argv++;
|
2000-02-08 19:58:47 +00:00
|
|
|
}
|
|
|
|
}
|
2006-07-12 19:17:55 +00:00
|
|
|
|
2002-09-16 06:22:25 +00:00
|
|
|
return errors;
|
2000-02-08 19:58:47 +00:00
|
|
|
}
|