hush: fix bug 353 (wrong handling of \x in assignments)

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2009-05-28 16:49:11 +02:00
parent 5d7cca2090
commit e640cb4ad1
3 changed files with 31 additions and 22 deletions

View File

@ -2055,10 +2055,10 @@ static char *expand_pseudo_dquoted(const char *str)
* 'echo -$*-'. If you play here, you must run testsuite afterwards! */ * 'echo -$*-'. If you play here, you must run testsuite afterwards! */
static int expand_vars_to_list(o_string *output, int n, char *arg, char or_mask) static int expand_vars_to_list(o_string *output, int n, char *arg, char or_mask)
{ {
/* or_mask is either 0 (normal case) or 0x80 /* or_mask is either 0 (normal case) or 0x80 -
* (expansion of right-hand side of assignment == 1-element expand. * expansion of right-hand side of assignment == 1-element expand.
* It will also do no globbing, and thus we must not backslash-quote!) */ * It will also do no globbing, and thus we must not backslash-quote!
*/
char ored_ch; char ored_ch;
char *p; char *p;
@ -2412,6 +2412,7 @@ static char *expand_string_to_string(const char *str)
bb_error_msg_and_die("BUG in varexp2"); bb_error_msg_and_die("BUG in varexp2");
/* actually, just move string 2*sizeof(char*) bytes back */ /* actually, just move string 2*sizeof(char*) bytes back */
overlapping_strcpy((char*)list, list[0]); overlapping_strcpy((char*)list, list[0]);
unbackslash((char*)list);
debug_printf_expand("string_to_string='%s'\n", (char*)list); debug_printf_expand("string_to_string='%s'\n", (char*)list);
return (char*)list; return (char*)list;
} }
@ -3906,7 +3907,6 @@ static int run_list(struct pipe *pi)
#endif #endif
#if ENABLE_HUSH_LOOPS #if ENABLE_HUSH_LOOPS
struct pipe *loop_top = NULL; struct pipe *loop_top = NULL;
char *for_varname = NULL;
char **for_lcur = NULL; char **for_lcur = NULL;
char **for_list = NULL; char **for_list = NULL;
#endif #endif
@ -4042,22 +4042,18 @@ static int run_list(struct pipe *pi)
for_list = expand_strvec_to_strvec(vals); for_list = expand_strvec_to_strvec(vals);
for_lcur = for_list; for_lcur = for_list;
debug_print_strings("for_list", for_list); debug_print_strings("for_list", for_list);
for_varname = pi->cmds[0].argv[0];
pi->cmds[0].argv[0] = NULL;
} }
free(pi->cmds[0].argv[0]);
if (!*for_lcur) { if (!*for_lcur) {
/* "for" loop is over, clean up */ /* "for" loop is over, clean up */
free(for_list); free(for_list);
for_list = NULL; for_list = NULL;
for_lcur = NULL; for_lcur = NULL;
pi->cmds[0].argv[0] = for_varname;
break; break;
} }
/* Insert next value from for_lcur */ /* Insert next value from for_lcur */
/* note: *for_lcur already has quotes removed, $var expanded, etc */ /* note: *for_lcur already has quotes removed, $var expanded, etc */
pi->cmds[0].argv[0] = xasprintf("%s=%s", for_varname, *for_lcur++); set_local_var(xasprintf("%s=%s", pi->cmds[0].argv[0], *for_lcur++), 0, 0);
pi->cmds[0].assignment_cnt = 1; continue;
} }
if (rword == RES_IN) { if (rword == RES_IN) {
continue; /* "for v IN list;..." - "in" has no cmds anyway */ continue; /* "for v IN list;..." - "in" has no cmds anyway */
@ -4544,13 +4540,13 @@ static int done_word(o_string *word, struct parse_context *ctx)
* Same with heredocs: * Same with heredocs:
* for <<\H delim is H; <<\\H, <<"\H", <<"\\H" - \H * for <<\H delim is H; <<\\H, <<"\H", <<"\\H" - \H
*/ */
if (ctx->pending_redirect->rd_type == REDIRECT_HEREDOC) {
unbackslash(ctx->pending_redirect->rd_filename); unbackslash(ctx->pending_redirect->rd_filename);
/* Is it <<"HEREDOC"? */ /* Is it <<"HEREDOC"? */
if (ctx->pending_redirect->rd_type == REDIRECT_HEREDOC if (word->o_quoted) {
&& word->o_quoted
) {
ctx->pending_redirect->rd_dup |= HEREDOC_QUOTED; ctx->pending_redirect->rd_dup |= HEREDOC_QUOTED;
} }
}
debug_printf_parse("word stored in rd_filename: '%s'\n", word->data); debug_printf_parse("word stored in rd_filename: '%s'\n", word->data);
ctx->pending_redirect = NULL; ctx->pending_redirect = NULL;
} else { } else {
@ -5465,7 +5461,7 @@ static int parse_stream_dquoted(o_string *as_string,
* "The backslash retains its special meaning [in "..."] * "The backslash retains its special meaning [in "..."]
* only when followed by one of the following characters: * only when followed by one of the following characters:
* $, `, ", \, or <newline>. A double quote may be quoted * $, `, ", \, or <newline>. A double quote may be quoted
* within double quotes by preceding it with a backslash. * within double quotes by preceding it with a backslash."
*/ */
if (strchr("$`\"\\\n", next) != NULL) { if (strchr("$`\"\\\n", next) != NULL) {
ch = i_getch(input); ch = i_getch(input);
@ -5834,10 +5830,7 @@ static struct pipe *parse_stream(char **pstring,
nommu_addchr(&ctx.as_string, ch); nommu_addchr(&ctx.as_string, ch);
if (ch == '\'') if (ch == '\'')
break; break;
if (dest.o_assignment == NOT_ASSIGNMENT)
o_addqchr(&dest, ch); o_addqchr(&dest, ch);
else
o_addchr(&dest, ch);
} }
break; break;
case '"': case '"':

View File

@ -0,0 +1,9 @@
a\nb\nc\n
a
b
c
a\nb\nc\n
a
b
c
Done

View File

@ -0,0 +1,7 @@
v="a\nb\nc\n"
echo "$v"
printf "$v"
v='a\nb\nc\n'
echo "$v"
printf "$v"
echo Done