609 lines
17 KiB
C
609 lines
17 KiB
C
/*
|
|
* VAS_EBOOT -- GRand Unified Bootloader
|
|
* Copyright (C) 2008,2010 Free Software Foundation, Inc.
|
|
*
|
|
* VAS_EBOOT 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.
|
|
*
|
|
* VAS_EBOOT 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 VAS_EBOOT. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <VasEBoot/loader.h>
|
|
#include <VasEBoot/file.h>
|
|
#include <VasEBoot/disk.h>
|
|
#include <VasEBoot/err.h>
|
|
#include <VasEBoot/misc.h>
|
|
#include <VasEBoot/types.h>
|
|
#include <VasEBoot/command.h>
|
|
#include <VasEBoot/dl.h>
|
|
#include <VasEBoot/mm.h>
|
|
#include <VasEBoot/cache.h>
|
|
#include <VasEBoot/kernel.h>
|
|
#include <VasEBoot/efi/api.h>
|
|
#include <VasEBoot/efi/efi.h>
|
|
#include <VasEBoot/elf.h>
|
|
#include <VasEBoot/i18n.h>
|
|
#include <VasEBoot/env.h>
|
|
#include <VasEBoot/linux.h>
|
|
#include <VasEBoot/verify.h>
|
|
|
|
VAS_EBOOT_MOD_LICENSE ("GPLv3+");
|
|
|
|
#pragma GCC diagnostic ignored "-Wcast-align"
|
|
|
|
#define ALIGN_MIN (256*1024*1024)
|
|
|
|
#define VAS_EBOOT_ELF_SEARCH 1024
|
|
|
|
#define BOOT_PARAM_SIZE 16384
|
|
|
|
struct ia64_boot_param
|
|
{
|
|
VasEBoot_uint64_t command_line; /* physical address of command line. */
|
|
VasEBoot_uint64_t efi_systab; /* physical address of EFI system table */
|
|
VasEBoot_uint64_t efi_memmap; /* physical address of EFI memory map */
|
|
VasEBoot_uint64_t efi_memmap_size; /* size of EFI memory map */
|
|
VasEBoot_uint64_t efi_memdesc_size; /* size of an EFI memory map descriptor */
|
|
VasEBoot_uint32_t efi_memdesc_version; /* memory descriptor version */
|
|
struct
|
|
{
|
|
VasEBoot_uint16_t num_cols; /* number of columns on console output dev */
|
|
VasEBoot_uint16_t num_rows; /* number of rows on console output device */
|
|
VasEBoot_uint16_t orig_x; /* cursor's x position */
|
|
VasEBoot_uint16_t orig_y; /* cursor's y position */
|
|
} console_info;
|
|
VasEBoot_uint64_t fpswa; /* physical address of the fpswa interface */
|
|
VasEBoot_uint64_t initrd_start;
|
|
VasEBoot_uint64_t initrd_size;
|
|
};
|
|
|
|
typedef struct
|
|
{
|
|
VasEBoot_uint32_t revision;
|
|
VasEBoot_uint32_t reserved;
|
|
void *fpswa;
|
|
} fpswa_interface_t;
|
|
static fpswa_interface_t *fpswa;
|
|
|
|
#define NEXT_MEMORY_DESCRIPTOR(desc, size) \
|
|
((VasEBoot_efi_memory_descriptor_t *) ((char *) (desc) + (size)))
|
|
|
|
static VasEBoot_dl_t my_mod;
|
|
|
|
static int loaded;
|
|
|
|
/* Kernel base and size. */
|
|
static void *kernel_mem;
|
|
static VasEBoot_efi_uintn_t kernel_pages;
|
|
static VasEBoot_uint64_t entry;
|
|
|
|
/* Initrd base and size. */
|
|
static void *initrd_mem;
|
|
static VasEBoot_efi_uintn_t initrd_pages;
|
|
static VasEBoot_efi_uintn_t initrd_size;
|
|
|
|
static struct ia64_boot_param *boot_param;
|
|
static VasEBoot_efi_uintn_t boot_param_pages;
|
|
|
|
static inline VasEBoot_size_t
|
|
page_align (VasEBoot_size_t size)
|
|
{
|
|
return (size + (1 << 12) - 1) & (~((1 << 12) - 1));
|
|
}
|
|
|
|
static void
|
|
query_fpswa (void)
|
|
{
|
|
VasEBoot_efi_handle_t fpswa_image;
|
|
VasEBoot_efi_boot_services_t *bs;
|
|
VasEBoot_efi_status_t status;
|
|
VasEBoot_efi_uintn_t size;
|
|
static const VasEBoot_guid_t fpswa_protocol =
|
|
{ 0xc41b6531, 0x97b9, 0x11d3,
|
|
{0x9a, 0x29, 0x0, 0x90, 0x27, 0x3f, 0xc1, 0x4d} };
|
|
|
|
if (fpswa != NULL)
|
|
return;
|
|
|
|
size = sizeof(VasEBoot_efi_handle_t);
|
|
|
|
bs = VasEBoot_efi_system_table->boot_services;
|
|
status = bs->locate_handle (VAS_EBOOT_EFI_BY_PROTOCOL,
|
|
(void *) &fpswa_protocol,
|
|
NULL, &size, &fpswa_image);
|
|
if (status != VAS_EBOOT_EFI_SUCCESS)
|
|
{
|
|
VasEBoot_printf ("%s\n", _("Could not locate FPSWA driver"));
|
|
return;
|
|
}
|
|
status = bs->handle_protocol (fpswa_image,
|
|
(void *) &fpswa_protocol, (void *) &fpswa);
|
|
if (status != VAS_EBOOT_EFI_SUCCESS)
|
|
{
|
|
VasEBoot_printf ("%s\n",
|
|
_("FPSWA protocol wasn't able to find the interface"));
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
free_pages (void)
|
|
{
|
|
if (kernel_mem)
|
|
{
|
|
VasEBoot_efi_free_pages ((VasEBoot_addr_t) kernel_mem, kernel_pages);
|
|
kernel_mem = 0;
|
|
}
|
|
|
|
if (initrd_mem)
|
|
{
|
|
VasEBoot_efi_free_pages ((VasEBoot_addr_t) initrd_mem, initrd_pages);
|
|
initrd_mem = 0;
|
|
}
|
|
|
|
if (boot_param)
|
|
{
|
|
/* Free bootparam. */
|
|
VasEBoot_efi_free_pages ((VasEBoot_efi_physical_address_t) boot_param,
|
|
boot_param_pages);
|
|
boot_param = 0;
|
|
}
|
|
}
|
|
|
|
static void *
|
|
allocate_pages (VasEBoot_uint64_t align, VasEBoot_uint64_t size_pages,
|
|
VasEBoot_uint64_t nobase)
|
|
{
|
|
VasEBoot_uint64_t size;
|
|
VasEBoot_efi_uintn_t desc_size;
|
|
VasEBoot_efi_memory_descriptor_t *mmap, *mmap_end;
|
|
VasEBoot_efi_uintn_t mmap_size, tmp_mmap_size;
|
|
VasEBoot_efi_memory_descriptor_t *desc;
|
|
void *mem = NULL;
|
|
|
|
size = size_pages << 12;
|
|
|
|
mmap_size = VasEBoot_efi_find_mmap_size ();
|
|
if (!mmap_size)
|
|
return 0;
|
|
|
|
/* Read the memory map temporarily, to find free space. */
|
|
mmap = VasEBoot_malloc (mmap_size);
|
|
if (! mmap)
|
|
return 0;
|
|
|
|
tmp_mmap_size = mmap_size;
|
|
if (VasEBoot_efi_get_memory_map (&tmp_mmap_size, mmap, 0, &desc_size, 0) <= 0)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_IO, "cannot get memory map");
|
|
goto fail;
|
|
}
|
|
|
|
mmap_end = NEXT_MEMORY_DESCRIPTOR (mmap, tmp_mmap_size);
|
|
|
|
/* First, find free pages for the real mode code
|
|
and the memory map buffer. */
|
|
for (desc = mmap;
|
|
desc < mmap_end;
|
|
desc = NEXT_MEMORY_DESCRIPTOR (desc, desc_size))
|
|
{
|
|
VasEBoot_uint64_t start, end;
|
|
VasEBoot_uint64_t aligned_start;
|
|
|
|
if (desc->type != VAS_EBOOT_EFI_CONVENTIONAL_MEMORY)
|
|
continue;
|
|
|
|
start = desc->physical_start;
|
|
end = start + (desc->num_pages << 12);
|
|
/* Align is a power of 2. */
|
|
aligned_start = (start + align - 1) & ~(align - 1);
|
|
if (aligned_start + size > end)
|
|
continue;
|
|
if (aligned_start == nobase)
|
|
aligned_start += align;
|
|
if (aligned_start + size > end)
|
|
continue;
|
|
mem = VasEBoot_efi_allocate_fixed (aligned_start, size_pages);
|
|
if (! mem)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_OUT_OF_MEMORY, "cannot allocate memory");
|
|
goto fail;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if (! mem)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_OUT_OF_MEMORY, "cannot allocate memory");
|
|
goto fail;
|
|
}
|
|
|
|
VasEBoot_free (mmap);
|
|
return mem;
|
|
|
|
fail:
|
|
VasEBoot_free (mmap);
|
|
free_pages ();
|
|
return 0;
|
|
}
|
|
|
|
static void
|
|
set_boot_param_console (void)
|
|
{
|
|
VasEBoot_efi_simple_text_output_interface_t *conout;
|
|
VasEBoot_efi_uintn_t cols, rows;
|
|
|
|
conout = VasEBoot_efi_system_table->con_out;
|
|
if (conout->query_mode (conout, conout->mode->mode, &cols, &rows)
|
|
!= VAS_EBOOT_EFI_SUCCESS)
|
|
return;
|
|
|
|
VasEBoot_dprintf ("linux",
|
|
"Console info: cols=%lu rows=%lu x=%u y=%u\n",
|
|
cols, rows,
|
|
conout->mode->cursor_column, conout->mode->cursor_row);
|
|
|
|
boot_param->console_info.num_cols = cols;
|
|
boot_param->console_info.num_rows = rows;
|
|
boot_param->console_info.orig_x = conout->mode->cursor_column;
|
|
boot_param->console_info.orig_y = conout->mode->cursor_row;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_linux_boot (void)
|
|
{
|
|
VasEBoot_efi_uintn_t mmap_size;
|
|
VasEBoot_efi_uintn_t map_key;
|
|
VasEBoot_efi_uintn_t desc_size;
|
|
VasEBoot_efi_uint32_t desc_version;
|
|
VasEBoot_efi_memory_descriptor_t *mmap_buf;
|
|
VasEBoot_err_t err;
|
|
|
|
/* FPSWA. */
|
|
query_fpswa ();
|
|
boot_param->fpswa = (VasEBoot_uint64_t)fpswa;
|
|
|
|
/* Initrd. */
|
|
boot_param->initrd_start = (VasEBoot_uint64_t)initrd_mem;
|
|
boot_param->initrd_size = (VasEBoot_uint64_t)initrd_size;
|
|
|
|
set_boot_param_console ();
|
|
|
|
VasEBoot_dprintf ("linux", "Jump to %016lx\n", entry);
|
|
|
|
/* MDT.
|
|
Must be done after VasEBoot_machine_fini because map_key is used by
|
|
exit_boot_services. */
|
|
mmap_size = VasEBoot_efi_find_mmap_size ();
|
|
if (! mmap_size)
|
|
return VasEBoot_errno;
|
|
mmap_buf = VasEBoot_efi_allocate_any_pages (page_align (mmap_size) >> 12);
|
|
if (! mmap_buf)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_IO, "cannot allocate memory map");
|
|
err = VasEBoot_efi_finish_boot_services (&mmap_size, mmap_buf, &map_key,
|
|
&desc_size, &desc_version);
|
|
if (err)
|
|
return err;
|
|
|
|
boot_param->efi_memmap = (VasEBoot_uint64_t)mmap_buf;
|
|
boot_param->efi_memmap_size = mmap_size;
|
|
boot_param->efi_memdesc_size = desc_size;
|
|
boot_param->efi_memdesc_version = desc_version;
|
|
|
|
/* See you next boot. */
|
|
asm volatile ("mov r28=%1; br.sptk.few %0" :: "b"(entry),"r"(boot_param));
|
|
|
|
/* Never reach here. */
|
|
return VAS_EBOOT_ERR_NONE;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_linux_unload (void)
|
|
{
|
|
free_pages ();
|
|
VasEBoot_dl_unref (my_mod);
|
|
loaded = 0;
|
|
return VAS_EBOOT_ERR_NONE;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_load_elf64 (VasEBoot_file_t file, void *buffer, const char *filename)
|
|
{
|
|
Elf64_Ehdr *ehdr = (Elf64_Ehdr *) buffer;
|
|
Elf64_Phdr *phdr;
|
|
int i;
|
|
VasEBoot_uint64_t low_addr;
|
|
VasEBoot_uint64_t high_addr;
|
|
VasEBoot_uint64_t align;
|
|
VasEBoot_uint64_t reloc_offset;
|
|
const char *relocate;
|
|
|
|
if (ehdr->e_ident[EI_MAG0] != ELFMAG0
|
|
|| ehdr->e_ident[EI_MAG1] != ELFMAG1
|
|
|| ehdr->e_ident[EI_MAG2] != ELFMAG2
|
|
|| ehdr->e_ident[EI_MAG3] != ELFMAG3
|
|
|| ehdr->e_ident[EI_DATA] != ELFDATA2LSB)
|
|
return VasEBoot_error(VAS_EBOOT_ERR_UNKNOWN_OS,
|
|
N_("invalid arch-independent ELF magic"));
|
|
|
|
if (ehdr->e_ident[EI_CLASS] != ELFCLASS64
|
|
|| ehdr->e_version != EV_CURRENT
|
|
|| ehdr->e_machine != EM_IA_64)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_UNKNOWN_OS,
|
|
N_("invalid arch-dependent ELF magic"));
|
|
|
|
if (ehdr->e_type != ET_EXEC)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_UNKNOWN_OS,
|
|
N_("this ELF file is not of the right type"));
|
|
|
|
/* FIXME: Should we support program headers at strange locations? */
|
|
if (ehdr->e_phoff + ehdr->e_phnum * ehdr->e_phentsize > VAS_EBOOT_ELF_SEARCH)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_BAD_OS, "program header at a too high offset");
|
|
|
|
entry = ehdr->e_entry;
|
|
|
|
/* Compute low, high and align addresses. */
|
|
low_addr = ~0UL;
|
|
high_addr = 0;
|
|
align = 0;
|
|
for (i = 0; i < ehdr->e_phnum; i++)
|
|
{
|
|
phdr = (Elf64_Phdr *) ((char *) buffer + ehdr->e_phoff
|
|
+ i * ehdr->e_phentsize);
|
|
if (phdr->p_type == PT_LOAD)
|
|
{
|
|
if (phdr->p_paddr < low_addr)
|
|
low_addr = phdr->p_paddr;
|
|
if (phdr->p_paddr + phdr->p_memsz > high_addr)
|
|
high_addr = phdr->p_paddr + phdr->p_memsz;
|
|
if (phdr->p_align > align)
|
|
align = phdr->p_align;
|
|
}
|
|
}
|
|
|
|
if (align < ALIGN_MIN)
|
|
align = ALIGN_MIN;
|
|
|
|
if (high_addr == 0)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_BAD_OS, "no program entries");
|
|
|
|
kernel_pages = page_align (high_addr - low_addr) >> 12;
|
|
|
|
/* Undocumented on purpose. */
|
|
relocate = VasEBoot_env_get ("linux_relocate");
|
|
if (!relocate || VasEBoot_strcmp (relocate, "force") != 0)
|
|
{
|
|
kernel_mem = VasEBoot_efi_allocate_fixed (low_addr, kernel_pages);
|
|
reloc_offset = 0;
|
|
VasEBoot_errno = VAS_EBOOT_ERR_NONE;
|
|
}
|
|
/* Try to relocate. */
|
|
if (! kernel_mem && (!relocate || VasEBoot_strcmp (relocate, "off") != 0))
|
|
{
|
|
kernel_mem = allocate_pages (align, kernel_pages, low_addr);
|
|
if (kernel_mem)
|
|
{
|
|
reloc_offset = (VasEBoot_uint64_t)kernel_mem - low_addr;
|
|
VasEBoot_dprintf ("linux", " Relocated at %p (offset=%016lx)\n",
|
|
kernel_mem, reloc_offset);
|
|
entry += reloc_offset;
|
|
}
|
|
}
|
|
if (! kernel_mem)
|
|
return VasEBoot_error (VAS_EBOOT_ERR_OUT_OF_MEMORY,
|
|
"cannot allocate memory for OS");
|
|
|
|
/* Load every loadable segment in memory. */
|
|
for (i = 0; i < ehdr->e_phnum; i++)
|
|
{
|
|
phdr = (Elf64_Phdr *) ((char *) buffer + ehdr->e_phoff
|
|
+ i * ehdr->e_phentsize);
|
|
if (phdr->p_type == PT_LOAD)
|
|
{
|
|
VasEBoot_dprintf ("linux", " [paddr=%lx load=%lx memsz=%08lx "
|
|
"off=%lx flags=%x]\n",
|
|
phdr->p_paddr, phdr->p_paddr + reloc_offset,
|
|
phdr->p_memsz, phdr->p_offset, phdr->p_flags);
|
|
|
|
if (VasEBoot_file_seek (file, phdr->p_offset) == (VasEBoot_off_t)-1)
|
|
return VasEBoot_errno;
|
|
|
|
if (VasEBoot_file_read (file, (void *) (phdr->p_paddr + reloc_offset),
|
|
phdr->p_filesz)
|
|
!= (VasEBoot_ssize_t) phdr->p_filesz)
|
|
{
|
|
if (!VasEBoot_errno)
|
|
VasEBoot_error (VAS_EBOOT_ERR_BAD_OS, N_("premature end of file %s"),
|
|
filename);
|
|
return VasEBoot_errno;
|
|
}
|
|
|
|
if (phdr->p_filesz < phdr->p_memsz)
|
|
VasEBoot_memset
|
|
((char *)(phdr->p_paddr + reloc_offset + phdr->p_filesz),
|
|
0, phdr->p_memsz - phdr->p_filesz);
|
|
|
|
/* Sync caches if necessary. */
|
|
if (phdr->p_flags & PF_X)
|
|
VasEBoot_arch_sync_caches
|
|
((void *)(phdr->p_paddr + reloc_offset), phdr->p_memsz);
|
|
}
|
|
}
|
|
loaded = 1;
|
|
return 0;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_cmd_linux (VasEBoot_command_t cmd __attribute__ ((unused)),
|
|
int argc, char *argv[])
|
|
{
|
|
VasEBoot_file_t file = 0;
|
|
char buffer[VAS_EBOOT_ELF_SEARCH];
|
|
char *cmdline, *p;
|
|
VasEBoot_ssize_t len;
|
|
int i;
|
|
|
|
VasEBoot_dl_ref (my_mod);
|
|
|
|
VasEBoot_loader_unset ();
|
|
|
|
if (argc == 0)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_BAD_ARGUMENT, N_("filename expected"));
|
|
goto fail;
|
|
}
|
|
|
|
file = VasEBoot_file_open (argv[0], VAS_EBOOT_FILE_TYPE_LINUX_KERNEL);
|
|
if (! file)
|
|
goto fail;
|
|
|
|
len = VasEBoot_file_read (file, buffer, sizeof (buffer));
|
|
if (len < (VasEBoot_ssize_t) sizeof (Elf64_Ehdr))
|
|
{
|
|
if (!VasEBoot_errno)
|
|
VasEBoot_error (VAS_EBOOT_ERR_BAD_OS, N_("premature end of file %s"),
|
|
argv[0]);
|
|
goto fail;
|
|
}
|
|
|
|
VasEBoot_dprintf ("linux", "Loading linux: %s\n", argv[0]);
|
|
|
|
if (VasEBoot_load_elf64 (file, buffer, argv[0]))
|
|
goto fail;
|
|
|
|
len = sizeof("BOOT_IMAGE=") + 8;
|
|
for (i = 0; i < argc; i++)
|
|
len += VasEBoot_strlen (argv[i]) + 1;
|
|
len += sizeof (struct ia64_boot_param) + 512; /* Room for extensions. */
|
|
boot_param_pages = page_align (len) >> 12;
|
|
boot_param = VasEBoot_efi_allocate_any_pages (boot_param_pages);
|
|
if (boot_param == 0)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_OUT_OF_MEMORY,
|
|
"cannot allocate memory for bootparams");
|
|
goto fail;
|
|
}
|
|
|
|
VasEBoot_memset (boot_param, 0, len);
|
|
cmdline = ((char *)(boot_param + 1)) + 256;
|
|
|
|
/* Build cmdline. */
|
|
p = VasEBoot_stpcpy (cmdline, "BOOT_IMAGE");
|
|
for (i = 0; i < argc; i++)
|
|
{
|
|
*p++ = ' ';
|
|
p = VasEBoot_stpcpy (p, argv[i]);
|
|
}
|
|
cmdline[10] = '=';
|
|
|
|
*p = '\0';
|
|
|
|
if (VasEBoot_verify_string (cmdline, VAS_EBOOT_VERIFY_KERNEL_CMDLINE))
|
|
goto fail;
|
|
|
|
boot_param->command_line = (VasEBoot_uint64_t) cmdline;
|
|
boot_param->efi_systab = (VasEBoot_uint64_t) VasEBoot_efi_system_table;
|
|
|
|
VasEBoot_errno = VAS_EBOOT_ERR_NONE;
|
|
|
|
VasEBoot_loader_set (VasEBoot_linux_boot, VasEBoot_linux_unload, 0);
|
|
|
|
fail:
|
|
if (file)
|
|
VasEBoot_file_close (file);
|
|
|
|
if (VasEBoot_errno != VAS_EBOOT_ERR_NONE)
|
|
{
|
|
VasEBoot_efi_free_pages ((VasEBoot_efi_physical_address_t) boot_param,
|
|
boot_param_pages);
|
|
VasEBoot_dl_unref (my_mod);
|
|
}
|
|
return VasEBoot_errno;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_cmd_initrd (VasEBoot_command_t cmd __attribute__ ((unused)),
|
|
int argc, char *argv[])
|
|
{
|
|
struct VasEBoot_linux_initrd_context initrd_ctx = { 0, 0, 0 };
|
|
|
|
if (argc == 0)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_BAD_ARGUMENT, N_("filename expected"));
|
|
goto fail;
|
|
}
|
|
|
|
if (! loaded)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_BAD_ARGUMENT, N_("you need to load the kernel first"));
|
|
goto fail;
|
|
}
|
|
|
|
if (VasEBoot_initrd_init (argc, argv, &initrd_ctx))
|
|
goto fail;
|
|
|
|
initrd_size = VasEBoot_get_initrd_size (&initrd_ctx);
|
|
VasEBoot_dprintf ("linux", "Loading initrd\n");
|
|
|
|
initrd_pages = (page_align (initrd_size) >> 12);
|
|
initrd_mem = VasEBoot_efi_allocate_any_pages (initrd_pages);
|
|
if (! initrd_mem)
|
|
{
|
|
VasEBoot_error (VAS_EBOOT_ERR_OUT_OF_MEMORY, "cannot allocate pages");
|
|
goto fail;
|
|
}
|
|
|
|
VasEBoot_dprintf ("linux", "[addr=0x%lx, size=0x%lx]\n",
|
|
(VasEBoot_uint64_t) initrd_mem, initrd_size);
|
|
|
|
if (VasEBoot_initrd_load (&initrd_ctx, initrd_mem))
|
|
goto fail;
|
|
fail:
|
|
VasEBoot_initrd_close (&initrd_ctx);
|
|
return VasEBoot_errno;
|
|
}
|
|
|
|
static VasEBoot_err_t
|
|
VasEBoot_cmd_fpswa (VasEBoot_command_t cmd __attribute__ ((unused)),
|
|
int argc __attribute__((unused)),
|
|
char *argv[] __attribute__((unused)))
|
|
{
|
|
query_fpswa ();
|
|
if (fpswa == NULL)
|
|
VasEBoot_puts_ (N_("No FPSWA found"));
|
|
else
|
|
VasEBoot_printf (_("FPSWA revision: %x\n"), fpswa->revision);
|
|
return VAS_EBOOT_ERR_NONE;
|
|
}
|
|
|
|
static VasEBoot_command_t cmd_linux, cmd_initrd, cmd_fpswa;
|
|
|
|
VAS_EBOOT_MOD_INIT(linux)
|
|
{
|
|
cmd_linux = VasEBoot_register_command ("linux", VasEBoot_cmd_linux,
|
|
N_("FILE [ARGS...]"), N_("Load Linux."));
|
|
|
|
cmd_initrd = VasEBoot_register_command ("initrd", VasEBoot_cmd_initrd,
|
|
N_("FILE"), N_("Load initrd."));
|
|
|
|
cmd_fpswa = VasEBoot_register_command ("fpswa", VasEBoot_cmd_fpswa,
|
|
"", N_("Display FPSWA version."));
|
|
|
|
my_mod = mod;
|
|
}
|
|
|
|
VAS_EBOOT_MOD_FINI(linux)
|
|
{
|
|
VasEBoot_unregister_command (cmd_linux);
|
|
VasEBoot_unregister_command (cmd_initrd);
|
|
VasEBoot_unregister_command (cmd_fpswa);
|
|
}
|