grub-devel
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

[PATCH] Remove nested functions from memory map iterators


From: Colin Watson
Subject: [PATCH] Remove nested functions from memory map iterators
Date: Mon, 14 Jan 2013 15:26:53 +0000
User-agent: Mutt/1.5.21 (2010-09-15)

Here's the next part of my nested functions patch set.  Size changes in
the platforms I tested are as follows:

  DOWN: obj/i386-coreboot/grub-core/kernel.img (27976 > 27936) - change: -40
  UP:   obj/i386-ieee1275/grub-core/kernel.img (48392 < 48428) - change: 36
  DOWN: obj/i386-pc/grub-core/kernel.img (28952 > 28920) - change: -32
  DOWN: obj/amd64-efi/grub-core/acpi.mod (17072 > 17040) - change: -32
  DOWN: obj/amd64-efi/grub-core/bsd.mod (49760 > 49696) - change: -64
  DOWN: obj/amd64-efi/grub-core/linux.mod (17640 > 17504) - change: -136
  DOWN: obj/amd64-efi/grub-core/mmap.mod (9808 > 9560) - change: -248
  DOWN: obj/amd64-efi/grub-core/multiboot.mod (18192 > 18120) - change: -72
  DOWN: obj/amd64-efi/grub-core/multiboot2.mod (19344 > 19296) - change: -48
  DOWN: obj/amd64-efi/grub-core/relocator.mod (26432 > 26392) - change: -40
  DOWN: obj/i386-coreboot/grub-core/acpi.mod (10024 > 9996) - change: -28
  DOWN: obj/i386-coreboot/grub-core/bsd.mod (29604 > 29548) - change: -56
  DOWN: obj/i386-coreboot/grub-core/linux.mod (9392 > 9332) - change: -60
  DOWN: obj/i386-coreboot/grub-core/mmap.mod (6048 > 5884) - change: -164
  DOWN: obj/i386-coreboot/grub-core/multiboot.mod (11244 > 11192) - change: -52
  DOWN: obj/i386-coreboot/grub-core/multiboot2.mod (11828 > 11780) - change: -48
  DOWN: obj/i386-coreboot/grub-core/relocator.mod (15004 > 14948) - change: -56
  DOWN: obj/i386-efi/grub-core/acpi.mod (9884 > 9860) - change: -24
  DOWN: obj/i386-efi/grub-core/bsd.mod (30720 > 30672) - change: -48
  DOWN: obj/i386-efi/grub-core/linux.mod (10560 > 10508) - change: -52
  DOWN: obj/i386-efi/grub-core/mmap.mod (6984 > 6852) - change: -132
  DOWN: obj/i386-efi/grub-core/multiboot.mod (11452 > 11408) - change: -44
  DOWN: obj/i386-efi/grub-core/multiboot2.mod (12200 > 12152) - change: -48
  DOWN: obj/i386-efi/grub-core/relocator.mod (18192 > 18140) - change: -52
  DOWN: obj/i386-ieee1275/grub-core/bsd.mod (29596 > 29540) - change: -56
  UP:   obj/i386-ieee1275/grub-core/efiemu.mod (23248 < 23252) - change: 4
  DOWN: obj/i386-ieee1275/grub-core/linux.mod (9744 > 9684) - change: -60
  DOWN: obj/i386-ieee1275/grub-core/mmap.mod (6048 > 5884) - change: -164
  DOWN: obj/i386-ieee1275/grub-core/multiboot.mod (11084 > 11032) - change: -52
  DOWN: obj/i386-ieee1275/grub-core/multiboot2.mod (11448 > 11400) - change: -48
  DOWN: obj/i386-ieee1275/grub-core/relocator.mod (16432 > 16316) - change: -116
  DOWN: obj/i386-pc/grub-core/acpi.mod (10024 > 9996) - change: -28
  DOWN: obj/i386-pc/grub-core/bsd.mod (30084 > 30028) - change: -56
  DOWN: obj/i386-pc/grub-core/linux.mod (11524 > 11352) - change: -172
  DOWN: obj/i386-pc/grub-core/mmap.mod (8808 > 8604) - change: -204
  DOWN: obj/i386-pc/grub-core/multiboot.mod (12304 > 12256) - change: -48
  DOWN: obj/i386-pc/grub-core/multiboot2.mod (12732 > 12684) - change: -48
  DOWN: obj/i386-pc/grub-core/relocator.mod (15060 > 15004) - change: -56

An i386-pc core image with "biosdisk ext2 part_msdos" gets 24 bytes
smaller.  One with "biosdisk ext2 part_msdos lvm mdraid1x" gets 30 bytes
smaller.

=== modified file 'ChangeLog'
--- ChangeLog   2013-01-13 21:45:16 +0000
+++ ChangeLog   2013-01-14 14:17:55 +0000
@@ -1,3 +1,113 @@
+2013-01-14  Colin Watson  <address@hidden>
+
+       Remove nested functions from memory map iterators.
+
+       * grub-core/efiemu/mm.c (grub_efiemu_mmap_iterate): Add hook_data
+       argument, passed to hook.
+       * grub-core/kern/i386/coreboot/mmap.c
+       (grub_linuxbios_table_iterate): Likewise.
+       (grub_machine_mmap_iterate: iterate_linuxbios_table): Make static
+       instead of nested.
+       (grub_machine_mmap_iterate): Add hook_data argument.
+       * grub-core/kern/i386/multiboot_mmap.c (grub_machine_mmap_iterate):
+       Add hook_data argument, passed to hook.
+       * grub-core/kern/i386/pc/mmap.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/kern/i386/qemu/mmap.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/kern/ieee1275/mmap.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/kern/mips/arc/init.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/kern/mips/loongson/init.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/kern/mips/qemu_mips/init.c (grub_machine_mmap_iterate):
+       Likewise.
+       * grub-core/mmap/efi/mmap.c (grub_efi_mmap_iterate): Likewise.
+       (grub_machine_mmap_iterate): Likewise.
+       * grub-core/mmap/mmap.c (grub_mmap_iterate): Likewise.
+       * include/grub/efiemu/efiemu.h (grub_efiemu_mmap_iterate): Update
+       prototype.
+       * include/grub/memory.h (grub_memory_hook_t): Add data argument.
+       Remove NESTED_FUNC_ATTR from here and from all users.
+       (grub_mmap_iterate): Update prototype.
+       (grub_efi_mmap_iterate): Update prototype.  Update all callers to
+       pass appropriate hook data.
+       (grub_machine_mmap_iterate): Likewise.
+
+       * grub-core/commands/acpi.c (grub_acpi_create_ebda: find_hook): Make
+       static instead of nested.
+       * grub-core/commands/lsmmap.c (grub_cmd_lsmmap: hook): Likewise.
+       Rename to ...
+       (lsmmap_hook): ... this.
+       * grub-core/efiemu/mm.c (grub_efiemu_mmap_init: bounds_hook):
+       Likewise.
+       (grub_efiemu_mmap_fill: fill_hook): Likewise.
+       * grub-core/kern/i386/coreboot/init.c (grub_machine_init:
+       heap_init): Likewise.
+       * grub-core/kern/i386/pc/init.c (grub_machine_init: hook): Likewise.
+       Rename to ...
+       (mmap_iterate_hook): ... this.
+       * grub-core/kern/ieee1275/init.c (grub_claim_heap: heap_init):
+       Likewise.
+       * grub-core/lib/ieee1275/relocator.c
+       (grub_relocator_firmware_get_max_events: count): Likewise.
+       (grub_relocator_firmware_fill_events: fill): Likewise.  Rename
+       to ...
+       (grub_relocator_firmware_fill_events_iter): ... this.
+       * grub-core/lib/relocator.c (grub_relocator_alloc_chunk_align:
+       hook): Likewise.  Rename to ...
+       (grub_relocator_alloc_chunk_align_iter): ... this.
+       * grub-core/loader/i386/bsd.c (generate_e820_mmap: hook): Likewise.
+       Rename to ...
+       (generate_e820_mmap_iter): ... this.
+       * grub-core/loader/i386/linux.c (find_mmap_size: hook): Likewise.
+       Rename to ...
+       (count_hook): ... this.
+       (grub_linux_boot: hook): Likewise.  Rename to ...
+       (grub_linux_boot_mmap_find): ... this.
+       (grub_linux_boot: hook_fill): Likewise.  Rename to ...
+       (grub_linux_boot_mmap_fill): ... this.
+       * grub-core/loader/i386/multiboot_mbi.c (grub_fill_multiboot_mmap:
+       hook): Likewise.  Rename to ...
+       (grub_fill_multiboot_mmap_iter): ... this.
+       * grub-core/loader/multiboot.c (grub_get_multiboot_mmap_count:
+       hook): Likewise.  Rename to ...
+       (count_hook): ... this.
+       * grub-core/loader/multiboot_mbi2.c (grub_fill_multiboot_mmap:
+       hook): Likewise.  Rename to ...
+       (grub_fill_multiboot_mmap_iter): ... this.
+       * grub-core/loader/powerpc/ieee1275/linux.c
+       (grub_linux_claimmap_iterate: alloc_mem): Likewise.
+       * grub-core/loader/sparc64/ieee1275/linux.c (alloc_phys: choose):
+       Likewise.  Rename to ...
+       (alloc_phys_choose): ... this.
+       (determine_phys_base: get_physbase): Likewise.
+       * grub-core/mmap/i386/mmap.c (grub_mmap_malign_and_register:
+       find_hook): Likewise.
+       * grub-core/mmap/i386/pc/mmap.c (preboot: fill_hook): Likewise.
+       (malloc_hook: count_hook): Likewise.
+       * grub-core/mmap/i386/uppermem.c (grub_mmap_get_lower: hook):
+       Likewise.  Rename to ...
+       (lower_hook): ... this.
+       (grub_mmap_get_upper: hook): Likewise.  Rename to ...
+       (upper_hook): ... this.
+       (grub_mmap_get_post64: hook): Likewise.  Rename to ...
+       (post64_hook): ... this.
+       * grub-core/mmap/mips/uppermem.c (grub_mmap_get_lower: hook):
+       Likewise.  Rename to ...
+       (lower_hook): ... this.
+       (grub_mmap_get_upper: hook): Likewise.  Rename to ...
+       (upper_hook): ... this.
+       * grub-core/mmap/mmap.c (grub_mmap_iterate: count_hook): Likewise.
+       (grub_mmap_iterate: fill_hook): Likewise.
+       (fill_mask): Pass addr and mask within a single struct.
+       (grub_cmd_badram: hook): Make static instead of nested.  Rename
+       to ...
+       (badram_iter): ... this.
+       (grub_cmd_cutmem: hook): Likewise.  Rename to ...
+       (cutmem_iter): ... this.
+
 2013-01-13  Vladimir Serbinenko  <address@hidden>
 
        * grub-core/kern/emu/hostdisk.c (read_device_map): Explicitly

=== modified file 'grub-core/commands/acpi.c'
--- grub-core/commands/acpi.c   2012-03-06 14:03:33 +0000
+++ grub-core/commands/acpi.c   2013-01-13 01:49:43 +0000
@@ -142,49 +142,58 @@ iszero (grub_uint8_t *reg, int size)
 }
 
 #if defined (__i386__) || defined (__x86_64__)
+/* Context for grub_acpi_create_ebda.  */
+struct grub_acpi_create_ebda_ctx {
+  int ebda_len;
+  grub_uint64_t highestlow;
+};
+
+/* Helper for grub_acpi_create_ebda.  */
+static int
+find_hook (grub_uint64_t start, grub_uint64_t size, grub_memory_type_t type,
+          void *data)
+{
+  struct grub_acpi_create_ebda_ctx *ctx = data;
+  grub_uint64_t end = start + size;
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (end > 0x100000)
+    end = 0x100000;
+  if (end > start + ctx->ebda_len
+      && ctx->highestlow < ((end - ctx->ebda_len) & (~0xf)) )
+    ctx->highestlow = (end - ctx->ebda_len) & (~0xf);
+  return 0;
+}
+
 grub_err_t
 grub_acpi_create_ebda (void)
 {
+  struct grub_acpi_create_ebda_ctx ctx = {
+    .highestlow = 0
+  };
   int ebda_kb_len;
-  int ebda_len;
   int mmapregion = 0;
   grub_uint8_t *ebda, *v1inebda = 0, *v2inebda = 0;
-  grub_uint64_t highestlow = 0;
   grub_uint8_t *targetebda, *target;
   struct grub_acpi_rsdp_v10 *v1;
   struct grub_acpi_rsdp_v20 *v2;
-  auto int NESTED_FUNC_ATTR find_hook (grub_uint64_t, grub_uint64_t,
-                                      grub_uint32_t);
-  int NESTED_FUNC_ATTR find_hook (grub_uint64_t start, grub_uint64_t size,
-                                 grub_memory_type_t type)
-  {
-    grub_uint64_t end = start + size;
-    if (type != GRUB_MEMORY_AVAILABLE)
-      return 0;
-    if (end > 0x100000)
-      end = 0x100000;
-    if (end > start + ebda_len
-       && highestlow < ((end - ebda_len) & (~0xf)) )
-      highestlow = (end - ebda_len) & (~0xf);
-    return 0;
-  }
 
   ebda = (grub_uint8_t *) (grub_addr_t) ((*((grub_uint16_t *)0x40e)) << 4);
   ebda_kb_len = *(grub_uint16_t *) ebda;
   if (! ebda || ebda_kb_len > 16)
     ebda_kb_len = 0;
-  ebda_len = (ebda_kb_len + 1) << 10;
+  ctx.ebda_len = (ebda_kb_len + 1) << 10;
 
   /* FIXME: use low-memory mm allocation once it's available. */
-  grub_mmap_iterate (find_hook);
-  targetebda = (grub_uint8_t *) (grub_addr_t) highestlow;
+  grub_mmap_iterate (find_hook, &ctx);
+  targetebda = (grub_uint8_t *) (grub_addr_t) ctx.highestlow;
   grub_dprintf ("acpi", "creating ebda @%llx\n",
-               (unsigned long long) highestlow);
-  if (! highestlow)
+               (unsigned long long) ctx.highestlow);
+  if (! ctx.highestlow)
     return grub_error (GRUB_ERR_OUT_OF_MEMORY,
                       "couldn't find space for the new EBDA");
 
-  mmapregion = grub_mmap_register ((grub_addr_t) targetebda, ebda_len,
+  mmapregion = grub_mmap_register ((grub_addr_t) targetebda, ctx.ebda_len,
                                   GRUB_MEMORY_RESERVED);
   if (! mmapregion)
     return grub_errno;

=== modified file 'grub-core/commands/lsmmap.c'
--- grub-core/commands/lsmmap.c 2012-03-02 14:09:10 +0000
+++ grub-core/commands/lsmmap.c 2013-01-13 01:49:43 +0000
@@ -21,6 +21,7 @@
 #include <grub/command.h>
 #include <grub/i18n.h>
 #include <grub/memory.h>
+#include <grub/mm.h>
 
 GRUB_MOD_LICENSE ("GPLv3+");
 
@@ -39,26 +40,30 @@ static const char *names[] =
     [GRUB_MEMORY_HOLE] = N_("Address range not associated with RAM")
   };
 
+#ifndef GRUB_MACHINE_EMU
+/* Helper for grub_cmd_lsmmap.  */
+static int
+lsmmap_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+            void *data __attribute__ ((unused)))
+{
+  if (type < ARRAY_SIZE (names) && names[type])
+    grub_printf_ (N_("base_addr = 0x%llx, length = 0x%llx, %s\n"),
+                 (long long) addr, (long long) size, _(names[type]));
+  else
+    grub_printf_ (N_("base_addr = 0x%llx, length = 0x%llx, type = 0x%x\n"),
+                 (long long) addr, (long long) size, type);
+  return 0;
+}
+#endif
+
 static grub_err_t
 grub_cmd_lsmmap (grub_command_t cmd __attribute__ ((unused)),
                 int argc __attribute__ ((unused)),
                 char **args __attribute__ ((unused)))
 
 {
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t, 
grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size, 
-                            grub_memory_type_t type)
-    {
-      if (type < ARRAY_SIZE (names) && names[type])
-       grub_printf_ (N_("base_addr = 0x%llx, length = 0x%llx, %s\n"),
-                     (long long) addr, (long long) size, _(names[type]));
-      else
-       grub_printf_ (N_("base_addr = 0x%llx, length = 0x%llx, type = 0x%x\n"),
-                     (long long) addr, (long long) size, type);
-      return 0;
-    }
 #ifndef GRUB_MACHINE_EMU
-  grub_machine_mmap_iterate (hook);
+  grub_machine_mmap_iterate (lsmmap_hook, NULL);
 #endif
 
   return 0;

=== modified file 'grub-core/efiemu/mm.c'
--- grub-core/efiemu/mm.c       2012-02-08 18:26:01 +0000
+++ grub-core/efiemu/mm.c       2013-01-13 01:49:43 +0000
@@ -268,26 +268,26 @@ grub_efiemu_mm_return_request (int handl
       }
 }
 
+/* Helper for grub_efiemu_mmap_init.  */
+static int
+bounds_hook (grub_uint64_t addr __attribute__ ((unused)),
+            grub_uint64_t size __attribute__ ((unused)),
+            grub_memory_type_t type __attribute__ ((unused)),
+            void *data __attribute__ ((unused)))
+{
+  mmap_reserved_size++;
+  return 0;
+}
+
 /* Reserve space for memory map */
 static grub_err_t
 grub_efiemu_mmap_init (void)
 {
-  auto int NESTED_FUNC_ATTR bounds_hook (grub_uint64_t, grub_uint64_t,
-                                        grub_memory_type_t);
-  int NESTED_FUNC_ATTR bounds_hook (grub_uint64_t addr __attribute__ 
((unused)),
-                                   grub_uint64_t size __attribute__ ((unused)),
-                                   grub_memory_type_t type
-                                   __attribute__ ((unused)))
-    {
-      mmap_reserved_size++;
-      return 0;
-    }
-
   // the place for memory used by efiemu itself
   mmap_reserved_size = GRUB_EFI_MAX_MEMORY_TYPE + 1;
 
 #ifndef GRUB_MACHINE_EMU
-  grub_machine_mmap_iterate (bounds_hook);
+  grub_machine_mmap_iterate (bounds_hook, NULL);
 #endif
 
   return GRUB_ERR_NONE;
@@ -383,48 +383,47 @@ grub_efiemu_mm_init (void)
   return GRUB_ERR_NONE;
 }
 
+/* Helper for grub_efiemu_mmap_fill.  */
+static int
+fill_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+          void *data __attribute__ ((unused)))
+  {
+    switch (type)
+      {
+      case GRUB_MEMORY_AVAILABLE:
+       return grub_efiemu_add_to_mmap (addr, size,
+                                       GRUB_EFI_CONVENTIONAL_MEMORY);
+
+      case GRUB_MEMORY_ACPI:
+       return grub_efiemu_add_to_mmap (addr, size,
+                                       GRUB_EFI_ACPI_RECLAIM_MEMORY);
+
+      case GRUB_MEMORY_NVS:
+       return grub_efiemu_add_to_mmap (addr, size,
+                                       GRUB_EFI_ACPI_MEMORY_NVS);
+
+      default:
+       grub_dprintf ("efiemu",
+                     "Unknown memory type %d. Assuming unusable\n", type);
+      case GRUB_MEMORY_RESERVED:
+       return grub_efiemu_add_to_mmap (addr, size,
+                                       GRUB_EFI_UNUSABLE_MEMORY);
+      }
+  }
+
 /* Copy host memory map */
 static grub_err_t
 grub_efiemu_mmap_fill (void)
 {
-  auto int NESTED_FUNC_ATTR fill_hook (grub_uint64_t, grub_uint64_t,
-                                      grub_memory_type_t);
-  int NESTED_FUNC_ATTR fill_hook (grub_uint64_t addr,
-                                 grub_uint64_t size,
-                                 grub_memory_type_t type)
-    {
-      switch (type)
-       {
-       case GRUB_MEMORY_AVAILABLE:
-         return grub_efiemu_add_to_mmap (addr, size,
-                                         GRUB_EFI_CONVENTIONAL_MEMORY);
-
-       case GRUB_MEMORY_ACPI:
-         return grub_efiemu_add_to_mmap (addr, size,
-                                         GRUB_EFI_ACPI_RECLAIM_MEMORY);
-
-       case GRUB_MEMORY_NVS:
-         return grub_efiemu_add_to_mmap (addr, size,
-                                         GRUB_EFI_ACPI_MEMORY_NVS);
-
-       default:
-         grub_dprintf ("efiemu",
-                       "Unknown memory type %d. Assuming unusable\n", type);
-       case GRUB_MEMORY_RESERVED:
-         return grub_efiemu_add_to_mmap (addr, size,
-                                         GRUB_EFI_UNUSABLE_MEMORY);
-       }
-    }
-
 #ifndef GRUB_MACHINE_EMU
-  grub_machine_mmap_iterate (fill_hook);
+  grub_machine_mmap_iterate (fill_hook, NULL);
 #endif
 
   return GRUB_ERR_NONE;
 }
 
 grub_err_t
-grub_efiemu_mmap_iterate (grub_memory_hook_t hook)
+grub_efiemu_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   unsigned i;
 
@@ -433,12 +432,12 @@ grub_efiemu_mmap_iterate (grub_memory_ho
       {
       case GRUB_EFI_RUNTIME_SERVICES_CODE:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_CODE);
+             GRUB_MEMORY_CODE, hook_data);
        break;
 
       case GRUB_EFI_UNUSABLE_MEMORY:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_BADRAM);
+             GRUB_MEMORY_BADRAM, hook_data);
        break;
 
       case GRUB_EFI_RESERVED_MEMORY_TYPE:
@@ -448,7 +447,7 @@ grub_efiemu_mmap_iterate (grub_memory_ho
       case GRUB_EFI_PAL_CODE:
       case GRUB_EFI_MAX_MEMORY_TYPE:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_RESERVED);
+             GRUB_MEMORY_RESERVED, hook_data);
        break;
 
       case GRUB_EFI_LOADER_CODE:
@@ -457,17 +456,17 @@ grub_efiemu_mmap_iterate (grub_memory_ho
       case GRUB_EFI_BOOT_SERVICES_DATA:
       case GRUB_EFI_CONVENTIONAL_MEMORY:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_AVAILABLE);
+             GRUB_MEMORY_AVAILABLE, hook_data);
        break;
 
       case GRUB_EFI_ACPI_RECLAIM_MEMORY:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_ACPI);
+             GRUB_MEMORY_ACPI, hook_data);
        break;
 
       case GRUB_EFI_ACPI_MEMORY_NVS:
        hook (efiemu_mmap[i].physical_start, efiemu_mmap[i].num_pages * 4096,
-             GRUB_MEMORY_NVS);
+             GRUB_MEMORY_NVS, hook_data);
        break;
       }
 

=== modified file 'grub-core/kern/i386/coreboot/init.c'
--- grub-core/kern/i386/coreboot/init.c 2012-06-11 18:44:38 +0000
+++ grub-core/kern/i386/coreboot/init.c 2013-01-13 01:49:43 +0000
@@ -57,6 +57,39 @@ grub_addr_t grub_modbase;
 grub_addr_t grub_modbase = ALIGN_UP((grub_addr_t) _end, 
GRUB_KERNEL_MACHINE_MOD_ALIGN);
 #endif
 
+/* Helper for grub_machine_init.  */
+static int
+heap_init (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+          void *data __attribute__ ((unused)))
+{
+#if GRUB_CPU_SIZEOF_VOID_P == 4
+  /* Restrict ourselves to 32-bit memory space.  */
+  if (addr > GRUB_ULONG_MAX)
+    return 0;
+  if (addr + size > GRUB_ULONG_MAX)
+    size = GRUB_ULONG_MAX - addr;
+#endif
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+
+  /* Avoid the lower memory.  */
+  if (addr < GRUB_MEMORY_MACHINE_LOWER_SIZE)
+    {
+      if (addr + size <= GRUB_MEMORY_MACHINE_LOWER_SIZE)
+       return 0;
+      else
+       {
+         size -= GRUB_MEMORY_MACHINE_LOWER_SIZE - addr;
+         addr = GRUB_MEMORY_MACHINE_LOWER_SIZE;
+       }
+    }
+
+  grub_mm_init_region ((void *) (grub_addr_t) addr, (grub_size_t) size);
+
+  return 0;
+}
+
 void
 grub_machine_init (void)
 {
@@ -68,43 +101,10 @@ grub_machine_init (void)
   /* Initialize the console as early as possible.  */
   grub_vga_text_init ();
 
-  auto int NESTED_FUNC_ATTR heap_init (grub_uint64_t, grub_uint64_t, 
-                                      grub_memory_type_t);
-  int NESTED_FUNC_ATTR heap_init (grub_uint64_t addr, grub_uint64_t size,
-                                 grub_memory_type_t type)
-  {
-#if GRUB_CPU_SIZEOF_VOID_P == 4
-    /* Restrict ourselves to 32-bit memory space.  */
-    if (addr > GRUB_ULONG_MAX)
-      return 0;
-    if (addr + size > GRUB_ULONG_MAX)
-      size = GRUB_ULONG_MAX - addr;
-#endif
-
-    if (type != GRUB_MEMORY_AVAILABLE)
-      return 0;
-
-    /* Avoid the lower memory.  */
-    if (addr < GRUB_MEMORY_MACHINE_LOWER_SIZE)
-      {
-       if (addr + size <= GRUB_MEMORY_MACHINE_LOWER_SIZE)
-         return 0;
-       else
-         {
-           size -= GRUB_MEMORY_MACHINE_LOWER_SIZE - addr;
-           addr = GRUB_MEMORY_MACHINE_LOWER_SIZE;
-         }
-      }
-
-    grub_mm_init_region ((void *) (grub_addr_t) addr, (grub_size_t) size);
-
-    return 0;
-  }
-
 #if defined (GRUB_MACHINE_MULTIBOOT) || defined (GRUB_MACHINE_QEMU)
   grub_machine_mmap_init ();
 #endif
-  grub_machine_mmap_iterate (heap_init);
+  grub_machine_mmap_iterate (heap_init, NULL);
 
   grub_tsc_init ();
 }

=== modified file 'grub-core/kern/i386/coreboot/mmap.c'
--- grub-core/kern/i386/coreboot/mmap.c 2012-12-31 17:31:38 +0000
+++ grub-core/kern/i386/coreboot/mmap.c 2013-01-13 01:49:43 +0000
@@ -33,7 +33,9 @@ check_signature (grub_linuxbios_table_he
 }
 
 static grub_err_t
-grub_linuxbios_table_iterate (int (*hook) (grub_linuxbios_table_item_t))
+grub_linuxbios_table_iterate (int (*hook) (grub_linuxbios_table_item_t,
+                                          void *),
+                             void *hook_data)
 {
   grub_linuxbios_table_header_t table_header;
   grub_linuxbios_table_item_t table_item;
@@ -68,42 +70,54 @@ signature_found:
            *(grub_uint64_t *) (table_item + 1);
          goto signature_found;   
        }
-      if (hook (table_item))
+      if (hook (table_item, hook_data))
        return 1;
     }
 
   return 0;
 }
 
-grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+/* Context for grub_machine_mmap_iterate.  */
+struct grub_machine_mmap_iterate_ctx
+{
+  grub_memory_hook_t hook;
+  void *hook_data;
+};
+
+/* Helper for grub_machine_mmap_iterate.  */
+static int
+iterate_linuxbios_table (grub_linuxbios_table_item_t table_item, void *data)
 {
+  struct grub_machine_mmap_iterate_ctx *ctx = data;
   mem_region_t mem_region;
 
-  auto int iterate_linuxbios_table (grub_linuxbios_table_item_t);
-  int iterate_linuxbios_table (grub_linuxbios_table_item_t table_item)
-  {
-    if (table_item->tag != GRUB_LINUXBIOS_MEMBER_MEMORY)
-      return 0;
-
-    mem_region =
-      (mem_region_t) ((long) table_item +
-                                sizeof (struct grub_linuxbios_table_item));
-    while ((long) mem_region < (long) table_item + (long) table_item->size)
-      {
-       if (hook (mem_region->addr, mem_region->size,
-                 /* Multiboot mmaps match with the coreboot mmap definition.
-                    Therefore, we can just pass type through.  */
-                 mem_region->type))
-         return 1;
+  if (table_item->tag != GRUB_LINUXBIOS_MEMBER_MEMORY)
+    return 0;
 
-       mem_region++;
-      }
+  mem_region =
+    (mem_region_t) ((long) table_item +
+                              sizeof (struct grub_linuxbios_table_item));
+  while ((long) mem_region < (long) table_item + (long) table_item->size)
+    {
+      if (ctx->hook (mem_region->addr, mem_region->size,
+                    /* Multiboot mmaps match with the coreboot mmap
+                       definition.  Therefore, we can just pass type
+                       through.  */
+                    mem_region->type, ctx->hook_data))
+       return 1;
 
-    return 0;
-  }
+      mem_region++;
+    }
+
+  return 0;
+}
+
+grub_err_t
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
+{
+  struct grub_machine_mmap_iterate_ctx ctx = { hook, hook_data };
 
-  grub_linuxbios_table_iterate (iterate_linuxbios_table);
+  grub_linuxbios_table_iterate (iterate_linuxbios_table, &ctx);
 
   return 0;
 }

=== modified file 'grub-core/kern/i386/multiboot_mmap.c'
--- grub-core/kern/i386/multiboot_mmap.c        2012-02-10 15:48:48 +0000
+++ grub-core/kern/i386/multiboot_mmap.c        2013-01-13 01:49:43 +0000
@@ -57,13 +57,13 @@ grub_machine_mmap_init (void)
 }
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   struct multiboot_mmap_entry *entry = (void *) kern_multiboot_info.mmap_addr;
 
   while ((unsigned long) entry < kern_multiboot_info.mmap_addr + 
kern_multiboot_info.mmap_length)
     {
-      if (hook (entry->addr, entry->len, entry->type))
+      if (hook (entry->addr, entry->len, entry->type, hook_data))
        break;
 
       entry = (void *) ((grub_addr_t) entry + entry->size + sizeof 
(entry->size));

=== modified file 'grub-core/kern/i386/pc/init.c'
--- grub-core/kern/i386/pc/init.c       2012-06-11 18:44:38 +0000
+++ grub-core/kern/i386/pc/init.c       2013-01-13 01:49:43 +0000
@@ -154,6 +154,36 @@ compact_mem_regions (void)
 grub_addr_t grub_modbase;
 extern grub_uint8_t _start[], _edata[];
 
+/* Helper for grub_machine_init.  */
+static int
+mmap_iterate_hook (grub_uint64_t addr, grub_uint64_t size,
+                  grub_memory_type_t type,
+                  void *data __attribute__ ((unused)))
+{
+  /* Avoid the lower memory.  */
+  if (addr < 0x100000)
+    {
+      if (size <= 0x100000 - addr)
+       return 0;
+
+      size -= 0x100000 - addr;
+      addr = 0x100000;
+    }
+
+  /* Ignore >4GB.  */
+  if (addr <= 0xFFFFFFFF && type == GRUB_MEMORY_AVAILABLE)
+    {
+      grub_size_t len;
+
+      len = (grub_size_t) ((addr + size > 0xFFFFFFFF)
+            ? 0xFFFFFFFF - addr
+            : size);
+      add_mem_region (addr, len);
+    }
+
+  return 0;
+}
+
 void
 grub_machine_init (void)
 {
@@ -188,36 +218,7 @@ grub_machine_init (void)
                    grub_lower_mem - GRUB_MEMORY_MACHINE_RESERVED_END);
 #endif
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      /* Avoid the lower memory.  */
-      if (addr < 0x100000)
-       {
-         if (size <= 0x100000 - addr)
-           return 0;
-
-         size -= 0x100000 - addr;
-         addr = 0x100000;
-       }
-
-      /* Ignore >4GB.  */
-      if (addr <= 0xFFFFFFFF && type == GRUB_MEMORY_AVAILABLE)
-       {
-         grub_size_t len;
-
-         len = (grub_size_t) ((addr + size > 0xFFFFFFFF)
-                ? 0xFFFFFFFF - addr
-                : size);
-         add_mem_region (addr, len);
-       }
-
-      return 0;
-    }
-
-  grub_machine_mmap_iterate (hook);
+  grub_machine_mmap_iterate (mmap_iterate_hook, NULL);
 
   compact_mem_regions ();
 

=== modified file 'grub-core/kern/i386/pc/mmap.c'
--- grub-core/kern/i386/pc/mmap.c       2011-01-10 16:51:06 +0000
+++ grub-core/kern/i386/pc/mmap.c       2013-01-13 01:49:43 +0000
@@ -139,7 +139,7 @@ grub_get_mmap_entry (struct grub_machine
 }
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   grub_uint32_t cont;
   struct grub_machine_mmap_entry *entry
@@ -156,7 +156,8 @@ grub_machine_mmap_iterate (grub_memory_h
        if (hook (entry->addr, entry->len,
                  /* GRUB mmaps have been defined to match with the E820 
definition.
                     Therefore, we can just pass type through.  */
-                 ((entry->type <= GRUB_MACHINE_MEMORY_BADRAM) && (entry->type 
>= GRUB_MACHINE_MEMORY_AVAILABLE)) ? entry->type : GRUB_MEMORY_RESERVED))
+                 ((entry->type <= GRUB_MACHINE_MEMORY_BADRAM) && (entry->type 
>= GRUB_MACHINE_MEMORY_AVAILABLE)) ? entry->type : GRUB_MEMORY_RESERVED,
+                 hook_data))
          break;
 
        if (! cont)
@@ -172,18 +173,19 @@ grub_machine_mmap_iterate (grub_memory_h
       grub_uint32_t eisa_mmap = grub_get_eisa_mmap ();
 
       if (hook (0x0, ((grub_uint32_t) grub_get_conv_memsize ()) << 10,
-               GRUB_MEMORY_AVAILABLE))
+               GRUB_MEMORY_AVAILABLE, hook_data))
        return 0;
 
       if (eisa_mmap)
        {
          if (hook (0x100000, (eisa_mmap & 0xFFFF) << 10,
-                   GRUB_MEMORY_AVAILABLE) == 0)
-           hook (0x1000000, eisa_mmap & ~0xFFFF, GRUB_MEMORY_AVAILABLE);
+                   GRUB_MEMORY_AVAILABLE, hook_data) == 0)
+           hook (0x1000000, eisa_mmap & ~0xFFFF, GRUB_MEMORY_AVAILABLE,
+                 hook_data);
        }
       else
        hook (0x100000, ((grub_uint32_t) grub_get_ext_memsize ()) << 10,
-             GRUB_MEMORY_AVAILABLE);
+             GRUB_MEMORY_AVAILABLE, hook_data);
     }
 
   return 0;

=== modified file 'grub-core/kern/i386/qemu/mmap.c'
--- grub-core/kern/i386/qemu/mmap.c     2012-02-09 22:25:14 +0000
+++ grub-core/kern/i386/qemu/mmap.c     2013-01-13 01:49:43 +0000
@@ -69,38 +69,38 @@ grub_machine_mmap_init (void)
 }
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   if (hook (0x0,
            (grub_addr_t) _start,
-           GRUB_MEMORY_AVAILABLE))
+           GRUB_MEMORY_AVAILABLE, hook_data))
     return 1;
 
   if (hook ((grub_addr_t) _end,
            0xa0000 - (grub_addr_t) _end,
-           GRUB_MEMORY_AVAILABLE))
+           GRUB_MEMORY_AVAILABLE, hook_data))
     return 1;
 
   if (hook (GRUB_MEMORY_MACHINE_UPPER,
            0x100000 - GRUB_MEMORY_MACHINE_UPPER,
-           GRUB_MEMORY_RESERVED))
+           GRUB_MEMORY_RESERVED, hook_data))
     return 1;
 
   /* Everything else is free.  */
   if (hook (0x100000,
            min (mem_size, (grub_uint32_t) -GRUB_BOOT_MACHINE_SIZE) - 0x100000,
-           GRUB_MEMORY_AVAILABLE))
+           GRUB_MEMORY_AVAILABLE, hook_data))
     return 1;
 
   /* Protect boot.img, which contains the gdt.  It is mapped at the top of 
memory
      (it is also mapped below 0x100000, but we already reserved that area).  */
   if (hook ((grub_uint32_t) -GRUB_BOOT_MACHINE_SIZE,
            GRUB_BOOT_MACHINE_SIZE,
-           GRUB_MEMORY_RESERVED))
+           GRUB_MEMORY_RESERVED, hook_data))
     return 1;
 
   if (above_4g != 0 && hook (0x100000000ULL, above_4g,
-                            GRUB_MEMORY_AVAILABLE))
+                            GRUB_MEMORY_AVAILABLE, hook_data))
     return 1;
 
   return 0;

=== modified file 'grub-core/kern/ieee1275/init.c'
--- grub-core/kern/ieee1275/init.c      2012-12-10 15:23:16 +0000
+++ grub-core/kern/ieee1275/init.c      2013-01-13 01:49:43 +0000
@@ -156,74 +156,76 @@ grub_claim_heap (void)
                                 + GRUB_KERNEL_MACHINE_STACK_SIZE), 0x200000);
 }
 #else
+/* Helper for grub_claim_heap.  */
+static int
+heap_init (grub_uint64_t addr, grub_uint64_t len, grub_memory_type_t type,
+          void *data)
+{
+  unsigned long *total = data;
+
+  if (type != 1)
+    return 0;
+
+  if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_NO_PRE1_5M_CLAIM))
+    {
+      if (addr + len <= 0x180000)
+       return 0;
+
+      if (addr < 0x180000)
+       {
+         len = addr + len - 0x180000;
+         addr = 0x180000;
+       }
+    }
+  len -= 1; /* Required for some firmware.  */
+
+  /* Never exceed HEAP_MAX_SIZE  */
+  if (*total + len > HEAP_MAX_SIZE)
+    len = HEAP_MAX_SIZE - *total;
+
+  /* Avoid claiming anything above HEAP_MAX_ADDR, if possible. */
+  if ((addr < HEAP_MAX_ADDR) &&                                /* if it's too 
late, don't bother */
+      (addr + len > HEAP_MAX_ADDR) &&                          /* if it wasn't 
available anyway, don't bother */
+      (*total + (HEAP_MAX_ADDR - addr) > HEAP_MIN_SIZE))       /* only limit 
ourselves when we can afford to */
+     len = HEAP_MAX_ADDR - addr;
+
+  /* In theory, firmware should already prevent this from happening by not
+     listing our own image in /memory/available.  The check below is intended
+     as a safeguard in case that doesn't happen.  However, it doesn't protect
+     us from corrupting our module area, which extends up to a
+     yet-undetermined region above _end.  */
+  if ((addr < (grub_addr_t) _end) && ((addr + len) > (grub_addr_t) _start))
+    {
+      grub_printf ("Warning: attempt to claim over our own code!\n");
+      len = 0;
+    }
+
+  if (len)
+    {
+      grub_err_t err;
+      /* Claim and use it.  */
+      err = grub_claimmap (addr, len);
+      if (err)
+       return err;
+      grub_mm_init_region ((void *) (grub_addr_t) addr, len);
+    }
+
+  *total += len;
+  if (*total >= HEAP_MAX_SIZE)
+    return 1;
+
+  return 0;
+}
+
 static void 
 grub_claim_heap (void)
 {
   unsigned long total = 0;
 
-  auto int NESTED_FUNC_ATTR heap_init (grub_uint64_t addr, grub_uint64_t len,
-                                      grub_memory_type_t type);
-  int NESTED_FUNC_ATTR heap_init (grub_uint64_t addr, grub_uint64_t len,
-                                 grub_memory_type_t type)
-  {
-    if (type != 1)
-      return 0;
-
-    if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_NO_PRE1_5M_CLAIM))
-      {
-       if (addr + len <= 0x180000)
-         return 0;
-
-       if (addr < 0x180000)
-         {
-           len = addr + len - 0x180000;
-           addr = 0x180000;
-         }
-      }
-    len -= 1; /* Required for some firmware.  */
-
-    /* Never exceed HEAP_MAX_SIZE  */
-    if (total + len > HEAP_MAX_SIZE)
-      len = HEAP_MAX_SIZE - total;
-
-    /* Avoid claiming anything above HEAP_MAX_ADDR, if possible. */
-    if ((addr < HEAP_MAX_ADDR) &&                              /* if it's too 
late, don't bother */
-        (addr + len > HEAP_MAX_ADDR) &&                                /* if 
it wasn't available anyway, don't bother */
-        (total + (HEAP_MAX_ADDR - addr) > HEAP_MIN_SIZE))      /* only limit 
ourselves when we can afford to */
-       len = HEAP_MAX_ADDR - addr;
-
-    /* In theory, firmware should already prevent this from happening by not
-       listing our own image in /memory/available.  The check below is intended
-       as a safeguard in case that doesn't happen.  However, it doesn't protect
-       us from corrupting our module area, which extends up to a
-       yet-undetermined region above _end.  */
-    if ((addr < (grub_addr_t) _end) && ((addr + len) > (grub_addr_t) _start))
-      {
-        grub_printf ("Warning: attempt to claim over our own code!\n");
-        len = 0;
-      }
-
-    if (len)
-      {
-       grub_err_t err;
-       /* Claim and use it.  */
-       err = grub_claimmap (addr, len);
-       if (err)
-         return err;
-       grub_mm_init_region ((void *) (grub_addr_t) addr, len);
-      }
-
-    total += len;
-    if (total >= HEAP_MAX_SIZE)
-      return 1;
-
-    return 0;
-  }
-
   if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_CANNOT_INTERPRET))
-    heap_init (HEAP_MAX_ADDR - HEAP_MIN_SIZE, HEAP_MIN_SIZE, 1);
+    heap_init (HEAP_MAX_ADDR - HEAP_MIN_SIZE, HEAP_MIN_SIZE, 1, &total);
   else
-    grub_machine_mmap_iterate (heap_init);
+    grub_machine_mmap_iterate (heap_init, &total);
 }
 #endif
 

=== modified file 'grub-core/kern/ieee1275/mmap.c'
--- grub-core/kern/ieee1275/mmap.c      2011-02-12 07:22:55 +0000
+++ grub-core/kern/ieee1275/mmap.c      2013-01-13 01:49:43 +0000
@@ -21,7 +21,7 @@
 #include <grub/types.h>
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   grub_ieee1275_phandle_t root;
   grub_ieee1275_phandle_t memory;
@@ -72,7 +72,7 @@ grub_machine_mmap_iterate (grub_memory_h
       if (size_cells == 2)
        size = (size << 32) | available[i++];
 
-      if (hook (address, size, GRUB_MEMORY_AVAILABLE))
+      if (hook (address, size, GRUB_MEMORY_AVAILABLE, hook_data))
        break;
     }
 

=== modified file 'grub-core/kern/mips/arc/init.c'
--- grub-core/kern/mips/arc/init.c      2012-06-26 21:28:23 +0000
+++ grub-core/kern/mips/arc/init.c      2013-01-13 01:49:43 +0000
@@ -91,7 +91,7 @@ grub_arc_iterate_devs (int (*hook) (cons
 }
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   struct grub_arc_memory_descriptor *cur = NULL;
   while (1)
@@ -120,7 +120,7 @@ grub_machine_mmap_iterate (grub_memory_h
          break;
        }
       if (hook (((grub_uint64_t) cur->start_page) << 12,
-               ((grub_uint64_t) cur->num_pages)  << 12, type))
+               ((grub_uint64_t) cur->num_pages)  << 12, type, hook_data))
        return GRUB_ERR_NONE;
     }
 }

=== modified file 'grub-core/kern/mips/loongson/init.c'
--- grub-core/kern/mips/loongson/init.c 2013-01-13 01:10:41 +0000
+++ grub-core/kern/mips/loongson/init.c 2013-01-13 01:49:43 +0000
@@ -40,12 +40,12 @@
 #include <grub/at_keyboard.h>
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
   hook (GRUB_ARCH_LOWMEMPSTART, grub_arch_memsize << 20,
-       GRUB_MEMORY_AVAILABLE);
+       GRUB_MEMORY_AVAILABLE, hook_data);
   hook (GRUB_ARCH_HIGHMEMPSTART, grub_arch_highmemsize << 20,
-       GRUB_MEMORY_AVAILABLE);
+       GRUB_MEMORY_AVAILABLE, hook_data);
   return GRUB_ERR_NONE;
 }
 

=== modified file 'grub-core/kern/mips/qemu_mips/init.c'
--- grub-core/kern/mips/qemu_mips/init.c        2012-06-06 10:28:25 +0000
+++ grub-core/kern/mips/qemu_mips/init.c        2013-01-13 01:49:43 +0000
@@ -92,9 +92,9 @@ grub_halt (void)
 }
 
 grub_err_t 
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
-  hook (0, grub_arch_memsize, GRUB_MEMORY_AVAILABLE);
+  hook (0, grub_arch_memsize, GRUB_MEMORY_AVAILABLE, hook_data);
   return GRUB_ERR_NONE;
 }
 

=== modified file 'grub-core/lib/ieee1275/relocator.c'
--- grub-core/lib/ieee1275/relocator.c  2012-02-04 13:00:23 +0000
+++ grub-core/lib/ieee1275/relocator.c  2013-01-13 01:49:43 +0000
@@ -21,65 +21,81 @@
 #include <grub/memory.h>
 #include <grub/ieee1275/ieee1275.h>
 
+/* Helper for grub_relocator_firmware_get_max_events.  */
+static int
+count (grub_uint64_t addr __attribute__ ((unused)),
+       grub_uint64_t len __attribute__ ((unused)),
+       grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  int *counter = data;
+
+  (*counter)++;
+  return 0;
+}
+
 unsigned 
 grub_relocator_firmware_get_max_events (void)
 {
   int counter = 0;
-  auto int NESTED_FUNC_ATTR count (grub_uint64_t addr __attribute__ ((unused)),
-                                  grub_uint64_t len __attribute__ ((unused)),
-                                  grub_memory_type_t type __attribute__ 
((unused)));
-  int NESTED_FUNC_ATTR count (grub_uint64_t addr __attribute__ ((unused)), 
-                             grub_uint64_t len __attribute__ ((unused)), 
-                             grub_memory_type_t type __attribute__ ((unused)))
-  {
-    counter++;
-    return 0;
-  }
 
   if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_CANNOT_INTERPRET))
     return 0;
-  grub_machine_mmap_iterate (count);
+  grub_machine_mmap_iterate (count, &counter);
   return 2 * counter;
 }
 
-unsigned 
-grub_relocator_firmware_fill_events (struct grub_relocator_mmap_event *events)
+/* Context for grub_relocator_firmware_fill_events.  */
+struct grub_relocator_firmware_fill_events_ctx
 {
-  int counter = 0;
-  auto int NESTED_FUNC_ATTR fill (grub_uint64_t addr, grub_uint64_t len,
-                                 grub_memory_type_t type);
-  int NESTED_FUNC_ATTR fill (grub_uint64_t addr, grub_uint64_t len,
-                            grub_memory_type_t type)
-  {
-    if (type != GRUB_MEMORY_AVAILABLE)
-      return 0;
-
-    if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_NO_PRE1_5M_CLAIM))
-      {
-       if (addr + len <= 0x180000)
-         return 0;
-
-       if (addr < 0x180000)
-         {
-           len = addr + len - 0x180000;
-           addr = 0x180000;
-         }
-      }
-
-    events[counter].type = REG_FIRMWARE_START;
-    events[counter].pos = addr;
-    counter++;
-    events[counter].type = REG_FIRMWARE_END;
-    events[counter].pos = addr + len;
-    counter++;
+  struct grub_relocator_mmap_event *events;
+  int counter;
+};
+
+/* Helper for grub_relocator_firmware_fill_events.  */
+static int
+grub_relocator_firmware_fill_events_iter (grub_uint64_t addr,
+                                         grub_uint64_t len,
+                                         grub_memory_type_t type, void *data)
+{
+  struct grub_relocator_firmware_fill_events_ctx *ctx = data;
 
+  if (type != GRUB_MEMORY_AVAILABLE)
     return 0;
-  }
+
+  if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_NO_PRE1_5M_CLAIM))
+    {
+      if (addr + len <= 0x180000)
+       return 0;
+
+      if (addr < 0x180000)
+       {
+         len = addr + len - 0x180000;
+         addr = 0x180000;
+       }
+    }
+
+  ctx->events[ctx->counter].type = REG_FIRMWARE_START;
+  ctx->events[ctx->counter].pos = addr;
+  ctx->counter++;
+  ctx->events[ctx->counter].type = REG_FIRMWARE_END;
+  ctx->events[ctx->counter].pos = addr + len;
+  ctx->counter++;
+
+  return 0;
+}
+
+unsigned 
+grub_relocator_firmware_fill_events (struct grub_relocator_mmap_event *events)
+{
+  struct grub_relocator_firmware_fill_events_ctx ctx = {
+    .events = events,
+    .counter = 0
+  };
 
   if (grub_ieee1275_test_flag (GRUB_IEEE1275_FLAG_CANNOT_INTERPRET))
     return 0;
-  grub_machine_mmap_iterate (fill);
-  return counter;
+  grub_machine_mmap_iterate (grub_relocator_firmware_fill_events_iter, &ctx);
+  return ctx.counter;
 }
 
 int

=== modified file 'grub-core/lib/relocator.c'
--- grub-core/lib/relocator.c   2012-04-13 14:55:20 +0000
+++ grub-core/lib/relocator.c   2013-01-13 01:49:43 +0000
@@ -1313,6 +1313,45 @@ grub_relocator_alloc_chunk_addr (struct
   return GRUB_ERR_NONE;
 }
 
+/* Context for grub_relocator_alloc_chunk_align.  */
+struct grub_relocator_alloc_chunk_align_ctx
+{
+  grub_phys_addr_t min_addr, max_addr;
+  grub_size_t size, align;
+  int preference;
+  struct grub_relocator_chunk *chunk;
+  int found;
+};
+
+/* Helper for grub_relocator_alloc_chunk_align.  */
+static int
+grub_relocator_alloc_chunk_align_iter (grub_uint64_t addr, grub_uint64_t sz,
+                                      grub_memory_type_t type, void *data)
+{
+  struct grub_relocator_alloc_chunk_align_ctx *ctx = data;
+  grub_uint64_t candidate;
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  candidate = ALIGN_UP (addr, ctx->align);
+  if (candidate < ctx->min_addr)
+    candidate = ALIGN_UP (ctx->min_addr, ctx->align);
+  if (candidate + ctx->size > addr + sz
+      || candidate > ALIGN_DOWN (ctx->max_addr, ctx->align))
+    return 0;
+  if (ctx->preference == GRUB_RELOCATOR_PREFERENCE_HIGH)
+    candidate = ALIGN_DOWN (min (addr + sz - ctx->size, ctx->max_addr),
+                           ctx->align);
+  if (!ctx->found || (ctx->preference == GRUB_RELOCATOR_PREFERENCE_HIGH
+                     && candidate > ctx->chunk->target))
+    ctx->chunk->target = candidate;
+  if (!ctx->found || (ctx->preference == GRUB_RELOCATOR_PREFERENCE_LOW
+                     && candidate < ctx->chunk->target))
+    ctx->chunk->target = candidate;
+  ctx->found = 1;
+  return 0;
+}
+
 grub_err_t
 grub_relocator_alloc_chunk_align (struct grub_relocator *rel,
                                  grub_relocator_chunk_t *out,
@@ -1322,8 +1361,15 @@ grub_relocator_alloc_chunk_align (struct
                                  int preference,
                                  int avoid_efi_boot_services)
 {
+  struct grub_relocator_alloc_chunk_align_ctx ctx = {
+    .min_addr = min_addr,
+    .max_addr = max_addr,
+    .size = size,
+    .align = align,
+    .preference = preference,
+    .found = 0
+  };
   grub_addr_t min_addr2 = 0, max_addr2;
-  struct grub_relocator_chunk *chunk;
 
   if (max_addr > ~size)
     max_addr = ~size;
@@ -1335,24 +1381,24 @@ grub_relocator_alloc_chunk_align (struct
 
   grub_dprintf ("relocator", "chunks = %p\n", rel->chunks);
 
-  chunk = grub_malloc (sizeof (struct grub_relocator_chunk));
-  if (!chunk)
+  ctx.chunk = grub_malloc (sizeof (struct grub_relocator_chunk));
+  if (!ctx.chunk)
     return grub_errno;
 
   if (malloc_in_range (rel, min_addr, max_addr, align,
-                      size, chunk,
+                      size, ctx.chunk,
                       preference != GRUB_RELOCATOR_PREFERENCE_HIGH, 1))
     {
       grub_dprintf ("relocator", "allocated 0x%llx/0x%llx\n",
-                   (unsigned long long) chunk->src,
-                   (unsigned long long) chunk->src);
+                   (unsigned long long) ctx.chunk->src,
+                   (unsigned long long) ctx.chunk->src);
       grub_dprintf ("relocator", "chunks = %p\n", rel->chunks);
-      chunk->target = chunk->src;
-      chunk->size = size;
-      chunk->next = rel->chunks;
-      rel->chunks = chunk;
-      chunk->srcv = grub_map_memory (chunk->src, chunk->size);
-      *out = chunk;
+      ctx.chunk->target = ctx.chunk->src;
+      ctx.chunk->size = size;
+      ctx.chunk->next = rel->chunks;
+      rel->chunks = ctx.chunk;
+      ctx.chunk->srcv = grub_map_memory (ctx.chunk->src, ctx.chunk->size);
+      *out = ctx.chunk;
       return GRUB_ERR_NONE;
     }
 
@@ -1364,14 +1410,14 @@ grub_relocator_alloc_chunk_align (struct
   do
     {
       if (malloc_in_range (rel, min_addr2, max_addr2, align,
-                          size, chunk, 1, 1))
+                          size, ctx.chunk, 1, 1))
        break;
 
       if (malloc_in_range (rel, rel->highestnonpostaddr, ~(grub_addr_t)0, 1,
-                          size, chunk, 0, 1))
+                          size, ctx.chunk, 0, 1))
        {
-         if (rel->postchunks > chunk->src)
-           rel->postchunks = chunk->src;
+         if (rel->postchunks > ctx.chunk->src)
+           rel->postchunks = ctx.chunk->src;
          break;
        }
 
@@ -1380,58 +1426,33 @@ grub_relocator_alloc_chunk_align (struct
   while (0);
 
   {
-    int found = 0;
-    auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                   grub_memory_type_t);
-    int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t sz,
-                              grub_memory_type_t type)
-    {
-      grub_uint64_t candidate;
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      candidate = ALIGN_UP (addr, align);
-      if (candidate < min_addr)
-       candidate = ALIGN_UP (min_addr, align);
-      if (candidate + size > addr + sz
-         || candidate > ALIGN_DOWN (max_addr, align))
-       return 0;
-      if (preference == GRUB_RELOCATOR_PREFERENCE_HIGH)
-       candidate = ALIGN_DOWN (min (addr + sz - size, max_addr), align);
-      if (!found || (preference == GRUB_RELOCATOR_PREFERENCE_HIGH
-                    && candidate > chunk->target))
-       chunk->target = candidate;
-      if (!found || (preference == GRUB_RELOCATOR_PREFERENCE_LOW
-                    && candidate < chunk->target))
-       chunk->target = candidate;
-      found = 1;
-      return 0;
-    }
-
 #ifdef GRUB_MACHINE_EFI
-    grub_efi_mmap_iterate (hook, avoid_efi_boot_services);
+    grub_efi_mmap_iterate (grub_relocator_alloc_chunk_align_iter, &ctx,
+                          avoid_efi_boot_services);
 #elif defined (__powerpc__)
     (void) avoid_efi_boot_services;
-    grub_machine_mmap_iterate (hook);
+    grub_machine_mmap_iterate (grub_relocator_alloc_chunk_align_iter, &ctx);
 #else
     (void) avoid_efi_boot_services;
-    grub_mmap_iterate (hook);
+    grub_mmap_iterate (grub_relocator_alloc_chunk_align_iter, &ctx);
 #endif
-    if (!found)
+    if (!ctx.found)
       return grub_error (GRUB_ERR_BAD_OS, "couldn't find suitable memory 
target");
   }
   while (1)
     {
       struct grub_relocator_chunk *chunk2;
       for (chunk2 = rel->chunks; chunk2; chunk2 = chunk2->next)
-       if ((chunk2->target <= chunk->target
-            && chunk->target < chunk2->target + chunk2->size)
-           || (chunk->target <= chunk2->target && chunk2->target
-               < chunk->target + size))
+       if ((chunk2->target <= ctx.chunk->target
+            && ctx.chunk->target < chunk2->target + chunk2->size)
+           || (ctx.chunk->target <= chunk2->target && chunk2->target
+               < ctx.chunk->target + size))
          {
            if (preference == GRUB_RELOCATOR_PREFERENCE_HIGH)
-             chunk->target = ALIGN_DOWN (chunk2->target, align);
+             ctx.chunk->target = ALIGN_DOWN (chunk2->target, align);
            else
-             chunk->target = ALIGN_UP (chunk2->target + chunk2->size, align);
+             ctx.chunk->target = ALIGN_UP (chunk2->target + chunk2->size,
+                                           align);
            break;
          }
       if (!chunk2)
@@ -1441,23 +1462,23 @@ grub_relocator_alloc_chunk_align (struct
   grub_dprintf ("relocator", "relocators_size=%ld\n",
                (unsigned long) rel->relocators_size);
 
-  if (chunk->src < chunk->target)
+  if (ctx.chunk->src < ctx.chunk->target)
     rel->relocators_size += grub_relocator_backward_size;
-  if (chunk->src > chunk->target)
+  if (ctx.chunk->src > ctx.chunk->target)
     rel->relocators_size += grub_relocator_forward_size;
 
   grub_dprintf ("relocator", "relocators_size=%ld\n",
                (unsigned long) rel->relocators_size);
 
-  chunk->size = size;
-  chunk->next = rel->chunks;
-  rel->chunks = chunk;
+  ctx.chunk->size = size;
+  ctx.chunk->next = rel->chunks;
+  rel->chunks = ctx.chunk;
   grub_dprintf ("relocator", "cur = %p, next = %p\n", rel->chunks,
                rel->chunks->next);
-  chunk->srcv = grub_map_memory (chunk->src, chunk->size);
-  *out = chunk;
+  ctx.chunk->srcv = grub_map_memory (ctx.chunk->src, ctx.chunk->size);
+  *out = ctx.chunk;
 #ifdef DEBUG_RELOCATOR
-  grub_memset (chunk->srcv, 0xfa, chunk->size);
+  grub_memset (ctx.chunk->srcv, 0xfa, ctx.chunk->size);
   grub_mm_check ();
 #endif
   return GRUB_ERR_NONE;

=== modified file 'grub-core/loader/i386/bsd.c'
--- grub-core/loader/i386/bsd.c 2012-07-22 14:30:48 +0000
+++ grub-core/loader/i386/bsd.c 2013-01-13 01:49:43 +0000
@@ -269,81 +269,93 @@ struct grub_e820_mmap
 #define GRUB_E820_NVS        4
 #define GRUB_E820_BADRAM     5
 
-static void
-generate_e820_mmap (grub_size_t *len, grub_size_t *cnt, void *buf)
+/* Context for generate_e820_mmap.  */
+struct generate_e820_mmap_ctx
 {
-  int count = 0;
-  struct grub_e820_mmap *mmap = buf;
+  int count;
+  struct grub_e820_mmap *mmap;
   struct grub_e820_mmap prev, cur;
+};
+
+/* Helper for generate_e820_mmap.  */
+static int
+generate_e820_mmap_iter (grub_uint64_t addr, grub_uint64_t size,
+                        grub_memory_type_t type, void *data)
+{
+  struct generate_e820_mmap_ctx *ctx = data;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
+  ctx->cur.addr = addr;
+  ctx->cur.size = size;
+  switch (type)
     {
-      cur.addr = addr;
-      cur.size = size;
-      switch (type)
-       {
-       case GRUB_MEMORY_AVAILABLE:
-         cur.type = GRUB_E820_RAM;
-         break;
-
-       case GRUB_MEMORY_ACPI:
-         cur.type = GRUB_E820_ACPI;
-         break;
-
-       case GRUB_MEMORY_NVS:
-         cur.type = GRUB_E820_NVS;
-         break;
-
-       default:
-       case GRUB_MEMORY_CODE:
-       case GRUB_MEMORY_RESERVED:
-         cur.type = GRUB_E820_RESERVED;
-         break;
-       }
+    case GRUB_MEMORY_AVAILABLE:
+      ctx->cur.type = GRUB_E820_RAM;
+      break;
+
+    case GRUB_MEMORY_ACPI:
+      ctx->cur.type = GRUB_E820_ACPI;
+      break;
+
+    case GRUB_MEMORY_NVS:
+      ctx->cur.type = GRUB_E820_NVS;
+      break;
+
+    default:
+    case GRUB_MEMORY_CODE:
+    case GRUB_MEMORY_RESERVED:
+      ctx->cur.type = GRUB_E820_RESERVED;
+      break;
+    }
 
-      /* Merge regions if possible. */
-      if (count && cur.type == prev.type && cur.addr == prev.addr + prev.size)
-       {
-         prev.size += cur.size;
-         if (mmap)
-           mmap[-1] = prev;
-       }
-      else
-       {
-         if (mmap)
-           *mmap++ = cur;
-         prev = cur;
-         count++;
-       }
+  /* Merge regions if possible. */
+  if (ctx->count && ctx->cur.type == ctx->prev.type
+      && ctx->cur.addr == ctx->prev.addr + ctx->prev.size)
+    {
+      ctx->prev.size += ctx->cur.size;
+      if (ctx->mmap)
+       ctx->mmap[-1] = ctx->prev;
+    }
+  else
+    {
+      if (ctx->mmap)
+       *ctx->mmap++ = ctx->cur;
+      ctx->prev = ctx->cur;
+      ctx->count++;
+    }
 
-      if (kernel_type == KERNEL_TYPE_OPENBSD && prev.addr < 0x100000
-         && prev.addr + prev.size > 0x100000)
+  if (kernel_type == KERNEL_TYPE_OPENBSD && ctx->prev.addr < 0x100000
+      && ctx->prev.addr + ctx->prev.size > 0x100000)
+    {
+      ctx->cur.addr = 0x100000;
+      ctx->cur.size = ctx->prev.addr + ctx->prev.size - 0x100000;
+      ctx->cur.type = ctx->prev.type;
+      ctx->prev.size = 0x100000 - ctx->prev.addr;
+      if (ctx->mmap)
        {
-         cur.addr = 0x100000;
-         cur.size = prev.addr + prev.size - 0x100000;
-         cur.type = prev.type;
-         prev.size = 0x100000 - prev.addr;
-         if (mmap)
-           {
-             mmap[-1] = prev;
-             mmap[0] = cur;
-             mmap++;
-           }
-         prev = cur;
-         count++;
+         ctx->mmap[-1] = ctx->prev;
+         ctx->mmap[0] = ctx->cur;
+         ctx->mmap++;
        }
-
-      return 0;
+      ctx->prev = ctx->cur;
+      ctx->count++;
     }
 
-  grub_mmap_iterate (hook);
+  return 0;
+}
+
+static void
+generate_e820_mmap (grub_size_t *len, grub_size_t *cnt, void *buf)
+{
+  struct generate_e820_mmap_ctx ctx = {
+    .count = 0,
+    .mmap = buf
+  };
+
+  grub_mmap_iterate (generate_e820_mmap_iter, &ctx);
 
   if (len)
-    *len = count * sizeof (struct grub_e820_mmap);
-  *cnt = count;
+    *len = ctx.count * sizeof (struct grub_e820_mmap);
+  *cnt = ctx.count;
 
   return;
 }

=== modified file 'grub-core/loader/i386/linux.c'
--- grub-core/loader/i386/linux.c       2012-10-05 12:09:19 +0000
+++ grub-core/loader/i386/linux.c       2013-01-13 01:49:43 +0000
@@ -150,23 +150,25 @@ find_efi_mmap_size (void)
 
 #endif
 
+/* Helper for find_mmap_size.  */
+static int
+count_hook (grub_uint64_t addr __attribute__ ((unused)),
+           grub_uint64_t size __attribute__ ((unused)),
+           grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  grub_size_t *count = data;
+
+  (*count)++;
+  return 0;
+}
+
 /* Find the optimal number of pages for the memory map. */
 static grub_size_t
 find_mmap_size (void)
 {
   grub_size_t count = 0, mmap_size;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr __attribute__ ((unused)),
-                            grub_uint64_t size __attribute__ ((unused)),
-                            grub_memory_type_t type __attribute__ ((unused)))
-    {
-      count++;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (count_hook, &count);
 
   mmap_size = count * sizeof (struct grub_e820_mmap);
 
@@ -372,17 +374,96 @@ grub_linux_setup_video (struct linux_ker
   return GRUB_ERR_NONE;
 }
 
+/* Context for grub_linux_boot.  */
+struct grub_linux_boot_ctx
+{
+  grub_addr_t real_mode_target;
+  grub_size_t real_size;
+  struct linux_kernel_params *params;
+  int e820_num;
+};
+
+/* Helper for grub_linux_boot.  */
+static int
+grub_linux_boot_mmap_find (grub_uint64_t addr, grub_uint64_t size,
+                          grub_memory_type_t type, void *data)
+{
+  struct grub_linux_boot_ctx *ctx = data;
+
+  /* We must put real mode code in the traditional space.  */
+  if (type != GRUB_MEMORY_AVAILABLE || addr > 0x90000)
+    return 0;
+
+  if (addr + size < 0x10000)
+    return 0;
+
+  if (addr < 0x10000)
+    {
+      size += addr - 0x10000;
+      addr = 0x10000;
+    }
+
+  if (addr + size > 0x90000)
+    size = 0x90000 - addr;
+
+  if (ctx->real_size + efi_mmap_size > size)
+    return 0;
+
+  grub_dprintf ("linux", "addr = %lx, size = %x, need_size = %x\n",
+               (unsigned long) addr,
+               (unsigned) size,
+               (unsigned) (ctx->real_size + efi_mmap_size));
+  ctx->real_mode_target = ((addr + size) - (ctx->real_size + efi_mmap_size));
+  return 1;
+}
+
+static int
+grub_linux_boot_mmap_fill (grub_uint64_t addr, grub_uint64_t size,
+                          grub_memory_type_t type, void *data)
+{
+  struct grub_linux_boot_ctx *ctx = data;
+
+  grub_uint32_t e820_type;
+  switch (type)
+    {
+    case GRUB_MEMORY_AVAILABLE:
+      e820_type = GRUB_E820_RAM;
+      break;
+
+    case GRUB_MEMORY_ACPI:
+      e820_type = GRUB_E820_ACPI;
+      break;
+
+    case GRUB_MEMORY_NVS:
+      e820_type = GRUB_E820_NVS;
+      break;
+
+    case GRUB_MEMORY_BADRAM:
+      e820_type = GRUB_E820_BADRAM;
+      break;
+
+    default:
+      e820_type = GRUB_E820_RESERVED;
+    }
+  if (grub_e820_add_region (ctx->params->e820_map, &ctx->e820_num,
+                           addr, size, e820_type))
+    return 1;
+
+  return 0;
+}
+
 static grub_err_t
 grub_linux_boot (void)
 {
-  int e820_num;
   grub_err_t err = 0;
   const char *modevar;
   char *tmp;
   struct grub_relocator32_state state;
   void *real_mode_mem;
-  grub_addr_t real_mode_target = 0;
-  grub_size_t real_size, mmap_size;
+  struct grub_linux_boot_ctx ctx = {
+    .real_mode_target = 0
+  };
+  grub_size_t mmap_size;
   grub_size_t cl_offset;
 
 #ifdef GRUB_MACHINE_IEEE1275
@@ -484,7 +565,7 @@ grub_linux_boot (void)
   if (cl_offset < ((grub_size_t) linux_params.setup_sects << 
GRUB_DISK_SECTOR_BITS))
     cl_offset = ALIGN_UP ((grub_size_t) (linux_params.setup_sects
                                         << GRUB_DISK_SECTOR_BITS), 4096);
-  real_size = ALIGN_UP (cl_offset + maximal_cmdline_size, 4096);
+  ctx.real_size = ALIGN_UP (cl_offset + maximal_cmdline_size, 4096);
 
 #ifdef GRUB_MACHINE_EFI
   efi_mmap_size = find_efi_mmap_size ();
@@ -493,118 +574,51 @@ grub_linux_boot (void)
 #endif
 
   grub_dprintf ("linux", "real_size = %x, mmap_size = %x\n",
-               (unsigned) real_size, (unsigned) mmap_size);
-
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      /* We must put real mode code in the traditional space.  */
-      if (type != GRUB_MEMORY_AVAILABLE || addr > 0x90000)
-       return 0;
+               (unsigned) ctx.real_size, (unsigned) mmap_size);
 
-      if (addr + size < 0x10000)
-       return 0;
-
-      if (addr < 0x10000)
-       {
-         size += addr - 0x10000;
-         addr = 0x10000;
-       }
-
-      if (addr + size > 0x90000)
-       size = 0x90000 - addr;
-
-      if (real_size + efi_mmap_size > size)
-       return 0;
-
-      grub_dprintf ("linux", "addr = %lx, size = %x, need_size = %x\n",
-                   (unsigned long) addr,
-                   (unsigned) size,
-                   (unsigned) (real_size + efi_mmap_size));
-      real_mode_target = ((addr + size) - (real_size + efi_mmap_size));
-      return 1;
-    }
 #ifdef GRUB_MACHINE_EFI
-  grub_efi_mmap_iterate (hook, 1);
-  if (! real_mode_target)
-    grub_efi_mmap_iterate (hook, 0);
+  grub_efi_mmap_iterate (grub_linux_boot_mmap_find, &ctx, 1);
+  if (! ctx.real_mode_target)
+    grub_efi_mmap_iterate (grub_linux_boot_mmap_find, &ctx, 0);
 #else
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (grub_linux_boot_mmap_find, &ctx);
 #endif
   grub_dprintf ("linux", "real_mode_target = %lx, real_size = %x, 
efi_mmap_size = %x\n",
-                (unsigned long) real_mode_target,
-               (unsigned) real_size,
+                (unsigned long) ctx.real_mode_target,
+               (unsigned) ctx.real_size,
                (unsigned) efi_mmap_size);
 
-  if (! real_mode_target)
+  if (! ctx.real_mode_target)
     return grub_error (GRUB_ERR_OUT_OF_MEMORY, "cannot allocate real mode 
pages");
 
   {
     grub_relocator_chunk_t ch;
     err = grub_relocator_alloc_chunk_addr (relocator, &ch,
-                                          real_mode_target,
-                                          (real_size + efi_mmap_size));
+                                          ctx.real_mode_target,
+                                          (ctx.real_size + efi_mmap_size));
     if (err)
      return err;
     real_mode_mem = get_virtual_current_address (ch);
   }
-  efi_mmap_buf = (grub_uint8_t *) real_mode_mem + real_size;
+  efi_mmap_buf = (grub_uint8_t *) real_mode_mem + ctx.real_size;
 
   grub_dprintf ("linux", "real_mode_mem = %lx\n",
                 (unsigned long) real_mode_mem);
 
-  struct linux_kernel_params *params;
-
-  params = real_mode_mem;
+  ctx.params = real_mode_mem;
 
-  *params = linux_params;
-  params->cmd_line_ptr = real_mode_target + cl_offset;
-  grub_memcpy ((char *) params + cl_offset, linux_cmdline,
+  *ctx.params = linux_params;
+  ctx.params->cmd_line_ptr = ctx.real_mode_target + cl_offset;
+  grub_memcpy ((char *) ctx.params + cl_offset, linux_cmdline,
               maximal_cmdline_size);
 
   grub_dprintf ("linux", "code32_start = %x\n",
-               (unsigned) params->code32_start);
-
-  auto int NESTED_FUNC_ATTR hook_fill (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook_fill (grub_uint64_t addr, grub_uint64_t size, 
-                                 grub_memory_type_t type)
-    {
-      grub_uint32_t e820_type;
-      switch (type)
-        {
-        case GRUB_MEMORY_AVAILABLE:
-         e820_type = GRUB_E820_RAM;
-         break;
-
-        case GRUB_MEMORY_ACPI:
-         e820_type = GRUB_E820_ACPI;
-         break;
-
-        case GRUB_MEMORY_NVS:
-         e820_type = GRUB_E820_NVS;
-         break;
-
-        case GRUB_MEMORY_BADRAM:
-         e820_type = GRUB_E820_BADRAM;
-         break;
-
-        default:
-          e820_type = GRUB_E820_RESERVED;
-        }
-      if (grub_e820_add_region (params->e820_map, &e820_num,
-                               addr, size, e820_type))
-       return 1;
-
-      return 0;
-    }
+               (unsigned) ctx.params->code32_start);
 
-  e820_num = 0;
-  if (grub_mmap_iterate (hook_fill))
+  ctx.e820_num = 0;
+  if (grub_mmap_iterate (grub_linux_boot_mmap_fill, &ctx))
     return grub_errno;
-  params->mmap_size = e820_num;
+  ctx.params->mmap_size = ctx.e820_num;
 
 #ifdef GRUB_MACHINE_EFI
   {
@@ -617,33 +631,33 @@ grub_linux_boot (void)
       return err;
     
     /* Note that no boot services are available from here.  */
-    efi_mmap_target = real_mode_target 
+    efi_mmap_target = ctx.real_mode_target 
       + ((grub_uint8_t *) efi_mmap_buf - (grub_uint8_t *) real_mode_mem);
     /* Pass EFI parameters.  */
-    if (grub_le_to_cpu16 (params->version) >= 0x0208)
+    if (grub_le_to_cpu16 (ctx.params->version) >= 0x0208)
       {
-       params->v0208.efi_mem_desc_size = efi_desc_size;
-       params->v0208.efi_mem_desc_version = efi_desc_version;
-       params->v0208.efi_mmap = efi_mmap_target;
-       params->v0208.efi_mmap_size = efi_mmap_size;
+       ctx.params->v0208.efi_mem_desc_size = efi_desc_size;
+       ctx.params->v0208.efi_mem_desc_version = efi_desc_version;
+       ctx.params->v0208.efi_mmap = efi_mmap_target;
+       ctx.params->v0208.efi_mmap_size = efi_mmap_size;
 
 #ifdef __x86_64__
-       params->v0208.efi_mmap_hi = (efi_mmap_target >> 32);
+       ctx.params->v0208.efi_mmap_hi = (efi_mmap_target >> 32);
 #endif
       }
-    else if (grub_le_to_cpu16 (params->version) >= 0x0206)
+    else if (grub_le_to_cpu16 (ctx.params->version) >= 0x0206)
       {
-       params->v0206.efi_mem_desc_size = efi_desc_size;
-       params->v0206.efi_mem_desc_version = efi_desc_version;
-       params->v0206.efi_mmap = efi_mmap_target;
-       params->v0206.efi_mmap_size = efi_mmap_size;
+       ctx.params->v0206.efi_mem_desc_size = efi_desc_size;
+       ctx.params->v0206.efi_mem_desc_version = efi_desc_version;
+       ctx.params->v0206.efi_mmap = efi_mmap_target;
+       ctx.params->v0206.efi_mmap_size = efi_mmap_size;
       }
-    else if (grub_le_to_cpu16 (params->version) >= 0x0204)
+    else if (grub_le_to_cpu16 (ctx.params->version) >= 0x0204)
       {
-       params->v0204.efi_mem_desc_size = efi_desc_size;
-       params->v0204.efi_mem_desc_version = efi_desc_version;
-       params->v0204.efi_mmap = efi_mmap_target;
-       params->v0204.efi_mmap_size = efi_mmap_size;
+       ctx.params->v0204.efi_mem_desc_size = efi_desc_size;
+       ctx.params->v0204.efi_mem_desc_version = efi_desc_version;
+       ctx.params->v0204.efi_mmap = efi_mmap_target;
+       ctx.params->v0204.efi_mmap_size = efi_mmap_size;
       }
   }
 #endif
@@ -651,9 +665,9 @@ grub_linux_boot (void)
   /* FIXME.  */
   /*  asm volatile ("lidt %0" : : "m" (idt_desc)); */
   state.ebp = state.edi = state.ebx = 0;
-  state.esi = real_mode_target;
-  state.esp = real_mode_target;
-  state.eip = params->code32_start;
+  state.esi = ctx.real_mode_target;
+  state.esp = ctx.real_mode_target;
+  state.eip = ctx.params->code32_start;
   return grub_relocator32_boot (relocator, state, 0);
 }
 

=== modified file 'grub-core/loader/i386/multiboot_mbi.c'
--- grub-core/loader/i386/multiboot_mbi.c       2012-03-03 19:06:41 +0000
+++ grub-core/loader/i386/multiboot_mbi.c       2013-01-13 01:49:43 +0000
@@ -224,48 +224,50 @@ grub_multiboot_get_mbi_size (void)
   return ret;
 }
 
+/* Helper for grub_fill_multiboot_mmap.  */
+static int
+grub_fill_multiboot_mmap_iter (grub_uint64_t addr, grub_uint64_t size,
+                              grub_memory_type_t type, void *data)
+{
+  struct multiboot_mmap_entry **mmap_entry = data;
+
+  (*mmap_entry)->addr = addr;
+  (*mmap_entry)->len = size;
+  switch (type)
+    {
+    case GRUB_MEMORY_AVAILABLE:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_AVAILABLE;
+      break;
+
+    case GRUB_MEMORY_ACPI:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_ACPI_RECLAIMABLE;
+      break;
+
+    case GRUB_MEMORY_NVS:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_NVS;
+      break;
+
+    case GRUB_MEMORY_BADRAM:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_BADRAM;
+      break;
+      
+    default:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_RESERVED;
+      break;
+    }
+  (*mmap_entry)->size = sizeof (struct multiboot_mmap_entry) - sizeof 
((*mmap_entry)->size);
+  (*mmap_entry)++;
+
+  return 0;
+}
+
 /* Fill previously allocated Multiboot mmap.  */
 static void
 grub_fill_multiboot_mmap (struct multiboot_mmap_entry *first_entry)
 {
   struct multiboot_mmap_entry *mmap_entry = (struct multiboot_mmap_entry *) 
first_entry;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size, 
-                            grub_memory_type_t type)
-    {
-      mmap_entry->addr = addr;
-      mmap_entry->len = size;
-      switch (type)
-       {
-       case GRUB_MEMORY_AVAILABLE:
-         mmap_entry->type = MULTIBOOT_MEMORY_AVAILABLE;
-         break;
-
-       case GRUB_MEMORY_ACPI:
-         mmap_entry->type = MULTIBOOT_MEMORY_ACPI_RECLAIMABLE;
-         break;
-
-       case GRUB_MEMORY_NVS:
-         mmap_entry->type = MULTIBOOT_MEMORY_NVS;
-         break;
-
-       case GRUB_MEMORY_BADRAM:
-         mmap_entry->type = MULTIBOOT_MEMORY_BADRAM;
-         break;
-         
-       default:
-         mmap_entry->type = MULTIBOOT_MEMORY_RESERVED;
-         break;
-       }
-      mmap_entry->size = sizeof (struct multiboot_mmap_entry) - sizeof 
(mmap_entry->size);
-      mmap_entry++;
-
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (grub_fill_multiboot_mmap_iter, &mmap_entry);
 }
 
 #if GRUB_MACHINE_HAS_VBE || GRUB_MACHINE_HAS_VGA_TEXT

=== modified file 'grub-core/loader/multiboot.c'
--- grub-core/loader/multiboot.c        2012-03-03 19:53:54 +0000
+++ grub-core/loader/multiboot.c        2013-01-13 01:49:43 +0000
@@ -63,6 +63,18 @@ static int console_required;
 static grub_dl_t my_mod;
 
 
+/* Helper for grub_get_multiboot_mmap_count.  */
+static int
+count_hook (grub_uint64_t addr __attribute__ ((unused)),
+           grub_uint64_t size __attribute__ ((unused)),
+           grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  grub_size_t *count = data;
+
+  (*count)++;
+  return 0;
+}
+
 /* Return the length of the Multiboot mmap that will be needed to allocate
    our platform's map.  */
 grub_uint32_t
@@ -70,16 +82,7 @@ grub_get_multiboot_mmap_count (void)
 {
   grub_size_t count = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t, grub_uint32_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr __attribute__ ((unused)),
-                            grub_uint64_t size __attribute__ ((unused)),
-                            grub_memory_type_t type __attribute__ ((unused)))
-    {
-      count++;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (count_hook, &count);
 
   return count;
 }

=== modified file 'grub-core/loader/multiboot_mbi2.c'
--- grub-core/loader/multiboot_mbi2.c   2012-03-03 19:06:41 +0000
+++ grub-core/loader/multiboot_mbi2.c   2013-01-13 01:49:43 +0000
@@ -322,53 +322,55 @@ grub_multiboot_get_mbi_size (void)
     + sizeof (struct multiboot_tag_apm) + MULTIBOOT_TAG_ALIGN - 1;
 }
 
+/* Helper for grub_fill_multiboot_mmap.  */
+static int
+grub_fill_multiboot_mmap_iter (grub_uint64_t addr, grub_uint64_t size,
+                              grub_memory_type_t type, void *data)
+{
+  struct multiboot_mmap_entry **mmap_entry = data;
+
+  (*mmap_entry)->addr = addr;
+  (*mmap_entry)->len = size;
+  switch (type)
+    {
+    case GRUB_MEMORY_AVAILABLE:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_AVAILABLE;
+      break;
+
+    case GRUB_MEMORY_ACPI:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_ACPI_RECLAIMABLE;
+      break;
+
+    case GRUB_MEMORY_NVS:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_NVS;
+      break;
+
+    case GRUB_MEMORY_BADRAM:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_BADRAM;
+      break;
+
+    default:
+      (*mmap_entry)->type = MULTIBOOT_MEMORY_RESERVED;
+      break;
+    }
+  (*mmap_entry)++;
+
+  return 0;
+}
+
 /* Fill previously allocated Multiboot mmap.  */
 static void
 grub_fill_multiboot_mmap (struct multiboot_tag_mmap *tag)
 {
   struct multiboot_mmap_entry *mmap_entry = tag->entries;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      mmap_entry->addr = addr;
-      mmap_entry->len = size;
-      switch (type)
-       {
-       case GRUB_MEMORY_AVAILABLE:
-         mmap_entry->type = MULTIBOOT_MEMORY_AVAILABLE;
-         break;
-
-       case GRUB_MEMORY_ACPI:
-         mmap_entry->type = MULTIBOOT_MEMORY_ACPI_RECLAIMABLE;
-         break;
-
-       case GRUB_MEMORY_NVS:
-         mmap_entry->type = MULTIBOOT_MEMORY_NVS;
-         break;
-
-       case GRUB_MEMORY_BADRAM:
-         mmap_entry->type = MULTIBOOT_MEMORY_BADRAM;
-         break;
-
-       default:
-         mmap_entry->type = MULTIBOOT_MEMORY_RESERVED;
-         break;
-       }
-      mmap_entry++;
-
-      return 0;
-    }
-
   tag->type = MULTIBOOT_TAG_TYPE_MMAP;
   tag->size = sizeof (struct multiboot_tag_mmap)
     + sizeof (struct multiboot_mmap_entry) * grub_get_multiboot_mmap_count (); 
   tag->entry_size = sizeof (struct multiboot_mmap_entry);
   tag->entry_version = 0;
 
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (grub_fill_multiboot_mmap_iter, &mmap_entry);
 }
 
 #if defined (GRUB_MACHINE_PCBIOS)

=== modified file 'grub-core/loader/powerpc/ieee1275/linux.c'
--- grub-core/loader/powerpc/ieee1275/linux.c   2012-06-12 23:15:15 +0000
+++ grub-core/loader/powerpc/ieee1275/linux.c   2013-01-13 01:49:43 +0000
@@ -51,49 +51,51 @@ static char *linux_args;
 typedef void (*kernel_entry_t) (void *, unsigned long, int (void *),
                                unsigned long, unsigned long);
 
+/* 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)
+{
+  grub_addr_t *found_addr = data;
+
+  grub_uint64_t end = addr + len;
+  addr = ALIGN_UP (addr, align);
+  target = ALIGN_UP (target, align);
+
+  /* Target above the memory chunk.  */
+  if (type != GRUB_MEMORY_AVAILABLE || target > end)
+    return 0;
+
+  /* Target inside the memory chunk.  */
+  if (target >= addr && target < end && size <= end - target)
+    {
+      if (grub_claimmap (target, size) == GRUB_ERR_NONE)
+       {
+         *found_addr = target;
+         return 1;
+       }
+      grub_print_error ();
+    }
+  /* Target below the memory chunk.  */
+  if (target < addr && addr + size <= end)
+    {
+      if (grub_claimmap (addr, size) == GRUB_ERR_NONE)
+       {
+         *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)
 {
   grub_addr_t found_addr = (grub_addr_t) -1;
 
-  auto int NESTED_FUNC_ATTR alloc_mem (grub_uint64_t addr, grub_uint64_t len,
-                                      grub_memory_type_t type);
-  int NESTED_FUNC_ATTR alloc_mem (grub_uint64_t addr, grub_uint64_t len,
-                                 grub_memory_type_t type)
-  {
-    grub_uint64_t end = addr + len;
-    addr = ALIGN_UP (addr, align);
-    target = ALIGN_UP (target, align);
-
-    /* Target above the memory chunk.  */
-    if (type != GRUB_MEMORY_AVAILABLE || target > end)
-      return 0;
-
-    /* Target inside the memory chunk.  */
-    if (target >= addr && target < end && size <= end - target)
-      {
-       if (grub_claimmap (target, size) == GRUB_ERR_NONE)
-         {
-           found_addr = target;
-           return 1;
-         }
-       grub_print_error ();
-      }
-    /* Target below the memory chunk.  */
-    if (target < addr && addr + size <= end)
-      {
-       if (grub_claimmap (addr, size) == GRUB_ERR_NONE)
-         {
-           found_addr = addr;
-           return 1;
-         }
-       grub_print_error ();
-      }
-    return 0;
-  }
-
-  grub_machine_mmap_iterate (alloc_mem);
+  grub_machine_mmap_iterate (alloc_mem, &found_addr);
 
   return found_addr;
 }

=== modified file 'grub-core/loader/sparc64/ieee1275/linux.c'
--- grub-core/loader/sparc64/ieee1275/linux.c   2012-03-05 00:17:55 +0000
+++ grub-core/loader/sparc64/ieee1275/linux.c   2013-01-13 01:49:43 +0000
@@ -180,63 +180,64 @@ grub_linux_unload (void)
 
 #define FOUR_MB        (4 * 1024 * 1024)
 
-static grub_addr_t
-alloc_phys (grub_addr_t size)
+/* Helper for alloc_phys.  */
+static int
+alloc_phys_choose (grub_uint64_t addr, grub_uint64_t len,
+                  grub_memory_type_t type, void *data)
 {
-  grub_addr_t ret = (grub_addr_t) -1;
+  grub_addr_t *ret = data;
+  grub_addr_t end = addr + len;
 
-  auto int NESTED_FUNC_ATTR choose (grub_uint64_t addr, grub_uint64_t len,
-                                   grub_memory_type_t type);
-  int NESTED_FUNC_ATTR choose (grub_uint64_t addr, grub_uint64_t len,
-                              grub_memory_type_t type)
-  {
-    grub_addr_t end = addr + len;
+  if (type != 1)
+    return 0;
 
-    if (type != 1)
-      return 0;
+  addr = ALIGN_UP (addr, FOUR_MB);
+  if (addr + size >= end)
+    return 0;
 
-    addr = ALIGN_UP (addr, FOUR_MB);
-    if (addr + size >= end)
-      return 0;
+  if (addr >= grub_phys_start && addr < grub_phys_end)
+    {
+      addr = ALIGN_UP (grub_phys_end, FOUR_MB);
+      if (addr + size >= end)
+       return 0;
+    }
+  if ((addr + size) >= grub_phys_start
+      && (addr + size) < grub_phys_end)
+    {
+      addr = ALIGN_UP (grub_phys_end, FOUR_MB);
+      if (addr + size >= end)
+       return 0;
+    }
 
-    if (addr >= grub_phys_start && addr < grub_phys_end)
-      {
-       addr = ALIGN_UP (grub_phys_end, FOUR_MB);
-       if (addr + size >= end)
-         return 0;
-      }
-    if ((addr + size) >= grub_phys_start
-       && (addr + size) < grub_phys_end)
-      {
-       addr = ALIGN_UP (grub_phys_end, FOUR_MB);
-       if (addr + size >= end)
-         return 0;
-      }
+  if (loaded)
+    {
+      grub_addr_t linux_end = ALIGN_UP (linux_paddr + linux_size, FOUR_MB);
 
-    if (loaded)
-      {
-       grub_addr_t linux_end = ALIGN_UP (linux_paddr + linux_size, FOUR_MB);
-
-       if (addr >= linux_paddr && addr < linux_end)
-         {
-           addr = linux_end;
-           if (addr + size >= end)
-             return 0;
-         }
-       if ((addr + size) >= linux_paddr
-           && (addr + size) < linux_end)
-         {
-           addr = linux_end;
-           if (addr + size >= end)
-             return 0;
-         }
-      }
-
-    ret = addr;
-    return 1;
-  }
+      if (addr >= linux_paddr && addr < linux_end)
+       {
+         addr = linux_end;
+         if (addr + size >= end)
+           return 0;
+       }
+      if ((addr + size) >= linux_paddr
+         && (addr + size) < linux_end)
+       {
+         addr = linux_end;
+         if (addr + size >= end)
+           return 0;
+       }
+    }
+
+  *ret = addr;
+  return 1;
+}
+
+static grub_addr_t
+alloc_phys (grub_addr_t size)
+{
+  grub_addr_t ret = (grub_addr_t) -1;
 
-  grub_machine_mmap_iterate (choose);
+  grub_machine_mmap_iterate (alloc_phys_choose, &ret);
 
   return ret;
 }
@@ -454,21 +455,23 @@ grub_cmd_initrd (grub_command_t cmd __at
   return grub_errno;
 }
 
-static void
-determine_phys_base (void)
+/* Helper for determine_phys_base.  */
+static int
+get_physbase (grub_uint64_t addr, grub_uint64_t len __attribute__ ((unused)),
+             grub_uint32_t type, void *data __attribute__ ((unused)))
 {
-  auto int NESTED_FUNC_ATTR get_physbase (grub_uint64_t addr, grub_uint64_t 
len __attribute__((unused)), grub_uint32_t type);
-  int NESTED_FUNC_ATTR get_physbase (grub_uint64_t addr, grub_uint64_t len 
__attribute__((unused)), grub_uint32_t type)
-  {
-    if (type != 1)
-      return 0;
-    if (addr < phys_base)
-      phys_base = addr;
+  if (type != 1)
     return 0;
-  }
+  if (addr < phys_base)
+    phys_base = addr;
+  return 0;
+}
 
+static void
+determine_phys_base (void)
+{
   phys_base = ~(grub_uint64_t) 0;
-  grub_machine_mmap_iterate (get_physbase);
+  grub_machine_mmap_iterate (get_physbase, NULL);
 }
 
 static void

=== modified file 'grub-core/mmap/efi/mmap.c'
--- grub-core/mmap/efi/mmap.c   2012-03-03 19:12:03 +0000
+++ grub-core/mmap/efi/mmap.c   2013-01-13 01:49:43 +0000
@@ -29,7 +29,8 @@
   ((grub_efi_memory_descriptor_t *) ((char *) (desc) + (size)))
 
 grub_err_t
-grub_efi_mmap_iterate (grub_memory_hook_t hook, int avoid_efi_boot_services)
+grub_efi_mmap_iterate (grub_memory_hook_t hook, void *hook_data,
+                      int avoid_efi_boot_services)
 {
   grub_efi_uintn_t mmap_size = 0;
   grub_efi_memory_descriptor_t *map_buf = 0;
@@ -69,17 +70,17 @@ grub_efi_mmap_iterate (grub_memory_hook_
          if (!avoid_efi_boot_services)
            {
              hook (desc->physical_start, desc->num_pages * 4096,
-                   GRUB_MEMORY_AVAILABLE);
+                   GRUB_MEMORY_AVAILABLE, hook_data);
              break;
            }
        case GRUB_EFI_RUNTIME_SERVICES_CODE:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_CODE);
+               GRUB_MEMORY_CODE, hook_data);
          break;
 
        case GRUB_EFI_UNUSABLE_MEMORY:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_BADRAM);
+               GRUB_MEMORY_BADRAM, hook_data);
          break;
 
        default:
@@ -90,7 +91,7 @@ grub_efi_mmap_iterate (grub_memory_hook_
          if (!avoid_efi_boot_services)
            {
              hook (desc->physical_start, desc->num_pages * 4096,
-                   GRUB_MEMORY_AVAILABLE);
+                   GRUB_MEMORY_AVAILABLE, hook_data);
              break;
            }
        case GRUB_EFI_RESERVED_MEMORY_TYPE:
@@ -99,24 +100,24 @@ grub_efi_mmap_iterate (grub_memory_hook_
        case GRUB_EFI_MEMORY_MAPPED_IO_PORT_SPACE:
        case GRUB_EFI_PAL_CODE:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_RESERVED);
+               GRUB_MEMORY_RESERVED, hook_data);
          break;
 
        case GRUB_EFI_LOADER_CODE:
        case GRUB_EFI_LOADER_DATA:
        case GRUB_EFI_CONVENTIONAL_MEMORY:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_AVAILABLE);
+               GRUB_MEMORY_AVAILABLE, hook_data);
          break;
 
        case GRUB_EFI_ACPI_RECLAIM_MEMORY:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_ACPI);
+               GRUB_MEMORY_ACPI, hook_data);
          break;
 
        case GRUB_EFI_ACPI_MEMORY_NVS:
          hook (desc->physical_start, desc->num_pages * 4096,
-               GRUB_MEMORY_NVS);
+               GRUB_MEMORY_NVS, hook_data);
          break;
        }
     }
@@ -125,9 +126,9 @@ grub_efi_mmap_iterate (grub_memory_hook_
 }
 
 grub_err_t
-grub_machine_mmap_iterate (grub_memory_hook_t hook)
+grub_machine_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
 {
-  return grub_efi_mmap_iterate (hook, 0);
+  return grub_efi_mmap_iterate (hook, hook_data, 0);
 }
 
 static inline grub_efi_memory_type_t

=== modified file 'grub-core/mmap/i386/mmap.c'
--- grub-core/mmap/i386/mmap.c  2011-12-15 18:59:49 +0000
+++ grub-core/mmap/i386/mmap.c  2013-01-13 01:49:43 +0000
@@ -27,34 +27,48 @@
 
 #ifndef GRUB_MMAP_REGISTER_BY_FIRMWARE
 
+/* Context for grub_mmap_malign_and_register.  */
+struct grub_mmap_malign_and_register_ctx
+{
+  grub_uint64_t align, size, highestlow;
+};
+
+/* Helper for grub_mmap_malign_and_register.  */
+static int
+find_hook (grub_uint64_t start, grub_uint64_t rangesize,
+          grub_memory_type_t memtype, void *data)
+{
+  struct grub_mmap_malign_and_register_ctx *ctx = data;
+  grub_uint64_t end = start + rangesize;
+
+  if (memtype != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (end > 0x100000)
+    end = 0x100000;
+  if (end > start + ctx->size
+      && ctx->highestlow < ((end - ctx->size)
+                           - ((end - ctx->size) & (ctx->align - 1))))
+    ctx->highestlow = (end - ctx->size)
+                     - ((end - ctx->size) & (ctx->align - 1));
+  return 0;
+}
+
 void *
 grub_mmap_malign_and_register (grub_uint64_t align, grub_uint64_t size,
                               int *handle, int type, int flags)
 {
-  grub_uint64_t highestlow = 0;
-
-  auto int NESTED_FUNC_ATTR find_hook (grub_uint64_t, grub_uint64_t,
-                                      grub_memory_type_t);
-  int NESTED_FUNC_ATTR find_hook (grub_uint64_t start, grub_uint64_t rangesize,
-                                 grub_memory_type_t memtype)
-  {
-    grub_uint64_t end = start + rangesize;
-    if (memtype != GRUB_MEMORY_AVAILABLE)
-      return 0;
-    if (end > 0x100000)
-      end = 0x100000;
-    if (end > start + size
-       && highestlow < ((end - size) - ((end - size) & (align - 1))))
-      highestlow = (end - size)  - ((end - size) & (align - 1));
-    return 0;
-  }
+  struct grub_mmap_malign_and_register_ctx ctx = {
+    .align = align,
+    .size = size,
+    .highestlow = 0
+  };
 
   void *ret;
   if (flags & GRUB_MMAP_MALLOC_LOW)
     {
       /* FIXME: use low-memory mm allocation once it's available. */
-      grub_mmap_iterate (find_hook);
-      ret = (void *) (grub_addr_t) highestlow;
+      grub_mmap_iterate (find_hook, &ctx);
+      ret = (void *) (grub_addr_t) ctx.highestlow;
     }
   else
     ret = grub_memalign (align, size);

=== modified file 'grub-core/mmap/i386/pc/mmap.c'
--- grub-core/mmap/i386/pc/mmap.c       2012-06-03 16:00:37 +0000
+++ grub-core/mmap/i386/pc/mmap.c       2013-01-13 01:49:43 +0000
@@ -50,22 +50,23 @@ struct grub_e820_mmap_entry
 } __attribute__((packed));
 
 
+/* Helper for preboot.  */
+static int fill_hook (grub_uint64_t addr, grub_uint64_t size,
+                     grub_memory_type_t type, void *data)
+{
+  struct grub_e820_mmap_entry **hookmmapcur = data;
+  grub_dprintf ("mmap", "mmap chunk %llx-%llx:%x\n", addr, addr + size, type);
+  (*hookmmapcur)->addr = addr;
+  (*hookmmapcur)->len = size;
+  (*hookmmapcur)->type = type;
+  (*hookmmapcur)++;
+  return 0;
+}
+
 static grub_err_t
 preboot (int noreturn __attribute__ ((unused)))
 {
   struct grub_e820_mmap_entry *hookmmap, *hookmmapcur;
-  auto int NESTED_FUNC_ATTR fill_hook (grub_uint64_t, grub_uint64_t,
-                                      grub_uint32_t);
-  int NESTED_FUNC_ATTR fill_hook (grub_uint64_t addr, grub_uint64_t size,
-                                 grub_memory_type_t type)
-  {
-    grub_dprintf ("mmap", "mmap chunk %llx-%llx:%x\n", addr, addr + size, 
type);
-    hookmmapcur->addr = addr;
-    hookmmapcur->len = size;
-    hookmmapcur->type = type;
-    hookmmapcur++;
-    return 0;
-  }
 
   if (! hooktarget)
     return grub_error (GRUB_ERR_OUT_OF_MEMORY,
@@ -77,7 +78,7 @@ preboot (int noreturn __attribute__ ((un
     ((grub_uint8_t *) hooktarget + (&grub_machine_mmaphook_end
                                    - &grub_machine_mmaphook_start));
 
-  grub_mmap_iterate (fill_hook);
+  grub_mmap_iterate (fill_hook, &hookmmapcur);
   grub_machine_mmaphook_mmap_num = hookmmapcur - hookmmap;
 
   grub_machine_mmaphook_kblow = grub_mmap_get_lower () >> 10;
@@ -123,6 +124,17 @@ preboot_rest (void)
   return GRUB_ERR_NONE;
 }
 
+/* Helper for malloc_hook.  */
+static int
+count_hook (grub_uint64_t addr __attribute__ ((unused)),
+           grub_uint64_t size __attribute__ ((unused)),
+           grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  int *regcount = data;
+  (*regcount)++;
+  return 0;
+}
+
 static grub_err_t
 malloc_hook (void)
 {
@@ -131,22 +143,13 @@ malloc_hook (void)
   static int slots_available = 0;
   int hooksize;
   int regcount = 0;
-  auto int NESTED_FUNC_ATTR count_hook (grub_uint64_t, grub_uint64_t,
-                                       grub_uint32_t);
-  int NESTED_FUNC_ATTR count_hook (grub_uint64_t addr __attribute__ ((unused)),
-                                  grub_uint64_t size __attribute__ ((unused)),
-                                  grub_memory_type_t type __attribute__ 
((unused)))
-  {
-    regcount++;
-    return 0;
-  }
 
   if (reentry)
     return GRUB_ERR_NONE;
 
   grub_dprintf ("mmap", "registering\n");
 
-  grub_mmap_iterate (count_hook);
+  grub_mmap_iterate (count_hook, &regcount);
 
   /* Mapping hook itself may introduce up to 2 additional regions. */
   regcount += 2;

=== modified file 'grub-core/mmap/i386/uppermem.c'
--- grub-core/mmap/i386/uppermem.c      2010-09-04 15:10:10 +0000
+++ grub-core/mmap/i386/uppermem.c      2013-01-13 01:49:43 +0000
@@ -22,68 +22,73 @@
 #include <grub/mm.h>
 #include <grub/misc.h>
 
+/* Helper for grub_mmap_get_lower.  */
+static int
+lower_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+           void *data)
+{
+  grub_uint64_t *lower = data;
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (addr == 0)
+    *lower = size;
+  return 0;
+}
+
 grub_uint64_t
 grub_mmap_get_lower (void)
 {
   grub_uint64_t lower = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      if (addr == 0)
-       lower = size;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (lower_hook, &lower);
   if (lower > 0x100000)
     lower =  0x100000;
   return lower;
 }
 
+/* Helper for grub_mmap_get_upper.  */
+static int
+upper_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+           void *data)
+{
+  grub_uint64_t *upper = data;
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (addr <= 0x100000 && addr + size > 0x100000)
+    *upper = addr + size - 0x100000;
+  return 0;
+}
+
 grub_uint64_t
 grub_mmap_get_upper (void)
 {
   grub_uint64_t upper = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      if (addr <= 0x100000 && addr + size > 0x100000)
-       upper = addr + size - 0x100000;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (upper_hook, &upper);
   return upper;
 }
 
+/* Helper for grub_mmap_get_post64.  */
+static int
+post64_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+            void *data)
+{
+  grub_uint64_t *post64 = data;
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (addr <= 0x4000000 && addr + size > 0x4000000)
+    *post64 = addr + size - 0x4000000;
+  return 0;
+}
+
 /* Count the continuous bytes after 64 MiB. */
 grub_uint64_t
 grub_mmap_get_post64 (void)
 {
   grub_uint64_t post64 = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      if (addr <= 0x4000000 && addr + size > 0x4000000)
-       post64 = addr + size - 0x4000000;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (post64_hook, &post64);
   return post64;
 }

=== modified file 'grub-core/mmap/mips/uppermem.c'
--- grub-core/mmap/mips/uppermem.c      2011-05-13 14:36:05 +0000
+++ grub-core/mmap/mips/uppermem.c      2013-01-13 01:49:43 +0000
@@ -22,47 +22,51 @@
 #include <grub/misc.h>
 #include <grub/cpu/memory.h>
 
+/* Helper for grub_mmap_get_lower.  */
+static int
+lower_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+           void *data)
+{
+  grub_uint64_t *lower = data;
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (addr == 0)
+    *lower = size;
+  return 0;
+}
+
 grub_uint64_t
 grub_mmap_get_lower (void)
 {
   grub_uint64_t lower = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      if (addr == 0)
-       lower = size;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (lower_hook, &lower);
   if (lower > GRUB_ARCH_LOWMEMMAXSIZE)
     lower = GRUB_ARCH_LOWMEMMAXSIZE;
   return lower;
 }
 
+/* Helper for grub_mmap_get_upper.  */
+static int
+upper_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+           void *data)
+{
+  grub_uint64_t *upper = data;
+
+  if (type != GRUB_MEMORY_AVAILABLE)
+    return 0;
+  if (addr <= GRUB_ARCH_HIGHMEMPSTART && addr + size
+      > GRUB_ARCH_HIGHMEMPSTART)
+    *upper = addr + size - GRUB_ARCH_HIGHMEMPSTART;
+  return 0;
+}
+
 grub_uint64_t
 grub_mmap_get_upper (void)
 {
   grub_uint64_t upper = 0;
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                 grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr, grub_uint64_t size,
-                            grub_memory_type_t type)
-    {
-      if (type != GRUB_MEMORY_AVAILABLE)
-       return 0;
-      if (addr <= GRUB_ARCH_HIGHMEMPSTART && addr + size
-         > GRUB_ARCH_HIGHMEMPSTART)
-       upper = addr + size - GRUB_ARCH_HIGHMEMPSTART;
-      return 0;
-    }
-
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (upper_hook, &upper);
   return upper;
 }

=== modified file 'grub-core/mmap/mmap.c'
--- grub-core/mmap/mmap.c       2012-02-12 14:25:25 +0000
+++ grub-core/mmap/mmap.c       2013-01-14 11:21:39 +0000
@@ -35,43 +35,90 @@ static int curhandle = 1;
 
 #endif
 
-grub_err_t
-grub_mmap_iterate (grub_memory_hook_t hook)
+/* If same page is used by multiple types it's resolved
+   according to priority:
+   1 - free memory
+   2 - memory usable by firmware-aware code
+   3 - unusable memory
+   4 - a range deliberately empty
+*/
+static const int priority[] =
+{
+  [GRUB_MEMORY_AVAILABLE] = 1,
+  [GRUB_MEMORY_RESERVED] = 3,
+  [GRUB_MEMORY_ACPI] = 2,
+  [GRUB_MEMORY_CODE] = 3,
+  [GRUB_MEMORY_NVS] = 3,
+  [GRUB_MEMORY_HOLE] = 4,
+};
+
+/* Scanline events. */
+struct grub_mmap_scan
+{
+  /* At which memory address. */
+  grub_uint64_t pos;
+  /* 0 = region starts, 1 = region ends. */
+  int type;
+  /* Which type of memory region? */
+  int memtype;
+};
+
+/* Context for grub_mmap_iterate.  */
+struct grub_mmap_iterate_ctx
+{
+  struct grub_mmap_scan *scanline_events;
+  int i;
+};
+
+/* Helper for grub_mmap_iterate.  */
+static int
+count_hook (grub_uint64_t addr __attribute__ ((unused)),
+           grub_uint64_t size __attribute__ ((unused)),
+           grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  int *mmap_num = data;
+
+  (*mmap_num)++;
+  return 0;
+}
+
+/* Helper for grub_mmap_iterate.  */
+static int
+fill_hook (grub_uint64_t addr, grub_uint64_t size, grub_memory_type_t type,
+          void *data)
 {
+  struct grub_mmap_iterate_ctx *ctx = data;
+
+  ctx->scanline_events[ctx->i].pos = addr;
+  ctx->scanline_events[ctx->i].type = 0;
+  if (type < ARRAY_SIZE (priority) && priority[type])
+    ctx->scanline_events[ctx->i].memtype = type;
+  else
+    {
+      grub_dprintf ("mmap", "Unknown memory type %d. Assuming unusable\n",
+                   type);
+      ctx->scanline_events[ctx->i].memtype = GRUB_MEMORY_RESERVED;
+    }
+  ctx->i++;
+
+  ctx->scanline_events[ctx->i].pos = addr + size;
+  ctx->scanline_events[ctx->i].type = 1;
+  ctx->scanline_events[ctx->i].memtype =
+    ctx->scanline_events[ctx->i - 1].memtype;
+  ctx->i++;
+
+  return 0;
+}
 
+grub_err_t
+grub_mmap_iterate (grub_memory_hook_t hook, void *hook_data)
+{
   /* This function resolves overlapping regions and sorts the memory map.
      It uses scanline (sweeping) algorithm.
   */
-  /* If same page is used by multiple types it's resolved
-     according to priority:
-     1 - free memory
-     2 - memory usable by firmware-aware code
-     3 - unusable memory
-     4 - a range deliberately empty
-  */
-  int priority[] =
-    {
-      [GRUB_MEMORY_AVAILABLE] = 1,
-      [GRUB_MEMORY_RESERVED] = 3,
-      [GRUB_MEMORY_ACPI] = 2,
-      [GRUB_MEMORY_CODE] = 3,
-      [GRUB_MEMORY_NVS] = 3,
-      [GRUB_MEMORY_HOLE] = 4,
-    };
-
+  struct grub_mmap_iterate_ctx ctx;
   int i, done;
 
-  /* Scanline events. */
-  struct grub_mmap_scan
-  {
-    /* At which memory address. */
-    grub_uint64_t pos;
-    /* 0 = region starts, 1 = region ends. */
-    int type;
-    /* Which type of memory region? */
-    int memtype;
-  };
-  struct grub_mmap_scan *scanline_events;
   struct grub_mmap_scan t;
 
   /* Previous scanline event. */
@@ -88,42 +135,6 @@ grub_mmap_iterate (grub_memory_hook_t ho
   struct grub_mmap_region *cur;
 #endif
 
-  auto int NESTED_FUNC_ATTR count_hook (grub_uint64_t, grub_uint64_t,
-                                       grub_uint32_t);
-  int NESTED_FUNC_ATTR count_hook (grub_uint64_t addr __attribute__ ((unused)),
-                                  grub_uint64_t size __attribute__ ((unused)),
-                                  grub_memory_type_t type __attribute__ 
((unused)))
-  {
-    mmap_num++;
-    return 0;
-  }
-
-  auto int NESTED_FUNC_ATTR fill_hook (grub_uint64_t, grub_uint64_t,
-                                       grub_uint32_t);
-  int NESTED_FUNC_ATTR fill_hook (grub_uint64_t addr,
-                                 grub_uint64_t size,
-                                 grub_memory_type_t type)
-  {
-    scanline_events[i].pos = addr;
-    scanline_events[i].type = 0;
-    if (type < ARRAY_SIZE (priority) && priority[type])
-      scanline_events[i].memtype = type;
-    else
-      {
-       grub_dprintf ("mmap", "Unknown memory type %d. Assuming unusable\n",
-                     type);
-       scanline_events[i].memtype = GRUB_MEMORY_RESERVED;
-      }
-    i++;
-
-    scanline_events[i].pos = addr + size;
-    scanline_events[i].type = 1;
-    scanline_events[i].memtype = scanline_events[i - 1].memtype;
-    i++;
-
-    return 0;
-  }
-
   mmap_num = 0;
 
 #ifndef GRUB_MMAP_REGISTER_BY_FIRMWARE
@@ -131,37 +142,38 @@ grub_mmap_iterate (grub_memory_hook_t ho
     mmap_num++;
 #endif
 
-  grub_machine_mmap_iterate (count_hook);
+  grub_machine_mmap_iterate (count_hook, &mmap_num);
 
   /* Initialize variables. */
   grub_memset (present, 0, sizeof (present));
-  scanline_events = (struct grub_mmap_scan *)
+  ctx.scanline_events = (struct grub_mmap_scan *)
     grub_malloc (sizeof (struct grub_mmap_scan) * 2 * mmap_num);
 
-  if (! scanline_events)
+  if (! ctx.scanline_events)
     return grub_errno;
 
-  i = 0;
+  ctx.i = 0;
 #ifndef GRUB_MMAP_REGISTER_BY_FIRMWARE
   /* Register scanline events. */
   for (cur = grub_mmap_overlays; cur; cur = cur->next)
     {
-      scanline_events[i].pos = cur->start;
-      scanline_events[i].type = 0;
+      ctx.scanline_events[ctx.i].pos = cur->start;
+      ctx.scanline_events[ctx.i].type = 0;
       if (cur->type < ARRAY_SIZE (priority) && priority[cur->type])
-       scanline_events[i].memtype = cur->type;
+       ctx.scanline_events[ctx.i].memtype = cur->type;
       else
-       scanline_events[i].memtype = GRUB_MEMORY_RESERVED;
-      i++;
+       ctx.scanline_events[ctx.i].memtype = GRUB_MEMORY_RESERVED;
+      ctx.i++;
 
-      scanline_events[i].pos = cur->end;
-      scanline_events[i].type = 1;
-      scanline_events[i].memtype = scanline_events[i - 1].memtype;
-      i++;
+      ctx.scanline_events[ctx.i].pos = cur->end;
+      ctx.scanline_events[ctx.i].type = 1;
+      ctx.scanline_events[ctx.i].memtype =
+       ctx.scanline_events[ctx.i - 1].memtype;
+      ctx.i++;
     }
 #endif /* ! GRUB_MMAP_REGISTER_BY_FIRMWARE */
 
-  grub_machine_mmap_iterate (fill_hook);
+  grub_machine_mmap_iterate (fill_hook, &ctx);
 
   /* Primitive bubble sort. It has complexity O(n^2) but since we're
      unlikely to have more than 100 chunks it's probably one of the
@@ -171,28 +183,28 @@ grub_mmap_iterate (grub_memory_hook_t ho
     {
       done = 0;
       for (i = 0; i < 2 * mmap_num - 1; i++)
-       if (scanline_events[i + 1].pos < scanline_events[i].pos
-           || (scanline_events[i + 1].pos == scanline_events[i].pos
-               && scanline_events[i + 1].type == 0
-               && scanline_events[i].type == 1))
+       if (ctx.scanline_events[i + 1].pos < ctx.scanline_events[i].pos
+           || (ctx.scanline_events[i + 1].pos == ctx.scanline_events[i].pos
+               && ctx.scanline_events[i + 1].type == 0
+               && ctx.scanline_events[i].type == 1))
          {
-           t = scanline_events[i + 1];
-           scanline_events[i + 1] = scanline_events[i];
-           scanline_events[i] = t;
+           t = ctx.scanline_events[i + 1];
+           ctx.scanline_events[i + 1] = ctx.scanline_events[i];
+           ctx.scanline_events[i] = t;
            done = 1;
          }
     }
 
-  lastaddr = scanline_events[0].pos;
-  lasttype = scanline_events[0].memtype;
+  lastaddr = ctx.scanline_events[0].pos;
+  lasttype = ctx.scanline_events[0].memtype;
   for (i = 0; i < 2 * mmap_num; i++)
     {
       unsigned k;
       /* Process event. */
-      if (scanline_events[i].type)
-       present[scanline_events[i].memtype]--;
+      if (ctx.scanline_events[i].type)
+       present[ctx.scanline_events[i].memtype]--;
       else
-       present[scanline_events[i].memtype]++;
+       present[ctx.scanline_events[i].memtype]++;
 
       /* Determine current region type. */
       curtype = -1;
@@ -202,12 +214,13 @@ grub_mmap_iterate (grub_memory_hook_t ho
 
       /* Announce region to the hook if necessary. */
       if ((curtype == -1 || curtype != lasttype)
-         && lastaddr != scanline_events[i].pos
+         && lastaddr != ctx.scanline_events[i].pos
          && lasttype != -1
          && lasttype != GRUB_MEMORY_HOLE
-         && hook (lastaddr, scanline_events[i].pos - lastaddr, lasttype))
+         && hook (lastaddr, ctx.scanline_events[i].pos - lastaddr, lasttype,
+                  hook_data))
        {
-         grub_free (scanline_events);
+         grub_free (ctx.scanline_events);
          return GRUB_ERR_NONE;
        }
 
@@ -215,11 +228,11 @@ grub_mmap_iterate (grub_memory_hook_t ho
       if (curtype == -1 || curtype != lasttype)
        {
          lasttype = curtype;
-         lastaddr = scanline_events[i].pos;
+         lastaddr = ctx.scanline_events[i].pos;
        }
     }
 
-  grub_free (scanline_events);
+  grub_free (ctx.scanline_events);
   return GRUB_ERR_NONE;
 }
 
@@ -280,19 +293,23 @@ grub_mmap_unregister (int handle)
 
 #define CHUNK_SIZE     0x400
 
+struct badram_entry {
+  grub_uint64_t addr, mask;
+};
+
 static inline grub_uint64_t
-fill_mask (grub_uint64_t addr, grub_uint64_t mask, grub_uint64_t iterator)
+fill_mask (struct badram_entry *entry, grub_uint64_t iterator)
 {
   int i, j;
-  grub_uint64_t ret = (addr & mask);
+  grub_uint64_t ret = (entry->addr & entry->mask);
 
   /* Find first fixed bit. */
   for (i = 0; i < 64; i++)
-    if ((mask & (1ULL << i)) != 0)
+    if ((entry->mask & (1ULL << i)) != 0)
       break;
   j = 0;
   for (; i < 64; i++)
-    if ((mask & (1ULL << i)) == 0)
+    if ((entry->mask & (1ULL << i)) == 0)
       {
        if ((iterator & (1ULL << j)) != 0)
          ret |= 1ULL << i;
@@ -301,64 +318,64 @@ fill_mask (grub_uint64_t addr, grub_uint
   return ret;
 }
 
+/* Helper for grub_cmd_badram.  */
+static int
+badram_iter (grub_uint64_t addr, grub_uint64_t size,
+            grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  struct badram_entry *entry = data;
+  grub_uint64_t iterator, low, high, cur;
+  int tail, var;
+  int i;
+  grub_dprintf ("badram", "hook %llx+%llx\n", (unsigned long long) addr,
+               (unsigned long long) size);
+
+  /* How many trailing zeros? */
+  for (tail = 0; ! (entry->mask & (1ULL << tail)); tail++);
+
+  /* How many zeros in mask? */
+  var = 0;
+  for (i = 0; i < 64; i++)
+    if (! (entry->mask & (1ULL << i)))
+      var++;
+
+  if (fill_mask (entry, 0) >= addr)
+    iterator = 0;
+  else
+    {
+      low = 0;
+      high = ~0ULL;
+      /* Find starting value. Keep low and high such that
+        fill_mask (low) < addr and fill_mask (high) >= addr;
+      */
+      while (high - low > 1)
+       {
+         cur = (low + high) / 2;
+         if (fill_mask (entry, cur) >= addr)
+           high = cur;
+         else
+           low = cur;
+       }
+      iterator = high;
+    }
+
+  for (; iterator < (1ULL << (var - tail))
+        && (cur = fill_mask (entry, iterator)) < addr + size;
+       iterator++)
+    {
+      grub_dprintf ("badram", "%llx (size %llx) is a badram range\n",
+                   (unsigned long long) cur, (1ULL << tail));
+      grub_mmap_register (cur, (1ULL << tail), GRUB_MEMORY_HOLE);
+    }
+  return 0;
+}
+
 static grub_err_t
 grub_cmd_badram (grub_command_t cmd __attribute__ ((unused)),
                 int argc, char **args)
 {
   char * str;
-  grub_uint64_t badaddr, badmask;
-
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr,
-                            grub_uint64_t size,
-                            grub_memory_type_t type __attribute__ ((unused)))
-  {
-    grub_uint64_t iterator, low, high, cur;
-    int tail, var;
-    int i;
-    grub_dprintf ("badram", "hook %llx+%llx\n", (unsigned long long) addr,
-                 (unsigned long long) size);
-
-    /* How many trailing zeros? */
-    for (tail = 0; ! (badmask & (1ULL << tail)); tail++);
-
-    /* How many zeros in mask? */
-    var = 0;
-    for (i = 0; i < 64; i++)
-      if (! (badmask & (1ULL << i)))
-       var++;
-
-    if (fill_mask (badaddr, badmask, 0) >= addr)
-      iterator = 0;
-    else
-      {
-       low = 0;
-       high = ~0ULL;
-       /* Find starting value. Keep low and high such that
-          fill_mask (low) < addr and fill_mask (high) >= addr;
-       */
-       while (high - low > 1)
-         {
-           cur = (low + high) / 2;
-           if (fill_mask (badaddr, badmask, cur) >= addr)
-             high = cur;
-           else
-             low = cur;
-         }
-       iterator = high;
-      }
-
-    for (; iterator < (1ULL << (var - tail))
-          && (cur = fill_mask (badaddr, badmask, iterator)) < addr + size;
-        iterator++)
-      {
-       grub_dprintf ("badram", "%llx (size %llx) is a badram range\n",
-                     (unsigned long long) cur, (1ULL << tail));
-       grub_mmap_register (cur, (1ULL << tail), GRUB_MEMORY_HOLE);
-      }
-    return 0;
-  }
+  struct badram_entry entry;
 
   if (argc != 1)
     return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("one argument expected"));
@@ -370,10 +387,10 @@ grub_cmd_badram (grub_command_t cmd __at
   while (1)
     {
       /* Parse address and mask.  */
-      badaddr = grub_strtoull (str, &str, 16);
+      entry.addr = grub_strtoull (str, &str, 16);
       if (*str == ',')
        str++;
-      badmask = grub_strtoull (str, &str, 16);
+      entry.mask = grub_strtoull (str, &str, 16);
       if (*str == ',')
        str++;
 
@@ -385,12 +402,13 @@ grub_cmd_badram (grub_command_t cmd __at
 
       /* When part of a page is tainted, we discard the whole of it.  There's
         no point in providing sub-page chunks.  */
-      badmask &= ~(CHUNK_SIZE - 1);
+      entry.mask &= ~(CHUNK_SIZE - 1);
 
       grub_dprintf ("badram", "badram %llx:%llx\n",
-                   (unsigned long long) badaddr, (unsigned long long) badmask);
+                   (unsigned long long) entry.addr,
+                   (unsigned long long) entry.mask);
 
-      grub_mmap_iterate (hook);
+      grub_mmap_iterate (badram_iter, &entry);
     }
 }
 
@@ -416,44 +434,48 @@ parsemem (const char *str)
   return ret;
 }
 
-static grub_err_t
-grub_cmd_cutmem (grub_command_t cmd __attribute__ ((unused)),
-                int argc, char **args)
-{
+struct cutmem_range {
   grub_uint64_t from, to;
+};
 
-  auto int NESTED_FUNC_ATTR hook (grub_uint64_t, grub_uint64_t,
-                                grub_memory_type_t);
-  int NESTED_FUNC_ATTR hook (grub_uint64_t addr,
-                            grub_uint64_t size,
-                            grub_memory_type_t type __attribute__ ((unused)))
-  {
-    grub_uint64_t end = addr + size;
-
-    if (addr <= from)
-      addr = from;
-    if (end >= to)
-      end = to;
+/* Helper for grub_cmd_cutmem.  */
+static int
+cutmem_iter (grub_uint64_t addr, grub_uint64_t size,
+            grub_memory_type_t type __attribute__ ((unused)), void *data)
+{
+  struct cutmem_range *range = data;
+  grub_uint64_t end = addr + size;
 
-    if (end <= addr)
-      return 0;
+  if (addr <= range->from)
+    addr = range->from;
+  if (end >= range->to)
+    end = range->to;
 
-    grub_mmap_register (addr, end - addr, GRUB_MEMORY_HOLE);
+  if (end <= addr)
     return 0;
-  }
+
+  grub_mmap_register (addr, end - addr, GRUB_MEMORY_HOLE);
+  return 0;
+}
+
+static grub_err_t
+grub_cmd_cutmem (grub_command_t cmd __attribute__ ((unused)),
+                int argc, char **args)
+{
+  struct cutmem_range range;
 
   if (argc != 2)
     return grub_error (GRUB_ERR_BAD_ARGUMENT, N_("two arguments expected"));
 
-  from = parsemem (args[0]);
+  range.from = parsemem (args[0]);
   if (grub_errno)
     return grub_errno;
 
-  to = parsemem (args[1]);
+  range.to = parsemem (args[1]);
   if (grub_errno)
     return grub_errno;
 
-  grub_mmap_iterate (hook);
+  grub_mmap_iterate (cutmem_iter, &range);
 
   return GRUB_ERR_NONE;
 }

=== modified file 'include/grub/efiemu/efiemu.h'
--- include/grub/efiemu/efiemu.h        2012-02-12 17:43:39 +0000
+++ include/grub/efiemu/efiemu.h        2013-01-13 01:49:43 +0000
@@ -226,7 +226,7 @@ grub_efiemu_finish_boot_services (grub_e
                                  grub_efi_uint32_t *descriptor_version);
 
 grub_err_t
-grub_efiemu_mmap_iterate (grub_memory_hook_t hook);
+grub_efiemu_mmap_iterate (grub_memory_hook_t hook, void *hook_data);
 int grub_efiemu_sizeof_uintn_t (void);
 grub_err_t
 grub_efiemu_get_lower_upper_memory (grub_uint64_t *lower, grub_uint64_t 
*upper);

=== modified file 'include/grub/memory.h'
--- include/grub/memory.h       2012-03-03 19:12:03 +0000
+++ include/grub/memory.h       2013-01-13 01:49:43 +0000
@@ -36,21 +36,25 @@ typedef enum grub_memory_type
     GRUB_MEMORY_HOLE = 21
   } grub_memory_type_t;
 
-typedef int NESTED_FUNC_ATTR (*grub_memory_hook_t) (grub_uint64_t,
-                                                   grub_uint64_t,
-                                                   grub_memory_type_t);
+typedef int (*grub_memory_hook_t) (grub_uint64_t,
+                                  grub_uint64_t,
+                                  grub_memory_type_t,
+                                  void *);
 
-grub_err_t grub_mmap_iterate (grub_memory_hook_t hook);
+grub_err_t grub_mmap_iterate (grub_memory_hook_t hook, void *hook_data);
 
 #ifdef GRUB_MACHINE_EFI
 grub_err_t
-grub_efi_mmap_iterate (grub_memory_hook_t hook, int avoid_efi_boot_services);
+grub_efi_mmap_iterate (grub_memory_hook_t hook, void *hook_data,
+                      int avoid_efi_boot_services);
 #endif
 
 #if !defined (GRUB_MACHINE_EMU) && !defined (GRUB_MACHINE_EFI)
-grub_err_t EXPORT_FUNC(grub_machine_mmap_iterate) (grub_memory_hook_t hook);
+grub_err_t EXPORT_FUNC(grub_machine_mmap_iterate) (grub_memory_hook_t hook,
+                                                  void *hook_data);
 #else
-grub_err_t grub_machine_mmap_iterate (grub_memory_hook_t hook);
+grub_err_t grub_machine_mmap_iterate (grub_memory_hook_t hook,
+                                     void *hook_data);
 #endif
 
 int grub_mmap_register (grub_uint64_t start, grub_uint64_t size, int type);

Thanks,

-- 
Colin Watson                                       address@hidden



reply via email to

[Prev in Thread] Current Thread [Next in Thread]