aboutsummaryrefslogtreecommitdiff
path: root/02_process_init/entry.c
blob: a6b0918c413d39d2f016732273f25016d31a410b (plain) (blame)
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
70
71
72
73
74
75
// Copyright (c) 2020 Johannes Stoelp

#include <asm/unistd.h>
#include <elf.h>
#include <fmt.h>
#include <stdint.h>
#include <syscall.h>

#if !defined(__linux__) || !defined(__x86_64__)
#    error "Only supported in linux(x86_64)!"
#endif

int dynld_printf(const char* fmt, ...) {
    va_list ap;
    va_start(ap, fmt);
    char buf[64];
    int ret = dynld_vsnprintf(buf, sizeof(buf), fmt, ap);
    va_end(ap);
    syscall3(__NR_write, 1 /* stdout */, buf, ret);
    return ret;
}

void entry(long* prctx) {
    // Interpret data on the stack passed by the OS kernel as specified in the
    // x86_64 SysV ABI.

    long argc = *prctx;
    const char** argv = (const char**)(prctx + 1);
    const char** envv = (const char**)(argv + argc + 1);

    int envc = 0;
    for (const char** env = envv; *env; ++env) {
        ++envc;
    }

    uint64_t auxv[AT_MAX_CNT];
    for (unsigned i = 0; i < AT_MAX_CNT; ++i) {
        auxv[i] = 0;
    }

    const uint64_t* auxvp = (const uint64_t*)(envv + envc + 1);
    for (unsigned i = 0; auxvp[i] != AT_NULL; i += 2) {
        if (auxvp[i] < AT_MAX_CNT) {
            auxv[auxvp[i]] = auxvp[i + 1];
        }
    }

    // Print for demonstration

    dynld_printf("Got %d arg(s)\n", argc);
    for (const char** arg = argv; *arg; ++arg) {
        dynld_printf("\targ = %s\n", *arg);
    }

    const int max_env = 10;
    dynld_printf("Print first %d env var(s)\n", max_env - 1);
    for (const char** env = envv; *env && (env - envv < max_env); ++env) {
        dynld_printf("\tenv = %s\n", *env);
    }

    dynld_printf("Print auxiliary vector\n");
    dynld_printf("\tAT_EXECFD: %ld\n", auxv[AT_EXECFD]);
    dynld_printf("\tAT_PHDR  : %p\n", auxv[AT_PHDR]);
    dynld_printf("\tAT_PHENT : %ld\n", auxv[AT_PHENT]);
    dynld_printf("\tAT_PHNUM : %ld\n", auxv[AT_PHNUM]);
    dynld_printf("\tAT_PAGESZ: %ld\n", auxv[AT_PAGESZ]);
    dynld_printf("\tAT_BASE  : %lx\n", auxv[AT_BASE]);
    dynld_printf("\tAT_FLAGS : %ld\n", auxv[AT_FLAGS]);
    dynld_printf("\tAT_ENTRY : %p\n", auxv[AT_ENTRY]);
    dynld_printf("\tAT_NOTELF: %lx\n", auxv[AT_NOTELF]);
    dynld_printf("\tAT_UID   : %ld\n", auxv[AT_UID]);
    dynld_printf("\tAT_EUID  : %ld\n", auxv[AT_EUID]);
    dynld_printf("\tAT_GID   : %ld\n", auxv[AT_GID]);
    dynld_printf("\tAT_EGID  : %ld\n", auxv[AT_EGID]);
}