Home | History | Annotate | Line # | Download | only in kill
kill.c revision 1.21
      1  1.21  christos /* $NetBSD: kill.c,v 1.21 2002/11/25 14:23:07 christos Exp $ */
      2  1.10       cgd 
      3   1.1       cgd /*
      4   1.7   mycroft  * Copyright (c) 1988, 1993, 1994
      5   1.7   mycroft  *	The Regents of the University of California.  All rights reserved.
      6   1.1       cgd  *
      7   1.1       cgd  * Redistribution and use in source and binary forms, with or without
      8   1.1       cgd  * modification, are permitted provided that the following conditions
      9   1.1       cgd  * are met:
     10   1.1       cgd  * 1. Redistributions of source code must retain the above copyright
     11   1.1       cgd  *    notice, this list of conditions and the following disclaimer.
     12   1.1       cgd  * 2. Redistributions in binary form must reproduce the above copyright
     13   1.1       cgd  *    notice, this list of conditions and the following disclaimer in the
     14   1.1       cgd  *    documentation and/or other materials provided with the distribution.
     15   1.1       cgd  * 3. All advertising materials mentioning features or use of this software
     16   1.1       cgd  *    must display the following acknowledgement:
     17   1.1       cgd  *	This product includes software developed by the University of
     18   1.1       cgd  *	California, Berkeley and its contributors.
     19   1.1       cgd  * 4. Neither the name of the University nor the names of its contributors
     20   1.1       cgd  *    may be used to endorse or promote products derived from this software
     21   1.1       cgd  *    without specific prior written permission.
     22   1.1       cgd  *
     23   1.1       cgd  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
     24   1.1       cgd  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     25   1.1       cgd  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     26   1.1       cgd  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
     27   1.1       cgd  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     28   1.1       cgd  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     29   1.1       cgd  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     30   1.1       cgd  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     31   1.1       cgd  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     32   1.1       cgd  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     33   1.1       cgd  * SUCH DAMAGE.
     34   1.1       cgd  */
     35   1.1       cgd 
     36  1.13  christos #include <sys/cdefs.h>
     37  1.20  christos #if !defined(lint) && !defined(SHELL)
     38  1.13  christos __COPYRIGHT("@(#) Copyright (c) 1988, 1993, 1994\n\
     39  1.13  christos 	The Regents of the University of California.  All rights reserved.\n");
     40   1.1       cgd #endif /* not lint */
     41   1.1       cgd 
     42   1.1       cgd #ifndef lint
     43  1.10       cgd #if 0
     44  1.11       jtc static char sccsid[] = "@(#)kill.c	8.4 (Berkeley) 4/28/95";
     45  1.10       cgd #else
     46  1.21  christos __RCSID("$NetBSD: kill.c,v 1.21 2002/11/25 14:23:07 christos Exp $");
     47  1.10       cgd #endif
     48   1.1       cgd #endif /* not lint */
     49   1.1       cgd 
     50   1.7   mycroft #include <ctype.h>
     51   1.7   mycroft #include <err.h>
     52   1.7   mycroft #include <errno.h>
     53   1.1       cgd #include <signal.h>
     54   1.1       cgd #include <stdio.h>
     55   1.1       cgd #include <stdlib.h>
     56   1.1       cgd #include <string.h>
     57  1.20  christos #include <termios.h>
     58  1.20  christos #include <unistd.h>
     59  1.20  christos #include <locale.h>
     60  1.20  christos #include <sys/ioctl.h>
     61  1.20  christos 
     62  1.20  christos #ifdef SHELL            /* sh (aka ash) builtin */
     63  1.20  christos #define main killcmd
     64  1.20  christos #include "../../bin/sh/bltin/bltin.h"
     65  1.20  christos #endif /* SHELL */
     66  1.20  christos 
     67  1.20  christos static void nosig(char *);
     68  1.20  christos static void printsignals(FILE *);
     69  1.20  christos static int signame_to_signum(char *);
     70  1.20  christos static void usage(void);
     71  1.18       wiz int main(int, char *[]);
     72   1.7   mycroft 
     73   1.7   mycroft int
     74  1.18       wiz main(int argc, char *argv[])
     75   1.1       cgd {
     76   1.7   mycroft 	int errors, numsig, pid;
     77   1.1       cgd 	char *ep;
     78   1.1       cgd 
     79  1.19       wiz 	setprogname(argv[0]);
     80  1.20  christos 	setlocale(LC_ALL, "");
     81   1.1       cgd 	if (argc < 2)
     82   1.1       cgd 		usage();
     83   1.1       cgd 
     84   1.4       jtc 	numsig = SIGTERM;
     85   1.7   mycroft 
     86   1.4       jtc 	argc--, argv++;
     87  1.18       wiz 	if (strcmp(*argv, "-l") == 0) {
     88   1.7   mycroft 		argc--, argv++;
     89   1.7   mycroft 		if (argc > 1)
     90   1.7   mycroft 			usage();
     91   1.7   mycroft 		if (argc == 1) {
     92  1.18       wiz 			if (isdigit((unsigned char)**argv) == 0)
     93   1.7   mycroft 				usage();
     94   1.7   mycroft 			numsig = strtol(*argv, &ep, 10);
     95  1.18       wiz 			if (*ep != '\0')
     96   1.7   mycroft 				errx(1, "illegal signal number: %s", *argv);
     97   1.7   mycroft 			if (numsig >= 128)
     98   1.7   mycroft 				numsig -= 128;
     99   1.7   mycroft 			if (numsig <= 0 || numsig >= NSIG)
    100   1.7   mycroft 				nosig(*argv);
    101  1.20  christos 			printf("%s\n", sys_signame[numsig]);
    102   1.7   mycroft 			exit(0);
    103   1.4       jtc 		}
    104   1.4       jtc 		printsignals(stdout);
    105   1.1       cgd 		exit(0);
    106   1.7   mycroft 	}
    107   1.7   mycroft 
    108   1.7   mycroft 	if (!strcmp(*argv, "-s")) {
    109   1.7   mycroft 		argc--, argv++;
    110   1.7   mycroft 		if (argc < 1) {
    111   1.7   mycroft 			warnx("option requires an argument -- s");
    112   1.4       jtc 			usage();
    113   1.4       jtc 		}
    114   1.7   mycroft 		if (strcmp(*argv, "0")) {
    115   1.7   mycroft 			if ((numsig = signame_to_signum(*argv)) < 0)
    116   1.7   mycroft 				nosig(*argv);
    117   1.7   mycroft 		} else
    118   1.4       jtc 			numsig = 0;
    119   1.7   mycroft 		argc--, argv++;
    120   1.4       jtc 	} else if (**argv == '-') {
    121   1.1       cgd 		++*argv;
    122  1.17  christos 		if (isalpha((unsigned char)**argv)) {
    123   1.7   mycroft 			if ((numsig = signame_to_signum(*argv)) < 0)
    124   1.4       jtc 				nosig(*argv);
    125  1.17  christos 		} else if (isdigit((unsigned char)**argv)) {
    126   1.1       cgd 			numsig = strtol(*argv, &ep, 10);
    127   1.7   mycroft 			if (!*argv || *ep)
    128   1.7   mycroft 				errx(1, "illegal signal number: %s", *argv);
    129   1.9       jtc 			if (numsig < 0 || numsig >= NSIG)
    130   1.1       cgd 				nosig(*argv);
    131   1.1       cgd 		} else
    132   1.1       cgd 			nosig(*argv);
    133   1.7   mycroft 		argc--, argv++;
    134   1.1       cgd 	}
    135   1.1       cgd 
    136   1.7   mycroft 	if (argc == 0)
    137   1.1       cgd 		usage();
    138   1.1       cgd 
    139   1.7   mycroft 	for (errors = 0; argc; argc--, argv++) {
    140  1.20  christos #ifdef SHELL
    141  1.20  christos 		extern int getjobpgrp(const char *);
    142  1.20  christos 		if (*argv[0] == '%') {
    143  1.20  christos 			pid = getjobpgrp(*argv);
    144  1.20  christos 			if (pid == 0) {
    145  1.20  christos 				warnx("illegal job id: %s", *argv);
    146  1.20  christos 				errors = 1;
    147  1.20  christos 				continue;
    148  1.20  christos 			}
    149  1.20  christos 		} else
    150  1.20  christos #endif
    151  1.20  christos 		{
    152  1.20  christos 			pid = strtol(*argv, &ep, 10);
    153  1.20  christos 			if (!**argv || *ep) {
    154  1.20  christos 				warnx("illegal process id: %s", *argv);
    155  1.20  christos 				errors = 1;
    156  1.20  christos 				continue;
    157  1.20  christos 			}
    158  1.20  christos 		}
    159  1.20  christos 		if (kill(pid, numsig) == -1) {
    160   1.7   mycroft 			warn("%s", *argv);
    161   1.1       cgd 			errors = 1;
    162   1.1       cgd 		}
    163  1.21  christos #ifdef SHELL
    164  1.21  christos 		/* Wakeup the process if it was suspended, so it can
    165  1.21  christos 		   exit without an explicit 'fg'. */
    166  1.21  christos 		if (numsig == SIGTERM || numsig == SIGHUP)
    167  1.21  christos 			kill(pid, SIGCONT);
    168  1.21  christos #endif
    169   1.1       cgd 	}
    170   1.7   mycroft 
    171   1.1       cgd 	exit(errors);
    172  1.14   mycroft 	/* NOTREACHED */
    173   1.1       cgd }
    174   1.1       cgd 
    175  1.20  christos static int
    176  1.18       wiz signame_to_signum(char *sig)
    177   1.4       jtc {
    178   1.6   mycroft 	int n;
    179   1.4       jtc 
    180  1.18       wiz 	if (strncasecmp(sig, "sig", 3) == 0)
    181   1.4       jtc 		sig += 3;
    182   1.6   mycroft 	for (n = 1; n < NSIG; n++) {
    183   1.6   mycroft 		if (!strcasecmp(sys_signame[n], sig))
    184   1.7   mycroft 			return (n);
    185   1.4       jtc 	}
    186   1.7   mycroft 	return (-1);
    187   1.4       jtc }
    188   1.4       jtc 
    189  1.20  christos static void
    190  1.18       wiz nosig(char *name)
    191   1.1       cgd {
    192   1.7   mycroft 
    193   1.7   mycroft 	warnx("unknown signal %s; valid signals:", name);
    194   1.4       jtc 	printsignals(stderr);
    195   1.1       cgd 	exit(1);
    196  1.15   mycroft 	/* NOTREACHED */
    197   1.1       cgd }
    198   1.1       cgd 
    199  1.20  christos static void
    200  1.18       wiz printsignals(FILE *fp)
    201   1.1       cgd {
    202  1.20  christos 	int sig;
    203  1.20  christos 	int len, nl;
    204  1.20  christos 	const char *name;
    205  1.20  christos 	int termwidth = 80;
    206  1.20  christos 
    207  1.20  christos 	if (isatty(fileno(fp))) {
    208  1.20  christos 		struct winsize win;
    209  1.20  christos 		if (ioctl(fileno(fp), TIOCGWINSZ, &win) == 0 && win.ws_col > 0)
    210  1.20  christos 			termwidth = win.ws_col;
    211  1.20  christos 	}
    212   1.1       cgd 
    213  1.20  christos 	for (len = 0, sig = 1; sig < NSIG; sig++) {
    214  1.20  christos 		name = sys_signame[sig];
    215  1.20  christos 		nl = 1 + strlen(name);
    216  1.20  christos 
    217  1.20  christos 		if (len + nl >= termwidth) {
    218  1.20  christos 			fprintf(fp, "\n");
    219  1.20  christos 			len = 0;
    220  1.20  christos 		} else
    221  1.20  christos 			if (len != 0)
    222  1.20  christos 				fprintf(fp, " ");
    223  1.20  christos 		len += nl;
    224  1.20  christos 		fprintf(fp, "%s", name);
    225   1.7   mycroft 	}
    226  1.20  christos 	if (len != 0)
    227  1.20  christos 		fprintf(fp, "\n");
    228   1.1       cgd }
    229   1.1       cgd 
    230  1.20  christos static void
    231  1.18       wiz usage(void)
    232   1.1       cgd {
    233   1.7   mycroft 
    234  1.20  christos 	fprintf(stderr, "usage: %s [-s signal_name] pid ...\n"
    235  1.19       wiz 	    "       %s -l [exit_status]\n"
    236  1.19       wiz 	    "       %s -signal_name pid ...\n"
    237  1.19       wiz 	    "       %s -signal_number pid ...\n",
    238  1.19       wiz 	    getprogname(), getprogname(), getprogname(), getprogname());
    239   1.1       cgd 	exit(1);
    240  1.15   mycroft 	/* NOTREACHED */
    241   1.1       cgd }
    242