2 Copyright © 1995-2007, The AROS Development Team. All rights reserved.
5 Desc: Code to dynamically load ELF executables
8 1997/12/13: Changed filename to internalloadseg_elf.c
9 Original file was created by digulla.
14 #include <exec/memory.h>
15 #include <proto/exec.h>
16 #include <dos/dosasl.h>
17 #include <proto/dos.h>
18 #include <proto/arossupport.h>
19 #include <aros/asmcall.h>
20 #include "internalloadseg.h"
21 #include "dos_intern.h"
23 #include <aros/debug.h>
27 #include <aros/macros.h>
29 #define SHT_PROGBITS 1
35 #define SHT_SYMTAB_SHNDX 18
43 #define EM_X86_64 62 /* AMD x86-64 */
49 /* AMD x86-64 relocations. */
50 #define R_X86_64_NONE 0 /* No reloc */
51 #define R_X86_64_64 1 /* Direct 64 bit */
52 #define R_X86_64_PC32 2 /* PC relative 32 bit signed */
59 #define R_PPC_ADDR32 1
60 #define R_PPC_ADDR16_LO 4
61 #define R_PPC_ADDR16_HA 6
62 #define R_PPC_REL24 10
63 #define R_PPC_REL32 26
64 #define R_PPC_REL16_LO 250
65 #define R_PPC_REL16_HA 252
75 #define SHN_LORESERVE 0xff00
76 #define SHN_ABS 0xfff1
77 #define SHN_COMMON 0xfff2
78 #define SHN_XINDEX 0xffff
79 #define SHN_HIRESERVE 0xffff
81 #define SHF_ALLOC (1 << 1)
82 #define SHF_EXECINSTR (1 << 2)
84 #define ELF32_ST_TYPE(i) ((i) & 0x0F)
95 #define ELFCLASS64 2 /* 64-bit objects */
97 #define ELF32_R_SYM(val) ((val) >> 8)
98 #define ELF32_R_TYPE(val) ((val) & 0xff)
99 #define ELF32_R_INFO(sym, type) (((sym) << 8) + ((type) & 0xff))
119 /* these are internal, and not part of the header proper. they are wider
120 * versions of shnum and shstrndx for when they don't fit in the header
121 * and we need to get them from the first section header. see
122 * load_header() for details
144 ULONG name
; /* Offset of the name string in the string table */
145 ULONG value
; /* Varies; eg. the offset of the symbol in its hunk */
146 ULONG size
; /* How much memory does the symbol occupy */
147 UBYTE info
; /* What kind of symbol is this ? (global, variable, etc) */
148 UBYTE other
; /* undefined */
149 UWORD shindex
; /* In which section is the symbol defined ? */
154 ULONG offset
; /* Address of the relocation relative to the section it refers to */
155 ULONG info
; /* Type of the relocation */
156 #if defined(__mc68000__) || defined (__x86_64__) || defined (__ppc__) || defined (__powerpc__) || defined(__arm__)
157 LONG addend
; /* Constant addend used to compute value */
166 } __attribute__((packed
));
168 #define BPTR2HUNK(bptr) ((struct hunk *)((char *)BADDR(bptr) - offsetof(struct hunk, next)))
169 #define HUNK2BPTR(hunk) MKBADDR(&hunk->next)
171 /* convert section header number to array index */
173 ((n) < SHN_LORESERVE ? (n) : ((n) <= SHN_HIRESERVE ? 0 : (n) - (SHN_HIRESERVE + 1 - SHN_LORESERVE)))
175 /* convert section header array index to section number */
177 ((i) < SHN_LORESERVE ? (i) : (i) + (SHN_HIRESERVE + 1 - SHN_LORESERVE))
184 #define MyRead(file, buf, size) \
187 LONG, funcarray[0], \
188 AROS_LCA(BPTR, file, D1), \
189 AROS_LCA(void *, buf, D2), \
190 AROS_LCA(LONG, size, D3), \
191 struct DosLibrary *, DOSBase \
195 #define MyAlloc(size, flags) \
198 void *, funcarray[1], \
199 AROS_LCA(ULONG, size, D0), \
200 AROS_LCA(ULONG, flags, D1), \
201 struct ExecBase *, SysBase \
205 #define MyFree(addr, size) \
208 void, funcarray[2], \
209 AROS_LCA(void *, addr, A1), \
210 AROS_LCA(ULONG, size, D0), \
211 struct ExecBase *, SysBase \
214 static int read_block
221 struct DosLibrary
*DOSBase
224 UBYTE
*buf
= (UBYTE
*)buffer
;
227 if (Seek(file
, offset
, OFFSET_BEGINNING
) < 0)
232 subsize
= MyRead(file
, buf
, size
);
237 SetIoErr(ERROR_BAD_HUNK
);
249 static void * load_block
255 struct DosLibrary
*DOSBase
258 D(bug("[ELF Loader] Load Block\n"));
259 D(bug("[ELF Loader] (size=%d)\n",size
));
260 D(bug("[ELF Loader] (funcarray=0x%x)\n",funcarray
));
261 D(bug("[ELF Loader] (funcarray[1]=0x%x)\n",funcarray
[1]));
262 void *block
= MyAlloc(size
, MEMF_ANY
);
265 if (read_block(file
, offset
, block
, size
, funcarray
, DOSBase
))
271 SetIoErr(ERROR_NO_FREE_STORE
);
276 static int load_header(BPTR file
, struct elfheader
*eh
, SIPTR
*funcarray
, struct DosLibrary
*DOSBase
) {
277 if (!read_block(file
, 0, eh
, offsetof(struct elfheader
, int_shnum
), funcarray
, DOSBase
))
280 if (eh
->ident
[0] != 0x7f || eh
->ident
[1] != 'E' ||
281 eh
->ident
[2] != 'L' || eh
->ident
[3] != 'F') {
282 D(bug("[ELF Loader] Not an ELF object\n"));
283 SetIoErr(ERROR_NOT_EXECUTABLE
);
286 D(bug("[ELF Loader] ELF object\n"));
288 eh
->int_shnum
= eh
->shnum
;
289 eh
->int_shstrndx
= eh
->shstrndx
;
291 /* the ELF header only uses 16 bits to store the count of section headers,
292 * so it can't handle more than 65535 headers. if the count is 0, and an
293 * offset is defined, then the real count can be found in the first
294 * section header (which always exists).
296 * similarly, if the string table index is SHN_XINDEX, then the actual
297 * index is found in the first section header also.
299 * see the System V ABI 2001-04-24 draft for more details.
301 if (eh
->int_shnum
== 0 || eh
->int_shstrndx
== SHN_XINDEX
) {
302 if (eh
->shoff
== 0) {
303 SetIoErr(ERROR_NOT_EXECUTABLE
);
308 if (!read_block(file
, eh
->shoff
, &sh
, sizeof(sh
), funcarray
, DOSBase
))
311 /* wider section header count is in the size field */
312 if (eh
->int_shnum
== 0)
313 eh
->int_shnum
= sh
.size
;
315 /* wider string table index is in the link field */
316 if (eh
->int_shstrndx
== SHN_XINDEX
)
317 eh
->int_shstrndx
= sh
.link
;
319 /* sanity, if they're still invalid then this isn't elf */
320 if (eh
->int_shnum
== 0 || eh
->int_shstrndx
== SHN_XINDEX
) {
321 SetIoErr(ERROR_NOT_EXECUTABLE
);
328 eh
->ident
[EI_CLASS
] != ELFCLASS32
||
329 eh
->ident
[EI_VERSION
] != EV_CURRENT
||
330 eh
->type
!= ET_REL
||
332 #if defined(__i386__)
333 eh
->ident
[EI_DATA
] != ELFDATA2LSB
||
334 eh
->machine
!= EM_386
336 #elif defined(__x86_64__)
337 eh
->ident
[EI_DATA
] != ELFDATA2LSB
||
338 eh
->machine
!= EM_X86_64
340 #elif defined(__mc68000__)
341 eh
->ident
[EI_DATA
] != ELFDATA2MSB
||
342 eh
->machine
!= EM_68K
344 #elif defined(__ppc__) || defined(__powerpc__)
345 eh
->ident
[EI_DATA
] != ELFDATA2MSB
||
346 eh
->machine
!= EM_PPC
348 #elif defined(__arm__)
349 eh
->ident
[EI_DATA
] != ELFDATA2LSB
||
350 eh
->machine
!= EM_ARM
351 #warning ARM has not been tested, yet!
354 # error Your architecture is not supported
358 D(bug("[ELF Loader] Object is of wrong type\n"));
359 #if defined(__x86_64__)
360 D(bug("[ELF Loader] EI_CLASS is %d - should be %d\n", eh
->ident
[EI_CLASS
], ELFCLASS64
));
362 D(bug("[ELF Loader] EI_CLASS is %d - should be %d\n", eh
->ident
[EI_CLASS
], ELFCLASS32
));
364 D(bug("[ELF Loader] EI_VERSION is %d - should be %d\n", eh
->ident
[EI_VERSION
], EV_CURRENT
));
365 D(bug("[ELF Loader] type is %d - should be %d\n", eh
->type
, ET_REL
));
366 #if defined (__i386__)
367 D(bug("[ELF Loader] EI_DATA is %d - should be %d\n", eh
->ident
[EI_DATA
],ELFDATA2LSB
));
368 #elif defined (__mc68000__)
369 D(bug("[ELF Loader] EI_DATA is %d - should be %d\n", eh
->ident
[EI_DATA
],ELFDATA2MSB
));
370 #elif defined(__ppc__) || defined(__powerpc__)
371 D(bug("[ELF Loader] EI_DATA is %d - should be %d\n", eh
->ident
[EI_DATA
],ELFDATA2MSB
));
372 #elif defined (__arm__)
373 D(bug("[ELF Loader] EI_DATA is %d - should be %d\n", eh
->ident
[EI_DATA
],ELFDATA2MSB
));
376 #if defined (__i386__)
377 D(bug("[ELF Loader] machine is %d - should be %d\n", eh
->machine
, EM_386
));
378 #elif defined(__mc68000__)
379 D(bug("[ELF Loader] machine is %d - should be %d\n", eh
->machine
, EM_68K
));
380 #elif defined(__ppc__) || defined(__powerpc__)
381 D(bug("[ELF Loader] machine is %d - should be %d\n", eh
->machine
, EM_PPC
));
382 #elif defined(__arm__)
383 D(bug("[ELF Loader] machine is %d - should be %d\n", eh
->machine
, EM_ARM
));
386 SetIoErr(ERROR_NOT_EXECUTABLE
);
396 BPTR
**next_hunk_ptr
,
400 struct DosLibrary
*DOSBase
409 /* The size of the hunk is the size of the section, plus
410 the size of the hunk structure, plus the size of the alignment (if necessary)*/
411 hunk_size
= sh
->size
+ sizeof(struct hunk
);
415 hunk_size
+= sh
->addralign
;
417 /* Also create space for a trampoline, if necessary */
418 if (sh
->flags
& SHF_EXECINSTR
)
419 hunk_size
+= sizeof(struct FullJumpVec
);
422 hunk
= MyAlloc(hunk_size
, MEMF_ANY
| (sh
->type
== SHT_NOBITS
) ? MEMF_CLEAR
: 0);
426 hunk
->size
= hunk_size
;
428 /* In case we are required to honour alignment, and If this section contains
429 executable code, create a trampoline to its beginning, so that even if the
430 alignment requirements make the actual code go much after the end of the
431 hunk structure, the code can still be reached in the usual way. */
434 if (sh
->flags
& SHF_EXECINSTR
)
436 sh
->addr
= (char *)AROS_ROUNDUP2
438 (ULONG
)hunk
->data
+ sizeof(struct FullJumpVec
), sh
->addralign
440 __AROS_SET_FULLJMP((struct FullJumpVec
*)hunk
->data
, sh
->addr
);
443 sh
->addr
= (char *)AROS_ROUNDUP2((ULONG
)hunk
->data
, sh
->addralign
);
446 sh
->addr
= hunk
->data
;
448 /* Link the previous one with the new one */
449 BPTR2HUNK(*next_hunk_ptr
)->next
= HUNK2BPTR(hunk
);
451 /* Update the pointer to the previous one, which is now the current one */
452 *next_hunk_ptr
= HUNK2BPTR(hunk
);
454 if (sh
->type
!= SHT_NOBITS
)
455 return read_block(file
, sh
->offset
, sh
->addr
, sh
->size
, funcarray
, DOSBase
);
461 SetIoErr(ERROR_NO_FREE_STORE
);
468 struct elfheader
*eh
,
471 struct sheader
*symtab_shndx
,
472 struct DosLibrary
*DOSBase
475 struct sheader
*shrel
= &sh
[shrel_idx
];
476 struct sheader
*shsymtab
= &sh
[SHINDEX(shrel
->link
)];
477 struct sheader
*toreloc
= &sh
[SHINDEX(shrel
->info
)];
479 struct symbol
*symtab
= (struct symbol
*)shsymtab
->addr
;
480 struct relo
*rel
= (struct relo
*)shrel
->addr
;
481 char *section
= toreloc
->addr
;
483 /* this happens if the target section has no allocation. that can happen
484 * eg. with a .debug PROGBITS and a .rel.debug section */
488 ULONG numrel
= shrel
->size
/ shrel
->entsize
;
491 struct symbol
*SysBase_sym
= NULL
;
493 for (i
=0; i
<numrel
; i
++, rel
++)
495 struct symbol
*sym
= &symtab
[ELF32_R_SYM(rel
->info
)];
496 ULONG
*p
= (ULONG
*)§ion
[rel
->offset
];
500 if (sym
->shindex
!= SHN_XINDEX
)
501 shindex
= sym
->shindex
;
504 if (symtab_shndx
== NULL
) {
505 D(bug("[ELF Loader] got symbol with shndx 0xfff, but there's no symtab shndx table\n"));
506 SetIoErr(ERROR_BAD_HUNK
);
509 shindex
= ((ULONG
*)symtab_shndx
->addr
)[ELF32_R_SYM(rel
->info
)];
516 D(bug("[ELF Loader] Undefined symbol '%s' while relocating the section '%s'\n",
517 (STRPTR
)sh
[SHINDEX(shsymtab
->link
)].addr
+ sym
->name
,
518 (STRPTR
)sh
[SHINDEX(eh
->int_shstrndx
)].addr
+ toreloc
->name
));
519 SetIoErr(ERROR_BAD_HUNK
);
523 D(bug("[ELF Loader] COMMON symbol '%s' while relocating the section '%s'\n",
524 (STRPTR
)sh
[SHINDEX(shsymtab
->link
)].addr
+ sym
->name
,
525 (STRPTR
)sh
[SHINDEX(eh
->int_shstrndx
)].addr
+ toreloc
->name
));
526 SetIoErr(ERROR_BAD_HUNK
);
531 if (SysBase_sym
== NULL
)
533 if (strncmp((STRPTR
)sh
[SHINDEX(shsymtab
->link
)].addr
+ sym
->name
, "SysBase", 8) == 0)
542 if (SysBase_sym
== sym
)
544 SysBase_yes
: s
= (ULONG
)&SysBase
;
547 SysBase_no
: s
= sym
->value
;
551 s
= (ULONG
)sh
[SHINDEX(shindex
)].addr
+ sym
->value
;
554 switch (ELF32_R_TYPE(rel
->info
))
556 #if defined(__i386__)
558 case R_386_32
: /* 32bit absolute */
562 case R_386_PC32
: /* 32bit PC relative */
569 #elif defined(__x86_64__)
570 /* These weren't tested */
571 case R_X86_64_64
: /* 64bit direct/absolute */
572 *p
= s
+ rel
->addend
;
575 case R_X86_64_PC32
: /* PC relative 32 bit signed */
576 *p
= s
+ rel
->addend
- (ULONG
)p
;
579 case R_X86_64_NONE
: /* No reloc */
582 #elif defined(__mc68000__)
585 *p
= s
+ rel
->addend
;
589 *p
= s
+ rel
->addend
- (ULONG
)p
;
595 #elif defined(__ppc__) || defined(__powerpc__)
598 *p
= s
+ rel
->addend
;
601 case R_PPC_ADDR16_LO
:
603 unsigned short *c
= (unsigned short *) p
;
604 *c
= (s
+ rel
->addend
) & 0xffff;
608 case R_PPC_ADDR16_HA
:
610 unsigned short *c
= (unsigned short *) p
;
611 ULONG temp
= s
+ rel
->addend
;
613 if ((temp
& 0x8000) != 0)
620 unsigned short *c
= (unsigned short *) p
;
621 *c
= (s
+ rel
->addend
- (ULONG
) p
) & 0xffff;
627 unsigned short *c
= (unsigned short *) p
;
628 ULONG temp
= s
+ rel
->addend
- (ULONG
) p
;
630 if ((temp
& 0x8000) != 0)
637 *p
|= (s
+ rel
->addend
- (ULONG
) p
) & 0x3fffffc;
641 *p
= s
+ rel
->addend
- (ULONG
) p
;
647 #elif defined(__arm__)
650 * This has not been tested. Taken from ARMELF.pdf
651 * from arm.com page 33ff.
654 *p
= s
+ rel
->addend
- (ULONG
)p
;
658 *p
= s
+ rel
->addend
;
665 # error Your architecture is not supported
669 D(bug("[ELF Loader] Unrecognized relocation type %d %d\n", i
, ELF32_R_TYPE(rel
->info
)));
670 SetIoErr(ERROR_BAD_HUNK
);
678 BPTR InternalLoadSeg_ELF
683 SIPTR
*stack __unused
,
684 struct MinList
*seginfos
,
685 struct DosLibrary
*DOSBase
690 struct sheader
*symtab_shndx
= NULL
;
692 BPTR
*next_hunk_ptr
= MKBADDR(&hunks
);
694 BOOL exec_hunk_seen
= FALSE
;
696 /* load and validate ELF header */
697 if (!load_header(file
, &eh
, funcarray
, DOSBase
))
700 /* load section headers */
701 if (!(sh
= load_block(file
, eh
.shoff
, eh
.int_shnum
* eh
.shentsize
, funcarray
, DOSBase
)))
704 /* load the string table */
706 struct sheader
*shstr
= sh
+ SHINDEX(eh
.int_shstrndx
);
707 if (shstr
->size
!= 0)
709 st
= MyAlloc(shstr
->size
, MEMF_ANY
| MEMF_CLEAR
);
710 read_block(file
, shstr
->offset
, st
, shstr
->size
, funcarray
, DOSBase
);
713 /* Iterate over the section headers in order to do some stuff... */
714 for (i
= 0; i
< eh
.int_shnum
; i
++)
717 Load the symbol and string table(s).
719 NOTICE: the ELF standard, at the moment (Nov 2002) explicitely states
720 that only one symbol table per file is allowed. However, it
721 also states that this may change in future... we already handle it.
723 if (sh
[i
].type
== SHT_SYMTAB
|| sh
[i
].type
== SHT_STRTAB
|| sh
[i
].type
== SHT_SYMTAB_SHNDX
)
725 sh
[i
].addr
= load_block(file
, sh
[i
].offset
, sh
[i
].size
, funcarray
, DOSBase
);
729 if (sh
[i
].type
== SHT_SYMTAB_SHNDX
) {
730 if (symtab_shndx
== NULL
)
731 symtab_shndx
= &sh
[i
];
733 D(bug("[ELF Loader] file contains multiple symtab shndx tables. only using the first one\n"));
737 /* Load the section in memory if needed, and make an hunk out of it */
738 if (sh
[i
].flags
& SHF_ALLOC
)
742 /* Only allow alignment if this is an executable hunk
743 or if an executable hunk has been loaded already,
744 so to avoid the situation in which a data hunk has its
745 content displaced from the hunk's header in case it's the
746 first hunk (this happens with Keymaps, for instance). */
747 if (sh
[i
].flags
& SHF_EXECINSTR
)
748 exec_hunk_seen
= TRUE
;
750 if (!load_hunk(file
, &next_hunk_ptr
, &sh
[i
], funcarray
, exec_hunk_seen
, DOSBase
))
755 STRPTR name
= st
+ sh
[i
].name
;
756 ULONG size
= sizeof(struct seginfo
);
757 struct seginfo
*si
= MyAlloc(size
, MEMF_ANY
);
759 D(bug("[ELF Loader] seg %s at 0x%x\n", name
, sh
[i
].addr
));
761 si
->addr
= sh
[i
].addr
;
762 size
= sizeof(si
->name
) - 1;
763 strncpy(si
->name
, name
, size
);
764 si
->name
[size
] = '\0';
766 ADDTAIL(seginfos
, &si
->node
);
773 /* Relocate the sections */
774 for (i
= 0; i
< eh
.int_shnum
; i
++)
778 #if defined(__i386__)
780 sh
[i
].type
== SHT_REL
&&
782 #elif defined(__x86_64__)
784 sh
[i
].type
== SHT_RELA
&&
786 #elif defined(__mc68000__)
788 sh
[i
].type
== SHT_RELA
&&
790 #elif defined(__ppc__) || defined(__powerpc__)
792 sh
[i
].type
== SHT_RELA
&&
794 #elif defined(__arm__)
795 #warning Missing code for ARM
799 # error Your architecture is not supported
802 /* Does this relocation section refer to a hunk? If so, addr must be != 0 */
803 sh
[SHINDEX(sh
[i
].info
)].addr
806 sh
[i
].addr
= load_block(file
, sh
[i
].offset
, sh
[i
].size
, funcarray
, DOSBase
);
807 if (!sh
[i
].addr
|| !relocate(&eh
, sh
, i
, symtab_shndx
, DOSBase
))
810 MyFree(sh
[i
].addr
, sh
[i
].size
);
820 /* There were some errors, deallocate The hunks */
822 InternalUnLoadSeg(hunks
, (VOID_FUNC
)funcarray
[2]);
827 /* Clear the caches to let the CPU see the new data and instructions */
832 struct hunk
*hunk
= BPTR2HUNK(curr
);
834 CacheClearE(hunk
->data
, hunk
->size
, CACRF_ClearD
| CACRF_ClearI
);
840 /* deallocate the symbol tables */
841 for (i
= 0; i
< eh
.int_shnum
; i
++)
843 if (((sh
[i
].type
== SHT_SYMTAB
) || (sh
[i
].type
== SHT_STRTAB
)) && (sh
[i
].addr
!= NULL
))
844 MyFree(sh
[i
].addr
, sh
[i
].size
);
847 /* Free the string table */
848 MyFree(st
, shstr
->size
);
850 /* Free the section headers */
851 MyFree(sh
, eh
.int_shnum
* eh
.shentsize
);