summaryrefslogtreecommitdiff
path: root/src/usertestelf.c
blob: 1cbfcc565fe9740b339ba798a7784ec7a20fd84e (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
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
#include <shared/printf.h>
#include <user/lib/elf.h>
#include <user/lib/syscall.c>

const char *str = "Hello!\n", *str2 = "World.\n";

__attribute__((visibility("hidden")))
extern struct Elf64_Dyn _DYNAMIC[];

__attribute__((visibility("hidden")))
extern char _image_base[];

static void printf_backend(void *arg, const char *buf, size_t len) {
	(void)arg;
	_syscall_write(1, buf, len, -1);
	_syscall_debug_klog(buf, len);
}

int printf(const char *fmt, ...) {
	int ret = 0;
	va_list argp;
	va_start(argp, fmt);
	ret = __printf_internal(fmt, argp, printf_backend, 0);
	va_end(argp);
	return ret;
}


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;
	}
}

static void reloc(void) {
	// TODO DT_REL
	if (dyn_gettag(DT_PLTGOT) || dyn_gettag(DT_JMPREL)) {
		printf("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:
					printf("elf: unsupported relocation type\n");
			}
		}
	}
}

int main(void) {
	printf("loaded at %x\n", &_image_base);
	reloc();
	printf(str);
	printf(str2);
	_syscall_exit(0);
}