]> CyberLeo.Net >> Repos - FreeBSD/stable/10.git/blob - bin/kill/kill.c
MFC r368207,368607:
[FreeBSD/stable/10.git] / bin / kill / kill.c
1 /*-
2  * Copyright (c) 1988, 1993, 1994
3  *      The Regents of the University of California.  All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  * 4. Neither the name of the University nor the names of its contributors
14  *    may be used to endorse or promote products derived from this software
15  *    without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  */
29 /*
30  * Important: This file is used both as a standalone program /bin/kill and
31  * as a builtin for /bin/sh (#define SHELL).
32  */
33
34 #if 0
35 #ifndef lint
36 static char const copyright[] =
37 "@(#) Copyright (c) 1988, 1993, 1994\n\
38         The Regents of the University of California.  All rights reserved.\n";
39 #endif /* not lint */
40
41 #ifndef lint
42 static char sccsid[] = "@(#)kill.c      8.4 (Berkeley) 4/28/95";
43 #endif /* not lint */
44 #endif
45 #include <sys/cdefs.h>
46 __FBSDID("$FreeBSD$");
47
48 #include <ctype.h>
49 #include <err.h>
50 #include <errno.h>
51 #include <signal.h>
52 #include <stdio.h>
53 #include <stdlib.h>
54 #include <string.h>
55
56 #ifdef SHELL
57 #define main killcmd
58 #include "bltin/bltin.h"
59 #include "error.h"
60 #endif
61
62 static void nosig(const char *);
63 static void printsignals(FILE *);
64 static int signame_to_signum(const char *);
65 static void usage(void);
66
67 int
68 main(int argc, char *argv[])
69 {
70         long pidl;
71         pid_t pid;
72         int errors, numsig;
73         char *ep;
74
75         if (argc < 2)
76                 usage();
77
78         numsig = SIGTERM;
79
80         argc--, argv++;
81         if (!strcmp(*argv, "-l")) {
82                 argc--, argv++;
83                 if (argc > 1)
84                         usage();
85                 if (argc == 1) {
86                         if (!isdigit(**argv))
87                                 usage();
88                         numsig = strtol(*argv, &ep, 10);
89                         if (!**argv || *ep)
90                                 errx(2, "illegal signal number: %s", *argv);
91                         if (numsig >= 128)
92                                 numsig -= 128;
93                         if (numsig <= 0 || numsig >= sys_nsig)
94                                 nosig(*argv);
95                         printf("%s\n", sys_signame[numsig]);
96                         return (0);
97                 }
98                 printsignals(stdout);
99                 return (0);
100         }
101
102         if (!strcmp(*argv, "-s")) {
103                 argc--, argv++;
104                 if (argc < 1) {
105                         warnx("option requires an argument -- s");
106                         usage();
107                 }
108                 if (strcmp(*argv, "0")) {
109                         if ((numsig = signame_to_signum(*argv)) < 0)
110                                 nosig(*argv);
111                 } else
112                         numsig = 0;
113                 argc--, argv++;
114         } else if (**argv == '-' && *(*argv + 1) != '-') {
115                 ++*argv;
116                 if (isalpha(**argv)) {
117                         if ((numsig = signame_to_signum(*argv)) < 0)
118                                 nosig(*argv);
119                 } else if (isdigit(**argv)) {
120                         numsig = strtol(*argv, &ep, 10);
121                         if (!**argv || *ep)
122                                 errx(2, "illegal signal number: %s", *argv);
123                         if (numsig < 0)
124                                 nosig(*argv);
125                 } else
126                         nosig(*argv);
127                 argc--, argv++;
128         }
129
130         if (argc > 0 && strncmp(*argv, "--", 2) == 0)
131                 argc--, argv++;
132
133         if (argc == 0)
134                 usage();
135
136         for (errors = 0; argc; argc--, argv++) {
137 #ifdef SHELL
138                 if (**argv == '%')
139                         pid = getjobpgrp(*argv);
140                 else
141 #endif
142                 {
143                         pidl = strtol(*argv, &ep, 10);
144                         /* Check for overflow of pid_t. */
145                         pid = (pid_t)pidl;
146                         if (!**argv || *ep || pid != pidl)
147                                 errx(2, "illegal process id: %s", *argv);
148                 }
149                 if (kill(pid, numsig) == -1) {
150                         warn("%s", *argv);
151                         errors = 1;
152                 }
153         }
154
155         return (errors);
156 }
157
158 static int
159 signame_to_signum(const char *sig)
160 {
161         int n;
162
163         if (strncasecmp(sig, "SIG", 3) == 0)
164                 sig += 3;
165         for (n = 1; n < sys_nsig; n++) {
166                 if (!strcasecmp(sys_signame[n], sig))
167                         return (n);
168         }
169         return (-1);
170 }
171
172 static void
173 nosig(const char *name)
174 {
175
176         warnx("unknown signal %s; valid signals:", name);
177         printsignals(stderr);
178 #ifdef SHELL
179         error(NULL);
180 #else
181         exit(2);
182 #endif
183 }
184
185 static void
186 printsignals(FILE *fp)
187 {
188         int n;
189
190         for (n = 1; n < sys_nsig; n++) {
191                 (void)fprintf(fp, "%s", sys_signame[n]);
192                 if (n == (sys_nsig / 2) || n == (sys_nsig - 1))
193                         (void)fprintf(fp, "\n");
194                 else
195                         (void)fprintf(fp, " ");
196         }
197 }
198
199 static void
200 usage(void)
201 {
202
203         (void)fprintf(stderr, "%s\n%s\n%s\n%s\n",
204                 "usage: kill [-s signal_name] pid ...",
205                 "       kill -l [exit_status]",
206                 "       kill -signal_name pid ...",
207                 "       kill -signal_number pid ...");
208 #ifdef SHELL
209         error(NULL);
210 #else
211         exit(2);
212 #endif
213 }