1*344aa361SAndroid Build Coastguard Worker /*
2*344aa361SAndroid Build Coastguard Worker * Copyright (c) 2012 Travis Geiselbrecht
3*344aa361SAndroid Build Coastguard Worker *
4*344aa361SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining
5*344aa361SAndroid Build Coastguard Worker * a copy of this software and associated documentation files
6*344aa361SAndroid Build Coastguard Worker * (the "Software"), to deal in the Software without restriction,
7*344aa361SAndroid Build Coastguard Worker * including without limitation the rights to use, copy, modify, merge,
8*344aa361SAndroid Build Coastguard Worker * publish, distribute, sublicense, and/or sell copies of the Software,
9*344aa361SAndroid Build Coastguard Worker * and to permit persons to whom the Software is furnished to do so,
10*344aa361SAndroid Build Coastguard Worker * subject to the following conditions:
11*344aa361SAndroid Build Coastguard Worker *
12*344aa361SAndroid Build Coastguard Worker * The above copyright notice and this permission notice shall be
13*344aa361SAndroid Build Coastguard Worker * included in all copies or substantial portions of the Software.
14*344aa361SAndroid Build Coastguard Worker *
15*344aa361SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
16*344aa361SAndroid Build Coastguard Worker * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
17*344aa361SAndroid Build Coastguard Worker * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
18*344aa361SAndroid Build Coastguard Worker * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
19*344aa361SAndroid Build Coastguard Worker * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
20*344aa361SAndroid Build Coastguard Worker * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
21*344aa361SAndroid Build Coastguard Worker * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
22*344aa361SAndroid Build Coastguard Worker */
23*344aa361SAndroid Build Coastguard Worker #include <arch.h>
24*344aa361SAndroid Build Coastguard Worker #include <arch/arm.h>
25*344aa361SAndroid Build Coastguard Worker #include <arch/arm/mmu.h>
26*344aa361SAndroid Build Coastguard Worker #include <debug.h>
27*344aa361SAndroid Build Coastguard Worker #include <dev/interrupt/arm_gic.h>
28*344aa361SAndroid Build Coastguard Worker #include <dev/timer/arm_generic.h>
29*344aa361SAndroid Build Coastguard Worker #include <err.h>
30*344aa361SAndroid Build Coastguard Worker #include <kernel/thread.h>
31*344aa361SAndroid Build Coastguard Worker #include <kernel/vm.h>
32*344aa361SAndroid Build Coastguard Worker #include <lk/init.h>
33*344aa361SAndroid Build Coastguard Worker #include <platform.h>
34*344aa361SAndroid Build Coastguard Worker #include <platform/gic.h>
35*344aa361SAndroid Build Coastguard Worker #include <platform/interrupts.h>
36*344aa361SAndroid Build Coastguard Worker #include <platform/vexpress-a15.h>
37*344aa361SAndroid Build Coastguard Worker #include <reg.h>
38*344aa361SAndroid Build Coastguard Worker #include <string.h>
39*344aa361SAndroid Build Coastguard Worker #include "platform_p.h"
40*344aa361SAndroid Build Coastguard Worker
41*344aa361SAndroid Build Coastguard Worker #if WITH_LIB_KMAP
42*344aa361SAndroid Build Coastguard Worker #include <lib/kmap.h>
43*344aa361SAndroid Build Coastguard Worker #endif
44*344aa361SAndroid Build Coastguard Worker
45*344aa361SAndroid Build Coastguard Worker #if WITH_SMP
46*344aa361SAndroid Build Coastguard Worker void platform_secondary_entry(void);
47*344aa361SAndroid Build Coastguard Worker paddr_t platform_secondary_entry_paddr;
48*344aa361SAndroid Build Coastguard Worker #endif
49*344aa361SAndroid Build Coastguard Worker
50*344aa361SAndroid Build Coastguard Worker /* initial memory mappings. parsed by start.S */
51*344aa361SAndroid Build Coastguard Worker struct mmu_initial_mapping mmu_initial_mappings[] = {
52*344aa361SAndroid Build Coastguard Worker /* Mark next entry as dynamic as it might be updated
53*344aa361SAndroid Build Coastguard Worker by platform_reset code to specify actual size and
54*344aa361SAndroid Build Coastguard Worker location of RAM to use */
55*344aa361SAndroid Build Coastguard Worker {.phys = MEMBASE + KERNEL_LOAD_OFFSET,
56*344aa361SAndroid Build Coastguard Worker .virt = KERNEL_BASE + KERNEL_LOAD_OFFSET,
57*344aa361SAndroid Build Coastguard Worker .size = MEMSIZE,
58*344aa361SAndroid Build Coastguard Worker .flags = MMU_INITIAL_MAPPING_FLAG_DYNAMIC,
59*344aa361SAndroid Build Coastguard Worker .name = "ram"},
60*344aa361SAndroid Build Coastguard Worker
61*344aa361SAndroid Build Coastguard Worker {.phys = REGISTER_BANK_0_PADDR,
62*344aa361SAndroid Build Coastguard Worker .virt = REGISTER_BANK_0_VADDR,
63*344aa361SAndroid Build Coastguard Worker .size = 0x00100000,
64*344aa361SAndroid Build Coastguard Worker .flags = MMU_INITIAL_MAPPING_FLAG_DEVICE,
65*344aa361SAndroid Build Coastguard Worker .name = "bank-0"},
66*344aa361SAndroid Build Coastguard Worker
67*344aa361SAndroid Build Coastguard Worker {.phys = REGISTER_BANK_1_PADDR,
68*344aa361SAndroid Build Coastguard Worker .virt = REGISTER_BANK_1_VADDR,
69*344aa361SAndroid Build Coastguard Worker .size = 0x00100000,
70*344aa361SAndroid Build Coastguard Worker .flags = MMU_INITIAL_MAPPING_FLAG_DEVICE,
71*344aa361SAndroid Build Coastguard Worker .name = "bank-1"},
72*344aa361SAndroid Build Coastguard Worker
73*344aa361SAndroid Build Coastguard Worker {.phys = REGISTER_BANK_2_PADDR,
74*344aa361SAndroid Build Coastguard Worker .virt = REGISTER_BANK_2_VADDR,
75*344aa361SAndroid Build Coastguard Worker .size = 0x00100000,
76*344aa361SAndroid Build Coastguard Worker .flags = MMU_INITIAL_MAPPING_FLAG_DEVICE,
77*344aa361SAndroid Build Coastguard Worker .name = "bank-2"},
78*344aa361SAndroid Build Coastguard Worker
79*344aa361SAndroid Build Coastguard Worker /* null entry to terminate the list */
80*344aa361SAndroid Build Coastguard Worker {0}};
81*344aa361SAndroid Build Coastguard Worker
82*344aa361SAndroid Build Coastguard Worker static pmm_arena_t ram_arena = {.name = "ram",
83*344aa361SAndroid Build Coastguard Worker .base = MEMBASE + KERNEL_LOAD_OFFSET,
84*344aa361SAndroid Build Coastguard Worker .size = MEMSIZE,
85*344aa361SAndroid Build Coastguard Worker .flags = PMM_ARENA_FLAG_KMAP};
86*344aa361SAndroid Build Coastguard Worker
get_cpuid(void)87*344aa361SAndroid Build Coastguard Worker static uint32_t get_cpuid(void) {
88*344aa361SAndroid Build Coastguard Worker u_int cpuid;
89*344aa361SAndroid Build Coastguard Worker
90*344aa361SAndroid Build Coastguard Worker __asm__ volatile("mrc p15, 0, %0, c0, c0, 5 @ read MPIDR\n" : "=r"(cpuid));
91*344aa361SAndroid Build Coastguard Worker
92*344aa361SAndroid Build Coastguard Worker return cpuid & 0xF;
93*344aa361SAndroid Build Coastguard Worker }
94*344aa361SAndroid Build Coastguard Worker
platform_init_mmu_mappings(void)95*344aa361SAndroid Build Coastguard Worker void platform_init_mmu_mappings(void) {
96*344aa361SAndroid Build Coastguard Worker if (get_cpuid())
97*344aa361SAndroid Build Coastguard Worker return;
98*344aa361SAndroid Build Coastguard Worker
99*344aa361SAndroid Build Coastguard Worker /* go through mmu_initial_mapping to find dynamic entry
100*344aa361SAndroid Build Coastguard Worker * matching ram_arena (by name) and adjust it.
101*344aa361SAndroid Build Coastguard Worker */
102*344aa361SAndroid Build Coastguard Worker struct mmu_initial_mapping* m = mmu_initial_mappings;
103*344aa361SAndroid Build Coastguard Worker for (size_t i = 0; i < countof(mmu_initial_mappings); i++, m++) {
104*344aa361SAndroid Build Coastguard Worker if (!(m->flags & MMU_INITIAL_MAPPING_FLAG_DYNAMIC))
105*344aa361SAndroid Build Coastguard Worker continue;
106*344aa361SAndroid Build Coastguard Worker
107*344aa361SAndroid Build Coastguard Worker if (strcmp(m->name, ram_arena.name) == 0) {
108*344aa361SAndroid Build Coastguard Worker /* update ram_arena */
109*344aa361SAndroid Build Coastguard Worker ram_arena.base = m->phys;
110*344aa361SAndroid Build Coastguard Worker ram_arena.size = m->size;
111*344aa361SAndroid Build Coastguard Worker ram_arena.flags = PMM_ARENA_FLAG_KMAP;
112*344aa361SAndroid Build Coastguard Worker
113*344aa361SAndroid Build Coastguard Worker break;
114*344aa361SAndroid Build Coastguard Worker }
115*344aa361SAndroid Build Coastguard Worker }
116*344aa361SAndroid Build Coastguard Worker pmm_add_arena(&ram_arena);
117*344aa361SAndroid Build Coastguard Worker }
118*344aa361SAndroid Build Coastguard Worker
read_mpidr(void)119*344aa361SAndroid Build Coastguard Worker static uint32_t read_mpidr(void) {
120*344aa361SAndroid Build Coastguard Worker int mpidr;
121*344aa361SAndroid Build Coastguard Worker __asm__ volatile("mrc p15, 0, %0, c0, c0, 5" : "=r"(mpidr));
122*344aa361SAndroid Build Coastguard Worker return mpidr;
123*344aa361SAndroid Build Coastguard Worker }
124*344aa361SAndroid Build Coastguard Worker
platform_early_init(void)125*344aa361SAndroid Build Coastguard Worker void platform_early_init(void) {
126*344aa361SAndroid Build Coastguard Worker /* initialize the interrupt controller */
127*344aa361SAndroid Build Coastguard Worker arm_gic_init();
128*344aa361SAndroid Build Coastguard Worker
129*344aa361SAndroid Build Coastguard Worker /* initialize the timer block */
130*344aa361SAndroid Build Coastguard Worker arm_generic_timer_init(ARM_GENERIC_TIMER_INT, 0);
131*344aa361SAndroid Build Coastguard Worker
132*344aa361SAndroid Build Coastguard Worker #if WITH_SMP
133*344aa361SAndroid Build Coastguard Worker dprintf(INFO, "Booting secondary CPUs. Main CPU MPIDR = %x\n",
134*344aa361SAndroid Build Coastguard Worker read_mpidr());
135*344aa361SAndroid Build Coastguard Worker platform_secondary_entry_paddr = vaddr_to_paddr(platform_secondary_entry);
136*344aa361SAndroid Build Coastguard Worker writel(platform_secondary_entry_paddr, SECONDARY_BOOT_ADDR);
137*344aa361SAndroid Build Coastguard Worker arm_gic_sgi(0, ARM_GIC_SGI_FLAG_TARGET_FILTER_NOT_SENDER, 0);
138*344aa361SAndroid Build Coastguard Worker #endif
139*344aa361SAndroid Build Coastguard Worker }
140*344aa361SAndroid Build Coastguard Worker
141*344aa361SAndroid Build Coastguard Worker #if WITH_SMP
142*344aa361SAndroid Build Coastguard Worker
143*344aa361SAndroid Build Coastguard Worker #define GICC_CTLR (GICC_OFFSET + 0x0000)
144*344aa361SAndroid Build Coastguard Worker #define GICC_IAR (GICC_OFFSET + 0x000c)
145*344aa361SAndroid Build Coastguard Worker #define GICC_EOIR (GICC_OFFSET + 0x0010)
146*344aa361SAndroid Build Coastguard Worker
platform_secondary_init(uint level)147*344aa361SAndroid Build Coastguard Worker static void platform_secondary_init(uint level) {
148*344aa361SAndroid Build Coastguard Worker u_int val;
149*344aa361SAndroid Build Coastguard Worker dprintf(INFO, "Booted secondary CPU, MPIDR = %x\n", read_mpidr());
150*344aa361SAndroid Build Coastguard Worker val = *REG32(GICBASE(0) + GICC_IAR);
151*344aa361SAndroid Build Coastguard Worker if (val)
152*344aa361SAndroid Build Coastguard Worker dprintf(INFO, "bad interrupt number on secondary CPU: %x\n", val);
153*344aa361SAndroid Build Coastguard Worker *REG32(GICBASE(0) + GICC_EOIR) = val & 0x3ff;
154*344aa361SAndroid Build Coastguard Worker }
155*344aa361SAndroid Build Coastguard Worker
156*344aa361SAndroid Build Coastguard Worker LK_INIT_HOOK_FLAGS(vexpress_a15,
157*344aa361SAndroid Build Coastguard Worker platform_secondary_init,
158*344aa361SAndroid Build Coastguard Worker LK_INIT_LEVEL_PLATFORM,
159*344aa361SAndroid Build Coastguard Worker LK_INIT_FLAG_SECONDARY_CPUS);
160*344aa361SAndroid Build Coastguard Worker #endif
161*344aa361SAndroid Build Coastguard Worker
platform_init(void)162*344aa361SAndroid Build Coastguard Worker void platform_init(void) {}
163