123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246 |
- /* boot.c - command to boot an operating system */
- /*
- * GRUB -- GRand Unified Bootloader
- * Copyright (C) 2002,2003,2004,2005,2007,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/normal.h>
- #include <grub/dl.h>
- #include <grub/misc.h>
- #include <grub/loader.h>
- #include <grub/kernel.h>
- #include <grub/mm.h>
- #include <grub/i18n.h>
- GRUB_MOD_LICENSE ("GPLv3+");
- static grub_err_t (*grub_loader_boot_func) (void *context);
- static grub_err_t (*grub_loader_unload_func) (void *context);
- static void *grub_loader_context;
- static int grub_loader_flags;
- struct grub_simple_loader_hooks
- {
- grub_err_t (*boot) (void);
- grub_err_t (*unload) (void);
- };
- /* Don't heap allocate this to avoid making grub_loader_set() fallible. */
- static struct grub_simple_loader_hooks simple_loader_hooks;
- struct grub_preboot
- {
- grub_err_t (*preboot_func) (int);
- grub_err_t (*preboot_rest_func) (void);
- grub_loader_preboot_hook_prio_t prio;
- struct grub_preboot *next;
- struct grub_preboot *prev;
- };
- static int grub_loader_loaded;
- static struct grub_preboot *preboots_head = 0,
- *preboots_tail = 0;
- static grub_err_t
- grub_simple_boot_hook (void *context)
- {
- struct grub_simple_loader_hooks *hooks;
- hooks = (struct grub_simple_loader_hooks *) context;
- return hooks->boot ();
- }
- static grub_err_t
- grub_simple_unload_hook (void *context)
- {
- struct grub_simple_loader_hooks *hooks;
- grub_err_t ret;
- hooks = (struct grub_simple_loader_hooks *) context;
- ret = hooks->unload ();
- grub_memset (hooks, 0, sizeof (*hooks));
- return ret;
- }
- int
- grub_loader_is_loaded (void)
- {
- return grub_loader_loaded;
- }
- /* Register a preboot hook. */
- struct grub_preboot *
- grub_loader_register_preboot_hook (grub_err_t (*preboot_func) (int flags),
- grub_err_t (*preboot_rest_func) (void),
- grub_loader_preboot_hook_prio_t prio)
- {
- struct grub_preboot *cur, *new_preboot;
- if (! preboot_func && ! preboot_rest_func)
- return 0;
- new_preboot = (struct grub_preboot *)
- grub_malloc (sizeof (struct grub_preboot));
- if (! new_preboot)
- return 0;
- new_preboot->preboot_func = preboot_func;
- new_preboot->preboot_rest_func = preboot_rest_func;
- new_preboot->prio = prio;
- for (cur = preboots_head; cur && cur->prio > prio; cur = cur->next);
- if (cur)
- {
- new_preboot->next = cur;
- new_preboot->prev = cur->prev;
- cur->prev = new_preboot;
- }
- else
- {
- new_preboot->next = 0;
- new_preboot->prev = preboots_tail;
- preboots_tail = new_preboot;
- }
- if (new_preboot->prev)
- new_preboot->prev->next = new_preboot;
- else
- preboots_head = new_preboot;
- return new_preboot;
- }
- void
- grub_loader_unregister_preboot_hook (struct grub_preboot *hnd)
- {
- struct grub_preboot *preb = hnd;
- if (preb->next)
- preb->next->prev = preb->prev;
- else
- preboots_tail = preb->prev;
- if (preb->prev)
- preb->prev->next = preb->next;
- else
- preboots_head = preb->next;
- grub_free (preb);
- }
- void
- grub_loader_set_ex (grub_err_t (*boot) (void *context),
- grub_err_t (*unload) (void *context),
- void *context,
- int flags)
- {
- if (grub_loader_loaded && grub_loader_unload_func)
- grub_loader_unload_func (grub_loader_context);
- grub_loader_boot_func = boot;
- grub_loader_unload_func = unload;
- grub_loader_context = context;
- grub_loader_flags = flags;
- grub_loader_loaded = 1;
- }
- void
- grub_loader_set (grub_err_t (*boot) (void),
- grub_err_t (*unload) (void),
- int flags)
- {
- grub_loader_set_ex (grub_simple_boot_hook,
- grub_simple_unload_hook,
- &simple_loader_hooks,
- flags);
- simple_loader_hooks.boot = boot;
- simple_loader_hooks.unload = unload;
- }
- void
- grub_loader_unset(void)
- {
- if (grub_loader_loaded && grub_loader_unload_func)
- grub_loader_unload_func (grub_loader_context);
- grub_loader_boot_func = 0;
- grub_loader_unload_func = 0;
- grub_loader_context = 0;
- grub_loader_loaded = 0;
- }
- grub_err_t
- grub_loader_boot (void)
- {
- grub_err_t err = GRUB_ERR_NONE;
- struct grub_preboot *cur;
- if (! grub_loader_loaded)
- return grub_error (GRUB_ERR_NO_KERNEL,
- N_("you need to load the kernel first"));
- grub_machine_fini (grub_loader_flags);
- for (cur = preboots_head; cur; cur = cur->next)
- {
- err = cur->preboot_func (grub_loader_flags);
- if (err)
- {
- for (cur = cur->prev; cur; cur = cur->prev)
- cur->preboot_rest_func ();
- return err;
- }
- }
- err = (grub_loader_boot_func) (grub_loader_context);
- for (cur = preboots_tail; cur; cur = cur->prev)
- if (! err)
- err = cur->preboot_rest_func ();
- else
- cur->preboot_rest_func ();
- return err;
- }
- /* boot */
- static grub_err_t
- grub_cmd_boot (struct grub_command *cmd __attribute__ ((unused)),
- int argc __attribute__ ((unused)),
- char *argv[] __attribute__ ((unused)))
- {
- return grub_loader_boot ();
- }
- static grub_command_t cmd_boot;
- GRUB_MOD_INIT(boot)
- {
- cmd_boot =
- grub_register_command ("boot", grub_cmd_boot,
- 0, N_("Boot an operating system."));
- }
- GRUB_MOD_FINI(boot)
- {
- grub_unregister_command (cmd_boot);
- }
|