| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | This file is kill.def, from which is created kill.c. | 
					
						
							|  |  |  | It implements the builtin "kill" in Bash. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Copyright (C) 1987, 1989, 1991 Free Software Foundation, Inc. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This file is part of GNU Bash, the Bourne Again SHell. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Bash is free software; you can redistribute it and/or modify it under | 
					
						
							|  |  |  | the terms of the GNU General Public License as published by the Free | 
					
						
							|  |  |  | Software Foundation; either version 1, or (at your option) any later | 
					
						
							|  |  |  | version. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Bash is distributed in the hope that it will be useful, but WITHOUT ANY | 
					
						
							|  |  |  | WARRANTY; without even the implied warranty of MERCHANTABILITY or | 
					
						
							|  |  |  | FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License | 
					
						
							|  |  |  | for more details. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  | with Bash; see the file COPYING.  If not, write to the Free Software | 
					
						
							|  |  |  | Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $PRODUCES kill.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $BUILTIN kill | 
					
						
							|  |  |  | $FUNCTION kill_builtin | 
					
						
							|  |  |  | $DEPENDS_ON JOB_CONTROL | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | $SHORT_DOC kill [-s sigspec | -n signum | -sigspec] [pid | job]... or kill -l [sigspec] | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | Send the processes named by PID (or JOB) the signal SIGSPEC.  If | 
					
						
							|  |  |  | SIGSPEC is not present, then SIGTERM is assumed.  An argument of `-l' | 
					
						
							|  |  |  | lists the signal names; if arguments follow `-l' they are assumed to | 
					
						
							|  |  |  | be signal numbers for which names should be listed.  Kill is a shell | 
					
						
							|  |  |  | builtin for two reasons: it allows job IDs to be used instead of | 
					
						
							|  |  |  | process IDs, and, if you have reached the limit on processes that | 
					
						
							|  |  |  | you can create, you don't have to start a process to kill another one. | 
					
						
							|  |  |  | $END | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include <config.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <stdio.h> | 
					
						
							|  |  |  | #include <errno.h> | 
					
						
							|  |  |  | #if defined (HAVE_UNISTD_H) | 
					
						
							|  |  |  | #  include <unistd.h> | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../bashansi.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "../shell.h" | 
					
						
							|  |  |  | #include "../trap.h" | 
					
						
							|  |  |  | #include "../jobs.h" | 
					
						
							|  |  |  | #include "common.h" | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Not all systems declare ERRNO in errno.h... and some systems #define it! */ | 
					
						
							|  |  |  | #if !defined (errno) | 
					
						
							|  |  |  | extern int errno; | 
					
						
							|  |  |  | #endif /* !errno */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL) | 
					
						
							|  |  |  | extern int interactive; | 
					
						
							|  |  |  | extern int posixly_correct; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (CONTINUE_AFTER_KILL_ERROR) | 
					
						
							|  |  |  | #  define CONTINUE_OR_FAIL return (EXECUTION_FAILURE) | 
					
						
							|  |  |  | #else | 
					
						
							|  |  |  | #  define CONTINUE_OR_FAIL goto continue_killing | 
					
						
							|  |  |  | #endif /* CONTINUE_AFTER_KILL_ERROR */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Here is the kill builtin.  We only have it so that people can type | 
					
						
							|  |  |  |    kill -KILL %1?  No, if you fill up the process table this way you | 
					
						
							|  |  |  |    can still kill some. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | kill_builtin (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int signal, any_succeeded, listing, saw_signal; | 
					
						
							|  |  |  |   char *sigspec, *word; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   pid_t pid; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (list == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return (EXECUTION_SUCCESS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   any_succeeded = listing = saw_signal = 0; | 
					
						
							|  |  |  |   signal = SIGTERM; | 
					
						
							|  |  |  |   sigspec = "TERM"; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Process options. */ | 
					
						
							|  |  |  |   while (list) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       word = list->word->word; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (ISOPTION (word, 'l')) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  listing++; | 
					
						
							|  |  |  | 	  list = list->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (ISOPTION (word, 's') || ISOPTION (word, 'n')) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  list = list->next; | 
					
						
							|  |  |  | 	  if (list) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      sigspec = list->word->word; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      if (sigspec[0] == '0' && sigspec[1] == '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		signal = 0; | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		signal = decode_signal (sigspec); | 
					
						
							|  |  |  | 	      list = list->next; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      builtin_error ("%s requires an argument", word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      return (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (ISOPTION (word, '-')) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  list = list->next; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (ISOPTION (word, '?')) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  builtin_usage (); | 
					
						
							|  |  |  | 	  return (EXECUTION_SUCCESS); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       /* If this is a signal specification then process it.  We only process | 
					
						
							|  |  |  | 	 the first one seen; other arguments may signify process groups (e.g, | 
					
						
							|  |  |  | 	 -num == process group num). */ | 
					
						
							|  |  |  |       else if ((*word == '-') && !saw_signal) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  sigspec = word + 1; | 
					
						
							|  |  |  | 	  signal = decode_signal (sigspec); | 
					
						
							|  |  |  | 	  saw_signal++; | 
					
						
							|  |  |  | 	  list = list->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (listing) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     return (display_signal_list (list, 0)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* OK, we are killing processes. */ | 
					
						
							|  |  |  |   if (signal == NO_SIG) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       builtin_error ("bad signal spec `%s'", sigspec); | 
					
						
							|  |  |  |       return (EXECUTION_FAILURE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (list) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       word = list->word->word; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (*word == '-') | 
					
						
							|  |  |  | 	word++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (all_digits (word)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* Use the entire argument in case of minus sign presence. */ | 
					
						
							|  |  |  | 	  pid = (pid_t) atoi (list->word->word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (kill_pid (pid, signal, 0) < 0) | 
					
						
							|  |  |  | 	    goto signal_error; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    any_succeeded++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (*list->word->word != '%') | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  builtin_error ("%s: no such pid", list->word->word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  CONTINUE_OR_FAIL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (interactive) | 
					
						
							|  |  |  | 	/* Posix.2 says you can kill without job control active (4.32.4) */ | 
					
						
							|  |  |  | 	{			/* Must be a job spec.  Check it out. */ | 
					
						
							|  |  |  | 	  int job; | 
					
						
							|  |  |  | 	  sigset_t set, oset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  BLOCK_CHILD (set, oset); | 
					
						
							|  |  |  | 	  job = get_job_spec (list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (job < 0 || job >= job_slots || !jobs[job]) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (job != DUP_JOB) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		builtin_error ("%s: no such job", list->word->word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      UNBLOCK_CHILD (oset); | 
					
						
							|  |  |  | 	      CONTINUE_OR_FAIL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Job spec used.  Kill the process group. If the job was started | 
					
						
							|  |  |  | 	     without job control, then its pgrp == shell_pgrp, so we have | 
					
						
							|  |  |  | 	     to be careful.  We take the pid of the first job in the pipeline | 
					
						
							|  |  |  | 	     in that case. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  pid = IS_JOBCONTROL (job) ? jobs[job]->pgrp : jobs[job]->pipe->pid; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  UNBLOCK_CHILD (oset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (kill_pid (pid, signal, 1) < 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	    signal_error: | 
					
						
							|  |  |  | 	      if (errno == EPERM) | 
					
						
							|  |  |  | 		builtin_error ("(%d) - Not owner", (int)pid); | 
					
						
							|  |  |  | 	      else if (errno == ESRCH) | 
					
						
							|  |  |  | 		builtin_error ("(%d) - No such pid", (int)pid); | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		builtin_error ("Invalid signal %d", signal); | 
					
						
							|  |  |  | 	      CONTINUE_OR_FAIL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    any_succeeded++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  builtin_error ("`%s' is not a pid or valid job spec", list->word->word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  CONTINUE_OR_FAIL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     continue_killing: | 
					
						
							|  |  |  |       list = list->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (any_succeeded ? EXECUTION_SUCCESS : EXECUTION_FAILURE); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif /* JOB_CONTROL */ |