xref: /aosp_15_r20/trusty/kernel/platform/vexpress-a15/platform.c (revision 344aa361028b423587d4ef3fa52a23d194628137)
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