hush: fix source1.tests

function                                             old     new   delta
parse_and_run_stream                                  95     139     +44
static_get                                            22      25      +3
file_get                                             260     263      +3
builtin_umask                                        133     132      -1
parse_stream                                        2442    2425     -17
------------------------------------------------------------------------------
(add/remove: 0/0 grow/shrink: 3/2 up/down: 50/-18)             Total: 32 bytes

Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
This commit is contained in:
Denys Vlasenko 2011-03-30 18:54:52 +02:00
parent 0e89fd9c39
commit cecbc98611

View File

@ -445,8 +445,6 @@ enum {
/* Used for initialization: o_string foo = NULL_O_STRING; */ /* Used for initialization: o_string foo = NULL_O_STRING; */
#define NULL_O_STRING { NULL } #define NULL_O_STRING { NULL }
/* I can almost use ordinary FILE*. Is open_memstream() universally
* available? Where is it documented? */
typedef struct in_str { typedef struct in_str {
const char *p; const char *p;
/* eof_flag=1: last char in ->p is really an EOF */ /* eof_flag=1: last char in ->p is really an EOF */
@ -455,6 +453,7 @@ typedef struct in_str {
#if ENABLE_HUSH_INTERACTIVE #if ENABLE_HUSH_INTERACTIVE
smallint promptmode; /* 0: PS1, 1: PS2 */ smallint promptmode; /* 0: PS1, 1: PS2 */
#endif #endif
int last_char;
FILE *file; FILE *file;
int (*get) (struct in_str *) FAST_FUNC; int (*get) (struct in_str *) FAST_FUNC;
int (*peek) (struct in_str *) FAST_FUNC; int (*peek) (struct in_str *) FAST_FUNC;
@ -1077,22 +1076,22 @@ static void die_if_script(unsigned lineno, const char *fmt, ...)
xfunc_die(); xfunc_die();
} }
static void syntax_error(unsigned lineno, const char *msg) static void syntax_error(unsigned lineno UNUSED_PARAM, const char *msg)
{ {
if (msg) if (msg)
die_if_script(lineno, "syntax error: %s", msg); bb_error_msg("syntax error: %s", msg);
else else
die_if_script(lineno, "syntax error", NULL); bb_error_msg("syntax error");
} }
static void syntax_error_at(unsigned lineno, const char *msg) static void syntax_error_at(unsigned lineno UNUSED_PARAM, const char *msg)
{ {
die_if_script(lineno, "syntax error at '%s'", msg); bb_error_msg("syntax error at '%s'", msg);
} }
static void syntax_error_unterm_str(unsigned lineno, const char *s) static void syntax_error_unterm_str(unsigned lineno UNUSED_PARAM, const char *s)
{ {
die_if_script(lineno, "syntax error: unterminated %s", s); bb_error_msg("syntax error: unterminated %s", s);
} }
static void syntax_error_unterm_ch(unsigned lineno, char ch) static void syntax_error_unterm_ch(unsigned lineno, char ch)
@ -1101,12 +1100,12 @@ static void syntax_error_unterm_ch(unsigned lineno, char ch)
syntax_error_unterm_str(lineno, msg); syntax_error_unterm_str(lineno, msg);
} }
static void syntax_error_unexpected_ch(unsigned lineno, int ch) static void syntax_error_unexpected_ch(unsigned lineno UNUSED_PARAM, int ch)
{ {
char msg[2]; char msg[2];
msg[0] = ch; msg[0] = ch;
msg[1] = '\0'; msg[1] = '\0';
die_if_script(lineno, "syntax error: unexpected %s", ch == EOF ? "EOF" : msg); bb_error_msg("syntax error: unexpected %s", ch == EOF ? "EOF" : msg);
} }
#if HUSH_DEBUG < 2 #if HUSH_DEBUG < 2
@ -1843,6 +1842,7 @@ static int FAST_FUNC static_get(struct in_str *i)
int ch = *i->p; int ch = *i->p;
if (ch != '\0') { if (ch != '\0') {
i->p++; i->p++;
i->last_char = ch;
return ch; return ch;
} }
return EOF; return EOF;
@ -1964,6 +1964,7 @@ static int FAST_FUNC file_get(struct in_str *i)
do ch = fgetc(i->file); while (ch == '\0'); do ch = fgetc(i->file); while (ch == '\0');
} }
debug_printf("file_get: got '%c' %d\n", ch, ch); debug_printf("file_get: got '%c' %d\n", ch, ch);
i->last_char = ch;
return ch; return ch;
} }
@ -4008,7 +4009,7 @@ static int encode_string(o_string *as_string,
* Scan input until EOF or end_trigger char. * Scan input until EOF or end_trigger char.
* Return a list of pipes to execute, or NULL on EOF * Return a list of pipes to execute, or NULL on EOF
* or if end_trigger character is met. * or if end_trigger character is met.
* On syntax error, exit is shell is not interactive, * On syntax error, exit if shell is not interactive,
* reset parsing machinery and start parsing anew, * reset parsing machinery and start parsing anew,
* or return ERR_PTR. * or return ERR_PTR.
*/ */
@ -4037,8 +4038,6 @@ static struct pipe *parse_stream(char **pstring,
* here we should use blank chars as separators, not $IFS * here we should use blank chars as separators, not $IFS
*/ */
reset: /* we come back here only on syntax errors in interactive shell */
if (MAYBE_ASSIGNMENT != 0) if (MAYBE_ASSIGNMENT != 0)
dest.o_assignment = MAYBE_ASSIGNMENT; dest.o_assignment = MAYBE_ASSIGNMENT;
initialize_context(&ctx); initialize_context(&ctx);
@ -4532,20 +4531,17 @@ static struct pipe *parse_stream(char **pstring,
} while (HAS_KEYWORDS && pctx); } while (HAS_KEYWORDS && pctx);
/* Free text, clear all dest fields */ /* Free text, clear all dest fields */
o_free(&dest); o_free(&dest);
G.last_exitcode = 1;
/* If we are not in top-level parse, we return, /* If we are not in top-level parse, we return,
* our caller will propagate error. * our caller will propagate error.
*/ */
if (end_trigger != ';') {
#if !BB_MMU #if !BB_MMU
if (pstring) if (pstring)
*pstring = NULL; *pstring = NULL;
#endif #endif
debug_leave(); debug_leave();
return ERR_PTR; return ERR_PTR;
}
/* Discard cached input, force prompt */
input->p = NULL;
goto reset;
} }
} }
@ -5550,8 +5546,24 @@ static void parse_and_run_stream(struct in_str *inp, int end_trigger)
inp->promptmode = 0; /* PS1 */ inp->promptmode = 0; /* PS1 */
#endif #endif
pipe_list = parse_stream(NULL, inp, end_trigger); pipe_list = parse_stream(NULL, inp, end_trigger);
if (!pipe_list) { /* EOF */ if (!pipe_list || pipe_list == ERR_PTR) { /* EOF/error */
if (empty) /* If we are in "big" script
* (not in `cmd` or something similar)...
*/
if (pipe_list == ERR_PTR && end_trigger == ';') {
/* Discard cached input (rest of line) */
int ch = inp->last_char;
while (ch != EOF && ch != '\n') {
//bb_error_msg("Discarded:'%c'", ch);
ch = i_getch(inp);
}
/* Force prompt */
inp->p = NULL;
/* This stream isn't empty */
empty = 0;
continue;
}
if (!pipe_list && empty)
G.last_exitcode = 0; G.last_exitcode = 0;
break; break;
} }
@ -8630,8 +8642,6 @@ static int FAST_FUNC builtin_source(char **argv)
#endif #endif
save_and_replace_G_args(&sv, argv); save_and_replace_G_args(&sv, argv);
//TODO: syntax errors in sourced file should never abort the "calling" script.
//Try: bash -c '. ./bad_file; echo YES'
parse_and_run_file(input); parse_and_run_file(input);
fclose(input); fclose(input);