]> git.ipfire.org Git - thirdparty/grub.git/commitdiff
Working hello.mod with extcmd.mod
authorphcoder <phcoder@debian>
Sun, 2 Jan 2011 18:20:28 +0000 (19:20 +0100)
committerphcoder <phcoder@debian>
Sun, 2 Jan 2011 18:20:28 +0000 (19:20 +0100)
grub-core/kern/dl.c
grub-core/kern/ia64/dl.c

index 9dfff52870da2e49c09c65e198d14739c2e896f3..0ccdbe03b2b388fde10218091b238461ba37ca37 100644 (file)
@@ -238,28 +238,32 @@ grub_dl_load_segments (grub_dl_t mod, const Elf_Ehdr *e)
        {
          grub_dl_segment_t seg;
          grub_size_t tramp_size = 0;
+         grub_size_t alsize, align;
 
          seg = (grub_dl_segment_t) grub_malloc (sizeof (*seg));
          if (! seg)
            return grub_errno;
 
+         alsize = s->sh_size;
+         align = s->sh_addralign;
          tramp_size = grub_arch_dl_get_tramp_size (e, i);
-         if (tramp_size && s->sh_addralign < GRUB_ARCH_DL_TRAMP_ALIGN)
+         if (tramp_size && align < GRUB_ARCH_DL_TRAMP_ALIGN)
            {
-             s->sh_addralign = GRUB_ARCH_DL_TRAMP_ALIGN;
-             s->sh_size = ALIGN_UP (s->sh_size, GRUB_ARCH_DL_TRAMP_ALIGN) + tramp_size;
+             align = GRUB_ARCH_DL_TRAMP_ALIGN;
+             alsize = ALIGN_UP (alsize, GRUB_ARCH_DL_TRAMP_ALIGN);
            }
+         alsize += tramp_size;
 #ifdef GRUB_MACHINE_EMU
-         if (s->sh_addralign < 8192)
-           s->sh_addralign = 8192;
-         s->sh_size = ALIGN_UP (s->sh_size, 8192);
+         if (align < 8192 * 16)
+           align = 8192 * 16;
+         alsize = ALIGN_UP (alsize, 8192 * 16);
 #endif
 
-         if (s->sh_size)
+         if (alsize)
            {
              void *addr;
 
-             addr = grub_memalign (s->sh_addralign, s->sh_size);
+             addr = grub_memalign (align, alsize);
              if (! addr)
                {
                  grub_free (seg);
@@ -279,7 +283,7 @@ grub_dl_load_segments (grub_dl_t mod, const Elf_Ehdr *e)
              seg->addr = addr;
 #ifdef GRUB_MACHINE_EMU
              if (s->sh_flags & SHF_EXECINSTR)
-               mprotect (addr, s->sh_size, PROT_READ | PROT_WRITE | PROT_EXEC);
+               mprotect (addr, alsize, PROT_READ | PROT_WRITE | PROT_EXEC);
 #endif
            }
          else
@@ -359,9 +363,8 @@ grub_dl_resolve_symbols (grub_dl_t mod, Elf_Ehdr *e)
        case STT_FUNC:
          sym->st_value += (Elf_Addr) grub_dl_get_section_addr (mod,
                                                                sym->st_shndx);
-         if (bind != STB_LOCAL)
-           {
 #ifdef __ia64__
+         {
              /* FIXME: free descriptor once it's not used anymore. */
              char **desc;
              desc = grub_malloc (2 * sizeof (char *));
@@ -371,10 +374,13 @@ grub_dl_resolve_symbols (grub_dl_t mod, Elf_Ehdr *e)
              desc[1] = mod->gp;
              if (grub_dl_register_symbol (name, (void *) desc, mod))
                return grub_errno;
-#else
+             sym->st_value = (grub_addr_t) desc;
+         }
+#endif
+         if (bind != STB_LOCAL)
+           {
              if (grub_dl_register_symbol (name, (void *) sym->st_value, mod))
                return grub_errno;
-#endif
            }
          if (grub_strcmp (name, "grub_mod_init") == 0)
            mod->init = sym->st_value;
@@ -405,14 +411,7 @@ grub_dl_call_init (grub_dl_t mod)
 {
   if (mod->init)
     {
-#ifndef __ia64__
       ((void (*) (grub_dl_t)) mod->init) (mod);
-#else
-      char *jmp[2];
-      jmp[0] = (char *) mod->init;
-      jmp[1] = mod->gp;
-      ((void (*) (grub_dl_t)) jmp) (mod);
-#endif
     }
 }
 
@@ -678,14 +677,7 @@ grub_dl_unload (grub_dl_t mod)
 
   if (mod->fini)
     {
-#ifndef __ia64__
       ((void (*) (void)) mod->fini) ();
-#else
-      char *jmp[2];
-      jmp[0] = (char *) mod->fini;
-      jmp[1] = mod->gp;
-      ((void (*) (void)) jmp) ();
-#endif
     }
 
   grub_dl_remove (mod);
index 1fbe185b40afe1d4167faf9b9d4ed24295c679f1..55b50ff69da4da8497a762d6fd3e98faad9c1770 100644 (file)
@@ -41,23 +41,62 @@ grub_arch_dl_check_header (void *ehdr)
 #define MASK20 ((1 << 20) - 1)
 #define MASK19 ((1 << 19) - 1)
 
+struct unaligned_uint32
+{
+  grub_uint32_t val;
+}  __attribute__ ((packed));
+
+static void
+add_value_to_slot_20b (grub_addr_t addr, grub_uint32_t value)
+{
+  struct unaligned_uint32 *p;
+  switch (addr & 3)
+    {
+    case 0:
+      p = (struct unaligned_uint32 *) ((addr & ~3ULL) + 2);
+      p->val = (((((p->val >> 2) & MASK20) + value) & MASK20) << 2) | (p->val & ~(MASK20 << 2));
+      break;
+    case 1:
+      p = (struct unaligned_uint32 *) ((grub_uint8_t *) (addr & ~3ULL) + 7);
+      p->val = (((((p->val >> 3) & MASK20) + value) & MASK20) << 3) | (p->val & ~(MASK20 << 3));
+      break;
+    case 2:
+      p = (struct unaligned_uint32 *) ((grub_uint8_t *) (addr & ~3ULL) + 12);
+      p->val = (((((p->val >> 4) & MASK20) + value) & MASK20) << 4) | (p->val & ~(MASK20 << 4));
+      break;
+    }
+}
+
+#define MASKF21 ( ((1 << 23) - 1) & ~((1 << 7) | (1 << 8)) )
+
+static grub_uint32_t
+add_value_to_slot_21_real (grub_uint32_t a, grub_uint32_t value)
+{
+  grub_uint32_t high, mid, low, c;
+  low  = (a & 0x00007f);
+  mid  = (a & 0x7fc000) >> 7;
+  high = (a & 0x003e00) << 5;
+  c = (low | mid | high) + value;
+  return (c & 0x7f) | ((c << 7) & 0x7fc000) | ((c >> 5) & 0x003e00);
+}
+
 static void
-add_value_to_slot13_20 (Elf_Word *addr, grub_uint32_t value, int slot)
+add_value_to_slot_21 (grub_addr_t addr, grub_uint32_t value)
 {
-  grub_uint32_t *p __attribute__ ((aligned (1)));
-  switch (slot)
+  struct unaligned_uint32 *p;
+  switch (addr & 3)
     {
     case 0:
-      p = (grub_uint32_t *) (addr + 2);
-      *p = (((((*p >> 2) & MASK20) + value) & MASK20) << 2) | (*p & ~(MASK20 << 2));
+      p = (struct unaligned_uint32 *) ((addr & ~3ULL) + 2);
+      p->val = ((add_value_to_slot_21_real (((p->val >> 2) & MASKF21), value) & MASKF21) << 2) | (p->val & ~(MASKF21 << 2));
       break;
     case 1:
-      p = (grub_uint32_t *) ((grub_uint8_t *) addr + 7);
-      *p = (((((*p >> 3) & MASK20) + value) & MASK20) << 3) | (*p & ~(MASK20 << 3));
+      p = (struct unaligned_uint32 *) ((grub_uint8_t *) (addr & ~3ULL) + 7);
+      p->val = ((add_value_to_slot_21_real (((p->val >> 3) & MASKF21), value) & MASKF21) << 3) | (p->val & ~(MASKF21 << 3));
       break;
     case 2:
-      p = (grub_uint32_t *) ((grub_uint8_t *) addr + 12);
-      *p = (((((*p >> 4) & MASK20) + value) & MASK20) << 4) | (*p & ~(MASK20 << 4));
+      p = (struct unaligned_uint32 *) ((grub_uint8_t *) (addr & ~3ULL) + 12);
+      p->val = ((add_value_to_slot_21_real (((p->val >> 4) & MASKF21), value) & MASKF21) << 4) | (p->val & ~(MASKF21 << 4));
       break;
     }
 }
@@ -147,8 +186,8 @@ grub_arch_dl_get_tramp_size (const void *ehdr, unsigned sec)
        for (rel = (Elf_Rela *) ((char *) e + s->sh_offset),
               max = rel + s->sh_size / s->sh_entsize;
             rel < max; rel++)
-           if (ELF_R_TYPE (rel->r_info) == R_IA64_PCREL21B)
-             cnt++;
+         if (ELF_R_TYPE (rel->r_info) == R_IA64_PCREL21B)
+           cnt++;
       }
 
   return cnt * sizeof (struct ia64_trampoline);
@@ -165,7 +204,7 @@ grub_arch_dl_allocate_gp (grub_dl_t mod, const void *ehdr)
   for (i = 0, s = (Elf_Shdr *) ((char *) e + e->e_shoff);
        i < e->e_shnum;
        i++, s = (Elf_Shdr *) ((char *) s + e->e_shentsize))
-    if (s->sh_type == SHT_REL)
+    if (s->sh_type == SHT_RELA)
       {
        grub_dl_segment_t seg;
 
@@ -176,16 +215,18 @@ grub_arch_dl_allocate_gp (grub_dl_t mod, const void *ehdr)
 
        if (seg)
          {
-           Elf_Rel *rel, *max;
+           Elf_Rela *rel, *max;
 
-           for (rel = (Elf_Rel *) ((char *) e + s->sh_offset),
+           for (rel = (Elf_Rela *) ((char *) e + s->sh_offset),
                   max = rel + s->sh_size / s->sh_entsize;
                 rel < max;
                 rel++)
                switch (ELF_R_TYPE (rel->r_info))
                  {
+                 case R_IA64_LTOFF_FPTR22:
                  case R_IA64_LTOFF22X:
                  case R_IA64_LTOFF22:
+                 case R_IA64_GPREL22:
                    gp_size += 8;
                    break;
                  default: break;
@@ -243,29 +284,22 @@ grub_arch_dl_relocate_symbols (grub_dl_t mod, void *ehdr)
            Elf_Rela *rel, *max;
            struct ia64_trampoline *tr;
 
+           tr = (void *) ((char *) seg->addr + ALIGN_UP (seg->size, GRUB_ARCH_DL_TRAMP_ALIGN));
+
            for (rel = (Elf_Rela *) ((char *) e + s->sh_offset),
                   max = rel + s->sh_size / s->sh_entsize;
                 rel < max;
                 rel++)
              {
-               Elf_Word *addr;
+               grub_addr_t addr;
                Elf_Sym *sym;
                grub_uint64_t value;
-               int slot = 0;
 
                if (seg->size < (rel->r_offset & ~3))
                  return grub_error (GRUB_ERR_BAD_MODULE,
                                     "reloc offset is out of the segment");
 
-               tr = (void *) ((char *) seg->addr + ALIGN_UP (seg->size, GRUB_ARCH_DL_TRAMP_ALIGN));
-
-               if (ELF_R_TYPE (rel->r_info) == R_IA64_PCREL21B)
-                 {
-                   addr = (Elf_Word *) ((char *) seg->addr + (rel->r_offset & ~3));
-                   slot = rel->r_offset & 3;
-                 }
-               else
-                 addr = (Elf_Word *) ((char *) seg->addr + rel->r_offset);
+               addr = (grub_addr_t) seg->addr + rel->r_offset;
                sym = (Elf_Sym *) ((char *) mod->symtab
                                     + entsize * ELF_R_SYM (rel->r_info));
 
@@ -278,21 +312,26 @@ grub_arch_dl_relocate_symbols (grub_dl_t mod, void *ehdr)
                    {
                      grub_uint64_t noff;
                      make_trampoline (tr, value);
-                     noff = ((char *) tr - (char *) addr) >> 4;
+                     noff = ((char *) tr - (char *) (addr & ~3)) >> 4;
                      tr++;
                      if (noff & ~MASK19)
                        return grub_error (GRUB_ERR_BAD_OS,
-                                          "trampoline offset too big");
-                     add_value_to_slot13_20 (addr, noff, slot);
+                                          "trampoline offset too big (%lx)", noff);
+                     add_value_to_slot_20b (addr, noff);
                    }
                    break;
                  case R_IA64_SEGREL64LSB:
                    *(grub_uint64_t *) addr += value - rel->r_offset;
                    break;
+                 case R_IA64_DIR64LSB:
+                   *(grub_uint64_t *) addr += value;
+                   break;
+                 case R_IA64_LTOFF_FPTR22:
                  case R_IA64_LTOFF22X:
                  case R_IA64_LTOFF22:
+                 case R_IA64_GPREL22:
                    *gpptr = value;
-                   add_value_to_slot13_20 (addr, (gpptr - gp) * sizeof (grub_uint64_t), slot);
+                   add_value_to_slot_21 (addr, (gpptr - gp) * sizeof (grub_uint64_t));
                    gpptr++;
                    break;