e30274a
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
e30274a
From: Peter Jones <pjones@redhat.com>
e30274a
Date: Wed, 12 Sep 2018 16:03:55 -0400
e30274a
Subject: [PATCH] x86-efi: Make our own allocator for kernel stuff
e30274a
e30274a
This helps enable allocations above 4GB.
e30274a
e30274a
Signed-off-by: Peter Jones <pjones@redhat.com>
e30274a
---
a3bfe35
 grub-core/loader/i386/efi/linux.c | 167 +++++++++++++++++++++-----------------
a3bfe35
 1 file changed, 94 insertions(+), 73 deletions(-)
e30274a
e30274a
diff --git a/grub-core/loader/i386/efi/linux.c b/grub-core/loader/i386/efi/linux.c
a3bfe35
index de5f839ff0f..1811f4b3d56 100644
e30274a
--- a/grub-core/loader/i386/efi/linux.c
e30274a
+++ b/grub-core/loader/i386/efi/linux.c
a3bfe35
@@ -48,6 +48,65 @@ static char *linux_cmdline;
e30274a
 
e30274a
 #define BYTES_TO_PAGES(bytes)   (((bytes) + 0xfff) >> 12)
e30274a
 
e30274a
+struct allocation_choice {
e30274a
+    grub_efi_physical_address_t addr;
e30274a
+    grub_efi_allocate_type_t alloc_type;
e30274a
+};
e30274a
+
e30274a
+static struct allocation_choice max_addresses[] =
e30274a
+  {
e30274a
+    { GRUB_EFI_MAX_ALLOCATION_ADDRESS, GRUB_EFI_ALLOCATE_MAX_ADDRESS },
e30274a
+    { GRUB_EFI_MAX_ALLOCATION_ADDRESS, GRUB_EFI_ALLOCATE_MAX_ADDRESS },
e30274a
+    { GRUB_EFI_MAX_ALLOCATION_ADDRESS, GRUB_EFI_ALLOCATE_MAX_ADDRESS },
e30274a
+    { 0, 0 }
e30274a
+  };
e30274a
+
e30274a
+static inline void
e30274a
+kernel_free(void *addr, grub_efi_uintn_t size)
e30274a
+{
e30274a
+  if (addr && size)
e30274a
+    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)addr,
e30274a
+			 BYTES_TO_PAGES(size));
e30274a
+}
e30274a
+
e30274a
+static void *
e30274a
+kernel_alloc(grub_efi_uintn_t size, const char * const errmsg)
e30274a
+{
e30274a
+  void *addr = 0;
e30274a
+  unsigned int i;
e30274a
+  grub_efi_physical_address_t prev_max = 0;
e30274a
+
e30274a
+  for (i = 0; max_addresses[i].addr != 0 && addr == 0; i++)
e30274a
+    {
e30274a
+      grub_uint64_t max = max_addresses[i].addr;
e30274a
+      grub_efi_uintn_t pages;
e30274a
+
e30274a
+      if (max == prev_max)
e30274a
+	continue;
e30274a
+
e30274a
+      pages = BYTES_TO_PAGES(size);
e30274a
+      grub_dprintf ("linux", "Trying to allocate %lu pages from %p\n",
e30274a
+		    pages, (void *)max);
e30274a
+
e30274a
+      prev_max = max;
e30274a
+      addr = grub_efi_allocate_pages_real (max, pages,
e30274a
+					   max_addresses[i].alloc_type,
e30274a
+					   GRUB_EFI_LOADER_DATA);
e30274a
+      if (addr)
e30274a
+	grub_dprintf ("linux", "Allocated at %p\n", addr);
e30274a
+    }
e30274a
+
a3bfe35
+  while (grub_error_pop ())
a3bfe35
+    {
a3bfe35
+      ;
a3bfe35
+    }
a3bfe35
+
e30274a
+  if (addr == NULL)
e30274a
+    grub_error (GRUB_ERR_OUT_OF_MEMORY, errmsg);
e30274a
+
e30274a
+  return addr;
e30274a
+}
e30274a
+
e30274a
 static grub_err_t
e30274a
 grub_linuxefi_boot (void)
e30274a
 {
a3bfe35
@@ -63,19 +122,12 @@ grub_linuxefi_unload (void)
e30274a
 {
e30274a
   grub_dl_unref (my_mod);
e30274a
   loaded = 0;
e30274a
-  if (initrd_mem)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)initrd_mem,
e30274a
-			 BYTES_TO_PAGES(params->ramdisk_size));
e30274a
-  if (linux_cmdline)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)
e30274a
-			 linux_cmdline,
e30274a
-			 BYTES_TO_PAGES(params->cmdline_size + 1));
e30274a
-  if (kernel_mem)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)kernel_mem,
e30274a
-			 BYTES_TO_PAGES(kernel_size));
e30274a
-  if (params)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)params,
e30274a
-			 BYTES_TO_PAGES(16384));
e30274a
+
e30274a
+  kernel_free(initrd_mem, params->ramdisk_size);
e30274a
+  kernel_free(linux_cmdline, params->cmdline_size + 1);
e30274a
+  kernel_free(kernel_mem, kernel_size);
e30274a
+  kernel_free(params, sizeof(*params));
e30274a
+
e30274a
   return GRUB_ERR_NONE;
e30274a
 }
e30274a
 
a3bfe35
@@ -152,19 +204,13 @@ grub_cmd_initrd (grub_command_t cmd __attribute__ ((unused)),
e30274a
       size += ALIGN_UP (grub_file_size (files[i]), 4);
e30274a
     }
e30274a
 
e30274a
-  initrd_mem = grub_efi_allocate_pages_max (GRUB_EFI_MAX_ALLOCATION_ADDRESS, BYTES_TO_PAGES(size));
e30274a
-  if (!initrd_mem)
e30274a
-    initrd_mem = grub_efi_allocate_pages_max (GRUB_EFI_MAX_USABLE_ADDRESS, BYTES_TO_PAGES(size));
e30274a
-  if (!initrd_mem)
e30274a
-    {
e30274a
-      grub_error (GRUB_ERR_OUT_OF_MEMORY, N_("can't allocate initrd"));
e30274a
-      goto fail;
e30274a
-    }
e30274a
-
e30274a
-  grub_dprintf ("linux", "initrd_mem = %lx\n", (unsigned long) initrd_mem);
e30274a
+  initrd_mem = kernel_alloc(size, N_("can't allocate initrd"));
e30274a
+  if (initrd_mem == NULL)
e30274a
+    goto fail;
e30274a
+  grub_dprintf ("linux", "initrd_mem = %p\n", initrd_mem);
e30274a
 
e30274a
   params->ramdisk_size = size;
e30274a
-  params->ramdisk_image = (grub_uint32_t)(grub_addr_t) initrd_mem;
e30274a
+  params->ramdisk_image = initrd_mem;
e30274a
 
e30274a
   ptr = initrd_mem;
e30274a
 
a3bfe35
@@ -225,7 +271,6 @@ grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
e30274a
   filelen = grub_file_size (file);
e30274a
 
e30274a
   kernel = grub_malloc(filelen);
e30274a
-
e30274a
   if (!kernel)
e30274a
     {
e30274a
       grub_error (GRUB_ERR_OUT_OF_MEMORY, N_("cannot allocate kernel buffer"));
a3bfe35
@@ -281,7 +326,7 @@ grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
e30274a
       goto fail;
e30274a
     }
e30274a
 
e30274a
-#if defined(__x86_64__) || defined(__aarch64__)
e30274a
+#if defined(__x86_64__)
e30274a
   grub_dprintf ("linux", "checking lh->xloadflags\n");
e30274a
   if (!(lh->xloadflags & LINUX_XLF_KERNEL_64))
e30274a
     {
a3bfe35
@@ -300,17 +345,9 @@ grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
e30274a
     }
e30274a
 #endif
e30274a
 
e30274a
-  params = grub_efi_allocate_pages_max (GRUB_EFI_MAX_ALLOCATION_ADDRESS,
e30274a
-					BYTES_TO_PAGES(sizeof(*params)));
e30274a
+  params = kernel_alloc (sizeof(*params), "cannot allocate kernel parameters");
e30274a
   if (!params)
e30274a
-    params = grub_efi_allocate_pages_max (GRUB_EFI_MAX_USABLE_ADDRESS,
e30274a
-					  BYTES_TO_PAGES(sizeof(*params)));
e30274a
-  if (! params)
e30274a
-    {
e30274a
-      grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate kernel parameters");
e30274a
-      goto fail;
e30274a
-    }
e30274a
-
e30274a
+    goto fail;
e30274a
   grub_dprintf ("linux", "params = %p\n", params);
e30274a
 
e30274a
   grub_memset (params, 0, sizeof(*params));
a3bfe35
@@ -329,19 +366,10 @@ grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
e30274a
   grub_dprintf ("linux", "new lh is at %p\n", lh);
e30274a
 
e30274a
   grub_dprintf ("linux", "setting up cmdline\n");
e30274a
-  linux_cmdline = grub_efi_allocate_pages_max(GRUB_EFI_MAX_ALLOCATION_ADDRESS,
e30274a
-					      BYTES_TO_PAGES(lh->cmdline_size + 1));
e30274a
+  linux_cmdline = kernel_alloc (lh->cmdline_size + 1, N_("can't allocate cmdline"));
e30274a
   if (!linux_cmdline)
e30274a
-    linux_cmdline = grub_efi_allocate_pages_max(GRUB_EFI_MAX_USABLE_ADDRESS,
e30274a
-						BYTES_TO_PAGES(lh->cmdline_size + 1));
e30274a
-  if (!linux_cmdline)
e30274a
-    {
e30274a
-      grub_error (GRUB_ERR_OUT_OF_MEMORY, N_("can't allocate cmdline"));
e30274a
-      goto fail;
e30274a
-    }
e30274a
-
e30274a
-  grub_dprintf ("linux", "linux_cmdline = %lx\n",
e30274a
-		(unsigned long)linux_cmdline);
e30274a
+    goto fail;
e30274a
+  grub_dprintf ("linux", "linux_cmdline = %p\n", linux_cmdline);
e30274a
 
e30274a
   grub_memcpy (linux_cmdline, LINUX_IMAGE, sizeof (LINUX_IMAGE));
e30274a
   grub_create_loader_cmdline (argc, argv,
a3bfe35
@@ -349,27 +377,24 @@ grub_cmd_linux (grub_command_t cmd __attribute__ ((unused)),
e30274a
 			      lh->cmdline_size - (sizeof (LINUX_IMAGE) - 1));
e30274a
 
e30274a
   grub_dprintf ("linux", "cmdline:%s\n", linux_cmdline);
e30274a
-  grub_dprintf ("linux", "setting lh->cmd_line_ptr\n");
e30274a
-  lh->cmd_line_ptr = (grub_uint32_t)(grub_addr_t)linux_cmdline;
e30274a
+  grub_dprintf ("linux", "setting lh->cmd_line_ptr to 0x%08x\n",
e30274a
+		linux_cmdline);
e30274a
+  lh->cmd_line_ptr = linux_cmdline;
e30274a
 
e30274a
   handover_offset = lh->handover_offset;
e30274a
-  grub_dprintf("linux", "handover_offset: %08x\n", handover_offset);
e30274a
+  grub_dprintf("linux", "handover_offset: 0x%08x\n", handover_offset);
e30274a
 
e30274a
   start = (lh->setup_sects + 1) * 512;
e30274a
 
e30274a
-  kernel_mem = grub_efi_allocate_pages_max(lh->pref_address,
e30274a
-					   BYTES_TO_PAGES(lh->init_size));
e30274a
-  if (!kernel_mem)
e30274a
-    kernel_mem = grub_efi_allocate_pages_max(GRUB_EFI_MAX_ALLOCATION_ADDRESS,
e30274a
-					     BYTES_TO_PAGES(lh->init_size));
e30274a
-  if (!kernel_mem)
e30274a
-    kernel_mem = grub_efi_allocate_pages_max(GRUB_EFI_MAX_USABLE_ADDRESS,
e30274a
-					     BYTES_TO_PAGES(lh->init_size));
e30274a
-  if (!kernel_mem)
e30274a
+  grub_dprintf ("linux", "lh->pref_address: %p\n", (void *)(grub_addr_t)lh->pref_address);
e30274a
+  if (lh->pref_address < (grub_uint64_t)GRUB_EFI_MAX_ALLOCATION_ADDRESS)
e30274a
     {
e30274a
-      grub_error (GRUB_ERR_OUT_OF_MEMORY, N_("can't allocate kernel"));
e30274a
-      goto fail;
e30274a
+      max_addresses[0].addr = lh->pref_address;
e30274a
+      max_addresses[0].alloc_type = GRUB_EFI_ALLOCATE_ADDRESS;
e30274a
     }
e30274a
+  kernel_mem = kernel_alloc (lh->init_size, N_("can't allocate kernel"));
e30274a
+  if (!kernel_mem)
e30274a
+    goto fail;
e30274a
   grub_dprintf("linux", "kernel_mem = %p\n", kernel_mem);
e30274a
 
e30274a
   grub_loader_set (grub_linuxefi_boot, grub_linuxefi_unload, 0);
a3bfe35
@@ -404,18 +429,14 @@ fail:
e30274a
       loaded = 0;
e30274a
     }
e30274a
 
e30274a
-  if (linux_cmdline && lh && !loaded)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)
e30274a
-			 linux_cmdline,
e30274a
-			 BYTES_TO_PAGES(lh->cmdline_size + 1));
e30274a
+  if (!loaded)
e30274a
+    {
e30274a
+      if (lh)
e30274a
+	kernel_free (linux_cmdline, lh->cmdline_size + 1);
e30274a
 
e30274a
-  if (kernel_mem && !loaded)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)kernel_mem,
e30274a
-			 BYTES_TO_PAGES(kernel_size));
e30274a
-
e30274a
-  if (params && !loaded)
e30274a
-    grub_efi_free_pages ((grub_efi_physical_address_t)(grub_addr_t)params,
e30274a
-			 BYTES_TO_PAGES(16384));
e30274a
+      kernel_free (kernel_mem, kernel_size);
e30274a
+      kernel_free (params, sizeof(*params));
e30274a
+    }
e30274a
 
e30274a
   return grub_errno;
e30274a
 }