mirror of
				https://git.proxmox.com/git/grub2
				synced 2025-11-04 14:37:14 +00:00 
			
		
		
		
	* normal/main.c: Add include to grub/menu_viewer.h. (free_menu_entry_classes): Added. (grub_normal_menu_addentry): Added class property handling. (grub_normal_execute): Changed to use new menu viewer for menu viewing. (GRUB_MOD_INIT(normal)): Added register for text based menu viewer. * normal/menu_viewer.c: New file. * normal/menu.c (run_menu_entry): Renamed to ... (grub_menu_execute_entry): ... this and made it as global. (grub_menu_run): Renamed to ... (show_text_menu): ... this and made it local. (show_text_menu): Adapt to new function names. (grub_normal_terminal_menu_viewer): New global variable. * include/grub/menu.h: New file. * include/grub/menu_viewer.h: New file. * include/grub/normal.h: Added include to grub/menu.h. (grub_menu_entry): Moved to include/grub/menu.h. (grub_menu_entry_t): Likewise. (grub_menu): Likewise. (grub_menu_t): Likewise. (grub_normal_terminal_menu_viewer): Added. (grub_menu_execute_entry): Likewise. (grub_menu_run): Removed. * DISTLIST: Added include/grub/menu.h. Added include/grub/menu_viewer.h. Added normal/menu_viewer.c. 2009-01-31 Vesa Jääskeläinen <chaac@nic.fi> * normal/execute.c (grub_script_execute_menuentry): Changed to use arglist for menutitle arguments. * normal/main.c (grub_normal_menu_addentry): Likewise. * normal/parser.y (menuentry): Likewise. * normal/script.c (grub_script_create_cmdmenu): Likewise. * include/grub/script.h (grub_script_cmd_menuentry): Likewise. (grub_script_create_cmdmenu): Likewise. * include/grub/normal.h (grub_normal_menu_addentry): Likewise. * conf/i386-pc.rmk (normal_mod_SOURCES): Adapt Colin D Bennett's changes. * conf/x86_64-efi.rmk (normal_mod_SOURCES): Likewise. * conf/i386-coreboot.rmk (normal_mod_SOURCES): Likewise. * conf/i386-efi.rmk (normal_mod_SOURCES): Likewise. * conf/i386-ieee1275.rmk (normal_mod_SOURCES): Likewise. * conf/powerpc-ieee1275.rmk (normal_mod_SOURCES): Likewise. * conf/sparc64-ieee1275.rmk (normal_mod_SOURCES): Likewise.
		
			
				
	
	
		
			276 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			276 lines
		
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* execute.c -- Execute a GRUB script.  */
 | 
						|
/*
 | 
						|
 *  GRUB  --  GRand Unified Bootloader
 | 
						|
 *  Copyright (C) 2005,2007,2008,2009  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/>.
 | 
						|
 */
 | 
						|
 | 
						|
#include <grub/misc.h>
 | 
						|
#include <grub/mm.h>
 | 
						|
#include <grub/normal.h>
 | 
						|
#include <grub/arg.h>
 | 
						|
#include <grub/env.h>
 | 
						|
#include <grub/script.h>
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_script_execute_cmd (struct grub_script_cmd *cmd)
 | 
						|
{
 | 
						|
  if (cmd == 0)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  return cmd->exec (cmd);
 | 
						|
}
 | 
						|
 | 
						|
/* Parse ARG and return the textual representation.  Add strings are
 | 
						|
   concatenated and all values of the variables are filled in.  */
 | 
						|
static char *
 | 
						|
grub_script_execute_argument_to_string (struct grub_script_arg *arg)
 | 
						|
{
 | 
						|
  int size = 0;
 | 
						|
  char *val;
 | 
						|
  char *chararg;
 | 
						|
  struct grub_script_arg *argi;
 | 
						|
 | 
						|
  /* First determine the size of the argument.  */
 | 
						|
  for (argi = arg; argi; argi = argi->next)
 | 
						|
    {
 | 
						|
      if (argi->type == 1)
 | 
						|
	{
 | 
						|
	  val = grub_env_get (argi->str);
 | 
						|
	  if (val)
 | 
						|
	    size += grub_strlen (val);
 | 
						|
	}
 | 
						|
      else
 | 
						|
	size += grub_strlen (argi->str);
 | 
						|
    }
 | 
						|
 | 
						|
  /* Create the argument.  */
 | 
						|
  chararg = grub_malloc (size + 1);
 | 
						|
  if (! chararg)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  *chararg = '\0';
 | 
						|
  /* First determine the size of the argument.  */
 | 
						|
  for (argi = arg; argi; argi = argi->next)
 | 
						|
    {
 | 
						|
      if (argi->type == 1)
 | 
						|
	{
 | 
						|
	  val = grub_env_get (argi->str);
 | 
						|
	  if (val)
 | 
						|
	    grub_strcat (chararg, val);
 | 
						|
	}
 | 
						|
      else
 | 
						|
	grub_strcat (chararg, argi->str);
 | 
						|
    }
 | 
						|
 | 
						|
  return chararg;
 | 
						|
}
 | 
						|
 | 
						|
/* Execute a single command line.  */
 | 
						|
grub_err_t
 | 
						|
grub_script_execute_cmdline (struct grub_script_cmd *cmd)
 | 
						|
{
 | 
						|
  struct grub_script_cmdline *cmdline = (struct grub_script_cmdline *) cmd;
 | 
						|
  struct grub_script_arglist *arglist;
 | 
						|
  char **args = 0;
 | 
						|
  int i = 0;
 | 
						|
  grub_command_t grubcmd;
 | 
						|
  struct grub_arg_list *state;
 | 
						|
  struct grub_arg_option *parser;
 | 
						|
  int maxargs = 0;
 | 
						|
  char **parsed_arglist;
 | 
						|
  int numargs;
 | 
						|
  grub_err_t ret = 0;
 | 
						|
  int argcount = 0;
 | 
						|
  grub_script_function_t func = 0;
 | 
						|
  char errnobuf[6];
 | 
						|
 | 
						|
  /* Lookup the command.  */
 | 
						|
  grubcmd = grub_command_find (cmdline->cmdname);
 | 
						|
  if (! grubcmd)
 | 
						|
    {
 | 
						|
      /* Ignore errors.  */
 | 
						|
      grub_errno = GRUB_ERR_NONE;
 | 
						|
 | 
						|
      /* It's not a GRUB command, try all functions.  */
 | 
						|
      func = grub_script_function_find (cmdline->cmdname);
 | 
						|
      if (! func)
 | 
						|
	{
 | 
						|
	  /* As a last resort, try if it is an assignment.  */
 | 
						|
	  char *assign = grub_strdup (cmdline->cmdname);
 | 
						|
	  char *eq = grub_strchr (assign, '=');
 | 
						|
 | 
						|
	  if (eq)
 | 
						|
	    {
 | 
						|
	      /* Create two strings and set the variable.  */
 | 
						|
	      *eq = '\0';
 | 
						|
	      eq++;
 | 
						|
	      grub_env_set (assign, eq);
 | 
						|
 | 
						|
	      /* This was set because the command was not found.  */
 | 
						|
	      grub_errno = GRUB_ERR_NONE;
 | 
						|
	    }
 | 
						|
	  grub_free (assign);
 | 
						|
	  return 0;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  if (cmdline->arglist)
 | 
						|
    {
 | 
						|
      argcount = cmdline->arglist->argcount;
 | 
						|
 | 
						|
      /* Create argv from the arguments.  */
 | 
						|
      args = grub_malloc (sizeof (char *) * argcount);
 | 
						|
      for (arglist = cmdline->arglist; arglist; arglist = arglist->next)
 | 
						|
	{
 | 
						|
	  char *str;
 | 
						|
	  str = grub_script_execute_argument_to_string (arglist->arg);
 | 
						|
	  args[i++] = str;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  /* Execute the GRUB command or function.  */
 | 
						|
  if (grubcmd)
 | 
						|
    {
 | 
						|
      /* Count the amount of options the command has.  */
 | 
						|
      parser = (struct grub_arg_option *) grubcmd->options;
 | 
						|
      while (parser && (parser++)->doc)
 | 
						|
	maxargs++;
 | 
						|
 | 
						|
      /* Set up the option state.  */
 | 
						|
      state = grub_malloc (sizeof (struct grub_arg_list) * maxargs);
 | 
						|
      grub_memset (state, 0, sizeof (struct grub_arg_list) * maxargs);
 | 
						|
 | 
						|
      /* Start the command.  */
 | 
						|
      if (! (grubcmd->flags & GRUB_COMMAND_FLAG_NO_ARG_PARSE))
 | 
						|
	{
 | 
						|
	  if (grub_arg_parse (grubcmd, argcount, args, state, &parsed_arglist, &numargs))
 | 
						|
	    ret = (grubcmd->func) (state, numargs, parsed_arglist);
 | 
						|
	}
 | 
						|
      else
 | 
						|
	ret = (grubcmd->func) (state, argcount, args);
 | 
						|
 | 
						|
      grub_free (state);
 | 
						|
    }
 | 
						|
  else
 | 
						|
    ret = grub_script_function_call (func, argcount, args);
 | 
						|
 | 
						|
  /* Free arguments.  */
 | 
						|
  for (i = 0; i < argcount; i++)
 | 
						|
    grub_free (args[i]);
 | 
						|
  grub_free (args);
 | 
						|
 | 
						|
  grub_sprintf (errnobuf, "%d", ret);
 | 
						|
  grub_env_set ("?", errnobuf);
 | 
						|
 | 
						|
  return ret;
 | 
						|
}
 | 
						|
 | 
						|
/* Execute a block of one or more commands.  */
 | 
						|
grub_err_t
 | 
						|
grub_script_execute_cmdblock (struct grub_script_cmd *cmd)
 | 
						|
{
 | 
						|
  struct grub_script_cmdblock *cmdblock = (struct grub_script_cmdblock *) cmd;
 | 
						|
 | 
						|
  /* Loop over every command and execute it.  */
 | 
						|
  for (cmd = cmdblock->cmdlist; cmd; cmd = cmd->next)
 | 
						|
    grub_script_execute_cmd (cmd);
 | 
						|
 | 
						|
  return 0;
 | 
						|
}
 | 
						|
 | 
						|
/* Execute an if statement.  */
 | 
						|
grub_err_t
 | 
						|
grub_script_execute_cmdif (struct grub_script_cmd *cmd)
 | 
						|
{
 | 
						|
  struct grub_script_cmdif *cmdif = (struct grub_script_cmdif *) cmd;
 | 
						|
  char *result;
 | 
						|
 | 
						|
  /* Check if the commands results in a true or a false.  The value is
 | 
						|
     read from the env variable `?'.  */
 | 
						|
  grub_script_execute_cmd (cmdif->exec_to_evaluate);
 | 
						|
  result = grub_env_get ("?");
 | 
						|
 | 
						|
  /* Execute the `if' or the `else' part depending on the value of
 | 
						|
     `?'.  */
 | 
						|
  if (result && ! grub_strcmp (result, "0"))
 | 
						|
    return grub_script_execute_cmd (cmdif->exec_on_true);
 | 
						|
  else
 | 
						|
    return grub_script_execute_cmd (cmdif->exec_on_false);
 | 
						|
}
 | 
						|
 | 
						|
/* Execute the menu entry generate statement.  */
 | 
						|
grub_err_t
 | 
						|
grub_script_execute_menuentry (struct grub_script_cmd *cmd)
 | 
						|
{
 | 
						|
  struct grub_script_cmd_menuentry *cmd_menuentry;
 | 
						|
  struct grub_script_arglist *arglist;
 | 
						|
  struct grub_script *script;
 | 
						|
  char **args = 0;
 | 
						|
  int argcount = 0;
 | 
						|
  int i = 0;
 | 
						|
 | 
						|
  cmd_menuentry = (struct grub_script_cmd_menuentry *) cmd;
 | 
						|
 | 
						|
  if (cmd_menuentry->arglist)
 | 
						|
    {
 | 
						|
      argcount = cmd_menuentry->arglist->argcount;
 | 
						|
 | 
						|
      /* Create argv from the arguments.  */
 | 
						|
      args = grub_malloc (sizeof (char *) * argcount);
 | 
						|
 | 
						|
      if (! args)
 | 
						|
	{
 | 
						|
	  return grub_errno;
 | 
						|
	}
 | 
						|
 | 
						|
      for (arglist = cmd_menuentry->arglist; arglist; arglist = arglist->next)
 | 
						|
	{
 | 
						|
	  char *str;
 | 
						|
	  str = grub_script_execute_argument_to_string (arglist->arg);
 | 
						|
	  args[i++] = str;
 | 
						|
	}
 | 
						|
    }
 | 
						|
 | 
						|
  /* Parse the menu entry *again*.  */
 | 
						|
  script = grub_script_parse ((char *) cmd_menuentry->sourcecode, 0);
 | 
						|
 | 
						|
  /* Add new menu entry.  */
 | 
						|
  if (script)
 | 
						|
    {
 | 
						|
      grub_normal_menu_addentry (argcount, (const char **)args,
 | 
						|
				 script, cmd_menuentry->sourcecode);
 | 
						|
    }
 | 
						|
 | 
						|
  /* Free arguments.  */
 | 
						|
  for (i = 0; i < argcount; i++)
 | 
						|
    grub_free (args[i]);
 | 
						|
  grub_free (args);
 | 
						|
 | 
						|
  return grub_errno;
 | 
						|
}
 | 
						|
 | 
						|
 | 
						|
 | 
						|
/* Execute any GRUB pre-parsed command or script.  */
 | 
						|
grub_err_t
 | 
						|
grub_script_execute (struct grub_script *script)
 | 
						|
{
 | 
						|
  if (script == 0)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  return grub_script_execute_cmd (script->cmd);
 | 
						|
}
 |