mirror of
				https://git.proxmox.com/git/grub2
				synced 2025-11-04 12:55:16 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			401 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			401 lines
		
	
	
		
			9.5 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
/* loadenv.c - command to load/save environment variable.  */
 | 
						|
/*
 | 
						|
 *  GRUB  --  GRand Unified Bootloader
 | 
						|
 *  Copyright (C) 2008,2009,2010  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/dl.h>
 | 
						|
#include <grub/mm.h>
 | 
						|
#include <grub/file.h>
 | 
						|
#include <grub/disk.h>
 | 
						|
#include <grub/misc.h>
 | 
						|
#include <grub/env.h>
 | 
						|
#include <grub/partition.h>
 | 
						|
#include <grub/lib/envblk.h>
 | 
						|
#include <grub/extcmd.h>
 | 
						|
#include <grub/i18n.h>
 | 
						|
 | 
						|
static const struct grub_arg_option options[] =
 | 
						|
  {
 | 
						|
    {"file", 'f', 0, N_("Specify filename."), 0, ARG_TYPE_PATHNAME},
 | 
						|
    {0, 0, 0, 0, 0, 0}
 | 
						|
  };
 | 
						|
 | 
						|
static grub_file_t
 | 
						|
open_envblk_file (char *filename)
 | 
						|
{
 | 
						|
  grub_file_t file;
 | 
						|
 | 
						|
  if (! filename)
 | 
						|
    {
 | 
						|
      char *prefix;
 | 
						|
 | 
						|
      prefix = grub_env_get ("prefix");
 | 
						|
      if (prefix)
 | 
						|
        {
 | 
						|
          int len;
 | 
						|
 | 
						|
          len = grub_strlen (prefix);
 | 
						|
          filename = grub_malloc (len + 1 + sizeof (GRUB_ENVBLK_DEFCFG));
 | 
						|
          if (! filename)
 | 
						|
            return 0;
 | 
						|
 | 
						|
          grub_strcpy (filename, prefix);
 | 
						|
          filename[len] = '/';
 | 
						|
          grub_strcpy (filename + len + 1, GRUB_ENVBLK_DEFCFG);
 | 
						|
          file = grub_file_open (filename);
 | 
						|
          grub_free (filename);
 | 
						|
          return file;
 | 
						|
        }
 | 
						|
      else
 | 
						|
        {
 | 
						|
          grub_error (GRUB_ERR_FILE_NOT_FOUND, "prefix is not found");
 | 
						|
          return 0;
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
  return grub_file_open (filename);
 | 
						|
}
 | 
						|
 | 
						|
static grub_envblk_t
 | 
						|
read_envblk_file (grub_file_t file)
 | 
						|
{
 | 
						|
  grub_off_t offset = 0;
 | 
						|
  char *buf;
 | 
						|
  grub_size_t size = grub_file_size (file);
 | 
						|
  grub_envblk_t envblk;
 | 
						|
 | 
						|
  buf = grub_malloc (size);
 | 
						|
  if (! buf)
 | 
						|
    return 0;
 | 
						|
 | 
						|
  while (size > 0)
 | 
						|
    {
 | 
						|
      grub_ssize_t ret;
 | 
						|
 | 
						|
      ret = grub_file_read (file, buf + offset, size);
 | 
						|
      if (ret <= 0)
 | 
						|
        {
 | 
						|
          if (grub_errno == GRUB_ERR_NONE)
 | 
						|
            grub_error (GRUB_ERR_FILE_READ_ERROR, "cannot read");
 | 
						|
          grub_free (buf);
 | 
						|
          return 0;
 | 
						|
        }
 | 
						|
 | 
						|
      size -= ret;
 | 
						|
      offset += ret;
 | 
						|
    }
 | 
						|
 | 
						|
  envblk = grub_envblk_open (buf, offset);
 | 
						|
  if (! envblk)
 | 
						|
    {
 | 
						|
      grub_free (buf);
 | 
						|
      grub_error (GRUB_ERR_BAD_FILE_TYPE, "invalid environment block");
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
 | 
						|
  return envblk;
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_cmd_load_env (grub_extcmd_t cmd,
 | 
						|
		   int argc __attribute__ ((unused)),
 | 
						|
		   char **args __attribute__ ((unused)))
 | 
						|
{
 | 
						|
  struct grub_arg_list *state = cmd->state;
 | 
						|
  grub_file_t file;
 | 
						|
  grub_envblk_t envblk;
 | 
						|
 | 
						|
  auto int set_var (const char *name, const char *value);
 | 
						|
  int set_var (const char *name, const char *value)
 | 
						|
  {
 | 
						|
    grub_env_set (name, value);
 | 
						|
    return 0;
 | 
						|
  }
 | 
						|
 | 
						|
  file = open_envblk_file ((state[0].set) ? state[0].arg : 0);
 | 
						|
  if (! file)
 | 
						|
    return grub_errno;
 | 
						|
 | 
						|
  envblk = read_envblk_file (file);
 | 
						|
  if (! envblk)
 | 
						|
    goto fail;
 | 
						|
 | 
						|
  grub_envblk_iterate (envblk, set_var);
 | 
						|
  grub_envblk_close (envblk);
 | 
						|
 | 
						|
 fail:
 | 
						|
  grub_file_close (file);
 | 
						|
  return grub_errno;
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_cmd_list_env (grub_extcmd_t cmd,
 | 
						|
		   int argc __attribute__ ((unused)),
 | 
						|
		   char **args __attribute__ ((unused)))
 | 
						|
{
 | 
						|
  struct grub_arg_list *state = cmd->state;
 | 
						|
  grub_file_t file;
 | 
						|
  grub_envblk_t envblk;
 | 
						|
 | 
						|
  /* Print all variables in current context.  */
 | 
						|
  auto int print_var (const char *name, const char *value);
 | 
						|
  int print_var (const char *name, const char *value)
 | 
						|
    {
 | 
						|
      grub_printf ("%s=%s\n", name, value);
 | 
						|
      return 0;
 | 
						|
    }
 | 
						|
 | 
						|
  file = open_envblk_file ((state[0].set) ? state[0].arg : 0);
 | 
						|
  if (! file)
 | 
						|
    return grub_errno;
 | 
						|
 | 
						|
  envblk = read_envblk_file (file);
 | 
						|
  if (! envblk)
 | 
						|
    goto fail;
 | 
						|
 | 
						|
  grub_envblk_iterate (envblk, print_var);
 | 
						|
  grub_envblk_close (envblk);
 | 
						|
 | 
						|
 fail:
 | 
						|
  grub_file_close (file);
 | 
						|
  return grub_errno;
 | 
						|
}
 | 
						|
 | 
						|
/* Used to maintain a variable length of blocklists internally.  */
 | 
						|
struct blocklist
 | 
						|
{
 | 
						|
  grub_disk_addr_t sector;
 | 
						|
  unsigned offset;
 | 
						|
  unsigned length;
 | 
						|
  struct blocklist *next;
 | 
						|
};
 | 
						|
 | 
						|
static void
 | 
						|
free_blocklists (struct blocklist *p)
 | 
						|
{
 | 
						|
  struct blocklist *q;
 | 
						|
 | 
						|
  for (; p; p = q)
 | 
						|
    {
 | 
						|
      q = p->next;
 | 
						|
      grub_free (p);
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
check_blocklists (grub_envblk_t envblk, struct blocklist *blocklists,
 | 
						|
                  grub_file_t file)
 | 
						|
{
 | 
						|
  grub_size_t total_length;
 | 
						|
  grub_size_t index;
 | 
						|
  grub_disk_t disk;
 | 
						|
  grub_disk_addr_t part_start;
 | 
						|
  struct blocklist *p;
 | 
						|
  char *buf;
 | 
						|
 | 
						|
  /* Sanity checks.  */
 | 
						|
  total_length = 0;
 | 
						|
  for (p = blocklists; p; p = p->next)
 | 
						|
    {
 | 
						|
      struct blocklist *q;
 | 
						|
      for (q = p->next; q; q = q->next)
 | 
						|
        {
 | 
						|
          /* Check if any pair of blocks overlap.  */
 | 
						|
          if (p->sector == q->sector)
 | 
						|
            {
 | 
						|
              /* This might be actually valid, but it is unbelievable that
 | 
						|
                 any filesystem makes such a silly allocation.  */
 | 
						|
              return grub_error (GRUB_ERR_BAD_FS, "malformed file");
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
      total_length += p->length;
 | 
						|
    }
 | 
						|
 | 
						|
  if (total_length != grub_file_size (file))
 | 
						|
    {
 | 
						|
      /* Maybe sparse, unallocated sectors. No way in GRUB.  */
 | 
						|
      return grub_error (GRUB_ERR_BAD_FILE_TYPE, "sparse file not allowed");
 | 
						|
    }
 | 
						|
 | 
						|
  /* One more sanity check. Re-read all sectors by blocklists, and compare
 | 
						|
     those with the data read via a file.  */
 | 
						|
  disk = file->device->disk;
 | 
						|
 | 
						|
  part_start = grub_partition_get_start (disk->partition);
 | 
						|
 | 
						|
  buf = grub_envblk_buffer (envblk);
 | 
						|
  for (p = blocklists, index = 0; p; index += p->length, p = p->next)
 | 
						|
    {
 | 
						|
      char blockbuf[GRUB_DISK_SECTOR_SIZE];
 | 
						|
 | 
						|
      if (grub_disk_read (disk, p->sector - part_start,
 | 
						|
                          p->offset, p->length, blockbuf))
 | 
						|
        return grub_errno;
 | 
						|
 | 
						|
      if (grub_memcmp (buf + index, blockbuf, p->length) != 0)
 | 
						|
	return grub_error (GRUB_ERR_FILE_READ_ERROR, "invalid blocklist");
 | 
						|
    }
 | 
						|
 | 
						|
  return GRUB_ERR_NONE;
 | 
						|
}
 | 
						|
 | 
						|
static int
 | 
						|
write_blocklists (grub_envblk_t envblk, struct blocklist *blocklists,
 | 
						|
                  grub_file_t file)
 | 
						|
{
 | 
						|
  char *buf;
 | 
						|
  grub_disk_t disk;
 | 
						|
  grub_disk_addr_t part_start;
 | 
						|
  struct blocklist *p;
 | 
						|
  grub_size_t index;
 | 
						|
 | 
						|
  buf = grub_envblk_buffer (envblk);
 | 
						|
  disk = file->device->disk;
 | 
						|
  part_start = grub_partition_get_start (disk->partition);
 | 
						|
 | 
						|
  index = 0;
 | 
						|
  for (p = blocklists; p; index += p->length, p = p->next)
 | 
						|
    {
 | 
						|
      if (grub_disk_write (disk, p->sector - part_start,
 | 
						|
                           p->offset, p->length, buf + index))
 | 
						|
        return 0;
 | 
						|
    }
 | 
						|
 | 
						|
  return 1;
 | 
						|
}
 | 
						|
 | 
						|
static grub_err_t
 | 
						|
grub_cmd_save_env (grub_extcmd_t cmd, int argc, char **args)
 | 
						|
{
 | 
						|
  struct grub_arg_list *state = cmd->state;
 | 
						|
  grub_file_t file;
 | 
						|
  grub_envblk_t envblk;
 | 
						|
  struct blocklist *head = 0;
 | 
						|
  struct blocklist *tail = 0;
 | 
						|
 | 
						|
  /* Store blocklists in a linked list.  */
 | 
						|
  auto void NESTED_FUNC_ATTR read_hook (grub_disk_addr_t sector,
 | 
						|
                                        unsigned offset,
 | 
						|
                                        unsigned length);
 | 
						|
  void NESTED_FUNC_ATTR read_hook (grub_disk_addr_t sector,
 | 
						|
                                   unsigned offset, unsigned length)
 | 
						|
    {
 | 
						|
      struct blocklist *block;
 | 
						|
 | 
						|
      if (offset + length > GRUB_DISK_SECTOR_SIZE)
 | 
						|
        /* Seemingly a bug.  */
 | 
						|
        return;
 | 
						|
 | 
						|
      block = grub_malloc (sizeof (*block));
 | 
						|
      if (! block)
 | 
						|
        return;
 | 
						|
 | 
						|
      block->sector = sector;
 | 
						|
      block->offset = offset;
 | 
						|
      block->length = length;
 | 
						|
 | 
						|
      /* Slightly complicated, because the list should be FIFO.  */
 | 
						|
      block->next = 0;
 | 
						|
      if (tail)
 | 
						|
        tail->next = block;
 | 
						|
      tail = block;
 | 
						|
      if (! head)
 | 
						|
        head = block;
 | 
						|
    }
 | 
						|
 | 
						|
  if (! argc)
 | 
						|
    return grub_error (GRUB_ERR_BAD_ARGUMENT, "no variable is specified");
 | 
						|
 | 
						|
  file = open_envblk_file ((state[0].set) ? state[0].arg : 0);
 | 
						|
  if (! file)
 | 
						|
    return grub_errno;
 | 
						|
 | 
						|
  if (! file->device->disk)
 | 
						|
    {
 | 
						|
      grub_file_close (file);
 | 
						|
      return grub_error (GRUB_ERR_BAD_DEVICE, "disk device required");
 | 
						|
    }
 | 
						|
 | 
						|
  file->read_hook = read_hook;
 | 
						|
  envblk = read_envblk_file (file);
 | 
						|
  file->read_hook = 0;
 | 
						|
  if (! envblk)
 | 
						|
    goto fail;
 | 
						|
 | 
						|
  if (check_blocklists (envblk, head, file))
 | 
						|
    goto fail;
 | 
						|
 | 
						|
  while (argc)
 | 
						|
    {
 | 
						|
      char *value;
 | 
						|
 | 
						|
      value = grub_env_get (args[0]);
 | 
						|
      if (value)
 | 
						|
        {
 | 
						|
          if (! grub_envblk_set (envblk, args[0], value))
 | 
						|
            {
 | 
						|
              grub_error (GRUB_ERR_BAD_ARGUMENT, "environment block too small");
 | 
						|
              goto fail;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
      argc--;
 | 
						|
      args++;
 | 
						|
    }
 | 
						|
 | 
						|
  write_blocklists (envblk, head, file);
 | 
						|
 | 
						|
 fail:
 | 
						|
  if (envblk)
 | 
						|
    grub_envblk_close (envblk);
 | 
						|
  free_blocklists (head);
 | 
						|
  grub_file_close (file);
 | 
						|
  return grub_errno;
 | 
						|
}
 | 
						|
 | 
						|
static grub_extcmd_t cmd_load, cmd_list, cmd_save;
 | 
						|
 | 
						|
GRUB_MOD_INIT(loadenv)
 | 
						|
{
 | 
						|
  cmd_load =
 | 
						|
    grub_register_extcmd ("load_env", grub_cmd_load_env,
 | 
						|
			  GRUB_COMMAND_FLAG_BOTH,
 | 
						|
			  N_("[-f FILE]"),
 | 
						|
			  N_("Load variables from environment block file."),
 | 
						|
			  options);
 | 
						|
  cmd_list =
 | 
						|
    grub_register_extcmd ("list_env", grub_cmd_list_env,
 | 
						|
			  GRUB_COMMAND_FLAG_BOTH,
 | 
						|
			  N_("[-f FILE]"),
 | 
						|
			  N_("List variables from environment block file."),
 | 
						|
			  options);
 | 
						|
  cmd_save =
 | 
						|
    grub_register_extcmd ("save_env", grub_cmd_save_env,
 | 
						|
			  GRUB_COMMAND_FLAG_BOTH,
 | 
						|
			  N_("[-f FILE] variable_name [...]"),
 | 
						|
			  N_("Save variables to environment block file."),
 | 
						|
			  options);
 | 
						|
}
 | 
						|
 | 
						|
GRUB_MOD_FINI(loadenv)
 | 
						|
{
 | 
						|
  grub_unregister_extcmd (cmd_load);
 | 
						|
  grub_unregister_extcmd (cmd_list);
 | 
						|
  grub_unregister_extcmd (cmd_save);
 | 
						|
}
 |