2018-02-28 22:29:17 +01:00
|
|
|
#include <string.h>
|
2018-02-25 03:34:15 +01:00
|
|
|
#include "utils.h"
|
|
|
|
#include "mmu.h"
|
|
|
|
#include "memory_map.h"
|
2018-03-01 00:05:18 +01:00
|
|
|
#include "arm.h"
|
2018-02-25 03:34:15 +01:00
|
|
|
|
2018-02-27 13:26:51 +01:00
|
|
|
extern const uint8_t __start_cold[];
|
|
|
|
|
2018-03-01 19:11:09 +01:00
|
|
|
extern const uint8_t __warmboot_crt0_start__[], __warmboot_crt0_end__[];
|
|
|
|
extern const uint8_t __main_start__[], __main_bss_start__[], __main_end__[];
|
|
|
|
extern const uint8_t __pk2ldr_start__[], __pk2ldr_bss_start__[], __pk2ldr_end__[];
|
|
|
|
extern const uint8_t __vectors_start__[], __vectors_end__[];
|
|
|
|
|
|
|
|
extern const size_t __warmboot_crt0_offset, __main_offset, __pk2ldr_offset, __vectors_offset;
|
2018-02-27 13:26:51 +01:00
|
|
|
|
2018-02-28 13:32:18 +01:00
|
|
|
/* warmboot_init.c */
|
|
|
|
void set_memory_registers_enable_mmu(void);
|
2018-02-25 03:34:15 +01:00
|
|
|
|
2018-02-26 22:09:35 +01:00
|
|
|
static void identity_map_all_mappings(uintptr_t *mmu_l1_tbl, uintptr_t *mmu_l3_tbl) {
|
|
|
|
static const uintptr_t addrs[] = { TUPLE_FOLD_LEFT_0(EVAL(IDENTIY_MAPPING_ID_MAX), _MMAPID, COMMA) };
|
|
|
|
static const size_t sizes[] = { TUPLE_FOLD_LEFT_1(EVAL(IDENTIY_MAPPING_ID_MAX), _MMAPID, COMMA) };
|
|
|
|
static const uint64_t attribs[] = { TUPLE_FOLD_LEFT_2(EVAL(IDENTIY_MAPPING_ID_MAX), _MMAPID, COMMA) };
|
|
|
|
static const uint64_t is_block[] = { TUPLE_FOLD_LEFT_3(EVAL(IDENTIY_MAPPING_ID_MAX), _MMAPID, COMMA) };
|
|
|
|
|
|
|
|
for(size_t i = 0; i < IDENTIY_MAPPING_ID_MAX; i++) {
|
|
|
|
identity_map_mapping(mmu_l1_tbl, mmu_l3_tbl, addrs[i], sizes[i], attribs[i], is_block[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void mmio_map_all_devices(uintptr_t *mmu_l3_tbl) {
|
|
|
|
static const uintptr_t pas[] = { TUPLE_FOLD_LEFT_0(EVAL(MMIO_DEVID_MAX), _MMAPDEV, COMMA) };
|
|
|
|
static const size_t sizes[] = { TUPLE_FOLD_LEFT_1(EVAL(MMIO_DEVID_MAX), _MMAPDEV, COMMA) };
|
|
|
|
static const bool is_secure[] = { TUPLE_FOLD_LEFT_2(EVAL(MMIO_DEVID_MAX), _MMAPDEV, COMMA) };
|
|
|
|
|
|
|
|
for(size_t i = 0, offset = 0; i < MMIO_DEVID_MAX; i++) {
|
|
|
|
mmio_map_device(mmu_l3_tbl, MMIO_BASE + offset, pas[i], sizes[i], is_secure[i]);
|
|
|
|
offset += sizes[i];
|
|
|
|
offset += 0x1000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void lp0_entry_map_all_ram_segments(uintptr_t *mmu_l3_tbl) {
|
|
|
|
static const uintptr_t pas[] = { TUPLE_FOLD_LEFT_0(EVAL(LP0_ENTRY_RAM_SEGMENT_ID_MAX), _MMAPLP0ES, COMMA) };
|
|
|
|
static const size_t sizes[] = { TUPLE_FOLD_LEFT_1(EVAL(LP0_ENTRY_RAM_SEGMENT_ID_MAX), _MMAPLP0ES, COMMA) };
|
|
|
|
static const uint64_t attribs[] = { TUPLE_FOLD_LEFT_2(EVAL(LP0_ENTRY_RAM_SEGMENT_ID_MAX), _MMAPLP0ES, COMMA) };
|
|
|
|
|
|
|
|
for(size_t i = 0, offset = 0; i < LP0_ENTRY_RAM_SEGMENT_ID_MAX; i++) {
|
|
|
|
lp0_entry_map_ram_segment(mmu_l3_tbl, LP0_ENTRY_RAM_SEGMENT_BASE + offset, pas[i], sizes[i], attribs[i]);
|
|
|
|
offset += 0x10000;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void warmboot_map_all_ram_segments(uintptr_t *mmu_l3_tbl) {
|
|
|
|
static const uintptr_t pas[] = { TUPLE_FOLD_LEFT_0(EVAL(WARMBOOT_RAM_SEGMENT_ID_MAX), _MMAPWBS, COMMA) };
|
|
|
|
static const size_t sizes[] = { TUPLE_FOLD_LEFT_1(EVAL(WARMBOOT_RAM_SEGMENT_ID_MAX), _MMAPWBS, COMMA) };
|
|
|
|
static const uint64_t attribs[] = { TUPLE_FOLD_LEFT_2(EVAL(WARMBOOT_RAM_SEGMENT_ID_MAX), _MMAPWBS, COMMA) };
|
|
|
|
|
|
|
|
for(size_t i = 0, offset = 0; i < WARMBOOT_RAM_SEGMENT_ID_MAX; i++) {
|
|
|
|
warmboot_map_ram_segment(mmu_l3_tbl, WARMBOOT_RAM_SEGMENT_BASE + offset, pas[i], sizes[i], attribs[i]);
|
|
|
|
offset += sizes[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void tzram_map_all_segments(uintptr_t *mmu_l3_tbl) {
|
|
|
|
static const uintptr_t offs[] = { TUPLE_FOLD_LEFT_0(EVAL(TZRAM_SEGMENT_ID_MAX), _MMAPTZS, COMMA) };
|
|
|
|
static const size_t sizes[] = { TUPLE_FOLD_LEFT_1(EVAL(TZRAM_SEGMENT_ID_MAX), _MMAPTZS, COMMA) };
|
|
|
|
static const size_t increments[] = { TUPLE_FOLD_LEFT_2(EVAL(TZRAM_SEGMENT_ID_MAX), _MMAPTZS, COMMA) };
|
|
|
|
static const bool is_executable[] = { TUPLE_FOLD_LEFT_3(EVAL(TZRAM_SEGMENT_ID_MAX), _MMAPTZS, COMMA) };
|
|
|
|
|
|
|
|
for(size_t i = 0, offset = 0; i < TZRAM_SEGMENT_ID_MAX; i++) {
|
2018-03-01 02:36:16 +01:00
|
|
|
tzram_map_segment(mmu_l3_tbl, TZRAM_SEGMENT_BASE + offset, 0x7C010000ull + offs[i], sizes[i], is_executable[i]);
|
2018-02-26 22:09:35 +01:00
|
|
|
offset += increments[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-25 03:34:15 +01:00
|
|
|
static void configure_ttbls(void) {
|
2018-02-26 22:09:35 +01:00
|
|
|
uintptr_t *mmu_l1_tbl = (uintptr_t *)(TZRAM_GET_SEGMENT_PA(TZRAM_SEGEMENT_ID_SECMON_EVT) + 0x800 - 64);
|
|
|
|
uintptr_t *mmu_l2_tbl = (uintptr_t *)TZRAM_GET_SEGMENT_PA(TZRAM_SEGMENT_ID_L2_TRANSLATION_TABLE);
|
|
|
|
uintptr_t *mmu_l3_tbl = (uintptr_t *)TZRAM_GET_SEGMENT_PA(TZRAM_SEGMENT_ID_L3_TRANSLATION_TABLE);
|
|
|
|
|
2018-02-25 03:34:15 +01:00
|
|
|
mmu_init_table(mmu_l1_tbl, 64); /* 33-bit address space */
|
|
|
|
mmu_init_table(mmu_l2_tbl, 4096);
|
|
|
|
/*
|
|
|
|
Nintendo uses the same L3 table for everything, but they make sure
|
|
|
|
nothing clashes.
|
|
|
|
*/
|
|
|
|
mmu_init_table(mmu_l3_tbl, 4096);
|
|
|
|
|
|
|
|
mmu_map_table(1, mmu_l1_tbl, 0x40000000, mmu_l2_tbl, 0);
|
|
|
|
mmu_map_table(1, mmu_l1_tbl, 0x1C0000000, mmu_l2_tbl, 0);
|
|
|
|
|
|
|
|
mmu_map_table(2, mmu_l2_tbl, 0x40000000, mmu_l3_tbl, 0);
|
|
|
|
mmu_map_table(2, mmu_l2_tbl, 0x7C000000, mmu_l3_tbl, 0);
|
|
|
|
mmu_map_table(2, mmu_l2_tbl, 0x1F0000000ull, mmu_l3_tbl, 0);
|
|
|
|
|
|
|
|
identity_map_all_mappings(mmu_l1_tbl, mmu_l3_tbl);
|
|
|
|
mmio_map_all_devices(mmu_l3_tbl);
|
2018-02-26 22:09:35 +01:00
|
|
|
lp0_entry_map_all_ram_segments(mmu_l3_tbl);
|
2018-02-25 21:41:16 +01:00
|
|
|
warmboot_map_all_ram_segments(mmu_l3_tbl);
|
2018-02-25 03:34:15 +01:00
|
|
|
tzram_map_all_segments(mmu_l3_tbl);
|
|
|
|
}
|
|
|
|
|
2018-03-01 19:11:09 +01:00
|
|
|
__attribute__((noinline)) static void copy_lma_to_vma(const void *vma, size_t offset, size_t size) {
|
2018-02-25 03:34:15 +01:00
|
|
|
uint64_t *p_vma = (uint64_t *)vma;
|
2018-03-01 19:11:09 +01:00
|
|
|
const uint64_t *p_lma = (const uint64_t *)(__start_cold + offset);
|
2018-02-25 03:34:15 +01:00
|
|
|
for (size_t i = 0; i < size / 8; i++) {
|
2018-02-27 13:26:51 +01:00
|
|
|
p_vma[i] = p_lma[i];
|
2018-02-25 03:34:15 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-28 23:35:30 +01:00
|
|
|
FAR_REACHING static void copy_warmboot_crt0(void) {
|
2018-03-01 19:11:09 +01:00
|
|
|
copy_lma_to_vma(__warmboot_crt0_start__, __warmboot_crt0_offset, __warmboot_crt0_end__ - __warmboot_crt0_start__);
|
2018-02-27 13:26:51 +01:00
|
|
|
}
|
|
|
|
|
2018-02-28 23:35:30 +01:00
|
|
|
FAR_REACHING static void copy_other_sections(void) {
|
2018-03-01 19:11:09 +01:00
|
|
|
copy_lma_to_vma(__main_start__, __main_offset, __main_end__ - __main_start__);
|
|
|
|
copy_lma_to_vma(__pk2ldr_start__, __pk2ldr_offset, __pk2ldr_end__ - __pk2ldr_start__);
|
|
|
|
copy_lma_to_vma(__vectors_start__, __vectors_offset, __vectors_end__ - __vectors_start__);
|
2018-02-27 13:26:51 +01:00
|
|
|
}
|
|
|
|
|
2018-03-01 00:05:18 +01:00
|
|
|
FAR_REACHING static void set_memory_registers_enable_mmu_tzram_pa(void) {
|
|
|
|
volatile uintptr_t v = (uintptr_t)set_memory_registers_enable_mmu;
|
|
|
|
((void (*)(void))v)();
|
|
|
|
}
|
|
|
|
|
|
|
|
FAR_REACHING static void flush_dcache_all_tzram_pa(void) {
|
|
|
|
uintptr_t pa = TZRAM_GET_SEGMENT_PA(TZRAM_SEGMENT_ID_WARMBOOT_CRT0_AND_MAIN);
|
|
|
|
uintptr_t main_pa = pa | ((uintptr_t)__main_start__ & 0xFFF);
|
|
|
|
uintptr_t v = (uintptr_t)flush_dcache_all - (uintptr_t)__main_start__ + (uintptr_t)main_pa;
|
|
|
|
((void (*)(void))v)();
|
|
|
|
}
|
|
|
|
|
2018-03-01 19:11:09 +01:00
|
|
|
FAR_REACHING static void invalidate_icache_all_tzram_pa(void) {
|
2018-03-01 00:05:18 +01:00
|
|
|
uintptr_t pa = TZRAM_GET_SEGMENT_PA(TZRAM_SEGMENT_ID_WARMBOOT_CRT0_AND_MAIN);
|
|
|
|
uintptr_t main_pa = pa | ((uintptr_t)__main_start__ & 0xFFF);
|
2018-03-01 19:11:09 +01:00
|
|
|
uintptr_t v = (uintptr_t)invalidate_icache_all - (uintptr_t)__main_start__ + (uintptr_t)main_pa;
|
2018-03-01 00:05:18 +01:00
|
|
|
((void (*)(void))v)();
|
|
|
|
}
|
|
|
|
|
2018-02-28 23:35:30 +01:00
|
|
|
FAR_REACHING static void clear_bss(void) {
|
2018-03-01 00:59:08 -08:00
|
|
|
volatile uintptr_t v = (uintptr_t)memset;
|
|
|
|
((void (*)(void *, int, size_t))v)((void *)__pk2ldr_bss_start__, 0, __pk2ldr_end__ - __pk2ldr_bss_start__);
|
|
|
|
((void (*)(void *, int, size_t))v)((void *)__main_bss_start__, 0, __main_end__ - __main_bss_start__);
|
2018-02-28 22:29:17 +01:00
|
|
|
}
|
|
|
|
|
2018-03-01 00:05:18 +01:00
|
|
|
uintptr_t get_coldboot_crt0_stack_address(void) {
|
|
|
|
return TZRAM_GET_SEGMENT_PA(TZRAM_SEGMENT_ID_CORE3_STACK) + 0x800;
|
|
|
|
}
|
|
|
|
|
2018-02-25 03:34:15 +01:00
|
|
|
void coldboot_init(void) {
|
|
|
|
/* TODO: Set NX BOOTLOADER clock time field */
|
2018-02-28 22:29:17 +01:00
|
|
|
|
2018-02-27 13:26:51 +01:00
|
|
|
copy_warmboot_crt0();
|
2018-02-28 13:32:18 +01:00
|
|
|
/* At this point, we can (and will) access functions located in .warm_crt0 */
|
2018-02-28 22:29:17 +01:00
|
|
|
|
2018-02-28 13:32:18 +01:00
|
|
|
/* TODO: initialize DMA controllers, etc. */
|
2018-02-25 03:34:15 +01:00
|
|
|
configure_ttbls();
|
2018-03-01 00:05:18 +01:00
|
|
|
set_memory_registers_enable_mmu_tzram_pa();
|
2018-03-01 00:59:08 -08:00
|
|
|
|
2018-03-01 01:13:10 +01:00
|
|
|
copy_other_sections();
|
2018-02-25 03:34:15 +01:00
|
|
|
|
|
|
|
flush_dcache_all_tzram_pa();
|
2018-03-01 19:11:09 +01:00
|
|
|
invalidate_icache_all_tzram_pa();
|
2018-02-28 22:29:17 +01:00
|
|
|
/* At this point we can access all the mapped segments (all other functions, data...) normally */
|
|
|
|
|
|
|
|
clear_bss();
|
2018-02-25 03:34:15 +01:00
|
|
|
}
|