| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | This file is command.def, from which is created command.c. | 
					
						
							|  |  |  | It implements the builtin "command" 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 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | Software Foundation; either version 2, or (at your option) any later | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 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 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | Foundation, 59 Temple Place, Suite 330, Boston, MA 02111 USA. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | $PRODUCES command.c | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | $BUILTIN command | 
					
						
							|  |  |  | $FUNCTION command_builtin | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | $SHORT_DOC command [-pVv] command [arg ...] | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | Runs COMMAND with ARGS ignoring shell functions.  If you have a shell | 
					
						
							|  |  |  | function called `ls', and you wish to call the command `ls', you can | 
					
						
							|  |  |  | say "command ls".  If the -p option is given, a default value is used | 
					
						
							|  |  |  | for PATH that is guaranteed to find all of the standard utilities.  If | 
					
						
							|  |  |  | the -V or -v option is given, a string is printed describing COMMAND. | 
					
						
							|  |  |  | The -V option produces a more verbose description. | 
					
						
							|  |  |  | $END | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include <config.h> | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_UNISTD_H) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #  ifdef _MINIX | 
					
						
							|  |  |  | #    include <sys/types.h> | 
					
						
							|  |  |  | #  endif | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #  include <unistd.h> | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../bashansi.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "../shell.h" | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "../execute_cmd.h" | 
					
						
							|  |  |  | #include "../flags.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "bashgetopt.h" | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "common.h" | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #if defined (_CS_PATH) && defined (HAVE_CONFSTR) && !HAVE_DECL_CONFSTR | 
					
						
							|  |  |  | extern size_t confstr __P((int, char *, size_t)); | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | extern int subshell_environment; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static void restore_path __P((char *)); | 
					
						
							|  |  |  | static char *get_standard_path __P((void)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Run the commands mentioned in LIST without paying attention to shell | 
					
						
							|  |  |  |    functions. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | command_builtin (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int result, verbose, use_standard_path, opt; | 
					
						
							|  |  |  |   char *old_path, *standard_path; | 
					
						
							|  |  |  |   COMMAND *command; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   verbose = use_standard_path = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   reset_internal_getopt (); | 
					
						
							|  |  |  |   while ((opt = internal_getopt (list, "pvV")) != -1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       switch (opt) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case 'p': | 
					
						
							|  |  |  | 	  use_standard_path = 1; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case 'V': | 
					
						
							|  |  |  | 	  verbose = 2; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case 'v': | 
					
						
							|  |  |  | 	  verbose = 4; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  builtin_usage (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  return (EX_USAGE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   list = loptend; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (list == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return (EXECUTION_SUCCESS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (verbose) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       int found, any_found; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       for (any_found = 0; list; list = list->next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  found = describe_command (list->word->word, verbose, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (found == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    builtin_error ("%s: not found", list->word->word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  any_found += found; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       return (any_found ? EXECUTION_SUCCESS : EXECUTION_FAILURE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (RESTRICTED_SHELL) | 
					
						
							|  |  |  |   if (use_standard_path && restricted) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       builtin_error ("restricted: cannot use -p"); | 
					
						
							|  |  |  |       return (EXECUTION_FAILURE); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   begin_unwind_frame ("command_builtin"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* We don't want this to be reparsed (consider command echo 'foo &'), so | 
					
						
							|  |  |  |      just make a simple_command structure and call execute_command with it. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (use_standard_path) | 
					
						
							|  |  |  |     {       | 
					
						
							|  |  |  |       old_path = get_string_value ("PATH"); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       /* If old_path is NULL, $PATH is unset.  If so, we want to make sure | 
					
						
							|  |  |  | 	 it's unset after this command completes. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (old_path) | 
					
						
							|  |  |  | 	old_path = savestring (old_path); | 
					
						
							|  |  |  |       add_unwind_protect ((Function *)restore_path, old_path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       standard_path = get_standard_path (); | 
					
						
							|  |  |  |       bind_variable ("PATH", standard_path ? standard_path : ""); | 
					
						
							|  |  |  |       FREE (standard_path); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #define COMMAND_BUILTIN_FLAGS (CMD_NO_FUNCTIONS | CMD_INHIBIT_EXPANSION | CMD_COMMAND_BUILTIN) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   command = make_bare_simple_command (); | 
					
						
							|  |  |  |   command->value.Simple->words = (WORD_LIST *)copy_word_list (list); | 
					
						
							|  |  |  |   command->value.Simple->redirects = (REDIRECT *)NULL; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   command->flags |= COMMAND_BUILTIN_FLAGS; | 
					
						
							|  |  |  |   command->value.Simple->flags |= COMMAND_BUILTIN_FLAGS; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if 0 | 
					
						
							|  |  |  |   /* This breaks for things like ( cd /tmp ; command z ababa ; echo next ) | 
					
						
							|  |  |  |      or $(command echo a ; command echo b;) or even | 
					
						
							|  |  |  |      { command echo a; command echo b; } & */ | 
					
						
							|  |  |  |   /* If we're in a subshell, see if we can get away without forking | 
					
						
							|  |  |  |      again, since we've already forked to run this builtin. */ | 
					
						
							|  |  |  |   if (subshell_environment) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       command->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  |       command->value.Simple->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif | 
					
						
							|  |  |  |   add_unwind_protect ((char *)dispose_command, command); | 
					
						
							|  |  |  |   result = execute_command (command); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   run_unwind_frame ("command_builtin"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Restore the value of the $PATH variable after replacing it when | 
					
						
							|  |  |  |    executing `command -p'. */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | restore_path (var) | 
					
						
							|  |  |  |      char *var; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   if (var) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       bind_variable ("PATH", var); | 
					
						
							|  |  |  |       free (var); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     unbind_variable ("PATH"); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return a value for PATH that is guaranteed to find all of the standard | 
					
						
							|  |  |  |    utilities.  This uses Posix.2 configuration variables, if present.  It | 
					
						
							|  |  |  |    uses a value defined in config.h as a last resort. */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | get_standard_path () | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (_CS_PATH) && defined (HAVE_CONFSTR) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   char *p; | 
					
						
							|  |  |  |   size_t len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   len = (size_t)confstr (_CS_PATH, (char *)NULL, (size_t)0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (len > 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       p = (char *)xmalloc (len + 2); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       *p = '\0'; | 
					
						
							|  |  |  |       confstr (_CS_PATH, p, len); | 
					
						
							|  |  |  |       return (p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return (savestring (STANDARD_UTILS_PATH)); | 
					
						
							|  |  |  | #else /* !_CS_PATH || !HAVE_CONFSTR  */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #  if defined (CS_PATH) | 
					
						
							|  |  |  |   return (savestring (CS_PATH)); | 
					
						
							|  |  |  | #  else | 
					
						
							|  |  |  |   return (savestring (STANDARD_UTILS_PATH)); | 
					
						
							|  |  |  | #  endif /* !CS_PATH */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* !_CS_PATH || !HAVE_CONFSTR */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } |