| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | /* subst.c -- The part of the shell that does parameter, command, arithmetic,
 | 
					
						
							|  |  |  |    and globbing substitutions. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | /* ``Have a little faith, there's magic in the night.  You ain't a
 | 
					
						
							|  |  |  |      beauty, but, hey, you're alright.'' */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /* Copyright (C) 1987-2016 Free Software Foundation, Inc.
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +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-08-26 18:22:31 +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. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |    along with Bash.  If not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "config.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "bashtypes.h"
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include "chartypes.h"
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (HAVE_PWD_H)
 | 
					
						
							|  |  |  | #  include <pwd.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include <signal.h>
 | 
					
						
							|  |  |  | #include <errno.h>
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_UNISTD_H)
 | 
					
						
							|  |  |  | #  include <unistd.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #define NEED_FPURGE_DECL
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "bashansi.h"
 | 
					
						
							|  |  |  | #include "posixstat.h"
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #include "bashintl.h"
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "shell.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #include "parser.h"
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #include "flags.h"
 | 
					
						
							|  |  |  | #include "jobs.h"
 | 
					
						
							|  |  |  | #include "execute_cmd.h"
 | 
					
						
							|  |  |  | #include "filecntl.h"
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "trap.h"
 | 
					
						
							|  |  |  | #include "pathexp.h"
 | 
					
						
							|  |  |  | #include "mailcheck.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #include "shmbutil.h"
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (HAVE_MBSTR_H) && defined (HAVE_MBSCHR)
 | 
					
						
							|  |  |  | #  include <mbstr.h>		/* mbschr */
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #include "typemax.h"
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #include "builtins/getopt.h"
 | 
					
						
							|  |  |  | #include "builtins/common.h"
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #include "builtins/builtext.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #include <tilde/tilde.h>
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #include <glob/strmatch.h>
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if !defined (errno)
 | 
					
						
							|  |  |  | extern int errno; | 
					
						
							|  |  |  | #endif /* !errno */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* The size that strings change by. */ | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #define DEFAULT_INITIAL_ARRAY_SIZE 112
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #define DEFAULT_ARRAY_SIZE 128
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Variable types. */ | 
					
						
							|  |  |  | #define VT_VARIABLE	0
 | 
					
						
							|  |  |  | #define VT_POSPARMS	1
 | 
					
						
							|  |  |  | #define VT_ARRAYVAR	2
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #define VT_ARRAYMEMBER	3
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #define VT_ASSOCVAR	4
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #define VT_STARSUB	128	/* $* or ${array[*]} -- used to split */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Flags for quoted_strchr */ | 
					
						
							|  |  |  | #define ST_BACKSL	0x01
 | 
					
						
							|  |  |  | #define ST_CTLESC	0x02
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #define ST_SQUOTE	0x04	/* unused yet */
 | 
					
						
							|  |  |  | #define ST_DQUOTE	0x08	/* unused yet */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* These defs make it easier to use the editor. */ | 
					
						
							|  |  |  | #define LBRACE		'{'
 | 
					
						
							|  |  |  | #define RBRACE		'}'
 | 
					
						
							|  |  |  | #define LPAREN		'('
 | 
					
						
							|  |  |  | #define RPAREN		')'
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #define LBRACK		'['
 | 
					
						
							|  |  |  | #define RBRACK		']'
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | #define WLPAREN		L'('
 | 
					
						
							|  |  |  | #define WRPAREN		L')'
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #define DOLLAR_AT_STAR(c)	((c) == '@' || (c) == '*')
 | 
					
						
							|  |  |  | #define STR_DOLLAR_AT_STAR(s)	(DOLLAR_AT_STAR ((s)[0]) && (s)[1] == '\0')
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | /* Evaluates to 1 if C is one of the shell's special parameters whose length
 | 
					
						
							|  |  |  |    can be taken, but is also one of the special expansion characters. */ | 
					
						
							|  |  |  | #define VALID_SPECIAL_LENGTH_PARAM(c) \
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   ((c) == '-' || (c) == '?' || (c) == '#' || (c) == '@') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Evaluates to 1 if C is one of the shell's special parameters for which an
 | 
					
						
							|  |  |  |    indirect variable reference may be made. */ | 
					
						
							|  |  |  | #define VALID_INDIR_PARAM(c) \
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   ((posixly_correct == 0 && (c) == '#') || (posixly_correct == 0 && (c) == '?') || (c) == '@' || (c) == '*') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Evaluates to 1 if C is one of the OP characters that follows the parameter
 | 
					
						
							|  |  |  |    in ${parameter[:]OPword}. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #define VALID_PARAM_EXPAND_CHAR(c) (sh_syntaxtab[(unsigned char)c] & CSUBSTOP)
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | /* Evaluates to 1 if this is one of the shell's special variables. */ | 
					
						
							|  |  |  | #define SPECIAL_VAR(name, wi) \
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |  (*name && ((DIGIT (*name) && all_digits (name)) || \ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       (name[1] == '\0' && (sh_syntaxtab[(unsigned char)*name] & CSPECVAR)) || \ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       (wi && name[2] == '\0' && VALID_INDIR_PARAM (name[1])))) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This can be used by all of the *_extract_* functions that have a similar
 | 
					
						
							|  |  |  |    structure.  It can't just be wrapped in a do...while(0) loop because of | 
					
						
							|  |  |  |    the embedded `break'. The dangling else accommodates a trailing semicolon; | 
					
						
							|  |  |  |    we could also put in a do ; while (0) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     | 
					
						
							|  |  |  | #define CHECK_STRING_OVERRUN(oind, ind, len, ch) \
 | 
					
						
							|  |  |  |   if (ind >= len) \ | 
					
						
							|  |  |  |     { \ | 
					
						
							|  |  |  |       oind = len; \ | 
					
						
							|  |  |  |       ch = 0; \ | 
					
						
							|  |  |  |       break; \ | 
					
						
							|  |  |  |     } \ | 
					
						
							|  |  |  |   else \ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | /* An expansion function that takes a string and a quoted flag and returns
 | 
					
						
							|  |  |  |    a WORD_LIST *.  Used as the type of the third argument to | 
					
						
							|  |  |  |    expand_string_if_necessary(). */ | 
					
						
							|  |  |  | typedef WORD_LIST *EXPFUNC __P((char *, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Process ID of the last command executed within command substitution. */ | 
					
						
							|  |  |  | pid_t last_command_subst_pid = NO_PID; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | pid_t current_command_subst_pid = NO_PID; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* Variables used to keep track of the characters in IFS. */ | 
					
						
							|  |  |  | SHELL_VAR *ifs_var; | 
					
						
							|  |  |  | char *ifs_value; | 
					
						
							|  |  |  | unsigned char ifs_cmap[UCHAR_MAX + 1]; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | int ifs_is_set, ifs_is_null; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | unsigned char ifs_firstc[MB_LEN_MAX]; | 
					
						
							|  |  |  | size_t ifs_firstc_len; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | unsigned char ifs_firstc; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /* If non-zero, command substitution inherits the value of errexit option */ | 
					
						
							|  |  |  | int inherit_errexit = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | /* Sentinel to tell when we are performing variable assignments preceding a
 | 
					
						
							|  |  |  |    command name and putting them into the environment.  Used to make sure | 
					
						
							|  |  |  |    we use the temporary environment when looking up variable values. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | int assigning_in_environment; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | /* Used to hold a list of variable assignments preceding a command.  Global
 | 
					
						
							|  |  |  |    so the SIGCHLD handler in jobs.c can unwind-protect it when it runs a | 
					
						
							|  |  |  |    SIGCHLD trap and so it can be saved and restored by the trap handlers. */ | 
					
						
							|  |  |  | WORD_LIST *subst_assign_varlist = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /* Tell the expansion functions to not longjmp back to top_level on fatal
 | 
					
						
							|  |  |  |    errors.  Enabled when doing completion and prompt string expansion. */ | 
					
						
							|  |  |  | int no_longjmp_on_fatal_error = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Extern functions and variables from different files. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | extern int last_command_exit_value, last_command_exit_signal; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | extern int subshell_environment, running_in_background; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | extern int subshell_level, parse_and_execute_level, sourcelevel; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | extern int eof_encountered, line_number; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | extern int return_catch_flag, return_catch_value; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | extern pid_t dollar_dollar_pid; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | extern int posixly_correct; | 
					
						
							|  |  |  | extern char *this_command_name; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | extern struct fd_bitmap *current_fds_to_close; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | extern int wordexp_only; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | extern int expanding_redir; | 
					
						
							|  |  |  | extern int tempenv_assign_error; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | extern int builtin_ignoring_errexit; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (JOB_CONTROL) && defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  | extern PROCESS *last_procsub_child; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #if !defined (HAVE_WCSDUP) && defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | extern wchar_t *wcsdup __P((const wchar_t *)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Non-zero means to allow unmatched globbed filenames to expand to
 | 
					
						
							|  |  |  |    a null file. */ | 
					
						
							|  |  |  | int allow_null_glob_expansion; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | /* Non-zero means to throw an error when globbing fails to match anything. */ | 
					
						
							|  |  |  | int fail_glob_expansion; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Variables to keep track of which words in an expanded word list (the
 | 
					
						
							|  |  |  |    output of expand_word_list_internal) are the result of globbing | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |    expansions.  GLOB_ARGV_FLAGS is used by execute_cmd.c. | 
					
						
							|  |  |  |    (CURRENTLY UNUSED). */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char *glob_argv_flags; | 
					
						
							|  |  |  | static int glob_argv_flags_size; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static WORD_LIST *cached_quoted_dollar_at = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static WORD_LIST expand_word_error, expand_word_fatal; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC expand_wdesc_error, expand_wdesc_fatal; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char expand_param_error, expand_param_fatal; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static char extract_string_error, extract_string_fatal; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | /* Set by expand_word_unsplit; used to inhibit splitting and re-joining
 | 
					
						
							|  |  |  |    $* on $IFS, primarily when doing assignment statements. */ | 
					
						
							|  |  |  | static int expand_no_split_dollar_star = 0; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* A WORD_LIST of words to be expanded by expand_word_list_internal,
 | 
					
						
							|  |  |  |    without any leading variable assignments. */ | 
					
						
							|  |  |  | static WORD_LIST *garglist = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *quoted_substring __P((char *, int, int)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static int quoted_strlen __P((char *)); | 
					
						
							|  |  |  | static char *quoted_strchr __P((char *, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *expand_string_if_necessary __P((char *, int, EXPFUNC *)); | 
					
						
							|  |  |  | static inline char *expand_string_to_string_internal __P((char *, int, EXPFUNC *)); | 
					
						
							|  |  |  | static WORD_LIST *call_expand_word_internal __P((WORD_DESC *, int, int, int *, int *)); | 
					
						
							|  |  |  | static WORD_LIST *expand_string_internal __P((char *, int)); | 
					
						
							|  |  |  | static WORD_LIST *expand_string_leave_quoted __P((char *, int)); | 
					
						
							|  |  |  | static WORD_LIST *expand_string_for_rhs __P((char *, int, int *, int *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST *list_quote_escapes __P((WORD_LIST *)); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | static WORD_LIST *list_dequote_escapes __P((WORD_LIST *)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *make_quoted_char __P((int)); | 
					
						
							|  |  |  | static WORD_LIST *quote_list __P((WORD_LIST *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int unquoted_substring __P((char *, char *)); | 
					
						
							|  |  |  | static int unquoted_member __P((int, char *)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static SHELL_VAR *do_compound_assignment __P((char *, char *, int)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | static int do_assignment_internal __P((const WORD_DESC *, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static char *string_extract_verbatim __P((char *, size_t, int *, char *, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *string_extract __P((char *, int *, char *, int)); | 
					
						
							|  |  |  | static char *string_extract_double_quoted __P((char *, int *, int)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static inline char *string_extract_single_quoted __P((char *, int *)); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static inline int skip_single_quoted __P((const char *, size_t, int, int)); | 
					
						
							|  |  |  | static int skip_double_quoted __P((char *, size_t, int, int)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static char *extract_delimited_string __P((char *, int *, char *, char *, char *, int)); | 
					
						
							|  |  |  | static char *extract_dollar_brace_string __P((char *, int *, int, int)); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | static int skip_matched_pair __P((const char *, int, int, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *pos_params __P((char *, int, int, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static unsigned char *mb_getcharlens __P((char *, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char *remove_upattern __P((char *, char *, int)); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE) 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static wchar_t *remove_wpattern __P((wchar_t *, size_t, wchar_t *, int)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *remove_pattern __P((char *, char *, int)); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int match_upattern __P((char *, char *, int, char **, char **)); | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | static int match_wpattern __P((wchar_t *, char **, size_t, wchar_t *, int, char **, char **)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static int match_pattern __P((char *, char *, int, char **, char **)); | 
					
						
							|  |  |  | static int getpatspec __P((int, char *)); | 
					
						
							|  |  |  | static char *getpattern __P((char *, int, int)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static char *variable_remove_pattern __P((char *, char *, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *list_remove_pattern __P((WORD_LIST *, char *, int, int, int)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static char *parameter_list_remove_pattern __P((int, char *, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #ifdef ARRAY_VARS
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static char *array_remove_pattern __P((SHELL_VAR *, char *, int, char *, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static char *parameter_brace_remove_pattern __P((char *, char *, int, char *, int, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static char *string_var_assignment __P((SHELL_VAR *, char *)); | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static char *array_var_assignment __P((SHELL_VAR *, int, int)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | static char *pos_params_assignment __P((WORD_LIST *, int, int)); | 
					
						
							|  |  |  | static char *string_transform __P((int, SHELL_VAR *, char *)); | 
					
						
							|  |  |  | static char *list_transform __P((int, SHELL_VAR *, WORD_LIST *, int, int)); | 
					
						
							|  |  |  | static char *parameter_list_transform __P((int, int, int)); | 
					
						
							|  |  |  | #if defined ARRAY_VARS
 | 
					
						
							|  |  |  | static char *array_transform __P((int, SHELL_VAR *, char *, int)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | static char *parameter_brace_transform __P((char *, char *, int, char *, int, int, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static char *process_substitute __P((char *, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static char *read_comsub __P((int, int, int *)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef ARRAY_VARS
 | 
					
						
							|  |  |  | static arrayind_t array_length_reference __P((char *)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int valid_brace_expansion_word __P((char *, int)); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static int chk_atstar __P((char *, int, int *, int *)); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | static int chk_arithsub __P((const char *, int)); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static WORD_DESC *parameter_brace_expand_word __P((char *, int, int, int, arrayind_t *)); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | static char *parameter_brace_find_indir __P((char *, int, int, int)); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC *parameter_brace_expand_indir __P((char *, int, int, int *, int *)); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static WORD_DESC *parameter_brace_expand_rhs __P((char *, char *, int, int, int, int *, int *)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static void parameter_brace_expand_error __P((char *, char *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int valid_length_expression __P((char *)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static intmax_t parameter_brace_expand_length __P((char *)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *skiparith __P((char *, int)); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static int verify_substring_values __P((SHELL_VAR *, char *, char *, int, intmax_t *, intmax_t *)); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static int get_var_and_type __P((char *, char *, arrayind_t, int, int, SHELL_VAR **, char **)); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static char *mb_substring __P((char *, int, int)); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static char *parameter_brace_substring __P((char *, char *, int, char *, int, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int shouldexp_replacement __P((char *)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *pos_params_pat_subst __P((char *, char *, char *, int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static char *parameter_brace_patsub __P((char *, char *, int, char *, int, int, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | static char *pos_params_casemod __P((char *, char *, int, int)); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static char *parameter_brace_casemod __P((char *, char *, int, int, char *, int, int)); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | static WORD_DESC *parameter_brace_expand __P((char *, int *, int, int, int *, int *)); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC *param_expand __P((char *, int *, int, int *, int *, int *, int *, int)); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST *expand_word_internal __P((WORD_DESC *, int, int, int *, int *)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST *word_list_split __P((WORD_LIST *)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static void exp_jump_to_top_level __P((int)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static WORD_LIST *separate_out_assignments __P((WORD_LIST *)); | 
					
						
							|  |  |  | static WORD_LIST *glob_expand_word_list __P((WORD_LIST *, int)); | 
					
						
							|  |  |  | #ifdef BRACE_EXPANSION
 | 
					
						
							|  |  |  | static WORD_LIST *brace_expand_word_list __P((WORD_LIST *, int)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | static int make_internal_declare __P((char *, char *, char *)); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static WORD_LIST *shell_expand_word_list __P((WORD_LIST *, int)); | 
					
						
							|  |  |  | static WORD_LIST *expand_word_list_internal __P((WORD_LIST *, int)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /*			Utility Functions			    */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (DEBUG)
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | dump_word_flags (flags) | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   f = flags; | 
					
						
							|  |  |  |   fprintf (stderr, "%d -> ", f); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (f & W_ARRAYIND) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ARRAYIND; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ARRAYIND%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (f & W_ASSIGNASSOC) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSIGNASSOC; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSIGNASSOC%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (f & W_ASSIGNARRAY) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSIGNARRAY; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSIGNARRAY%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (f & W_HASCTLESC) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_HASCTLESC; | 
					
						
							|  |  |  |       fprintf (stderr, "W_HASCTLESC%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOPROCSUB) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOPROCSUB; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOPROCSUB%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_DQUOTE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_DQUOTE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_DQUOTE%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_HASQUOTEDNULL) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_HASQUOTEDNULL; | 
					
						
							|  |  |  |       fprintf (stderr, "W_HASQUOTEDNULL%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_ASSIGNARG) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSIGNARG; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSIGNARG%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_ASSNBLTIN) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSNBLTIN; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSNBLTIN%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2012-05-07 16:22:49 -04:00
										 |  |  |   if (f & W_ASSNGLOBAL) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSNGLOBAL; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSNGLOBAL%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (f & W_COMPASSIGN) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_COMPASSIGN; | 
					
						
							|  |  |  |       fprintf (stderr, "W_COMPASSIGN%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOEXPAND) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOEXPAND; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOEXPAND%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_ITILDE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ITILDE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ITILDE%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOTILDE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOTILDE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOTILDE%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_ASSIGNRHS) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSIGNRHS; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSIGNRHS%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOCOMSUB) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOCOMSUB; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOCOMSUB%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_DOLLARSTAR) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_DOLLARSTAR; | 
					
						
							|  |  |  |       fprintf (stderr, "W_DOLLARSTAR%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_DOLLARAT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_DOLLARAT; | 
					
						
							|  |  |  |       fprintf (stderr, "W_DOLLARAT%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_TILDEEXP) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_TILDEEXP; | 
					
						
							|  |  |  |       fprintf (stderr, "W_TILDEEXP%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOSPLIT2) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOSPLIT2; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOSPLIT2%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOSPLIT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOSPLIT; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOSPLIT%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (f & W_NOBRACE) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       f &= ~W_NOBRACE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOBRACE%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_NOGLOB) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_NOGLOB; | 
					
						
							|  |  |  |       fprintf (stderr, "W_NOGLOB%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_SPLITSPACE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_SPLITSPACE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_SPLITSPACE%s", f ? "|" : ""); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |   if (f & W_ASSIGNMENT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_ASSIGNMENT; | 
					
						
							|  |  |  |       fprintf (stderr, "W_ASSIGNMENT%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_QUOTED) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_QUOTED; | 
					
						
							|  |  |  |       fprintf (stderr, "W_QUOTED%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   if (f & W_HASDOLLAR) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_HASDOLLAR; | 
					
						
							|  |  |  |       fprintf (stderr, "W_HASDOLLAR%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (f & W_COMPLETE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       f &= ~W_COMPLETE; | 
					
						
							|  |  |  |       fprintf (stderr, "W_COMPLETE%s", f ? "|" : ""); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   fprintf (stderr, "\n"); | 
					
						
							|  |  |  |   fflush (stderr); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #ifdef INCLUDE_UNUSED
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static char * | 
					
						
							|  |  |  | quoted_substring (string, start, end) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int start, end; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int len, l; | 
					
						
							|  |  |  |   register char *result, *s, *r; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   len = end - start; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Move to string[start], skipping quoted characters. */ | 
					
						
							|  |  |  |   for (s = string, l = 0; *s && l < start; ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*s == CTLESC) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  s++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       l++; | 
					
						
							|  |  |  |       if (*s == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   r = result = (char *)xmalloc (2*len + 1);      /* save room for quotes */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Copy LEN characters, including quote characters. */ | 
					
						
							|  |  |  |   s = string + l; | 
					
						
							|  |  |  |   for (l = 0; l < len; s++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*s == CTLESC) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	*r++ = *s++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       *r++ = *s; | 
					
						
							|  |  |  |       l++; | 
					
						
							|  |  |  |       if (*s == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   *r = '\0'; | 
					
						
							|  |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef INCLUDE_UNUSED
 | 
					
						
							|  |  |  | /* Return the length of S, skipping over quoted characters */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | quoted_strlen (s) | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *p; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = 0; | 
					
						
							|  |  |  |   for (p = s; *p; p++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*p == CTLESC) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  p++; | 
					
						
							|  |  |  | 	  if (*p == 0) | 
					
						
							|  |  |  | 	    return (i + 1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       i++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return i; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Find the first occurrence of character C in string S, obeying shell
 | 
					
						
							|  |  |  |    quoting rules.  If (FLAGS & ST_BACKSL) is non-zero, backslash-escaped | 
					
						
							|  |  |  |    characters are skipped.  If (FLAGS & ST_CTLESC) is non-zero, characters | 
					
						
							|  |  |  |    escaped with CTLESC are skipped. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | quoted_strchr (s, c, flags) | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  |      int c, flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (p = s; *p; p++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (((flags & ST_BACKSL) && *p == '\\') | 
					
						
							|  |  |  | 	    || ((flags & ST_CTLESC) && *p == CTLESC)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  p++; | 
					
						
							|  |  |  | 	  if (*p == '\0') | 
					
						
							|  |  |  | 	    return ((char *)NULL); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (*p == c) | 
					
						
							|  |  |  | 	return p; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return ((char *)NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (INCLUDE_UNUSED)
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Return 1 if CHARACTER appears in an unquoted portion of
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    STRING.  Return 0 otherwise.  CHARACTER must be a single-byte character. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | unquoted_member (character, string) | 
					
						
							|  |  |  |      int character; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int sindex, c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   sindex = 0; | 
					
						
							|  |  |  |   while (c = string[sindex]) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (c == character) | 
					
						
							|  |  |  | 	return (1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       switch (c) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, sindex); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\\': | 
					
						
							|  |  |  | 	  sindex++; | 
					
						
							|  |  |  | 	  if (string[sindex]) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	    ADVANCE_CHAR (string, slen, sindex); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\'': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sindex = skip_single_quoted (string, slen, ++sindex, 0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '"': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sindex = skip_double_quoted (string, slen, ++sindex, 0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Return 1 if SUBSTR appears in an unquoted portion of STRING. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | unquoted_substring (substr, string) | 
					
						
							|  |  |  |      char *substr, *string; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int sindex, c, sublen; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (substr == 0 || *substr == '\0') | 
					
						
							|  |  |  |     return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   sublen = strlen (substr); | 
					
						
							|  |  |  |   for (sindex = 0; c = string[sindex]; ) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (STREQN (string + sindex, substr, sublen)) | 
					
						
							|  |  |  | 	return (1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       switch (c) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case '\\': | 
					
						
							|  |  |  | 	  sindex++; | 
					
						
							|  |  |  | 	  if (string[sindex]) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	    ADVANCE_CHAR (string, slen, sindex); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\'': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sindex = skip_single_quoted (string, slen, ++sindex, 0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '"': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sindex = skip_double_quoted (string, slen, ++sindex, 0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, sindex); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Most of the substitutions must be done in parallel.  In order
 | 
					
						
							|  |  |  |    to avoid using tons of unclear goto's, I have some functions | 
					
						
							|  |  |  |    for manipulating malloc'ed strings.  They all take INDX, a | 
					
						
							|  |  |  |    pointer to an integer which is the offset into the string | 
					
						
							|  |  |  |    where manipulation is taking place.  They also take SIZE, a | 
					
						
							|  |  |  |    pointer to an integer which is the current length of the | 
					
						
							|  |  |  |    character array for this string. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Append SOURCE to TARGET at INDEX.  SIZE is the current amount
 | 
					
						
							|  |  |  |    of space allocated to TARGET.  SOURCE can be NULL, in which | 
					
						
							|  |  |  |    case nothing happens.  Gets rid of SOURCE by freeing it. | 
					
						
							|  |  |  |    Returns TARGET in case the location has changed. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | INLINE char * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | sub_append_string (source, target, indx, size) | 
					
						
							|  |  |  |      char *source, *target; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int *indx; | 
					
						
							|  |  |  |      size_t *size; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   if (source) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       int n; | 
					
						
							|  |  |  |       size_t srclen; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       srclen = STRLEN (source); | 
					
						
							|  |  |  |       if (srclen >= (int)(*size - *indx)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  n = srclen + *indx; | 
					
						
							|  |  |  | 	  n = (n + DEFAULT_ARRAY_SIZE) - (n % DEFAULT_ARRAY_SIZE); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	  target = (char *)xrealloc (target, (*size = n)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       FASTCOPY (source, target + *indx, srclen); | 
					
						
							|  |  |  |       *indx += srclen; | 
					
						
							|  |  |  |       target[*indx] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       free (source); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (target); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | /* UNUSED */ | 
					
						
							|  |  |  | /* Append the textual representation of NUMBER to TARGET.
 | 
					
						
							|  |  |  |    INDX and SIZE are as in SUB_APPEND_STRING. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | sub_append_number (number, target, indx, size) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      intmax_t number; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      char *target; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int *indx; | 
					
						
							|  |  |  |      size_t *size; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   char *temp; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   temp = itos (number); | 
					
						
							|  |  |  |   return (sub_append_string (temp, target, indx, size)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Extract a substring from STRING, starting at SINDEX and ending with
 | 
					
						
							|  |  |  |    one of the characters in CHARLIST.  Don't make the ending character | 
					
						
							|  |  |  |    part of the string.  Leave SINDEX pointing at the ending character. | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    Understand about backslashes in the string.  If (flags & SX_VARNAME) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    is non-zero, and array variables have been compiled into the shell, | 
					
						
							|  |  |  |    everything between a `[' and a corresponding `]' is skipped over. | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    If (flags & SX_NOALLOC) is non-zero, don't return the substring, just | 
					
						
							|  |  |  |    update SINDEX.  If (flags & SX_REQMATCH) is non-zero, the string must | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |    contain a closing character from CHARLIST. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | string_extract (string, sindex, charlist, flags) | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							|  |  |  |      char *charlist; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register int c, i; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   int found; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   char *temp; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   slen = (MB_CUR_MAX > 1) ? strlen (string + *sindex) + *sindex : 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = *sindex; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   found = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   while (c = string[i]) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (c == '\\') | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  if (string[i + 1]) | 
					
						
							|  |  |  | 	    i++; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       else if ((flags & SX_VARNAME) && c == '[') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  int ni; | 
					
						
							|  |  |  | 	  /* If this is an array subscript, skip over it and continue. */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  ni = skipsubscript (string, i, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (string[ni] == ']') | 
					
						
							|  |  |  | 	    i = ni; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       else if (MEMBER (c, charlist)) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  found = 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* If we had to have a matching delimiter and didn't find one, return an
 | 
					
						
							|  |  |  |      error and let the caller deal with it. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if ((flags & SX_REQMATCH) && found == 0) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       *sindex = i; | 
					
						
							|  |  |  |       return (&extract_string_error); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   temp = (flags & SX_NOALLOC) ? (char *)NULL : substring (string, *sindex, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   *sindex = i; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Extract the contents of STRING as if it is enclosed in double quotes.
 | 
					
						
							|  |  |  |    SINDEX, when passed in, is the offset of the character immediately | 
					
						
							|  |  |  |    following the opening double quote; on exit, SINDEX is left pointing after | 
					
						
							|  |  |  |    the closing double quote.  If STRIPDQ is non-zero, unquoted double | 
					
						
							|  |  |  |    quotes are stripped and the string is terminated by a null byte. | 
					
						
							|  |  |  |    Backslashes between the embedded double quotes are processed.  If STRIPDQ | 
					
						
							|  |  |  |    is zero, an unquoted `"' terminates the string. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | string_extract_double_quoted (string, sindex, flags) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int *sindex, flags; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *send; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int j, i, t; | 
					
						
							|  |  |  |   unsigned char c; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *temp, *ret;		/* The new string we return. */ | 
					
						
							|  |  |  |   int pass_next, backquote, si;	/* State variables for the machine. */ | 
					
						
							|  |  |  |   int dquote; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int stripdq; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string + *sindex) + *sindex; | 
					
						
							|  |  |  |   send = string + slen; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   stripdq = (flags & SX_STRIPDQ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   pass_next = backquote = dquote = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   temp = (char *)xmalloc (1 + slen - *sindex); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   j = 0; | 
					
						
							|  |  |  |   i = *sindex; | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Process a character that was quoted by a backslash. */ | 
					
						
							|  |  |  |       if (pass_next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* XXX - take another look at this in light of Interp 221 */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* Posix.2 sez:
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	     ``The backslash shall retain its special meaning as an escape | 
					
						
							|  |  |  | 	     character only when followed by one of the characters: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 		$	`	"	\	<newline>''. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	     If STRIPDQ is zero, we handle the double quotes here and let | 
					
						
							|  |  |  | 	     expand_word_internal handle the rest.  If STRIPDQ is non-zero, | 
					
						
							|  |  |  | 	     we have already been through one round of backslash stripping, | 
					
						
							|  |  |  | 	     and want to strip these backslashes only if DQUOTE is non-zero, | 
					
						
							|  |  |  | 	     indicating that we are inside an embedded double-quoted string. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	     /* If we are in an embedded quoted string, then don't strip
 | 
					
						
							|  |  |  | 		backslashes before characters for which the backslash | 
					
						
							|  |  |  | 		retains its special meaning, but remove backslashes in | 
					
						
							|  |  |  | 		front of other characters.  If we are not in an | 
					
						
							|  |  |  | 		embedded quoted string, don't strip backslashes at all. | 
					
						
							|  |  |  | 		This mess is necessary because the string was already | 
					
						
							|  |  |  | 		surrounded by double quotes (and sh has some really weird | 
					
						
							|  |  |  | 		quoting rules). | 
					
						
							|  |  |  | 		The returned string will be run through expansion as if | 
					
						
							|  |  |  | 		it were double-quoted. */ | 
					
						
							|  |  |  | 	  if ((stripdq == 0 && c != '"') || | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	      (stripdq && ((dquote && (sh_syntaxtab[c] & CBSDQUOTE)) || dquote == 0))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    temp[j++] = '\\'; | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | add_one_character: | 
					
						
							|  |  |  | 	  COPY_CHAR_I (temp, j, string, send, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* A backslash protects the next character.  The code just above
 | 
					
						
							|  |  |  | 	 handles preserving the backslash in front of any character but | 
					
						
							|  |  |  | 	 a double quote. */ | 
					
						
							|  |  |  |       if (c == '\\') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  pass_next++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Inside backquotes, ``the portion of the quoted string from the
 | 
					
						
							|  |  |  | 	 initial backquote and the characters up to the next backquote | 
					
						
							|  |  |  | 	 that is not preceded by a backslash, having escape characters | 
					
						
							|  |  |  | 	 removed, defines that command''. */ | 
					
						
							|  |  |  |       if (backquote) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (c == '`') | 
					
						
							|  |  |  | 	    backquote = 0; | 
					
						
							|  |  |  | 	  temp[j++] = c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (c == '`') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  temp[j++] = c; | 
					
						
							|  |  |  | 	  backquote++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Pass everything between `$(' and the matching `)' or a quoted
 | 
					
						
							|  |  |  | 	 ${ ... } pair through according to the Posix.2 specification. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (c == '$' && ((string[i + 1] == LPAREN) || (string[i + 1] == LBRACE))) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  int free_ret = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  si = i + 2; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  if (string[i + 1] == LPAREN) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	    ret = extract_command_subst (string, &si, (flags & SX_COMPLETE)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	    ret = extract_dollar_brace_string (string, &si, Q_DOUBLE_QUOTES, 0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  temp[j++] = '$'; | 
					
						
							|  |  |  | 	  temp[j++] = string[i + 1]; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  /* Just paranoia; ret will not be 0 unless no_longjmp_on_fatal_error
 | 
					
						
							|  |  |  | 	     is set. */ | 
					
						
							|  |  |  | 	  if (ret == 0 && no_longjmp_on_fatal_error) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      free_ret = 0; | 
					
						
							|  |  |  | 	      ret = string + i + 2; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  for (t = 0; ret[t]; t++, j++) | 
					
						
							|  |  |  | 	    temp[j] = ret[t]; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  temp[j] = string[si]; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (string[si]) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      j++; | 
					
						
							|  |  |  | 	      i = si + 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    i = si; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (free_ret) | 
					
						
							|  |  |  | 	    free (ret); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Add any character but a double quote to the quoted string we're
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 accumulating. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (c != '"') | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	goto add_one_character; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* c == '"' */ | 
					
						
							|  |  |  |       if (stripdq) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  dquote ^= 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   temp[j] = '\0'; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* Point to after the closing quote. */ | 
					
						
							|  |  |  |   if (c) | 
					
						
							|  |  |  |     i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   *sindex = i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This should really be another option to string_extract_double_quoted. */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | skip_double_quoted (string, slen, sind, flags) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      size_t slen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      int sind; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int c, i; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *ret; | 
					
						
							|  |  |  |   int pass_next, backquote, si; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   pass_next = backquote = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = sind; | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (backquote) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (c == '`') | 
					
						
							|  |  |  | 	    backquote = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '`') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  backquote++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       else if (c == '$' && ((string[i + 1] == LPAREN) || (string[i + 1] == LBRACE))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 2; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  if (string[i + 1] == LPAREN) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	    ret = extract_command_subst (string, &si, SX_NOALLOC|(flags&SX_COMPLETE)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	    ret = extract_dollar_brace_string (string, &si, Q_DOUBLE_QUOTES, SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  /* These can consume the entire string if they are unterminated */ | 
					
						
							|  |  |  | 	  CHECK_STRING_OVERRUN (i, si, slen, c); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c != '"') | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (c) | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Extract the contents of STRING as if it is enclosed in single quotes.
 | 
					
						
							|  |  |  |    SINDEX, when passed in, is the offset of the character immediately | 
					
						
							|  |  |  |    following the opening single quote; on exit, SINDEX is left pointing after | 
					
						
							|  |  |  |    the closing single quote. */ | 
					
						
							|  |  |  | static inline char * | 
					
						
							|  |  |  | string_extract_single_quoted (string, sindex) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   register int i; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *t; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Don't need slen for ADVANCE_CHAR unless multibyte chars possible. */ | 
					
						
							|  |  |  |   slen = (MB_CUR_MAX > 1) ? strlen (string + *sindex) + *sindex : 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = *sindex; | 
					
						
							|  |  |  |   while (string[i] && string[i] != '\'') | 
					
						
							|  |  |  |     ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   t = substring (string, *sindex, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (string[i]) | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  |   *sindex = i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (t); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /* Skip over a single-quoted string.  We overload the SX_COMPLETE flag to mean
 | 
					
						
							|  |  |  |    that we are splitting out words for completion and have encountered a $'...' | 
					
						
							|  |  |  |    string, which allows backslash-escaped single quotes. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static inline int | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | skip_single_quoted (string, slen, sind, flags) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |      const char *string; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      size_t slen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      int sind; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   register int c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   c = sind; | 
					
						
							|  |  |  |   while (string[c] && string[c] != '\'') | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       if ((flags & SX_COMPLETE) && string[c] == '\\' && string[c+1] == '\'' && string[c+2]) | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, c); | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, c); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   if (string[c]) | 
					
						
							|  |  |  |     c++; | 
					
						
							|  |  |  |   return c; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Just like string_extract, but doesn't hack backslashes or any of
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |    that other stuff.  Obeys CTLESC quoting.  Used to do splitting on $IFS. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | string_extract_verbatim (string, slen, sindex, charlist, flags) | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |      size_t slen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      int *sindex; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      char *charlist; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   register int i; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   size_t clen; | 
					
						
							|  |  |  |   wchar_t *wcharlist; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int c; | 
					
						
							|  |  |  |   char *temp; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if ((flags & SX_NOCTLESC) && charlist[0] == '\'' && charlist[1] == '\0') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       temp = string_extract_single_quoted (string, sindex); | 
					
						
							|  |  |  |       --*sindex;	/* leave *sindex at separator character */ | 
					
						
							|  |  |  |       return temp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   i = *sindex; | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  |   /* See how the MBLEN and ADVANCE_CHAR macros work to understand why we need
 | 
					
						
							|  |  |  |      this only if MB_CUR_MAX > 1. */ | 
					
						
							|  |  |  |   slen = (MB_CUR_MAX > 1) ? strlen (string + *sindex) + *sindex : 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   clen = strlen (charlist); | 
					
						
							|  |  |  |   wcharlist = 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |       size_t mblength; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if ((flags & SX_NOCTLESC) == 0 && c == CTLESC) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  i += 2; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       /* Even if flags contains SX_NOCTLESC, we let CTLESC quoting CTLNUL
 | 
					
						
							|  |  |  | 	 through, to protect the CTLNULs from later calls to | 
					
						
							|  |  |  | 	 remove_quoted_nulls. */ | 
					
						
							|  |  |  |       else if ((flags & SX_NOESCCTLNUL) == 0 && c == CTLESC && string[i+1] == CTLNUL) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  i += 2; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |       mblength = MBLEN (string + i, slen - i); | 
					
						
							|  |  |  |       if (mblength > 1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  wchar_t wc; | 
					
						
							|  |  |  | 	  mblength = mbtowc (&wc, string + i, slen - i); | 
					
						
							|  |  |  | 	  if (MB_INVALIDCH (mblength)) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (MEMBER (c, charlist)) | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (wcharlist == 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  size_t len; | 
					
						
							|  |  |  | 		  len = mbstowcs (wcharlist, charlist, 0); | 
					
						
							|  |  |  | 		  if (len == -1) | 
					
						
							|  |  |  | 		    len = 0; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 		  wcharlist = (wchar_t *)xmalloc (sizeof (wchar_t) * (len + 1)); | 
					
						
							|  |  |  | 		  mbstowcs (wcharlist, charlist, len + 1); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      if (wcschr (wcharlist, wc)) | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else		 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (MEMBER (c, charlist)) | 
					
						
							|  |  |  | 	break; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   FREE (wcharlist); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   temp = substring (string, *sindex, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   *sindex = i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Extract the $( construct in STRING, and return a new string.
 | 
					
						
							|  |  |  |    Start extracting at (SINDEX) as if we had just seen "$(". | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    Make (SINDEX) get the position of the matching ")". ) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |    XFLAGS is additional flags to pass to other extraction functions. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | extract_command_subst (string, sindex, xflags) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      int xflags; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   char *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string[*sindex] == LPAREN || (xflags & SX_COMPLETE)) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     return (extract_delimited_string (string, sindex, "$(", "(", ")", xflags|SX_COMMAND)); /*)*/ | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       xflags |= (no_longjmp_on_fatal_error ? SX_NOLONGJMP : 0); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       ret = xparse_dolparen (string, string+*sindex, sindex, xflags); | 
					
						
							|  |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | /* Extract the $[ construct in STRING, and return a new string. (])
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |    Start extracting at (SINDEX) as if we had just seen "$[". | 
					
						
							|  |  |  |    Make (SINDEX) get the position of the matching "]". */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | extract_arithmetic_subst (string, sindex) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   return (extract_delimited_string (string, sindex, "$[", "[", "]", 0)); /*]*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  | /* Extract the <( or >( construct in STRING, and return a new string.
 | 
					
						
							|  |  |  |    Start extracting at (SINDEX) as if we had just seen "<(". | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    Make (SINDEX) get the position of the matching ")". */ /*))*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  | extract_process_subst (string, starter, sindex, xflags) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      char *starter; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  |      int xflags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   return (extract_delimited_string (string, sindex, starter, "(", ")", SX_COMMAND)); | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  | #else
 | 
					
						
							|  |  |  |   xflags |= (no_longjmp_on_fatal_error ? SX_NOLONGJMP : 0); | 
					
						
							|  |  |  |   return (xparse_dolparen (string, string+*sindex, sindex, xflags)); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif /* PROCESS_SUBSTITUTION */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | /* This can be fooled by unquoted right parens in the passed string. If
 | 
					
						
							|  |  |  |    each caller verifies that the last character in STRING is a right paren, | 
					
						
							|  |  |  |    we don't even need to call extract_delimited_string. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | extract_array_assignment_list (string, sindex) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   int slen; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string);	/* ( */ | 
					
						
							|  |  |  |   if (string[slen - 1] == ')') | 
					
						
							|  |  |  |    { | 
					
						
							|  |  |  |       ret = substring (string, *sindex, slen - 1); | 
					
						
							|  |  |  |       *sindex = slen - 1; | 
					
						
							|  |  |  |       return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return 0;   | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Extract and create a new string from the contents of STRING, a
 | 
					
						
							|  |  |  |    character string delimited with OPENER and CLOSER.  SINDEX is | 
					
						
							|  |  |  |    the address of an int describing the current offset in STRING; | 
					
						
							|  |  |  |    it should point to just after the first OPENER found.  On exit, | 
					
						
							|  |  |  |    SINDEX gets the position of the last character of the matching CLOSER. | 
					
						
							|  |  |  |    If OPENER is more than a single character, ALT_OPENER, if non-null, | 
					
						
							|  |  |  |    contains a character string that can also match CLOSER and thus | 
					
						
							|  |  |  |    needs to be skipped. */ | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | extract_delimited_string (string, sindex, opener, alt_opener, closer, flags) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex; | 
					
						
							|  |  |  |      char *opener, *alt_opener, *closer; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int i, c, si; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *t, *result; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   int pass_character, nesting_level, in_comment; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int len_closer, len_opener, len_alt_opener; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string + *sindex) + *sindex; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   len_opener = STRLEN (opener); | 
					
						
							|  |  |  |   len_alt_opener = STRLEN (alt_opener); | 
					
						
							|  |  |  |   len_closer = STRLEN (closer); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   pass_character = in_comment = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   nesting_level = 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   i = *sindex; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   while (nesting_level) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       c = string[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* If a recursive call or a call to ADVANCE_CHAR leaves the index beyond
 | 
					
						
							|  |  |  | 	 the end of the string, catch it and cut the loop. */ | 
					
						
							|  |  |  |       if (i > slen) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  i = slen; | 
					
						
							|  |  |  | 	  c = string[i = slen]; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (c == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       if (in_comment) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (c == '\n') | 
					
						
							|  |  |  | 	    in_comment = 0; | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (pass_character)	/* previous char was backslash */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_character = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       /* Not exactly right yet; should handle shell metacharacters and
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	 multibyte characters, too.  See COMMENT_BEGIN define in parse.y */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if ((flags & SX_COMMAND) && c == '#' && (i == 0 || string[i - 1] == '\n' || shellblank (string[i - 1]))) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  |           in_comment = 1; | 
					
						
							|  |  |  |           ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (c == CTLESC || c == '\\') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  pass_character++; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       /* Process a nested command substitution, but only if we're parsing an
 | 
					
						
							|  |  |  | 	 arithmetic substitution. */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if ((flags & SX_COMMAND) && string[i] == '$' && string[i+1] == LPAREN) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           si = i + 2; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |           t = extract_command_subst (string, &si, flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |           i = si + 1; | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Process a nested OPENER. */ | 
					
						
							|  |  |  |       if (STREQN (string + i, opener, len_opener)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  si = i + len_opener; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  t = extract_delimited_string (string, &si, opener, alt_opener, closer, flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Process a nested ALT_OPENER */ | 
					
						
							|  |  |  |       if (len_alt_opener && STREQN (string + i, alt_opener, len_alt_opener)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  si = i + len_alt_opener; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  t = extract_delimited_string (string, &si, alt_opener, alt_opener, closer, flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* If the current substring terminates the delimited string, decrement
 | 
					
						
							|  |  |  | 	 the nesting level. */ | 
					
						
							|  |  |  |       if (STREQN (string + i, closer, len_closer)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i += len_closer - 1;	/* move to last byte of the closer */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  nesting_level--; | 
					
						
							|  |  |  | 	  if (nesting_level == 0) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* Pass old-style command substitution through verbatim. */ | 
					
						
							|  |  |  |       if (c == '`') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  t = string_extract (string, &si, "`", flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       /* Pass single-quoted and double-quoted strings through verbatim. */ | 
					
						
							|  |  |  |       if (c == '\'' || c == '"') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 1; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = (c == '\'') ? skip_single_quoted (string, slen, si, 0) | 
					
						
							|  |  |  | 			  : skip_double_quoted (string, slen, si, 0); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       /* move past this character, which was not special. */ | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (c == 0 && nesting_level) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (no_longjmp_on_fatal_error == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  report_error (_("bad substitution: no closing `%s' in %s"), closer, string); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  exp_jump_to_top_level (DISCARD); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  *sindex = i; | 
					
						
							|  |  |  | 	  return (char *)NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   si = i - *sindex - len_closer + 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (flags & SX_NOALLOC) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     result = (char *)NULL; | 
					
						
							|  |  |  |   else     | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       result = (char *)xmalloc (1 + si); | 
					
						
							|  |  |  |       strncpy (result, string + *sindex, si); | 
					
						
							|  |  |  |       result[si] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   *sindex = i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Extract a parameter expansion expression within ${ and } from STRING.
 | 
					
						
							|  |  |  |    Obey the Posix.2 rules for finding the ending `}': count braces while | 
					
						
							|  |  |  |    skipping over enclosed quoted strings and command substitutions. | 
					
						
							|  |  |  |    SINDEX is the address of an int describing the current offset in STRING; | 
					
						
							|  |  |  |    it should point to just after the first `{' found.  On exit, SINDEX | 
					
						
							|  |  |  |    gets the position of the matching `}'.  QUOTED is non-zero if this | 
					
						
							|  |  |  |    occurs inside double quotes. */ | 
					
						
							|  |  |  | /* XXX -- this is very similar to extract_delimited_string -- XXX */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | extract_dollar_brace_string (string, sindex, quoted, flags) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      int *sindex, quoted, flags; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   register int i, c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   int pass_character, nesting_level, si, dolbrace_state; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *result, *t; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   pass_character = 0; | 
					
						
							|  |  |  |   nesting_level = 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string + *sindex) + *sindex; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   /* The handling of dolbrace_state needs to agree with the code in parse.y:
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:59:44 -05:00
										 |  |  |      parse_matched_pair().  The different initial value is to handle the | 
					
						
							|  |  |  |      case where this function is called to parse the word in | 
					
						
							|  |  |  |      ${param op word} (SX_WORD). */ | 
					
						
							|  |  |  |   dolbrace_state = (flags & SX_WORD) ? DOLBRACE_WORD : DOLBRACE_PARAM; | 
					
						
							|  |  |  |   if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && (flags & SX_POSIXEXP)) | 
					
						
							|  |  |  |     dolbrace_state = DOLBRACE_QUOTE; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = *sindex; | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (pass_character) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  pass_character = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* CTLESCs and backslashes quote the next character. */ | 
					
						
							|  |  |  |       if (c == CTLESC || c == '\\') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  pass_character++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (string[i] == '$' && string[i+1] == LBRACE) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  nesting_level++; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i += 2; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (c == RBRACE) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  nesting_level--; | 
					
						
							|  |  |  | 	  if (nesting_level == 0) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Pass the contents of old-style command substitutions through
 | 
					
						
							|  |  |  | 	 verbatim. */ | 
					
						
							|  |  |  |       if (c == '`') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  si = i + 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  t = string_extract (string, &si, "`", flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  CHECK_STRING_OVERRUN (i, si, slen, c); | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* Pass the contents of new-style command substitutions and
 | 
					
						
							|  |  |  | 	 arithmetic substitutions through verbatim. */ | 
					
						
							|  |  |  |       if (string[i] == '$' && string[i+1] == LPAREN) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  si = i + 2; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  t = extract_command_subst (string, &si, flags|SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i = si + 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       /* Pass the contents of double-quoted strings through verbatim. */ | 
					
						
							|  |  |  |       if (c == '"') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 1; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = skip_double_quoted (string, slen, si, 0); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* skip_XXX_quoted leaves index one past close quote */ | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (c == '\'') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | /*itrace("extract_dollar_brace_string: c == single quote flags = %d quoted = %d dolbrace_state = %d", flags, quoted, dolbrace_state);*/ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  if (posixly_correct && shell_compatibility_level > 42 && dolbrace_state != DOLBRACE_QUOTE && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	    ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      si = i + 1; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      i = skip_single_quoted (string, slen, si, 0); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* move past this character, which was not special. */ | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* This logic must agree with parse.y:parse_matched_pair, since they
 | 
					
						
							|  |  |  | 	 share the same defines. */ | 
					
						
							|  |  |  |       if (dolbrace_state == DOLBRACE_PARAM && c == '%' && (i - *sindex) > 1) | 
					
						
							|  |  |  | 	dolbrace_state = DOLBRACE_QUOTE; | 
					
						
							|  |  |  |       else if (dolbrace_state == DOLBRACE_PARAM && c == '#' && (i - *sindex) > 1) | 
					
						
							|  |  |  |         dolbrace_state = DOLBRACE_QUOTE; | 
					
						
							|  |  |  |       else if (dolbrace_state == DOLBRACE_PARAM && c == '/' && (i - *sindex) > 1) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |         dolbrace_state = DOLBRACE_QUOTE2;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       else if (dolbrace_state == DOLBRACE_PARAM && c == '^' && (i - *sindex) > 1) | 
					
						
							|  |  |  |         dolbrace_state = DOLBRACE_QUOTE; | 
					
						
							|  |  |  |       else if (dolbrace_state == DOLBRACE_PARAM && c == ',' && (i - *sindex) > 1) | 
					
						
							|  |  |  |         dolbrace_state = DOLBRACE_QUOTE; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* This is intended to handle all of the [:]op expansions and the substring/
 | 
					
						
							|  |  |  | 	 length/pattern removal/pattern substitution expansions. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       else if (dolbrace_state == DOLBRACE_PARAM && strchr ("#%^,~:-=?+/", c) != 0) | 
					
						
							|  |  |  | 	dolbrace_state = DOLBRACE_OP; | 
					
						
							|  |  |  |       else if (dolbrace_state == DOLBRACE_OP && strchr ("#%^,~:-=?+/", c) == 0) | 
					
						
							|  |  |  | 	dolbrace_state = DOLBRACE_WORD; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (c == 0 && nesting_level) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (no_longjmp_on_fatal_error == 0) | 
					
						
							|  |  |  | 	{			/* { */ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  report_error (_("bad substitution: no closing `%s' in %s"), "}", string); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  exp_jump_to_top_level (DISCARD); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  *sindex = i; | 
					
						
							|  |  |  | 	  return ((char *)NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   result = (flags & SX_NOALLOC) ? (char *)NULL : substring (string, *sindex, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   *sindex = i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Remove backslashes which are quoting backquotes from STRING.  Modifies
 | 
					
						
							|  |  |  |    STRING, and returns a pointer to it. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | de_backslash (string) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   register size_t slen; | 
					
						
							|  |  |  |   register int i, j, prev_i; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   i = j = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Loop copying string[i] to string[j], i >= j. */ | 
					
						
							|  |  |  |   while (i < slen) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (string[i] == '\\' && (string[i + 1] == '`' || string[i + 1] == '\\' || | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 			      string[i + 1] == '$')) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	i++; | 
					
						
							|  |  |  |       prev_i = i; | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |       if (j < prev_i) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	do string[j++] = string[prev_i++]; while (prev_i < i); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	j = i; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   string[j] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (string); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /*UNUSED*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Replace instances of \! in a string with !. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | unquote_bang (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i, j; | 
					
						
							|  |  |  |   register char *temp; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   temp = (char *)xmalloc (1 + strlen (string)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (i = 0, j = 0; (temp[j] = string[i]); i++, j++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (string[i] == '\\' && string[i + 1] == '!') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  temp[j] = '!'; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   strcpy (string, temp); | 
					
						
							|  |  |  |   free (temp); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #define CQ_RETURN(x) do { no_longjmp_on_fatal_error = oldjmp; return (x); } while (0)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | /* This function assumes s[i] == open; returns with s[ret] == close; used to
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |    parse array subscripts.  FLAGS & 1 means to not attempt to skip over | 
					
						
							|  |  |  |    matched pairs of quotes or backquotes, or skip word expansions; it is | 
					
						
							|  |  |  |    intended to be used after expansion has been performed and during final | 
					
						
							|  |  |  |    assignment parsing (see arrayfunc.c:assign_compound_array_list()). */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | static int | 
					
						
							|  |  |  | skip_matched_pair (string, start, open, close, flags) | 
					
						
							|  |  |  |      const char *string; | 
					
						
							|  |  |  |      int start, open, close, flags; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int i, pass_next, backq, si, c, count, oldjmp; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *temp, *ss; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string + start) + start; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   oldjmp = no_longjmp_on_fatal_error; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  |   no_longjmp_on_fatal_error = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = start + 1;		/* skip over leading bracket */ | 
					
						
							|  |  |  |   count = 1; | 
					
						
							|  |  |  |   pass_next = backq = 0; | 
					
						
							|  |  |  |   ss = (char *)string; | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							|  |  |  | 	  if (c == 0) | 
					
						
							|  |  |  | 	    CQ_RETURN(i); | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (backq) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (c == '`') | 
					
						
							|  |  |  | 	    backq = 0; | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if ((flags & 1) == 0 && c == '`') | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  backq = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if ((flags & 1) == 0 && c == open) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  count++; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == close) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  count--; | 
					
						
							|  |  |  | 	  if (count == 0) | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if ((flags & 1) == 0 && (c == '\'' || c == '"')) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = (c == '\'') ? skip_single_quoted (ss, slen, ++i, 0) | 
					
						
							|  |  |  | 			  : skip_double_quoted (ss, slen, ++i, 0); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	  /* no increment, the skip functions increment past the closing quote. */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if ((flags&1) == 0 && c == '$' && (string[i+1] == LPAREN || string[i+1] == LBRACE)) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 2; | 
					
						
							|  |  |  | 	  if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (string[i+1] == LPAREN) | 
					
						
							|  |  |  | 	    temp = extract_delimited_string (ss, &si, "$(", "(", ")", SX_NOALLOC|SX_COMMAND); /* ) */ | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = extract_dollar_brace_string (ss, &si, 0, SX_NOALLOC); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  CHECK_STRING_OVERRUN (i, si, slen, c); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0')	/* don't increment i past EOS in loop */ | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CQ_RETURN(i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | skipsubscript (string, start, flags) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  |      const char *string; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |      int start, flags; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   return (skip_matched_pair (string, start, '[', ']', flags)); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* Skip characters in STRING until we find a character in DELIMS, and return
 | 
					
						
							|  |  |  |    the index of that character.  START is the index into string at which we | 
					
						
							|  |  |  |    begin.  This is similar in spirit to strpbrk, but it returns an index into | 
					
						
							|  |  |  |    STRING and takes a starting index.  This little piece of code knows quite | 
					
						
							|  |  |  |    a lot of shell syntax.  It's very similar to skip_double_quoted and other | 
					
						
							|  |  |  |    functions of that ilk. */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | skip_to_delim (string, start, delims, flags) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int start; | 
					
						
							|  |  |  |      char *delims; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int i, pass_next, backq, dquote, si, c, oldjmp; | 
					
						
							|  |  |  |   int invert, skipquote, skipcmd, noprocsub, completeflag; | 
					
						
							|  |  |  |   int arithexp, skipcol; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   char *temp, open[3]; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string + start) + start; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   oldjmp = no_longjmp_on_fatal_error; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (flags & SD_NOJMP) | 
					
						
							|  |  |  |     no_longjmp_on_fatal_error = 1; | 
					
						
							|  |  |  |   invert = (flags & SD_INVERT); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   skipcmd = (flags & SD_NOSKIPCMD) == 0; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   noprocsub = (flags & SD_NOPROCSUB); | 
					
						
							|  |  |  |   completeflag = (flags & SD_COMPLETE) ? SX_COMPLETE : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   arithexp = (flags & SD_ARITHEXP); | 
					
						
							|  |  |  |   skipcol = 0; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   i = start; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   pass_next = backq = dquote = 0; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   while (c = string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       /* If this is non-zero, we should not let quote characters be delimiters
 | 
					
						
							|  |  |  | 	 and the current character is a single or double quote.  We should not | 
					
						
							|  |  |  | 	 test whether or not it's a delimiter until after we skip single- or | 
					
						
							|  |  |  | 	 double-quoted strings. */ | 
					
						
							|  |  |  |       skipquote = ((flags & SD_NOQUOTEDELIM) && (c == '\'' || c =='"')); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							|  |  |  | 	  if (c == 0) | 
					
						
							|  |  |  | 	    CQ_RETURN(i); | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (backq) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (c == '`') | 
					
						
							|  |  |  | 	    backq = 0; | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '`') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  backq = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       else if (arithexp && skipcol && c == ':') | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  skipcol--; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       else if (arithexp && c == '?') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  skipcol++; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (skipquote == 0 && invert == 0 && member (c, delims)) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       /* the usual case is to use skip_xxx_quoted, but we don't skip over double
 | 
					
						
							|  |  |  | 	 quoted strings when looking for the history expansion character as a | 
					
						
							|  |  |  | 	 delimiter. */ | 
					
						
							|  |  |  |       /* special case for programmable completion which takes place before
 | 
					
						
							|  |  |  |          parser converts backslash-escaped single quotes between $'...' to | 
					
						
							|  |  |  |          `regular' single-quoted strings. */ | 
					
						
							|  |  |  |       else if (completeflag && i > 0 && string[i-1] == '$' && c == '\'') | 
					
						
							|  |  |  | 	i = skip_single_quoted (string, slen, ++i, SX_COMPLETE); | 
					
						
							|  |  |  |       else if (c == '\'') | 
					
						
							|  |  |  | 	i = skip_single_quoted (string, slen, ++i, 0); | 
					
						
							|  |  |  |       else if (c == '"') | 
					
						
							|  |  |  | 	i = skip_double_quoted (string, slen, ++i, completeflag); | 
					
						
							|  |  |  |       else if (c == LPAREN && arithexp) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           si = i + 1; | 
					
						
							|  |  |  |           if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  temp = extract_delimited_string (string, &si, "(", "(", ")", SX_NOALLOC); /* ) */ | 
					
						
							|  |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0')	/* don't increment i past EOS in loop */ | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue;          | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if (c == '$' && ((skipcmd && string[i+1] == LPAREN) || string[i+1] == LBRACE)) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 2; | 
					
						
							|  |  |  | 	  if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (string[i+1] == LPAREN) | 
					
						
							|  |  |  | 	    temp = extract_delimited_string (string, &si, "$(", "(", ")", SX_NOALLOC|SX_COMMAND); /* ) */ | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = extract_dollar_brace_string (string, &si, 0, SX_NOALLOC); | 
					
						
							|  |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0')	/* don't increment i past EOS in loop */ | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       else if (skipcmd && noprocsub == 0 && (c == '<' || c == '>') && string[i+1] == LPAREN) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 2; | 
					
						
							|  |  |  | 	  if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  | 	  temp = extract_delimited_string (string, &si, (c == '<') ? "<(" : ">(", "(", ")", SX_COMMAND|SX_NOALLOC); /* )) */ | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  | 	  temp = extract_process_subst (string, (c == '<') ? "<(" : ">(", &si, 0); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  free (temp);		/* XXX - not using SX_NOALLOC here yet */ | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0') | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif /* PROCESS_SUBSTITUTION */
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #if defined (EXTENDED_GLOB)
 | 
					
						
							|  |  |  |       else if ((flags & SD_EXTGLOB) && extended_glob && string[i+1] == LPAREN && member (c, "?*+!@")) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 2; | 
					
						
							|  |  |  | 	  if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  open[0] = c; | 
					
						
							|  |  |  | 	  open[1] = LPAREN; | 
					
						
							|  |  |  | 	  open[2] = '\0'; | 
					
						
							|  |  |  | 	  temp = extract_delimited_string (string, &si, open, "(", ")", SX_NOALLOC); /* ) */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0')	/* don't increment i past EOS in loop */ | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       else if ((flags & SD_GLOB) && c == LBRACK) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  si = i + 1; | 
					
						
							|  |  |  | 	  if (string[si] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(si); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  temp = extract_delimited_string (string, &si, "[", "[", "]", SX_NOALLOC); /* ] */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  i = si; | 
					
						
							|  |  |  | 	  if (string[i] == '\0')	/* don't increment i past EOS in loop */ | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if ((skipquote || invert) && (member (c, delims) == 0)) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	break; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CQ_RETURN(i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (BANG_HISTORY)
 | 
					
						
							|  |  |  | /* Skip to the history expansion character (delims[0]), paying attention to
 | 
					
						
							|  |  |  |    quoted strings and command and process substitution.  This is a stripped- | 
					
						
							|  |  |  |    down version of skip_to_delims.  The essential difference is that this | 
					
						
							|  |  |  |    resets the quoting state when starting a command substitution */ | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | skip_to_histexp (string, start, delims, flags) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int start; | 
					
						
							|  |  |  |      char *delims; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i, pass_next, backq, dquote, si, c, oldjmp; | 
					
						
							|  |  |  |   int histexp_comsub, histexp_backq, old_dquote; | 
					
						
							|  |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *temp, open[3]; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string + start) + start; | 
					
						
							|  |  |  |   oldjmp = no_longjmp_on_fatal_error; | 
					
						
							|  |  |  |   if (flags & SD_NOJMP) | 
					
						
							|  |  |  |     no_longjmp_on_fatal_error = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   histexp_comsub = histexp_backq = old_dquote = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = start; | 
					
						
							|  |  |  |   pass_next = backq = dquote = 0; | 
					
						
							|  |  |  |   while (c = string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							|  |  |  | 	  if (c == 0) | 
					
						
							|  |  |  | 	    CQ_RETURN(i); | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (backq && c == '`') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  backq = 0; | 
					
						
							|  |  |  | 	  histexp_backq--; | 
					
						
							|  |  |  | 	  dquote = old_dquote; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '`') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  backq = 1; | 
					
						
							|  |  |  | 	  histexp_backq++; | 
					
						
							|  |  |  | 	  old_dquote = dquote;		/* simple - one level for now */ | 
					
						
							|  |  |  | 	  dquote = 0; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       /* When in double quotes, act as if the double quote is a member of
 | 
					
						
							|  |  |  | 	 history_no_expand_chars, like the history library does */ | 
					
						
							|  |  |  |       else if (dquote && c == delims[0] && string[i+1] == '"') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == delims[0]) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       /* the usual case is to use skip_xxx_quoted, but we don't skip over double
 | 
					
						
							|  |  |  | 	 quoted strings when looking for the history expansion character as a | 
					
						
							|  |  |  | 	 delimiter. */ | 
					
						
							|  |  |  |       else if (dquote && c == '\'') | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (c == '\'') | 
					
						
							|  |  |  | 	i = skip_single_quoted (string, slen, ++i, 0); | 
					
						
							|  |  |  |       /* The posixly_correct test makes posix-mode shells allow double quotes
 | 
					
						
							|  |  |  | 	 to quote the history expansion character */ | 
					
						
							|  |  |  |       else if (posixly_correct == 0 && c == '"') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  dquote = 1 - dquote; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	}      | 
					
						
							|  |  |  |       else if (c == '"') | 
					
						
							|  |  |  | 	i = skip_double_quoted (string, slen, ++i, 0); | 
					
						
							|  |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  |       else if ((c == '$' || c == '<' || c == '>') && string[i+1] == LPAREN && string[i+2] != LPAREN) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |       else if (c == '$' && string[i+1] == LPAREN && string[i+2] != LPAREN) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  | 	  if (string[i+2] == '\0') | 
					
						
							|  |  |  | 	    CQ_RETURN(i+2); | 
					
						
							|  |  |  | 	  i += 2; | 
					
						
							|  |  |  | 	  histexp_comsub++; | 
					
						
							|  |  |  | 	  old_dquote = dquote; | 
					
						
							|  |  |  | 	  dquote = 0; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (histexp_comsub && c == RPAREN) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  histexp_comsub--; | 
					
						
							|  |  |  | 	  dquote = old_dquote; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (backq)		/* placeholder */ | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   CQ_RETURN(i); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* BANG_HISTORY */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (READLINE)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Return 1 if the portion of STRING ending at EINDEX is quoted (there is
 | 
					
						
							|  |  |  |    an unclosed quoted string), or if the character at EINDEX is quoted | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |    by a backslash. NO_LONGJMP_ON_FATAL_ERROR is used to flag that the various | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |    single and double-quoted string parsing functions should not return an | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    error if there are unclosed quotes or braces.  The characters that this | 
					
						
							|  |  |  |    recognizes need to be the same as the contents of | 
					
						
							|  |  |  |    rl_completer_quote_characters. */ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | int | 
					
						
							|  |  |  | char_is_quoted (string, eindex) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int eindex; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int i, pass_next, c, oldjmp; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   oldjmp = no_longjmp_on_fatal_error; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   no_longjmp_on_fatal_error = 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = pass_next = 0; | 
					
						
							|  |  |  |   while (i <= eindex) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       c = string[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							|  |  |  | 	  if (i >= eindex)	/* XXX was if (i >= eindex - 1) */ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	    CQ_RETURN(1); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else if (c == '\\') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       else if (c == '$' && string[i+1] == '\'' && string[i+2]) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  i += 2; | 
					
						
							|  |  |  | 	  i = skip_single_quoted (string, slen, i, SX_COMPLETE); | 
					
						
							|  |  |  | 	  if (i > eindex) | 
					
						
							|  |  |  | 	    CQ_RETURN (i); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else if (c == '\'' || c == '"') | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = (c == '\'') ? skip_single_quoted (string, slen, ++i, 0) | 
					
						
							|  |  |  | 			  : skip_double_quoted (string, slen, ++i, SX_COMPLETE); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  if (i > eindex) | 
					
						
							|  |  |  | 	    CQ_RETURN(1); | 
					
						
							|  |  |  | 	  /* no increment, the skip_xxx functions go one past end */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   CQ_RETURN(0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							|  |  |  | unclosed_pair (string, eindex, openstr) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int eindex; | 
					
						
							|  |  |  |      char *openstr; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int i, pass_next, openc, olen; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   olen = strlen (openstr); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = pass_next = openc = 0; | 
					
						
							|  |  |  |   while (i <= eindex) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (pass_next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 0; | 
					
						
							|  |  |  | 	  if (i >= eindex)	/* XXX was if (i >= eindex - 1) */ | 
					
						
							|  |  |  | 	    return 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (string[i] == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pass_next = 1; | 
					
						
							|  |  |  | 	  i++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (STREQN (string + i, openstr, olen)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  openc = 1 - openc; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  i += olen; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* XXX - may want to handle $'...' specially here */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (string[i] == '\'' || string[i] == '"') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = (string[i] == '\'') ? skip_single_quoted (string, slen, i, 0) | 
					
						
							|  |  |  | 				  : skip_double_quoted (string, slen, i, SX_COMPLETE); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  if (i > eindex) | 
					
						
							|  |  |  | 	    return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return (openc); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Split STRING (length SLEN) at DELIMS, and return a WORD_LIST with the
 | 
					
						
							|  |  |  |    individual words.  If DELIMS is NULL, the current value of $IFS is used | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |    to split the string, and the function follows the shell field splitting | 
					
						
							|  |  |  |    rules.  SENTINEL is an index to look for.  NWP, if non-NULL, | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |    gets the number of words in the returned list.  CWP, if non-NULL, gets | 
					
						
							|  |  |  |    the index of the word containing SENTINEL.  Non-whitespace chars in | 
					
						
							|  |  |  |    DELIMS delimit separate fields. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | split_at_delims (string, slen, delims, sentinel, flags, nwp, cwp) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int slen; | 
					
						
							|  |  |  |      char *delims; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |      int sentinel, flags; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |      int *nwp, *cwp; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   int ts, te, i, nw, cw, ifs_split, dflags; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   char *token, *d, *d2; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   WORD_LIST *ret, *tl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (nwp) | 
					
						
							|  |  |  | 	*nwp = 0; | 
					
						
							|  |  |  |       if (cwp) | 
					
						
							|  |  |  | 	*cwp = 0;	 | 
					
						
							|  |  |  |       return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   d = (delims == 0) ? ifs_value : delims; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   ifs_split = delims == 0; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Make d2 the non-whitespace characters in delims */ | 
					
						
							|  |  |  |   d2 = 0; | 
					
						
							|  |  |  |   if (delims) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       size_t slength; | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |       size_t mblength = 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       slength = strlen (delims); | 
					
						
							|  |  |  |       d2 = (char *)xmalloc (slength + 1); | 
					
						
							|  |  |  |       i = ts = 0; | 
					
						
							|  |  |  |       while (delims[i]) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	  mbstate_t state_bak; | 
					
						
							|  |  |  | 	  state_bak = state; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  mblength = MBRLEN (delims + i, slength, &state); | 
					
						
							|  |  |  | 	  if (MB_INVALIDCH (mblength)) | 
					
						
							|  |  |  | 	    state = state_bak; | 
					
						
							|  |  |  | 	  else if (mblength > 1) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      memcpy (d2 + ts, delims + i, mblength); | 
					
						
							|  |  |  | 	      ts += mblength; | 
					
						
							|  |  |  | 	      i += mblength; | 
					
						
							|  |  |  | 	      slength -= mblength; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  if (whitespace (delims[i]) == 0) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	    d2[ts++] = delims[i]; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  slength--; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       d2[ts] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   /* Remove sequences of whitespace characters at the start of the string, as
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |      long as those characters are delimiters. */ | 
					
						
							|  |  |  |   for (i = 0; member (string[i], d) && spctabnl (string[i]); i++) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |     ; | 
					
						
							|  |  |  |   if (string[i] == '\0') | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       FREE (d2); | 
					
						
							|  |  |  |       return (ret); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ts = i; | 
					
						
							|  |  |  |   nw = 0; | 
					
						
							|  |  |  |   cw = -1; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   dflags = flags|SD_NOJMP; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   while (1) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       te = skip_to_delim (string, ts, d, dflags); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* If we have a non-whitespace delimiter character, use it to make a
 | 
					
						
							|  |  |  | 	 separate field.  This is just about what $IFS splitting does and | 
					
						
							|  |  |  | 	 is closer to the behavior of the shell parser. */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       if (ts == te && d2 && member (string[ts], d2)) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  te = ts + 1; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  /* If we're using IFS splitting, the non-whitespace delimiter char
 | 
					
						
							|  |  |  | 	     and any additional IFS whitespace delimits a field. */ | 
					
						
							|  |  |  | 	  if (ifs_split) | 
					
						
							|  |  |  | 	    while (member (string[te], d) && spctabnl (string[te])) | 
					
						
							|  |  |  | 	      te++; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    while (member (string[te], d2)) | 
					
						
							|  |  |  | 	      te++; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       token = substring (string, ts, te); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ret = add_string_to_list (token, ret); | 
					
						
							|  |  |  |       free (token); | 
					
						
							|  |  |  |       nw++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (sentinel >= ts && sentinel <= te) | 
					
						
							|  |  |  | 	cw = nw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If the cursor is at whitespace just before word start, set the
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 sentinel word to the current word. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       if (cwp && cw == -1 && sentinel == ts-1) | 
					
						
							|  |  |  | 	cw = nw; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If the cursor is at whitespace between two words, make a new, empty
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 word, add it before (well, after, since the list is in reverse order) | 
					
						
							|  |  |  | 	 the word we just added, and set the current word to that one. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       if (cwp && cw == -1 && sentinel < ts) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  tl = make_word_list (make_word (""), ret->next); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  ret->next = tl; | 
					
						
							|  |  |  | 	  cw = nw; | 
					
						
							|  |  |  | 	  nw++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (string[te] == 0) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       i = te; | 
					
						
							|  |  |  |       while (member (string[i], d) && (ifs_split || spctabnl(string[i]))) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	i++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (string[i]) | 
					
						
							|  |  |  | 	ts = i; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Special case for SENTINEL at the end of STRING.  If we haven't found
 | 
					
						
							|  |  |  |      the word containing SENTINEL yet, and the index we're looking for is at | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |      the end of STRING (or past the end of the previously-found token, | 
					
						
							|  |  |  |      possible if the end of the line is composed solely of IFS whitespace) | 
					
						
							|  |  |  |      add an additional null argument and set the current word pointer to that. */ | 
					
						
							|  |  |  |   if (cwp && cw == -1 && (sentinel >= slen || sentinel >= te)) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (whitespace (string[sentinel - 1])) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  token = ""; | 
					
						
							|  |  |  | 	  ret = add_string_to_list (token, ret); | 
					
						
							|  |  |  | 	  nw++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       cw = nw; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (nwp) | 
					
						
							|  |  |  |     *nwp = nw; | 
					
						
							|  |  |  |   if (cwp) | 
					
						
							|  |  |  |     *cwp = cw; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   FREE (d2); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   return (REVERSE_LIST (ret, WORD_LIST *)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #endif /* READLINE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | /* UNUSED */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Extract the name of the variable to bind to from the assignment string. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | assignment_name (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int offset; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   char *temp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   offset = assignment (string, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (offset == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return (char *)NULL; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   temp = substring (string, 0, offset); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /*     Functions to convert strings to WORD_LISTs and vice versa    */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Return a single string of all the words in LIST.  SEP is the separator
 | 
					
						
							|  |  |  |    to put between individual elements of LIST in the output string. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | string_list_internal (list, sep) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      char *sep; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *t; | 
					
						
							|  |  |  |   char *result, *r; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   size_t word_len, sep_len, result_size; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (list == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Short-circuit quickly if we don't need to separate anything. */ | 
					
						
							|  |  |  |   if (list->next == 0) | 
					
						
							|  |  |  |     return (savestring (list->word->word)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* This is nearly always called with either sep[0] == 0 or sep[1] == 0. */ | 
					
						
							|  |  |  |   sep_len = STRLEN (sep); | 
					
						
							|  |  |  |   result_size = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (t = list; t; t = t->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (t != list) | 
					
						
							|  |  |  | 	result_size += sep_len; | 
					
						
							|  |  |  |       result_size += strlen (t->word->word); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   r = result = (char *)xmalloc (result_size + 1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (t = list; t; t = t->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (t != list && sep_len) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (sep_len > 1) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      FASTCOPY (sep, r, sep_len); | 
					
						
							|  |  |  | 	      r += sep_len; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    *r++ = sep[0]; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       word_len = strlen (t->word->word); | 
					
						
							|  |  |  |       FASTCOPY (t->word->word, r, word_len); | 
					
						
							|  |  |  |       r += word_len; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   *r = '\0'; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return a single string of all the words present in LIST, separating
 | 
					
						
							|  |  |  |    each word with a space. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | string_list (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (string_list_internal (list, " ")); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* An external interface that can be used by the rest of the shell to
 | 
					
						
							|  |  |  |    obtain a string containing the first character in $IFS.  Handles all | 
					
						
							|  |  |  |    the multibyte complications.  If LENP is non-null, it is set to the | 
					
						
							|  |  |  |    length of the returned string. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | ifs_firstchar (lenp) | 
					
						
							|  |  |  |      int *lenp; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   int len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = xmalloc (MB_LEN_MAX + 1); | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (ifs_firstc_len == 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret[0] = ifs_firstc[0]; | 
					
						
							|  |  |  |       ret[1] = '\0'; | 
					
						
							|  |  |  |       len = ret[0] ? 1 : 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       memcpy (ret, ifs_firstc, ifs_firstc_len); | 
					
						
							|  |  |  |       ret[len = ifs_firstc_len] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   ret[0] = ifs_firstc; | 
					
						
							|  |  |  |   ret[1] = '\0'; | 
					
						
							|  |  |  |   len = ret[0] ? 0 : 1; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (lenp) | 
					
						
							|  |  |  |     *lenp = len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Return a single string of all the words present in LIST, obeying the
 | 
					
						
							|  |  |  |    quoting rules for "$*", to wit: (P1003.2, draft 11, 3.5.2) "If the | 
					
						
							|  |  |  |    expansion [of $*] appears within a double quoted string, it expands | 
					
						
							|  |  |  |    to a single field with the value of each parameter separated by the | 
					
						
							|  |  |  |    first character of the IFS variable, or by a <space> if IFS is unset." */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | string_list_dollar_star (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   char *ret; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  if defined (__GNUC__)
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   char sep[MB_CUR_MAX + 1]; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  else
 | 
					
						
							|  |  |  |   char *sep = 0; | 
					
						
							|  |  |  | #  endif
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   char sep[2]; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  if !defined (__GNUC__)
 | 
					
						
							|  |  |  |   sep = (char *)xmalloc (MB_CUR_MAX + 1); | 
					
						
							|  |  |  | #  endif /* !__GNUC__ */
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (ifs_firstc_len == 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       sep[0] = ifs_firstc[0]; | 
					
						
							|  |  |  |       sep[1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       memcpy (sep, ifs_firstc, ifs_firstc_len); | 
					
						
							|  |  |  |       sep[ifs_firstc_len] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   sep[0] = ifs_firstc; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   sep[1] = '\0'; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   ret = string_list_internal (list, sep); | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE) && !defined (__GNUC__)
 | 
					
						
							|  |  |  |   free (sep); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Turn $@ into a string.  If (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))
 | 
					
						
							|  |  |  |    is non-zero, the $@ appears within double quotes, and we should quote | 
					
						
							|  |  |  |    the list before converting it into a string.  If IFS is unset, and the | 
					
						
							|  |  |  |    word is not quoted, we just need to quote CTLESC and CTLNUL characters | 
					
						
							|  |  |  |    in the words in the list, because the default value of $IFS is | 
					
						
							|  |  |  |    <space><tab><newline>, IFS characters in the words in the list should | 
					
						
							|  |  |  |    also be split.  If IFS is null, and the word is not quoted, we need | 
					
						
							|  |  |  |    to quote the words in the list to preserve the positional parameters | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |    exactly. | 
					
						
							|  |  |  |    Valid values for the FLAGS argument are the PF_ flags in command.h, | 
					
						
							|  |  |  |    the only one we care about is PF_ASSIGNRHS.  $@ is supposed to expand | 
					
						
							|  |  |  |    to the positional parameters separated by spaces no matter what IFS is | 
					
						
							|  |  |  |    set to if in a context where word splitting is not performed.  The only | 
					
						
							|  |  |  |    one that we didn't handle before is assignment statement arguments to | 
					
						
							|  |  |  |    declaration builtins like `declare'. */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | string_list_dollar_at (list, quoted, flags) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   char *ifs, *ret; | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  if defined (__GNUC__)
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   char sep[MB_CUR_MAX + 1]; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  else
 | 
					
						
							|  |  |  |   char *sep = 0; | 
					
						
							|  |  |  | #  endif /* !__GNUC__ */
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |   char sep[2]; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   WORD_LIST *tlist; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* XXX this could just be ifs = ifs_value; */ | 
					
						
							|  |  |  |   ifs = ifs_var ? value_cell (ifs_var) : (char *)0; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #  if !defined (__GNUC__)
 | 
					
						
							|  |  |  |   sep = (char *)xmalloc (MB_CUR_MAX + 1); | 
					
						
							|  |  |  | #  endif /* !__GNUC__ */
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* XXX - bash-4.4/bash-5.0 testing PF_ASSIGNRHS */ | 
					
						
							|  |  |  |   if (flags & PF_ASSIGNRHS) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       sep[0] = ' '; | 
					
						
							|  |  |  |       sep[1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (ifs && *ifs) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (ifs_firstc_len == 1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  sep[0] = ifs_firstc[0]; | 
					
						
							|  |  |  | 	  sep[1] = '\0'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  memcpy (sep, ifs_firstc, ifs_firstc_len); | 
					
						
							|  |  |  | 	  sep[ifs_firstc_len] = '\0'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       sep[0] = ' '; | 
					
						
							|  |  |  |       sep[1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* XXX - bash-4.4/bash-5.0 test PF_ASSIGNRHS */ | 
					
						
							|  |  |  |   sep[0] = ((flags & PF_ASSIGNRHS) || ifs == 0 || *ifs == 0) ? ' ' : *ifs; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   sep[1] = '\0'; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   /* XXX -- why call quote_list if ifs == 0?  we can get away without doing
 | 
					
						
							|  |  |  |      it now that quote_escapes quotes spaces */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   tlist = (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES|Q_PATQUOTE)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		? quote_list (list) | 
					
						
							|  |  |  | 		: list_quote_escapes (list); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ret = string_list_internal (tlist, sep); | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE) && !defined (__GNUC__)
 | 
					
						
							|  |  |  |   free (sep); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | /* Turn the positional parameters into a string, understanding quoting and
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    the various subtleties of using the first character of $IFS as the | 
					
						
							|  |  |  |    separator.  Calls string_list_dollar_at, string_list_dollar_star, and | 
					
						
							|  |  |  |    string_list as appropriate. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | string_list_pos_params (pchar, list, quoted) | 
					
						
							|  |  |  |      int pchar; | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   WORD_LIST *tlist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (pchar == '*' && (quoted & Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tlist = quote_list (list); | 
					
						
							|  |  |  |       word_list_remove_quoted_nulls (tlist); | 
					
						
							|  |  |  |       ret = string_list_dollar_star (tlist); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (pchar == '*' && (quoted & Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tlist = quote_list (list); | 
					
						
							|  |  |  |       word_list_remove_quoted_nulls (tlist); | 
					
						
							|  |  |  |       ret = string_list (tlist); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (pchar == '*') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* Even when unquoted, string_list_dollar_star does the right thing
 | 
					
						
							|  |  |  | 	 making sure that the first character of $IFS is used as the | 
					
						
							|  |  |  | 	 separator. */ | 
					
						
							|  |  |  |       ret = string_list_dollar_star (list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (pchar == '@' && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  |     /* We use string_list_dollar_at, but only if the string is quoted, since
 | 
					
						
							|  |  |  |        that quotes the escapes if it's not, which we don't want.  We could | 
					
						
							|  |  |  |        use string_list (the old code did), but that doesn't do the right | 
					
						
							|  |  |  |        thing if the first character of $IFS is not a space.  We use | 
					
						
							|  |  |  |        string_list_dollar_star if the string is unquoted so we make sure that | 
					
						
							|  |  |  |        the elements of $@ are separated by the first character of $IFS for | 
					
						
							|  |  |  |        later splitting. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     ret = string_list_dollar_at (list, quoted, 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   else if (pchar == '@') | 
					
						
							|  |  |  |     ret = string_list_dollar_star (list); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     ret = string_list ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) ? quote_list (list) : list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Return the list of words present in STRING.  Separate the string into
 | 
					
						
							|  |  |  |    words at any of the characters found in SEPARATORS.  If QUOTED is | 
					
						
							|  |  |  |    non-zero then word in the list will have its quoted flag set, otherwise | 
					
						
							|  |  |  |    the quoted flag is left as make_word () deemed fit. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This obeys the P1003.2 word splitting semantics.  If `separators' is | 
					
						
							|  |  |  |    exactly <space><tab><newline>, then the splitting algorithm is that of | 
					
						
							|  |  |  |    the Bourne shell, which treats any sequence of characters from `separators' | 
					
						
							|  |  |  |    as a delimiter.  If IFS is unset, which results in `separators' being set | 
					
						
							|  |  |  |    to "", no splitting occurs.  If separators has some other value, the | 
					
						
							|  |  |  |    following rules are applied (`IFS white space' means zero or more | 
					
						
							|  |  |  |    occurrences of <space>, <tab>, or <newline>, as long as those characters | 
					
						
							|  |  |  |    are in `separators'): | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	1) IFS white space is ignored at the start and the end of the | 
					
						
							|  |  |  | 	   string. | 
					
						
							|  |  |  | 	2) Each occurrence of a character in `separators' that is not | 
					
						
							|  |  |  | 	   IFS white space, along with any adjacent occurrences of | 
					
						
							|  |  |  | 	   IFS white space delimits a field. | 
					
						
							|  |  |  | 	3) Any nonzero-length sequence of IFS white space delimits a field. | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* BEWARE!  list_string strips null arguments.  Don't call it twice and
 | 
					
						
							|  |  |  |    expect to have "" preserved! */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This performs word splitting and quoted null character removal on
 | 
					
						
							|  |  |  |    STRING. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #define issep(c) \
 | 
					
						
							|  |  |  | 	(((separators)[0]) ? ((separators)[1] ? isifs(c) \ | 
					
						
							|  |  |  | 					      : (c) == (separators)[0]) \ | 
					
						
							|  |  |  | 			   : 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | list_string (string, separators, quoted) | 
					
						
							|  |  |  |      register char *string, *separators; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  |   WORD_DESC *t; | 
					
						
							|  |  |  |   char *current_word, *s; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int sindex, sh_style_split, whitesep, xflags; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!string || !*string) | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   sh_style_split = separators && separators[0] == ' ' && | 
					
						
							|  |  |  | 				 separators[1] == '\t' && | 
					
						
							|  |  |  | 				 separators[2] == '\n' && | 
					
						
							|  |  |  | 				 separators[3] == '\0'; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   for (xflags = 0, s = ifs_value; s && *s; s++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*s == CTLESC) xflags |= SX_NOCTLESC; | 
					
						
							|  |  |  |       else if (*s == CTLNUL) xflags |= SX_NOESCCTLNUL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   slen = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Remove sequences of whitespace at the beginning of STRING, as
 | 
					
						
							|  |  |  |      long as those characters appear in IFS.  Do not do this if | 
					
						
							|  |  |  |      STRING is quoted or if there are no separator characters. */ | 
					
						
							|  |  |  |   if (!quoted || !separators || !*separators) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       for (s = string; *s && spctabnl (*s) && issep (*s); s++); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!*s) | 
					
						
							|  |  |  | 	return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       string = s; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* OK, now STRING points to a word that does not begin with white space.
 | 
					
						
							|  |  |  |      The splitting algorithm is: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	extract a word, stopping at a separator | 
					
						
							|  |  |  | 	skip sequences of spc, tab, or nl as long as they are separators | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      This obeys the field splitting rules in Posix.2. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   slen = (MB_CUR_MAX > 1) ? STRLEN (string) : 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (result = (WORD_LIST *)NULL, sindex = 0; string[sindex]; ) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       /* Don't need string length in ADVANCE_CHAR or string_extract_verbatim
 | 
					
						
							|  |  |  | 	 unless multibyte chars are possible. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       current_word = string_extract_verbatim (string, slen, &sindex, separators, xflags); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (current_word == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If we have a quoted empty string, add a quoted null argument.  We
 | 
					
						
							|  |  |  | 	 want to preserve the quoted null character iff this is a quoted | 
					
						
							|  |  |  | 	 empty string; otherwise the quoted null characters are removed | 
					
						
							|  |  |  | 	 below. */ | 
					
						
							|  |  |  |       if (QUOTED_NULL (current_word)) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  t = alloc_word_desc (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  t->word = make_quoted_char ('\0'); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  t->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  result = make_word_list (t, result); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (current_word[0] != '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If we have something, then add it regardless.  However,
 | 
					
						
							|  |  |  | 	     perform quoted null character removal on the current word. */ | 
					
						
							|  |  |  | 	  remove_quoted_nulls (current_word); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  result = add_string_to_list (current_word, result); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  result->word->flags &= ~W_HASQUOTEDNULL;	/* just to be sure */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  | 	    result->word->flags |= W_QUOTED; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If we're not doing sequences of separators in the traditional
 | 
					
						
							|  |  |  | 	 Bourne shell style, then add a quoted null argument. */ | 
					
						
							|  |  |  |       else if (!sh_style_split && !spctabnl (string[sindex])) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  t = alloc_word_desc (); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  t->word = make_quoted_char ('\0'); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  t->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  result = make_word_list (t, result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       free (current_word); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       /* Note whether or not the separator is IFS whitespace, used later. */ | 
					
						
							|  |  |  |       whitesep = string[sindex] && spctabnl (string[sindex]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       /* Move past the current separator character. */ | 
					
						
							|  |  |  |       if (string[sindex]) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  DECLARE_MBSTATE; | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (string, slen, sindex); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* Now skip sequences of space, tab, or newline characters if they are
 | 
					
						
							|  |  |  | 	 in the list of separators. */ | 
					
						
							|  |  |  |       while (string[sindex] && spctabnl (string[sindex]) && issep (string[sindex])) | 
					
						
							|  |  |  | 	sindex++; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       /* If the first separator was IFS whitespace and the current character
 | 
					
						
							|  |  |  | 	 is a non-whitespace IFS character, it should be part of the current | 
					
						
							|  |  |  | 	 field delimiter, not a separate delimiter that would result in an | 
					
						
							|  |  |  | 	 empty field.  Look at POSIX.2, 3.6.5, (3)(b). */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       if (string[sindex] && whitesep && issep (string[sindex]) && !spctabnl (string[sindex])) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  sindex++; | 
					
						
							|  |  |  | 	  /* An IFS character that is not IFS white space, along with any
 | 
					
						
							|  |  |  | 	     adjacent IFS white space, shall delimit a field. (SUSv3) */ | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	  while (string[sindex] && spctabnl (string[sindex]) && isifs (string[sindex])) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    sindex++; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return (REVERSE_LIST (result, WORD_LIST *)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Parse a single word from STRING, using SEPARATORS to separate fields.
 | 
					
						
							|  |  |  |    ENDPTR is set to the first character after the word.  This is used by | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    the `read' builtin.  This is never called with SEPARATORS != $IFS; | 
					
						
							|  |  |  |    it should be simplified. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    XXX - this function is very similar to list_string; they should be | 
					
						
							|  |  |  | 	 combined - XXX */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | get_word_from_string (stringp, separators, endptr) | 
					
						
							|  |  |  |      char **stringp, *separators, **endptr; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *s; | 
					
						
							|  |  |  |   char *current_word; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int sindex, sh_style_split, whitesep, xflags; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (!stringp || !*stringp || !**stringp) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   sh_style_split = separators && separators[0] == ' ' && | 
					
						
							|  |  |  | 				 separators[1] == '\t' && | 
					
						
							|  |  |  | 				 separators[2] == '\n' && | 
					
						
							|  |  |  | 				 separators[3] == '\0'; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   for (xflags = 0, s = ifs_value; s && *s; s++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*s == CTLESC) xflags |= SX_NOCTLESC; | 
					
						
							|  |  |  |       if (*s == CTLNUL) xflags |= SX_NOESCCTLNUL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   s = *stringp; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   slen = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Remove sequences of whitespace at the beginning of STRING, as
 | 
					
						
							|  |  |  |      long as those characters appear in IFS. */ | 
					
						
							|  |  |  |   if (sh_style_split || !separators || !*separators) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       for (; *s && spctabnl (*s) && isifs (*s); s++); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* If the string is nothing but whitespace, update it and return. */ | 
					
						
							|  |  |  |       if (!*s) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  *stringp = s; | 
					
						
							|  |  |  | 	  if (endptr) | 
					
						
							|  |  |  | 	    *endptr = s; | 
					
						
							|  |  |  | 	  return ((char *)NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* OK, S points to a word that does not begin with white space.
 | 
					
						
							|  |  |  |      Now extract a word, stopping at a separator, save a pointer to | 
					
						
							|  |  |  |      the first character after the word, then skip sequences of spc, | 
					
						
							|  |  |  |      tab, or nl as long as they are separators. | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      This obeys the field splitting rules in Posix.2. */ | 
					
						
							|  |  |  |   sindex = 0; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Don't need string length in ADVANCE_CHAR or string_extract_verbatim
 | 
					
						
							|  |  |  |      unless multibyte chars are possible. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   slen = (MB_CUR_MAX > 1) ? STRLEN (s) : 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   current_word = string_extract_verbatim (s, slen, &sindex, separators, xflags); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Set ENDPTR to the first character after the end of the word. */ | 
					
						
							|  |  |  |   if (endptr) | 
					
						
							|  |  |  |     *endptr = s + sindex; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* Note whether or not the separator is IFS whitespace, used later. */ | 
					
						
							|  |  |  |   whitesep = s[sindex] && spctabnl (s[sindex]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Move past the current separator character. */ | 
					
						
							|  |  |  |   if (s[sindex]) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       DECLARE_MBSTATE; | 
					
						
							|  |  |  |       ADVANCE_CHAR (s, slen, sindex); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Now skip sequences of space, tab, or newline characters if they are
 | 
					
						
							|  |  |  |      in the list of separators. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   while (s[sindex] && spctabnl (s[sindex]) && isifs (s[sindex])) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     sindex++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* If the first separator was IFS whitespace and the current character is
 | 
					
						
							|  |  |  |      a non-whitespace IFS character, it should be part of the current field | 
					
						
							|  |  |  |      delimiter, not a separate delimiter that would result in an empty field. | 
					
						
							|  |  |  |      Look at POSIX.2, 3.6.5, (3)(b). */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (s[sindex] && whitesep && isifs (s[sindex]) && !spctabnl (s[sindex])) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       sindex++; | 
					
						
							|  |  |  |       /* An IFS character that is not IFS white space, along with any adjacent
 | 
					
						
							|  |  |  | 	 IFS white space, shall delimit a field. */ | 
					
						
							|  |  |  |       while (s[sindex] && spctabnl (s[sindex]) && isifs (s[sindex])) | 
					
						
							|  |  |  | 	sindex++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Update STRING to point to the next field. */ | 
					
						
							|  |  |  |   *stringp = s + sindex; | 
					
						
							|  |  |  |   return (current_word); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Remove IFS white space at the end of STRING.  Start at the end
 | 
					
						
							|  |  |  |    of the string and walk backwards until the beginning of the string | 
					
						
							|  |  |  |    or we find a character that's not IFS white space and not CTLESC. | 
					
						
							|  |  |  |    Only let CTLESC escape a white space character if SAW_ESCAPE is | 
					
						
							|  |  |  |    non-zero.  */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | strip_trailing_ifs_whitespace (string, separators, saw_escape) | 
					
						
							|  |  |  |      char *string, *separators; | 
					
						
							|  |  |  |      int saw_escape; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *s; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   s = string + STRLEN (string) - 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   while (s > string && ((spctabnl (*s) && isifs (*s)) || | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 			(saw_escape && *s == CTLESC && spctabnl (s[1])))) | 
					
						
							|  |  |  |     s--; | 
					
						
							|  |  |  |   *++s = '\0'; | 
					
						
							|  |  |  |   return string; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | /* UNUSED */ | 
					
						
							|  |  |  | /* Split STRING into words at whitespace.  Obeys shell-style quoting with
 | 
					
						
							|  |  |  |    backslashes, single and double quotes. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | WORD_LIST * | 
					
						
							|  |  |  | list_string_with_quotes (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  |   char *token, *s; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t s_len; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int c, i, tokstart, len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (s = string; s && *s && spctabnl (*s); s++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   if (s == 0 || *s == 0) | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   s_len = strlen (s); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   tokstart = i = 0; | 
					
						
							|  |  |  |   list = (WORD_LIST *)NULL; | 
					
						
							|  |  |  |   while (1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       c = s[i]; | 
					
						
							|  |  |  |       if (c == '\\') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  if (s[i]) | 
					
						
							|  |  |  | 	    i++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (c == '\'') | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	i = skip_single_quoted (s, s_len, ++i, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (c == '"') | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	i = skip_double_quoted (s, s_len, ++i, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else if (c == 0 || spctabnl (c)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* We have found the end of a token.  Make a word out of it and
 | 
					
						
							|  |  |  | 	     add it to the word list. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  token = substring (s, tokstart, i); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  list = add_string_to_list (token, list); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  free (token); | 
					
						
							|  |  |  | 	  while (spctabnl (s[i])) | 
					
						
							|  |  |  | 	    i++; | 
					
						
							|  |  |  | 	  if (s[i]) | 
					
						
							|  |  |  | 	    tokstart = i; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	i++;	/* normal character */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (REVERSE_LIST (list, WORD_LIST *)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /********************************************************/ | 
					
						
							|  |  |  | /*							*/ | 
					
						
							|  |  |  | /*	Functions to perform assignment statements	*/ | 
					
						
							|  |  |  | /*							*/ | 
					
						
							|  |  |  | /********************************************************/ | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static SHELL_VAR * | 
					
						
							|  |  |  | do_compound_assignment (name, value, flags) | 
					
						
							|  |  |  |      char *name, *value; | 
					
						
							|  |  |  |      int flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   int mklocal, mkassoc, mkglobal; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   WORD_LIST *list; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   mklocal = flags & ASS_MKLOCAL; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   mkassoc = flags & ASS_MKASSOC; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   mkglobal = flags & ASS_MKGLOBAL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (mklocal && variable_context) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       v = find_variable (name); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (v && ((readonly_p (v) && (flags & ASS_FORCE) == 0) || noassign_p (v))) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (v && readonly_p (v)) | 
					
						
							|  |  |  | 	    err_readonly (name); | 
					
						
							|  |  |  | 	  return (v);	/* XXX */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       list = expand_compound_array_assignment (v, value, flags); | 
					
						
							|  |  |  |       if (mkassoc) | 
					
						
							|  |  |  | 	v = make_local_assoc_variable (name); | 
					
						
							|  |  |  |       else if (v == 0 || (array_p (v) == 0 && assoc_p (v) == 0) || v->context != variable_context) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |         v = make_local_array_variable (name, 0); | 
					
						
							|  |  |  |       if (v) | 
					
						
							|  |  |  | 	assign_compound_array_list (v, list, flags); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (list) | 
					
						
							|  |  |  | 	dispose_words (list); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |   /* In a function but forcing assignment in global context */ | 
					
						
							|  |  |  |   else if (mkglobal && variable_context) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       v = find_global_variable (name); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (v && ((readonly_p (v) && (flags & ASS_FORCE) == 0) || noassign_p (v))) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (v && readonly_p (v)) | 
					
						
							|  |  |  | 	    err_readonly (name); | 
					
						
							|  |  |  | 	  return (v);	/* XXX */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       list = expand_compound_array_assignment (v, value, flags); | 
					
						
							|  |  |  |       if (v == 0 && mkassoc) | 
					
						
							|  |  |  | 	v = make_new_assoc_variable (name); | 
					
						
							|  |  |  |       else if (v && mkassoc && assoc_p (v) == 0) | 
					
						
							|  |  |  | 	v = convert_var_to_assoc (v); | 
					
						
							|  |  |  |       else if (v == 0) | 
					
						
							|  |  |  | 	v = make_new_array_variable (name); | 
					
						
							|  |  |  |       else if (v && mkassoc == 0 && array_p (v) == 0) | 
					
						
							|  |  |  | 	v = convert_var_to_array (v); | 
					
						
							|  |  |  |       if (v) | 
					
						
							|  |  |  | 	assign_compound_array_list (v, list, flags); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (list) | 
					
						
							|  |  |  | 	dispose_words (list); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       v = assign_array_from_string (name, value, flags); | 
					
						
							|  |  |  |       if (v && ((readonly_p (v) && (flags & ASS_FORCE) == 0) || noassign_p (v))) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (v && readonly_p (v)) | 
					
						
							|  |  |  | 	    err_readonly (name); | 
					
						
							|  |  |  | 	  return (v);	/* XXX */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (v); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Given STRING, an assignment string, get the value of the right side
 | 
					
						
							|  |  |  |    of the `=', and bind it to the left side.  If EXPAND is true, then | 
					
						
							|  |  |  |    perform parameter expansion, command substitution, and arithmetic | 
					
						
							|  |  |  |    expansion on the right-hand side.  Perform tilde expansion in any | 
					
						
							|  |  |  |    case.  Do not perform word splitting on the result of expansion. */ | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | do_assignment_internal (word, expand) | 
					
						
							|  |  |  |      const WORD_DESC *word; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      int expand; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   int offset, appendop, assign_list, aflags, retval; | 
					
						
							|  |  |  |   char *name, *value, *temp; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *entry; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |   char *t; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   int ni; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   const char *string; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (word == 0 || word->word == 0) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   appendop = assign_list = aflags = 0; | 
					
						
							|  |  |  |   string = word->word; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   offset = assignment (string, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   name = savestring (string); | 
					
						
							|  |  |  |   value = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (name[offset] == '=') | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (name[offset - 1] == '+') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  appendop = 1; | 
					
						
							|  |  |  | 	  name[offset - 1] = '\0'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       name[offset] = 0;		/* might need this set later */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       temp = name + offset + 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (expand && (word->flags & W_COMPASSIGN)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  assign_list = ni = 1; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  value = extract_array_assignment_list (temp, &ni); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       if (expand && temp[0]) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	value = expand_string_if_necessary (temp, 0, expand_string_assignment); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	value = savestring (temp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value == 0) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       value = (char *)xmalloc (1); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       value[0] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (echo_command_at_execute) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (appendop) | 
					
						
							|  |  |  | 	name[offset - 1] = '+'; | 
					
						
							|  |  |  |       xtrace_print_assignment (name, value, assign_list, 1); | 
					
						
							|  |  |  |       if (appendop) | 
					
						
							|  |  |  | 	name[offset - 1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | #define ASSIGN_RETURN(r)	do { FREE (value); free (name); return (r); } while (0)
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (appendop) | 
					
						
							|  |  |  |     aflags |= ASS_APPEND; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (t = mbschr (name, '['))	/*]*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (assign_list) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  report_error (_("%s: cannot assign list to array member"), name); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  ASSIGN_RETURN (0); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       entry = assign_array_element (name, value, aflags); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (entry == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	ASSIGN_RETURN (0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else if (assign_list) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2012-05-07 16:22:49 -04:00
										 |  |  |       if ((word->flags & W_ASSIGNARG) && (word->flags & W_ASSNGLOBAL) == 0) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	aflags |= ASS_MKLOCAL; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if ((word->flags & W_ASSIGNARG) && (word->flags & W_ASSNGLOBAL)) | 
					
						
							|  |  |  | 	aflags |= ASS_MKGLOBAL; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (word->flags & W_ASSIGNASSOC) | 
					
						
							|  |  |  | 	aflags |= ASS_MKASSOC; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       entry = do_compound_assignment (name, value, aflags); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   else | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   entry = bind_variable (name, value, aflags); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   stupidly_hack_special_variables (name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   /* Return 1 if the assignment seems to have been performed correctly. */ | 
					
						
							|  |  |  |   if (entry == 0 || readonly_p (entry)) | 
					
						
							|  |  |  |     retval = 0;		/* assignment failure */ | 
					
						
							|  |  |  |   else if (noassign_p (entry)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  |       retval = 1;	/* error status, but not assignment failure */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     retval = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (entry && retval != 0 && noassign_p (entry) == 0) | 
					
						
							|  |  |  |     VUNSETATTR (entry, att_invisible); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ASSIGN_RETURN (retval); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform the assignment statement in STRING, and expand the
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |    right side by doing tilde, command and parameter expansion. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | do_assignment (string) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC td; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   td.flags = W_ASSIGNMENT; | 
					
						
							|  |  |  |   td.word = string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return do_assignment_internal (&td, 1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | do_word_assignment (word, flags) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |      WORD_DESC *word; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      int flags; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   return do_assignment_internal (word, 1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Given STRING, an assignment string, get the value of the right side
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |    of the `=', and bind it to the left side.  Do not perform any word | 
					
						
							|  |  |  |    expansions on the right hand side. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | do_assignment_no_expand (string) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC td; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   td.flags = W_ASSIGNMENT; | 
					
						
							|  |  |  |   td.word = string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (do_assignment_internal (&td, 0)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /***************************************************
 | 
					
						
							|  |  |  |  *						   * | 
					
						
							|  |  |  |  *  Functions to manage the positional parameters  * | 
					
						
							|  |  |  |  *						   * | 
					
						
							|  |  |  |  ***************************************************/ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Return the word list that corresponds to `$*'. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | list_rest_of_args () | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register WORD_LIST *list, *args; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Break out of the loop as soon as one of the dollar variables is null. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (i = 1, list = (WORD_LIST *)NULL; i < 10 && dollar_vars[i]; i++) | 
					
						
							|  |  |  |     list = make_word_list (make_bare_word (dollar_vars[i]), list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (args = rest_of_args; args; args = args->next) | 
					
						
							|  |  |  |     list = make_word_list (make_bare_word (args->word->word), list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (REVERSE_LIST (list, WORD_LIST *)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | int | 
					
						
							|  |  |  | number_of_args () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *list; | 
					
						
							|  |  |  |   int n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (n = 0; n < 9 && dollar_vars[n+1]; n++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   for (list = rest_of_args; list; list = list->next) | 
					
						
							|  |  |  |     n++; | 
					
						
							|  |  |  |   return n; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Return the value of a positional parameter.  This handles values > 10. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | get_dollar_var_value (ind) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      intmax_t ind; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   char *temp; | 
					
						
							|  |  |  |   WORD_LIST *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (ind < 10) | 
					
						
							|  |  |  |     temp = dollar_vars[ind] ? savestring (dollar_vars[ind]) : (char *)NULL; | 
					
						
							|  |  |  |   else	/* We want something like ${11} */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ind -= 10; | 
					
						
							|  |  |  |       for (p = rest_of_args; p && ind--; p = p->next) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       temp = p ? savestring (p->word->word) : (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Make a single large string out of the dollar digit variables,
 | 
					
						
							|  |  |  |    and the rest_of_args.  If DOLLAR_STAR is 1, then obey the special | 
					
						
							|  |  |  |    case of "$*" with respect to IFS. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | string_rest_of_args (dollar_star) | 
					
						
							|  |  |  |      int dollar_star; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register WORD_LIST *list; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   char *string; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   list = list_rest_of_args (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   string = dollar_star ? string_list_dollar_star (list) : string_list (list); | 
					
						
							|  |  |  |   dispose_words (list); | 
					
						
							|  |  |  |   return (string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Return a string containing the positional parameters from START to
 | 
					
						
							|  |  |  |    END, inclusive.  If STRING[0] == '*', we obey the rules for $*, | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    which only makes a difference if QUOTED is non-zero.  If QUOTED includes | 
					
						
							|  |  |  |    Q_HERE_DOCUMENT or Q_DOUBLE_QUOTES, this returns a quoted list, otherwise | 
					
						
							|  |  |  |    no quoting chars are added. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static char * | 
					
						
							|  |  |  | pos_params (string, start, end, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int start, end, quoted; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   WORD_LIST *save, *params, *h, *t; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   int i; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   /* see if we can short-circuit.  if start == end, we want 0 parameters. */ | 
					
						
							|  |  |  |   if (start == end) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   save = params = list_rest_of_args (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (save == 0 && start > 0) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (start == 0)		/* handle ${@:0[:x]} specially */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t = make_word_list (make_word (dollar_vars[0]), params); | 
					
						
							|  |  |  |       save = params = t; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   for (i = start ? 1 : 0; params && i < start; i++) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     params = params->next; | 
					
						
							|  |  |  |   if (params == 0) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       dispose_words (save); | 
					
						
							|  |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   for (h = t = params; params && i < end; i++) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       t = params; | 
					
						
							|  |  |  |       params = params->next; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   t->next = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ret = string_list_pos_params (string[0], h, quoted); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (t != params) | 
					
						
							|  |  |  |     t->next = params; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   dispose_words (save); | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /******************************************************************/ | 
					
						
							|  |  |  | /*								  */ | 
					
						
							|  |  |  | /*	Functions to expand strings to strings or WORD_LISTs      */ | 
					
						
							|  |  |  | /*								  */ | 
					
						
							|  |  |  | /******************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #define EXP_CHAR(s) (s == '$' || s == '`' || s == '<' || s == '>' || s == CTLESC || s == '~')
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #define EXP_CHAR(s) (s == '$' || s == '`' || s == CTLESC || s == '~')
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* If there are any characters in STRING that require full expansion,
 | 
					
						
							|  |  |  |    then call FUNC to expand STRING; otherwise just perform quote | 
					
						
							|  |  |  |    removal if necessary.  This returns a new string. */ | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | expand_string_if_necessary (string, quoted, func) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      EXPFUNC *func; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int i, saw_quote; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Don't need string length for ADVANCE_CHAR unless multibyte chars possible. */ | 
					
						
							|  |  |  |   slen = (MB_CUR_MAX > 1) ? strlen (string) : 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   i = saw_quote = 0; | 
					
						
							|  |  |  |   while (string[i]) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (EXP_CHAR (string[i])) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       else if (string[i] == '\'' || string[i] == '\\' || string[i] == '"') | 
					
						
							|  |  |  | 	saw_quote = 1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       list = (*func) (string, quoted); | 
					
						
							|  |  |  |       if (list) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  ret = string_list (list); | 
					
						
							|  |  |  | 	  dispose_words (list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ret = (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (saw_quote && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							|  |  |  |     ret = string_quote_removal (string, quoted); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     ret = savestring (string); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline char * | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | expand_string_to_string_internal (string, quoted, func) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      EXPFUNC *func; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   list = (*func) (string, quoted); | 
					
						
							|  |  |  |   if (list) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = string_list (list); | 
					
						
							|  |  |  |       dispose_words (list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     ret = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | expand_string_to_string (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (expand_string_to_string_internal (string, quoted, expand_string)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | expand_string_unsplit_to_string (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (expand_string_to_string_internal (string, quoted, expand_string_unsplit)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | expand_assignment_string_to_string (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return (expand_string_to_string_internal (string, quoted, expand_string_assignment)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | expand_arith_string (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   WORD_DESC td; | 
					
						
							|  |  |  |   WORD_LIST *list, *tlist; | 
					
						
							|  |  |  |   size_t slen; | 
					
						
							|  |  |  |   int i, saw_quote; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Don't need string length for ADVANCE_CHAR unless multibyte chars possible. */ | 
					
						
							|  |  |  |   slen = (MB_CUR_MAX > 1) ? strlen (string) : 0; | 
					
						
							|  |  |  |   i = saw_quote = 0; | 
					
						
							|  |  |  |   while (string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (EXP_CHAR (string[i])) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       else if (string[i] == '\'' || string[i] == '\\' || string[i] == '"') | 
					
						
							|  |  |  | 	saw_quote = 1; | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* This is expanded version of expand_string_internal as it's called by
 | 
					
						
							|  |  |  | 	 expand_string_leave_quoted  */ | 
					
						
							|  |  |  |       td.flags = W_NOPROCSUB;	/* don't want process substitution */ | 
					
						
							|  |  |  |       td.word = savestring (string); | 
					
						
							|  |  |  |       list = call_expand_word_internal (&td, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							|  |  |  |       /* This takes care of the calls from expand_string_leave_quoted and
 | 
					
						
							|  |  |  | 	 expand_string */ | 
					
						
							|  |  |  |       if (list) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tlist = word_list_split (list); | 
					
						
							|  |  |  | 	  dispose_words (list); | 
					
						
							|  |  |  | 	  list = tlist; | 
					
						
							|  |  |  | 	  if (list) | 
					
						
							|  |  |  | 	    dequote_list (list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       /* This comes from expand_string_if_necessary */ | 
					
						
							|  |  |  |       if (list) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  ret = string_list (list); | 
					
						
							|  |  |  | 	  dispose_words (list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	ret = (char *)NULL; | 
					
						
							|  |  |  |       FREE (td.word); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (saw_quote && (quoted & Q_ARITH)) | 
					
						
							|  |  |  |     ret = string_quote_removal (string, quoted); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   else if (saw_quote && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							|  |  |  |     ret = string_quote_removal (string, quoted); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     ret = savestring (string); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #if defined (COND_COMMAND)
 | 
					
						
							|  |  |  | /* Just remove backslashes in STRING.  Returns a new string. */ | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | remove_backslashes (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *r, *ret, *s; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   r = ret = (char *)xmalloc (strlen (string) + 1); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   for (s = string; s && *s; ) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*s == '\\') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	s++; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (*s == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       *r++ = *s++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   *r = '\0'; | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* This needs better error handling. */ | 
					
						
							|  |  |  | /* Expand W for use as an argument to a unary or binary operator in a
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |    [[...]] expression.  If SPECIAL is 1, this is the rhs argument | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    to the != or == operator, and should be treated as a pattern.  In | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |    this case, we quote the string specially for the globbing code.  If | 
					
						
							|  |  |  |    SPECIAL is 2, this is an rhs argument for the =~ operator, and should | 
					
						
							|  |  |  |    be quoted appropriately for regcomp/regexec.  The caller is responsible | 
					
						
							|  |  |  |    for removing the backslashes if the unquoted word is needed later. */    | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | cond_expand_word (w, special) | 
					
						
							|  |  |  |      WORD_DESC *w; | 
					
						
							|  |  |  |      int special; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *r, *p; | 
					
						
							|  |  |  |   WORD_LIST *l; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   int qflags; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (w->word == 0 || w->word[0] == '\0') | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   w->flags |= W_NOSPLIT2; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   l = call_expand_word_internal (w, 0, 0, (int *)0, (int *)0); | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (l) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (special == 0)			/* LHS */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  dequote_list (l); | 
					
						
							|  |  |  | 	  r = string_list (l); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  /* Need to figure out whether or not we should call dequote_escapes
 | 
					
						
							|  |  |  | 	     or a new dequote_ctlnul function here, and under what | 
					
						
							|  |  |  | 	     circumstances. */ | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | 	  qflags = QGLOB_CVTNULL; | 
					
						
							|  |  |  | 	  if (special == 2) | 
					
						
							|  |  |  | 	    qflags |= QGLOB_REGEXP; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  word_list_remove_quoted_nulls (l); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  p = string_list (l); | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | 	  r = quote_string_for_globbing (p, qflags); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  free (p); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       dispose_words (l); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     r = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return r; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Call expand_word_internal to expand W and handle error returns.
 | 
					
						
							|  |  |  |    A convenience function for functions that don't want to handle | 
					
						
							|  |  |  |    any errors or free any memory before aborting. */ | 
					
						
							|  |  |  | static WORD_LIST * | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | call_expand_word_internal (w, q, i, c, e) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      WORD_DESC *w; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |      int q, i, *c, *e; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   result = expand_word_internal (w, q, i, c, e); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (result == &expand_word_error || result == &expand_word_fatal) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* By convention, each time this error is returned, w->word has
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	 already been freed (it sometimes may not be in the fatal case, | 
					
						
							|  |  |  | 	 but that doesn't result in a memory leak because we're going | 
					
						
							|  |  |  | 	 to exit in most cases). */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       w->word = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       exp_jump_to_top_level ((result == &expand_word_error) ? DISCARD : FORCE_EOF); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* NOTREACHED */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       return (NULL); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform parameter expansion, command substitution, and arithmetic
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |    expansion on STRING, as if it were a word.  Leave the result quoted. | 
					
						
							|  |  |  |    Since this does not perform word splitting, it leaves quoted nulls | 
					
						
							|  |  |  |    in the result.  */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | expand_string_internal (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_DESC td; | 
					
						
							|  |  |  |   WORD_LIST *tresult; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == 0) | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   td.flags = 0; | 
					
						
							|  |  |  |   td.word = savestring (string); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   tresult = call_expand_word_internal (&td, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   FREE (td.word); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (tresult); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand STRING by performing parameter expansion, command substitution,
 | 
					
						
							|  |  |  |    and arithmetic expansion.  Dequote the resulting WORD_LIST before | 
					
						
							|  |  |  |    returning it, but do not perform word splitting.  The call to | 
					
						
							|  |  |  |    remove_quoted_nulls () is in here because word splitting normally | 
					
						
							|  |  |  |    takes care of quote removal. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_string_unsplit (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *value; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   if (string == 0 || *string == '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   value = expand_string_internal (string, quoted); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (value->word) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  remove_quoted_nulls (value->word->word); | 
					
						
							|  |  |  | 	  value->word->flags &= ~W_HASQUOTEDNULL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       dequote_list (value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand the rhs of an assignment statement */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_string_assignment (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_DESC td; | 
					
						
							|  |  |  |   WORD_LIST *value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   td.flags = W_ASSIGNRHS; | 
					
						
							|  |  |  |   td.word = savestring (string); | 
					
						
							|  |  |  |   value = call_expand_word_internal (&td, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							|  |  |  |   FREE (td.word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (value->word) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  remove_quoted_nulls (value->word->word); | 
					
						
							|  |  |  | 	  value->word->flags &= ~W_HASQUOTEDNULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       dequote_list (value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* Expand one of the PS? prompt strings. This is a sort of combination of
 | 
					
						
							|  |  |  |    expand_string_unsplit and expand_string_internal, but returns the | 
					
						
							|  |  |  |    passed string when an error occurs.  Might want to trap other calls | 
					
						
							|  |  |  |    to jump_to_top_level here so we don't endlessly loop. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | expand_prompt_string (string, quoted, wflags) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |      int wflags; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *value; | 
					
						
							|  |  |  |   WORD_DESC td; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == 0) | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   td.flags = wflags; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   td.word = savestring (string); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   no_longjmp_on_fatal_error = 1; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   value = expand_word_internal (&td, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   no_longjmp_on_fatal_error = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (value == &expand_word_error || value == &expand_word_fatal) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       value = make_word_list (make_bare_word (string), (WORD_LIST *)NULL); | 
					
						
							|  |  |  |       return value; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   FREE (td.word); | 
					
						
							|  |  |  |   if (value) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (value->word) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  remove_quoted_nulls (value->word->word); | 
					
						
							|  |  |  | 	  value->word->flags &= ~W_HASQUOTEDNULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       dequote_list (value); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Expand STRING just as if you were expanding a word, but do not dequote
 | 
					
						
							|  |  |  |    the resultant WORD_LIST.  This is called only from within this file, | 
					
						
							|  |  |  |    and is used to correctly preserve quoted characters when expanding | 
					
						
							|  |  |  |    things like ${1+"$@"}.  This does parameter expansion, command | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |    substitution, arithmetic expansion, and word splitting. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | expand_string_leave_quoted (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *tlist; | 
					
						
							|  |  |  |   WORD_LIST *tresult; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (string == 0 || *string == '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   tlist = expand_string_internal (string, quoted); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (tlist) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tresult = word_list_split (tlist); | 
					
						
							|  |  |  |       dispose_words (tlist); | 
					
						
							|  |  |  |       return (tresult); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* This does not perform word splitting or dequote the WORD_LIST
 | 
					
						
							|  |  |  |    it returns. */ | 
					
						
							|  |  |  | static WORD_LIST * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | expand_string_for_rhs (string, quoted, dollar_at_p, expanded_p) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int quoted, *dollar_at_p, *expanded_p; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_DESC td; | 
					
						
							|  |  |  |   WORD_LIST *tresult; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string == 0 || *string == '\0') | 
					
						
							|  |  |  |     return (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							| 
									
										
										
										
											2011-11-22 20:00:10 -05:00
										 |  |  |   td.flags = W_NOSPLIT2;		/* no splitting, remove "" and '' */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   td.word = string; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   tresult = call_expand_word_internal (&td, quoted, 1, dollar_at_p, expanded_p); | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (tresult); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Expand STRING just as if you were expanding a word.  This also returns
 | 
					
						
							|  |  |  |    a list of words.  Note that filename globbing is *NOT* done for word | 
					
						
							|  |  |  |    or string expansion, just when the shell is expanding a command.  This | 
					
						
							|  |  |  |    does parameter expansion, command substitution, arithmetic expansion, | 
					
						
							|  |  |  |    and word splitting.  Dequote the resultant WORD_LIST before returning. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_string (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   if (string == 0 || *string == '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   result = expand_string_leave_quoted (string, quoted); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (result ? dequote_list (result) : result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /***************************************************
 | 
					
						
							|  |  |  |  *						   * | 
					
						
							|  |  |  |  *	Functions to handle quoting chars	   * | 
					
						
							|  |  |  |  *						   * | 
					
						
							|  |  |  |  ***************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Conventions:
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |      A string with s[0] == CTLNUL && s[1] == 0 is a quoted null string. | 
					
						
							|  |  |  |      The parser passes CTLNUL as CTLESC CTLNUL. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Quote escape characters in string s, but no other characters.  This is
 | 
					
						
							|  |  |  |    used to protect CTLESC and CTLNUL in variable values from the rest of | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |    the word expansion process after the variable is expanded (word splitting | 
					
						
							|  |  |  |    and filename generation).  If IFS is null, we quote spaces as well, just | 
					
						
							|  |  |  |    in case we split on spaces later (in the case of unquoted $@, we will | 
					
						
							|  |  |  |    eventually attempt to split the entire word on spaces).  Corresponding | 
					
						
							|  |  |  |    code exists in dequote_escapes.  Even if we don't end up splitting on | 
					
						
							|  |  |  |    spaces, quoting spaces is not a problem.  This should never be called on | 
					
						
							|  |  |  |    a string that is quoted with single or double quotes or part of a here | 
					
						
							|  |  |  |    document (effectively double-quoted). */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | quote_escapes (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *s, *t; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *result, *send; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int quote_spaces, skip_ctlesc, skip_ctlnul; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE;  | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   send = string + slen; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   quote_spaces = (ifs_value && *ifs_value == 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (skip_ctlesc = skip_ctlnul = 0, s = ifs_value; s && *s; s++) | 
					
						
							|  |  |  |     skip_ctlesc |= *s == CTLESC, skip_ctlnul |= *s == CTLNUL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   t = result = (char *)xmalloc ((slen * 2) + 1); | 
					
						
							|  |  |  |   s = string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (*s) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if ((skip_ctlesc == 0 && *s == CTLESC) || (skip_ctlnul == 0 && *s == CTLNUL) || (quote_spaces && *s == ' ')) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	*t++ = CTLESC; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       COPY_CHAR_P (t, s, send); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   *t = '\0'; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST * | 
					
						
							|  |  |  | list_quote_escapes (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *w; | 
					
						
							|  |  |  |   char *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (w = list; w; w = w->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t = w->word->word; | 
					
						
							|  |  |  |       w->word->word = quote_escapes (t); | 
					
						
							|  |  |  |       free (t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return list; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | /* Inverse of quote_escapes; remove CTLESC protecting CTLESC or CTLNUL.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    The parser passes us CTLESC as CTLESC CTLESC and CTLNUL as CTLESC CTLNUL. | 
					
						
							|  |  |  |    This is necessary to make unquoted CTLESC and CTLNUL characters in the | 
					
						
							|  |  |  |    data stream pass through properly. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    We need to remove doubled CTLESC characters inside quoted strings before | 
					
						
							|  |  |  |    quoting the entire string, so we do not double the number of CTLESC | 
					
						
							|  |  |  |    characters. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    Also used by parts of the pattern substitution code. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | dequote_escapes (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   register char *s, *t, *s1; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *result, *send; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   int quote_spaces; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (string == 0) | 
					
						
							|  |  |  |     return string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   send = string + slen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   t = result = (char *)xmalloc (slen + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (strchr (string, CTLESC) == 0) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     return (strcpy (result, string)); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   quote_spaces = (ifs_value && *ifs_value == 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   s = string; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   while (*s) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |       if (*s == CTLESC && (s[1] == CTLESC || s[1] == CTLNUL || (quote_spaces && s[1] == ' '))) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  s++; | 
					
						
							|  |  |  | 	  if (*s == '\0') | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       COPY_CHAR_P (t, s, send); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   *t = '\0'; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return result; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | list_dequote_escapes (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *w; | 
					
						
							|  |  |  |   char *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (w = list; w; w = w->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t = w->word->word; | 
					
						
							|  |  |  |       w->word->word = dequote_escapes (t); | 
					
						
							|  |  |  |       free (t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return list; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | /* Return a new string with the quoted representation of character C.
 | 
					
						
							|  |  |  |    This turns "" into QUOTED_NULL, so the W_HASQUOTEDNULL flag needs to be | 
					
						
							|  |  |  |    set in any resultant WORD_DESC where this value is the word. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							|  |  |  | make_quoted_char (c) | 
					
						
							|  |  |  |      int c; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *temp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   temp = (char *)xmalloc (3); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (c == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp[0] = CTLNUL; | 
					
						
							|  |  |  |       temp[1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp[0] = CTLESC; | 
					
						
							|  |  |  |       temp[1] = c; | 
					
						
							|  |  |  |       temp[2] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (temp); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | /* Quote STRING, returning a new string.  This turns "" into QUOTED_NULL, so
 | 
					
						
							|  |  |  |    the W_HASQUOTEDNULL flag needs to be set in any resultant WORD_DESC where | 
					
						
							|  |  |  |    this value is the word. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | quote_string (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register char *t; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *result, *send; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (*string == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       result = (char *)xmalloc (2); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       result[0] = CTLNUL; | 
					
						
							|  |  |  |       result[1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       slen = strlen (string); | 
					
						
							|  |  |  |       send = string + slen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       result = (char *)xmalloc ((slen * 2) + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (t = result; string < send; ) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  *t++ = CTLESC; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  COPY_CHAR_P (t, string, send); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       *t = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | /* De-quote quoted characters in STRING. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | dequote_string (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   register char *s, *t; | 
					
						
							|  |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *result, *send; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if defined (DEBUG)
 | 
					
						
							|  |  |  |   if (string[0] == CTLESC && string[1] == 0) | 
					
						
							|  |  |  |     internal_inform ("dequote_string: string with bare CTLESC"); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   t = result = (char *)xmalloc (slen + 1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (QUOTED_NULL (string)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       result[0] = '\0'; | 
					
						
							|  |  |  |       return (result); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* A string consisting of only a single CTLESC should pass through unchanged */ | 
					
						
							|  |  |  |   if (string[0] == CTLESC && string[1] == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       result[0] = CTLESC; | 
					
						
							|  |  |  |       result[1] = '\0'; | 
					
						
							|  |  |  |       return (result); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* If no character in the string can be quoted, don't bother examining
 | 
					
						
							|  |  |  |      each character.  Just return a copy of the string passed to us. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (strchr (string, CTLESC) == NULL) | 
					
						
							|  |  |  |     return (strcpy (result, string)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   send = string + slen; | 
					
						
							|  |  |  |   s = string; | 
					
						
							|  |  |  |   while (*s) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (*s == CTLESC) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  s++; | 
					
						
							|  |  |  | 	  if (*s == '\0') | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       COPY_CHAR_P (t, s, send); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   *t = '\0'; | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Quote the entire WORD_LIST list. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | quote_list (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *w; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *t; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (w = list; w; w = w->next) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       t = w->word->word; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       w->word->word = quote_string (t); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (*t == 0) | 
					
						
							|  |  |  | 	w->word->flags |= W_HASQUOTEDNULL;	/* XXX - turn on W_HASQUOTEDNULL here? */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       w->word->flags |= W_QUOTED; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       free (t); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return list; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | /* De-quote quoted characters in each word in LIST. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | dequote_list (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *s; | 
					
						
							|  |  |  |   register WORD_LIST *tlist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (tlist = list; tlist; tlist = tlist->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       s = dequote_string (tlist->word->word); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (QUOTED_NULL (tlist->word->word)) | 
					
						
							|  |  |  | 	tlist->word->flags &= ~W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       free (tlist->word->word); | 
					
						
							|  |  |  |       tlist->word->word = s; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return list; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Remove CTLESC protecting a CTLESC or CTLNUL in place.  Return the passed
 | 
					
						
							|  |  |  |    string. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | remove_quoted_escapes (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t = dequote_escapes (string); | 
					
						
							|  |  |  |       strcpy (string, t); | 
					
						
							|  |  |  |       free (t); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Perform quoted null character removal on STRING.  We don't allow any
 | 
					
						
							|  |  |  |    quoted null characters in the middle or at the ends of strings because | 
					
						
							|  |  |  |    of how expand_word_internal works.  remove_quoted_nulls () turns | 
					
						
							|  |  |  |    STRING into an empty string iff it only consists of a quoted null, | 
					
						
							|  |  |  |    and removes all unquoted CTLNUL characters. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | char * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | remove_quoted_nulls (string) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   register size_t slen; | 
					
						
							|  |  |  |   register int i, j, prev_i; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (strchr (string, CTLNUL) == 0)		/* XXX */ | 
					
						
							|  |  |  |     return string;				/* XXX */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   i = j = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   while (i < slen) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (string[i] == CTLESC) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  /* Old code had j++, but we cannot assume that i == j at this
 | 
					
						
							|  |  |  | 	     point -- what if a CTLNUL has already been removed from the | 
					
						
							|  |  |  | 	     string?  We don't want to drop the CTLESC or recopy characters | 
					
						
							|  |  |  | 	     that we've already copied down. */ | 
					
						
							|  |  |  | 	  i++; string[j++] = CTLESC; | 
					
						
							|  |  |  | 	  if (i == slen) | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else if (string[i] == CTLNUL) | 
					
						
							| 
									
										
										
										
											2011-11-22 20:02:41 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       prev_i = i; | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, slen, i); | 
					
						
							|  |  |  |       if (j < prev_i) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  do string[j++] = string[prev_i++]; while (prev_i < i); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	j = i; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   string[j] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (string); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform quoted null character removal on each element of LIST.
 | 
					
						
							|  |  |  |    This modifies LIST. */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | word_list_remove_quoted_nulls (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (t = list; t; t = t->next) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       remove_quoted_nulls (t->word->word); | 
					
						
							|  |  |  |       t->word->flags &= ~W_HASQUOTEDNULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /*	   Functions for Matching and Removing Patterns		    */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | #if 0 /* Currently unused */
 | 
					
						
							|  |  |  | static unsigned char * | 
					
						
							|  |  |  | mb_getcharlens (string, len) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int len; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i, offset, last; | 
					
						
							|  |  |  |   unsigned char *ret; | 
					
						
							|  |  |  |   char *p; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = offset = 0; | 
					
						
							|  |  |  |   last = 0; | 
					
						
							|  |  |  |   ret = (unsigned char *)xmalloc (len); | 
					
						
							|  |  |  |   memset (ret, 0, len); | 
					
						
							|  |  |  |   while (string[last]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ADVANCE_CHAR (string, len, offset); | 
					
						
							|  |  |  |       ret[last] = offset - last; | 
					
						
							|  |  |  |       last = offset; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Remove the portion of PARAM matched by PATTERN according to OP, where OP
 | 
					
						
							|  |  |  |    can have one of 4 values: | 
					
						
							|  |  |  | 	RP_LONG_LEFT	remove longest matching portion at start of PARAM | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	RP_SHORT_LEFT	remove shortest matching portion at start of PARAM | 
					
						
							|  |  |  | 	RP_LONG_RIGHT	remove longest matching portion at end of PARAM | 
					
						
							|  |  |  | 	RP_SHORT_RIGHT	remove shortest matching portion at end of PARAM | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define RP_LONG_LEFT	1
 | 
					
						
							|  |  |  | #define RP_SHORT_LEFT	2
 | 
					
						
							|  |  |  | #define RP_LONG_RIGHT	3
 | 
					
						
							|  |  |  | #define RP_SHORT_RIGHT	4
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | /* Returns its first argument if nothing matched; new memory otherwise */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | remove_upattern (param, pattern, op) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *param, *pattern; | 
					
						
							|  |  |  |      int op; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   register size_t len; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register char *end; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   register char *p, *ret, c; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   len = STRLEN (param); | 
					
						
							|  |  |  |   end = param + len; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (op) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       case RP_LONG_LEFT:	/* remove longest match at start */ | 
					
						
							|  |  |  | 	for (p = end; p >= param; p--) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    c = *p; *p = '\0'; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	    if (strmatch (pattern, param, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      { | 
					
						
							|  |  |  | 		*p = c; | 
					
						
							|  |  |  | 		return (savestring (p)); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	    *p = c; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RP_SHORT_LEFT:	/* remove shortest match at start */ | 
					
						
							|  |  |  | 	for (p = param; p <= end; p++) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    c = *p; *p = '\0'; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	    if (strmatch (pattern, param, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      { | 
					
						
							|  |  |  | 		*p = c; | 
					
						
							|  |  |  | 		return (savestring (p)); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	    *p = c; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       case RP_LONG_RIGHT:	/* remove longest match at end */ | 
					
						
							|  |  |  | 	for (p = param; p <= end; p++) | 
					
						
							|  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	    if (strmatch (pattern, p, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      { | 
					
						
							|  |  |  | 		c = *p; *p = '\0'; | 
					
						
							|  |  |  | 		ret = savestring (param); | 
					
						
							|  |  |  | 		*p = c; | 
					
						
							|  |  |  | 		return (ret); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RP_SHORT_RIGHT:	/* remove shortest match at end */ | 
					
						
							|  |  |  | 	for (p = end; p >= param; p--) | 
					
						
							|  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	    if (strmatch (pattern, p, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      { | 
					
						
							|  |  |  | 		c = *p; *p = '\0'; | 
					
						
							|  |  |  | 		ret = savestring (param); | 
					
						
							|  |  |  | 		*p = c; | 
					
						
							|  |  |  | 		return (ret); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   return (param);	/* no match, return original string */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | /* Returns its first argument if nothing matched; new memory otherwise */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static wchar_t * | 
					
						
							|  |  |  | remove_wpattern (wparam, wstrlen, wpattern, op) | 
					
						
							|  |  |  |      wchar_t *wparam; | 
					
						
							|  |  |  |      size_t wstrlen; | 
					
						
							|  |  |  |      wchar_t *wpattern; | 
					
						
							|  |  |  |      int op; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   wchar_t wc, *ret; | 
					
						
							|  |  |  |   int n; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (op) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       case RP_LONG_LEFT:	/* remove longest match at start */ | 
					
						
							|  |  |  |         for (n = wstrlen; n >= 0; n--) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    wc = wparam[n]; wparam[n] = L'\0'; | 
					
						
							|  |  |  | 	    if (wcsmatch (wpattern, wparam, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		wparam[n] = wc; | 
					
						
							|  |  |  | 		return (wcsdup (wparam + n)); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	    wparam[n] = wc; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RP_SHORT_LEFT:	/* remove shortest match at start */ | 
					
						
							|  |  |  | 	for (n = 0; n <= wstrlen; n++) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    wc = wparam[n]; wparam[n] = L'\0'; | 
					
						
							|  |  |  | 	    if (wcsmatch (wpattern, wparam, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		wparam[n] = wc; | 
					
						
							|  |  |  | 		return (wcsdup (wparam + n)); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	    wparam[n] = wc; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RP_LONG_RIGHT:	/* remove longest match at end */ | 
					
						
							|  |  |  |         for (n = 0; n <= wstrlen; n++) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    if (wcsmatch (wpattern, wparam + n, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		wc = wparam[n]; wparam[n] = L'\0'; | 
					
						
							|  |  |  | 		ret = wcsdup (wparam); | 
					
						
							|  |  |  | 		wparam[n] = wc; | 
					
						
							|  |  |  | 		return (ret); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       case RP_SHORT_RIGHT:	/* remove shortest match at end */ | 
					
						
							|  |  |  | 	for (n = wstrlen; n >= 0; n--) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    if (wcsmatch (wpattern, wparam + n, FNMATCH_EXTFLAG) != FNM_NOMATCH) | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		wc = wparam[n]; wparam[n] = L'\0'; | 
					
						
							|  |  |  | 		ret = wcsdup (wparam); | 
					
						
							|  |  |  | 		wparam[n] = wc; | 
					
						
							|  |  |  | 		return (ret); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   return (wparam);	/* no match, return original string */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | } | 
					
						
							|  |  |  | #endif /* HANDLE_MULTIBYTE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | remove_pattern (param, pattern, op) | 
					
						
							|  |  |  |      char *param, *pattern; | 
					
						
							|  |  |  |      int op; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   char *xret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (param == NULL) | 
					
						
							|  |  |  |     return (param); | 
					
						
							|  |  |  |   if (*param == '\0' || pattern == NULL || *pattern == '\0')	/* minor optimization */ | 
					
						
							|  |  |  |     return (savestring (param)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (MB_CUR_MAX > 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       wchar_t *ret, *oret; | 
					
						
							|  |  |  |       size_t n; | 
					
						
							|  |  |  |       wchar_t *wparam, *wpattern; | 
					
						
							|  |  |  |       mbstate_t ps; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       n = xdupmbstowcs (&wpattern, NULL, pattern); | 
					
						
							|  |  |  |       if (n == (size_t)-1) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  xret = remove_upattern (param, pattern, op); | 
					
						
							|  |  |  | 	  return ((xret == param) ? savestring (param) : xret); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       n = xdupmbstowcs (&wparam, NULL, param); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (n == (size_t)-1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  free (wpattern); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  xret = remove_upattern (param, pattern, op); | 
					
						
							|  |  |  | 	  return ((xret == param) ? savestring (param) : xret); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       oret = ret = remove_wpattern (wparam, n, wpattern, op); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       /* Don't bother to convert wparam back to multibyte string if nothing
 | 
					
						
							|  |  |  | 	 matched; just return copy of original string */ | 
					
						
							|  |  |  |       if (ret == wparam) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           free (wparam); | 
					
						
							|  |  |  |           free (wpattern); | 
					
						
							|  |  |  |           return (savestring (param)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       free (wparam); | 
					
						
							|  |  |  |       free (wpattern); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       n = strlen (param); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       xret = (char *)xmalloc (n + 1); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       memset (&ps, '\0', sizeof (mbstate_t)); | 
					
						
							|  |  |  |       n = wcsrtombs (xret, (const wchar_t **)&ret, n, &ps); | 
					
						
							|  |  |  |       xret[n] = '\0';		/* just to make sure */ | 
					
						
							|  |  |  |       free (oret); | 
					
						
							|  |  |  |       return xret;       | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       xret = remove_upattern (param, pattern, op); | 
					
						
							|  |  |  |       return ((xret == param) ? savestring (param) : xret); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Match PAT anywhere in STRING and return the match boundaries.
 | 
					
						
							|  |  |  |    This returns 1 in case of a successful match, 0 otherwise.  SP | 
					
						
							|  |  |  |    and EP are pointers into the string where the match begins and | 
					
						
							|  |  |  |    ends, respectively.  MTYPE controls what kind of match is attempted. | 
					
						
							|  |  |  |    MATCH_BEG and MATCH_END anchor the match at the beginning and end | 
					
						
							|  |  |  |    of the string, respectively.  The longest match is returned. */ | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | match_upattern (string, pat, mtype, sp, ep) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string, *pat; | 
					
						
							|  |  |  |      int mtype; | 
					
						
							|  |  |  |      char **sp, **ep; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int c, mlen; | 
					
						
							|  |  |  |   size_t len; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   register char *p, *p1, *npat; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *end; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   int n1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* If the pattern doesn't match anywhere in the string, go ahead and
 | 
					
						
							|  |  |  |      short-circuit right away.  A minor optimization, saves a bunch of | 
					
						
							|  |  |  |      unnecessary calls to strmatch (up to N calls for a string of N | 
					
						
							|  |  |  |      characters) if the match is unsuccessful.  To preserve the semantics | 
					
						
							|  |  |  |      of the substring matches below, we make sure that the pattern has | 
					
						
							|  |  |  |      `*' as first and last character, making a new pattern if necessary. */ | 
					
						
							|  |  |  |   /* XXX - check this later if I ever implement `**' with special meaning,
 | 
					
						
							|  |  |  |      since this will potentially result in `**' at the beginning or end */ | 
					
						
							|  |  |  |   len = STRLEN (pat); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (pat[0] != '*' || (pat[0] == '*' && pat[1] == LPAREN && extended_glob) || pat[len - 1] != '*') | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       int unescaped_backslash; | 
					
						
							|  |  |  |       char *pp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       p = npat = (char *)xmalloc (len + 3); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       p1 = pat; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if (*p1 != '*' || (*p1 == '*' && p1[1] == LPAREN && extended_glob)) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	*p++ = '*'; | 
					
						
							|  |  |  |       while (*p1) | 
					
						
							|  |  |  | 	*p++ = *p1++; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  |       /* Need to also handle a pattern that ends with an unescaped backslash.
 | 
					
						
							|  |  |  | 	 For right now, we ignore it because the pattern matching code will | 
					
						
							|  |  |  | 	 fail the match anyway */ | 
					
						
							|  |  |  |       /* If the pattern ends with a `*' we leave it alone if it's preceded by
 | 
					
						
							|  |  |  | 	 an even number of backslashes, but if it's escaped by a backslash | 
					
						
							|  |  |  | 	 we need to add another `*'. */ | 
					
						
							|  |  |  |       if (p1[-1] == '*' && (unescaped_backslash = p1[-2] == '\\')) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  pp = p1 - 3; | 
					
						
							|  |  |  | 	  while (pp >= pat && *pp-- == '\\') | 
					
						
							|  |  |  | 	    unescaped_backslash = 1 - unescaped_backslash; | 
					
						
							|  |  |  | 	  if (unescaped_backslash) | 
					
						
							|  |  |  | 	    *p++ = '*'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if (p1[-1] != '*') | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	*p++ = '*'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #else 
 | 
					
						
							|  |  |  |       if (p1[-1] != '*' || p1[-2] == '\\') | 
					
						
							|  |  |  | 	*p++ = '*'; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       *p = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     npat = pat; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   c = strmatch (npat, string, FNMATCH_EXTFLAG | FNMATCH_IGNCASE); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (npat != pat) | 
					
						
							|  |  |  |     free (npat); | 
					
						
							|  |  |  |   if (c == FNM_NOMATCH) | 
					
						
							|  |  |  |     return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   len = STRLEN (string); | 
					
						
							|  |  |  |   end = string + len; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   mlen = umatchlen (pat, len); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   switch (mtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case MATCH_ANY: | 
					
						
							|  |  |  |       for (p = string; p <= end; p++) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (match_pattern_char (pat, p, FNMATCH_IGNCASE)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      p1 = (mlen == -1) ? end : p + mlen; | 
					
						
							|  |  |  | 	      /* p1 - p = length of portion of string to be considered
 | 
					
						
							|  |  |  | 	         p = current position in string | 
					
						
							|  |  |  | 	         mlen = number of characters consumed by match (-1 for entire string) | 
					
						
							|  |  |  | 	         end = end of string | 
					
						
							|  |  |  | 	         we want to break immediately if the potential match len | 
					
						
							|  |  |  | 	         is greater than the number of characters remaining in the | 
					
						
							|  |  |  | 	         string | 
					
						
							|  |  |  | 	      */ | 
					
						
							|  |  |  | 	      if (p1 > end) | 
					
						
							|  |  |  | 		break; | 
					
						
							|  |  |  | 	      for ( ; p1 >= p; p1--) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  c = *p1; *p1 = '\0'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  if (strmatch (pat, p, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		    { | 
					
						
							|  |  |  | 		      *p1 = c; | 
					
						
							|  |  |  | 		      *sp = p; | 
					
						
							|  |  |  | 		      *ep = p1; | 
					
						
							|  |  |  | 		      return 1; | 
					
						
							|  |  |  | 		    } | 
					
						
							|  |  |  | 		  *p1 = c; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  | 		  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 		  if (mlen != -1) | 
					
						
							|  |  |  | 		    break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case MATCH_BEG: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (match_pattern_char (pat, string, FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	return (0); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       for (p = (mlen == -1) ? end : string + mlen; p >= string; p--) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  c = *p; *p = '\0'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (strmatch (pat, string, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      *p = c; | 
					
						
							|  |  |  | 	      *sp = string; | 
					
						
							|  |  |  | 	      *ep = p; | 
					
						
							|  |  |  | 	      return 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  *p = c; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 	  if (mlen != -1) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       return (0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     case MATCH_END: | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       for (p = end - ((mlen == -1) ? len : mlen); p <= end; p++) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (strmatch (pat, p, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      *sp = p; | 
					
						
							|  |  |  | 	      *ep = end; | 
					
						
							|  |  |  | 	      return 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 	  if (mlen != -1) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define WFOLD(c) (match_ignore_case && iswupper (c) ? towlower (c) : (c))
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | /* Match WPAT anywhere in WSTRING and return the match boundaries.
 | 
					
						
							|  |  |  |    This returns 1 in case of a successful match, 0 otherwise.  Wide | 
					
						
							|  |  |  |    character version. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | match_wpattern (wstring, indices, wstrlen, wpat, mtype, sp, ep) | 
					
						
							|  |  |  |      wchar_t *wstring; | 
					
						
							|  |  |  |      char **indices; | 
					
						
							|  |  |  |      size_t wstrlen; | 
					
						
							|  |  |  |      wchar_t *wpat; | 
					
						
							|  |  |  |      int mtype; | 
					
						
							|  |  |  |      char **sp, **ep; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   wchar_t wc, *wp, *nwpat, *wp1; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   size_t len; | 
					
						
							|  |  |  |   int mlen; | 
					
						
							|  |  |  |   int n, n1, n2, simple; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   simple = (wpat[0] != L'\\' && wpat[0] != L'*' && wpat[0] != L'?' && wpat[0] != L'['); | 
					
						
							|  |  |  | #if defined (EXTENDED_GLOB)
 | 
					
						
							|  |  |  |   if (extended_glob) | 
					
						
							| 
									
										
										
										
											2012-05-07 16:23:18 -04:00
										 |  |  |     simple &= (wpat[1] != L'(' || (wpat[0] != L'*' && wpat[0] != L'?' && wpat[0] != L'+' && wpat[0] != L'!' && wpat[0] != L'@')); /*)*/ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* If the pattern doesn't match anywhere in the string, go ahead and
 | 
					
						
							|  |  |  |      short-circuit right away.  A minor optimization, saves a bunch of | 
					
						
							|  |  |  |      unnecessary calls to strmatch (up to N calls for a string of N | 
					
						
							|  |  |  |      characters) if the match is unsuccessful.  To preserve the semantics | 
					
						
							|  |  |  |      of the substring matches below, we make sure that the pattern has | 
					
						
							|  |  |  |      `*' as first and last character, making a new pattern if necessary. */ | 
					
						
							|  |  |  |   len = wcslen (wpat); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (wpat[0] != L'*' || (wpat[0] == L'*' && wpat[1] == WLPAREN && extended_glob) || wpat[len - 1] != L'*') | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       int unescaped_backslash; | 
					
						
							|  |  |  |       wchar_t *wpp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       wp = nwpat = (wchar_t *)xmalloc ((len + 3) * sizeof (wchar_t)); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       wp1 = wpat; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if (*wp1 != L'*' || (*wp1 == '*' && wp1[1] == WLPAREN && extended_glob)) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	*wp++ = L'*'; | 
					
						
							|  |  |  |       while (*wp1 != L'\0') | 
					
						
							|  |  |  | 	*wp++ = *wp1++; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  |       /* See comments above in match_upattern. */ | 
					
						
							|  |  |  |       if (wp1[-1] == L'*' && (unescaped_backslash = wp1[-2] == L'\\')) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  |           wpp = wp1 - 3; | 
					
						
							|  |  |  |           while (wpp >= wpat && *wpp-- == L'\\') | 
					
						
							|  |  |  |             unescaped_backslash = 1 - unescaped_backslash; | 
					
						
							|  |  |  |           if (unescaped_backslash) | 
					
						
							|  |  |  |             *wp++ = L'*'; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else if (wp1[-1] != L'*') | 
					
						
							|  |  |  |         *wp++ = L'*'; | 
					
						
							|  |  |  | #else      
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (wp1[-1] != L'*' || wp1[-2] == L'\\') | 
					
						
							|  |  |  |         *wp++ = L'*'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       *wp = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     nwpat = wpat; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   len = wcsmatch (nwpat, wstring, FNMATCH_EXTFLAG | FNMATCH_IGNCASE); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (nwpat != wpat) | 
					
						
							|  |  |  |     free (nwpat); | 
					
						
							|  |  |  |   if (len == FNM_NOMATCH) | 
					
						
							|  |  |  |     return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   mlen = wmatchlen (wpat, wstrlen); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* itrace("wmatchlen (%ls) -> %d", wpat, mlen); */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   switch (mtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case MATCH_ANY: | 
					
						
							|  |  |  |       for (n = 0; n <= wstrlen; n++) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  n2 = simple ? (WFOLD(*wpat) == WFOLD(wstring[n])) : match_pattern_wchar (wpat, wstring + n, FNMATCH_IGNCASE); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (n2) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      n1 = (mlen == -1) ? wstrlen : n + mlen; | 
					
						
							|  |  |  | 	      if (n1 > wstrlen) | 
					
						
							|  |  |  | 	        break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      for ( ; n1 >= n; n1--) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  wc = wstring[n1]; wstring[n1] = L'\0'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  if (wcsmatch (wpat, wstring + n, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		    { | 
					
						
							|  |  |  | 		      wstring[n1] = wc; | 
					
						
							|  |  |  | 		      *sp = indices[n]; | 
					
						
							|  |  |  | 		      *ep = indices[n1]; | 
					
						
							|  |  |  | 		      return 1; | 
					
						
							|  |  |  | 		    } | 
					
						
							|  |  |  | 		  wstring[n1] = wc; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 		  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 		  if (mlen != -1) | 
					
						
							|  |  |  | 		    break; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case MATCH_BEG: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (match_pattern_wchar (wpat, wstring, FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       for (n = (mlen == -1) ? wstrlen : mlen; n >= 0; n--) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  wc = wstring[n]; wstring[n] = L'\0'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (wcsmatch (wpat, wstring, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      wstring[n] = wc; | 
					
						
							|  |  |  | 	      *sp = indices[0]; | 
					
						
							|  |  |  | 	      *ep = indices[n]; | 
					
						
							|  |  |  | 	      return 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  wstring[n] = wc; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 	  if (mlen != -1) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case MATCH_END: | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       for (n = wstrlen - ((mlen == -1) ? wstrlen : mlen); n <= wstrlen; n++) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (wcsmatch (wpat, wstring + n, FNMATCH_EXTFLAG | FNMATCH_IGNCASE) == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      *sp = indices[n]; | 
					
						
							|  |  |  | 	      *ep = indices[wstrlen]; | 
					
						
							|  |  |  | 	      return 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* If MLEN != -1, we have a fixed length pattern. */ | 
					
						
							|  |  |  | 	  if (mlen != -1) | 
					
						
							|  |  |  | 	    break; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       return (0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #undef WFOLD
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #endif /* HANDLE_MULTIBYTE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | match_pattern (string, pat, mtype, sp, ep) | 
					
						
							|  |  |  |      char *string, *pat; | 
					
						
							|  |  |  |      int mtype; | 
					
						
							|  |  |  |      char **sp, **ep; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   int ret; | 
					
						
							|  |  |  |   size_t n; | 
					
						
							|  |  |  |   wchar_t *wstring, *wpat; | 
					
						
							|  |  |  |   char **indices; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   size_t slen, plen, mslen, mplen; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (string == 0 || pat == 0 || *pat == 0) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     return (0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (MB_CUR_MAX > 1) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       if (mbsmbchar (string) == 0 && mbsmbchar (pat) == 0) | 
					
						
							|  |  |  |         return (match_upattern (string, pat, mtype, sp, ep)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       n = xdupmbstowcs (&wpat, NULL, pat); | 
					
						
							|  |  |  |       if (n == (size_t)-1) | 
					
						
							|  |  |  | 	return (match_upattern (string, pat, mtype, sp, ep)); | 
					
						
							|  |  |  |       n = xdupmbstowcs (&wstring, &indices, string); | 
					
						
							|  |  |  |       if (n == (size_t)-1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  free (wpat); | 
					
						
							|  |  |  | 	  return (match_upattern (string, pat, mtype, sp, ep)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       ret = match_wpattern (wstring, indices, n, wpat, mtype, sp, ep); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       free (wpat); | 
					
						
							|  |  |  |       free (wstring); | 
					
						
							|  |  |  |       free (indices); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (ret); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     return (match_upattern (string, pat, mtype, sp, ep)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | getpatspec (c, value) | 
					
						
							|  |  |  |      int c; | 
					
						
							|  |  |  |      char *value; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (c == '#') | 
					
						
							|  |  |  |     return ((*value == '#') ? RP_LONG_LEFT : RP_SHORT_LEFT); | 
					
						
							|  |  |  |   else	/* c == '%' */ | 
					
						
							|  |  |  |     return ((*value == '%') ? RP_LONG_RIGHT : RP_SHORT_RIGHT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Posix.2 says that the WORD should be run through tilde expansion,
 | 
					
						
							|  |  |  |    parameter expansion, command substitution and arithmetic expansion. | 
					
						
							|  |  |  |    This leaves the result quoted, so quote_string_for_globbing () has | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |    to be called to fix it up for strmatch ().  If QUOTED is non-zero, | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    it means that the entire expression was enclosed in double quotes. | 
					
						
							|  |  |  |    This means that quoting characters in the pattern do not make any | 
					
						
							|  |  |  |    special pattern characters quoted.  For example, the `*' in the | 
					
						
							|  |  |  |    following retains its special meaning: "${foo#'*'}". */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | getpattern (value, quoted, expandpat) | 
					
						
							|  |  |  |      char *value; | 
					
						
							|  |  |  |      int quoted, expandpat; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *pat, *tword; | 
					
						
							|  |  |  |   WORD_LIST *l; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int i; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* There is a problem here:  how to handle single or double quotes in the
 | 
					
						
							|  |  |  |      pattern string when the whole expression is between double quotes? | 
					
						
							|  |  |  |      POSIX.2 says that enclosing double quotes do not cause the pattern to | 
					
						
							|  |  |  |      be quoted, but does that leave us a problem with @ and array[@] and their | 
					
						
							|  |  |  |      expansions inside a pattern? */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (expandpat && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && *tword) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       i = 0; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       pat = string_extract_double_quoted (tword, &i, SX_STRIPDQ); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       free (tword); | 
					
						
							|  |  |  |       tword = pat; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* expand_string_for_rhs () leaves WORD quoted and does not perform
 | 
					
						
							|  |  |  |      word splitting. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   l = *value ? expand_string_for_rhs (value, | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 				      (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) ? Q_PATQUOTE : quoted, | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 				      (int *)NULL, (int *)NULL) | 
					
						
							|  |  |  | 	     : (WORD_LIST *)0; | 
					
						
							|  |  |  |   pat = string_list (l); | 
					
						
							|  |  |  |   dispose_words (l); | 
					
						
							|  |  |  |   if (pat) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tword = quote_string_for_globbing (pat, QGLOB_CVTNULL); | 
					
						
							|  |  |  |       free (pat); | 
					
						
							|  |  |  |       pat = tword; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (pat); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Handle removing a pattern from a string as a result of ${name%[%]value}
 | 
					
						
							|  |  |  |    or ${name#[#]value}. */ | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | variable_remove_pattern (value, pattern, patspec, quoted) | 
					
						
							|  |  |  |      char *value, *pattern; | 
					
						
							|  |  |  |      int patspec, quoted; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   char *tword; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   tword = remove_pattern (value, pattern, patspec); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return (tword); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | list_remove_pattern (list, pattern, patspec, itype, quoted) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      char *pattern; | 
					
						
							|  |  |  |      int patspec, itype, quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *new, *l; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   char *tword; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (new = (WORD_LIST *)NULL, l = list; l; l = l->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tword = remove_pattern (l->word->word, pattern, patspec); | 
					
						
							|  |  |  |       w = alloc_word_desc (); | 
					
						
							|  |  |  |       w->word = tword ? tword : savestring (""); | 
					
						
							|  |  |  |       new = make_word_list (w, new); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   l = REVERSE_LIST (new, WORD_LIST *); | 
					
						
							|  |  |  |   tword = string_list_pos_params (itype, l, quoted); | 
					
						
							|  |  |  |   dispose_words (l); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (tword); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | parameter_list_remove_pattern (itype, pattern, patspec, quoted) | 
					
						
							|  |  |  |      int itype; | 
					
						
							|  |  |  |      char *pattern; | 
					
						
							|  |  |  |      int patspec, quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   list = list_rest_of_args (); | 
					
						
							|  |  |  |   if (list == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  |   ret = list_remove_pattern (list, pattern, patspec, itype, quoted); | 
					
						
							|  |  |  |   dispose_words (list); | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | array_remove_pattern (var, pattern, patspec, varname, quoted) | 
					
						
							|  |  |  |      SHELL_VAR *var; | 
					
						
							|  |  |  |      char *pattern; | 
					
						
							|  |  |  |      int patspec; | 
					
						
							|  |  |  |      char *varname;	/* so we can figure out how it's indexed */ | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ARRAY *a; | 
					
						
							|  |  |  |   HASH_TABLE *h; | 
					
						
							|  |  |  |   int itype; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* compute itype from varname here */ | 
					
						
							|  |  |  |   v = array_variable_part (varname, &ret, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* XXX */ | 
					
						
							|  |  |  |   if (v && invisible_p (v)) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   itype = ret[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   a = (v && array_p (v)) ? array_cell (v) : 0; | 
					
						
							|  |  |  |   h = (v && assoc_p (v)) ? assoc_cell (v) : 0; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   list = a ? array_to_word_list (a) : (h ? assoc_to_word_list (h) : 0); | 
					
						
							|  |  |  |   if (list == 0) | 
					
						
							|  |  |  |    return ((char *)NULL); | 
					
						
							|  |  |  |   ret = list_remove_pattern (list, pattern, patspec, itype, quoted); | 
					
						
							|  |  |  |   dispose_words (list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | parameter_brace_remove_pattern (varname, value, ind, patstr, rtype, quoted, flags) | 
					
						
							|  |  |  |      char *varname, *value; | 
					
						
							|  |  |  |      int ind; | 
					
						
							|  |  |  |      char *patstr; | 
					
						
							|  |  |  |      int rtype, quoted, flags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int vtype, patspec, starsub; | 
					
						
							|  |  |  |   char *temp1, *val, *pattern; | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   this_command_name = varname; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   vtype = get_var_and_type (varname, value, ind, quoted, flags, &v, &val); | 
					
						
							|  |  |  |   if (vtype == -1) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   starsub = vtype & VT_STARSUB; | 
					
						
							|  |  |  |   vtype &= ~VT_STARSUB; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   patspec = getpatspec (rtype, patstr); | 
					
						
							|  |  |  |   if (patspec == RP_LONG_LEFT || patspec == RP_LONG_RIGHT) | 
					
						
							|  |  |  |     patstr++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Need to pass getpattern newly-allocated memory in case of expansion --
 | 
					
						
							|  |  |  |      the expansion code will free the passed string on an error. */ | 
					
						
							|  |  |  |   temp1 = savestring (patstr); | 
					
						
							|  |  |  |   pattern = getpattern (temp1, quoted, 1); | 
					
						
							|  |  |  |   free (temp1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   temp1 = (char *)NULL;		/* shut up gcc */ | 
					
						
							|  |  |  |   switch (vtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case VT_VARIABLE: | 
					
						
							|  |  |  |     case VT_ARRAYMEMBER: | 
					
						
							|  |  |  |       temp1 = remove_pattern (val, pattern, patspec); | 
					
						
							|  |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       if (temp1) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  val = (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 			? quote_string (temp1) | 
					
						
							|  |  |  | 			: quote_escapes (temp1); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							|  |  |  |       temp1 = array_remove_pattern (v, pattern, patspec, varname, quoted); | 
					
						
							|  |  |  |       if (temp1 && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  val = quote_escapes (temp1); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							|  |  |  |       temp1 = parameter_list_remove_pattern (varname[0], pattern, patspec, quoted); | 
					
						
							|  |  |  |       if (temp1 && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  val = quote_escapes (temp1); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   FREE (pattern); | 
					
						
							|  |  |  |   return temp1; | 
					
						
							|  |  |  | }     | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | string_var_assignment (v, s) | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char flags[MAX_ATTRIBUTES], *ret, *val; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   val = sh_quote_reusable (s, 0); | 
					
						
							|  |  |  |   i = var_attribute_string (v, 0, flags); | 
					
						
							|  |  |  |   ret = (char *)xmalloc (i + strlen (val) + strlen (v->name) + 16 + MAX_ATTRIBUTES); | 
					
						
							|  |  |  |   if (i > 0) | 
					
						
							|  |  |  |     sprintf (ret, "declare -%s %s=%s", flags, v->name, val); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     sprintf (ret, "%s=%s", v->name, val); | 
					
						
							|  |  |  |   free (val); | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | array_var_assignment (v, itype, quoted) | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							|  |  |  |      int itype, quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret, *val, flags[MAX_ATTRIBUTES]; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (v == 0) | 
					
						
							|  |  |  |     return (char *)NULL; | 
					
						
							|  |  |  |   val = array_p (v) ? array_to_assign (array_cell (v), 0) | 
					
						
							|  |  |  | 		    : assoc_to_assign (assoc_cell (v), 0); | 
					
						
							|  |  |  |   if (val == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       val = (char *)xmalloc (3); | 
					
						
							|  |  |  |       val[0] = '('; | 
					
						
							|  |  |  |       val[1] = ')'; | 
					
						
							|  |  |  |       val[2] = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) ? quote_string (val) : quote_escapes (val); | 
					
						
							|  |  |  |       free (val); | 
					
						
							|  |  |  |       val = ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   i = var_attribute_string (v, 0, flags); | 
					
						
							|  |  |  |   ret = (char *)xmalloc (i + strlen (val) + strlen (v->name) + 16); | 
					
						
							|  |  |  |   sprintf (ret, "declare -%s %s=%s", flags, v->name, val); | 
					
						
							|  |  |  |   free (val); | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | pos_params_assignment (list, itype, quoted) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      int itype; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *temp, *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* first, we transform the list to quote each word. */ | 
					
						
							|  |  |  |   temp = list_transform ('Q', (SHELL_VAR *)0, list, itype, quoted); | 
					
						
							|  |  |  |   ret = (char *)xmalloc (strlen (temp) + 8); | 
					
						
							|  |  |  |   strcpy (ret, "set -- "); | 
					
						
							|  |  |  |   strcpy (ret + 7, temp); | 
					
						
							|  |  |  |   free (temp); | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | string_transform (xc, v, s) | 
					
						
							|  |  |  |      int xc; | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret, flags[MAX_ATTRIBUTES]; | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (((xc == 'A' || xc == 'a') && v == 0) || (xc != 'a' && s == 0)) | 
					
						
							|  |  |  |     return (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   switch (xc) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* Transformations that interrogate the variable */ | 
					
						
							|  |  |  |       case 'a': | 
					
						
							|  |  |  | 	i = var_attribute_string (v, 0, flags); | 
					
						
							|  |  |  | 	ret = (i > 0) ? savestring (flags) : (char *)NULL; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       case 'A': | 
					
						
							|  |  |  | 	ret = string_var_assignment (v, s); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       /* Transformations that modify the variable's value */ | 
					
						
							|  |  |  |       case 'E': | 
					
						
							|  |  |  | 	ret = ansiexpand (s, 0, strlen (s), (int *)0); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       case 'P': | 
					
						
							|  |  |  | 	ret = decode_prompt_string (s); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       case 'Q': | 
					
						
							|  |  |  | 	ret = sh_quote_reusable (s, 0); | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  | 	ret = (char *)NULL; | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | list_transform (xc, v, list, itype, quoted) | 
					
						
							|  |  |  |      int xc; | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |      WORD_LIST *list; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int itype, quoted; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *new, *l; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   char *tword; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (new = (WORD_LIST *)NULL, l = list; l; l = l->next) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       tword = string_transform (xc, v, l->word->word); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       w = alloc_word_desc (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       w->word = tword ? tword : savestring ("");	/* XXX */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       new = make_word_list (w, new); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   l = REVERSE_LIST (new, WORD_LIST *); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   tword = string_list_pos_params (itype, l, quoted); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   dispose_words (l); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (tword); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | parameter_list_transform (xc, itype, quoted) | 
					
						
							|  |  |  |      int xc; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      int itype; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   char *ret; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   list = list_rest_of_args (); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (list == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (xc == 'A') | 
					
						
							|  |  |  |     return (pos_params_assignment (list, itype, quoted)); | 
					
						
							|  |  |  |   ret = list_transform (xc, (SHELL_VAR *)0, list, itype, quoted); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   dispose_words (list); | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | array_transform (xc, var, varname, quoted) | 
					
						
							|  |  |  |      int xc; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      SHELL_VAR *var; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      char *varname;	/* so we can figure out how it's indexed */ | 
					
						
							|  |  |  |      int quoted; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   ARRAY *a; | 
					
						
							|  |  |  |   HASH_TABLE *h; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   int itype; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* compute itype from varname here */ | 
					
						
							|  |  |  |   v = array_variable_part (varname, &ret, 0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* XXX */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (v && invisible_p (v)) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   itype = ret[0]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (xc == 'A') | 
					
						
							|  |  |  |     return (array_var_assignment (v, itype, quoted)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   a = (v && array_p (v)) ? array_cell (v) : 0; | 
					
						
							|  |  |  |   h = (v && assoc_p (v)) ? assoc_cell (v) : 0; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   list = a ? array_to_word_list (a) : (h ? assoc_to_word_list (h) : 0); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (list == 0) | 
					
						
							|  |  |  |    return ((char *)NULL); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   ret = list_transform (xc, v, list, itype, quoted); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   dispose_words (list); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | parameter_brace_transform (varname, value, ind, xform, rtype, quoted, flags) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      char *varname, *value; | 
					
						
							|  |  |  |      int ind; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      char *xform; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      int rtype, quoted, flags; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int vtype, xc; | 
					
						
							|  |  |  |   char *temp1, *val; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   xc = xform[0]; | 
					
						
							|  |  |  |   if (value == 0 && xc != 'A' && xc != 'a') | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   this_command_name = varname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   vtype = get_var_and_type (varname, value, ind, quoted, flags, &v, &val); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   if (vtype == -1) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* check for valid values of xc */ | 
					
						
							|  |  |  |   switch (xc) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case 'a':		/* expand to a string with just attributes */ | 
					
						
							|  |  |  |     case 'A':		/* expand as an assignment statement with attributes */ | 
					
						
							|  |  |  |     case 'E':		/* expand like $'...' */ | 
					
						
							|  |  |  |     case 'P':		/* expand like prompt string */ | 
					
						
							|  |  |  |     case 'Q':		/* quote reusably */ | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       return &expand_param_error; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   temp1 = (char *)NULL;		/* shut up gcc */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   switch (vtype & ~VT_STARSUB) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     case VT_VARIABLE: | 
					
						
							|  |  |  |     case VT_ARRAYMEMBER: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = string_transform (xc, v, val); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       if (temp1) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  val = (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 			? quote_string (temp1) | 
					
						
							|  |  |  | 			: quote_escapes (temp1); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = array_transform (xc, v, varname, quoted); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (temp1 && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  val = quote_escapes (temp1); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = parameter_list_transform (xc, varname[0], quoted); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (temp1 && ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) == 0)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  val = quote_escapes (temp1); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  temp1 = val; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   return temp1; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /*******************************************
 | 
					
						
							|  |  |  |  *					   * | 
					
						
							|  |  |  |  *	Functions to expand WORD_DESCs	   * | 
					
						
							|  |  |  |  *					   * | 
					
						
							|  |  |  |  *******************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand WORD, performing word splitting on the result.  This does
 | 
					
						
							|  |  |  |    parameter expansion, command substitution, arithmetic expansion, | 
					
						
							|  |  |  |    word splitting, and quote removal. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_word (word, quoted) | 
					
						
							|  |  |  |      WORD_DESC *word; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *result, *tresult; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   tresult = call_expand_word_internal (word, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   result = word_list_split (tresult); | 
					
						
							|  |  |  |   dispose_words (tresult); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (result ? dequote_list (result) : result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Expand WORD, but do not perform word splitting on the result.  This
 | 
					
						
							|  |  |  |    does parameter expansion, command substitution, arithmetic expansion, | 
					
						
							|  |  |  |    and quote removal. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | expand_word_unsplit (word, quoted) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      WORD_DESC *word; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (ifs_firstc[0] == 0) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (ifs_firstc == 0) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     word->flags |= W_NOSPLIT; | 
					
						
							| 
									
										
										
										
											2011-11-22 20:00:41 -05:00
										 |  |  |   word->flags |= W_NOSPLIT2; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   result = call_expand_word_internal (word, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (result ? dequote_list (result) : result); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform shell expansions on WORD, but do not perform word splitting or
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |    quote removal on the result.  Virtually identical to expand_word_unsplit; | 
					
						
							|  |  |  |    could be combined if implementations don't diverge. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_word_leave_quoted (word, quoted) | 
					
						
							|  |  |  |      WORD_DESC *word; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 1; | 
					
						
							|  |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (ifs_firstc[0] == 0) | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   if (ifs_firstc == 0) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     word->flags |= W_NOSPLIT; | 
					
						
							|  |  |  |   word->flags |= W_NOSPLIT2; | 
					
						
							|  |  |  |   result = call_expand_word_internal (word, quoted, 0, (int *)NULL, (int *)NULL); | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return result; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /*****************************************************************/ | 
					
						
							|  |  |  | /*								 */ | 
					
						
							|  |  |  | /*		    Hacking Process Substitution		 */ | 
					
						
							|  |  |  | /*								 */ | 
					
						
							|  |  |  | /*****************************************************************/ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if !defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  | /* Named pipes must be removed explicitly with `unlink'.  This keeps a list
 | 
					
						
							|  |  |  |    of FIFOs the shell has open.  unlink_fifo_list will walk the list and | 
					
						
							|  |  |  |    unlink all of them. add_fifo_list adds the name of an open FIFO to the | 
					
						
							|  |  |  |    list.  NFIFO is a count of the number of FIFOs in the list. */ | 
					
						
							|  |  |  | #define FIFO_INCR 20
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | struct temp_fifo { | 
					
						
							|  |  |  |   char *file; | 
					
						
							|  |  |  |   pid_t proc; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct temp_fifo *fifo_list = (struct temp_fifo *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static int nfifo; | 
					
						
							|  |  |  | static int fifo_list_size; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | void | 
					
						
							|  |  |  | clear_fifo_list () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | char * | 
					
						
							|  |  |  | copy_fifo_list (sizep) | 
					
						
							|  |  |  |      int *sizep; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (sizep) | 
					
						
							|  |  |  |     *sizep = 0; | 
					
						
							|  |  |  |   return (char *)NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | add_fifo_list (pathname) | 
					
						
							|  |  |  |      char *pathname; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (nfifo >= fifo_list_size - 1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       fifo_list_size += FIFO_INCR; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       fifo_list = (struct temp_fifo *)xrealloc (fifo_list, | 
					
						
							|  |  |  | 				fifo_list_size * sizeof (struct temp_fifo)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   fifo_list[nfifo].file = savestring (pathname); | 
					
						
							|  |  |  |   nfifo++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | void | 
					
						
							|  |  |  | unlink_fifo (i) | 
					
						
							|  |  |  |      int i; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if ((fifo_list[i].proc == -1) || (kill(fifo_list[i].proc, 0) == -1)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       unlink (fifo_list[i].file); | 
					
						
							|  |  |  |       free (fifo_list[i].file); | 
					
						
							|  |  |  |       fifo_list[i].file = (char *)NULL; | 
					
						
							|  |  |  |       fifo_list[i].proc = -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | void | 
					
						
							|  |  |  | unlink_fifo_list () | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int saved, i, j; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (nfifo == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   for (i = saved = 0; i < nfifo; i++) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       if ((fifo_list[i].proc == -1) || (kill(fifo_list[i].proc, 0) == -1)) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  unlink (fifo_list[i].file); | 
					
						
							|  |  |  | 	  free (fifo_list[i].file); | 
					
						
							|  |  |  | 	  fifo_list[i].file = (char *)NULL; | 
					
						
							|  |  |  | 	  fifo_list[i].proc = -1; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	saved++; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If we didn't remove some of the FIFOs, compact the list. */ | 
					
						
							|  |  |  |   if (saved) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       for (i = j = 0; i < nfifo; i++) | 
					
						
							|  |  |  | 	if (fifo_list[i].file) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    fifo_list[j].file = fifo_list[i].file; | 
					
						
							|  |  |  | 	    fifo_list[j].proc = fifo_list[i].proc; | 
					
						
							|  |  |  | 	    j++; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  |       nfifo = j; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     nfifo = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | /* Take LIST, which is a bitmap denoting active FIFOs in fifo_list
 | 
					
						
							|  |  |  |    from some point in the past, and close all open FIFOs in fifo_list | 
					
						
							|  |  |  |    that are not marked as active in LIST.  If LIST is NULL, close | 
					
						
							|  |  |  |    everything in fifo_list. LSIZE is the number of elements in LIST, in | 
					
						
							|  |  |  |    case it's larger than fifo_list_size (size of fifo_list). */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | close_new_fifos (list, lsize) | 
					
						
							|  |  |  |      char *list; | 
					
						
							|  |  |  |      int lsize; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (list == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       unlink_fifo_list (); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < lsize; i++) | 
					
						
							|  |  |  |     if (list[i] == 0 && i < fifo_list_size && fifo_list[i].proc != -1) | 
					
						
							|  |  |  |       unlink_fifo (i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = lsize; i < fifo_list_size; i++) | 
					
						
							|  |  |  |     unlink_fifo (i);   | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | int | 
					
						
							|  |  |  | fifos_pending () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return nfifo; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | int | 
					
						
							|  |  |  | num_fifos () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return nfifo; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static char * | 
					
						
							|  |  |  | make_named_pipe () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *tname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   tname = sh_mktmpname ("sh-np", MT_USERANDOM|MT_USETMPDIR); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (mkfifo (tname, 0600) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       free (tname); | 
					
						
							|  |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   add_fifo_list (tname); | 
					
						
							|  |  |  |   return (tname); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* DEV_FD_LIST is a bitmap of file descriptors attached to pipes the shell
 | 
					
						
							|  |  |  |    has open to children.  NFDS is a count of the number of bits currently | 
					
						
							|  |  |  |    set in DEV_FD_LIST.  TOTFDS is a count of the highest possible number | 
					
						
							|  |  |  |    of open files. */ | 
					
						
							|  |  |  | static char *dev_fd_list = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static int nfds; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static int totfds;	/* The highest possible number of open files. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | void | 
					
						
							|  |  |  | clear_fifo (i) | 
					
						
							|  |  |  |      int i; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (dev_fd_list[i]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       dev_fd_list[i] = 0; | 
					
						
							|  |  |  |       nfds--; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | clear_fifo_list () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (nfds == 0) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; nfds && i < totfds; i++) | 
					
						
							|  |  |  |     clear_fifo (i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   nfds = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | char * | 
					
						
							|  |  |  | copy_fifo_list (sizep) | 
					
						
							|  |  |  |      int *sizep; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (nfds == 0 || totfds == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (sizep) | 
					
						
							|  |  |  | 	*sizep = 0; | 
					
						
							|  |  |  |       return (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sizep) | 
					
						
							|  |  |  |     *sizep = totfds; | 
					
						
							|  |  |  |   ret = (char *)xmalloc (totfds); | 
					
						
							|  |  |  |   return (memcpy (ret, dev_fd_list, totfds)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | add_fifo_list (fd) | 
					
						
							|  |  |  |      int fd; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   if (dev_fd_list == 0 || fd >= totfds) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       int ofds; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ofds = totfds; | 
					
						
							|  |  |  |       totfds = getdtablesize (); | 
					
						
							|  |  |  |       if (totfds < 0 || totfds > 256) | 
					
						
							|  |  |  | 	totfds = 256; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (fd >= totfds) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	totfds = fd + 2; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       dev_fd_list = (char *)xrealloc (dev_fd_list, totfds); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       memset (dev_fd_list + ofds, '\0', totfds - ofds); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dev_fd_list[fd] = 1; | 
					
						
							|  |  |  |   nfds++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | int | 
					
						
							|  |  |  | fifos_pending () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return 0;	/* used for cleanup; not needed with /dev/fd */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | int | 
					
						
							|  |  |  | num_fifos () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return nfds; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | unlink_fifo (fd) | 
					
						
							|  |  |  |      int fd; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (dev_fd_list[fd]) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       close (fd); | 
					
						
							|  |  |  |       dev_fd_list[fd] = 0; | 
					
						
							|  |  |  |       nfds--; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | void | 
					
						
							|  |  |  | unlink_fifo_list () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (nfds == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; nfds && i < totfds; i++) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |     unlink_fifo (i); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   nfds = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | /* Take LIST, which is a snapshot copy of dev_fd_list from some point in
 | 
					
						
							|  |  |  |    the past, and close all open fds in dev_fd_list that are not marked | 
					
						
							|  |  |  |    as open in LIST.  If LIST is NULL, close everything in dev_fd_list. | 
					
						
							|  |  |  |    LSIZE is the number of elements in LIST, in case it's larger than | 
					
						
							|  |  |  |    totfds (size of dev_fd_list). */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | close_new_fifos (list, lsize) | 
					
						
							|  |  |  |      char *list; | 
					
						
							|  |  |  |      int lsize; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (list == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       unlink_fifo_list (); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < lsize; i++) | 
					
						
							|  |  |  |     if (list[i] == 0 && i < totfds && dev_fd_list[i]) | 
					
						
							|  |  |  |       unlink_fifo (i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = lsize; i < totfds; i++) | 
					
						
							|  |  |  |     unlink_fifo (i);   | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #if defined (NOTDEF)
 | 
					
						
							|  |  |  | print_dev_fd_list () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register int i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   fprintf (stderr, "pid %ld: dev_fd_list:", (long)getpid ()); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   fflush (stderr); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < totfds; i++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (dev_fd_list[i]) | 
					
						
							|  |  |  | 	fprintf (stderr, " %d", i); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   fprintf (stderr, "\n"); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* NOTDEF */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | make_dev_fd_filename (fd) | 
					
						
							|  |  |  |      int fd; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   char *ret, intbuf[INT_STRLEN_BOUND (int) + 1], *p; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  |   ret = (char *)xmalloc (sizeof (DEV_FD_PREFIX) + 8); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   strcpy (ret, DEV_FD_PREFIX); | 
					
						
							|  |  |  |   p = inttostr (fd, intbuf, sizeof (intbuf)); | 
					
						
							|  |  |  |   strcpy (ret + sizeof (DEV_FD_PREFIX) - 1, p); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   add_fifo_list (fd); | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return a filename that will open a connection to the process defined by
 | 
					
						
							|  |  |  |    executing STRING.  HAVE_DEV_FD, if defined, means open a pipe and return | 
					
						
							|  |  |  |    a filename in /dev/fd corresponding to a descriptor that is one of the | 
					
						
							|  |  |  |    ends of the pipe.  If not defined, we use named pipes on systems that have | 
					
						
							|  |  |  |    them.  Systems without /dev/fd and named pipes are out of luck. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    OPEN_FOR_READ_IN_CHILD, if 1, means open the named pipe for reading or | 
					
						
							|  |  |  |    use the read end of the pipe and dup that file descriptor to fd 0 in | 
					
						
							|  |  |  |    the child.  If OPEN_FOR_READ_IN_CHILD is 0, we open the named pipe for | 
					
						
							|  |  |  |    writing or use the write end of the pipe in the child, and dup that | 
					
						
							|  |  |  |    file descriptor to fd 1 in the child.  The parent does the opposite. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | process_substitute (string, open_for_read_in_child) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int open_for_read_in_child; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *pathname; | 
					
						
							|  |  |  |   int fd, result; | 
					
						
							|  |  |  |   pid_t old_pid, pid; | 
					
						
							|  |  |  | #if defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |   int parent_pipe_fd, child_pipe_fd; | 
					
						
							|  |  |  |   int fildes[2]; | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							|  |  |  |   pid_t old_pipeline_pgrp; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (!string || !*string || wordexp_only) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |   pathname = make_named_pipe (); | 
					
						
							|  |  |  | #else /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  |   if (pipe (fildes) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       sys_error ("%s", _("cannot make pipe for process substitution")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   /* If OPEN_FOR_READ_IN_CHILD == 1, we want to use the write end of
 | 
					
						
							|  |  |  |      the pipe in the parent, otherwise the read end. */ | 
					
						
							|  |  |  |   parent_pipe_fd = fildes[open_for_read_in_child]; | 
					
						
							|  |  |  |   child_pipe_fd = fildes[1 - open_for_read_in_child]; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   /* Move the parent end of the pipe to some high file descriptor, to
 | 
					
						
							|  |  |  |      avoid clashes with FDs used by the script. */ | 
					
						
							|  |  |  |   parent_pipe_fd = move_to_high_fd (parent_pipe_fd, 1, 64); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   pathname = make_dev_fd_filename (parent_pipe_fd); | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (pathname == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       sys_error ("%s", _("cannot make pipe for process substitution")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   old_pid = last_made_pid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							|  |  |  |   old_pipeline_pgrp = pipeline_pgrp; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (pipeline_pgrp == 0 || (subshell_environment & (SUBSHELL_PIPE|SUBSHELL_FORK|SUBSHELL_ASYNC)) == 0) | 
					
						
							|  |  |  |     pipeline_pgrp = shell_pgrp; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   save_pipeline (1); | 
					
						
							|  |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   pid = make_child ((char *)NULL, 1); | 
					
						
							|  |  |  |   if (pid == 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       reset_terminating_signals ();	/* XXX */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       free_pushed_string_input (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       /* Cancel traps, in trap.c. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       restore_original_signals ();	/* XXX - what about special builtins? bash-4.2 */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       QUIT;	/* catch any interrupts we got post-fork */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       setup_async_signals (); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       subshell_environment |= SUBSHELL_COMSUB|SUBSHELL_PROCSUB; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* if we're expanding a redirection, we shouldn't have access to the
 | 
					
						
							|  |  |  | 	 temporary environment, but commands in the subshell should have | 
					
						
							|  |  |  | 	 access to their own temporary environment. */ | 
					
						
							|  |  |  |       if (expanding_redir) | 
					
						
							|  |  |  |         flush_temporary_env (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   set_sigchld_handler (); | 
					
						
							|  |  |  |   stop_making_children (); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   /* XXX - should we only do this in the parent? (as in command subst) */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   pipeline_pgrp = old_pipeline_pgrp; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #else
 | 
					
						
							|  |  |  |   stop_making_children (); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (pid < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       sys_error ("%s", _("cannot make child for process substitution")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       free (pathname); | 
					
						
							|  |  |  | #if defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |       close (parent_pipe_fd); | 
					
						
							|  |  |  |       close (child_pipe_fd); | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (pid > 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (last_procsub_child) | 
					
						
							| 
									
										
										
										
											2016-11-14 14:27:35 -05:00
										 |  |  | 	discard_last_procsub_child (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       last_procsub_child = restore_pipeline (0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #if !defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |       fifo_list[nfifo-1].proc = pid; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       last_made_pid = old_pid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL) && defined (PGRP_PIPE)
 | 
					
						
							|  |  |  |       close_pgrp_pipe (); | 
					
						
							|  |  |  | #endif /* JOB_CONTROL && PGRP_PIPE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |       close (child_pipe_fd); | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return (pathname); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   set_sigint_handler (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							|  |  |  |   set_job_control (0); | 
					
						
							|  |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |   /* Open the named pipe in the child. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   fd = open (pathname, open_for_read_in_child ? O_RDONLY : O_WRONLY); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (fd < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       /* Two separate strings for ease of translation. */ | 
					
						
							|  |  |  |       if (open_for_read_in_child) | 
					
						
							|  |  |  | 	sys_error (_("cannot open named pipe %s for reading"), pathname); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	sys_error (_("cannot open named pipe %s for writing"), pathname); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       exit (127); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (open_for_read_in_child) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       if (sh_unset_nodelay_mode (fd) < 0) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  sys_error (_("cannot reset nodelay mode for fd %d"), fd); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  exit (127); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #else /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  |   fd = child_pipe_fd; | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* Discard  buffered stdio output before replacing the underlying file
 | 
					
						
							|  |  |  |      descriptor. */ | 
					
						
							|  |  |  |   if (open_for_read_in_child == 0) | 
					
						
							|  |  |  |     fpurge (stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (dup2 (fd, open_for_read_in_child ? 0 : 1) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       sys_error (_("cannot duplicate named pipe %s as fd %d"), pathname, | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	open_for_read_in_child ? 0 : 1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       exit (127); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   if (fd != (open_for_read_in_child ? 0 : 1)) | 
					
						
							|  |  |  |     close (fd); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Need to close any files that this process has open to pipes inherited
 | 
					
						
							|  |  |  |      from its parent. */ | 
					
						
							|  |  |  |   if (current_fds_to_close) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       close_fd_bitmap (current_fds_to_close); | 
					
						
							|  |  |  |       current_fds_to_close = (struct fd_bitmap *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |   /* Make sure we close the parent's end of the pipe and clear the slot
 | 
					
						
							|  |  |  |      in the fd list so it is not closed later, if reallocated by, for | 
					
						
							|  |  |  |      instance, pipe(2). */ | 
					
						
							|  |  |  |   close (parent_pipe_fd); | 
					
						
							|  |  |  |   dev_fd_list[parent_pipe_fd] = 0; | 
					
						
							|  |  |  | #endif /* HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 19:48:17 -05:00
										 |  |  |   /* subshells shouldn't have this flag, which controls using the temporary
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      environment for variable lookups.  We have already flushed the temporary | 
					
						
							|  |  |  |      environment above in the case we're expanding a redirection, so processes | 
					
						
							|  |  |  |      executed by this command need to be able to set it independently of their | 
					
						
							|  |  |  |      parent. */ | 
					
						
							| 
									
										
										
										
											2013-01-10 19:48:17 -05:00
										 |  |  |   expanding_redir = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   subshell_level++; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   result = parse_and_execute (string, "process substitution", (SEVAL_NONINT|SEVAL_NOHIST)); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   subshell_level--; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if !defined (HAVE_DEV_FD)
 | 
					
						
							|  |  |  |   /* Make sure we close the named pipe in the child before we exit. */ | 
					
						
							|  |  |  |   close (open_for_read_in_child ? 0 : 1); | 
					
						
							|  |  |  | #endif /* !HAVE_DEV_FD */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   last_command_exit_value = result; | 
					
						
							|  |  |  |   result = run_exit_trap (); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   exit (result); | 
					
						
							|  |  |  |   /*NOTREACHED*/ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif /* PROCESS_SUBSTITUTION */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /***********************************/ | 
					
						
							|  |  |  | /*				   */ | 
					
						
							|  |  |  | /*	Command Substitution	   */ | 
					
						
							|  |  |  | /*				   */ | 
					
						
							|  |  |  | /***********************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | read_comsub (fd, quoted, rflag) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |      int fd, quoted; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      int *rflag; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   char *istring, buf[128], *bufp, *s; | 
					
						
							|  |  |  |   int istring_index, istring_size, c, tflag, skip_ctlesc, skip_ctlnul; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   ssize_t bufn; | 
					
						
							| 
									
										
										
										
											2016-11-14 14:27:06 -05:00
										 |  |  |   int nullbyte; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   istring = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   istring_index = istring_size = bufn = tflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (skip_ctlesc = skip_ctlnul = 0, s = ifs_value; s && *s; s++) | 
					
						
							|  |  |  |     skip_ctlesc |= *s == CTLESC, skip_ctlnul |= *s == CTLNUL; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-14 14:27:06 -05:00
										 |  |  |   nullbyte = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   /* Read the output of the command through the pipe.  This may need to be
 | 
					
						
							|  |  |  |      changed to understand multibyte characters in the future. */ | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   while (1) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (fd < 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       if (--bufn <= 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  bufn = zread (fd, buf, sizeof (buf)); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	  if (bufn <= 0)  | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	  bufp = buf; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       c = *bufp++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       if (c == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2016-11-14 14:27:06 -05:00
										 |  |  | 	  if (nullbyte == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      internal_warning ("%s", _("command substitution: ignored null byte in input")); | 
					
						
							|  |  |  | 	      nullbyte = 1; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       /* Add the character to ISTRING, possibly after resizing it. */ | 
					
						
							|  |  |  |       RESIZE_MALLOCED_BUFFER (istring, istring_index, 2, istring_size, DEFAULT_ARRAY_SIZE); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |       /* This is essentially quote_string inline */ | 
					
						
							|  |  |  |       if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) /* || c == CTLESC || c == CTLNUL */) | 
					
						
							|  |  |  | 	istring[istring_index++] = CTLESC; | 
					
						
							|  |  |  |       /* Escape CTLESC and CTLNUL in the output to protect those characters
 | 
					
						
							|  |  |  | 	 from the rest of the word expansions (word splitting and globbing.) | 
					
						
							|  |  |  | 	 This is essentially quote_escapes inline. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       else if (skip_ctlesc == 0 && c == CTLESC) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tflag |= W_HASCTLESC; | 
					
						
							|  |  |  | 	  istring[istring_index++] = CTLESC; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else if ((skip_ctlnul == 0 && c == CTLNUL) || (c == ' ' && (ifs_value && *ifs_value == 0))) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	istring[istring_index++] = CTLESC; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       istring[istring_index++] = c; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | #if defined (__CYGWIN__)
 | 
					
						
							|  |  |  |       if (c == '\n' && istring_index > 1 && istring[istring_index - 2] == '\r') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  istring_index--; | 
					
						
							|  |  |  | 	  istring[istring_index - 1] = '\n'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (istring) | 
					
						
							|  |  |  |     istring[istring_index] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If we read no output, just return now and save ourselves some
 | 
					
						
							|  |  |  |      trouble. */ | 
					
						
							|  |  |  |   if (istring_index == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       FREE (istring); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (rflag) | 
					
						
							|  |  |  | 	*rflag = tflag; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       return (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Strip trailing newlines from the output of the command. */ | 
					
						
							|  |  |  |   if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       while (istring_index > 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (istring[istring_index - 1] == '\n') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      --istring_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      /* If the newline was quoted, remove the quoting char. */ | 
					
						
							|  |  |  | 	      if (istring[istring_index - 1] == CTLESC) | 
					
						
							|  |  |  | 		--istring_index; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       istring[istring_index] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     strip_trailing (istring, istring_index - 1, 1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (rflag) | 
					
						
							|  |  |  |     *rflag = tflag; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   return istring; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* Perform command substitution on STRING.  This returns a WORD_DESC * with the
 | 
					
						
							|  |  |  |    contained string possibly quoted. */ | 
					
						
							|  |  |  | WORD_DESC * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | command_substitute (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   pid_t pid, old_pid, old_pipeline_pgrp, old_async_pid; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   char *istring, *s; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int result, fildes[2], function_value, pflags, rc, tflag; | 
					
						
							|  |  |  |   WORD_DESC *ret; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   istring = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Don't fork () if there is no need to.  In the case of no command to
 | 
					
						
							|  |  |  |      run, just return NULL. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 1
 | 
					
						
							|  |  |  |   for (s = string; s && *s && (shellblank (*s) || *s == '\n'); s++) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   if (s == 0 || *s == 0) | 
					
						
							|  |  |  |     return ((WORD_DESC *)NULL); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (!string || !*string || (string[0] == '\n' && !string[1])) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     return ((WORD_DESC *)NULL); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (wordexp_only && read_but_dont_execute) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       last_command_exit_value = EX_WEXPCOMSUB; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       jump_to_top_level (EXITPROG); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   /* We're making the assumption here that the command substitution will
 | 
					
						
							|  |  |  |      eventually run a command from the file system.  Since we'll run | 
					
						
							|  |  |  |      maybe_make_export_env in this subshell before executing that command, | 
					
						
							|  |  |  |      the parent shell and any other shells it starts will have to remake | 
					
						
							|  |  |  |      the environment.  If we make it before we fork, other shells won't | 
					
						
							|  |  |  |      have to.  Don't bother if we have any temporary variable assignments, | 
					
						
							|  |  |  |      though, because the export environment will be remade after this | 
					
						
							|  |  |  |      command completes anyway, but do it if all the words to be expanded | 
					
						
							|  |  |  |      are variable assignments. */ | 
					
						
							|  |  |  |   if (subst_assign_varlist == 0 || garglist == 0) | 
					
						
							|  |  |  |     maybe_make_export_env ();	/* XXX */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Flags to pass to parse_and_execute() */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   pflags = (interactive && sourcelevel == 0) ? SEVAL_RESETLINE : 0; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Pipe the output of executing STRING into the current shell. */ | 
					
						
							|  |  |  |   if (pipe (fildes) < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       sys_error ("%s", _("cannot make pipe for command substitution")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       goto error_exit; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   old_pid = last_made_pid; | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   old_pipeline_pgrp = pipeline_pgrp; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* Don't reset the pipeline pgrp if we're already a subshell in a pipeline. */ | 
					
						
							|  |  |  |   if ((subshell_environment & SUBSHELL_PIPE) == 0) | 
					
						
							|  |  |  |     pipeline_pgrp = shell_pgrp; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   cleanup_the_pipeline (); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   old_async_pid = last_asynchronous_pid; | 
					
						
							|  |  |  |   pid = make_child ((char *)NULL, subshell_environment&SUBSHELL_ASYNC); | 
					
						
							|  |  |  |   last_asynchronous_pid = old_async_pid; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (pid == 0) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* Reset the signal handlers in the child, but don't free the
 | 
					
						
							|  |  |  | 	 trap strings.  Set a flag noting that we have to free the | 
					
						
							|  |  |  | 	 trap strings if we run trap to change a signal disposition. */ | 
					
						
							|  |  |  |       reset_signal_handlers (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (ISINTERRUPT) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  kill (getpid (), SIGINT); | 
					
						
							|  |  |  | 	  CLRINTERRUPT;		/* if we're ignoring SIGINT somehow */ | 
					
						
							|  |  |  | 	}	 | 
					
						
							|  |  |  |       QUIT;	/* catch any interrupts we got post-fork */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       subshell_environment |= SUBSHELL_RESETTRAP; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   /* XXX DO THIS ONLY IN PARENT ? XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   set_sigchld_handler (); | 
					
						
							|  |  |  |   stop_making_children (); | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   if (pid != 0) | 
					
						
							|  |  |  |     pipeline_pgrp = old_pipeline_pgrp; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |   stop_making_children (); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (pid < 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       sys_error (_("cannot make child for command substitution")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     error_exit: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       last_made_pid = old_pid; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       FREE (istring); | 
					
						
							|  |  |  |       close (fildes[0]); | 
					
						
							|  |  |  |       close (fildes[1]); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       return ((WORD_DESC *)NULL); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (pid == 0) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* The currently executing shell is not interactive. */ | 
					
						
							|  |  |  |       interactive = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       set_sigint_handler ();	/* XXX */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       free_pushed_string_input (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* Discard  buffered stdio output before replacing the underlying file
 | 
					
						
							|  |  |  | 	 descriptor. */ | 
					
						
							|  |  |  |       fpurge (stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       if (dup2 (fildes[1], 1) < 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  sys_error ("%s", _("command_substitute: cannot duplicate pipe as fd 1")); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  exit (EXECUTION_FAILURE); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If standard output is closed in the parent shell
 | 
					
						
							|  |  |  | 	 (such as after `exec >&-'), file descriptor 1 will be | 
					
						
							|  |  |  | 	 the lowest available file descriptor, and end up in | 
					
						
							|  |  |  | 	 fildes[0].  This can happen for stdin and stderr as well, | 
					
						
							|  |  |  | 	 but stdout is more important -- it will cause no output | 
					
						
							|  |  |  | 	 to be generated from this command. */ | 
					
						
							|  |  |  |       if ((fildes[1] != fileno (stdin)) && | 
					
						
							|  |  |  | 	  (fildes[1] != fileno (stdout)) && | 
					
						
							|  |  |  | 	  (fildes[1] != fileno (stderr))) | 
					
						
							|  |  |  | 	close (fildes[1]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if ((fildes[0] != fileno (stdin)) && | 
					
						
							|  |  |  | 	  (fildes[0] != fileno (stdout)) && | 
					
						
							|  |  |  | 	  (fildes[0] != fileno (stderr))) | 
					
						
							|  |  |  | 	close (fildes[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #ifdef __CYGWIN__
 | 
					
						
							|  |  |  |       /* Let stdio know the fd may have changed from text to binary mode, and
 | 
					
						
							|  |  |  | 	 make sure to preserve stdout line buffering. */ | 
					
						
							|  |  |  |       freopen (NULL, "w", stdout); | 
					
						
							|  |  |  |       sh_setlinebuf (stdout); | 
					
						
							|  |  |  | #endif /* __CYGWIN__ */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* This is a subshell environment. */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       subshell_environment |= SUBSHELL_COMSUB; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* Many shells do not appear to inherit the -v option for command
 | 
					
						
							|  |  |  | 	 substitutions. */ | 
					
						
							|  |  |  |       change_flag ('v', FLAG_OFF); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* When inherit_errexit option is not enabled, command substitution does
 | 
					
						
							|  |  |  | 	 not inherit the -e flag.  It is enabled when Posix mode is enabled */ | 
					
						
							|  |  |  |       if (inherit_errexit == 0) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |         { | 
					
						
							|  |  |  |           builtin_ignoring_errexit = 0; | 
					
						
							|  |  |  | 	  change_flag ('e', FLAG_OFF); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       set_shellopts (); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If we are expanding a redirection, we can dispose of any temporary
 | 
					
						
							|  |  |  | 	 environment we received, since redirections are not supposed to have | 
					
						
							|  |  |  | 	 access to the temporary environment.  We will have to see whether this | 
					
						
							|  |  |  | 	 affects temporary environments supplied to `eval', but the temporary | 
					
						
							|  |  |  | 	 environment gets copied to builtin_env at some point. */ | 
					
						
							|  |  |  |       if (expanding_redir) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  flush_temporary_env (); | 
					
						
							|  |  |  | 	  expanding_redir = 0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       remove_quoted_escapes (string); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       startup_state = 2;	/* see if we can avoid a fork */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       /* Give command substitution a place to jump back to on failure,
 | 
					
						
							|  |  |  | 	 so we don't go back up to main (). */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       result = setjmp_nosigs (top_level); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       /* If we're running a command substitution inside a shell function,
 | 
					
						
							|  |  |  | 	 trap `return' so we don't return from the function in the subshell | 
					
						
							|  |  |  | 	 and go off to never-never land. */ | 
					
						
							|  |  |  |       if (result == 0 && return_catch_flag) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	function_value = setjmp_nosigs (return_catch); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	function_value = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (result == ERREXIT) | 
					
						
							|  |  |  | 	rc = last_command_exit_value; | 
					
						
							|  |  |  |       else if (result == EXITPROG) | 
					
						
							|  |  |  | 	rc = last_command_exit_value; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else if (result) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	rc = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       else if (function_value) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	rc = return_catch_value; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  subshell_level++; | 
					
						
							|  |  |  | 	  rc = parse_and_execute (string, "command substitution", pflags|SEVAL_NOHIST); | 
					
						
							|  |  |  | 	  subshell_level--; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       last_command_exit_value = rc; | 
					
						
							|  |  |  |       rc = run_exit_trap (); | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  |       unlink_fifo_list (); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       exit (rc); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  | #if defined (JOB_CONTROL) && defined (PGRP_PIPE)
 | 
					
						
							|  |  |  |       close_pgrp_pipe (); | 
					
						
							|  |  |  | #endif /* JOB_CONTROL && PGRP_PIPE */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       close (fildes[1]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       tflag = 0; | 
					
						
							|  |  |  |       istring = read_comsub (fildes[0], quoted, &tflag); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       close (fildes[0]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |       current_command_subst_pid = pid; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       last_command_exit_value = wait_for (pid); | 
					
						
							|  |  |  |       last_command_subst_pid = pid; | 
					
						
							|  |  |  |       last_made_pid = old_pid; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (JOB_CONTROL)
 | 
					
						
							|  |  |  |       /* If last_command_exit_value > 128, then the substituted command
 | 
					
						
							|  |  |  | 	 was terminated by a signal.  If that signal was SIGINT, then send | 
					
						
							|  |  |  | 	 SIGINT to ourselves.  This will break out of loops, for instance. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (last_command_exit_value == (128 + SIGINT) && last_command_exit_signal == SIGINT) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	kill (getpid (), SIGINT); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* wait_for gives the terminal back to shell_pgrp.  If some other
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 process group should have it, give it away to that group here. | 
					
						
							|  |  |  | 	 pipeline_pgrp is non-zero only while we are constructing a | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	 pipeline, so what we are concerned about is whether or not that | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 pipeline was started in the background.  A pipeline started in | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	 the background should never get the tty back here.  We duplicate | 
					
						
							|  |  |  | 	 the conditions that wait_for tests to make sure we only give | 
					
						
							|  |  |  | 	 the terminal back to pipeline_pgrp under the conditions that wait_for | 
					
						
							|  |  |  | 	 gave it to shell_pgrp.  If wait_for doesn't mess with the terminal | 
					
						
							|  |  |  | 	 pgrp, we should not either. */ | 
					
						
							|  |  |  |       if (interactive && pipeline_pgrp != (pid_t)0 && running_in_background == 0 && | 
					
						
							|  |  |  | 	   (subshell_environment & (SUBSHELL_ASYNC|SUBSHELL_PIPE)) == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	give_terminal_to (pipeline_pgrp, 0); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | #endif /* JOB_CONTROL */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = istring; | 
					
						
							|  |  |  |       ret->flags = tflag; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /********************************************************
 | 
					
						
							|  |  |  |  *							* | 
					
						
							|  |  |  |  *	Utility functions for parameter expansion	* | 
					
						
							|  |  |  |  *							* | 
					
						
							|  |  |  |  ********************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | static arrayind_t | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | array_length_reference (s) | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int len; | 
					
						
							|  |  |  |   arrayind_t ind; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   char *akey; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   char *t, c; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   ARRAY *array; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   HASH_TABLE *h; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *var; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var = array_variable_part (s, &t, &len); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* If unbound variables should generate an error, report one and return
 | 
					
						
							|  |  |  |      failure. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if ((var == 0 || invisible_p (var) || (assoc_p (var) == 0 && array_p (var) == 0)) && unbound_vars_is_error) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       c = *--t; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       *t = '\0'; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       err_unboundvar (s); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       *t = c; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       return (-1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   else if (var == 0 || invisible_p (var)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* We support a couple of expansions for variables that are not arrays.
 | 
					
						
							|  |  |  |      We'll return the length of the value for v[0], and 1 for v[@] or | 
					
						
							|  |  |  |      v[*].  Return 0 for everything else. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   array = array_p (var) ? array_cell (var) : (ARRAY *)NULL; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   h = assoc_p (var) ? assoc_cell (var) : (HASH_TABLE *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (ALL_ELEMENT_SUB (t[0]) && t[1] == ']') | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (assoc_p (var)) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	return (h ? assoc_num_elements (h) : 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       else if (array_p (var)) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	return (array ? array_num_elements (array) : 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	return (var_isset (var) ? 1 : 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (assoc_p (var)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t[len - 1] = '\0'; | 
					
						
							|  |  |  |       akey = expand_assignment_string_to_string (t, 0);	/* [ */ | 
					
						
							|  |  |  |       t[len - 1] = ']'; | 
					
						
							|  |  |  |       if (akey == 0 || *akey == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  err_badarraysub (t); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  FREE (akey); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  return (-1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       t = assoc_reference (assoc_cell (var), akey); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       free (akey); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       ind = array_expand_index (var, t, len); | 
					
						
							|  |  |  |       /* negative subscripts to indexed arrays count back from end */ | 
					
						
							|  |  |  |       if (var && array_p (var) && ind < 0) | 
					
						
							|  |  |  | 	ind = array_max_index (array_cell (var)) + 1 + ind; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (ind < 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  err_badarraysub (t); | 
					
						
							|  |  |  | 	  return (-1); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       if (array_p (var)) | 
					
						
							|  |  |  | 	t = array_reference (array, ind); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	t = (ind == 0) ? value_cell (var) : (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   len = MB_STRLEN (t); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return (len); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | valid_brace_expansion_word (name, var_is_special) | 
					
						
							|  |  |  |      char *name; | 
					
						
							|  |  |  |      int var_is_special; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   if (DIGIT (*name) && all_digits (name)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return 1; | 
					
						
							|  |  |  |   else if (var_is_special) | 
					
						
							|  |  |  |     return 1; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (valid_array_reference (name, 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     return 1; | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   else if (legal_identifier (name)) | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | static int | 
					
						
							|  |  |  | chk_atstar (name, quoted, quoted_dollar_atp, contains_dollar_at) | 
					
						
							|  |  |  |      char *name; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  |      int *quoted_dollar_atp, *contains_dollar_at; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *temp1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (name == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (quoted_dollar_atp) | 
					
						
							|  |  |  | 	*quoted_dollar_atp = 0; | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 0; | 
					
						
							|  |  |  |       return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* check for $@ and $* */ | 
					
						
							|  |  |  |   if (name[0] == '@' && name[1] == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 	*quoted_dollar_atp = 1; | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 1; | 
					
						
							|  |  |  |       return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (name[0] == '*' && name[1] == '\0' && quoted == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 1; | 
					
						
							|  |  |  |       return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Now check for ${array[@]} and ${array[*]} */ | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (valid_array_reference (name, 0)) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       temp1 = mbschr (name, '['); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (temp1 && temp1[1] == '@' && temp1[2] == ']') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 	    *quoted_dollar_atp = 1; | 
					
						
							|  |  |  | 	  if (contains_dollar_at) | 
					
						
							|  |  |  | 	    *contains_dollar_at = 1; | 
					
						
							|  |  |  | 	  return 1; | 
					
						
							|  |  |  | 	}	/* [ */ | 
					
						
							|  |  |  |       /* ${array[*]}, when unquoted, should be treated like ${array[@]},
 | 
					
						
							|  |  |  | 	 which should result in separate words even when IFS is unset. */ | 
					
						
							|  |  |  |       if (temp1 && temp1[1] == '*' && temp1[2] == ']' && quoted == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (contains_dollar_at) | 
					
						
							|  |  |  | 	    *contains_dollar_at = 1; | 
					
						
							|  |  |  | 	  return 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Parameter expand NAME, and return a new string which is the expansion,
 | 
					
						
							|  |  |  |    or NULL if there was no expansion. | 
					
						
							|  |  |  |    VAR_IS_SPECIAL is non-zero if NAME is one of the special variables in | 
					
						
							|  |  |  |    the shell, e.g., "@", "$", "*", etc.  QUOTED, if non-zero, means that | 
					
						
							|  |  |  |    NAME was found inside of a double-quoted expression. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC * | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | parameter_brace_expand_word (name, var_is_special, quoted, pflags, indp) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *name; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  |      int var_is_special, quoted, pflags; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      arrayind_t *indp; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC *ret; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *temp, *tt; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   intmax_t arg_index; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *var; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   int atype, rflags; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   arrayind_t ind; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   ret = 0; | 
					
						
							|  |  |  |   temp = 0; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   rflags = 0; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   if (indp) | 
					
						
							|  |  |  |     *indp = INTMAX_MIN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Handle multiple digit arguments, as in ${11}. */   | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   if (legal_number (name, &arg_index)) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       tt = get_dollar_var_value (arg_index); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (tt) | 
					
						
							|  |  |  |  	temp = (*tt && (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT))) | 
					
						
							|  |  |  |  		  ? quote_string (tt) | 
					
						
							|  |  |  |  		  : quote_escapes (tt); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       FREE (tt); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   else if (var_is_special)      /* ${@} */ | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       int sindex; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       tt = (char *)xmalloc (2 + strlen (name)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       tt[sindex = 0] = '$'; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       strcpy (tt + 1, name); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       ret = param_expand (tt, &sindex, quoted, (int *)NULL, (int *)NULL, | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 			  (int *)NULL, (int *)NULL, pflags); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       free (tt); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (valid_array_reference (name, 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | expand_arrayref: | 
					
						
							|  |  |  |       if (pflags & PF_ASSIGNRHS) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  var = array_variable_part (name, &tt, (int *)0); | 
					
						
							|  |  |  | 	  if (ALL_ELEMENT_SUB (tt[0]) && tt[1] == ']') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Only treat as double quoted if array variable */ | 
					
						
							|  |  |  | 	      if (var && (array_p (var) || assoc_p (var))) | 
					
						
							|  |  |  | 	        /* XXX - bash-4.4/bash-5.0 pass AV_ASSIGNRHS */ | 
					
						
							|  |  |  | 		temp = array_value (name, quoted|Q_DOUBLE_QUOTES, AV_ASSIGNRHS, &atype, &ind); | 
					
						
							|  |  |  | 	      else		 | 
					
						
							|  |  |  | 		temp = array_value (name, quoted, 0, &atype, &ind); | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = array_value (name, quoted, 0, &atype, &ind); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       else | 
					
						
							|  |  |  | 	temp = array_value (name, quoted, 0, &atype, &ind); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (atype == 0 && temp) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = (*temp && (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT))) | 
					
						
							|  |  |  | 		    ? quote_string (temp) | 
					
						
							|  |  |  | 		    : quote_escapes (temp); | 
					
						
							|  |  |  | 	  rflags |= W_ARRAYIND; | 
					
						
							|  |  |  | 	  if (indp) | 
					
						
							|  |  |  | 	    *indp = ind; | 
					
						
							|  |  |  | 	} 		   | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |       else if (atype == 1 && temp && QUOTED_NULL (temp) && (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT))) | 
					
						
							|  |  |  | 	rflags |= W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   else if (var = find_variable (name)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (var_isset (var) && invisible_p (var) == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if (assoc_p (var)) | 
					
						
							|  |  |  | 	    temp = assoc_reference (assoc_cell (var), "0"); | 
					
						
							|  |  |  | 	  else if (array_p (var)) | 
					
						
							|  |  |  | 	    temp = array_reference (array_cell (var), 0); | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = value_cell (var); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  | 	  temp = value_cell (var); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (temp) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    temp = (*temp && (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT))) | 
					
						
							|  |  |  | 		      ? quote_string (temp) | 
					
						
							|  |  |  | 		      : quote_escapes (temp); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	temp = (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (var = find_variable_last_nameref (name, 0)) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |       temp = nameref_cell (var); | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |       /* Handle expanding nameref whose value is x[n] */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (temp && *temp && valid_array_reference (temp, 0)) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  name = temp; | 
					
						
							|  |  |  | 	  goto expand_arrayref; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       /* y=2 ; typeset -n x=y; echo ${x} is not the same as echo ${2} in ksh */ | 
					
						
							|  |  |  |       if (temp && *temp && legal_identifier (temp) == 0) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  report_error (_("%s: invalid variable name for name reference"), temp); | 
					
						
							|  |  |  | 	  temp = &expand_param_error; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	temp = (char *)NULL; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (ret == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |       ret->flags |= rflags; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | static char * | 
					
						
							|  |  |  | parameter_brace_find_indir (name, var_is_special, quoted, find_nameref) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *name; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |      int var_is_special, quoted, find_nameref; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   char *temp, *t; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC *w; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (find_nameref && var_is_special == 0 && (v = find_variable_last_nameref (name, 0)) && | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       nameref_p (v) && (t = nameref_cell (v)) && *t) | 
					
						
							|  |  |  |     return (savestring (t)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   /* If var_is_special == 0, and name is not an array reference, this does
 | 
					
						
							|  |  |  |      more expansion than necessary.  It should really look up the variable's | 
					
						
							|  |  |  |      value and not try to expand it. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   w = parameter_brace_expand_word (name, var_is_special, quoted, PF_IGNUNBOUND, 0); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   t = w->word; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Have to dequote here if necessary */ | 
					
						
							|  |  |  |   if (t) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp = (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  | 		? dequote_string (t) | 
					
						
							|  |  |  | 		: dequote_escapes (t); | 
					
						
							|  |  |  |       free (t); | 
					
						
							|  |  |  |       t = temp; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   dispose_word_desc (w); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   return t; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  | /* Expand an indirect reference to a variable: ${!NAME} expands to the
 | 
					
						
							|  |  |  |    value of the variable whose name is the value of NAME. */ | 
					
						
							|  |  |  | static WORD_DESC * | 
					
						
							|  |  |  | parameter_brace_expand_indir (name, var_is_special, quoted, quoted_dollar_atp, contains_dollar_at) | 
					
						
							|  |  |  |      char *name; | 
					
						
							|  |  |  |      int var_is_special, quoted; | 
					
						
							|  |  |  |      int *quoted_dollar_atp, *contains_dollar_at; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *temp, *t; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* See if it's a nameref first, behave in ksh93-compatible fashion.
 | 
					
						
							|  |  |  |      There is at least one incompatibility: given ${!foo[0]} where foo=bar, | 
					
						
							|  |  |  |      bash performs an indirect lookup on foo[0] and expands the result; | 
					
						
							|  |  |  |      ksh93 expands bar[0].  We could do that here -- there are enough usable | 
					
						
							|  |  |  |      primitives to do that -- but do not at this point. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (var_is_special == 0 && (v = find_variable_last_nameref (name, 0))) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (nameref_p (v) && (t = nameref_cell (v)) && *t) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  w = alloc_word_desc (); | 
					
						
							|  |  |  | 	  w->word = savestring (t); | 
					
						
							|  |  |  | 	  w->flags = 0; | 
					
						
							|  |  |  | 	  return w; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   t = parameter_brace_find_indir (name, var_is_special, quoted, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   chk_atstar (t, quoted, quoted_dollar_atp, contains_dollar_at); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (t == 0) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     return (WORD_DESC *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (valid_brace_expansion_word (t, SPECIAL_VAR (t, 0)) == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       report_error (_("%s: bad substitution"), t); | 
					
						
							|  |  |  |       free (t); | 
					
						
							|  |  |  |       w = alloc_word_desc (); | 
					
						
							|  |  |  |       w->word = &expand_param_error; | 
					
						
							|  |  |  |       w->flags = 0; | 
					
						
							|  |  |  |       return (w); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   w = parameter_brace_expand_word (t, SPECIAL_VAR(t, 0), quoted, 0, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   free (t); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return w; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Expand the right side of a parameter expansion of the form ${NAMEcVALUE},
 | 
					
						
							|  |  |  |    depending on the value of C, the separating character.  C can be one of | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |    "-", "+", or "=".  QUOTED is true if the entire brace expression occurs | 
					
						
							|  |  |  |    between double quotes. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | parameter_brace_expand_rhs (name, value, c, quoted, pflags, qdollaratp, hasdollarat) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      char *name, *value; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int c, quoted, pflags, *qdollaratp, *hasdollarat; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC *w; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   WORD_LIST *l; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   char *t, *t1, *temp, *vname; | 
					
						
							|  |  |  |   int l_hasdollat, sindex; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /*itrace("parameter_brace_expand_rhs: %s:%s pflags = %d", name, value, pflags);*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* If the entire expression is between double quotes, we want to treat
 | 
					
						
							|  |  |  |      the value as a double-quoted string, with the exception that we strip | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      embedded unescaped double quotes (for sh backwards compatibility). */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && *value) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       sindex = 0; | 
					
						
							|  |  |  |       temp = string_extract_double_quoted (value, &sindex, SX_STRIPDQ); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     temp = value; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   w = alloc_word_desc (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   l_hasdollat = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* XXX was 0 not quoted */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   l = *temp ? expand_string_for_rhs (temp, quoted, &l_hasdollat, (int *)NULL) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    : (WORD_LIST *)0; | 
					
						
							|  |  |  |   if (hasdollarat) | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     *hasdollarat = l_hasdollat || (l && l->next); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   if (temp != value) | 
					
						
							|  |  |  |     free (temp); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (l) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* If l->next is not null, we know that TEMP contained "$@", since that
 | 
					
						
							|  |  |  | 	 is the only expansion that creates more than one word. */ | 
					
						
							|  |  |  |       if (qdollaratp && ((l_hasdollat && quoted) || l->next)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | /*itrace("parameter_brace_expand_rhs: %s:%s: l != NULL, set *qdollaratp", name, value);*/ | 
					
						
							|  |  |  | 	  *qdollaratp = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* The expansion of TEMP returned something.  We need to treat things
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  slightly differently if L_HASDOLLAT is non-zero.  If we have "$@", | 
					
						
							|  |  |  | 	  the individual words have already been quoted.  We need to turn them | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  into a string with the words separated by the first character of | 
					
						
							|  |  |  | 	  $IFS without any additional quoting, so string_list_dollar_at won't | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  do the right thing.  If IFS is null, we want "$@" to split into | 
					
						
							|  |  |  | 	  separate arguments, not be concatenated, so we use string_list_internal | 
					
						
							|  |  |  | 	  and mark the word to be split on spaces later.  We use | 
					
						
							|  |  |  | 	  string_list_dollar_star for "$@" otherwise. */ | 
					
						
							|  |  |  |       if (l->next && ifs_is_null) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = string_list_internal (l, " "); | 
					
						
							|  |  |  | 	  w->flags |= W_SPLITSPACE; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	temp = (l_hasdollat || l->next) ? string_list_dollar_star (l) : string_list (l); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-07 16:23:39 -04:00
										 |  |  |       /* If we have a quoted null result (QUOTED_NULL(temp)) and the word is
 | 
					
						
							|  |  |  | 	 a quoted null (l->next == 0 && QUOTED_NULL(l->word->word)), the | 
					
						
							|  |  |  | 	 flags indicate it (l->word->flags & W_HASQUOTEDNULL), and the | 
					
						
							|  |  |  | 	 expansion is quoted (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 	 (which is more paranoia than anything else), we need to return the | 
					
						
							|  |  |  | 	 quoted null string and set the flags to indicate it. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (l->next == 0 && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && QUOTED_NULL (temp) && QUOTED_NULL (l->word->word) && (l->word->flags & W_HASQUOTEDNULL)) | 
					
						
							| 
									
										
										
										
											2012-05-07 16:23:39 -04:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  w->flags |= W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /*itrace("parameter_brace_expand_rhs (%s:%s): returning quoted null, turning off qdollaratp", name, value);*/ | 
					
						
							|  |  |  | 	  /* If we return a quoted null with L_HASDOLLARAT, we either have a
 | 
					
						
							|  |  |  | 	     construct like "${@-$@}" or "${@-${@-$@}}" with no positional | 
					
						
							|  |  |  | 	     parameters or a quoted expansion of "$@" with $1 == ''.  In either | 
					
						
							|  |  |  | 	     case, we don't want to enable special handling of $@. */ | 
					
						
							|  |  |  | 	  if (qdollaratp && l_hasdollat) | 
					
						
							|  |  |  | 	    *qdollaratp = 0; | 
					
						
							| 
									
										
										
										
											2012-05-07 16:23:39 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       dispose_words (l); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && l_hasdollat) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* Posix interp 221 changed the rules on this.  The idea is that
 | 
					
						
							|  |  |  | 	 something like "$xxx$@" should expand the same as "${foo-$xxx$@}" | 
					
						
							|  |  |  | 	 when foo and xxx are unset.  The problem is that it's not in any | 
					
						
							|  |  |  | 	 way backwards compatible and few other shells do it.  We're eventually | 
					
						
							|  |  |  | 	 going to try and split the difference (heh) a little bit here. */ | 
					
						
							|  |  |  |       /* l_hasdollat == 1 means we saw a quoted dollar at.  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* The brace expansion occurred between double quotes and there was
 | 
					
						
							|  |  |  | 	 a $@ in TEMP.  It does not matter if the $@ is quoted, as long as | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	 it does not expand to anything.  In this case, we want to return | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	 a quoted empty string.  Posix interp 888 */ | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       temp = make_quoted_char ('\0'); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       w->flags |= W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /*itrace("parameter_brace_expand_rhs (%s:%s): returning quoted null", name, value);*/ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     temp = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (c == '-' || c == '+') | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       w->word = temp; | 
					
						
							|  |  |  |       return w; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* c == '=' */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   t = temp ? savestring (temp) : savestring (""); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   t1 = dequote_string (t); | 
					
						
							|  |  |  |   free (t); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* bash-4.4/5.0 */ | 
					
						
							|  |  |  |   vname = name; | 
					
						
							|  |  |  |   if (*name == '!' && | 
					
						
							|  |  |  |       (legal_variable_starter ((unsigned char)name[1]) || DIGIT (name[1]) || VALID_INDIR_PARAM (name[1]))) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       vname = parameter_brace_find_indir (name + 1, SPECIAL_VAR (name, 1), quoted, 1); | 
					
						
							|  |  |  |       if (vname == 0 || *vname == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  report_error (_("%s: invalid indirect expansion"), name); | 
					
						
							|  |  |  | 	  free (vname); | 
					
						
							|  |  |  | 	  dispose_word (w); | 
					
						
							|  |  |  | 	  return &expand_wdesc_error; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       if (legal_identifier (vname) == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  report_error (_("%s: invalid variable name"), vname); | 
					
						
							|  |  |  | 	  free (vname); | 
					
						
							|  |  |  | 	  dispose_word (w); | 
					
						
							|  |  |  | 	  return &expand_wdesc_error; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (valid_array_reference (vname, 0)) | 
					
						
							|  |  |  |     assign_array_element (vname, t1, 0); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   else | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   bind_variable (vname, t1, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   stupidly_hack_special_variables (vname); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (vname != name) | 
					
						
							|  |  |  |     free (vname); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   /* From Posix group discussion Feb-March 2010.  Issue 7 0000221 */ | 
					
						
							|  |  |  |   free (temp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   w->word = t1; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   return w; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Deal with the right hand side of a ${name:?value} expansion in the case
 | 
					
						
							|  |  |  |    that NAME is null or not set.  If VALUE is non-null it is expanded and | 
					
						
							|  |  |  |    used as the error message to print, otherwise a standard message is | 
					
						
							|  |  |  |    printed. */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | parameter_brace_expand_error (name, value) | 
					
						
							|  |  |  |      char *name, *value; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   WORD_LIST *l; | 
					
						
							|  |  |  |   char *temp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   last_command_exit_value = EXECUTION_FAILURE;	/* ensure it's non-zero */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (value && *value) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       l = expand_string (value, 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       temp =  string_list (l); | 
					
						
							|  |  |  |       report_error ("%s: %s", name, temp ? temp : "");	/* XXX was value not "" */ | 
					
						
							|  |  |  |       FREE (temp); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       dispose_words (l); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     report_error (_("%s: parameter null or not set"), name); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Free the data we have allocated during this expansion, since we
 | 
					
						
							|  |  |  |      are about to longjmp out. */ | 
					
						
							|  |  |  |   free (name); | 
					
						
							|  |  |  |   FREE (value); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return 1 if NAME is something for which parameter_brace_expand_length is
 | 
					
						
							|  |  |  |    OK to do. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | valid_length_expression (name) | 
					
						
							|  |  |  |      char *name; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   return (name[1] == '\0' ||					/* ${#} */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	  ((sh_syntaxtab[(unsigned char) name[1]] & CSPECVAR) && name[2] == '\0') ||  /* special param */ | 
					
						
							|  |  |  | 	  (DIGIT (name[1]) && all_digits (name + 1)) ||	/* ${#11} */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  valid_array_reference (name + 1, 0) ||		/* ${#a[7]} */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  legal_identifier (name + 1));				/* ${#PS1} */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Handle the parameter brace expansion that requires us to return the
 | 
					
						
							|  |  |  |    length of a parameter. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | static intmax_t | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | parameter_brace_expand_length (name) | 
					
						
							|  |  |  |      char *name; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *t, *newname; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   intmax_t number, arg_index; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |   SHELL_VAR *var; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (name[1] == '\0')			/* ${#} */ | 
					
						
							|  |  |  |     number = number_of_args (); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (DOLLAR_AT_STAR (name[1]) && name[2] == '\0')	/* ${#@}, ${#*} */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     number = number_of_args (); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   else if ((sh_syntaxtab[(unsigned char) name[1]] & CSPECVAR) && name[2] == '\0') | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* Take the lengths of some of the shell's special parameters. */ | 
					
						
							|  |  |  |       switch (name[1]) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case '-': | 
					
						
							|  |  |  | 	  t = which_set_flags (); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case '?': | 
					
						
							|  |  |  | 	  t = itos (last_command_exit_value); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case '$': | 
					
						
							|  |  |  | 	  t = itos (dollar_dollar_pid); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	case '!': | 
					
						
							|  |  |  | 	  if (last_asynchronous_pid == NO_PID) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	    t = (char *)NULL;	/* XXX - error if set -u set? */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	    t = itos (last_asynchronous_pid); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 	case '#': | 
					
						
							|  |  |  | 	  t = itos (number_of_args ()); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       number = STRLEN (t); | 
					
						
							|  |  |  |       FREE (t); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (valid_array_reference (name + 1, 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     number = array_length_reference (name + 1); | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       number = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       if (legal_number (name + 1, &arg_index))		/* ${#1} */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	  t = get_dollar_var_value (arg_index); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (t == 0 && unbound_vars_is_error) | 
					
						
							|  |  |  | 	    return INTMAX_MIN; | 
					
						
							| 
									
										
										
										
											2004-11-09 21:37:25 +00:00
										 |  |  | 	  number = MB_STRLEN (t); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  FREE (t); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       else if ((var = find_variable (name + 1)) && (invisible_p (var) == 0) && (array_p (var) || assoc_p (var))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if (assoc_p (var)) | 
					
						
							|  |  |  | 	    t = assoc_reference (assoc_cell (var), "0"); | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    t = array_reference (array_cell (var), 0); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (t == 0 && unbound_vars_is_error) | 
					
						
							|  |  |  | 	    return INTMAX_MIN; | 
					
						
							| 
									
										
										
										
											2004-11-09 21:37:25 +00:00
										 |  |  | 	  number = MB_STRLEN (t); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |       else				/* ${#PS1} */ | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  newname = savestring (name); | 
					
						
							|  |  |  | 	  newname[0] = '$'; | 
					
						
							|  |  |  | 	  list = expand_string (newname, Q_DOUBLE_QUOTES); | 
					
						
							|  |  |  | 	  t = list ? string_list (list) : (char *)NULL; | 
					
						
							|  |  |  | 	  free (newname); | 
					
						
							|  |  |  | 	  if (list) | 
					
						
							|  |  |  | 	    dispose_words (list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  number = t ? MB_STRLEN (t) : 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  FREE (t); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (number); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | /* Skip characters in SUBSTR until DELIM.  SUBSTR is an arithmetic expression,
 | 
					
						
							|  |  |  |    so we do some ad-hoc parsing of an arithmetic expression to find | 
					
						
							|  |  |  |    the first DELIM, instead of using strchr(3).  Two rules: | 
					
						
							|  |  |  | 	1.  If the substring contains a `(', read until closing `)'. | 
					
						
							|  |  |  | 	2.  If the substring contains a `?', read past one `:' for each `?'. | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |    The SD_ARITHEXP flag to skip_to_delim takes care of doing this. | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | skiparith (substr, delim) | 
					
						
							|  |  |  |      char *substr; | 
					
						
							|  |  |  |      int delim; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int i; | 
					
						
							|  |  |  |   char delims[2]; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   delims[0] = delim; | 
					
						
							|  |  |  |   delims[1] = '\0'; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   i = skip_to_delim (substr, 0, delims, SD_ARITHEXP); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   return (substr + i); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Verify and limit the start and end of the desired substring.  If
 | 
					
						
							|  |  |  |    VTYPE == 0, a regular shell variable is being used; if it is 1, | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    then the positional parameters are being used; if it is 2, then | 
					
						
							| 
									
										
										
										
											1997-09-22 20:22:27 +00:00
										 |  |  |    VALUE is really a pointer to an array variable that should be used. | 
					
						
							|  |  |  |    Return value is 1 if both values were OK, 0 if there was a problem | 
					
						
							|  |  |  |    with an invalid expression, or -1 if the values were out of range. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | verify_substring_values (v, value, substr, vtype, e1p, e2p) | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *value, *substr; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |      int vtype; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      intmax_t *e1p, *e2p; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   char *t, *temp1, *temp2; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   arrayind_t len; | 
					
						
							|  |  |  |   int expok; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |  ARRAY *a; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |  HASH_TABLE *h; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   /* duplicate behavior of strchr(3) */ | 
					
						
							|  |  |  |   t = skiparith (substr, ':'); | 
					
						
							|  |  |  |   if (*t && *t == ':') | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |     *t = '\0'; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     t = (char *)0; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   temp1 = expand_arith_string (substr, Q_DOUBLE_QUOTES); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   *e1p = evalexp (temp1, &expok); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   free (temp1); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   if (expok == 0) | 
					
						
							|  |  |  |     return (0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   len = -1;	/* paranoia */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   switch (vtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case VT_VARIABLE: | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     case VT_ARRAYMEMBER: | 
					
						
							| 
									
										
										
										
											2004-11-09 21:37:25 +00:00
										 |  |  |       len = MB_STRLEN (value); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							|  |  |  |       len = number_of_args () + 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (*e1p == 0) | 
					
						
							|  |  |  | 	len++;		/* add one arg if counting from $0 */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							| 
									
										
										
										
											2004-11-09 21:37:25 +00:00
										 |  |  |       /* For arrays, the first value deals with array indices.  Negative
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	 offsets count from one past the array's maximum index.  Associative | 
					
						
							|  |  |  | 	 arrays treat the number of elements as the maximum index. */ | 
					
						
							|  |  |  |       if (assoc_p (v)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  h = assoc_cell (v); | 
					
						
							|  |  |  | 	  len = assoc_num_elements (h) + (*e1p < 0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  a = (ARRAY *)value; | 
					
						
							|  |  |  | 	  len = array_max_index (a) + (*e1p < 0);	/* arrays index from 0 to n - 1 */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   if (len == -1)	/* paranoia */ | 
					
						
							|  |  |  |     return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (*e1p < 0)		/* negative offsets count from end */ | 
					
						
							|  |  |  |     *e1p += len; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-11-09 21:37:25 +00:00
										 |  |  |   if (*e1p > len || *e1p < 0) | 
					
						
							| 
									
										
										
										
											1997-09-22 20:22:27 +00:00
										 |  |  |     return (-1); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |   /* For arrays, the second offset deals with the number of elements. */ | 
					
						
							|  |  |  |   if (vtype == VT_ARRAYVAR) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     len = assoc_p (v) ? assoc_num_elements (h) : array_num_elements (a); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (t) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       t++; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       temp2 = savestring (t); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |       temp1 = expand_arith_string (temp2, Q_DOUBLE_QUOTES); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       free (temp2); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       t[-1] = ':'; | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       *e2p = evalexp (temp1, &expok); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       free (temp1); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       if (expok == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	return (0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #if 1
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       if ((vtype == VT_ARRAYVAR || vtype == VT_POSPARMS) && *e2p < 0) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | #else
 | 
					
						
							|  |  |  |       /* bash-4.3: allow positional parameter length < 0 to count backwards
 | 
					
						
							|  |  |  | 	 from end of positional parameters */ | 
					
						
							|  |  |  |       if (vtype == VT_ARRAYVAR && *e2p < 0) | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  internal_error (_("%s: substring expression < 0"), t); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  return (0); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |       /* In order to deal with sparse arrays, push the intelligence about how
 | 
					
						
							|  |  |  | 	 to deal with the number of elements desired down to the array- | 
					
						
							|  |  |  | 	 specific functions.  */ | 
					
						
							|  |  |  |       if (vtype != VT_ARRAYVAR) | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (*e2p < 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      *e2p += len; | 
					
						
							|  |  |  | 	      if (*e2p < 0 || *e2p < *e1p) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  internal_error (_("%s: substring expression < 0"), t); | 
					
						
							|  |  |  | 		  return (0); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    *e2p += *e1p;		/* want E2 chars starting at E1 */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (*e2p > len) | 
					
						
							|  |  |  | 	    *e2p = len; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     *e2p = len; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (1); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Return the type of variable specified by VARNAME (simple variable,
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    positional param, or array variable).  Also return the value specified | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    by VARNAME (value of a variable or a reference to an array element). | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |    QUOTED is the standard description of quoting state, using Q_* defines. | 
					
						
							|  |  |  |    FLAGS is currently a set of flags to pass to array_value.  If IND is | 
					
						
							|  |  |  |    non-null and not INTMAX_MIN, and FLAGS includes AV_USEIND, IND is | 
					
						
							|  |  |  |    passed to array_value so the array index is not computed again. | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    If this returns VT_VARIABLE, the caller assumes that CTLESC and CTLNUL | 
					
						
							|  |  |  |    characters in the value are quoted with CTLESC and takes appropriate | 
					
						
							|  |  |  |    steps.  For convenience, *VALP is set to the dequoted VALUE. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static int | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | get_var_and_type (varname, value, ind, quoted, flags, varp, valp) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *varname, *value; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      arrayind_t ind; | 
					
						
							|  |  |  |      int quoted, flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      SHELL_VAR **varp; | 
					
						
							|  |  |  |      char **valp; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   int vtype, want_indir; | 
					
						
							|  |  |  |   char *temp, *vname; | 
					
						
							|  |  |  |   WORD_DESC *wd; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   arrayind_t lind; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   want_indir = *varname == '!' && | 
					
						
							|  |  |  |     (legal_variable_starter ((unsigned char)varname[1]) || DIGIT (varname[1]) | 
					
						
							|  |  |  | 					|| VALID_INDIR_PARAM (varname[1])); | 
					
						
							|  |  |  |   if (want_indir) | 
					
						
							|  |  |  |     vname = parameter_brace_find_indir (varname+1, SPECIAL_VAR (varname, 1), quoted, 1); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     /* XXX - what if vname == 0 || *vname == 0 ? */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   else | 
					
						
							|  |  |  |     vname = varname; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (vname == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       vtype = VT_VARIABLE; | 
					
						
							|  |  |  |       *varp = (SHELL_VAR *)NULL; | 
					
						
							|  |  |  |       *valp = (char *)NULL; | 
					
						
							|  |  |  |       return (vtype); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* This sets vtype to VT_VARIABLE or VT_POSPARMS */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   vtype = STR_DOLLAR_AT_STAR (vname); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (vtype == VT_POSPARMS && vname[0] == '*') | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     vtype |= VT_STARSUB; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   *varp = (SHELL_VAR *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (valid_array_reference (vname, 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       v = array_variable_part (vname, &temp, (int *)0); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       /* If we want to signal array_value to use an already-computed index,
 | 
					
						
							|  |  |  | 	 set LIND to that index */ | 
					
						
							|  |  |  |       lind = (ind != INTMAX_MIN && (flags & AV_USEIND)) ? ind : 0; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (v && invisible_p (v)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  vtype = VT_ARRAYMEMBER; | 
					
						
							|  |  |  | 	  *varp = (SHELL_VAR *)NULL; | 
					
						
							|  |  |  | 	  *valp = (char *)NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (v && (array_p (v) || assoc_p (v))) | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	{ /* [ */ | 
					
						
							|  |  |  | 	  if (ALL_ELEMENT_SUB (temp[0]) && temp[1] == ']') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      /* Callers have to differentiate between indexed and associative */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      vtype = VT_ARRAYVAR; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      if (temp[0] == '*') | 
					
						
							|  |  |  | 		vtype |= VT_STARSUB; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	      *valp = array_p (v) ? (char *)array_cell (v) : (char *)assoc_cell (v); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	      vtype = VT_ARRAYMEMBER; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      *valp = array_value (vname, Q_DOUBLE_QUOTES, flags, (int *)NULL, &lind); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  *varp = v; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       else if (v && (ALL_ELEMENT_SUB (temp[0]) && temp[1] == ']')) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  vtype = VT_VARIABLE; | 
					
						
							|  |  |  | 	  *varp = v; | 
					
						
							|  |  |  | 	  if (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  | 	    *valp = dequote_string (value); | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    *valp = dequote_escapes (value); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  vtype = VT_ARRAYMEMBER; | 
					
						
							|  |  |  | 	  *varp = v; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  *valp = array_value (vname, Q_DOUBLE_QUOTES, flags, (int *)NULL, &lind); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   else if ((v = find_variable (vname)) && (invisible_p (v) == 0) && (assoc_p (v) || array_p (v))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       vtype = VT_ARRAYMEMBER; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       *varp = v; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       *valp = assoc_p (v) ? assoc_reference (assoc_cell (v), "0") : array_reference (array_cell (v), 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (value && vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  *varp = find_variable (vname); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  | 	    *valp = dequote_string (value); | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    *valp = dequote_escapes (value); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	*valp = value; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (want_indir) | 
					
						
							|  |  |  |     free (vname); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   return vtype; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /******************************************************/ | 
					
						
							|  |  |  | /*						      */ | 
					
						
							|  |  |  | /* Functions to extract substrings of variable values */ | 
					
						
							|  |  |  | /*						      */ | 
					
						
							|  |  |  | /******************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | /* Character-oriented rather than strictly byte-oriented substrings.  S and
 | 
					
						
							|  |  |  |    E, rather being strict indices into STRING, indicate character (possibly | 
					
						
							|  |  |  |    multibyte character) positions that require calculation. | 
					
						
							|  |  |  |    Used by the ${param:offset[:length]} expansion. */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | mb_substring (string, s, e) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int s, e; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char *tt; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int start, stop, i; | 
					
						
							|  |  |  |   size_t slen; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   start = 0; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Don't need string length in ADVANCE_CHAR unless multibyte chars possible. */ | 
					
						
							|  |  |  |   slen = (MB_CUR_MAX > 1) ? STRLEN (string) : 0; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   i = s; | 
					
						
							|  |  |  |   while (string[start] && i--) | 
					
						
							|  |  |  |     ADVANCE_CHAR (string, slen, start); | 
					
						
							|  |  |  |   stop = start; | 
					
						
							|  |  |  |   i = e - s; | 
					
						
							|  |  |  |   while (string[stop] && i--) | 
					
						
							|  |  |  |     ADVANCE_CHAR (string, slen, stop); | 
					
						
							|  |  |  |   tt = substring (string, start, stop); | 
					
						
							|  |  |  |   return tt; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | /* Process a variable substring expansion: ${name:e1[:e2]}.  If VARNAME
 | 
					
						
							|  |  |  |    is `@', use the positional parameters; otherwise, use the value of | 
					
						
							|  |  |  |    VARNAME.  If VARNAME is an array variable, use the array elements. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | parameter_brace_substring (varname, value, ind, substr, quoted, flags) | 
					
						
							|  |  |  |      char *varname, *value; | 
					
						
							|  |  |  |      int ind; | 
					
						
							|  |  |  |      char *substr; | 
					
						
							|  |  |  |      int quoted, flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   intmax_t e1, e2; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   int vtype, r, starsub; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   char *temp, *val, *tt, *oname; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (value == 0 && ((varname[0] != '@' && varname[0] != '*') || varname[1])) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   oname = this_command_name; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   this_command_name = varname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   vtype = get_var_and_type (varname, value, ind, quoted, flags, &v, &val); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (vtype == -1) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       this_command_name = oname; | 
					
						
							|  |  |  |       return ((char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   starsub = vtype & VT_STARSUB; | 
					
						
							|  |  |  |   vtype &= ~VT_STARSUB; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   r = verify_substring_values (v, val, substr, vtype, &e1, &e2); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   this_command_name = oname; | 
					
						
							| 
									
										
										
										
											1997-09-22 20:22:27 +00:00
										 |  |  |   if (r <= 0) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       return ((r == 0) ? &expand_param_error : (char *)NULL); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (vtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case VT_VARIABLE: | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     case VT_ARRAYMEMBER: | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |       if (MB_CUR_MAX > 1) | 
					
						
							|  |  |  | 	tt = mb_substring (val, e1, e2); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       tt = substring (val, e1, e2); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       if (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) | 
					
						
							|  |  |  | 	temp = quote_string (tt); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	temp = tt ? quote_escapes (tt) : (char *)NULL; | 
					
						
							|  |  |  |       FREE (tt); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       tt = pos_params (varname, e1, e2, quoted); | 
					
						
							|  |  |  |       if ((quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = tt ? quote_escapes (tt) : (char *)NULL; | 
					
						
							|  |  |  | 	  FREE (tt); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	temp = tt; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (assoc_p (v)) | 
					
						
							|  |  |  | 	/* we convert to list and take first e2 elements starting at e1th
 | 
					
						
							|  |  |  | 	   element -- officially undefined for now */	 | 
					
						
							|  |  |  | 	temp = assoc_subrange (assoc_cell (v), e1, e2, starsub, quoted); | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       /* We want E2 to be the number of elements desired (arrays can be sparse,
 | 
					
						
							|  |  |  | 	 so verify_substring_values just returns the numbers specified and we | 
					
						
							|  |  |  | 	 rely on array_subrange to understand how to deal with them). */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	temp = array_subrange (array_cell (v), e1, e2, starsub, quoted); | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |       /* array_subrange now calls array_quote_escapes as appropriate, so the
 | 
					
						
							|  |  |  | 	 caller no longer needs to. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |     default: | 
					
						
							|  |  |  |       temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return temp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /****************************************************************/ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /* Functions to perform pattern substitution on variable values */ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /****************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #if 0	/* Unused */
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | static int | 
					
						
							|  |  |  | shouldexp_replacement (s) | 
					
						
							|  |  |  |      char *s; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (p = s; p && *p; p++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (*p == '\\') | 
					
						
							|  |  |  | 	p++; | 
					
						
							|  |  |  |       else if (*p == '&') | 
					
						
							|  |  |  | 	return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return 0; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | pat_subst (string, pat, rep, mflags) | 
					
						
							|  |  |  |      char *string, *pat, *rep; | 
					
						
							|  |  |  |      int mflags; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   char *ret, *s, *e, *str, *rstr, *mstr; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int rptr, mtype, rxpand, mlen; | 
					
						
							|  |  |  |   size_t rsize, l, replen, rslen; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (string  == 0) | 
					
						
							|  |  |  |     return (savestring ("")); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   mtype = mflags & MATCH_TYPEMASK; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | #if 0	/* bash-4.2 ? */
 | 
					
						
							|  |  |  |   rxpand = (rep && *rep) ? shouldexp_replacement (rep) : 0; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   rxpand = 0; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |   /* Special cases:
 | 
					
						
							|  |  |  |    * 	1.  A null pattern with mtype == MATCH_BEG means to prefix STRING | 
					
						
							|  |  |  |    *	    with REP and return the result. | 
					
						
							|  |  |  |    *	2.  A null pattern with mtype == MATCH_END means to append REP to | 
					
						
							|  |  |  |    *	    STRING and return the result. | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |    *	3.  A null STRING with a matching pattern means to append REP to | 
					
						
							|  |  |  |    *	    STRING and return the result. | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |    * These don't understand or process `&' in the replacement string. | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |    */ | 
					
						
							|  |  |  |   if ((pat == 0 || *pat == 0) && (mtype == MATCH_BEG || mtype == MATCH_END)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       replen = STRLEN (rep); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       l = STRLEN (string); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       ret = (char *)xmalloc (replen + l + 2); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       if (replen == 0) | 
					
						
							|  |  |  | 	strcpy (ret, string); | 
					
						
							|  |  |  |       else if (mtype == MATCH_BEG) | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  strcpy (ret, rep); | 
					
						
							|  |  |  | 	  strcpy (ret + replen, string); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  strcpy (ret, string); | 
					
						
							|  |  |  | 	  strcpy (ret + l, rep); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       return (ret); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   else if (*string == 0 && (match_pattern (string, pat, mtype, &s, &e) != 0)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       replen = STRLEN (rep); | 
					
						
							|  |  |  |       ret = (char *)xmalloc (replen + 1); | 
					
						
							|  |  |  |       if (replen == 0) | 
					
						
							|  |  |  | 	ret[0] = '\0'; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	strcpy (ret, rep); | 
					
						
							|  |  |  |       return (ret); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   ret = (char *)xmalloc (rsize = 64); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   ret[0] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   for (replen = STRLEN (rep), rptr = 0, str = string; *str;) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (match_pattern (str, pat, mtype, &s, &e) == 0) | 
					
						
							|  |  |  | 	break; | 
					
						
							|  |  |  |       l = s - str; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (rep && rxpand) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |         { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  int x; | 
					
						
							|  |  |  | 	  mlen = e - s; | 
					
						
							|  |  |  | 	  mstr = xmalloc (mlen + 1); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  for (x = 0; x < mlen; x++) | 
					
						
							|  |  |  | 	    mstr[x] = s[x]; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  mstr[mlen] = '\0'; | 
					
						
							|  |  |  | 	  rstr = strcreplace (rep, '&', mstr, 0); | 
					
						
							|  |  |  | 	  free (mstr); | 
					
						
							|  |  |  | 	  rslen = strlen (rstr); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  rstr = rep; | 
					
						
							|  |  |  | 	  rslen = replen; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |          | 
					
						
							|  |  |  |       RESIZE_MALLOCED_BUFFER (ret, rptr, (l + rslen), rsize, 64); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* OK, now copy the leading unmatched portion of the string (from
 | 
					
						
							|  |  |  | 	 str to s) to ret starting at rptr (the current offset).  Then copy | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 the replacement string at ret + rptr + (s - str).  Increment | 
					
						
							|  |  |  | 	 rptr (if necessary) and str and go on. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (l) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  strncpy (ret + rptr, str, l); | 
					
						
							|  |  |  | 	  rptr += l; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       if (replen) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  strncpy (ret + rptr, rstr, rslen); | 
					
						
							|  |  |  | 	  rptr += rslen; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       str = e;		/* e == end of match */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       if (rstr != rep) | 
					
						
							|  |  |  | 	free (rstr); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (((mflags & MATCH_GLOBREP) == 0) || mtype != MATCH_ANY) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	break; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (s == e) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  /* On a zero-length match, make sure we copy one character, since
 | 
					
						
							|  |  |  | 	     we increment one character to avoid infinite recursion. */ | 
					
						
							|  |  |  | 	  RESIZE_MALLOCED_BUFFER (ret, rptr, 1, rsize, 64); | 
					
						
							|  |  |  | 	  ret[rptr++] = *str++; | 
					
						
							|  |  |  | 	  e++;		/* avoid infinite recursion on zero-length match */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Now copy the unmatched portion of the input string */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   if (str && *str) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       RESIZE_MALLOCED_BUFFER (ret, rptr, STRLEN(str) + 1, rsize, 64); | 
					
						
							|  |  |  |       strcpy (ret + rptr, str); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   else | 
					
						
							|  |  |  |     ret[rptr] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Do pattern match and replacement on the positional parameters. */ | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | pos_params_pat_subst (string, pat, rep, mflags) | 
					
						
							|  |  |  |      char *string, *pat, *rep; | 
					
						
							|  |  |  |      int mflags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *save, *params; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   char *ret; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int pchar, qflags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   save = params = list_rest_of_args (); | 
					
						
							|  |  |  |   if (save == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for ( ; params; params = params->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = pat_subst (params->word->word, pat, rep, mflags); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       w = alloc_word_desc (); | 
					
						
							|  |  |  |       w->word = ret ? ret : savestring (""); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       dispose_word (params->word); | 
					
						
							|  |  |  |       params->word = w; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   pchar = (mflags & MATCH_STARSUB) == MATCH_STARSUB ? '*' : '@'; | 
					
						
							|  |  |  |   qflags = (mflags & MATCH_QUOTED) == MATCH_QUOTED ? Q_DOUBLE_QUOTES : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = string_list_pos_params (pchar, save, qflags); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   dispose_words (save); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Perform pattern substitution on VALUE, which is the expansion of
 | 
					
						
							|  |  |  |    VARNAME.  PATSUB is an expression supplying the pattern to match | 
					
						
							|  |  |  |    and the string to substitute.  QUOTED is a flags word containing | 
					
						
							|  |  |  |    the type of quoting currently in effect. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | static char * | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | parameter_brace_patsub (varname, value, ind, patsub, quoted, pflags, flags) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      char *varname, *value; | 
					
						
							|  |  |  |      int ind; | 
					
						
							|  |  |  |      char *patsub; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int quoted, pflags, flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int vtype, mflags, starsub, delim; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   char *val, *temp, *pat, *rep, *p, *lpatsub, *tt; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   this_command_name = varname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   vtype = get_var_and_type (varname, value, ind, quoted, flags, &v, &val); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (vtype == -1) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   starsub = vtype & VT_STARSUB; | 
					
						
							|  |  |  |   vtype &= ~VT_STARSUB; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   mflags = 0; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   /* PATSUB is never NULL when this is called. */ | 
					
						
							|  |  |  |   if (*patsub == '/') | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       mflags |= MATCH_GLOBREP; | 
					
						
							|  |  |  |       patsub++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Malloc this because expand_string_if_necessary or one of the expansion
 | 
					
						
							|  |  |  |      functions in its call chain may free it on a substitution error. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   lpatsub = savestring (patsub); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  |     mflags |= MATCH_QUOTED; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   if (starsub) | 
					
						
							|  |  |  |     mflags |= MATCH_STARSUB; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (pflags & PF_ASSIGNRHS) | 
					
						
							|  |  |  |     mflags |= MATCH_ASSIGNRHS; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   /* If the pattern starts with a `/', make sure we skip over it when looking
 | 
					
						
							|  |  |  |      for the replacement delimiter. */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   delim = skip_to_delim (lpatsub, ((*patsub == '/') ? 1 : 0), "/", 0); | 
					
						
							|  |  |  |   if (lpatsub[delim] == '/') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       lpatsub[delim] = 0; | 
					
						
							|  |  |  |       rep = lpatsub + delim + 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     rep = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (rep && *rep == '\0') | 
					
						
							|  |  |  |     rep = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Perform the same expansions on the pattern as performed by the
 | 
					
						
							|  |  |  |      pattern removal expansions. */ | 
					
						
							|  |  |  |   pat = getpattern (lpatsub, quoted, 1); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   if (rep) | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       /* We want to perform quote removal on the expanded replacement even if
 | 
					
						
							|  |  |  | 	 the entire expansion is double-quoted because the parser and string | 
					
						
							|  |  |  | 	 extraction functions treated quotes in the replacement string as | 
					
						
							|  |  |  | 	 special.  THIS IS NOT BACKWARDS COMPATIBLE WITH BASH-4.2. */ | 
					
						
							|  |  |  |       if (shell_compatibility_level > 42) | 
					
						
							|  |  |  | 	rep = expand_string_if_necessary (rep, quoted & ~(Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT), expand_string_unsplit); | 
					
						
							|  |  |  |       /* This is the bash-4.2 code. */       | 
					
						
							|  |  |  |       else if ((mflags & MATCH_QUOTED) == 0) | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	rep = expand_string_if_necessary (rep, quoted, expand_string_unsplit); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	rep = expand_string_to_string_internal (rep, quoted, expand_string_unsplit); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   /* ksh93 doesn't allow the match specifier to be a part of the expanded
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |      pattern.  This is an extension.  Make sure we don't anchor the pattern | 
					
						
							|  |  |  |      at the beginning or end of the string if we're doing global replacement, | 
					
						
							|  |  |  |      though. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   p = pat; | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   if (mflags & MATCH_GLOBREP) | 
					
						
							|  |  |  |     mflags |= MATCH_ANY; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   else if (pat && pat[0] == '#') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       mflags |= MATCH_BEG; | 
					
						
							|  |  |  |       p++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |   else if (pat && pat[0] == '%') | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       mflags |= MATCH_END; | 
					
						
							|  |  |  |       p++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     mflags |= MATCH_ANY; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* OK, we now want to substitute REP for PAT in VAL.  If
 | 
					
						
							|  |  |  |      flags & MATCH_GLOBREP is non-zero, the substitution is done | 
					
						
							|  |  |  |      everywhere, otherwise only the first occurrence of PAT is | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      replaced.  The pattern matching code doesn't understand | 
					
						
							|  |  |  |      CTLESC quoting CTLESC and CTLNUL so we use the dequoted variable | 
					
						
							|  |  |  |      values passed in (VT_VARIABLE) so the pattern substitution | 
					
						
							|  |  |  |      code works right.  We need to requote special chars after | 
					
						
							|  |  |  |      we're done for VT_VARIABLE and VT_ARRAYMEMBER, and for the | 
					
						
							|  |  |  |      other cases if QUOTED == 0, since the posparams and arrays | 
					
						
							|  |  |  |      indexed by * or @ do special things when QUOTED != 0. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   switch (vtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case VT_VARIABLE: | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  |     case VT_ARRAYMEMBER: | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       temp = pat_subst (val, p, rep, mflags); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       if (temp) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  tt = (mflags & MATCH_QUOTED) ? quote_string (temp) : quote_escapes (temp); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  free (temp); | 
					
						
							|  |  |  | 	  temp = tt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							|  |  |  |       temp = pos_params_pat_subst (val, p, rep, mflags); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (temp && (mflags & MATCH_QUOTED) == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tt = quote_escapes (temp); | 
					
						
							|  |  |  | 	  free (temp); | 
					
						
							|  |  |  | 	  temp = tt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       temp = assoc_p (v) ? assoc_patsub (assoc_cell (v), p, rep, mflags) | 
					
						
							|  |  |  | 			 : array_patsub (array_cell (v), p, rep, mflags); | 
					
						
							|  |  |  |       /* Don't call quote_escapes anymore; array_patsub calls
 | 
					
						
							|  |  |  | 	 array_quote_escapes as appropriate before adding the | 
					
						
							|  |  |  | 	 space separators; ditto for assoc_patsub. */ | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   FREE (pat); | 
					
						
							|  |  |  |   FREE (rep); | 
					
						
							|  |  |  |   free (lpatsub); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return temp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************/ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /*   Functions to perform case modification on variable values  */ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /****************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Do case modification on the positional parameters. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | pos_params_modcase (string, pat, modop, mflags) | 
					
						
							|  |  |  |      char *string, *pat; | 
					
						
							|  |  |  |      int modop; | 
					
						
							|  |  |  |      int mflags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *save, *params; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   char *ret; | 
					
						
							|  |  |  |   int pchar, qflags; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   save = params = list_rest_of_args (); | 
					
						
							|  |  |  |   if (save == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for ( ; params; params = params->next) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = sh_modcase (params->word->word, pat, modop); | 
					
						
							|  |  |  |       w = alloc_word_desc (); | 
					
						
							|  |  |  |       w->word = ret ? ret : savestring (""); | 
					
						
							|  |  |  |       dispose_word (params->word); | 
					
						
							|  |  |  |       params->word = w; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   pchar = (mflags & MATCH_STARSUB) == MATCH_STARSUB ? '*' : '@'; | 
					
						
							|  |  |  |   qflags = (mflags & MATCH_QUOTED) == MATCH_QUOTED ? Q_DOUBLE_QUOTES : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ret = string_list_pos_params (pchar, save, qflags); | 
					
						
							|  |  |  |   dispose_words (save); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (ret); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform case modification on VALUE, which is the expansion of
 | 
					
						
							|  |  |  |    VARNAME.  MODSPEC is an expression supplying the type of modification | 
					
						
							|  |  |  |    to perform.  QUOTED is a flags word containing the type of quoting | 
					
						
							|  |  |  |    currently in effect. */ | 
					
						
							|  |  |  | static char * | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | parameter_brace_casemod (varname, value, ind, modspec, patspec, quoted, flags) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      char *varname, *value; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      int ind, modspec; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      char *patspec; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |      int quoted, flags; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int vtype, starsub, modop, mflags, x; | 
					
						
							|  |  |  |   char *val, *temp, *pat, *p, *lpat, *tt; | 
					
						
							|  |  |  |   SHELL_VAR *v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (value == 0) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   this_command_name = varname; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   vtype = get_var_and_type (varname, value, ind, quoted, flags, &v, &val); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   if (vtype == -1) | 
					
						
							|  |  |  |     return ((char *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   starsub = vtype & VT_STARSUB; | 
					
						
							|  |  |  |   vtype &= ~VT_STARSUB; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   modop = 0; | 
					
						
							|  |  |  |   mflags = 0; | 
					
						
							|  |  |  |   if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  |     mflags |= MATCH_QUOTED; | 
					
						
							|  |  |  |   if (starsub) | 
					
						
							|  |  |  |     mflags |= MATCH_STARSUB; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   p = patspec; | 
					
						
							|  |  |  |   if (modspec == '^') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       x = p && p[0] == modspec; | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  |       modop = x ? CASE_UPPER : CASE_UPFIRST; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       p += x; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (modspec == ',') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       x = p && p[0] == modspec; | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  |       modop = x ? CASE_LOWER : CASE_LOWFIRST; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       p += x; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (modspec == '~') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       x = p && p[0] == modspec; | 
					
						
							|  |  |  |       modop = x ? CASE_TOGGLEALL : CASE_TOGGLE; | 
					
						
							|  |  |  |       p += x; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |   lpat = p ? savestring (p) : 0; | 
					
						
							|  |  |  |   /* Perform the same expansions on the pattern as performed by the
 | 
					
						
							|  |  |  |      pattern removal expansions.  FOR LATER */ | 
					
						
							|  |  |  |   pat = lpat ? getpattern (lpat, quoted, 1) : 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* OK, now we do the case modification. */ | 
					
						
							|  |  |  |   switch (vtype) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case VT_VARIABLE: | 
					
						
							|  |  |  |     case VT_ARRAYMEMBER: | 
					
						
							|  |  |  |       temp = sh_modcase (val, pat, modop); | 
					
						
							|  |  |  |       if (vtype == VT_VARIABLE) | 
					
						
							|  |  |  | 	FREE (val); | 
					
						
							|  |  |  |       if (temp) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tt = (mflags & MATCH_QUOTED) ? quote_string (temp) : quote_escapes (temp); | 
					
						
							|  |  |  | 	  free (temp); | 
					
						
							|  |  |  | 	  temp = tt; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case VT_POSPARMS: | 
					
						
							|  |  |  |       temp = pos_params_modcase (val, pat, modop, mflags); | 
					
						
							|  |  |  |       if (temp && (mflags & MATCH_QUOTED)  == 0) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  tt = quote_escapes (temp); | 
					
						
							|  |  |  | 	  free (temp); | 
					
						
							|  |  |  | 	  temp = tt; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |     case VT_ARRAYVAR: | 
					
						
							|  |  |  |       temp = assoc_p (v) ? assoc_modcase (assoc_cell (v), pat, modop, mflags) | 
					
						
							|  |  |  | 			 : array_modcase (array_cell (v), pat, modop, mflags); | 
					
						
							|  |  |  |       /* Don't call quote_escapes; array_modcase calls array_quote_escapes
 | 
					
						
							|  |  |  | 	 as appropriate before adding the space separators; ditto for | 
					
						
							|  |  |  | 	 assoc_modcase. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   FREE (pat); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   free (lpat); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return temp; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | /* Check for unbalanced parens in S, which is the contents of $(( ... )).  If
 | 
					
						
							|  |  |  |    any occur, this must be a nested command substitution, so return 0. | 
					
						
							|  |  |  |    Otherwise, return 1.  A valid arithmetic expression must always have a | 
					
						
							|  |  |  |    ( before a matching ), so any cases where there are more right parens | 
					
						
							|  |  |  |    means that this must not be an arithmetic expression, though the parser | 
					
						
							|  |  |  |    will not accept it without a balanced total number of parens. */ | 
					
						
							|  |  |  | static int | 
					
						
							|  |  |  | chk_arithsub (s, len) | 
					
						
							|  |  |  |      const char *s; | 
					
						
							|  |  |  |      int len; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i, count; | 
					
						
							|  |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   i = count = 0; | 
					
						
							|  |  |  |   while (i < len) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if (s[i] == LPAREN) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	count++; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if (s[i] == RPAREN) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  count--; | 
					
						
							|  |  |  | 	  if (count < 0) | 
					
						
							|  |  |  | 	    return 0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       switch (s[i]) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  ADVANCE_CHAR (s, len, i); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\\': | 
					
						
							|  |  |  | 	  i++; | 
					
						
							|  |  |  | 	  if (s[i]) | 
					
						
							|  |  |  | 	    ADVANCE_CHAR (s, len, i); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\'': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = skip_single_quoted (s, len, ++i, 0); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '"': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  i = skip_double_quoted ((char *)s, len, ++i, 0); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (count == 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /****************************************************************/ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /*	Functions to perform parameter expansion on a string	*/ | 
					
						
							|  |  |  | /*								*/ | 
					
						
							|  |  |  | /****************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | /* ${[#][!]name[[:][^[^]][,[,]]#[#]%[%]-=?+[word][:e1[:e2]]]} */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC * | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | parameter_brace_expand (string, indexp, quoted, pflags, quoted_dollar_atp, contains_dollar_at) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |      char *string; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      int *indexp, quoted, pflags, *quoted_dollar_atp, *contains_dollar_at; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   int check_nullness, var_is_set, var_is_null, var_is_special; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   int want_substring, want_indir, want_patsub, want_casemod; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *name, *value, *temp, *temp1; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC *tdesc, *ret; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int t_index, sindex, c, tflag, modspec, all_element_arrayref; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   intmax_t number; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   arrayind_t ind; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   temp = temp1 = value = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   var_is_set = var_is_null = var_is_special = check_nullness = 0; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   want_substring = want_indir = want_patsub = want_casemod = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   all_element_arrayref = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   sindex = *indexp; | 
					
						
							|  |  |  |   t_index = ++sindex; | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   /* ${#var} doesn't have any of the other parameter expansions on it. */ | 
					
						
							|  |  |  |   if (string[t_index] == '#' && legal_variable_starter (string[t_index+1]))		/* {{ */ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |     name = string_extract (string, &t_index, "}", SX_VARNAME); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (CASEMOD_EXPANSIONS)
 | 
					
						
							|  |  |  |     /* To enable case-toggling expansions using the `~' operator character
 | 
					
						
							|  |  |  |        change the 1 to 0. */ | 
					
						
							|  |  |  | #  if defined (CASEMOD_CAPCASE)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     name = string_extract (string, &t_index, "#%^,~:-=?+/@}", SX_VARNAME); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #  else
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     name = string_extract (string, &t_index, "#%^,:-=?+/@}", SX_VARNAME); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #  endif /* CASEMOD_CAPCASE */
 | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     name = string_extract (string, &t_index, "#%:-=?+/@}", SX_VARNAME); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #endif /* CASEMOD_EXPANSIONS */
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* Handle ${@[stuff]} now that @ is a word expansion operator.  Not exactly
 | 
					
						
							|  |  |  |      the cleanest code ever. */ | 
					
						
							|  |  |  |   if (*name == 0 && sindex == t_index && string[sindex] == '@') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name = (char *)xrealloc (name, 2); | 
					
						
							|  |  |  |       name[0] = '@'; | 
					
						
							|  |  |  |       name[1] = '\0'; | 
					
						
							|  |  |  |       t_index++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (*name == '!' && t_index > sindex && string[t_index] == '@' && string[t_index+1] == '}') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       name = (char *)xrealloc (name, t_index - sindex + 2); | 
					
						
							|  |  |  |       name[t_index - sindex] = '@'; | 
					
						
							|  |  |  |       name[t_index - sindex + 1] = '\0'; | 
					
						
							|  |  |  |       t_index++; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   ret = 0; | 
					
						
							|  |  |  |   tflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   ind = INTMAX_MIN; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* If the name really consists of a special variable, then make sure
 | 
					
						
							|  |  |  |      that we have the entire name.  We don't allow indirect references | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      to special variables except `#', `?', `@' and `*'.  This clause is | 
					
						
							|  |  |  |      designed to handle ${#SPECIAL} and ${!SPECIAL}, not anything more | 
					
						
							|  |  |  |      general. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   if ((sindex == t_index && VALID_SPECIAL_LENGTH_PARAM (string[t_index])) || | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       (sindex == t_index && string[sindex] == '#' && VALID_SPECIAL_LENGTH_PARAM (string[sindex + 1])) || | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       (sindex == t_index - 1 && string[sindex] == '!' && VALID_INDIR_PARAM (string[t_index]))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       t_index++; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = string_extract (string, &t_index, "#%:-=?+/@}", 0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       name = (char *)xrealloc (name, 3 + (strlen (temp1))); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       *name = string[sindex]; | 
					
						
							|  |  |  |       if (string[sindex] == '!') | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  /* indirect reference of $#, $?, $@, or $* */ | 
					
						
							|  |  |  | 	  name[1] = string[sindex + 1]; | 
					
						
							|  |  |  | 	  strcpy (name + 2, temp1); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       else	 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	strcpy (name + 1, temp1); | 
					
						
							|  |  |  |       free (temp1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   sindex = t_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Find out what character ended the variable name.  Then
 | 
					
						
							|  |  |  |      do the appropriate thing. */ | 
					
						
							|  |  |  |   if (c = string[sindex]) | 
					
						
							|  |  |  |     sindex++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If c is followed by one of the valid parameter expansion
 | 
					
						
							|  |  |  |      characters, move past it as normal.  If not, assume that | 
					
						
							|  |  |  |      a substring specification is being given, and do not move | 
					
						
							|  |  |  |      past it. */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   if (c == ':' && VALID_PARAM_EXPAND_CHAR (string[sindex])) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       check_nullness++; | 
					
						
							|  |  |  |       if (c = string[sindex]) | 
					
						
							|  |  |  | 	sindex++; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   else if (c == ':' && string[sindex] != RBRACE) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     want_substring = 1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   else if (c == '/' /* && string[sindex] != RBRACE */)	/* XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     want_patsub = 1; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (CASEMOD_EXPANSIONS)
 | 
					
						
							|  |  |  |   else if (c == '^' || c == ',' || c == '~') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       modspec = c; | 
					
						
							|  |  |  |       want_casemod = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Catch the valid and invalid brace expressions that made it through the
 | 
					
						
							|  |  |  |      tests above. */ | 
					
						
							|  |  |  |   /* ${#-} is a valid expansion and means to take the length of $-.
 | 
					
						
							|  |  |  |      Similarly for ${#?} and ${##}... */ | 
					
						
							|  |  |  |   if (name[0] == '#' && name[1] == '\0' && check_nullness == 0 && | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	VALID_SPECIAL_LENGTH_PARAM (c) && string[sindex] == RBRACE) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       name = (char *)xrealloc (name, 3); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       name[1] = c; | 
					
						
							|  |  |  |       name[2] = '\0'; | 
					
						
							|  |  |  |       c = string[sindex++]; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ...but ${#%}, ${#:}, ${#=}, ${#+}, and ${#/} are errors. */ | 
					
						
							|  |  |  |   if (name[0] == '#' && name[1] == '\0' && check_nullness == 0 && | 
					
						
							|  |  |  | 	member (c, "%:=+/") && string[sindex] == RBRACE) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       goto bad_substitution;	/* XXX - substitution error */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Indirect expansion begins with a `!'.  A valid indirect expansion is
 | 
					
						
							|  |  |  |      either a variable name, one of the positional parameters or a special | 
					
						
							|  |  |  |      variable that expands to one of the positional parameters. */ | 
					
						
							|  |  |  |   want_indir = *name == '!' && | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |     (legal_variable_starter ((unsigned char)name[1]) || DIGIT (name[1]) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 					|| VALID_INDIR_PARAM (name[1])); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Determine the value of this variable. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Check for special variables, directly referenced. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (SPECIAL_VAR (name, want_indir)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     var_is_special++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Check for special expansion things, like the length of a parameter */ | 
					
						
							|  |  |  |   if (*name == '#' && name[1]) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* If we are not pointing at the character just after the
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 closing brace, then we haven't gotten all of the name. | 
					
						
							|  |  |  | 	 Since it begins with a special character, this is a bad | 
					
						
							|  |  |  | 	 substitution.  Also check NAME for validity before trying | 
					
						
							|  |  |  | 	 to go on. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (string[sindex - 1] != RBRACE || (valid_length_expression (name) == 0)) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  goto bad_substitution;	/* substitution error */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       number = parameter_brace_expand_length (name); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       if (number == INTMAX_MIN && unbound_vars_is_error) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  err_unboundvar (name+1); | 
					
						
							|  |  |  | 	  free (name); | 
					
						
							|  |  |  | 	  return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       free (name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       *indexp = sindex; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (number < 0) | 
					
						
							|  |  |  |         return (&expand_wdesc_error); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  ret = alloc_word_desc (); | 
					
						
							|  |  |  | 	  ret->word = itos (number); | 
					
						
							|  |  |  | 	  return ret; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ${@} is identical to $@. */ | 
					
						
							|  |  |  |   if (name[0] == '@' && name[1] == '\0') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 	*quoted_dollar_atp = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  |       tflag |= W_DOLLARAT; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Process ${!PREFIX*} expansion. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (want_indir && string[sindex - 1] == RBRACE && | 
					
						
							|  |  |  |       (string[sindex - 2] == '*' || string[sindex - 2] == '@') && | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       legal_variable_starter ((unsigned char) name[1])) | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       char **x; | 
					
						
							|  |  |  |       WORD_LIST *xlist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       temp1 = savestring (name + 1); | 
					
						
							|  |  |  |       number = strlen (temp1); | 
					
						
							|  |  |  |       temp1[number - 1] = '\0'; | 
					
						
							|  |  |  |       x = all_variables_matching_prefix (temp1); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       xlist = strvec_to_word_list (x, 0, 0); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |       if (string[sindex - 2] == '*') | 
					
						
							|  |  |  | 	temp = string_list_dollar_star (xlist); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  temp = string_list_dollar_at (xlist, quoted, 0); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 	    *quoted_dollar_atp = 1; | 
					
						
							|  |  |  | 	  if (contains_dollar_at) | 
					
						
							|  |  |  | 	    *contains_dollar_at = 1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  tflag |= W_DOLLARAT; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       free (x); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  |       dispose_words (xlist); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       free (temp1); | 
					
						
							|  |  |  |       *indexp = sindex; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       free (name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       ret->flags = tflag;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)      
 | 
					
						
							|  |  |  |   /* Process ${!ARRAY[@]} and ${!ARRAY[*]} expansion. */ /* [ */ | 
					
						
							|  |  |  |   if (want_indir && string[sindex - 1] == RBRACE && | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       string[sindex - 2] == ']' && valid_array_reference (name+1, 0)) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       char *x, *x1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       temp1 = savestring (name + 1); | 
					
						
							|  |  |  |       x = array_variable_name (temp1, &x1, (int *)0);	/* [ */ | 
					
						
							|  |  |  |       FREE (x); | 
					
						
							|  |  |  |       if (ALL_ELEMENT_SUB (x1[0]) && x1[1] == ']') | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  temp = array_keys (temp1, quoted);	/* handles assoc vars too */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (x1[0] == '@') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 		*quoted_dollar_atp = 1; | 
					
						
							|  |  |  | 	      if (contains_dollar_at) | 
					
						
							|  |  |  | 		*contains_dollar_at = 1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	      tflag |= W_DOLLARAT; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    }	     | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  free (name); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  *indexp = sindex; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  ret = alloc_word_desc (); | 
					
						
							|  |  |  | 	  ret->word = temp; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  ret->flags = tflag;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  return ret; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       free (temp1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif /* ARRAY_VARS */
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |        | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* Make sure that NAME is valid before trying to go on. */ | 
					
						
							|  |  |  |   if (valid_brace_expansion_word (want_indir ? name + 1 : name, | 
					
						
							|  |  |  | 					var_is_special) == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       goto bad_substitution;		/* substitution error */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (want_indir) | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:31 -04:00
										 |  |  |     { | 
					
						
							|  |  |  |       tdesc = parameter_brace_expand_indir (name + 1, var_is_special, quoted, quoted_dollar_atp, contains_dollar_at); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (tdesc == &expand_wdesc_error || tdesc == &expand_wdesc_fatal) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = (char *)NULL; | 
					
						
							|  |  |  | 	  goto bad_substitution; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:31 -04:00
										 |  |  |       /* Turn off the W_ARRAYIND flag because there is no way for this function
 | 
					
						
							|  |  |  | 	 to return the index we're supposed to be using. */ | 
					
						
							|  |  |  |       if (tdesc && tdesc->flags) | 
					
						
							|  |  |  | 	tdesc->flags &= ~W_ARRAYIND; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |     tdesc = parameter_brace_expand_word (name, var_is_special, quoted, PF_IGNUNBOUND|(pflags&(PF_NOSPLIT2|PF_ASSIGNRHS)), &ind); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (tdesc) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       temp = tdesc->word; | 
					
						
							|  |  |  |       tflag = tdesc->flags; | 
					
						
							|  |  |  |       dispose_word_desc (tdesc); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |     temp = (char  *)0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   if (temp == &expand_param_error || temp == &expand_param_fatal) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       FREE (name); | 
					
						
							|  |  |  |       FREE (value); | 
					
						
							|  |  |  |       return (temp == &expand_param_error ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (valid_array_reference (name, 0)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       int qflags; | 
					
						
							|  |  |  |       char *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       qflags = quoted; | 
					
						
							|  |  |  |       /* If in a context where word splitting will not take place, treat as
 | 
					
						
							|  |  |  | 	 if double-quoted.  Has effects with $* and ${array[*]} */ | 
					
						
							|  |  |  |       if (pflags & PF_ASSIGNRHS) | 
					
						
							|  |  |  | 	qflags |= Q_DOUBLE_QUOTES; | 
					
						
							|  |  |  |       chk_atstar (name, qflags, quoted_dollar_atp, contains_dollar_at); | 
					
						
							|  |  |  |       /* We duplicate a little code here */ | 
					
						
							|  |  |  |       t = mbschr (name, '['); | 
					
						
							|  |  |  |       if (t && ALL_ELEMENT_SUB (t[1]) && t[2] == ']') | 
					
						
							|  |  |  |         all_element_arrayref = 1; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   var_is_set = temp != (char *)0; | 
					
						
							|  |  |  |   var_is_null = check_nullness && (var_is_set == 0 || *temp == 0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   /* XXX - this may not need to be restricted to special variables */ | 
					
						
							|  |  |  |   if (check_nullness) | 
					
						
							|  |  |  |     var_is_null |= var_is_set && var_is_special && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && QUOTED_NULL (temp); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Get the rest of the stuff inside the braces. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (c && c != RBRACE) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* Extract the contents of the ${ ... } expansion
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 according to the Posix.2 rules. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:59:44 -05:00
										 |  |  |       value = extract_dollar_brace_string (string, &sindex, quoted, (c == '%' || c == '#' || c =='/' || c == '^' || c == ',' || c ==':') ? SX_POSIXEXP|SX_WORD : SX_WORD); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (string[sindex] == RBRACE) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	sindex++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	goto bad_substitution;		/* substitution error */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     value = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   *indexp = sindex; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |   /* All the cases where an expansion can possibly generate an unbound
 | 
					
						
							|  |  |  |      variable error. */ | 
					
						
							|  |  |  |   if (want_substring || want_patsub || want_casemod || c == '#' || c == '%' || c == RBRACE) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (var_is_set == 0 && unbound_vars_is_error && ((name[0] != '@' && name[0] != '*') || name[1]) && all_element_arrayref == 0) | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  err_unboundvar (name); | 
					
						
							|  |  |  | 	  FREE (value); | 
					
						
							|  |  |  | 	  FREE (temp); | 
					
						
							|  |  |  | 	  free (name); | 
					
						
							|  |  |  | 	  return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* If this is a substring spec, process it and add the result. */ | 
					
						
							|  |  |  |   if (want_substring) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       temp1 = parameter_brace_substring (name, temp, ind, value, quoted, (tflag & W_ARRAYIND) ? AV_USEIND : 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       FREE (name); | 
					
						
							|  |  |  |       FREE (value); | 
					
						
							|  |  |  |       FREE (temp); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (temp1 == &expand_param_error) | 
					
						
							|  |  |  | 	return (&expand_wdesc_error); | 
					
						
							|  |  |  |       else if (temp1 == &expand_param_fatal) | 
					
						
							|  |  |  | 	return (&expand_wdesc_fatal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       /* We test quoted_dollar_atp because we want variants with double-quoted
 | 
					
						
							|  |  |  | 	 "$@" to take a different code path. In fact, we make sure at the end | 
					
						
							|  |  |  | 	 of expand_word_internal that we're only looking at these flags if | 
					
						
							|  |  |  | 	 quoted_dollar_at == 0. */ | 
					
						
							|  |  |  |       if (temp1 &&  | 
					
						
							|  |  |  |           (quoted_dollar_atp == 0 || *quoted_dollar_atp == 0) && | 
					
						
							|  |  |  | 	  QUOTED_NULL (temp1) && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	ret->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   else if (want_patsub) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = parameter_brace_patsub (name, temp, ind, value, quoted, pflags, (tflag & W_ARRAYIND) ? AV_USEIND : 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       FREE (name); | 
					
						
							|  |  |  |       FREE (value); | 
					
						
							|  |  |  |       FREE (temp); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (temp1 == &expand_param_error) | 
					
						
							|  |  |  | 	return (&expand_wdesc_error); | 
					
						
							|  |  |  |       else if (temp1 == &expand_param_fatal) | 
					
						
							|  |  |  | 	return (&expand_wdesc_fatal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (temp1 &&  | 
					
						
							|  |  |  |           (quoted_dollar_atp == 0 || *quoted_dollar_atp == 0) && | 
					
						
							|  |  |  | 	  QUOTED_NULL (temp1) && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	ret->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (CASEMOD_EXPANSIONS)
 | 
					
						
							|  |  |  |   else if (want_casemod) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       temp1 = parameter_brace_casemod (name, temp, ind, modspec, value, quoted, (tflag & W_ARRAYIND) ? AV_USEIND : 0); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       FREE (name); | 
					
						
							|  |  |  |       FREE (value); | 
					
						
							|  |  |  |       FREE (temp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (temp1 == &expand_param_error) | 
					
						
							|  |  |  | 	return (&expand_wdesc_error); | 
					
						
							|  |  |  |       else if (temp1 == &expand_param_fatal) | 
					
						
							|  |  |  | 	return (&expand_wdesc_fatal); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp1; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (temp1 && | 
					
						
							|  |  |  |           (quoted_dollar_atp == 0 || *quoted_dollar_atp == 0) && | 
					
						
							|  |  |  | 	  QUOTED_NULL (temp1) && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	ret->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							|  |  |  |       return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* Do the right thing based on which character ended the variable name. */ | 
					
						
							|  |  |  |   switch (c) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |     case '\0': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | bad_substitution: | 
					
						
							| 
									
										
										
										
											2012-03-13 15:12:07 -04:00
										 |  |  |       last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       report_error (_("%s: bad substitution"), string ? string : "??"); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       FREE (value); | 
					
						
							|  |  |  |       FREE (temp); | 
					
						
							|  |  |  |       free (name); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       if (shell_compatibility_level <= 43) | 
					
						
							|  |  |  | 	return &expand_wdesc_error; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	return ((posixly_correct && interactive_shell == 0) ? &expand_wdesc_fatal : &expand_wdesc_error); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     case RBRACE: | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |     case '@': | 
					
						
							|  |  |  |       temp1 = parameter_brace_transform (name, temp, ind, value, c, quoted, (tflag & W_ARRAYIND) ? AV_USEIND : 0); | 
					
						
							|  |  |  |       free (temp); | 
					
						
							|  |  |  |       free (value); | 
					
						
							|  |  |  |       free (name); | 
					
						
							|  |  |  |       if (temp1 == &expand_param_error || temp1 == &expand_param_fatal) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  report_error (_("%s: bad substitution"), string ? string : "??"); | 
					
						
							|  |  |  | 	  return (temp1 == &expand_param_error ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp1; | 
					
						
							|  |  |  |       if (temp1 && QUOTED_NULL (temp1) && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 	ret->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							|  |  |  |       return ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     case '#':	/* ${param#[#]pattern} */ | 
					
						
							|  |  |  |     case '%':	/* ${param%[%]pattern} */ | 
					
						
							|  |  |  |       if (value == 0 || *value == '\0' || temp == 0 || *temp == '\0') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  FREE (value); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       temp1 = parameter_brace_remove_pattern (name, temp, ind, value, c, quoted, (tflag & W_ARRAYIND) ? AV_USEIND : 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       free (temp); | 
					
						
							|  |  |  |       free (value); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       free (name); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->word = temp1; | 
					
						
							|  |  |  |       if (temp1 && QUOTED_NULL (temp1) && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 	ret->flags |= W_QUOTED|W_HASQUOTEDNULL; | 
					
						
							|  |  |  |       return ret; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case '-': | 
					
						
							|  |  |  |     case '=': | 
					
						
							|  |  |  |     case '?': | 
					
						
							|  |  |  |     case '+': | 
					
						
							|  |  |  |       if (var_is_set && var_is_null == 0) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If the operator is `+', we don't want the value of the named
 | 
					
						
							|  |  |  | 	     variable for anything, just the value of the right hand side. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (c == '+') | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	      /* XXX -- if we're double-quoted and the named variable is "$@",
 | 
					
						
							|  |  |  | 			we want to turn off any special handling of "$@" -- | 
					
						
							|  |  |  | 			we're not using it, so whatever is on the rhs applies. */ | 
					
						
							|  |  |  | 	      if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 		*quoted_dollar_atp = 0; | 
					
						
							|  |  |  | 	      if (contains_dollar_at) | 
					
						
							|  |  |  | 		*contains_dollar_at = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      FREE (temp); | 
					
						
							|  |  |  | 	      if (value) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 		  /* From Posix discussion on austin-group list.  Issue 221
 | 
					
						
							|  |  |  | 		     requires that backslashes escaping `}' inside | 
					
						
							|  |  |  | 		     double-quoted ${...} be removed. */ | 
					
						
							|  |  |  | 		  if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 		    quoted |= Q_DOLBRACE; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		  ret = parameter_brace_expand_rhs (name, value, c, | 
					
						
							|  |  |  | 						    quoted, | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 						    pflags, | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 						    quoted_dollar_atp, | 
					
						
							|  |  |  | 						    contains_dollar_at); | 
					
						
							|  |  |  | 		  /* XXX - fix up later, esp. noting presence of
 | 
					
						
							|  |  |  | 			   W_HASQUOTEDNULL in ret->flags */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  free (value); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      FREE (value); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  /* Otherwise do nothing; just use the value in TEMP. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else	/* VAR not set or VAR is NULL. */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  FREE (temp); | 
					
						
							|  |  |  | 	  temp = (char *)NULL; | 
					
						
							|  |  |  | 	  if (c == '=' && var_is_special) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      report_error (_("$%s: cannot assign in this way"), name); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      free (name); | 
					
						
							|  |  |  | 	      free (value); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      return &expand_wdesc_error; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (c == '?') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      parameter_brace_expand_error (name, value); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (c != '+') | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      /* XXX -- if we're double-quoted and the named variable is "$@",
 | 
					
						
							|  |  |  | 			we want to turn off any special handling of "$@" -- | 
					
						
							|  |  |  | 			we're not using it, so whatever is on the rhs applies. */ | 
					
						
							|  |  |  | 	      if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && quoted_dollar_atp) | 
					
						
							|  |  |  | 		*quoted_dollar_atp = 0; | 
					
						
							|  |  |  | 	      if (contains_dollar_at) | 
					
						
							|  |  |  | 		*contains_dollar_at = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      /* From Posix discussion on austin-group list.  Issue 221 requires
 | 
					
						
							|  |  |  | 		 that backslashes escaping `}' inside double-quoted ${...} be | 
					
						
							|  |  |  | 		 removed. */ | 
					
						
							|  |  |  | 	      if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 		quoted |= Q_DOLBRACE; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      ret = parameter_brace_expand_rhs (name, value, c, quoted, pflags, | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 						quoted_dollar_atp, | 
					
						
							|  |  |  | 						contains_dollar_at); | 
					
						
							|  |  |  | 	      /* XXX - fix up later, esp. noting presence of
 | 
					
						
							|  |  |  | 		       W_HASQUOTEDNULL in tdesc->flags */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  free (value); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   free (name); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (ret == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->flags = tflag; | 
					
						
							|  |  |  |       ret->word = temp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (ret); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Expand a single ${xxx} expansion.  The braces are optional.  When
 | 
					
						
							|  |  |  |    the braces are used, parameter_brace_expand() does the work, | 
					
						
							|  |  |  |    possibly calling param_expand recursively. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | static WORD_DESC * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | param_expand (string, sindex, quoted, expanded_something, | 
					
						
							|  |  |  | 	      contains_dollar_at, quoted_dollar_at_p, had_quoted_null_p, | 
					
						
							|  |  |  | 	      pflags) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int *sindex, quoted, *expanded_something, *contains_dollar_at; | 
					
						
							|  |  |  |      int *quoted_dollar_at_p, *had_quoted_null_p, pflags; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   char *temp, *temp1, uerror[3], *savecmd; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int zindex, t_index, expok; | 
					
						
							|  |  |  |   unsigned char c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   intmax_t number; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   SHELL_VAR *var; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   WORD_LIST *list; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_DESC *tdesc, *ret; | 
					
						
							|  |  |  |   int tflag; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | /*itrace("param_expand: `%s' pflags = %d", string+*sindex, pflags);*/ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   zindex = *sindex; | 
					
						
							|  |  |  |   c = string[++zindex]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   ret = tdesc = (WORD_DESC *)NULL; | 
					
						
							|  |  |  |   tflag = 0; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Do simple cases first. Switch on what follows '$'. */ | 
					
						
							|  |  |  |   switch (c) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     /* $0 .. $9? */ | 
					
						
							|  |  |  |     case '0': | 
					
						
							|  |  |  |     case '1': | 
					
						
							|  |  |  |     case '2': | 
					
						
							|  |  |  |     case '3': | 
					
						
							|  |  |  |     case '4': | 
					
						
							|  |  |  |     case '5': | 
					
						
							|  |  |  |     case '6': | 
					
						
							|  |  |  |     case '7': | 
					
						
							|  |  |  |     case '8': | 
					
						
							|  |  |  |     case '9': | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |       temp1 = dollar_vars[TODIGIT (c)]; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (unbound_vars_is_error && temp1 == (char *)NULL) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  uerror[0] = '$'; | 
					
						
							|  |  |  | 	  uerror[1] = c; | 
					
						
							|  |  |  | 	  uerror[2] = '\0'; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  err_unboundvar (uerror); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (temp1) | 
					
						
							|  |  |  | 	temp = (*temp1 && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 		  ? quote_string (temp1) | 
					
						
							|  |  |  | 		  : quote_escapes (temp1); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* $$ -- pid of the invoking shell. */ | 
					
						
							|  |  |  |     case '$': | 
					
						
							|  |  |  |       temp = itos (dollar_dollar_pid); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* $# -- number of positional parameters. */ | 
					
						
							|  |  |  |     case '#': | 
					
						
							|  |  |  |       temp = itos (number_of_args ()); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* $? -- return value of the last synchronous command. */ | 
					
						
							|  |  |  |     case '?': | 
					
						
							|  |  |  |       temp = itos (last_command_exit_value); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* $- -- flags supplied to the shell on invocation or by `set'. */ | 
					
						
							|  |  |  |     case '-': | 
					
						
							|  |  |  |       temp = which_set_flags (); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* $! -- Pid of the last asynchronous command. */ | 
					
						
							|  |  |  |     case '!': | 
					
						
							|  |  |  |       /* If no asynchronous pids have been created, expand to nothing.
 | 
					
						
							|  |  |  | 	 If `set -u' has been executed, and no async processes have | 
					
						
							|  |  |  | 	 been created, this is an expansion error. */ | 
					
						
							|  |  |  |       if (last_asynchronous_pid == NO_PID) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  if (expanded_something) | 
					
						
							|  |  |  | 	    *expanded_something = 0; | 
					
						
							|  |  |  | 	  temp = (char *)NULL; | 
					
						
							|  |  |  | 	  if (unbound_vars_is_error) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	      uerror[0] = '$'; | 
					
						
							|  |  |  | 	      uerror[1] = c; | 
					
						
							|  |  |  | 	      uerror[2] = '\0'; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	      err_unboundvar (uerror); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	temp = itos (last_asynchronous_pid); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* The only difference between this and $@ is when the arg is quoted. */ | 
					
						
							|  |  |  |     case '*':		/* `$*' */ | 
					
						
							|  |  |  |       list = list_rest_of_args (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  |       /* According to austin-group posix proposal by Geoff Clare in
 | 
					
						
							|  |  |  | 	 <20090505091501.GA10097@squonk.masqnet> of 5 May 2009: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  	"The shell shall write a message to standard error and | 
					
						
							|  |  |  |  	 immediately exit when it tries to expand an unset parameter | 
					
						
							|  |  |  |  	 other than the '@' and '*' special parameters." | 
					
						
							|  |  |  |       */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (list == 0 && unbound_vars_is_error && (pflags & PF_IGNUNBOUND) == 0) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  uerror[0] = '$'; | 
					
						
							|  |  |  | 	  uerror[1] = '*'; | 
					
						
							|  |  |  | 	  uerror[2] = '\0'; | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	  err_unboundvar (uerror); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* If there are no command-line arguments, this should just
 | 
					
						
							|  |  |  | 	 disappear if there are other characters in the expansion, | 
					
						
							|  |  |  | 	 even if it's quoted. */ | 
					
						
							|  |  |  |       if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && list == 0) | 
					
						
							|  |  |  | 	temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       else if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES|Q_PATQUOTE)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If we have "$*" we want to make a string of the positional
 | 
					
						
							|  |  |  | 	     parameters, separated by the first character of $IFS, and | 
					
						
							|  |  |  | 	     quote the whole string, including the separators.  If IFS | 
					
						
							|  |  |  | 	     is unset, the parameters are separated by ' '; if $IFS is | 
					
						
							|  |  |  | 	     null, the parameters are concatenated. */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  temp = (quoted & (Q_DOUBLE_QUOTES|Q_PATQUOTE)) ? string_list_dollar_star (list) : string_list (list); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (temp) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      temp1 = (quoted & Q_DOUBLE_QUOTES) ? quote_string (temp) : temp; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      if (*temp == 0) | 
					
						
							|  |  |  | 		tflag |= W_HASQUOTEDNULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      if (temp != temp1) | 
					
						
							|  |  |  | 		free (temp); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      temp = temp1; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  /* We check whether or not we're eventually going to split $* here,
 | 
					
						
							|  |  |  | 	     for example when IFS is empty and we are processing the rhs of | 
					
						
							|  |  |  | 	     an assignment statement.  In that case, we don't separate the | 
					
						
							|  |  |  | 	     arguments at all.  Otherwise, if the $* is not quoted it is | 
					
						
							|  |  |  | 	     identical to $@ */ | 
					
						
							|  |  |  | #  if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  | 	  if (expand_no_split_dollar_star && ifs_firstc[0] == 0) | 
					
						
							|  |  |  | #  else
 | 
					
						
							|  |  |  | 	  if (expand_no_split_dollar_star && ifs_firstc == 0) | 
					
						
							|  |  |  | #  endif
 | 
					
						
							|  |  |  | 	    temp = string_list_dollar_star (list); | 
					
						
							|  |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      temp = string_list_dollar_at (list, quoted, 0); | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      if (quoted == 0 && (ifs_is_set == 0 || ifs_is_null)) | 
					
						
							|  |  |  | 		tflag |= W_SPLITSPACE; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      /* If we're not quoted but we still don't want word splitting, make
 | 
					
						
							|  |  |  | 		 we quote the IFS characters to protect them from splitting (e.g., | 
					
						
							|  |  |  | 		 when $@ is in the string as well). */ | 
					
						
							|  |  |  | 	      else if (temp && quoted == 0 && ifs_is_set && (pflags & PF_ASSIGNRHS)) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  temp1 = quote_string (temp); | 
					
						
							|  |  |  | 		  free (temp); | 
					
						
							|  |  |  | 		  temp = temp1; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  if (expand_no_split_dollar_star == 0 && contains_dollar_at) | 
					
						
							|  |  |  | 	    *contains_dollar_at = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       dispose_words (list); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* When we have "$@" what we want is "$1" "$2" "$3" ... This
 | 
					
						
							|  |  |  |        means that we have to turn quoting off after we split into | 
					
						
							|  |  |  |        the individually quoted arguments so that the final split | 
					
						
							|  |  |  |        on the first character of $IFS is still done.  */ | 
					
						
							|  |  |  |     case '@':		/* `$@' */ | 
					
						
							|  |  |  |       list = list_rest_of_args (); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  |       /* According to austin-group posix proposal by Geoff Clare in
 | 
					
						
							|  |  |  | 	 <20090505091501.GA10097@squonk.masqnet> of 5 May 2009: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  	"The shell shall write a message to standard error and | 
					
						
							|  |  |  |  	 immediately exit when it tries to expand an unset parameter | 
					
						
							|  |  |  |  	 other than the '@' and '*' special parameters." | 
					
						
							|  |  |  |       */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (list == 0 && unbound_vars_is_error && (pflags & PF_IGNUNBOUND) == 0) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  uerror[0] = '$'; | 
					
						
							|  |  |  | 	  uerror[1] = '@'; | 
					
						
							|  |  |  | 	  uerror[2] = '\0'; | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | 	  err_unboundvar (uerror); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  return (interactive_shell ? &expand_wdesc_error : &expand_wdesc_fatal); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-11-21 20:49:12 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* We want to flag the fact that we saw this.  We can't turn
 | 
					
						
							|  |  |  | 	 off quoting entirely, because other characters in the | 
					
						
							|  |  |  | 	 string might need it (consider "\"$@\""), but we need some | 
					
						
							|  |  |  | 	 way to signal that the final split on the first character | 
					
						
							|  |  |  | 	 of $IFS should be done, even though QUOTED is 1. */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       /* XXX - should this test include Q_PATQUOTE? */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (quoted_dollar_at_p && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 	*quoted_dollar_at_p = 1; | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* We want to separate the positional parameters with the first
 | 
					
						
							|  |  |  | 	 character of $IFS in case $IFS is something other than a space. | 
					
						
							|  |  |  | 	 We also want to make sure that splitting is done no matter what -- | 
					
						
							|  |  |  | 	 according to POSIX.2, this expands to a list of the positional | 
					
						
							|  |  |  | 	 parameters no matter what IFS is set to. */ | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  |       /* XXX - what to do when in a context where word splitting is not
 | 
					
						
							|  |  |  | 	 performed? Even when IFS is not the default, posix seems to imply | 
					
						
							|  |  |  | 	 that we behave like unquoted $* ?  Maybe we should use PF_NOSPLIT2 | 
					
						
							|  |  |  | 	 here. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* XXX - bash-4.4/bash-5.0 passing PFLAGS */ | 
					
						
							|  |  |  |       temp = string_list_dollar_at (list, (pflags & PF_ASSIGNRHS) ? (quoted|Q_DOUBLE_QUOTES) : quoted, pflags); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       tflag |= W_DOLLARAT; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       dispose_words (list); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case LBRACE: | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       tdesc = parameter_brace_expand (string, &zindex, quoted, pflags, | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 				      quoted_dollar_at_p, | 
					
						
							|  |  |  | 				      contains_dollar_at); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (tdesc == &expand_wdesc_error || tdesc == &expand_wdesc_fatal) | 
					
						
							|  |  |  | 	return (tdesc); | 
					
						
							|  |  |  |       temp = tdesc ? tdesc->word : (char *)0; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* XXX */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       /* Quoted nulls should be removed if there is anything else
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 in the string. */ | 
					
						
							|  |  |  |       /* Note that we saw the quoted null so we can add one back at
 | 
					
						
							|  |  |  | 	 the end of this function if there are no other characters | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 in the string, discard TEMP, and go on.  The exception to | 
					
						
							|  |  |  | 	 this is when we have "${@}" and $1 is '', since $@ needs | 
					
						
							|  |  |  | 	 special handling. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (tdesc && tdesc->word && (tdesc->flags & W_HASQUOTEDNULL) && QUOTED_NULL (temp)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  if (had_quoted_null_p) | 
					
						
							|  |  |  | 	    *had_quoted_null_p = 1; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  if (*quoted_dollar_at_p == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      free (temp); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      tdesc->word = temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	     | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       ret = tdesc; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       goto return0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Do command or arithmetic substitution. */ | 
					
						
							|  |  |  |     case LPAREN: | 
					
						
							|  |  |  |       /* We have to extract the contents of this paren substitution. */ | 
					
						
							|  |  |  |       t_index = zindex + 1; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* XXX - might want to check for string[t_index+2] == LPAREN and parse
 | 
					
						
							|  |  |  | 	 as arithmetic substitution immediately. */ | 
					
						
							|  |  |  |       temp = extract_command_subst (string, &t_index, (pflags&PF_COMPLETE) ? SX_COMPLETE : 0); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       zindex = t_index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* For Posix.2-style `$(( ))' arithmetic substitution,
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 extract the expression and pass it to the evaluator. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (temp && *temp == LPAREN) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  char *temp2; | 
					
						
							|  |  |  | 	  temp1 = temp + 1; | 
					
						
							|  |  |  | 	  temp2 = savestring (temp1); | 
					
						
							|  |  |  | 	  t_index = strlen (temp2) - 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (temp2[t_index] != RPAREN) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      free (temp2); | 
					
						
							|  |  |  | 	      goto comsub; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Cut off ending `)' */ | 
					
						
							|  |  |  | 	  temp2[t_index] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	  if (chk_arithsub (temp2, t_index) == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      free (temp2); | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | 	      internal_warning (_("future versions of the shell will force evaluation as an arithmetic substitution")); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	      goto comsub; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  /* Expand variables found inside the expression. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  temp1 = expand_arith_string (temp2, Q_DOUBLE_QUOTES|Q_ARITH); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  free (temp2); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | arithsub: | 
					
						
							|  |  |  | 	  /* No error messages. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  savecmd = this_command_name; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  this_command_name = (char *)NULL; | 
					
						
							|  |  |  | 	  number = evalexp (temp1, &expok); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  this_command_name = savecmd; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  free (temp); | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  if (expok == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (interactive_shell == 0 && posixly_correct) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		  return (&expand_wdesc_fatal); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		return (&expand_wdesc_error); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  temp = itos (number); | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | comsub: | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       if (pflags & PF_NOCOMSUB) | 
					
						
							|  |  |  | 	/* we need zindex+1 because string[zindex] == RPAREN */ | 
					
						
							|  |  |  | 	temp1 = substring (string, *sindex, zindex+1); | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  tdesc = command_substitute (temp, quoted); | 
					
						
							|  |  |  | 	  temp1 = tdesc ? tdesc->word : (char *)NULL; | 
					
						
							|  |  |  | 	  if (tdesc) | 
					
						
							|  |  |  | 	    dispose_word_desc (tdesc); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       FREE (temp); | 
					
						
							|  |  |  |       temp = temp1; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* Do POSIX.2d9-style arithmetic substitution.  This will probably go
 | 
					
						
							|  |  |  |        away in a future bash release. */ | 
					
						
							|  |  |  |     case '[': | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       /* Extract the contents of this arithmetic substitution. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       t_index = zindex + 1; | 
					
						
							|  |  |  |       temp = extract_arithmetic_subst (string, &t_index); | 
					
						
							|  |  |  |       zindex = t_index; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |       if (temp == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = savestring (string); | 
					
						
							|  |  |  | 	  if (expanded_something) | 
					
						
							|  |  |  | 	    *expanded_something = 0; | 
					
						
							|  |  |  | 	  goto return0; | 
					
						
							|  |  |  | 	}	   | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |        /* Do initial variable expansion. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       temp1 = expand_arith_string (temp, Q_DOUBLE_QUOTES|Q_ARITH); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       goto arithsub; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |       /* Find the variable in VARIABLE_LIST. */ | 
					
						
							|  |  |  |       temp = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       for (t_index = zindex; (c = string[zindex]) && legal_variable_char (c); zindex++) | 
					
						
							|  |  |  | 	; | 
					
						
							|  |  |  |       temp1 = (zindex > t_index) ? substring (string, t_index, zindex) : (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If this isn't a variable name, then just output the `$'. */ | 
					
						
							|  |  |  |       if (temp1 == 0 || *temp1 == '\0') | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  FREE (temp1); | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	  temp = (char *)xmalloc (2); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  temp[0] = '$'; | 
					
						
							|  |  |  | 	  temp[1] = '\0'; | 
					
						
							|  |  |  | 	  if (expanded_something) | 
					
						
							|  |  |  | 	    *expanded_something = 0; | 
					
						
							|  |  |  | 	  goto return0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* If the variable exists, return its value cell. */ | 
					
						
							|  |  |  |       var = find_variable (temp1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       if (var && invisible_p (var) == 0 && var_isset (var)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if (assoc_p (var) || array_p (var)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	      temp = array_p (var) ? array_reference (array_cell (var), 0) | 
					
						
							|  |  |  | 				   : assoc_reference (assoc_cell (var), "0"); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      if (temp) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		temp = (*temp && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 			  ? quote_string (temp) | 
					
						
							|  |  |  | 			  : quote_escapes (temp); | 
					
						
							|  |  |  | 	      else if (unbound_vars_is_error) | 
					
						
							|  |  |  | 		goto unbound_variable; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      temp = value_cell (var); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      temp = (*temp && (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES))) | 
					
						
							|  |  |  | 			? quote_string (temp) | 
					
						
							|  |  |  | 			: quote_escapes (temp); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  free (temp1); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  goto return0; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-09-25 08:45:07 -04:00
										 |  |  |       else if (var && (invisible_p (var) || var_isset (var) == 0)) | 
					
						
							|  |  |  | 	temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       else if ((var = find_variable_last_nameref (temp1, 0)) && var_isset (var) && invisible_p (var) == 0) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  temp = nameref_cell (var); | 
					
						
							|  |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (temp && *temp && valid_array_reference (temp, 0)) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      tdesc = parameter_brace_expand_word (temp, SPECIAL_VAR (temp, 0), quoted, pflags, (arrayind_t *)NULL); | 
					
						
							|  |  |  | 	      if (tdesc == &expand_wdesc_error || tdesc == &expand_wdesc_fatal) | 
					
						
							|  |  |  | 		return (tdesc); | 
					
						
							|  |  |  | 	      ret = tdesc; | 
					
						
							|  |  |  | 	      goto return0; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  /* y=2 ; typeset -n x=y; echo $x is not the same as echo $2 in ksh */ | 
					
						
							|  |  |  | 	  if (temp && *temp && legal_identifier (temp) == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	      report_error (_("%s: invalid variable name for name reference"), temp); | 
					
						
							|  |  |  | 	      return (&expand_wdesc_error);	/* XXX */ | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = (char *)NULL; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       temp = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | unbound_variable: | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (unbound_vars_is_error) | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	  err_unboundvar (temp1); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  free (temp1); | 
					
						
							|  |  |  | 	  goto return0; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       free (temp1); | 
					
						
							|  |  |  |       last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  |       return ((unbound_vars_is_error && interactive_shell == 0) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		? &expand_wdesc_fatal | 
					
						
							|  |  |  | 		: &expand_wdesc_error); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (string[zindex]) | 
					
						
							|  |  |  |     zindex++; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | return0: | 
					
						
							|  |  |  |   *sindex = zindex; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (ret == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ret = alloc_word_desc (); | 
					
						
							|  |  |  |       ret->flags = tflag;	/* XXX */ | 
					
						
							|  |  |  |       ret->word = temp; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return ret; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | void | 
					
						
							|  |  |  | invalidate_cached_quoted_dollar_at () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   dispose_words (cached_quoted_dollar_at); | 
					
						
							|  |  |  |   cached_quoted_dollar_at = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Make a word list which is the result of parameter and variable
 | 
					
						
							|  |  |  |    expansion, command substitution, arithmetic substitution, and | 
					
						
							|  |  |  |    quote removal of WORD.  Return a pointer to a WORD_LIST which is | 
					
						
							|  |  |  |    the result of the expansion.  If WORD contains a null word, the | 
					
						
							|  |  |  |    word list returned is also null. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |    QUOTED contains flag values defined in shell.h. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |    ISEXP is used to tell expand_word_internal that the word should be | 
					
						
							|  |  |  |    treated as the result of an expansion.  This has implications for | 
					
						
							|  |  |  |    how IFS characters in the word are treated. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |    CONTAINS_DOLLAR_AT and EXPANDED_SOMETHING are return values; when non-null | 
					
						
							|  |  |  |    they point to an integer value which receives information about expansion. | 
					
						
							|  |  |  |    CONTAINS_DOLLAR_AT gets non-zero if WORD contained "$@", else zero. | 
					
						
							|  |  |  |    EXPANDED_SOMETHING get non-zero if WORD contained any parameter expansions, | 
					
						
							|  |  |  |    else zero. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    This only does word splitting in the case of $@ expansion.  In that | 
					
						
							|  |  |  |    case, we split on ' '. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Values for the local variable quoted_state. */ | 
					
						
							|  |  |  | #define UNQUOTED	 0
 | 
					
						
							|  |  |  | #define PARTIALLY_QUOTED 1
 | 
					
						
							|  |  |  | #define WHOLLY_QUOTED    2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static WORD_LIST * | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | expand_word_internal (word, quoted, isexp, contains_dollar_at, expanded_something) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      WORD_DESC *word; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |      int quoted, isexp; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      int *contains_dollar_at; | 
					
						
							|  |  |  |      int *expanded_something; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   WORD_LIST *list; | 
					
						
							|  |  |  |   WORD_DESC *tword; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* The intermediate string that we build while expanding. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *istring; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* The current size of the above object. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   size_t istring_size; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Index into ISTRING. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int istring_index; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Temporary string storage. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   char *temp, *temp1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* The text of WORD. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   register char *string; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   /* The size of STRING. */ | 
					
						
							|  |  |  |   size_t string_size; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* The index into STRING. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int sindex; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* This gets 1 if we see a $@ while quoted. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int quoted_dollar_at; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* One of UNQUOTED, PARTIALLY_QUOTED, or WHOLLY_QUOTED, depending on
 | 
					
						
							|  |  |  |      whether WORD contains no quoting characters, a partially quoted | 
					
						
							|  |  |  |      string (e.g., "xx"ab), or is fully quoted (e.g., "xxab"). */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int quoted_state; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* State flags */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   int had_quoted_null; | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  |   int has_dollar_at, temp_has_dollar_at; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   int split_on_spaces; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  |   int tflag; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   int pflags;			/* flags passed to param_expand */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int mb_cur_max; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   int assignoff;		/* If assignment, offset of `=' */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   register unsigned char c;	/* Current character. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   int t_index;			/* For calls to string_extract_xxx. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   char twochars[2]; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   /* OK, let's see if we can optimize a common idiom: "$@" */ | 
					
						
							|  |  |  |   if (STREQ (word->word, "\"$@\"") && | 
					
						
							|  |  |  |       (word->flags == (W_HASDOLLAR|W_QUOTED)) && | 
					
						
							|  |  |  |       dollar_vars[1])		/* XXX - check IFS here as well? */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (contains_dollar_at) | 
					
						
							|  |  |  | 	*contains_dollar_at = 1; | 
					
						
							|  |  |  |       if (expanded_something) | 
					
						
							|  |  |  | 	*expanded_something = 1; | 
					
						
							|  |  |  |       if (cached_quoted_dollar_at) | 
					
						
							|  |  |  | 	return (copy_word_list (cached_quoted_dollar_at)); | 
					
						
							|  |  |  |       list = list_rest_of_args (); | 
					
						
							|  |  |  |       list = quote_list (list); | 
					
						
							|  |  |  |       cached_quoted_dollar_at = copy_word_list (list); | 
					
						
							|  |  |  |       return (list); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   istring = (char *)xmalloc (istring_size = DEFAULT_INITIAL_ARRAY_SIZE); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   istring[istring_index = 0] = '\0'; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   quoted_dollar_at = had_quoted_null = has_dollar_at = 0; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   split_on_spaces = 0; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   quoted_state = UNQUOTED; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   string = word->word; | 
					
						
							|  |  |  |   if (string == 0) | 
					
						
							|  |  |  |     goto finished_with_string; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   mb_cur_max = MB_CUR_MAX; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Don't need the string length for the SADD... and COPY_ macros unless
 | 
					
						
							|  |  |  |      multibyte characters are possible. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   string_size = (mb_cur_max > 1) ? strlen (string) : 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (contains_dollar_at) | 
					
						
							|  |  |  |     *contains_dollar_at = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   assignoff = -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   /* Begin the expansion. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (sindex = 0; ;) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       c = string[sindex]; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       /* Case on top-level character. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       switch (c) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case '\0': | 
					
						
							|  |  |  | 	  goto finished_with_string; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case CTLESC: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  sindex++; | 
					
						
							|  |  |  | #if HANDLE_MULTIBYTE
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (mb_cur_max > 1 && string[sindex]) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      SADD_MBQCHAR_BODY(temp, string, sindex, string_size); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      temp = (char *)xmalloc (3); | 
					
						
							|  |  |  | 	      temp[0] = CTLESC; | 
					
						
							|  |  |  | 	      temp[1] = c = string[sindex]; | 
					
						
							|  |  |  | 	      temp[2] = '\0'; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | dollar_add_string: | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  if (string[sindex]) | 
					
						
							|  |  |  | 	    sindex++; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | add_string: | 
					
						
							|  |  |  | 	  if (temp) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      istring = sub_append_string (temp, istring, &istring_index, &istring_size); | 
					
						
							|  |  |  | 	      temp = (char *)0; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (PROCESS_SUBSTITUTION)
 | 
					
						
							|  |  |  | 	  /* Process substitution. */ | 
					
						
							|  |  |  | 	case '<': | 
					
						
							|  |  |  | 	case '>': | 
					
						
							|  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	    /* bash-4.4/bash-5.0
 | 
					
						
							|  |  |  | 	       XXX - technically this should only be expanded at the start | 
					
						
							|  |  |  | 	       of a word */ | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	    if (string[++sindex] != LPAREN || (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) || (word->flags & (W_DQUOTE|W_NOPROCSUB)) || posixly_correct) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 		sindex--;	/* add_character: label increments sindex */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		goto add_character; | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 	    else | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      t_index = sindex + 1; /* skip past both '<' and LPAREN */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-18 11:45:12 -04:00
										 |  |  | 	    temp1 = extract_process_subst (string, (c == '<') ? "<(" : ">(", &t_index, 0); /*))*/ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    sindex = t_index; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	    /* If the process substitution specification is `<()', we want to
 | 
					
						
							|  |  |  | 	       open the pipe for writing in the child and produce output; if | 
					
						
							|  |  |  | 	       it is `>()', we want to open the pipe for reading in the child | 
					
						
							|  |  |  | 	       and consume input. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    temp = temp1 ? process_substitute (temp1, (c == '>')) : (char *)0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	    FREE (temp1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	    goto dollar_add_string; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  | #endif /* PROCESS_SUBSTITUTION */
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	case '=': | 
					
						
							|  |  |  | 	  /* Posix.2 section 3.6.1 says that tildes following `=' in words
 | 
					
						
							|  |  |  | 	     which are not assignment statements are not expanded.  If the | 
					
						
							|  |  |  | 	     shell isn't in posix mode, though, we perform tilde expansion | 
					
						
							|  |  |  | 	     on `likely candidate' unquoted assignment statements (flags | 
					
						
							|  |  |  | 	     include W_ASSIGNMENT but not W_QUOTED).  A likely candidate | 
					
						
							|  |  |  | 	     contains an unquoted :~ or =~.  Something to think about: we | 
					
						
							|  |  |  | 	     now have a flag that says  to perform tilde expansion on arguments | 
					
						
							|  |  |  | 	     to `assignment builtins' like declare and export that look like | 
					
						
							|  |  |  | 	     assignment statements.  We now do tilde expansion on such words | 
					
						
							|  |  |  | 	     even in POSIX mode. */	 | 
					
						
							|  |  |  | 	  if (word->flags & (W_ASSIGNRHS|W_NOTILDE)) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	      if (isexp == 0 && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0 && isifs (c)) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 		goto add_ifs_character; | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		goto add_character; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  /* If we're not in posix mode or forcing assignment-statement tilde
 | 
					
						
							|  |  |  | 	     expansion, note where the `=' appears in the word and prepare to | 
					
						
							|  |  |  | 	     do tilde expansion following the first `='. */ | 
					
						
							|  |  |  | 	  if ((word->flags & W_ASSIGNMENT) && | 
					
						
							|  |  |  | 	      (posixly_correct == 0 || (word->flags & W_TILDEEXP)) && | 
					
						
							|  |  |  | 	      assignoff == -1 && sindex > 0) | 
					
						
							|  |  |  | 	    assignoff = sindex; | 
					
						
							|  |  |  | 	  if (sindex == assignoff && string[sindex+1] == '~')	/* XXX */ | 
					
						
							|  |  |  | 	    word->flags |= W_ITILDE; | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | 	  else if ((word->flags & W_ASSIGNMENT) && | 
					
						
							|  |  |  | 		   (posixly_correct == 0 || (word->flags & W_TILDEEXP)) && | 
					
						
							|  |  |  | 		   string[sindex+1] == '~') | 
					
						
							|  |  |  | 	    word->flags |= W_ITILDE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  /* XXX - bash-4.4/bash-5.0 */ | 
					
						
							|  |  |  | 	  if (word->flags & W_ASSIGNARG) | 
					
						
							|  |  |  | 	    word->flags |= W_ASSIGNRHS;		/* affects $@ */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  if (isexp == 0 && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0 && isifs (c)) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 	    goto add_ifs_character; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    goto add_character; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	case ':': | 
					
						
							|  |  |  | 	  if (word->flags & W_NOTILDE) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	      if (isexp == 0 && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0 && isifs (c)) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 		goto add_ifs_character; | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		goto add_character; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  if ((word->flags & (W_ASSIGNMENT|W_ASSIGNRHS|W_TILDEEXP)) && | 
					
						
							|  |  |  | 	      string[sindex+1] == '~') | 
					
						
							|  |  |  | 	    word->flags |= W_ITILDE; | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  if (isexp == 0 && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0 && isifs (c)) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 	    goto add_ifs_character; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    goto add_character; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	case '~': | 
					
						
							|  |  |  | 	  /* If the word isn't supposed to be tilde expanded, or we're not
 | 
					
						
							|  |  |  | 	     at the start of a word or after an unquoted : or = in an | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	     assignment statement, we don't do tilde expansion.  If we don't want | 
					
						
							|  |  |  | 	     tilde expansion when expanding words to be passed to the arithmetic | 
					
						
							|  |  |  | 	     evaluator, remove the check for Q_ARITH. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if ((word->flags & (W_NOTILDE|W_DQUOTE)) || | 
					
						
							|  |  |  | 	      (sindex > 0 && ((word->flags & W_ITILDE) == 0)) || | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      ((quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) && ((quoted & Q_ARITH) == 0))) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      word->flags &= ~W_ITILDE; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	      if (isexp == 0 && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0 && isifs (c) && (quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) == 0) | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 		goto add_ifs_character; | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		goto add_character; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (word->flags & W_ASSIGNRHS) | 
					
						
							|  |  |  | 	    tflag = 2; | 
					
						
							|  |  |  | 	  else if (word->flags & (W_ASSIGNMENT|W_TILDEEXP)) | 
					
						
							|  |  |  | 	    tflag = 1; | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    tflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  temp = bash_tilde_find_word (string + sindex, tflag, &t_index); | 
					
						
							|  |  |  | 	     | 
					
						
							|  |  |  | 	  word->flags &= ~W_ITILDE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (temp && *temp && t_index > 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      temp1 = bash_tilde_expand (temp, tflag); | 
					
						
							| 
									
										
										
										
											2006-10-10 14:15:34 +00:00
										 |  |  | 	      if  (temp1 && *temp1 == '~' && STREQ (temp, temp1)) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  FREE (temp); | 
					
						
							|  |  |  | 		  FREE (temp1); | 
					
						
							|  |  |  | 		  goto add_character;		/* tilde expansion failed */ | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      free (temp); | 
					
						
							|  |  |  | 	      temp = temp1; | 
					
						
							|  |  |  | 	      sindex += t_index; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	      goto add_quoted_string;		/* XXX was add_string */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      FREE (temp); | 
					
						
							|  |  |  | 	      goto add_character; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	case '$': | 
					
						
							|  |  |  | 	  if (expanded_something) | 
					
						
							|  |  |  | 	    *expanded_something = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  | 	  temp_has_dollar_at = 0; | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	  pflags = (word->flags & W_NOCOMSUB) ? PF_NOCOMSUB : 0; | 
					
						
							|  |  |  | 	  if (word->flags & W_NOSPLIT2) | 
					
						
							|  |  |  | 	    pflags |= PF_NOSPLIT2; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  if (word->flags & W_ASSIGNRHS) | 
					
						
							|  |  |  | 	    pflags |= PF_ASSIGNRHS; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (word->flags & W_COMPLETE) | 
					
						
							|  |  |  | 	    pflags |= PF_COMPLETE; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  tword = param_expand (string, &sindex, quoted, expanded_something, | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  | 			       &temp_has_dollar_at, "ed_dollar_at, | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 			       &had_quoted_null, pflags); | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  | 	  has_dollar_at += temp_has_dollar_at; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  split_on_spaces += (tword->flags & W_SPLITSPACE); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if (tword == &expand_wdesc_error || tword == &expand_wdesc_fatal) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      free (string); | 
					
						
							|  |  |  | 	      free (istring); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      return ((tword == &expand_wdesc_error) ? &expand_word_error | 
					
						
							|  |  |  | 						     : &expand_word_fatal); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  if (contains_dollar_at && has_dollar_at) | 
					
						
							|  |  |  | 	    *contains_dollar_at = 1; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  if (tword && (tword->flags & W_HASQUOTEDNULL)) | 
					
						
							|  |  |  | 	    had_quoted_null = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  temp = tword ? tword->word : (char *)NULL; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  dispose_word_desc (tword); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:07 -04:00
										 |  |  | 	  /* Kill quoted nulls; we will add them back at the end of
 | 
					
						
							|  |  |  | 	     expand_word_internal if nothing else in the string */ | 
					
						
							|  |  |  | 	  if (had_quoted_null && temp && QUOTED_NULL (temp)) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      FREE (temp); | 
					
						
							|  |  |  | 	      temp = (char *)NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  goto add_string; | 
					
						
							|  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	case '`':		/* Backquoted command substitution. */ | 
					
						
							|  |  |  | 	  { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    t_index = sindex++; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	    temp = string_extract (string, &sindex, "`", SX_REQMATCH); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    /* The test of sindex against t_index is to allow bare instances of
 | 
					
						
							|  |  |  | 	       ` to pass through, for backwards compatibility. */ | 
					
						
							|  |  |  | 	    if (temp == &extract_string_error || temp == &extract_string_fatal) | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		if (sindex - 1 == t_index) | 
					
						
							|  |  |  | 		  { | 
					
						
							|  |  |  | 		    sindex = t_index; | 
					
						
							|  |  |  | 		    goto add_character; | 
					
						
							|  |  |  | 		  } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 		last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 		report_error (_("bad substitution: no closing \"`\" in %s") , string+t_index); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		free (string); | 
					
						
							|  |  |  | 		free (istring); | 
					
						
							|  |  |  | 		return ((temp == &extract_string_error) ? &expand_word_error | 
					
						
							|  |  |  | 							: &expand_word_fatal); | 
					
						
							|  |  |  | 	      } | 
					
						
							|  |  |  | 		 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    if (expanded_something) | 
					
						
							|  |  |  | 	      *expanded_something = 1; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    if (word->flags & W_NOCOMSUB) | 
					
						
							|  |  |  | 	      /* sindex + 1 because string[sindex] == '`' */ | 
					
						
							|  |  |  | 	      temp1 = substring (string, t_index, sindex + 1); | 
					
						
							|  |  |  | 	    else | 
					
						
							|  |  |  | 	      { | 
					
						
							|  |  |  | 		de_backslash (temp); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 		tword = command_substitute (temp, quoted); | 
					
						
							|  |  |  | 		temp1 = tword ? tword->word : (char *)NULL; | 
					
						
							|  |  |  | 		if (tword) | 
					
						
							|  |  |  | 		  dispose_word_desc (tword); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    FREE (temp); | 
					
						
							|  |  |  | 	    temp = temp1; | 
					
						
							|  |  |  | 	    goto dollar_add_string; | 
					
						
							|  |  |  | 	  } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	case '\\': | 
					
						
							|  |  |  | 	  if (string[sindex + 1] == '\n') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      sindex += 2; | 
					
						
							|  |  |  | 	      continue; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  c = string[++sindex]; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  if (quoted & Q_HERE_DOCUMENT) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    tflag = CBSHDOC; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  else if (quoted & Q_DOUBLE_QUOTES) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    tflag = CBSDQUOTE; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	    tflag = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  /* From Posix discussion on austin-group list:  Backslash escaping
 | 
					
						
							|  |  |  | 	     a } in ${...} is removed.  Issue 0000221 */ | 
					
						
							|  |  |  | 	  if ((quoted & Q_DOLBRACE) && c == RBRACE) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      SCOPY_CHAR_I (twochars, CTLESC, c, string, sindex, string_size); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  /* This is the fix for " $@\ " */ | 
					
						
							|  |  |  | 	  else if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && ((sh_syntaxtab[c] & tflag) == 0) && isexp == 0 && isifs (c)) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      RESIZE_MALLOCED_BUFFER (istring, istring_index, 2, istring_size, | 
					
						
							|  |  |  | 				      DEFAULT_ARRAY_SIZE); | 
					
						
							|  |  |  | 	      istring[istring_index++] = CTLESC; | 
					
						
							|  |  |  | 	      istring[istring_index++] = '\\'; | 
					
						
							|  |  |  | 	      istring[istring_index] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	      SCOPY_CHAR_I (twochars, CTLESC, c, string, sindex, string_size); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) && ((sh_syntaxtab[c] & tflag) == 0)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	      SCOPY_CHAR_I (twochars, '\\', c, string, sindex, string_size); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (c == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      c = CTLNUL; | 
					
						
							|  |  |  | 	      sindex--;		/* add_character: label increments sindex */ | 
					
						
							|  |  |  | 	      goto add_character; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	      SCOPY_CHAR_I (twochars, CTLESC, c, string, sindex, string_size); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  sindex++; | 
					
						
							|  |  |  | add_twochars: | 
					
						
							|  |  |  | 	  /* BEFORE jumping here, we need to increment sindex if appropriate */ | 
					
						
							|  |  |  | 	  RESIZE_MALLOCED_BUFFER (istring, istring_index, 2, istring_size, | 
					
						
							|  |  |  | 				  DEFAULT_ARRAY_SIZE); | 
					
						
							|  |  |  | 	  istring[istring_index++] = twochars[0]; | 
					
						
							|  |  |  | 	  istring[istring_index++] = twochars[1]; | 
					
						
							|  |  |  | 	  istring[istring_index] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	case '"': | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if ((quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) && ((quoted & Q_ARITH) == 0)) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    goto add_character; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  t_index = ++sindex; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  temp = string_extract_double_quoted (string, &sindex, (word->flags & W_COMPLETE) ? SX_COMPLETE : 0); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  /* If the quotes surrounded the entire string, then the
 | 
					
						
							|  |  |  | 	     whole word was quoted. */ | 
					
						
							|  |  |  | 	  quoted_state = (t_index == 1 && string[sindex] == '\0') | 
					
						
							|  |  |  | 			    ? WHOLLY_QUOTED | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 			    : PARTIALLY_QUOTED; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  if (temp && *temp) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      tword = alloc_word_desc (); | 
					
						
							|  |  |  | 	      tword->word = temp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      /* XXX - bash-4.4/bash-5.0 */ | 
					
						
							|  |  |  | 	      if (word->flags & W_ASSIGNARG) | 
					
						
							|  |  |  | 		tword->flags |= word->flags & (W_ASSIGNARG|W_ASSIGNRHS);	/* affects $@ */ | 
					
						
							|  |  |  | 	      if (word->flags & W_COMPLETE) | 
					
						
							|  |  |  | 		tword->flags |= W_COMPLETE;	/* for command substitutions */ | 
					
						
							| 
									
										
										
										
											2017-01-20 11:47:55 -05:00
										 |  |  | 	      if (word->flags & W_NOCOMSUB) | 
					
						
							|  |  |  | 		tword->flags |= W_NOCOMSUB; | 
					
						
							|  |  |  | 	      if (word->flags & W_NOPROCSUB) | 
					
						
							|  |  |  | 		tword->flags |= W_NOPROCSUB; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      temp = (char *)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  | 	      temp_has_dollar_at = 0;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      /* Need to get W_HASQUOTEDNULL flag through this function. */ | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:54 -04:00
										 |  |  | 	      list = expand_word_internal (tword, Q_DOUBLE_QUOTES, 0, &temp_has_dollar_at, (int *)NULL); | 
					
						
							|  |  |  | 	      has_dollar_at += temp_has_dollar_at; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      if (list == &expand_word_error || list == &expand_word_fatal) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  free (istring); | 
					
						
							|  |  |  | 		  free (string); | 
					
						
							|  |  |  | 		  /* expand_word_internal has already freed temp_word->word
 | 
					
						
							|  |  |  | 		     for us because of the way it prints error messages. */ | 
					
						
							|  |  |  | 		  tword->word = (char *)NULL; | 
					
						
							|  |  |  | 		  dispose_word (tword); | 
					
						
							|  |  |  | 		  return list; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      dispose_word (tword); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      /* "$@" (a double-quoted dollar-at) expands into nothing,
 | 
					
						
							|  |  |  | 		 not even a NULL word, when there are no positional | 
					
						
							|  |  |  | 		 parameters. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      if (list == 0 && temp_has_dollar_at)	/* XXX - was has_dollar_at */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  quoted_dollar_at++; | 
					
						
							|  |  |  | 		  break; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      /* If we get "$@", we know we have expanded something, so we
 | 
					
						
							|  |  |  | 		 need to remember it for the final split on $IFS.  This is | 
					
						
							|  |  |  | 		 a special case; it's the only case where a quoted string | 
					
						
							|  |  |  | 		 can expand into more than one word.  It's going to come back | 
					
						
							|  |  |  | 		 from the above call to expand_word_internal as a list with | 
					
						
							|  |  |  | 		 a single word, in which all characters are quoted and | 
					
						
							|  |  |  | 		 separated by blanks.  What we want to do is to turn it back | 
					
						
							|  |  |  | 		 into a list for the next piece of code. */ | 
					
						
							|  |  |  | 	      if (list) | 
					
						
							|  |  |  | 		dequote_list (list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      if (list && list->word && (list->word->flags & W_HASQUOTEDNULL)) | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 		had_quoted_null = 1;		/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      if (temp_has_dollar_at)		/* XXX - was has_dollar_at */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  quoted_dollar_at++; | 
					
						
							|  |  |  | 		  if (contains_dollar_at) | 
					
						
							|  |  |  | 		    *contains_dollar_at = 1; | 
					
						
							|  |  |  | 		  if (expanded_something) | 
					
						
							|  |  |  | 		    *expanded_something = 1; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      /* What we have is "".  This is a minor optimization. */ | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | 	      FREE (temp); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      list = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* The code above *might* return a list (consider the case of "$@",
 | 
					
						
							|  |  |  | 	     where it returns "$1", "$2", etc.).  We can't throw away the | 
					
						
							|  |  |  | 	     rest of the list, and we have to make sure each word gets added | 
					
						
							|  |  |  | 	     as quoted.  We test on tresult->next:  if it is non-NULL, we | 
					
						
							|  |  |  | 	     quote the whole list, save it to a string with string_list, and | 
					
						
							|  |  |  | 	     add that string. We don't need to quote the results of this | 
					
						
							|  |  |  | 	     (and it would be wrong, since that would quote the separators | 
					
						
							|  |  |  | 	     as well), so we go directly to add_string. */ | 
					
						
							|  |  |  | 	  if (list) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      if (list->next) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											1998-07-23 14:37:54 +00:00
										 |  |  | 		  /* Testing quoted_dollar_at makes sure that "$@" is
 | 
					
						
							|  |  |  | 		     split correctly when $IFS does not contain a space. */ | 
					
						
							|  |  |  | 		  temp = quoted_dollar_at | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 				? string_list_dollar_at (list, Q_DOUBLE_QUOTES, 0) | 
					
						
							| 
									
										
										
										
											1998-07-23 14:37:54 +00:00
										 |  |  | 				: string_list (quote_list (list)); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  dispose_words (list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		  goto add_string; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  temp = savestring (list->word->word); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		  tflag = list->word->flags; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		  dispose_words (list); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		  /* If the string is not a quoted null string, we want
 | 
					
						
							|  |  |  | 		     to remove any embedded unquoted CTLNUL characters. | 
					
						
							|  |  |  | 		     We do not want to turn quoted null strings back into | 
					
						
							|  |  |  | 		     the empty string, though.  We do this because we | 
					
						
							|  |  |  | 		     want to remove any quoted nulls from expansions that | 
					
						
							|  |  |  | 		     contain other characters.  For example, if we have | 
					
						
							|  |  |  | 		     x"$*"y or "x$*y" and there are no positional parameters, | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 		     the $* should expand into nothing. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 		  /* We use the W_HASQUOTEDNULL flag to differentiate the
 | 
					
						
							|  |  |  | 		     cases:  a quoted null character as above and when | 
					
						
							|  |  |  | 		     CTLNUL is contained in the (non-null) expansion | 
					
						
							|  |  |  | 		     of some variable.  We use the had_quoted_null flag to | 
					
						
							|  |  |  | 		     pass the value through this function to its caller. */ | 
					
						
							|  |  |  | 		  if ((tflag & W_HASQUOTEDNULL) && QUOTED_NULL (temp) == 0) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		    remove_quoted_nulls (temp);	/* XXX */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  else | 
					
						
							|  |  |  | 	    temp = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* We do not want to add quoted nulls to strings that are only
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	     partially quoted; we can throw them away.  The exception to | 
					
						
							|  |  |  | 	     this is when we are going to be performing word splitting, | 
					
						
							|  |  |  | 	     since we have to preserve a null argument if the next character | 
					
						
							|  |  |  | 	     will cause word splitting. */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  if (temp == 0 && quoted_state == PARTIALLY_QUOTED && (word->flags & (W_NOSPLIT|W_NOSPLIT2))) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	add_quoted_string: | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (temp) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      temp1 = temp; | 
					
						
							|  |  |  | 	      temp = quote_string (temp); | 
					
						
							|  |  |  | 	      free (temp1); | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      goto add_string; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Add NULL arg. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      c = CTLNUL; | 
					
						
							|  |  |  | 	      sindex--;		/* add_character: label increments sindex */ | 
					
						
							|  |  |  | 	      goto add_character; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* break; */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	case '\'': | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if ((quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    goto add_character; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  t_index = ++sindex; | 
					
						
							|  |  |  | 	  temp = string_extract_single_quoted (string, &sindex); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* If the entire STRING was surrounded by single quotes,
 | 
					
						
							|  |  |  | 	     then the string is wholly quoted. */ | 
					
						
							|  |  |  | 	  quoted_state = (t_index == 1 && string[sindex] == '\0') | 
					
						
							|  |  |  | 			    ? WHOLLY_QUOTED | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 			    : PARTIALLY_QUOTED; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* If all we had was '', it is a null expansion. */ | 
					
						
							|  |  |  | 	  if (*temp == '\0') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      free (temp); | 
					
						
							|  |  |  | 	      temp = (char *)NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	    remove_quoted_escapes (temp);	/* ??? */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* We do not want to add quoted nulls to strings that are only
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	     partially quoted; such nulls are discarded.  See above for the | 
					
						
							|  |  |  | 	     exception, which is when the string is going to be split. */ | 
					
						
							|  |  |  | 	  if (temp == 0 && (quoted_state == PARTIALLY_QUOTED) && (word->flags & (W_NOSPLIT|W_NOSPLIT2))) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    continue; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  /* If we have a quoted null expansion, add a quoted NULL to istring. */ | 
					
						
							|  |  |  | 	  if (temp == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      c = CTLNUL; | 
					
						
							|  |  |  | 	      sindex--;		/* add_character: label increments sindex */ | 
					
						
							|  |  |  | 	      goto add_character; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    goto add_quoted_string; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* break; */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 	  /* This is the fix for " $@ " */ | 
					
						
							| 
									
										
										
										
											2009-02-19 22:21:29 +00:00
										 |  |  | 	add_ifs_character: | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) || (isexp == 0 && isifs (c) && (word->flags & (W_NOSPLIT|W_NOSPLIT2)) == 0)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      if (string[sindex])	/* from old goto dollar_add_string */ | 
					
						
							|  |  |  | 		sindex++; | 
					
						
							|  |  |  | 	      if (c == 0) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  c = CTLNUL; | 
					
						
							|  |  |  | 		  goto add_character; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	      else | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | #if HANDLE_MULTIBYTE
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  if (mb_cur_max > 1) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		    sindex--; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  if (mb_cur_max > 1) | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 		    { | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		      SADD_MBQCHAR_BODY(temp, string, sindex, string_size); | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 		    } | 
					
						
							|  |  |  | 		  else | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 		    { | 
					
						
							|  |  |  | 		      twochars[0] = CTLESC; | 
					
						
							|  |  |  | 		      twochars[1] = c; | 
					
						
							|  |  |  | 		      goto add_twochars; | 
					
						
							|  |  |  | 		    } | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  SADD_MBCHAR (temp, string, sindex, string_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	add_character: | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  RESIZE_MALLOCED_BUFFER (istring, istring_index, 1, istring_size, | 
					
						
							|  |  |  | 				  DEFAULT_ARRAY_SIZE); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  istring[istring_index++] = c; | 
					
						
							|  |  |  | 	  istring[istring_index] = '\0'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Next character. */ | 
					
						
							|  |  |  | 	  sindex++; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | finished_with_string: | 
					
						
							|  |  |  |   /* OK, we're ready to return.  If we have a quoted string, and
 | 
					
						
							|  |  |  |      quoted_dollar_at is not set, we do no splitting at all; otherwise | 
					
						
							|  |  |  |      we split on ' '.  The routines that call this will handle what to | 
					
						
							|  |  |  |      do if nothing has been expanded. */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Partially and wholly quoted strings which expand to the empty
 | 
					
						
							|  |  |  |      string are retained as an empty arguments.  Unquoted strings | 
					
						
							|  |  |  |      which expand to the empty string are discarded.  The single | 
					
						
							|  |  |  |      exception is the case of expanding "$@" when there are no | 
					
						
							|  |  |  |      positional parameters.  In that case, we discard the expansion. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Because of how the code that handles "" and '' in partially
 | 
					
						
							|  |  |  |      quoted strings works, we need to make ISTRING into a QUOTED_NULL | 
					
						
							|  |  |  |      if we saw quoting characters, but the expansion was empty. | 
					
						
							|  |  |  |      "" and '' are tossed away before we get to this point when | 
					
						
							|  |  |  |      processing partially quoted strings.  This makes "" and $xxx"" | 
					
						
							|  |  |  |      equivalent when xxx is unset.  We also look to see whether we | 
					
						
							|  |  |  |      saw a quoted null from a ${} expansion and add one back if we | 
					
						
							|  |  |  |      need to. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If we expand to nothing and there were no single or double quotes
 | 
					
						
							|  |  |  |      in the word, we throw it away.  Otherwise, we return a NULL word. | 
					
						
							|  |  |  |      The single exception is for $@ surrounded by double quotes when | 
					
						
							|  |  |  |      there are no positional parameters.  In that case, we also throw | 
					
						
							|  |  |  |      the word away. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (*istring == '\0') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (quoted_dollar_at == 0 && (had_quoted_null || quoted_state == PARTIALLY_QUOTED)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  istring[0] = CTLNUL; | 
					
						
							|  |  |  | 	  istring[1] = '\0'; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  tword = make_bare_word (istring); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  tword->flags |= W_HASQUOTEDNULL;		/* XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  list = make_word_list (tword, (WORD_LIST *)NULL); | 
					
						
							|  |  |  | 	  if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 	    tword->flags |= W_QUOTED; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* According to sh, ksh, and Posix.2, if a word expands into nothing
 | 
					
						
							|  |  |  | 	 and a double-quoted "$@" appears anywhere in it, then the entire | 
					
						
							|  |  |  | 	 word is removed. */ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* XXX - exception appears to be that quoted null strings result in
 | 
					
						
							|  |  |  | 	 null arguments */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else  if (quoted_state == UNQUOTED || quoted_dollar_at) | 
					
						
							|  |  |  | 	list = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  |       else | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  tword = make_bare_word (istring); | 
					
						
							|  |  |  | 	  if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							|  |  |  | 	    tword->flags |= W_QUOTED; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  list = make_word_list (tword, (WORD_LIST *)NULL); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	list = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else if (word->flags & W_NOSPLIT) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       tword = make_bare_word (istring); | 
					
						
							|  |  |  |       if (word->flags & W_ASSIGNMENT) | 
					
						
							|  |  |  | 	tword->flags |= W_ASSIGNMENT;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (word->flags & W_COMPASSIGN) | 
					
						
							|  |  |  | 	tword->flags |= W_COMPASSIGN;	/* XXX */ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |       if (word->flags & W_NOGLOB) | 
					
						
							|  |  |  | 	tword->flags |= W_NOGLOB;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (word->flags & W_NOBRACE) | 
					
						
							|  |  |  | 	tword->flags |= W_NOBRACE;	/* XXX */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (word->flags & W_NOEXPAND) | 
					
						
							|  |  |  | 	tword->flags |= W_NOEXPAND;	/* XXX */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       if (quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	tword->flags |= W_QUOTED; | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:07 -04:00
										 |  |  |       if (had_quoted_null && QUOTED_NULL (istring)) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	tword->flags |= W_HASQUOTEDNULL; | 
					
						
							|  |  |  |       list = make_word_list (tword, (WORD_LIST *)NULL); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       char *ifs_chars; | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  |       char *tstring; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       ifs_chars = (quoted_dollar_at || has_dollar_at) ? ifs_value : (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* If we have $@, we need to split the results no matter what.  If
 | 
					
						
							|  |  |  | 	 IFS is unset or NULL, string_list_dollar_at has separated the | 
					
						
							|  |  |  | 	 positional parameters with a space, so we split on space (we have | 
					
						
							|  |  |  | 	 set ifs_chars to " \t\n" above if ifs is unset).  If IFS is set, | 
					
						
							|  |  |  | 	 string_list_dollar_at has separated the positional parameters | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	 with the first character of $IFS, so we split on $IFS.  If | 
					
						
							|  |  |  | 	 SPLIT_ON_SPACES is set, we expanded $* (unquoted) with IFS either | 
					
						
							|  |  |  | 	 unset or null, and we want to make sure that we split on spaces | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	 regardless of what else has happened to IFS since the expansion, | 
					
						
							|  |  |  | 	 or we expanded "$@" with IFS null and we need to split the positional | 
					
						
							|  |  |  | 	 parameters into separate words. */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (split_on_spaces) | 
					
						
							|  |  |  | 	list = list_string (istring, " ", 1);	/* XXX quoted == 1? */ | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  |       /* If we have $@ (has_dollar_at != 0) and we are in a context where we
 | 
					
						
							|  |  |  | 	 don't want to split the result (W_NOSPLIT2), and we are not quoted, | 
					
						
							|  |  |  | 	 we have already separated the arguments with the first character of | 
					
						
							|  |  |  | 	 $IFS.  In this case, we want to return a list with a single word | 
					
						
							|  |  |  | 	 with the separator possibly replaced with a space (it's what other | 
					
						
							|  |  |  | 	 shells seem to do). | 
					
						
							|  |  |  | 	 quoted_dollar_at is internal to this function and is set if we are | 
					
						
							|  |  |  | 	 passed an argument that is unquoted (quoted == 0) but we encounter a | 
					
						
							|  |  |  | 	 double-quoted $@ while expanding it. */ | 
					
						
							|  |  |  |       else if (has_dollar_at && quoted_dollar_at == 0 && ifs_chars && quoted == 0 && (word->flags & W_NOSPLIT2)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* Only split and rejoin if we have to */ | 
					
						
							|  |  |  | 	  if (*ifs_chars && *ifs_chars != ' ') | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      list = list_string (istring, *ifs_chars ? ifs_chars : " ", 1); | 
					
						
							|  |  |  | 	      tstring = string_list (list); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    tstring = istring; | 
					
						
							|  |  |  | 	  tword = make_bare_word (tstring); | 
					
						
							|  |  |  | 	  if (tstring != istring) | 
					
						
							|  |  |  | 	    free (tstring); | 
					
						
							|  |  |  | 	  goto set_word_flags; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |       /* This is the attempt to make $* in an assignment context (a=$*) and
 | 
					
						
							|  |  |  | 	 array variables subscripted with * in an assignment context (a=${foo[*]}) | 
					
						
							|  |  |  | 	 behave similarly.  It has side effects that, though they increase | 
					
						
							|  |  |  | 	 compatibility with other shells, are not backwards compatible. */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  |       else if (has_dollar_at && quoted == 0 && ifs_chars && (word->flags & W_ASSIGNRHS)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tword = make_bare_word (istring); | 
					
						
							|  |  |  | 	  goto set_word_flags; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       else if (has_dollar_at && ifs_chars) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	list = list_string (istring, *ifs_chars ? ifs_chars : " ", 1); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  tword = make_bare_word (istring); | 
					
						
							| 
									
										
										
										
											2014-05-16 14:17:56 -04:00
										 |  |  | set_word_flags: | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if ((quoted & (Q_DOUBLE_QUOTES|Q_HERE_DOCUMENT)) || (quoted_state == WHOLLY_QUOTED)) | 
					
						
							|  |  |  | 	    tword->flags |= W_QUOTED; | 
					
						
							|  |  |  | 	  if (word->flags & W_ASSIGNMENT) | 
					
						
							|  |  |  | 	    tword->flags |= W_ASSIGNMENT; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if (word->flags & W_COMPASSIGN) | 
					
						
							|  |  |  | 	    tword->flags |= W_COMPASSIGN; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	  if (word->flags & W_NOGLOB) | 
					
						
							|  |  |  | 	    tword->flags |= W_NOGLOB; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  if (word->flags & W_NOBRACE) | 
					
						
							|  |  |  | 	    tword->flags |= W_NOBRACE; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if (word->flags & W_NOEXPAND) | 
					
						
							|  |  |  | 	    tword->flags |= W_NOEXPAND; | 
					
						
							| 
									
										
										
										
											2012-07-10 09:44:07 -04:00
										 |  |  | 	  if (had_quoted_null && QUOTED_NULL (istring)) | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	    tword->flags |= W_HASQUOTEDNULL;	/* XXX */ | 
					
						
							|  |  |  | 	  list = make_word_list (tword, (WORD_LIST *)NULL); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   free (istring); | 
					
						
							|  |  |  |   return (list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /*		   Functions for Quote Removal			    */ | 
					
						
							|  |  |  | /*								    */ | 
					
						
							|  |  |  | /* **************************************************************** */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform quote removal on STRING.  If QUOTED > 0, assume we are obeying the
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |    backslash quoting rules for within double quotes or a here document. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | char * | 
					
						
							|  |  |  | string_quote_removal (string, quoted) | 
					
						
							|  |  |  |      char *string; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   size_t slen; | 
					
						
							|  |  |  |   char *r, *result_string, *temp, *send; | 
					
						
							| 
									
										
										
										
											2001-11-13 17:56:06 +00:00
										 |  |  |   int sindex, tindex, dquote; | 
					
						
							|  |  |  |   unsigned char c; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   DECLARE_MBSTATE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* The result can be no longer than the original string. */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   slen = strlen (string); | 
					
						
							|  |  |  |   send = string + slen; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   r = result_string = (char *)xmalloc (slen + 1); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (dquote = sindex = 0; c = string[sindex];) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       switch (c) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case '\\': | 
					
						
							|  |  |  | 	  c = string[++sindex]; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  if (c == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      *r++ = '\\'; | 
					
						
							|  |  |  | 	      break; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  if (((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) || dquote) && (sh_syntaxtab[c] & CBSDQUOTE) == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    *r++ = '\\'; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* FALLTHROUGH */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	default: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	  SCOPY_CHAR_M (r, string, send, sindex); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '\'': | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if ((quoted & (Q_HERE_DOCUMENT|Q_DOUBLE_QUOTES)) || dquote) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      *r++ = c; | 
					
						
							|  |  |  | 	      sindex++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      break; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  tindex = sindex + 1; | 
					
						
							|  |  |  | 	  temp = string_extract_single_quoted (string, &tindex); | 
					
						
							|  |  |  | 	  if (temp) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	      strcpy (r, temp); | 
					
						
							|  |  |  | 	      r += strlen (r); | 
					
						
							|  |  |  | 	      free (temp); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  sindex = tindex; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	case '"': | 
					
						
							|  |  |  | 	  dquote = 1 - dquote; | 
					
						
							|  |  |  | 	  sindex++; | 
					
						
							|  |  |  | 	  break; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     *r = '\0'; | 
					
						
							|  |  |  |     return (result_string); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #if 0
 | 
					
						
							|  |  |  | /* UNUSED */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Perform quote removal on word WORD.  This allocates and returns a new
 | 
					
						
							|  |  |  |    WORD_DESC *. */ | 
					
						
							|  |  |  | WORD_DESC * | 
					
						
							|  |  |  | word_quote_removal (word, quoted) | 
					
						
							|  |  |  |      WORD_DESC *word; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   char *t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   t = string_quote_removal (word->word, quoted); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   w = alloc_word_desc (); | 
					
						
							|  |  |  |   w->word = t ? t : savestring (""); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (w); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform quote removal on all words in LIST.  If QUOTED is non-zero,
 | 
					
						
							|  |  |  |    the members of the list are treated as if they are surrounded by | 
					
						
							|  |  |  |    double quotes.  Return a new list, or NULL if LIST is NULL. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | word_list_quote_removal (list, quoted) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      int quoted; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_LIST *result, *t, *tresult, *e; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (t = list, result = (WORD_LIST *)NULL; t; t = t->next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       tresult = make_word_list (word_quote_removal (t->word, quoted), (WORD_LIST *)NULL); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if 0
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       result = (WORD_LIST *) list_append (result, tresult); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #else
 | 
					
						
							|  |  |  |       if (result == 0) | 
					
						
							|  |  |  | 	result = e = tresult; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  e->next = tresult; | 
					
						
							|  |  |  | 	  while (e->next) | 
					
						
							|  |  |  | 	    e = e->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /*******************************************
 | 
					
						
							|  |  |  |  *					   * | 
					
						
							|  |  |  |  *    Functions to perform word splitting  * | 
					
						
							|  |  |  |  *					   * | 
					
						
							|  |  |  |  *******************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | void | 
					
						
							|  |  |  | setifs (v) | 
					
						
							|  |  |  |      SHELL_VAR *v; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   char *t; | 
					
						
							|  |  |  |   unsigned char uc; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ifs_var = v; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   ifs_value = (v && value_cell (v)) ? value_cell (v) : " \t\n"; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   ifs_is_set = ifs_var != 0; | 
					
						
							|  |  |  |   ifs_is_null = ifs_is_set && (*ifs_value == 0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   /* Should really merge ifs_cmap with sh_syntaxtab.  XXX - doesn't yet
 | 
					
						
							|  |  |  |      handle multibyte chars in IFS */ | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   memset (ifs_cmap, '\0', sizeof (ifs_cmap)); | 
					
						
							|  |  |  |   for (t = ifs_value ; t && *t; t++) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       uc = *t; | 
					
						
							|  |  |  |       ifs_cmap[uc] = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (HANDLE_MULTIBYTE)
 | 
					
						
							|  |  |  |   if (ifs_value == 0) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ifs_firstc[0] = '\0'; | 
					
						
							|  |  |  |       ifs_firstc_len = 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       size_t ifs_len; | 
					
						
							|  |  |  |       ifs_len = strnlen (ifs_value, MB_CUR_MAX); | 
					
						
							|  |  |  |       ifs_firstc_len = MBLEN (ifs_value, ifs_len); | 
					
						
							|  |  |  |       if (ifs_firstc_len == 1 || ifs_firstc_len == 0 || MB_INVALIDCH (ifs_firstc_len)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  ifs_firstc[0] = ifs_value[0]; | 
					
						
							|  |  |  | 	  ifs_firstc[1] = '\0'; | 
					
						
							|  |  |  | 	  ifs_firstc_len = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	memcpy (ifs_firstc, ifs_value, ifs_firstc_len); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |   ifs_firstc = ifs_value ? *ifs_value : 0; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | char * | 
					
						
							|  |  |  | getifs () | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return ifs_value; | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* This splits a single word into a WORD LIST on $IFS, but only if the word
 | 
					
						
							|  |  |  |    is not quoted.  list_string () performs quote removal for us, even if we | 
					
						
							|  |  |  |    don't do any splitting. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | word_split (w, ifs_chars) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      WORD_DESC *w; | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |      char *ifs_chars; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *result; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (w) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       char *xifs; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       xifs = ((w->flags & W_QUOTED) || ifs_chars == 0) ? "" : ifs_chars; | 
					
						
							|  |  |  |       result = list_string (w->word, xifs, w->flags & W_QUOTED); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     result = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Perform word splitting on LIST and return the RESULT.  It is possible
 | 
					
						
							|  |  |  |    to return (WORD_LIST *)NULL. */ | 
					
						
							|  |  |  | static WORD_LIST * | 
					
						
							|  |  |  | word_list_split (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |   WORD_LIST *result, *t, *tresult, *e; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   for (t = list, result = (WORD_LIST *)NULL; t; t = t->next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       tresult = word_split (t->word, ifs_value); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       if (result == 0) | 
					
						
							|  |  |  |         result = e = tresult; | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  e->next = tresult; | 
					
						
							|  |  |  | 	  while (e->next) | 
					
						
							|  |  |  | 	    e = e->next; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   return (result); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**************************************************
 | 
					
						
							|  |  |  |  * 						  * | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |  *    Functions to expand an entire WORD_LIST	  * | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |  *						  * | 
					
						
							|  |  |  |  **************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | /* Do any word-expansion-specific cleanup and jump to top_level */ | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | exp_jump_to_top_level (v) | 
					
						
							|  |  |  |      int v; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   set_pipestatus_from_exit (last_command_exit_value); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   /* Cleanup code goes here. */ | 
					
						
							|  |  |  |   expand_no_split_dollar_star = 0;	/* XXX */ | 
					
						
							|  |  |  |   expanding_redir = 0; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   assigning_in_environment = 0; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  |   if (parse_and_execute_level == 0) | 
					
						
							|  |  |  |     top_level_cleanup ();			/* from sig.c */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |   jump_to_top_level (v); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* Put NLIST (which is a WORD_LIST * of only one element) at the front of
 | 
					
						
							|  |  |  |    ELIST, and set ELIST to the new list. */ | 
					
						
							|  |  |  | #define PREPEND_LIST(nlist, elist) \
 | 
					
						
							|  |  |  | 	do { nlist->next = elist; elist = nlist; } while (0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Separate out any initial variable assignments from TLIST.  If set -k has
 | 
					
						
							|  |  |  |    been executed, remove all assignment statements from TLIST.  Initial | 
					
						
							|  |  |  |    variable assignments and other environment assignments are placed | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |    on SUBST_ASSIGN_VARLIST. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | separate_out_assignments (tlist) | 
					
						
							|  |  |  |      WORD_LIST *tlist; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register WORD_LIST *vp, *lp; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |   if (tlist == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if (subst_assign_varlist) | 
					
						
							|  |  |  |     dispose_words (subst_assign_varlist);	/* Clean up after previous error */ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   subst_assign_varlist = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   vp = lp = tlist; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Separate out variable assignments at the start of the command.
 | 
					
						
							|  |  |  |      Loop invariant: vp->next == lp | 
					
						
							|  |  |  |      Loop postcondition: | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	lp = list of words left after assignment statements skipped | 
					
						
							|  |  |  | 	tlist = original list of words | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   while (lp && (lp->word->flags & W_ASSIGNMENT)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							|  |  |  |       vp = lp; | 
					
						
							|  |  |  |       lp = lp->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   /* If lp != tlist, we have some initial assignment statements.
 | 
					
						
							|  |  |  |      We make SUBST_ASSIGN_VARLIST point to the list of assignment | 
					
						
							|  |  |  |      words and TLIST point to the remaining words.  */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (lp != tlist) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       subst_assign_varlist = tlist; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       /* ASSERT(vp->next == lp); */ | 
					
						
							|  |  |  |       vp->next = (WORD_LIST *)NULL;	/* terminate variable list */ | 
					
						
							|  |  |  |       tlist = lp;			/* remainder of word list */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* vp == end of variable list */ | 
					
						
							|  |  |  |   /* tlist == remainder of original word list without variable assignments */ | 
					
						
							|  |  |  |   if (!tlist) | 
					
						
							|  |  |  |     /* All the words in tlist were assignment statements */ | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* ASSERT(tlist != NULL); */ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |   /* ASSERT((tlist->word->flags & W_ASSIGNMENT) == 0); */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* If the -k option is in effect, we need to go through the remaining
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |      words, separate out the assignment words, and place them on | 
					
						
							|  |  |  |      SUBST_ASSIGN_VARLIST. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |   if (place_keywords_in_env) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       WORD_LIST *tp;	/* tp == running pointer into tlist */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       tp = tlist; | 
					
						
							|  |  |  |       lp = tlist->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Loop Invariant: tp->next == lp */ | 
					
						
							|  |  |  |       /* Loop postcondition: tlist == word list without assignment statements */ | 
					
						
							|  |  |  |       while (lp) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  if (lp->word->flags & W_ASSIGNMENT) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Found an assignment statement, add this word to end of
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 		 subst_assign_varlist (vp). */ | 
					
						
							|  |  |  | 	      if (!subst_assign_varlist) | 
					
						
							|  |  |  | 		subst_assign_varlist = vp = lp; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	      else | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  vp->next = lp; | 
					
						
							|  |  |  | 		  vp = lp; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      /* Remove the word pointed to by LP from TLIST. */ | 
					
						
							|  |  |  | 	      tp->next = lp->next; | 
					
						
							|  |  |  | 	      /* ASSERT(vp == lp); */ | 
					
						
							|  |  |  | 	      lp->next = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 	      lp = tp->next; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      tp = lp; | 
					
						
							|  |  |  | 	      lp = lp->next; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return (tlist); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #define WEXP_VARASSIGN	0x001
 | 
					
						
							|  |  |  | #define WEXP_BRACEEXP	0x002
 | 
					
						
							|  |  |  | #define WEXP_TILDEEXP	0x004
 | 
					
						
							|  |  |  | #define WEXP_PARAMEXP	0x008
 | 
					
						
							|  |  |  | #define WEXP_PATHEXP	0x010
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* All of the expansions, including variable assignments at the start of
 | 
					
						
							|  |  |  |    the list. */ | 
					
						
							|  |  |  | #define WEXP_ALL	(WEXP_VARASSIGN|WEXP_BRACEEXP|WEXP_TILDEEXP|WEXP_PARAMEXP|WEXP_PATHEXP)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* All of the expansions except variable assignments at the start of
 | 
					
						
							|  |  |  |    the list. */ | 
					
						
							|  |  |  | #define WEXP_NOVARS	(WEXP_BRACEEXP|WEXP_TILDEEXP|WEXP_PARAMEXP|WEXP_PATHEXP)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* All of the `shell expansions': brace expansion, tilde expansion, parameter
 | 
					
						
							|  |  |  |    expansion, command substitution, arithmetic expansion, word splitting, and | 
					
						
							|  |  |  |    quote removal. */ | 
					
						
							|  |  |  | #define WEXP_SHELLEXP	(WEXP_BRACEEXP|WEXP_TILDEEXP|WEXP_PARAMEXP)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | /* Take the list of words in LIST and do the various substitutions.  Return
 | 
					
						
							|  |  |  |    a new list of words which is the expanded list, and without things like | 
					
						
							|  |  |  |    variable assignments. */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_words (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (expand_word_list_internal (list, WEXP_ALL)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Same as expand_words (), but doesn't hack variable or environment
 | 
					
						
							|  |  |  |    variables. */ | 
					
						
							|  |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_words_no_vars (list) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (expand_word_list_internal (list, WEXP_NOVARS)); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | WORD_LIST * | 
					
						
							|  |  |  | expand_words_shellexp (list) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (expand_word_list_internal (list, WEXP_SHELLEXP)); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | glob_expand_word_list (tlist, eflags) | 
					
						
							|  |  |  |      WORD_LIST *tlist; | 
					
						
							|  |  |  |      int eflags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   char **glob_array, *temp_string; | 
					
						
							|  |  |  |   register int glob_index; | 
					
						
							|  |  |  |   WORD_LIST *glob_list, *output_list, *disposables, *next; | 
					
						
							|  |  |  |   WORD_DESC *tword; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   output_list = disposables = (WORD_LIST *)NULL; | 
					
						
							|  |  |  |   glob_array = (char **)NULL; | 
					
						
							|  |  |  |   while (tlist) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* For each word, either globbing is attempted or the word is
 | 
					
						
							|  |  |  | 	 added to orig_list.  If globbing succeeds, the results are | 
					
						
							|  |  |  | 	 added to orig_list and the word (tlist) is added to the list | 
					
						
							|  |  |  | 	 of disposable words.  If globbing fails and failed glob | 
					
						
							|  |  |  | 	 expansions are left unchanged (the shell default), the | 
					
						
							|  |  |  | 	 original word is added to orig_list.  If globbing fails and | 
					
						
							|  |  |  | 	 failed glob expansions are removed, the original word is | 
					
						
							|  |  |  | 	 added to the list of disposable words.  orig_list ends up | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	 in reverse order and requires a call to REVERSE_LIST to | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 be set right.  After all words are examined, the disposable | 
					
						
							|  |  |  | 	 words are freed. */ | 
					
						
							|  |  |  |       next = tlist->next; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* If the word isn't an assignment and contains an unquoted
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	 pattern matching character, then glob it. */ | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |       if ((tlist->word->flags & W_NOGLOB) == 0 && | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  unquoted_glob_pattern_p (tlist->word->word)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  glob_array = shell_glob_filename (tlist->word->word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Handle error cases.
 | 
					
						
							|  |  |  | 	     I don't think we should report errors like "No such file | 
					
						
							|  |  |  | 	     or directory".  However, I would like to report errors | 
					
						
							|  |  |  | 	     like "Read failed". */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (glob_array == 0 || GLOB_FAILED (glob_array)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      glob_array = (char **)xmalloc (sizeof (char *)); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      glob_array[0] = (char *)NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Dequote the current word in case we have to use it. */ | 
					
						
							|  |  |  | 	  if (glob_array[0] == NULL) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      temp_string = dequote_string (tlist->word->word); | 
					
						
							|  |  |  | 	      free (tlist->word->word); | 
					
						
							|  |  |  | 	      tlist->word->word = temp_string; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Make the array into a word list. */ | 
					
						
							|  |  |  | 	  glob_list = (WORD_LIST *)NULL; | 
					
						
							|  |  |  | 	  for (glob_index = 0; glob_array[glob_index]; glob_index++) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      tword = make_bare_word (glob_array[glob_index]); | 
					
						
							|  |  |  | 	      glob_list = make_word_list (tword, glob_list); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if (glob_list) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      output_list = (WORD_LIST *)list_append (glob_list, output_list); | 
					
						
							|  |  |  | 	      PREPEND_LIST (tlist, disposables); | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  else if (fail_glob_expansion != 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	      report_error (_("no match: %s"), tlist->word->word); | 
					
						
							| 
									
										
										
										
											2008-11-18 13:15:12 +00:00
										 |  |  | 	      exp_jump_to_top_level (DISCARD); | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  else if (allow_null_glob_expansion == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Failed glob expressions are left unchanged. */ | 
					
						
							|  |  |  | 	      PREPEND_LIST (tlist, output_list); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Failed glob expressions are removed. */ | 
					
						
							|  |  |  | 	      PREPEND_LIST (tlist, disposables); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* Dequote the string. */ | 
					
						
							|  |  |  | 	  temp_string = dequote_string (tlist->word->word); | 
					
						
							|  |  |  | 	  free (tlist->word->word); | 
					
						
							|  |  |  | 	  tlist->word->word = temp_string; | 
					
						
							|  |  |  | 	  PREPEND_LIST (tlist, output_list); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  |       strvec_dispose (glob_array); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       glob_array = (char **)NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       tlist = next; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (disposables) | 
					
						
							|  |  |  |     dispose_words (disposables); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (output_list) | 
					
						
							|  |  |  |     output_list = REVERSE_LIST (output_list, WORD_LIST *); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (output_list); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #if defined (BRACE_EXPANSION)
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | brace_expand_word_list (tlist, eflags) | 
					
						
							|  |  |  |      WORD_LIST *tlist; | 
					
						
							|  |  |  |      int eflags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register char **expansions; | 
					
						
							|  |  |  |   char *temp_string; | 
					
						
							|  |  |  |   WORD_LIST *disposables, *output_list, *next; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  |   int eindex; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (disposables = output_list = (WORD_LIST *)NULL; tlist; tlist = next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       next = tlist->next; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |       if (tlist->word->flags & W_NOBRACE) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  | /*itrace("brace_expand_word_list: %s: W_NOBRACE", tlist->word->word);*/ | 
					
						
							|  |  |  | 	  PREPEND_LIST (tlist, output_list); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if ((tlist->word->flags & (W_COMPASSIGN|W_ASSIGNARG)) == (W_COMPASSIGN|W_ASSIGNARG)) | 
					
						
							|  |  |  |         { | 
					
						
							|  |  |  | /*itrace("brace_expand_word_list: %s: W_COMPASSIGN|W_ASSIGNARG", tlist->word->word);*/ | 
					
						
							|  |  |  | 	  PREPEND_LIST (tlist, output_list); | 
					
						
							|  |  |  | 	  continue; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       /* Only do brace expansion if the word has a brace character.  If
 | 
					
						
							|  |  |  | 	 not, just add the word list element to BRACES and continue.  In | 
					
						
							|  |  |  | 	 the common case, at least when running shell scripts, this will | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  | 	 degenerate to a bunch of calls to `mbschr', and then what is | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 basically a reversal of TLIST into BRACES, which is corrected | 
					
						
							| 
									
										
										
										
											2002-07-17 14:10:11 +00:00
										 |  |  | 	 by a call to REVERSE_LIST () on BRACES when the end of TLIST | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	 is reached. */ | 
					
						
							| 
									
										
										
										
											2011-11-21 20:51:19 -05:00
										 |  |  |       if (mbschr (tlist->word->word, LBRACE)) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  expansions = brace_expand (tlist->word->word); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  for (eindex = 0; temp_string = expansions[eindex]; eindex++) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      w = alloc_word_desc (); | 
					
						
							|  |  |  | 	      w->word = temp_string; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      /* If brace expansion didn't change the word, preserve
 | 
					
						
							|  |  |  | 		 the flags.  We may want to preserve the flags | 
					
						
							|  |  |  | 		 unconditionally someday -- XXX */ | 
					
						
							|  |  |  | 	      if (STREQ (temp_string, tlist->word->word)) | 
					
						
							|  |  |  | 		w->flags = tlist->word->flags; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	      else | 
					
						
							|  |  |  | 		w = make_word_flags (w, temp_string); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	      output_list = make_word_list (w, output_list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  free (expansions); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	  /* Add TLIST to the list of words to be freed after brace
 | 
					
						
							|  |  |  | 	     expansion has been performed. */ | 
					
						
							|  |  |  | 	  PREPEND_LIST (tlist, disposables); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	PREPEND_LIST (tlist, output_list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (disposables) | 
					
						
							|  |  |  |     dispose_words (disposables); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (output_list) | 
					
						
							|  |  |  |     output_list = REVERSE_LIST (output_list, WORD_LIST *); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return (output_list); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  | /* Take WORD, a compound associative array assignment, and internally run
 | 
					
						
							|  |  |  |    'declare -A w', where W is the variable name portion of WORD. */ | 
					
						
							|  |  |  | static int | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | make_internal_declare (word, option, cmd) | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |      char *word; | 
					
						
							|  |  |  |      char *option; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |      char *cmd; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   int t, r; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  |   WORD_LIST *wl; | 
					
						
							|  |  |  |   WORD_DESC *w; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   w = make_word (word); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   t = assignment (w->word, 0); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   if (w->word[t] == '=') | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       w->word[t] = '\0'; | 
					
						
							|  |  |  |       if (w->word[t - 1] == '+')	/* cut off any append op */ | 
					
						
							|  |  |  | 	w->word[t - 1] = '\0'; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   wl = make_word_list (w, (WORD_LIST *)NULL); | 
					
						
							|  |  |  |   wl = make_word_list (make_word (option), wl); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   r = declare_builtin (wl); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dispose_words (wl); | 
					
						
							|  |  |  |   return r; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | }   | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | shell_expand_word_list (tlist, eflags) | 
					
						
							|  |  |  |      WORD_LIST *tlist; | 
					
						
							|  |  |  |      int eflags; | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   WORD_LIST *expanded, *orig_list, *new_list, *next, *temp_list, *wcmd; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   int expanded_something, has_dollar_at; | 
					
						
							|  |  |  |   char *temp_string; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* We do tilde expansion all the time.  This is what 1003.2 says. */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   new_list = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   for (wcmd = tlist; wcmd; wcmd = wcmd->next) | 
					
						
							|  |  |  |     if (wcmd->word->flags & W_ASSNBLTIN) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   for (orig_list = tlist; tlist; tlist = next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       temp_string = tlist->word->word; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       next = tlist->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #if defined (ARRAY_VARS)
 | 
					
						
							|  |  |  |       /* If this is a compound array assignment to a builtin that accepts
 | 
					
						
							|  |  |  |          such assignments (e.g., `declare'), take the assignment and perform | 
					
						
							|  |  |  |          it separately, handling the semantics of declarations inside shell | 
					
						
							|  |  |  |          functions.  This avoids the double-evaluation of such arguments, | 
					
						
							|  |  |  |          because `declare' does some evaluation of compound assignments on | 
					
						
							|  |  |  |          its own. */ | 
					
						
							|  |  |  |       if ((tlist->word->flags & (W_COMPASSIGN|W_ASSIGNARG)) == (W_COMPASSIGN|W_ASSIGNARG)) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  int t; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  char opts[16], opti; | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  opti = 0; | 
					
						
							|  |  |  | 	  if (tlist->word->flags & (W_ASSIGNASSOC|W_ASSNGLOBAL|W_ASSIGNARRAY)) | 
					
						
							|  |  |  | 	    opts[opti++] = '-'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  if ((tlist->word->flags & (W_ASSIGNASSOC|W_ASSNGLOBAL)) == (W_ASSIGNASSOC|W_ASSNGLOBAL)) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      opts[opti++] = 'g'; | 
					
						
							|  |  |  | 	      opts[opti++] = 'A'; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (tlist->word->flags & W_ASSIGNASSOC) | 
					
						
							|  |  |  | 	    opts[opti++] = 'A'; | 
					
						
							|  |  |  | 	  else if ((tlist->word->flags & (W_ASSIGNARRAY|W_ASSNGLOBAL)) == (W_ASSIGNARRAY|W_ASSNGLOBAL)) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      opts[opti++] = 'g'; | 
					
						
							|  |  |  | 	      opts[opti++] = 'a'; | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 	  else if (tlist->word->flags & W_ASSIGNARRAY) | 
					
						
							|  |  |  | 	    opts[opti++] = 'a'; | 
					
						
							|  |  |  | 	  else if (tlist->word->flags & W_ASSNGLOBAL) | 
					
						
							|  |  |  | 	    opts[opti++] = 'g'; | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  /* If we have special handling note the integer attribute and others
 | 
					
						
							|  |  |  | 	     that transform the value upon assignment.  What we do is take all | 
					
						
							|  |  |  | 	     of the option arguments and scan through them looking for options | 
					
						
							|  |  |  | 	     that cause such transformations, and add them to the `opts' array. */ | 
					
						
							|  |  |  | /*	  if (opti > 0) */ | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      char omap[128]; | 
					
						
							|  |  |  | 	      int oind; | 
					
						
							|  |  |  | 	      WORD_LIST *l; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      memset (omap, '\0', sizeof (omap)); | 
					
						
							|  |  |  | 	      for (l = orig_list->next; l != tlist; l = l->next) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  if (l->word->word[0] != '-') | 
					
						
							|  |  |  | 		    break;	/* non-option argument */ | 
					
						
							|  |  |  | 		  if (l->word->word[0] == '-' && l->word->word[1] == '-' && l->word->word[2] == 0) | 
					
						
							|  |  |  | 		    break;	/* -- signals end of options */ | 
					
						
							|  |  |  | 		  for (oind = 1; l->word->word[oind]; oind++) | 
					
						
							|  |  |  | 		    switch (l->word->word[oind]) | 
					
						
							|  |  |  | 		      { | 
					
						
							|  |  |  | 			case 'i': | 
					
						
							|  |  |  | 			case 'l': | 
					
						
							|  |  |  | 			case 'u': | 
					
						
							|  |  |  | 			case 'c': | 
					
						
							|  |  |  | 			  omap[l->word->word[oind]] = 1; | 
					
						
							|  |  |  | 			  if (opti == 0) | 
					
						
							|  |  |  | 			    opts[opti++] = '-'; | 
					
						
							|  |  |  | 			  break; | 
					
						
							|  |  |  | 			default: | 
					
						
							|  |  |  | 			  break; | 
					
						
							|  |  |  | 		      } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      for (oind = 0; oind < sizeof (omap); oind++) | 
					
						
							|  |  |  | 		if (omap[oind]) | 
					
						
							|  |  |  | 		  opts[opti++] = oind; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	  opts[opti] = '\0'; | 
					
						
							|  |  |  | 	  if (opti > 0) | 
					
						
							| 
									
										
										
										
											2016-06-20 15:14:49 -04:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	      t = make_internal_declare (tlist->word->word, opts, wcmd ? wcmd->word->word : (char *)0); | 
					
						
							| 
									
										
										
										
											2016-06-20 15:14:49 -04:00
										 |  |  | 	      if (t != EXECUTION_SUCCESS) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 		  last_command_exit_value = t; | 
					
						
							|  |  |  | 		  exp_jump_to_top_level (DISCARD); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  t = do_word_assignment (tlist->word, 0); | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	  if (t == 0) | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							|  |  |  | 	      exp_jump_to_top_level (DISCARD); | 
					
						
							|  |  |  | 	    } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	  /* Now transform the word as ksh93 appears to do and go on */ | 
					
						
							|  |  |  | 	  t = assignment (tlist->word->word, 0); | 
					
						
							|  |  |  | 	  tlist->word->word[t] = '\0'; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  if (tlist->word->word[t - 1] == '+') | 
					
						
							|  |  |  | 	    tlist->word->word[t - 1] = '\0';	/* cut off append op */ | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  | 	  tlist->word->flags &= ~(W_ASSIGNMENT|W_NOSPLIT|W_COMPASSIGN|W_ASSIGNARG|W_ASSIGNASSOC|W_ASSIGNARRAY); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       expanded_something = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       expanded = expand_word_internal | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  | 	(tlist->word, 0, 0, &has_dollar_at, &expanded_something); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (expanded == &expand_word_error || expanded == &expand_word_fatal) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* By convention, each time this error is returned,
 | 
					
						
							|  |  |  | 	     tlist->word->word has already been freed. */ | 
					
						
							|  |  |  | 	  tlist->word->word = (char *)NULL; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  /* Dispose our copy of the original list. */ | 
					
						
							|  |  |  | 	  dispose_words (orig_list); | 
					
						
							| 
									
										
										
										
											1997-06-05 14:59:13 +00:00
										 |  |  | 	  /* Dispose the new list we're building. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  dispose_words (new_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  if (expanded == &expand_word_error) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    exp_jump_to_top_level (DISCARD); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	    exp_jump_to_top_level (FORCE_EOF); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       /* Don't split words marked W_NOSPLIT. */ | 
					
						
							|  |  |  |       if (expanded_something && (tlist->word->flags & W_NOSPLIT) == 0) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  temp_list = word_list_split (expanded); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	  dispose_words (expanded); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  /* If no parameter expansion, command substitution, process
 | 
					
						
							|  |  |  | 	     substitution, or arithmetic substitution took place, then | 
					
						
							|  |  |  | 	     do not do word splitting.  We still have to remove quoted | 
					
						
							|  |  |  | 	     null characters from the result. */ | 
					
						
							|  |  |  | 	  word_list_remove_quoted_nulls (expanded); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  temp_list = expanded; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  |       expanded = REVERSE_LIST (temp_list, WORD_LIST *); | 
					
						
							|  |  |  |       new_list = (WORD_LIST *)list_append (expanded, new_list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (orig_list)   | 
					
						
							|  |  |  |     dispose_words (orig_list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (new_list) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |     new_list = REVERSE_LIST (new_list, WORD_LIST *); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (new_list); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | /* The workhorse for expand_words () and expand_words_no_vars ().
 | 
					
						
							|  |  |  |    First arg is LIST, a WORD_LIST of words. | 
					
						
							| 
									
										
										
										
											1999-02-19 17:11:39 +00:00
										 |  |  |    Second arg EFLAGS is a flags word controlling which expansions are | 
					
						
							|  |  |  |    performed. | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |    This does all of the substitutions: brace expansion, tilde expansion, | 
					
						
							|  |  |  |    parameter expansion, command substitution, arithmetic expansion, | 
					
						
							|  |  |  |    process substitution, word splitting, and pathname expansion, according | 
					
						
							|  |  |  |    to the bits set in EFLAGS.  Words with the W_QUOTED or W_NOSPLIT bits | 
					
						
							|  |  |  |    set, or for which no expansion is done, do not undergo word splitting. | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |    Words with the W_NOGLOB bit set do not undergo pathname expansion; words | 
					
						
							|  |  |  |    with W_NOBRACE set do not undergo brace expansion (see | 
					
						
							|  |  |  |    brace_expand_word_list above). */ | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | static WORD_LIST * | 
					
						
							|  |  |  | expand_word_list_internal (list, eflags) | 
					
						
							|  |  |  |      WORD_LIST *list; | 
					
						
							|  |  |  |      int eflags; | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   WORD_LIST *new_list, *temp_list; | 
					
						
							|  |  |  |   int tint; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  |   char *savecmd; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-26 09:36:43 -05:00
										 |  |  |   tempenv_assign_error = 0; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (list == 0) | 
					
						
							|  |  |  |     return ((WORD_LIST *)NULL); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   garglist = new_list = copy_word_list (list); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   if (eflags & WEXP_VARASSIGN) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       garglist = new_list = separate_out_assignments (new_list); | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |       if (new_list == 0) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	  if (subst_assign_varlist) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      /* All the words were variable assignments, so they are placed
 | 
					
						
							|  |  |  | 		 into the shell's environment. */ | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      for (temp_list = subst_assign_varlist; temp_list; temp_list = temp_list->next) | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  savecmd = this_command_name; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		  this_command_name = (char *)NULL;	/* no arithmetic errors */ | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 		  tint = do_word_assignment (temp_list->word, 0); | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 		  this_command_name = savecmd; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		  /* Variable assignment errors in non-interactive shells
 | 
					
						
							|  |  |  | 		     running in Posix.2 mode cause the shell to exit. */ | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		  if (tint == 0) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		    { | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 		      last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		      if (interactive_shell == 0 && posixly_correct) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 			exp_jump_to_top_level (FORCE_EOF); | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 		      else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 			exp_jump_to_top_level (DISCARD); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 		    } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  | 	      dispose_words (subst_assign_varlist); | 
					
						
							|  |  |  | 	      subst_assign_varlist = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	    } | 
					
						
							|  |  |  | 	  return ((WORD_LIST *)NULL); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Begin expanding the words that remain.  The expansions take place on
 | 
					
						
							|  |  |  |      things that aren't really variable assignments. */ | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | #if defined (BRACE_EXPANSION)
 | 
					
						
							|  |  |  |   /* Do brace expansion on this word if there are any brace characters
 | 
					
						
							|  |  |  |      in the string. */ | 
					
						
							|  |  |  |   if ((eflags & WEXP_BRACEEXP) && brace_expansion && new_list) | 
					
						
							|  |  |  |     new_list = brace_expand_word_list (new_list, eflags); | 
					
						
							|  |  |  | #endif /* BRACE_EXPANSION */
 | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Perform the `normal' shell expansions: tilde expansion, parameter and
 | 
					
						
							|  |  |  |      variable substitution, command substitution, arithmetic expansion, | 
					
						
							|  |  |  |      and word splitting. */ | 
					
						
							|  |  |  |   new_list = shell_expand_word_list (new_list, eflags); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   /* Okay, we're almost done.  Now let's just do some filename
 | 
					
						
							|  |  |  |      globbing. */ | 
					
						
							|  |  |  |   if (new_list) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if ((eflags & WEXP_PATHEXP) && disallow_filename_globbing == 0) | 
					
						
							|  |  |  | 	/* Glob expand the word list unless globbing has been disabled. */ | 
					
						
							|  |  |  | 	new_list = glob_expand_word_list (new_list, eflags); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  | 	/* Dequote the words, because we're not performing globbing. */ | 
					
						
							|  |  |  | 	new_list = dequote_list (new_list); | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |   if ((eflags & WEXP_VARASSIGN) && subst_assign_varlist) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       sh_wassign_func_t *assign_func; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       int is_special_builtin, is_builtin_or_func; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* If the remainder of the words expand to nothing, Posix.2 requires
 | 
					
						
							|  |  |  | 	 that the variable and environment assignments affect the shell's | 
					
						
							|  |  |  | 	 environment. */ | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  |       assign_func = new_list ? assign_in_env : do_word_assignment; | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  |       tempenv_assign_error = 0; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  |       is_builtin_or_func = (new_list && new_list->word && (find_shell_builtin (new_list->word->word) || find_function (new_list->word->word))); | 
					
						
							|  |  |  |       /* Posix says that special builtins exit if a variable assignment error
 | 
					
						
							|  |  |  | 	 occurs in an assignment preceding it. */ | 
					
						
							|  |  |  |       is_special_builtin = (posixly_correct && new_list && new_list->word && find_special_builtin (new_list->word->word)); | 
					
						
							|  |  |  |        | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       for (temp_list = subst_assign_varlist; temp_list; temp_list = temp_list->next) | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  savecmd = this_command_name; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  this_command_name = (char *)NULL; | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  assigning_in_environment = (assign_func == assign_in_env); | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 	  tint = (*assign_func) (temp_list->word, is_builtin_or_func); | 
					
						
							| 
									
										
										
										
											2009-01-12 13:36:28 +00:00
										 |  |  | 	  assigning_in_environment = 0; | 
					
						
							| 
									
										
										
										
											2016-09-15 16:59:08 -04:00
										 |  |  | 	  this_command_name = savecmd; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	  /* Variable assignment errors in non-interactive shells running
 | 
					
						
							|  |  |  | 	     in Posix.2 mode cause the shell to exit. */ | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 	  if (tint == 0) | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2005-12-07 14:08:12 +00:00
										 |  |  | 	      if (assign_func == do_word_assignment) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		{ | 
					
						
							|  |  |  | 		  last_command_exit_value = EXECUTION_FAILURE; | 
					
						
							| 
									
										
										
										
											2011-11-22 19:11:26 -05:00
										 |  |  | 		  if (interactive_shell == 0 && posixly_correct && is_special_builtin) | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		    exp_jump_to_top_level (FORCE_EOF); | 
					
						
							|  |  |  | 		  else | 
					
						
							|  |  |  | 		    exp_jump_to_top_level (DISCARD); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2001-04-06 19:14:31 +00:00
										 |  |  | 	      else | 
					
						
							| 
									
										
										
										
											2004-07-27 13:29:18 +00:00
										 |  |  | 		tempenv_assign_error++; | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | 	    } | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-17 21:46:59 +00:00
										 |  |  |       dispose_words (subst_assign_varlist); | 
					
						
							|  |  |  |       subst_assign_varlist = (WORD_LIST *)NULL; | 
					
						
							| 
									
										
										
										
											1996-08-26 18:22:31 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-04-17 19:52:44 +00:00
										 |  |  |   return (new_list); | 
					
						
							| 
									
										
										
										
											1996-12-23 17:02:34 +00:00
										 |  |  | } |