| /* linux.c - boot Linux */ |
| /* |
| * GRUB -- GRand Unified Bootloader |
| * Copyright (C) 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/elf.h> |
| #include <grub/elfload.h> |
| #include <grub/loader.h> |
| #include <grub/dl.h> |
| #include <grub/mm.h> |
| #include <grub/misc.h> |
| #include <grub/ieee1275/ieee1275.h> |
| #include <grub/command.h> |
| #include <grub/i18n.h> |
| #include <grub/memory.h> |
| #include <grub/lib/cmdline.h> |
| #include <grub/cache.h> |
| #include <grub/linux.h> |
| |
| GRUB_MOD_LICENSE ("GPLv3+"); |
| |
| #define ELF32_LOADMASK (0xc0000000UL) |
| #define ELF64_LOADMASK (0xc000000000000000ULL) |
| |
| static grub_dl_t my_mod; |
| |
| static int loaded; |
| |
| static grub_addr_t initrd_addr; |
| static grub_size_t initrd_size; |
| |
| static grub_addr_t linux_addr; |
| static grub_addr_t linux_entry; |
| static grub_size_t linux_size; |
| |
| static char *linux_args; |
| |
| typedef void (*kernel_entry_t) (void *, unsigned long, int (void *), |
| unsigned long, unsigned long); |
| |
| /* Context for grub_linux_claimmap_iterate. */ |
| struct grub_linux_claimmap_iterate_ctx |
| { |
| grub_addr_t target; |
| grub_size_t size; |
| grub_size_t align; |
| grub_addr_t found_addr; |
| }; |
| |
| /* Helper for grub_linux_claimmap_iterate. */ |
| static int |
| alloc_mem (grub_uint64_t addr, grub_uint64_t len, grub_memory_type_t type, |
| void *data) |
| { |
| struct grub_linux_claimmap_iterate_ctx *ctx = data; |
| |
| grub_uint64_t end = addr + len; |
| addr = ALIGN_UP (addr, ctx->align); |
| ctx->target = ALIGN_UP (ctx->target, ctx->align); |
| |
| /* Target above the memory chunk. */ |
| if (type != GRUB_MEMORY_AVAILABLE || ctx->target > end) |
| return 0; |
| |
| /* Target inside the memory chunk. */ |
| if (ctx->target >= addr && ctx->target < end && |
| ctx->size <= end - ctx->target) |
| { |
| if (grub_claimmap (ctx->target, ctx->size) == GRUB_ERR_NONE) |
| { |
| ctx->found_addr = ctx->target; |
| return 1; |
| } |
| grub_print_error (); |
| } |
| /* Target below the memory chunk. */ |
| if (ctx->target < addr && addr + ctx->size <= end) |
| { |
| if (grub_claimmap (addr, ctx->size) == GRUB_ERR_NONE) |
| { |
| ctx->found_addr = addr; |
| return 1; |
| } |
| grub_print_error (); |
| } |
| return 0; |
| } |
| |
| static grub_addr_t |
| grub_linux_claimmap_iterate (grub_addr_t target, grub_size_t size, |
| grub_size_t align) |
| { |
| struct grub_linux_claimmap_iterate_ctx ctx = { |
| .target = target, |
| .size = size, |
| .align = align, |
| .found_addr = (grub_addr_t) -1 |
| }; |
| |
| if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_FORCE_CLAIM)) |
| { |
| grub_uint64_t addr = target; |
| if (addr < GRUB_IEEE1275_STATIC_HEAP_START |
| + GRUB_IEEE1275_STATIC_HEAP_LEN) |
| addr = GRUB_IEEE1275_STATIC_HEAP_START |
| + GRUB_IEEE1275_STATIC_HEAP_LEN; |
| addr = ALIGN_UP (addr, align); |
| if (grub_claimmap (addr, size) == GRUB_ERR_NONE) |
| return addr; |
| return (grub_addr_t) -1; |
| } |
| |
| |
| grub_machine_mmap_iterate (alloc_mem, &ctx); |
| |
| return ctx.found_addr; |
| } |
| |
| static grub_err_t |
| grub_linux_boot (void) |
| { |
| kernel_entry_t linuxmain; |
| grub_ssize_t actual; |
| |
| grub_arch_sync_caches ((void *) linux_addr, linux_size); |
| /* Set the command line arguments. */ |
| grub_ieee1275_set_property (grub_ieee1275_chosen, "bootargs", linux_args, |
| grub_strlen (linux_args) + 1, &actual); |
| |
| grub_dprintf ("loader", "Entry point: 0x%x\n", linux_entry); |
| grub_dprintf ("loader", "Initrd at: 0x%x, size 0x%x\n", initrd_addr, |
| initrd_size); |
| grub_dprintf ("loader", "Boot arguments: %s\n", linux_args); |
| grub_dprintf ("loader", "Jumping to Linux...\n"); |
| |
| /* Boot the kernel. */ |
| linuxmain = (kernel_entry_t) linux_entry; |
| linuxmain ((void *) initrd_addr, initrd_size, grub_ieee1275_entry_fn, 0, 0); |
| |
| return GRUB_ERR_NONE; |
| } |
| |
| static grub_err_t |
| grub_linux_release_mem (void) |
| { |
| grub_free (linux_args); |
| linux_args = 0; |
| |
| if (linux_addr && grub_ieee1275_release (linux_addr, linux_size)) |
| return grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot release memory"); |
| |
| if (initrd_addr && grub_ieee1275_release (initrd_addr, initrd_size)) |
| return grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot release memory"); |
| |
| linux_addr = 0; |
| initrd_addr = 0; |
| |
| return GRUB_ERR_NONE; |
| } |
| |
| static grub_err_t |
| grub_linux_unload (void) |
| { |
| grub_err_t err; |
| |
| err = grub_linux_release_mem (); |
| grub_dl_unref (my_mod); |
| |
| loaded = 0; |
| |
| return err; |
| } |
| |
| static grub_err_t |
| grub_linux_load32 (grub_elf_t elf, const char *filename) |
| { |
| Elf32_Addr base_addr; |
| grub_addr_t seg_addr; |
| grub_uint32_t align; |
| grub_uint32_t offset; |
| Elf32_Addr entry; |
| |
| linux_size = grub_elf32_size (elf, &base_addr, &align); |
| if (linux_size == 0) |
| return grub_errno; |
| /* Pad it; the kernel scribbles over memory beyond its load address. */ |
| linux_size += 0x100000; |
| |
| /* Linux's entry point incorrectly contains a virtual address. */ |
| entry = elf->ehdr.ehdr32.e_entry & ~ELF32_LOADMASK; |
| |
| /* Linux's incorrectly contains a virtual address. */ |
| base_addr &= ~ELF32_LOADMASK; |
| offset = entry - base_addr; |
| |
| /* On some systems, firmware occupies the memory we're trying to use. |
| * Happily, Linux can be loaded anywhere (it relocates itself). Iterate |
| * until we find an open area. */ |
| seg_addr = grub_linux_claimmap_iterate (base_addr & ~ELF32_LOADMASK, linux_size, align); |
| if (seg_addr == (grub_addr_t) -1) |
| return grub_error (GRUB_ERR_OUT_OF_MEMORY, "couldn't claim memory"); |
| |
| linux_entry = seg_addr + offset; |
| linux_addr = seg_addr; |
| |
| /* Now load the segments into the area we claimed. */ |
| return grub_elf32_load (elf, filename, (void *) (seg_addr - base_addr), GRUB_ELF_LOAD_FLAGS_30BITS, 0, 0); |
| } |
| |
| static grub_err_t |
| grub_linux_load64 (grub_elf_t elf, const char *filename) |
| { |
| Elf64_Addr base_addr; |
| grub_addr_t seg_addr; |
| grub_uint64_t align; |
| grub_uint64_t offset; |
| Elf64_Addr entry; |
| |
| linux_size = grub_elf64_size (elf, &base_addr, &align); |
| if (linux_size == 0) |
| return grub_errno; |
| /* Pad it; the kernel scribbles over memory beyond its load address. */ |
| linux_size += 0x100000; |
| |
| base_addr &= ~ELF64_LOADMASK; |
| entry = elf->ehdr.ehdr64.e_entry & ~ELF64_LOADMASK; |
| offset = entry - base_addr; |
| /* Linux's incorrectly contains a virtual address. */ |
| |
| /* On some systems, firmware occupies the memory we're trying to use. |
| * Happily, Linux can be loaded anywhere (it relocates itself). Iterate |
| * until we find an open area. */ |
| seg_addr = grub_linux_claimmap_iterate (base_addr & ~ELF64_LOADMASK, linux_size, align); |
| if (seg_addr == (grub_addr_t) -1) |
| return grub_error (GRUB_ERR_OUT_OF_MEMORY, "couldn't claim memory"); |
| |
| linux_entry = seg_addr + offset; |
| linux_addr = seg_addr; |
| |
| /* Now load the segments into the area we claimed. */ |
| return grub_elf64_load (elf, filename, (void *) (grub_addr_t) (seg_addr - base_addr), GRUB_ELF_LOAD_FLAGS_62BITS, 0, 0); |
| } |
| |
| static grub_err_t |
| grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)), |
| int argc, char *argv[]) |
| { |
| grub_elf_t elf = 0; |
| int size; |
| |
| grub_dl_ref (my_mod); |
| |
| if (argc == 0) |
| { |
| grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); |
| goto out; |
| } |
| |
| elf = grub_elf_open (argv[0]); |
| if (! elf) |
| goto out; |
| |
| if (elf->ehdr.ehdr32.e_type != ET_EXEC && elf->ehdr.ehdr32.e_type != ET_DYN) |
| { |
| grub_error (GRUB_ERR_UNKNOWN_OS, |
| N_("this ELF file is not of the right type")); |
| goto out; |
| } |
| |
| /* Release the previously used memory. */ |
| grub_loader_unset (); |
| |
| if (grub_elf_is_elf32 (elf)) |
| grub_linux_load32 (elf, argv[0]); |
| else |
| if (grub_elf_is_elf64 (elf)) |
| grub_linux_load64 (elf, argv[0]); |
| else |
| { |
| grub_error (GRUB_ERR_BAD_FILE_TYPE, N_("invalid arch-dependent ELF magic")); |
| goto out; |
| } |
| |
| size = grub_loader_cmdline_size(argc, argv); |
| linux_args = grub_malloc (size + sizeof (LINUX_IMAGE)); |
| if (! linux_args) |
| goto out; |
| |
| /* Create kernel command line. */ |
| grub_memcpy (linux_args, LINUX_IMAGE, sizeof (LINUX_IMAGE)); |
| if (grub_create_loader_cmdline (argc, argv, linux_args + sizeof (LINUX_IMAGE) - 1, |
| size)) |
| goto out; |
| |
| out: |
| |
| if (elf) |
| grub_elf_close (elf); |
| |
| if (grub_errno != GRUB_ERR_NONE) |
| { |
| grub_linux_release_mem (); |
| grub_dl_unref (my_mod); |
| loaded = 0; |
| } |
| else |
| { |
| grub_loader_set (grub_linux_boot, grub_linux_unload, 1); |
| initrd_addr = 0; |
| loaded = 1; |
| } |
| |
| return grub_errno; |
| } |
| |
| static grub_err_t |
| grub_cmd_initrd (grub_command_t cmd __attribute__ ((unused)), |
| int argc, char *argv[]) |
| { |
| grub_size_t size = 0; |
| grub_addr_t first_addr; |
| grub_addr_t addr; |
| struct grub_linux_initrd_context initrd_ctx = { 0, 0, 0 }; |
| |
| if (argc == 0) |
| { |
| grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected")); |
| goto fail; |
| } |
| |
| if (!loaded) |
| { |
| grub_error (GRUB_ERR_BAD_ARGUMENT, N_("you need to load the kernel first")); |
| goto fail; |
| } |
| |
| if (grub_initrd_init (argc, argv, &initrd_ctx)) |
| goto fail; |
| |
| size = grub_get_initrd_size (&initrd_ctx); |
| |
| first_addr = linux_addr + linux_size; |
| |
| /* Attempt to claim at a series of addresses until successful in |
| the same way that grub_rescue_cmd_linux does. */ |
| addr = grub_linux_claimmap_iterate (first_addr, size, 0x100000); |
| if (addr == (grub_addr_t) -1) |
| goto fail; |
| |
| grub_dprintf ("loader", "Loading initrd at 0x%x, size 0x%x\n", addr, size); |
| |
| if (grub_initrd_load (&initrd_ctx, argv, (void *) addr)) |
| goto fail; |
| |
| initrd_addr = addr; |
| initrd_size = size; |
| |
| fail: |
| grub_initrd_close (&initrd_ctx); |
| |
| return grub_errno; |
| } |
| |
| static grub_command_t cmd_linux, cmd_initrd; |
| |
| GRUB_MOD_INIT(linux) |
| { |
| cmd_linux = grub_register_command ("linux", grub_cmd_linux, |
| 0, N_("Load Linux.")); |
| cmd_initrd = grub_register_command ("initrd", grub_cmd_initrd, |
| 0, N_("Load initrd.")); |
| my_mod = mod; |
| } |
| |
| GRUB_MOD_FINI(linux) |
| { |
| grub_unregister_command (cmd_linux); |
| grub_unregister_command (cmd_initrd); |
| } |