diff --git a/ChangeLog b/ChangeLog index 38e8d4650..a8c30d971 100644 --- a/ChangeLog +++ b/ChangeLog @@ -1,3 +1,11 @@ +2009-08-24 Vladimir Serbinenko + + * script/sh/function.c (grub_script_function_find): Cut error message + not to flood terminal. + * script/sh/lexer.c (grub_script_yylex): Remove command line length + limit. + * script/sh/script.c (grub_script_arg_add): Duplicate string. + 2009-08-24 Colin Watson * term/usb_keyboard.c (grub_usb_keyboard_getreport): Make diff --git a/script/sh/function.c b/script/sh/function.c index 1e49c709f..a3950a8a0 100644 --- a/script/sh/function.c +++ b/script/sh/function.c @@ -99,7 +99,7 @@ grub_script_function_find (char *functionname) break; if (! func) - grub_error (GRUB_ERR_UNKNOWN_COMMAND, "unknown command `%s'", functionname); + grub_error (GRUB_ERR_UNKNOWN_COMMAND, "unknown command `%.20s'", functionname); return func; } diff --git a/script/sh/lexer.c b/script/sh/lexer.c index 44d7da2ad..a30e3c005 100644 --- a/script/sh/lexer.c +++ b/script/sh/lexer.c @@ -134,8 +134,6 @@ grub_script_yylex (union YYSTYPE *yylval, struct grub_parser_param *parsestate) { grub_parser_state_t newstate; char use; - char *buffer; - char *bp; struct grub_lexer_param *state = parsestate->lexerstate; int firstrun = 1; @@ -212,6 +210,14 @@ grub_script_yylex (union YYSTYPE *yylval, struct grub_parser_param *parsestate) /* Check if it is a text. */ if (check_textstate (newstate)) { + char *buffer = NULL; + int bufpos = 0; + /* Buffer is initially large enough to hold most commands + but extends automatically when needed. */ + int bufsize = 128; + + buffer = grub_malloc (bufsize); + /* In case the string is not quoted, this can be a one char length symbol. */ if (newstate == GRUB_PARSER_STATE_TEXT) @@ -254,16 +260,12 @@ grub_script_yylex (union YYSTYPE *yylval, struct grub_parser_param *parsestate) } } if (doexit) - break; + { + grub_free (buffer); + break; + } } - /* XXX: Use a better size. */ - buffer = grub_script_malloc (parsestate, 2048); - if (! buffer) - return 0; - - bp = buffer; - /* Read one token, possible quoted. */ while (*state->script) { @@ -295,32 +297,47 @@ grub_script_yylex (union YYSTYPE *yylval, struct grub_parser_param *parsestate) } if (breakout) break; - if (use) - *(bp++) = use; } - else if (use) - *(bp++) = use; + + if (use) + { + if (bufsize <= bufpos + 1) + { + bufsize <<= 1; + buffer = grub_realloc (buffer, bufsize); + } + buffer[bufpos++] = use; + } state->state = newstate; nextchar (state); } /* A string of text was read in. */ - *bp = '\0'; + if (bufsize <= bufpos + 1) + { + bufsize <<= 1; + buffer = grub_realloc (buffer, bufsize); + } + + buffer[bufpos++] = 0; + grub_dprintf ("scripting", "token=`%s'\n", buffer); yylval->arg = grub_script_arg_add (parsestate, yylval->arg, GRUB_SCRIPT_ARG_TYPE_STR, buffer); + grub_free (buffer); } else if (newstate == GRUB_PARSER_STATE_VAR || newstate == GRUB_PARSER_STATE_QVAR) { - /* XXX: Use a better size. */ - buffer = grub_script_malloc (parsestate, 2096); - if (! buffer) - return 0; + char *buffer = NULL; + int bufpos = 0; + /* Buffer is initially large enough to hold most commands + but extends automatically when needed. */ + int bufsize = 128; - bp = buffer; + buffer = grub_malloc (bufsize); /* This is a variable, read the variable name. */ while (*state->script) @@ -340,16 +357,33 @@ grub_script_yylex (union YYSTYPE *yylval, struct grub_parser_param *parsestate) } if (use) - *(bp++) = use; + { + if (bufsize <= bufpos + 1) + { + bufsize <<= 1; + buffer = grub_realloc (buffer, bufsize); + } + buffer[bufpos++] = use; + } + nextchar (state); state->state = newstate; } - *bp = '\0'; + if (bufsize <= bufpos + 1) + { + bufsize <<= 1; + buffer = grub_realloc (buffer, bufsize); + } + + buffer[bufpos++] = 0; + state->state = newstate; yylval->arg = grub_script_arg_add (parsestate, yylval->arg, GRUB_SCRIPT_ARG_TYPE_VAR, buffer); grub_dprintf ("scripting", "vartoken=`%s'\n", buffer); + + grub_free (buffer); } else { diff --git a/script/sh/script.c b/script/sh/script.c index cefafe605..c04a44966 100644 --- a/script/sh/script.c +++ b/script/sh/script.c @@ -110,10 +110,13 @@ grub_script_arg_add (struct grub_parser_param *state, struct grub_script_arg *ar { struct grub_script_arg *argpart; struct grub_script_arg *ll; + int len; argpart = (struct grub_script_arg *) grub_script_malloc (state, sizeof (*arg)); argpart->type = type; - argpart->str = str; + len = grub_strlen (str) + 1; + argpart->str = grub_script_malloc (state, len); + grub_memcpy (argpart->str, str, len); argpart->next = 0; if (! arg)