669 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			669 lines
		
	
	
	
		
			17 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  GRUB  --  GRand Unified Bootloader
 | |
|  *  Copyright (C) 2008,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/loader.h>
 | |
| #include <grub/file.h>
 | |
| #include <grub/disk.h>
 | |
| #include <grub/err.h>
 | |
| #include <grub/misc.h>
 | |
| #include <grub/types.h>
 | |
| #include <grub/command.h>
 | |
| #include <grub/dl.h>
 | |
| #include <grub/mm.h>
 | |
| #include <grub/cache.h>
 | |
| #include <grub/kernel.h>
 | |
| #include <grub/efi/api.h>
 | |
| #include <grub/efi/efi.h>
 | |
| #include <grub/elf.h>
 | |
| #include <grub/i18n.h>
 | |
| #include <grub/env.h>
 | |
| 
 | |
| GRUB_MOD_LICENSE ("GPLv3+");
 | |
| 
 | |
| #pragma GCC diagnostic ignored "-Wcast-align"
 | |
| 
 | |
| #define ALIGN_MIN (256*1024*1024)
 | |
| 
 | |
| #define GRUB_ELF_SEARCH 1024
 | |
| 
 | |
| #define BOOT_PARAM_SIZE	16384
 | |
| 
 | |
| struct ia64_boot_param
 | |
| {
 | |
|   grub_uint64_t command_line;	/* physical address of command line. */
 | |
|   grub_uint64_t efi_systab;	/* physical address of EFI system table */
 | |
|   grub_uint64_t efi_memmap;	/* physical address of EFI memory map */
 | |
|   grub_uint64_t efi_memmap_size;	/* size of EFI memory map */
 | |
|   grub_uint64_t efi_memdesc_size; /* size of an EFI memory map descriptor */
 | |
|   grub_uint32_t efi_memdesc_version;	/* memory descriptor version */
 | |
|   struct
 | |
|   {
 | |
|     grub_uint16_t num_cols;	/* number of columns on console output dev */
 | |
|     grub_uint16_t num_rows;	/* number of rows on console output device */
 | |
|     grub_uint16_t orig_x;	/* cursor's x position */
 | |
|     grub_uint16_t orig_y;	/* cursor's y position */
 | |
|   } console_info;
 | |
|   grub_uint64_t fpswa;		/* physical address of the fpswa interface */
 | |
|   grub_uint64_t initrd_start;
 | |
|   grub_uint64_t initrd_size;
 | |
| };
 | |
| 
 | |
| typedef struct
 | |
| {
 | |
|   grub_uint32_t	revision;
 | |
|   grub_uint32_t	reserved;
 | |
|   void *fpswa;
 | |
| } fpswa_interface_t;
 | |
| static fpswa_interface_t *fpswa;
 | |
| 
 | |
| #define NEXT_MEMORY_DESCRIPTOR(desc, size)      \
 | |
|   ((grub_efi_memory_descriptor_t *) ((char *) (desc) + (size)))
 | |
| 
 | |
| static grub_dl_t my_mod;
 | |
| 
 | |
| static int loaded;
 | |
| 
 | |
| /* Kernel base and size.  */
 | |
| static void *kernel_mem;
 | |
| static grub_efi_uintn_t kernel_pages;
 | |
| static grub_uint64_t entry;
 | |
| 
 | |
| /* Initrd base and size.  */
 | |
| static void *initrd_mem;
 | |
| static grub_efi_uintn_t initrd_pages;
 | |
| static grub_efi_uintn_t initrd_size;
 | |
| 
 | |
| static struct ia64_boot_param *boot_param;
 | |
| static grub_efi_uintn_t boot_param_pages;
 | |
| 
 | |
| static inline grub_size_t
 | |
| page_align (grub_size_t size)
 | |
| {
 | |
|   return (size + (1 << 12) - 1) & (~((1 << 12) - 1));
 | |
| }
 | |
| 
 | |
| static void
 | |
| query_fpswa (void)
 | |
| {
 | |
|   grub_efi_handle_t fpswa_image;
 | |
|   grub_efi_boot_services_t *bs;
 | |
|   grub_efi_status_t status;
 | |
|   grub_efi_uintn_t size;
 | |
|   static const grub_efi_guid_t fpswa_protocol = 
 | |
|     { 0xc41b6531, 0x97b9, 0x11d3,
 | |
|       {0x9a, 0x29, 0x0, 0x90, 0x27, 0x3f, 0xc1, 0x4d} };
 | |
| 
 | |
|   if (fpswa != NULL)
 | |
|     return;
 | |
| 
 | |
|   size = sizeof(grub_efi_handle_t);
 | |
|   
 | |
|   bs = grub_efi_system_table->boot_services;
 | |
|   status = bs->locate_handle (GRUB_EFI_BY_PROTOCOL,
 | |
| 			      (void *) &fpswa_protocol,
 | |
| 			      NULL, &size, &fpswa_image);
 | |
|   if (status != GRUB_EFI_SUCCESS)
 | |
|     {
 | |
|       grub_printf ("%s\n", _("Could not locate FPSWA driver"));
 | |
|       return;
 | |
|     }
 | |
|   status = bs->handle_protocol (fpswa_image,
 | |
| 				(void *) &fpswa_protocol, (void *) &fpswa);
 | |
|   if (status != GRUB_EFI_SUCCESS)
 | |
|     {
 | |
|       grub_printf ("%s\n",
 | |
| 		   _("FPSWA protocol wasn't able to find the interface"));
 | |
|       return;
 | |
|     } 
 | |
| }
 | |
| 
 | |
| /* Find the optimal number of pages for the memory map. Is it better to
 | |
|    move this code to efi/mm.c?  */
 | |
| static grub_efi_uintn_t
 | |
| find_mmap_size (void)
 | |
| {
 | |
|   static grub_efi_uintn_t mmap_size = 0;
 | |
| 
 | |
|   if (mmap_size != 0)
 | |
|     return mmap_size;
 | |
|   
 | |
|   mmap_size = (1 << 12);
 | |
|   while (1)
 | |
|     {
 | |
|       int ret;
 | |
|       grub_efi_memory_descriptor_t *mmap;
 | |
|       grub_efi_uintn_t desc_size;
 | |
|       
 | |
|       mmap = grub_malloc (mmap_size);
 | |
|       if (! mmap)
 | |
| 	return 0;
 | |
| 
 | |
|       ret = grub_efi_get_memory_map (&mmap_size, mmap, 0, &desc_size, 0);
 | |
|       grub_free (mmap);
 | |
|       
 | |
|       if (ret < 0)
 | |
| 	{
 | |
| 	  grub_error (GRUB_ERR_IO, "cannot get memory map");
 | |
| 	  return 0;
 | |
| 	}
 | |
|       else if (ret > 0)
 | |
| 	break;
 | |
| 
 | |
|       mmap_size += (1 << 12);
 | |
|     }
 | |
| 
 | |
|   /* Increase the size a bit for safety, because GRUB allocates more on
 | |
|      later, and EFI itself may allocate more.  */
 | |
|   mmap_size += (1 << 12);
 | |
| 
 | |
|   return page_align (mmap_size);
 | |
| }
 | |
| 
 | |
| static void
 | |
| free_pages (void)
 | |
| {
 | |
|   if (kernel_mem)
 | |
|     {
 | |
|       grub_efi_free_pages ((grub_addr_t) kernel_mem, kernel_pages);
 | |
|       kernel_mem = 0;
 | |
|     }
 | |
| 
 | |
|   if (initrd_mem)
 | |
|     {
 | |
|       grub_efi_free_pages ((grub_addr_t) initrd_mem, initrd_pages);
 | |
|       initrd_mem = 0;
 | |
|     }
 | |
| 
 | |
|   if (boot_param)
 | |
|     {
 | |
|       /* Free bootparam.  */
 | |
|       grub_efi_free_pages ((grub_efi_physical_address_t) boot_param,
 | |
| 			   boot_param_pages);
 | |
|       boot_param = 0;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void *
 | |
| allocate_pages (grub_uint64_t align, grub_uint64_t size_pages,
 | |
| 		grub_uint64_t nobase)
 | |
| {
 | |
|   grub_uint64_t size;
 | |
|   grub_efi_uintn_t desc_size;
 | |
|   grub_efi_memory_descriptor_t *mmap, *mmap_end;
 | |
|   grub_efi_uintn_t mmap_size, tmp_mmap_size;
 | |
|   grub_efi_memory_descriptor_t *desc;
 | |
|   void *mem = NULL;
 | |
| 
 | |
|   size = size_pages << 12;
 | |
| 
 | |
|   mmap_size = find_mmap_size ();
 | |
|   if (!mmap_size)
 | |
|     return 0;
 | |
| 
 | |
|     /* Read the memory map temporarily, to find free space.  */
 | |
|   mmap = grub_malloc (mmap_size);
 | |
|   if (! mmap)
 | |
|     return 0;
 | |
| 
 | |
|   tmp_mmap_size = mmap_size;
 | |
|   if (grub_efi_get_memory_map (&tmp_mmap_size, mmap, 0, &desc_size, 0) <= 0)
 | |
|     {
 | |
|       grub_error (GRUB_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))
 | |
|     {
 | |
|       grub_uint64_t start, end;
 | |
|       grub_uint64_t aligned_start;
 | |
| 
 | |
|       if (desc->type != GRUB_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 = grub_efi_allocate_pages (aligned_start, size_pages);
 | |
|       if (! mem)
 | |
| 	{
 | |
| 	  grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate memory");
 | |
| 	  goto fail;
 | |
| 	}
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|   if (! mem)
 | |
|     {
 | |
|       grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate memory");
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   grub_free (mmap);
 | |
|   return mem;
 | |
| 
 | |
|  fail:
 | |
|   grub_free (mmap);
 | |
|   free_pages ();
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static void
 | |
| set_boot_param_console (void)
 | |
| {
 | |
|   grub_efi_simple_text_output_interface_t *conout;
 | |
|   grub_efi_uintn_t cols, rows;
 | |
|   
 | |
|   conout = grub_efi_system_table->con_out;
 | |
|   if (conout->query_mode (conout, conout->mode->mode, &cols, &rows)
 | |
|       != GRUB_EFI_SUCCESS)
 | |
|     return;
 | |
| 
 | |
|   grub_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 grub_err_t
 | |
| grub_linux_boot (void)
 | |
| {
 | |
|   grub_efi_uintn_t mmap_size;
 | |
|   grub_efi_uintn_t map_key;
 | |
|   grub_efi_uintn_t desc_size;
 | |
|   grub_efi_uint32_t desc_version;
 | |
|   grub_efi_memory_descriptor_t *mmap_buf;
 | |
|   grub_err_t err;
 | |
| 
 | |
|   /* FPSWA.  */
 | |
|   query_fpswa ();
 | |
|   boot_param->fpswa = (grub_uint64_t)fpswa;
 | |
| 
 | |
|   /* Initrd.  */
 | |
|   boot_param->initrd_start = (grub_uint64_t)initrd_mem;
 | |
|   boot_param->initrd_size = (grub_uint64_t)initrd_size;
 | |
| 
 | |
|   set_boot_param_console ();
 | |
| 
 | |
|   grub_dprintf ("linux", "Jump to %016lx\n", entry);
 | |
| 
 | |
|   /* MDT.
 | |
|      Must be done after grub_machine_fini because map_key is used by
 | |
|      exit_boot_services.  */
 | |
|   mmap_size = find_mmap_size ();
 | |
|   if (! mmap_size)
 | |
|     return grub_errno;
 | |
|   mmap_buf = grub_efi_allocate_pages (0, page_align (mmap_size) >> 12);
 | |
|   if (! mmap_buf)
 | |
|     return grub_error (GRUB_ERR_IO, "cannot allocate memory map");
 | |
|   err = grub_efi_finish_boot_services (&mmap_size, mmap_buf, &map_key,
 | |
| 				       &desc_size, &desc_version);
 | |
|   if (err)
 | |
|     return err;
 | |
| 
 | |
|   boot_param->efi_memmap = (grub_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 GRUB_ERR_NONE;
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_linux_unload (void)
 | |
| {
 | |
|   free_pages ();
 | |
|   grub_dl_unref (my_mod);
 | |
|   loaded = 0;
 | |
|   return GRUB_ERR_NONE;
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_load_elf64 (grub_file_t file, void *buffer, const char *filename)
 | |
| {
 | |
|   Elf64_Ehdr *ehdr = (Elf64_Ehdr *) buffer;
 | |
|   Elf64_Phdr *phdr;
 | |
|   int i;
 | |
|   grub_uint64_t low_addr;
 | |
|   grub_uint64_t high_addr;
 | |
|   grub_uint64_t align;
 | |
|   grub_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 grub_error(GRUB_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 grub_error (GRUB_ERR_UNKNOWN_OS,
 | |
| 		       N_("invalid arch-dependent ELF magic"));
 | |
| 
 | |
|   if (ehdr->e_type != ET_EXEC)
 | |
|     return grub_error (GRUB_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 > GRUB_ELF_SEARCH)
 | |
|     return grub_error (GRUB_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 grub_error (GRUB_ERR_BAD_OS, "no program entries");
 | |
| 
 | |
|   kernel_pages = page_align (high_addr - low_addr) >> 12;
 | |
| 
 | |
|   /* Undocumented on purpose.  */
 | |
|   relocate = grub_env_get ("linux_relocate");
 | |
|   if (!relocate || grub_strcmp (relocate, "force") != 0)
 | |
|     {
 | |
|       kernel_mem = grub_efi_allocate_pages (low_addr, kernel_pages);
 | |
|       reloc_offset = 0;
 | |
|     }
 | |
|   /* Try to relocate.  */
 | |
|   if (! kernel_mem && (!relocate || grub_strcmp (relocate, "off") != 0))
 | |
|     {
 | |
|       kernel_mem = allocate_pages (align, kernel_pages, low_addr);
 | |
|       if (kernel_mem)
 | |
| 	{
 | |
| 	  reloc_offset = (grub_uint64_t)kernel_mem - low_addr;
 | |
| 	  grub_dprintf ("linux", "  Relocated at %p (offset=%016lx)\n",
 | |
| 			kernel_mem, reloc_offset);
 | |
| 	  entry += reloc_offset;
 | |
| 	}
 | |
|     }
 | |
|   if (! kernel_mem)
 | |
|     return grub_error (GRUB_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)
 | |
|         {
 | |
| 	  grub_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 (grub_file_seek (file, phdr->p_offset) == (grub_off_t)-1)
 | |
| 	    return grub_errno;
 | |
| 
 | |
| 	  if (grub_file_read (file, (void *) (phdr->p_paddr + reloc_offset),
 | |
| 			      phdr->p_filesz)
 | |
|               != (grub_ssize_t) phdr->p_filesz)
 | |
| 	    {
 | |
| 	      if (!grub_errno)
 | |
| 		grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
 | |
| 			    filename);
 | |
| 	      return grub_errno;
 | |
| 	    }
 | |
| 	  
 | |
|           if (phdr->p_filesz < phdr->p_memsz)
 | |
| 	    grub_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)
 | |
| 	    grub_arch_sync_caches
 | |
| 	      ((void *)(phdr->p_paddr + reloc_offset), phdr->p_memsz);
 | |
|         }
 | |
|     }
 | |
|   loaded = 1;
 | |
|   return 0;
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
 | |
| 		int argc, char *argv[])
 | |
| {
 | |
|   grub_file_t file = 0;
 | |
|   char buffer[GRUB_ELF_SEARCH];
 | |
|   char *cmdline, *p;
 | |
|   grub_ssize_t len;
 | |
|   int i;
 | |
| 
 | |
|   grub_dl_ref (my_mod);
 | |
| 
 | |
|   grub_loader_unset ();
 | |
|     
 | |
|   if (argc == 0)
 | |
|     {
 | |
|       grub_error (GRUB_ERR_BAD_ARGUMENT, N_("filename expected"));
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   file = grub_file_open (argv[0]);
 | |
|   if (! file)
 | |
|     goto fail;
 | |
| 
 | |
|   len = grub_file_read (file, buffer, sizeof (buffer));
 | |
|   if (len < (grub_ssize_t) sizeof (Elf64_Ehdr))
 | |
|     {
 | |
|       if (!grub_errno)
 | |
| 	grub_error (GRUB_ERR_BAD_OS, N_("premature end of file %s"),
 | |
| 		    argv[0]);
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   grub_dprintf ("linux", "Loading linux: %s\n", argv[0]);
 | |
| 
 | |
|   if (grub_load_elf64 (file, buffer, argv[0]))
 | |
|     goto fail;
 | |
| 
 | |
|   len = sizeof("BOOT_IMAGE=") + 8;
 | |
|   for (i = 0; i < argc; i++)
 | |
|     len += grub_strlen (argv[i]) + 1;
 | |
|   len += sizeof (struct ia64_boot_param) + 512; /* Room for extensions.  */
 | |
|   boot_param_pages = page_align (len) >> 12;
 | |
|   boot_param = grub_efi_allocate_pages (0, boot_param_pages);
 | |
|   if (boot_param == 0)
 | |
|     {
 | |
|       grub_error (GRUB_ERR_OUT_OF_MEMORY,
 | |
| 		  "cannot allocate memory for bootparams");
 | |
|       goto fail;
 | |
|     }
 | |
| 
 | |
|   grub_memset (boot_param, 0, len);
 | |
|   cmdline = ((char *)(boot_param + 1)) + 256;
 | |
| 
 | |
|   /* Build cmdline.  */
 | |
|   p = grub_stpcpy (cmdline, "BOOT_IMAGE");
 | |
|   for (i = 0; i < argc; i++)
 | |
|     {
 | |
|       *p++ = ' ';
 | |
|       p = grub_stpcpy (p, argv[i]);
 | |
|     }
 | |
|   cmdline[10] = '=';
 | |
|   
 | |
|   boot_param->command_line = (grub_uint64_t) cmdline;
 | |
|   boot_param->efi_systab = (grub_uint64_t) grub_efi_system_table;
 | |
| 
 | |
|   grub_errno = GRUB_ERR_NONE;
 | |
| 
 | |
|   grub_loader_set (grub_linux_boot, grub_linux_unload, 0);
 | |
| 
 | |
|  fail:
 | |
|   if (file)
 | |
|     grub_file_close (file);
 | |
| 
 | |
|   if (grub_errno != GRUB_ERR_NONE)
 | |
|     {
 | |
|       grub_efi_free_pages ((grub_efi_physical_address_t) boot_param,
 | |
| 			   boot_param_pages);
 | |
|       grub_dl_unref (my_mod);
 | |
|     }
 | |
|   return grub_errno;
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_cmd_initrd (grub_command_t cmd __attribute__ ((unused)),
 | |
| 		 int argc, char *argv[])
 | |
| {
 | |
|   grub_file_t *files = 0;
 | |
|   int i;
 | |
|   int nfiles = 0;
 | |
|   grub_uint8_t *ptr;
 | |
| 
 | |
|   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;
 | |
|     }
 | |
| 
 | |
|   files = grub_zalloc (argc * sizeof (files[0]));
 | |
|   if (!files)
 | |
|     goto fail;
 | |
| 
 | |
|   initrd_size = 0;
 | |
|   grub_dprintf ("linux", "Loading initrd\n");
 | |
|   for (i = 0; i < argc; i++)
 | |
|     {
 | |
|       grub_file_filter_disable_compression ();
 | |
|       files[i] = grub_file_open (argv[i]);
 | |
|       if (! files[i])
 | |
| 	goto fail;
 | |
|       nfiles++;
 | |
|       initrd_size += grub_file_size (files[i]);
 | |
|       grub_dprintf ("linux", "File %d: %s\n", i, argv[i]);
 | |
|     }
 | |
| 
 | |
|   initrd_pages = (page_align (initrd_size) >> 12);
 | |
|   initrd_mem = grub_efi_allocate_pages (0, initrd_pages);
 | |
|   if (! initrd_mem)
 | |
|     {
 | |
|       grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate pages");
 | |
|       goto fail;
 | |
|     }
 | |
|   
 | |
|   grub_dprintf ("linux", "[addr=0x%lx, size=0x%lx]\n",
 | |
| 		(grub_uint64_t) initrd_mem, initrd_size);
 | |
| 
 | |
|   ptr = initrd_mem;
 | |
|   for (i = 0; i < nfiles; i++)
 | |
|     {
 | |
|       grub_ssize_t cursize = grub_file_size (files[i]);
 | |
|       if (grub_file_read (files[i], ptr, cursize) != cursize)
 | |
| 	{
 | |
| 	  if (!grub_errno)
 | |
| 	    grub_error (GRUB_ERR_FILE_READ_ERROR, N_("premature end of file %s"),
 | |
| 			argv[i]);
 | |
| 	  goto fail;
 | |
| 	}
 | |
|       ptr += cursize;
 | |
|     }
 | |
|  fail:
 | |
|   for (i = 0; i < nfiles; i++)
 | |
|     grub_file_close (files[i]);
 | |
|   grub_free (files);
 | |
|   return grub_errno;
 | |
| }
 | |
| 
 | |
| static grub_err_t
 | |
| grub_cmd_fpswa (grub_command_t cmd __attribute__ ((unused)),
 | |
| 		int argc __attribute__((unused)),
 | |
| 		char *argv[] __attribute__((unused)))
 | |
| {
 | |
|   query_fpswa ();
 | |
|   if (fpswa == NULL)
 | |
|     grub_puts_ (N_("No FPSWA found"));
 | |
|   else
 | |
|     grub_printf (_("FPSWA revision: %x\n"), fpswa->revision);
 | |
|   return GRUB_ERR_NONE;
 | |
| }
 | |
| 
 | |
| static grub_command_t cmd_linux, cmd_initrd, cmd_fpswa;
 | |
| 
 | |
| GRUB_MOD_INIT(linux)
 | |
| {
 | |
|   cmd_linux = grub_register_command ("linux", grub_cmd_linux,
 | |
| 				     N_("FILE [ARGS...]"), N_("Load Linux."));
 | |
|   
 | |
|   cmd_initrd = grub_register_command ("initrd", grub_cmd_initrd,
 | |
| 				      N_("FILE"), N_("Load initrd."));
 | |
| 
 | |
|   cmd_fpswa = grub_register_command ("fpswa", grub_cmd_fpswa,
 | |
| 				     "", N_("Display FPSWA version."));
 | |
| 
 | |
|   my_mod = mod;
 | |
| }
 | |
| 
 | |
| GRUB_MOD_FINI(linux)
 | |
| {
 | |
|   grub_unregister_command (cmd_linux);
 | |
|   grub_unregister_command (cmd_initrd);
 | |
|   grub_unregister_command (cmd_fpswa);
 | |
| }
 |