restrict to only one block-arg (as last) param per command

This commit is contained in:
BVK Chaitanya 2010-07-21 03:47:30 +05:30
parent 15ee6f9dc7
commit 9ebedc24f2
8 changed files with 31 additions and 71 deletions

View file

@ -25,7 +25,7 @@
grub_err_t grub_err_t
grub_extcmd_dispatcher (struct grub_command *cmd, int argc, char **args, grub_extcmd_dispatcher (struct grub_command *cmd, int argc, char **args,
struct grub_script **scripts) struct grub_script *script)
{ {
int new_argc; int new_argc;
char **new_args; char **new_args;
@ -38,7 +38,7 @@ grub_extcmd_dispatcher (struct grub_command *cmd, int argc, char **args,
context.state = 0; context.state = 0;
context.extcmd = ext; context.extcmd = ext;
context.script_params = scripts; context.script = script;
/* Dynamic commands should not perform option parsing before /* Dynamic commands should not perform option parsing before
corresponding module gets loaded. */ corresponding module gets loaded. */

View file

@ -50,8 +50,8 @@ struct grub_extcmd_context
struct grub_arg_list *state; struct grub_arg_list *state;
/* Script parameters, if any. */ /* Script parameter, if any. */
struct grub_script **script_params; struct grub_script *script;
}; };
typedef struct grub_extcmd_context *grub_extcmd_context_t; typedef struct grub_extcmd_context *grub_extcmd_context_t;
@ -74,6 +74,6 @@ void grub_unregister_extcmd (grub_extcmd_t cmd);
grub_err_t grub_err_t
grub_extcmd_dispatcher (struct grub_command *cmd, int argc, char **args, grub_extcmd_dispatcher (struct grub_command *cmd, int argc, char **args,
struct grub_script **scripts); struct grub_script *script);
#endif /* ! GRUB_EXTCMD_HEADER */ #endif /* ! GRUB_EXTCMD_HEADER */

View file

@ -62,7 +62,7 @@ struct grub_script_arg
char *str; char *str;
/* Parsed block argument. */ /* Parsed block argument. */
struct grub_script *block; struct grub_script *script;
/* Next argument part. */ /* Next argument part. */
struct grub_script_arg *next; struct grub_script_arg *next;
@ -73,7 +73,7 @@ struct grub_script_argv
{ {
unsigned argc; unsigned argc;
char **args; char **args;
struct grub_script **scripts; struct grub_script *script;
}; };
/* A complete argument. It consists of a list of one or more `struct /* A complete argument. It consists of a list of one or more `struct
@ -234,8 +234,6 @@ void grub_script_argv_free (struct grub_script_argv *argv);
int grub_script_argv_next (struct grub_script_argv *argv); int grub_script_argv_next (struct grub_script_argv *argv);
int grub_script_argv_append (struct grub_script_argv *argv, const char *s); int grub_script_argv_append (struct grub_script_argv *argv, const char *s);
int grub_script_argv_split_append (struct grub_script_argv *argv, char *s); int grub_script_argv_split_append (struct grub_script_argv *argv, char *s);
int grub_script_argv_script_append (struct grub_script_argv *argv,
struct grub_script *s);
struct grub_script_arglist * struct grub_script_arglist *
grub_script_create_arglist (struct grub_parser_param *state); grub_script_create_arglist (struct grub_parser_param *state);

View file

@ -54,8 +54,7 @@ grub_dyncmd_dispatcher (struct grub_extcmd_context *ctxt,
{ {
if (cmd->flags & GRUB_COMMAND_FLAG_BLOCKS && if (cmd->flags & GRUB_COMMAND_FLAG_BLOCKS &&
cmd->flags & GRUB_COMMAND_FLAG_EXTCMD) cmd->flags & GRUB_COMMAND_FLAG_EXTCMD)
ret = grub_extcmd_dispatcher (cmd, argc, args, ret = grub_extcmd_dispatcher (cmd, argc, args, ctxt->script);
ctxt->script_params);
else else
ret = (cmd->func) (cmd, argc, args); ret = (cmd->func) (cmd, argc, args);
} }

View file

@ -18,6 +18,7 @@
*/ */
#include <grub/mm.h> #include <grub/mm.h>
#include <grub/misc.h>
#include <grub/script_sh.h> #include <grub/script_sh.h>
static unsigned static unsigned
@ -52,18 +53,8 @@ grub_script_argv_free (struct grub_script_argv *argv)
grub_free (argv->args); grub_free (argv->args);
} }
if (argv->scripts)
{
for (i = 0; i < argv->argc; i++)
if (argv->scripts[i])
grub_script_put (argv->scripts[i]);
grub_free (argv->scripts);
}
argv->argc = 0; argv->argc = 0;
argv->args = 0; argv->args = 0;
argv->scripts = 0;
} }
/* Prepare for next argc. */ /* Prepare for next argc. */
@ -71,7 +62,6 @@ int
grub_script_argv_next (struct grub_script_argv *argv) grub_script_argv_next (struct grub_script_argv *argv)
{ {
char **p = argv->args; char **p = argv->args;
struct grub_script **q = argv->scripts;
if (argv->args && argv->argc && argv->args[argv->argc - 1] == 0) if (argv->args && argv->argc && argv->args[argv->argc - 1] == 0)
return 0; return 0;
@ -80,24 +70,12 @@ grub_script_argv_next (struct grub_script_argv *argv)
if (! p) if (! p)
return 1; return 1;
q = grub_realloc (q, round_up_exp ((argv->argc + 2) * sizeof (struct grub_script *)));
if (! q)
{
grub_free (p);
return 1;
}
argv->argc++; argv->argc++;
argv->args = p; argv->args = p;
argv->scripts = q;
if (argv->argc == 1) if (argv->argc == 1)
{ argv->args[0] = 0;
argv->args[0] = 0;
argv->scripts[0] = 0;
}
argv->args[argv->argc] = 0; argv->args[argv->argc] = 0;
argv->scripts[argv->argc] = 0;
return 0; return 0;
} }
@ -123,18 +101,6 @@ grub_script_argv_append (struct grub_script_argv *argv, const char *s)
return 0; return 0;
} }
/* Append grub_script `s' as the last argument. */
int
grub_script_argv_script_append (struct grub_script_argv *argv,
struct grub_script *script)
{
if (argv->scripts[argv->argc - 1])
grub_script_put (argv->scripts[argv->argc - 1]);
argv->scripts[argv->argc - 1] = grub_script_get (script);
return 0;
}
/* Split `s' and append words as multiple arguments. */ /* Split `s' and append words as multiple arguments. */
int int
grub_script_argv_split_append (struct grub_script_argv *argv, char *s) grub_script_argv_split_append (struct grub_script_argv *argv, char *s)

View file

@ -199,9 +199,9 @@ grub_script_arglist_to_argv (struct grub_script_arglist *arglist,
case GRUB_SCRIPT_ARG_TYPE_BLOCK: case GRUB_SCRIPT_ARG_TYPE_BLOCK:
if (grub_script_argv_append (&result, "{") || if (grub_script_argv_append (&result, "{") ||
grub_script_argv_append (&result, arg->str) || grub_script_argv_append (&result, arg->str) ||
grub_script_argv_append (&result, "}") || grub_script_argv_append (&result, "}"))
grub_script_argv_script_append (&result, arg->block))
goto fail; goto fail;
result.script = arg->script;
break; break;
case GRUB_SCRIPT_ARG_TYPE_TEXT: case GRUB_SCRIPT_ARG_TYPE_TEXT:
@ -326,7 +326,7 @@ grub_script_execute_cmdline (struct grub_script_cmd *cmd)
if ((grubcmd->flags & GRUB_COMMAND_FLAG_BLOCKS) && if ((grubcmd->flags & GRUB_COMMAND_FLAG_BLOCKS) &&
(grubcmd->flags & GRUB_COMMAND_FLAG_EXTCMD)) (grubcmd->flags & GRUB_COMMAND_FLAG_EXTCMD))
ret = grub_extcmd_dispatcher (grubcmd, argv.argc - 1, argv.args + 1, ret = grub_extcmd_dispatcher (grubcmd, argv.argc - 1, argv.args + 1,
argv.scripts + 1); argv.script);
else else
ret = (grubcmd->func) (grubcmd, argv.argc - 1, argv.args + 1); ret = (grubcmd->func) (grubcmd, argv.argc - 1, argv.args + 1);
} }

View file

@ -79,7 +79,8 @@
%token <arg> GRUB_PARSER_TOKEN_NAME "name" %token <arg> GRUB_PARSER_TOKEN_NAME "name"
%token <arg> GRUB_PARSER_TOKEN_WORD "word" %token <arg> GRUB_PARSER_TOKEN_WORD "word"
%type <arglist> word argument block parameters0 parameters1 arguments0 arguments1 %type <arg> block block0
%type <arglist> word argument parameters0 parameters1 arguments0 arguments1
%type <cmd> script_init script %type <cmd> script_init script
%type <cmd> grubcmd ifclause ifcmd forcmd whilecmd untilcmd %type <cmd> grubcmd ifclause ifcmd forcmd whilecmd untilcmd
@ -160,21 +161,22 @@ block: "{"
commands1 delimiters0 "}" commands1 delimiters0 "}"
{ {
char *p; char *p;
struct grub_script_arg *arg;
struct grub_script_mem *memory; struct grub_script_mem *memory;
memory = grub_script_mem_record_stop (state, $<memory>2); memory = grub_script_mem_record_stop (state, $<memory>2);
if ((p = grub_script_lexer_record_stop (state, $<offset>2))) if ((p = grub_script_lexer_record_stop (state, $<offset>2)))
*grub_strrchr (p, '}') = '\0'; *grub_strrchr (p, '}') = '\0';
arg = grub_script_arg_add (state, 0, GRUB_SCRIPT_ARG_TYPE_BLOCK, p); $$ = grub_script_arg_add (state, 0, GRUB_SCRIPT_ARG_TYPE_BLOCK, p);
if (! arg || ! (arg->block = grub_script_create ($3, memory))) if (! $$ || ! ($$->script = grub_script_create ($3, memory)))
grub_script_mem_free (memory); grub_script_mem_free (memory);
$$ = grub_script_add_arglist (state, 0, arg);
grub_script_lexer_deref (state->lexerstate); grub_script_lexer_deref (state->lexerstate);
} }
; ;
block0: /* Empty */ { $$ = 0; }
| block { $$ = $1; }
;
arguments0: /* Empty */ { $$ = 0; } arguments0: /* Empty */ { $$ = 0; }
| arguments1 { $$ = $1; } | arguments1 { $$ = $1; }
@ -201,27 +203,22 @@ parameters1: argument parameters0
} }
$$ = $1; $$ = $1;
} }
| block parameters0
{
if ($1 && $2)
{
$1->next = $2;
$1->argcount += $2->argcount;
$2->argcount = 0;
}
$$ = $1;
}
; ;
parameters0: /* Empty */ { $$ = 0; } parameters0: /* Empty */ { $$ = 0; }
| parameters1 { $$ = $1; } | parameters1 { $$ = $1; }
; ;
grubcmd: word parameters0 grubcmd: word parameters0 block0
{ {
if ($1 && $2) { struct grub_script_arglist *x = $2;
$1->next = $2;
$1->argcount += $2->argcount; if ($3)
$2->argcount = 0; x = grub_script_add_arglist (state, $2, $3);
if ($1 && x) {
$1->next = x;
$1->argcount += x->argcount;
x->argcount = 0;
} }
$$ = grub_script_create_cmdline (state, $1); $$ = grub_script_create_cmdline (state, $1);
} }

View file

@ -122,7 +122,7 @@ grub_script_arg_add (struct grub_parser_param *state,
return arg; return arg;
argpart->type = type; argpart->type = type;
argpart->block = 0; argpart->script = 0;
len = grub_strlen (str) + 1; len = grub_strlen (str) + 1;
argpart->str = grub_script_malloc (state, len); argpart->str = grub_script_malloc (state, len);