| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | /* evalstring.c - evaluate a string as one or more shell commands. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /* Copyright (C) 1996-2015 Free Software Foundation, Inc.
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |    This file is part of GNU Bash, the Bourne Again SHell. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    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 3 of the License, or | 
					
						
							|  |  |  |    (at your option) any later version. | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    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.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											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 <stdio.h>
 | 
					
						
							|  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #include <errno.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #include "filecntl.h"
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "../bashansi.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../shell.h"
 | 
					
						
							|  |  |  | #include "../jobs.h"
 | 
					
						
							|  |  |  | #include "../builtins.h"
 | 
					
						
							|  |  |  | #include "../flags.h"
 | 
					
						
							|  |  |  | #include "../input.h"
 | 
					
						
							|  |  |  | #include "../execute_cmd.h"
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #include "../redir.h"
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include "../trap.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #include "../bashintl.h"
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #include <y.tab.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (HISTORY)
 | 
					
						
							|  |  |  | #  include "../bashhist.h"
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "common.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #include "builtext.h"
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #if !defined (errno)
 | 
					
						
							|  |  |  | extern int errno; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | #define IS_BUILTIN(s)	(builtin_address_internal(s, 0) != (struct builtin *)NULL)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | extern int indirection_level, subshell_environment; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | extern int line_number, line_number_for_err_trap; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | extern int current_token, shell_eof_token; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | extern int last_command_exit_value; | 
					
						
							|  |  |  | extern int running_trap; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | extern int loop_level; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | extern int executing_list; | 
					
						
							|  |  |  | extern int comsub_ignore_return; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | extern int posixly_correct; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | extern int return_catch_flag, return_catch_value; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | extern sh_builtin_func_t *this_shell_builtin; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | extern char *the_printed_command_except_trap; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | int parse_and_execute_level = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static int cat_file __P((REDIRECT *)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #define PE_TAG "parse_and_execute top"
 | 
					
						
							|  |  |  | #define PS_TAG "parse_string top"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | #if defined (HISTORY)
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | set_history_remembering () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   remember_on_history = enable_history_list; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | static void | 
					
						
							|  |  |  | restore_lastcom (x) | 
					
						
							|  |  |  |      char *x; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   FREE (the_printed_command_except_trap); | 
					
						
							|  |  |  |   the_printed_command_except_trap = x; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | int | 
					
						
							|  |  |  | should_suppress_fork (command) | 
					
						
							|  |  |  |      COMMAND *command; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (startup_state == 2 && parse_and_execute_level == 1 && | 
					
						
							|  |  |  | 	  running_trap == 0 && | 
					
						
							|  |  |  | 	  *bash_input.location.string == '\0' && | 
					
						
							|  |  |  | 	  command->type == cm_simple && | 
					
						
							|  |  |  | 	  signal_is_trapped (EXIT_TRAP) == 0 && | 
					
						
							|  |  |  | 	  signal_is_trapped (ERROR_TRAP) == 0 && | 
					
						
							|  |  |  | 	  any_signals_trapped () < 0 && | 
					
						
							|  |  |  | 	  command->redirects == 0 && command->value.Simple->redirects == 0 && | 
					
						
							|  |  |  | 	  ((command->flags & CMD_TIME_PIPELINE) == 0) && | 
					
						
							|  |  |  | 	  ((command->flags & CMD_INVERT_RETURN) == 0)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | optimize_fork (command) | 
					
						
							|  |  |  |      COMMAND *command; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (command->type == cm_connection && | 
					
						
							|  |  |  |       (command->value.Connection->connector == AND_AND || command->value.Connection->connector == OR_OR) && | 
					
						
							|  |  |  |       should_suppress_fork (command->value.Connection->second)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       command->value.Connection->second->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  |       command->value.Connection->second->value.Simple->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  |       | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* How to force parse_and_execute () to clean up after itself. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | parse_and_execute_cleanup () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (running_trap) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       run_trap_cleanup (running_trap - 1); | 
					
						
							|  |  |  |       unfreeze_jobs_list (); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (have_unwind_protects ()) | 
					
						
							|  |  |  |      run_unwind_frame (PE_TAG); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     parse_and_execute_level = 0;			/* XXX */ | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | parse_prologue (string, flags, tag) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      char *tag; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   char *orig_string, *lastcom; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int x; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   orig_string = string; | 
					
						
							|  |  |  |   /* Unwind protect this invocation of parse_and_execute (). */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   begin_unwind_frame (tag); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   unwind_protect_int (parse_and_execute_level); | 
					
						
							|  |  |  |   unwind_protect_jmp_buf (top_level); | 
					
						
							|  |  |  |   unwind_protect_int (indirection_level); | 
					
						
							|  |  |  |   unwind_protect_int (line_number); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   unwind_protect_int (line_number_for_err_trap); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   unwind_protect_int (loop_level); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   unwind_protect_int (executing_list); | 
					
						
							|  |  |  |   unwind_protect_int (comsub_ignore_return); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   if (flags & (SEVAL_NONINT|SEVAL_INTERACT)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     unwind_protect_int (interactive); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HISTORY)
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   if (parse_and_execute_level == 0) | 
					
						
							|  |  |  |     add_unwind_protect (set_history_remembering, (char *)NULL); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     unwind_protect_int (remember_on_history);	/* can be used in scripts */ | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #  if defined (BANG_HISTORY)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   unwind_protect_int (history_expansion_inhibited); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #  endif /* BANG_HISTORY */
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* HISTORY */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (interactive_shell) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       x = get_current_prompt_level (); | 
					
						
							|  |  |  |       add_unwind_protect (set_current_prompt_level, x); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (the_printed_command_except_trap) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       lastcom = savestring (the_printed_command_except_trap); | 
					
						
							|  |  |  |       add_unwind_protect (restore_lastcom, lastcom); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   add_unwind_protect (pop_stream, (char *)NULL); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (parser_expanding_alias ()) | 
					
						
							|  |  |  |     add_unwind_protect (parser_restore_alias, (char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (orig_string && ((flags & SEVAL_NOFREE) == 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     add_unwind_protect (xfree, orig_string); | 
					
						
							|  |  |  |   end_unwind_frame (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (flags & (SEVAL_NONINT|SEVAL_INTERACT)) | 
					
						
							|  |  |  |     interactive = (flags & SEVAL_NONINT) ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HISTORY)
 | 
					
						
							|  |  |  |   if (flags & SEVAL_NOHIST) | 
					
						
							|  |  |  |     bash_history_disable (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #  if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  |   if (flags & SEVAL_NOHISTEXP) | 
					
						
							|  |  |  |     history_expansion_inhibited = 1; | 
					
						
							|  |  |  | #  endif /* BANG_HISTORY */
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #endif /* HISTORY */
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Parse and execute the commands in STRING.  Returns whatever
 | 
					
						
							|  |  |  |    execute_command () returns.  This frees STRING.  FLAGS is a | 
					
						
							|  |  |  |    flags word; look in common.h for the possible values.  Actions | 
					
						
							|  |  |  |    are: | 
					
						
							|  |  |  |    	(flags & SEVAL_NONINT) -> interactive = 0; | 
					
						
							|  |  |  |    	(flags & SEVAL_INTERACT) -> interactive = 1; | 
					
						
							|  |  |  |    	(flags & SEVAL_NOHIST) -> call bash_history_disable () | 
					
						
							|  |  |  |    	(flags & SEVAL_NOFREE) -> don't free STRING when finished | 
					
						
							|  |  |  |    	(flags & SEVAL_RESETLINE) -> reset line_number to 1 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |    	(flags & SEVAL_NOHISTEXP) -> history_expansion_inhibited -> 1 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | parse_and_execute (string, from_file, flags) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      const char *from_file; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int code, lreset; | 
					
						
							|  |  |  |   volatile int should_jump_to_top_level, last_result; | 
					
						
							|  |  |  |   COMMAND *volatile command; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   volatile sigset_t pe_sigmask; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   parse_prologue (string, flags, PE_TAG); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   parse_and_execute_level++; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   lreset = flags & SEVAL_RESETLINE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #if defined (HAVE_POSIX_SIGNALS)
 | 
					
						
							|  |  |  |   /* If we longjmp and are going to go on, use this to restore signal mask */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   sigemptyset ((sigset_t *)&pe_sigmask); | 
					
						
							|  |  |  |   sigprocmask (SIG_BLOCK, (sigset_t *)NULL, (sigset_t *)&pe_sigmask); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Reset the line number if the caller wants us to.  If we don't reset the
 | 
					
						
							|  |  |  |      line number, we have to subtract one, because we will add one just | 
					
						
							|  |  |  |      before executing the next command (resetting the line number sets it to | 
					
						
							|  |  |  |      0; the first line number is 1). */ | 
					
						
							|  |  |  |   push_stream (lreset); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (parser_expanding_alias ()) | 
					
						
							|  |  |  |     /* push current shell_input_line */ | 
					
						
							|  |  |  |     parser_save_alias (); | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (lreset == 0) | 
					
						
							|  |  |  |     line_number--; | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   indirection_level++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   code = should_jump_to_top_level = 0; | 
					
						
							|  |  |  |   last_result = EXECUTION_SUCCESS; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* We need to reset enough of the token state so we can start fresh. */ | 
					
						
							|  |  |  |   if (current_token == yacc_EOF) | 
					
						
							|  |  |  |     current_token = '\n';		/* reset_parser() ? */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   with_input_from_string (string, from_file); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   clear_shell_input_line (); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   while (*(bash_input.location.string)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       command = (COMMAND *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (interrupt_state) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  last_result = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Provide a location for functions which `longjmp (top_level)' to
 | 
					
						
							|  |  |  | 	 jump to.  This prevents errors in substitution from restarting | 
					
						
							|  |  |  | 	 the reader loop directly, for example. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       code = setjmp_nosigs (top_level); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (code) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  should_jump_to_top_level = 0; | 
					
						
							|  |  |  | 	  switch (code) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    case ERREXIT: | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      /* variable_context -> 0 is what eval.c:reader_loop() does in
 | 
					
						
							|  |  |  | 		 these circumstances.  Don't bother with cleanup here because | 
					
						
							|  |  |  | 		 we don't want to run the function execution cleanup stuff | 
					
						
							|  |  |  | 		 that will cause pop_context and other functions to run. | 
					
						
							|  |  |  | 		 XXX - change that if we want the function context to be | 
					
						
							|  |  |  | 		 unwound. */ | 
					
						
							|  |  |  | 	      if (exit_immediately_on_error && variable_context) | 
					
						
							|  |  |  | 	        { | 
					
						
							|  |  |  | 	          discard_unwind_frame ("pe_dispose"); | 
					
						
							|  |  |  | 		  variable_context = 0;	/* not in a function */ | 
					
						
							|  |  |  | 	        } | 
					
						
							|  |  |  | 	      should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 	      goto out; | 
					
						
							|  |  |  | 	    case FORCE_EOF:	       | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    case EXITPROG: | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      if (command) | 
					
						
							|  |  |  | 		run_unwind_frame ("pe_dispose"); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      /* Remember to call longjmp (top_level) after the old
 | 
					
						
							|  |  |  | 		 value for it is restored. */ | 
					
						
							|  |  |  | 	      should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 	      goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    case DISCARD: | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      if (command) | 
					
						
							|  |  |  | 		run_unwind_frame ("pe_dispose"); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	      last_result = last_command_exit_value = EXECUTION_FAILURE; /* XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      if (subshell_environment) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 		  goto out; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											1997-09-22 20:22:27 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 		  dispose_command (command);	/* pe_dispose does this */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #if defined (HAVE_POSIX_SIGNALS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  sigprocmask (SIG_SETMASK, (sigset_t *)&pe_sigmask, (sigset_t *)NULL); | 
					
						
							| 
									
										
										
										
											1997-09-22 20:22:27 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  continue; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    default: | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	      command_error ("parse_and_execute", CMDERR_BADJUMP, code, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      break; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	   | 
					
						
							|  |  |  |       if (parse_command () == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if ((flags & SEVAL_PARSEONLY) || (interactive_shell == 0 && read_but_dont_execute)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      last_result = EXECUTION_SUCCESS; | 
					
						
							|  |  |  | 	      dispose_command (global_command); | 
					
						
							|  |  |  | 	      global_command = (COMMAND *)NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (command = global_command) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      struct fd_bitmap *bitmap; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-10-05 19:12:20 -04:00
										 |  |  | 	      if (flags & SEVAL_FUNCDEF) | 
					
						
							| 
									
										
										
										
											2014-09-25 08:45:07 -04:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2014-10-05 19:12:20 -04:00
										 |  |  | 		  char *x; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		  /* If the command parses to something other than a straight
 | 
					
						
							|  |  |  | 		     function definition, or if we have not consumed the entire | 
					
						
							|  |  |  | 		     string, or if the parser has transformed the function | 
					
						
							|  |  |  | 		     name (as parsing will if it begins or ends with shell | 
					
						
							|  |  |  | 		     whitespace, for example), reject the attempt */ | 
					
						
							|  |  |  | 		  if (command->type != cm_function_def || | 
					
						
							|  |  |  | 		      ((x = parser_remaining_input ()) && *x) || | 
					
						
							|  |  |  | 		      (STREQ (from_file, command->value.Function_def->name->word) == 0)) | 
					
						
							|  |  |  | 		    { | 
					
						
							|  |  |  | 		      internal_warning (_("%s: ignoring function definition attempt"), from_file); | 
					
						
							|  |  |  | 		      should_jump_to_top_level = 0; | 
					
						
							|  |  |  | 		      last_result = last_command_exit_value = EX_BADUSAGE; | 
					
						
							|  |  |  | 		      reset_parser (); | 
					
						
							|  |  |  | 		      break; | 
					
						
							|  |  |  | 		    } | 
					
						
							| 
									
										
										
										
											2014-09-25 08:45:07 -04:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      bitmap = new_fd_bitmap (FD_BITMAP_SIZE); | 
					
						
							|  |  |  | 	      begin_unwind_frame ("pe_dispose"); | 
					
						
							|  |  |  | 	      add_unwind_protect (dispose_fd_bitmap, bitmap); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	      add_unwind_protect (dispose_command, command);	/* XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	      global_command = (COMMAND *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	      if ((subshell_environment & SUBSHELL_COMSUB) && comsub_ignore_return) | 
					
						
							|  |  |  | 		command->flags |= CMD_IGNORE_RETURN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ONESHOT)
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	      /*
 | 
					
						
							|  |  |  | 	       * IF | 
					
						
							|  |  |  | 	       *   we were invoked as `bash -c' (startup_state == 2) AND | 
					
						
							|  |  |  | 	       *   parse_and_execute has not been called recursively AND | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	       *   we're not running a trap AND | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	       *   we have parsed the full command (string == '\0') AND | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | 	       *   we're not going to run the exit trap AND | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	       *   we have a simple command without redirections AND | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	       *   the command is not being timed AND | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	       *   the command's return status is not being inverted AND | 
					
						
							|  |  |  | 	       *   there aren't any traps in effect | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	       * THEN | 
					
						
							|  |  |  | 	       *   tell the execution code that we don't need to fork | 
					
						
							|  |  |  | 	       */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      if (should_suppress_fork (command)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  command->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  | 		  command->value.Simple->flags |= CMD_NO_FORK; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      else if (command->type == cm_connection) | 
					
						
							|  |  |  | 		optimize_fork (command); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* ONESHOT */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      /* See if this is a candidate for $( <file ). */ | 
					
						
							|  |  |  | 	      if (startup_state == 2 && | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		  (subshell_environment & SUBSHELL_COMSUB) && | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		  *bash_input.location.string == '\0' && | 
					
						
							|  |  |  | 		  command->type == cm_simple && !command->redirects && | 
					
						
							|  |  |  | 		  (command->flags & CMD_TIME_PIPELINE) == 0 && | 
					
						
							|  |  |  | 		  command->value.Simple->words == 0 && | 
					
						
							|  |  |  | 		  command->value.Simple->redirects && | 
					
						
							|  |  |  | 		  command->value.Simple->redirects->next == 0 && | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 		  command->value.Simple->redirects->instruction == r_input_direction && | 
					
						
							|  |  |  | 		  command->value.Simple->redirects->redirector.dest == 0) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  int r; | 
					
						
							|  |  |  | 		  r = cat_file (command->value.Simple->redirects); | 
					
						
							|  |  |  | 		  last_result = (r < 0) ? EXECUTION_FAILURE : EXECUTION_SUCCESS; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		last_result = execute_command_internal | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 				(command, 0, NO_PIPE, NO_PIPE, bitmap); | 
					
						
							|  |  |  | 	      dispose_command (command); | 
					
						
							|  |  |  | 	      dispose_fd_bitmap (bitmap); | 
					
						
							|  |  |  | 	      discard_unwind_frame ("pe_dispose"); | 
					
						
							| 
									
										
										
										
											2014-09-25 08:45:07 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	      if (flags & SEVAL_ONECMD) | 
					
						
							| 
									
										
										
										
											2014-10-05 19:12:20 -04:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  reset_parser (); | 
					
						
							|  |  |  | 		  break; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  last_result = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  if (interactive_shell == 0 && this_shell_builtin && | 
					
						
							|  |  |  | 	      (this_shell_builtin == source_builtin || this_shell_builtin == eval_builtin) && | 
					
						
							|  |  |  | 	      last_command_exit_value == EX_BADSYNTAX && posixly_correct) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 	      code = ERREXIT; | 
					
						
							|  |  |  | 	      last_command_exit_value = EX_BADUSAGE; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* Since we are shell compatible, syntax errors in a script
 | 
					
						
							|  |  |  | 	     abort the execution of the script.  Right? */ | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  out: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   run_unwind_frame (PE_TAG); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (interrupt_state && parse_and_execute_level == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* An interrupt during non-interactive execution in an
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 interactive shell (e.g. via $PROMPT_COMMAND) should | 
					
						
							|  |  |  | 	 not cause the shell to exit. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       interactive = interactive_shell; | 
					
						
							|  |  |  |       throw_to_top_level (); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (should_jump_to_top_level) | 
					
						
							|  |  |  |     jump_to_top_level (code); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (last_result); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* Parse a command contained in STRING according to FLAGS and return the
 | 
					
						
							|  |  |  |    number of characters consumed from the string.  If non-NULL, set *ENDP | 
					
						
							|  |  |  |    to the position in the string where the parse ended.  Used to validate | 
					
						
							|  |  |  |    command substitutions during parsing to obey Posix rules about finding | 
					
						
							|  |  |  |    the end of the command and balancing parens. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | parse_string (string, from_file, flags, endp) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      const char *from_file; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  |      char **endp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int code, nc; | 
					
						
							|  |  |  |   volatile int should_jump_to_top_level; | 
					
						
							|  |  |  |   COMMAND *volatile command, *oglobal; | 
					
						
							|  |  |  |   char *ostring; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   volatile sigset_t ps_sigmask; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   parse_prologue (string, flags, PS_TAG); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #if defined (HAVE_POSIX_SIGNALS)
 | 
					
						
							|  |  |  |   /* If we longjmp and are going to go on, use this to restore signal mask */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   sigemptyset ((sigset_t *)&ps_sigmask); | 
					
						
							|  |  |  |   sigprocmask (SIG_BLOCK, (sigset_t *)NULL, (sigset_t *)&ps_sigmask); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /*itrace("parse_string: `%s'", string);*/ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   /* Reset the line number if the caller wants us to.  If we don't reset the
 | 
					
						
							|  |  |  |      line number, we have to subtract one, because we will add one just | 
					
						
							|  |  |  |      before executing the next command (resetting the line number sets it to | 
					
						
							|  |  |  |      0; the first line number is 1). */ | 
					
						
							|  |  |  |   push_stream (0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (parser_expanding_alias ()) | 
					
						
							|  |  |  |     /* push current shell_input_line */ | 
					
						
							|  |  |  |     parser_save_alias (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   code = should_jump_to_top_level = 0; | 
					
						
							|  |  |  |   oglobal = global_command; | 
					
						
							|  |  |  |   ostring = string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   with_input_from_string (string, from_file); | 
					
						
							|  |  |  |   while (*(bash_input.location.string)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       command = (COMMAND *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  |       if (interrupt_state) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Provide a location for functions which `longjmp (top_level)' to
 | 
					
						
							|  |  |  | 	 jump to. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       code = setjmp_nosigs (top_level); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (code) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | #if defined (DEBUG)
 | 
					
						
							|  |  |  | itrace("parse_string: longjmp executed: code = %d", code); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  should_jump_to_top_level = 0; | 
					
						
							|  |  |  | 	  switch (code) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	    case FORCE_EOF: | 
					
						
							|  |  |  | 	    case ERREXIT: | 
					
						
							|  |  |  | 	    case EXITPROG: | 
					
						
							|  |  |  | 	    case DISCARD:		/* XXX */ | 
					
						
							|  |  |  | 	      if (command) | 
					
						
							|  |  |  | 		dispose_command (command); | 
					
						
							|  |  |  | 	      /* Remember to call longjmp (top_level) after the old
 | 
					
						
							|  |  |  | 		 value for it is restored. */ | 
					
						
							|  |  |  | 	      should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 	      goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    default: | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #if defined (HAVE_POSIX_SIGNALS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      sigprocmask (SIG_SETMASK, (sigset_t *)&ps_sigmask, (sigset_t *)NULL); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	      command_error ("parse_string", CMDERR_BADJUMP, code, 0); | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	   | 
					
						
							|  |  |  |       if (parse_command () == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  dispose_command (global_command); | 
					
						
							|  |  |  | 	  global_command = (COMMAND *)NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if ((flags & SEVAL_NOLONGJMP) == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      should_jump_to_top_level = 1; | 
					
						
							|  |  |  | 	      code = DISCARD; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    reset_parser ();	/* XXX - sets token_to_read */ | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (current_token == yacc_EOF || current_token == shell_eof_token) | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  out: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   global_command = oglobal; | 
					
						
							|  |  |  |   nc = bash_input.location.string - ostring; | 
					
						
							|  |  |  |   if (endp) | 
					
						
							|  |  |  |     *endp = bash_input.location.string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   run_unwind_frame (PS_TAG); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* If we return < 0, the caller (xparse_dolparen) will jump_to_top_level for
 | 
					
						
							|  |  |  |      us, after doing cleanup */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (should_jump_to_top_level) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (parse_and_execute_level == 0) | 
					
						
							|  |  |  | 	top_level_cleanup (); | 
					
						
							|  |  |  |       if (code == DISCARD) | 
					
						
							|  |  |  | 	return -DISCARD; | 
					
						
							|  |  |  |       jump_to_top_level (code); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (nc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Handle a $( < file ) command substitution.  This expands the filename,
 | 
					
						
							|  |  |  |    returning errors as appropriate, then just cats the file to the standard | 
					
						
							|  |  |  |    output. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | cat_file (r) | 
					
						
							|  |  |  |      REDIRECT *r; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   char *fn; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int fd, rval; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (r->instruction != r_input_direction) | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Get the filename. */ | 
					
						
							|  |  |  |   if (posixly_correct && !interactive_shell) | 
					
						
							|  |  |  |     disallow_filename_globbing++; | 
					
						
							|  |  |  |   fn = redirection_expand (r->redirectee.filename); | 
					
						
							|  |  |  |   if (posixly_correct && !interactive_shell) | 
					
						
							|  |  |  |     disallow_filename_globbing--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (fn == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       redirection_error (r, AMBIGUOUS_REDIRECT); | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   fd = open(fn, O_RDONLY); | 
					
						
							|  |  |  |   if (fd < 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       file_error (fn); | 
					
						
							|  |  |  |       free (fn); | 
					
						
							|  |  |  |       return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   rval = zcatfd (fd, 1, fn); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   free (fn); | 
					
						
							|  |  |  |   close (fd); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   return (rval); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | evalstring (string, from_file, flags) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      const char *from_file; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   volatile int r, rflag, rcatch; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   rcatch = 0; | 
					
						
							|  |  |  |   rflag = return_catch_flag; | 
					
						
							|  |  |  |   /* If we are in a place where `return' is valid, we have to catch
 | 
					
						
							|  |  |  |      `eval "... return"' and make sure parse_and_execute cleans up. Then | 
					
						
							|  |  |  |      we can trampoline to the previous saved return_catch location. */ | 
					
						
							|  |  |  |   if (rflag) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       begin_unwind_frame ("evalstring"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       unwind_protect_int (return_catch_flag); | 
					
						
							|  |  |  |       unwind_protect_jmp_buf (return_catch); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return_catch_flag++;	/* increment so we have a counter */ | 
					
						
							|  |  |  |       rcatch = setjmp_nosigs (return_catch); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (rcatch) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       parse_and_execute_cleanup (); | 
					
						
							|  |  |  |       r = return_catch_value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     /* Note that parse_and_execute () frees the string it is passed. */ | 
					
						
							|  |  |  |     r = parse_and_execute (string, from_file, flags); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (rflag) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       run_unwind_frame ("evalstring"); | 
					
						
							|  |  |  |       if (rcatch && return_catch_flag) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  return_catch_value = r; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sh_longjmp (return_catch, 1); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |   return (r); | 
					
						
							|  |  |  | } |