diff options
author | vetch <vetch97@gmail.com> | 2019-10-15 15:44:41 +0200 |
---|---|---|
committer | vetch <vetch97@gmail.com> | 2019-10-15 15:44:41 +0200 |
commit | e20129720adb334f46e329bfbe7090ee7188caac (patch) | |
tree | 4a8fab2da509d5e870ddbf64cf6521c110b7e0cf /loader_stage2.c | |
parent | 8de2ff74293a583fbf4cd9a87a8bf29c0c93d5a8 (diff) | |
parent | 0921cc609a1b411d2b6769a327fa11a675ac3d3a (diff) | |
download | rpi-MMU-example-e20129720adb334f46e329bfbe7090ee7188caac.tar.gz rpi-MMU-example-e20129720adb334f46e329bfbe7090ee7188caac.zip |
Merge branch 'bob' into alice
Diffstat (limited to 'loader_stage2.c')
-rw-r--r-- | loader_stage2.c | 40 |
1 files changed, 40 insertions, 0 deletions
diff --git a/loader_stage2.c b/loader_stage2.c new file mode 100644 index 0000000..e221dda --- /dev/null +++ b/loader_stage2.c @@ -0,0 +1,40 @@ +#include <stddef.h> +#include <stdint.h> +#include <uart.h> +#include <global.h> + +void *const kernel_load_addr = ((void*) 0x8000); + +void __attribute__((section(".text.stage2main"))) +stage2(uint32_t r0, uint32_t r1, uint32_t atags) +{ + // Declare as unused + (void) r0; + (void) r1; + (void) atags; + + uart_init(); + + // get kernel size via uart (little endian) + uint32_t b0, b1, b2, b3; + + b0 = uart_getc(); + b1 = uart_getc(); + b2 = uart_getc(); + b3 = uart_getc(); + + uint32_t kernel_size = b0 | (b1 << 8) | (b2 << 16) | (b3 << 24); + + // load kernel at kernel_load_addr + char *dst = kernel_load_addr, *end = dst + kernel_size; + + while (dst < end) + *(dst++) = uart_getc(); + + // jump to kernel + // TODO also forward arguments (r0, r1, atags) + asm volatile("bx %0" :: "r" (kernel_load_addr) : "memory"); +} + +void *const _start = ((void*) stage2); // for linker script + |