123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503 |
- /* 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 *e, *prefix;
- e = grub_env_get ("envfile");
- if (e)
- return grub_file_open (e);
- 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 int
- set_var (const char *name, const char *value)
- {
- grub_env_set (name, value);
- return 0;
- }
- 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;
- 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;
- }
- /* Print all variables in current context. */
- static int
- print_var (const char *name, const char *value)
- {
- grub_printf ("%s=%s\n", name, value);
- return 0;
- }
- 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;
- 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;
- }
- struct grub_cmd_save_env_closure
- {
- struct blocklist *head;
- struct blocklist *tail;
- };
- /* Store blocklists in a linked list. */
- static void
- read_hook (grub_disk_addr_t sector, unsigned offset, unsigned length,
- void *closure)
- {
- struct grub_cmd_save_env_closure *c = closure;
- 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 (c->tail)
- c->tail->next = block;
- c->tail = block;
- if (! c->head)
- c->head = block;
- }
- #ifdef GRUB_MACHINE_EMU
- #include <stdio.h>
- #include <unistd.h>
- #include <grub/util/misc.h>
- static char *
- check_host_file (char *filename)
- {
- char *buf = 0;
-
- if (! filename)
- {
- filename = grub_env_get ("envfile");
- if (! filename)
- {
- char *prefix;
- prefix = grub_env_get ("prefix");
- if (! prefix)
- return 0;
- buf = grub_xasprintf ("%s/%s", prefix, GRUB_ENVBLK_DEFCFG);
- if (! buf)
- return 0;
- filename = buf;
- }
- }
- if (filename[0] == '(')
- {
- if (grub_memcmp (filename, "(host)", 6))
- goto fail;
- filename += 6;
- }
- else
- {
- char *root;
- root = grub_env_get ("root");
- if ((! root) || (grub_strcmp (root, "host")))
- goto fail;
- }
- filename = grub_strdup (filename);
- grub_free (buf);
- return filename;
- fail:
- grub_free (buf);
- return 0;
- }
- #endif
- 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 grub_cmd_save_env_closure c;
- #ifdef GRUB_MACHINE_EMU
- char *host_file = check_host_file ((state[0].set) ? state[0].arg : 0);
- #endif
- c.head = 0;
- c.tail = 0;
- 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;
- file->closure = &c;
- envblk = read_envblk_file (file);
- file->read_hook = 0;
- if (! envblk)
- goto fail;
- #ifdef GRUB_MACHINE_EMU
- if ((host_file == NULL) && (check_blocklists (envblk, c.head, file)))
- goto fail;
- #else
- if (check_blocklists (envblk, c.head, file))
- goto fail;
- #endif
- 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++;
- }
- #ifdef GRUB_MACHINE_EMU
- if (host_file)
- {
- FILE *fp;
-
- grub_file_close (file);
- fp = fopen (host_file, "wb");
- if (! fp)
- grub_util_error ("cannot open the file %s", host_file);
- grub_free (host_file);
- if (fwrite (grub_envblk_buffer (envblk), 1,
- grub_envblk_size (envblk), fp) != grub_envblk_size (envblk))
- grub_util_error ("cannot write to the file %s", host_file);
- fsync (fileno (fp));
- fclose (fp);
- if (envblk)
- grub_envblk_close (envblk);
- return grub_errno;
- }
- #endif
- write_blocklists (envblk, c.head, file);
- fail:
- if (envblk)
- grub_envblk_close (envblk);
- free_blocklists (c.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);
- }
|