blob: b6e38442650cc4e33ec298777b76c3cbf3ff2057 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
|
#include <stdio.h>
#include <user/lib/elf.h>
__attribute__((visibility("hidden")))
extern struct Elf64_Dyn _DYNAMIC[];
__attribute__((visibility("hidden")))
extern char _image_base[];
static struct Elf64_Dyn *dyn_gettag(Elf64_Xword tag) {
for (size_t i = 0;; i++) {
if (_DYNAMIC[i].d_tag == tag) return &_DYNAMIC[i];
if (_DYNAMIC[i].d_tag == DT_NULL) return NULL;
}
}
void elf_selfreloc(void) {
// TODO DT_REL
if (dyn_gettag(DT_PLTGOT) || dyn_gettag(DT_JMPREL)) {
_klogf("elf: unimplemented tag in _DYNAMIC\n");
}
struct Elf64_Dyn *rela_tag = dyn_gettag(DT_RELA);
if (rela_tag) {
/* not checking pointer validity,
* crashing on an invalid elf is fine */
size_t relasz = dyn_gettag(DT_RELASZ)->d_val;
size_t relaent = dyn_gettag(DT_RELAENT)->d_val;
for (size_t o = 0; o < relasz; o += relaent) {
struct Elf64_Rela *r = (void*)_image_base + rela_tag->d_ptr + o;
uintptr_t *target = (void*)_image_base + r->r_offset;
switch (ELF64_R_TYPE(r->r_info)) {
case R_X86_64_RELATIVE:
*target = (uintptr_t)&_image_base + r->r_addend;
break;
default:
_klogf("elf: unsupported relocation type\n");
}
}
}
}
|