Add variable parsing in $"..." and fix several mismatches with bash.

* Makefile.util.def (grub_script_gettext): New test.
	* grub-core/script/execute.c (parse_string): New function.
	(gettext_append): Likewise.
	(grub_script_arglist_to_argv): Use gettext_append.
	* grub-core/script/yylex.l: Fix slash and newline handling in $"...".
	* tests/grub_script_gettext.in: New file.
This commit is contained in:
Vladimir 'phcoder' Serbinenko 2012-03-11 14:46:48 +01:00
parent 9fdb2d7b11
commit 546fbe9b5a
5 changed files with 298 additions and 4 deletions

View file

@ -1,3 +1,14 @@
2012-03-11 Vladimir Serbinenko <phcoder@gmail.com>
Add variable parsing in $"..." and fix several mismatches with bash.
* Makefile.util.def (grub_script_gettext): New test.
* grub-core/script/execute.c (parse_string): New function.
(gettext_append): Likewise.
(grub_script_arglist_to_argv): Use gettext_append.
* grub-core/script/yylex.l: Fix slash and newline handling in $"...".
* tests/grub_script_gettext.in: New file.
2012-03-11 Vladimir Serbinenko <phcoder@gmail.com> 2012-03-11 Vladimir Serbinenko <phcoder@gmail.com>
Fix handling of leading spaces in scripts. Fix handling of leading spaces in scripts.

View file

@ -666,6 +666,12 @@ script = {
common = tests/grub_cmd_echo.in; common = tests/grub_cmd_echo.in;
}; };
script = {
testcase;
name = grub_script_gettext;
common = tests/grub_script_gettext.in;
};
program = { program = {
testcase; testcase;
name = example_unit_test; name = example_unit_test;

View file

@ -321,6 +321,214 @@ grub_script_env_set (const char *name, const char *val)
return grub_env_set (name, val); return grub_env_set (name, val);
} }
static int
parse_string (const char *str,
int (*hook) (const char *var, grub_size_t varlen),
char **put)
{
const char *ptr;
int escaped = 0;
const char *optr;
for (ptr = str; ptr && *ptr; )
switch (*ptr)
{
case '\\':
escaped = !escaped;
if (!escaped && put)
*((*put)++) = '\\';
ptr++;
break;
case '$':
if (escaped)
{
escaped = 0;
if (put)
*((*put)++) = *ptr;
ptr++;
break;
}
ptr++;
switch (*ptr)
{
case '{':
{
optr = ptr + 1;
ptr = grub_strchr (optr, '}');
if (!ptr)
break;
if (hook (optr, ptr - optr))
return 1;
ptr++;
break;
}
case '0' ... '9':
optr = ptr;
while (*ptr >= '0' && *ptr <= '9')
ptr++;
if (hook (optr, ptr - optr))
return 1;
break;
case 'a' ... 'z':
case 'A' ... 'Z':
case '_':
optr = ptr;
while ((*ptr >= '0' && *ptr <= '9')
|| (*ptr >= 'a' && *ptr <= 'z')
|| (*ptr >= 'A' && *ptr <= 'Z')
|| *ptr == '_')
ptr++;
if (hook (optr, ptr - optr))
return 1;
break;
case '?':
case '#':
if (hook (ptr, 1))
return 1;
ptr++;
break;
default:
if (put)
*((*put)++) = '$';
}
break;
default:
if (escaped && put)
*((*put)++) = '\\';
escaped = 0;
if (put)
*((*put)++) = *ptr;
ptr++;
break;
}
return 0;
}
static int
gettext_append (struct grub_script_argv *result, const char *orig_str)
{
const char *template;
char *res = 0, *ptr;
char **allowed_strings;
grub_size_t nallowed_strings = 0;
grub_size_t additional_len = 1;
int rval = 1;
const char *iptr;
auto int save_allow (const char *str, grub_size_t len);
int save_allow (const char *str, grub_size_t len)
{
allowed_strings[nallowed_strings++] = grub_strndup (str, len);
if (!allowed_strings[nallowed_strings - 1])
return 1;
return 0;
}
auto int getlen (const char *str, grub_size_t len);
int getlen (const char *str, grub_size_t len)
{
const char *var;
grub_size_t i;
for (i = 0; i < nallowed_strings; i++)
if (grub_strncmp (allowed_strings[i], str, len) == 0
&& allowed_strings[i][len] == 0)
break;
if (i == nallowed_strings)
return 0;
/* Enough for any number. */
if (len == 1 && str[0] == '#')
{
additional_len += 30;
return 0;
}
var = grub_env_get (allowed_strings[i]);
if (var)
additional_len += grub_strlen (var);
return 0;
}
auto int putvar (const char *str, grub_size_t len);
int putvar (const char *str, grub_size_t len)
{
const char *var;
grub_size_t i;
for (i = 0; i < nallowed_strings; i++)
if (grub_strncmp (allowed_strings[i], str, len) == 0
&& allowed_strings[i][len] == 0)
{
break;
}
if (i == nallowed_strings)
return 0;
/* Enough for any number. */
if (len == 1 && str[0] == '#')
{
grub_snprintf (ptr, 30, "%u", scope->argv.argc);
ptr += grub_strlen (ptr);
return 0;
}
var = grub_env_get (allowed_strings[i]);
if (var)
ptr = grub_stpcpy (ptr, var);
return 0;
}
grub_size_t dollar_cnt = 0;
for (iptr = orig_str; *iptr; iptr++)
if (*iptr == '$')
dollar_cnt++;
allowed_strings = grub_malloc (sizeof (allowed_strings[0]) * dollar_cnt);
if (parse_string (orig_str, save_allow, 0))
goto fail;
template = _(orig_str);
if (parse_string (template, getlen, 0))
goto fail;
res = grub_malloc (grub_strlen (template) + additional_len);
if (!res)
goto fail;
ptr = res;
if (parse_string (template, putvar, &ptr))
goto fail;
*ptr = 0;
if (grub_wildcard_translator)
{
char *escaped = 0;
escaped = grub_wildcard_translator->escape (res);
if (grub_script_argv_append (result, escaped, grub_strlen (escaped)))
{
grub_free (escaped);
goto fail;
}
grub_free (escaped);
}
else
if (grub_script_argv_append (result, res, ptr - res))
goto fail;
rval = 0;
fail:
grub_free (res);
{
grub_size_t i;
for (i = 0; i < nallowed_strings; i++)
grub_free (allowed_strings[i]);
}
grub_free (allowed_strings);
return rval;
}
/* Convert arguments in ARGLIST into ARGV form. */ /* Convert arguments in ARGLIST into ARGV form. */
static int static int
grub_script_arglist_to_argv (struct grub_script_arglist *arglist, grub_script_arglist_to_argv (struct grub_script_arglist *arglist,
@ -406,8 +614,7 @@ grub_script_arglist_to_argv (struct grub_script_arglist *arglist,
case GRUB_SCRIPT_ARG_TYPE_GETTEXT: case GRUB_SCRIPT_ARG_TYPE_GETTEXT:
{ {
const char *t = _(arg->str); if (gettext_append (&result, arg->str))
if (grub_script_argv_append (&result, t, grub_strlen (t)))
goto fail; goto fail;
} }
break; break;

View file

@ -139,7 +139,7 @@ SPECIAL \?|\#|\*|\@
VARIABLE ${NAME}|$\{{NAME}\}|${DIGITS}|$\{{DIGITS}\}|${SPECIAL}|$\{{SPECIAL}\} VARIABLE ${NAME}|$\{{NAME}\}|${DIGITS}|$\{{DIGITS}\}|${SPECIAL}|$\{{SPECIAL}\}
WORD ({CHAR}|{DQSTR}|{SQSTR}|{ESC}|{VARIABLE}|{I18NSTR})+ WORD ({CHAR}|{DQSTR}|{SQSTR}|{ESC}|{VARIABLE}|{I18NSTR})+
MULTILINE {WORD}?((\"{DQCHR}*)|(\'{SQCHR}*)|(\\\n)) MULTILINE {WORD}?((\"{DQCHR}*)|(\$\"{DQCHR}*)|(\'{SQCHR}*)|(\\\n))
%x SPLIT %x SPLIT
%x DQUOTE %x DQUOTE
@ -289,7 +289,7 @@ MULTILINE {WORD}?((\"{DQCHR}*)|(\'{SQCHR}*)|(\\\n))
} }
<I18NQUOTE>{ <I18NQUOTE>{
\\\\ { COPY ("\\", 1); } \\\\ { COPY ("\\\\", 2); }
\\\" { COPY ("\"", 1); } \\\" { COPY ("\"", 1); }
\\\n { /* ignore */ } \\\n { /* ignore */ }
[^\"\\\n]+ { COPY (yytext, yyleng); } [^\"\\\n]+ { COPY (yytext, yyleng); }
@ -297,6 +297,7 @@ MULTILINE {WORD}?((\"{DQCHR}*)|(\'{SQCHR}*)|(\\\n))
yy_pop_state (yyscanner); yy_pop_state (yyscanner);
ARG (GRUB_SCRIPT_ARG_TYPE_GETTEXT); ARG (GRUB_SCRIPT_ARG_TYPE_GETTEXT);
} }
\\ { COPY ("\\", 1); }
(.|\n) { COPY (yytext, yyleng); } (.|\n) { COPY (yytext, yyleng); }
} }

View file

@ -0,0 +1,69 @@
#! @builddir@/grub-shell-tester
#
# Copyright (C) 2010,2012 Free Software Foundation, Inc.
#
# GRUB 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.
#
# GRUB is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with GRUB. If not, see <http://www.gnu.org/licenses/>.
echo $"foo"
echo $"foo bar"
echo -n $"foo"
echo -e $"foo\nbar"
echo -n -e $"foo\nbar"
x=5
echo $"$x"
echo $"\x\\y\$x$x\\$xx${x}x\"$x\""
echo $"$"
echo $"$,x"
echo $"one
"
echo $"one
\""
echo $"one
two"
echo one$"two
"three
echo one$"two
\""three
echo one$"two
\"three\"
four"
echo $"one\
"
echo $"one\
\""
echo $"one\
two"
echo one$"two\
"three
echo one$"two\
\""three
echo one$"two\
\"three\"\
four"
if test -n "$grubshell"; then insmod regexp; fi
echo $"*"
foo="*"
echo $"$foo"