1/* $NetBSD: locore.S,v 1.14 2003/04/20 16:21:40 thorpej Exp $ */ 2 3/*- 4 * Copyright (C) 1994-1997 Mark Brinicombe 5 * Copyright (C) 1994 Brini 6 * All rights reserved. 7 * 8 * Redistribution and use in source and binary forms, with or without 9 * modification, are permitted provided that the following conditions 10 * are met: 11 * 1. Redistributions of source code must retain the above copyright 12 * notice, this list of conditions and the following disclaimer. 13 * 2. Redistributions in binary form must reproduce the above copyright 14 * notice, this list of conditions and the following disclaimer in the 15 * documentation and/or other materials provided with the distribution. 16 * 3. All advertising materials mentioning features or use of this software 17 * must display the following acknowledgement: 18 * This product includes software developed by Brini. 19 * 4. The name of Brini may not be used to endorse or promote products 20 * derived from this software without specific prior written permission. 21 * 22 * THIS SOFTWARE IS PROVIDED BY BRINI ``AS IS'' AND ANY EXPRESS OR 23 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES 24 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. 25 * IN NO EVENT SHALL BRINI BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, 26 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, 27 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; 28 * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 29 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR 30 * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF 31 * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 32 * 33 */ 34 35#include "assym.s" 36#include <sys/syscall.h> 37#include <machine/asm.h> 38#include <machine/armreg.h> 39#include <machine/pte.h> 40__FBSDID("$FreeBSD$"); 41 42/* What size should this really be ? It is only used by initarm() */ 43#define INIT_ARM_STACK_SIZE 2048 44 45/* 46 * This is for kvm_mkdb, and should be the address of the beginning 47 * of the kernel text segment (not necessarily the same as kernbase). 48 */ 49 50 51#define CPWAIT_BRANCH \ 52 sub pc, pc, #4 53 54#define CPWAIT(tmp) \ 55 mrc p15, 0, tmp, c2, c0, 0 /* arbitrary read of CP15 */ ;\ 56 mov tmp, tmp /* wait for it to complete */ ;\ 57 CPWAIT_BRANCH /* branch to next insn */ 58 59 .text 60 .align 0 61.globl kernbase 62.set kernbase,KERNBASE 63.globl physaddr 64.set physaddr,PHYSADDR 65 66ENTRY_NP(btext) 67 68ASENTRY_NP(_start) 69 70/* 71 * Move metadata ptr to r12 (ip) 72 */ 73 74 mov ip, r0 75 76#if defined (FLASHADDR) && defined(LOADERRAMADDR) 77 /* Check if we're running from flash. */ 78 ldr r7, =FLASHADDR 79 /* 80 * If we're running with MMU disabled, test against the 81 * physical address instead. 82 */ 83 mrc p15, 0, r2, c1, c0, 0 84 ands r2, r2, #CPU_CONTROL_MMU_ENABLE 85 ldreq r8, =PHYSADDR 86 ldrne r8, =LOADERRAMADDR 87 cmp r7, r8 88 bls flash_lower 89 cmp r7, pc 90 bhi from_ram 91 b do_copy 92 93flash_lower: 94 cmp r8, pc 95 bls from_ram 96do_copy: 97 ldr r9, =KERNBASE 98 adr r1, _start 99 ldr r0, Lreal_start 100 ldr r2, Lend 101 sub r2, r2, r0 102 sub r0, r0, r9 103 add r0, r0, r8 104 mov r4, r0 105 bl memcpy 106 ldr r0, Lram_offset 107 add pc, r4, r0 108Lram_offset: .word from_ram-_C_LABEL(_start) 109from_ram: 110 nop 111#endif 112 adr r7, Lunmapped 113 bic r7, r7, #0xff000000 114 orr r7, r7, #PHYSADDR 115 116 117disable_mmu: 118 /* Disable MMU for a while */ 119 mrc p15, 0, r2, c1, c0, 0 120 bic r2, r2, #(CPU_CONTROL_MMU_ENABLE | CPU_CONTROL_DC_ENABLE |\ 121 CPU_CONTROL_WBUF_ENABLE) 122 bic r2, r2, #(CPU_CONTROL_IC_ENABLE) 123 bic r2, r2, #(CPU_CONTROL_BPRD_ENABLE) 124 mcr p15, 0, r2, c1, c0, 0 125 126 nop 127 nop 128 nop 129 mov pc, r7 130Lunmapped: 131#ifdef STARTUP_PAGETABLE_ADDR 132 /* build page table from scratch */ 133 ldr r0, Lstartup_pagetable 134 adr r4, mmu_init_table 135 b 3f 136 1372: 138 str r3, [r0, r2] 139 add r2, r2, #4 140 add r3, r3, #(L1_S_SIZE) 141 adds r1, r1, #-1 142 bhi 2b 1433: 144 ldmia r4!, {r1,r2,r3} /* # of sections, VA, PA|attr */ 145 cmp r1, #0 146 adrne r5, 2b 147 bicne r5, r5, #0xff000000 148 orrne r5, r5, #PHYSADDR 149 movne pc, r5 150 151 mcr p15, 0, r0, c2, c0, 0 /* Set TTB */ 152 mcr p15, 0, r0, c8, c7, 0 /* Flush TLB */ 153 154 /* Set the Domain Access register. Very important! */ 155 mov r0, #((DOMAIN_CLIENT << (PMAP_DOMAIN_KERNEL*2)) | DOMAIN_CLIENT) 156 mcr p15, 0, r0, c3, c0, 0 157 /* Enable MMU */ 158 mrc p15, 0, r0, c1, c0, 0 159 orr r0, r0, #CPU_CONTROL_MMU_ENABLE 160 mcr p15, 0, r0, c1, c0, 0 161 nop 162 nop 163 nop 164 CPWAIT(r0) 165 166#endif 167mmu_done: 168 nop 169 adr r1, .Lstart 170 ldmia r1, {r1, r2, sp} /* Set initial stack and */ 171 sub r2, r2, r1 /* get zero init data */ 172 mov r3, #0 173.L1: 174 str r3, [r1], #0x0004 /* get zero init data */ 175 subs r2, r2, #4 176 bgt .L1 177 ldr pc, .Lvirt_done 178 179virt_done: 180 mov r0, ip /* Load argument: metadata ptr */ 181 182 mov fp, #0 /* trace back starts here */ 183 bl _C_LABEL(initarm) /* Off we go */ 184 185 /* init arm will return the new stack pointer. */ 186 mov sp, r0 187 188 bl _C_LABEL(mi_startup) /* call mi_startup()! */ 189 190 adr r0, .Lmainreturned 191 b _C_LABEL(panic) 192 /* NOTREACHED */ 193#ifdef STARTUP_PAGETABLE_ADDR 194#define MMU_INIT(va,pa,n_sec,attr) \ 195 .word n_sec ; \ 196 .word 4*((va)>>L1_S_SHIFT) ; \ 197 .word (pa)|(attr) ; 198 199Lvirtaddr: 200 .word KERNVIRTADDR 201Lphysaddr: 202 .word KERNPHYSADDR 203Lreal_start: 204 .word _start 205Lend: 206 .word _edata 207Lstartup_pagetable: 208 .word STARTUP_PAGETABLE_ADDR 209mmu_init_table: 210 /* fill all table VA==PA */ 211 /* map SDRAM VA==PA, WT cacheable */ 212 MMU_INIT(PHYSADDR, PHYSADDR , 64, L1_TYPE_S|L1_S_C|L1_S_AP(AP_KRW)) 213 /* map VA 0xc0000000..0xc3ffffff to PA */ 214 MMU_INIT(KERNBASE, PHYSADDR, 64, L1_TYPE_S|L1_S_C|L1_S_AP(AP_KRW)) 215 216 .word 0 /* end of table */ 217#endif 218.Lstart: 219 .word _edata 220 .word _end 221 .word svcstk + INIT_ARM_STACK_SIZE 222 223#if defined(FLASHADDR) && defined(LOADERRAMADDR) 224.L_arm_memcpy: 225 .word _C_LABEL(_arm_memcpy) 226#endif 227 228.Lvirt_done: 229 .word virt_done 230.Lmainreturned: 231 .asciz "main() returned" 232 .align 0 233 234 .bss 235svcstk: 236 .space INIT_ARM_STACK_SIZE 237 238 .text 239 .align 0 240 241.Lcpufuncs: 242 .word _C_LABEL(cpufuncs) 243 244ENTRY_NP(cpu_halt) 245 mrs r2, cpsr 246 bic r2, r2, #(PSR_MODE) 247 orr r2, r2, #(PSR_SVC32_MODE) 248 orr r2, r2, #(I32_bit | F32_bit) 249 msr cpsr_all, r2 250 251 ldr r4, .Lcpu_reset_address 252 ldr r4, [r4] 253 254 ldr r0, .Lcpufuncs 255 mov lr, pc 256 ldr pc, [r0, #CF_IDCACHE_WBINV_ALL] 257 mov lr, pc 258 ldr pc, [r0, #CF_L2CACHE_WBINV_ALL] 259 260 /* 261 * Load the cpu_reset_needs_v4_MMU_disable flag to determine if it's 262 * necessary. 263 */ 264 265 ldr r1, .Lcpu_reset_needs_v4_MMU_disable 266 ldr r1, [r1] 267 cmp r1, #0 268 mov r2, #0 269 270 /* 271 * MMU & IDC off, 32 bit program & data space 272 * Hurl ourselves into the ROM 273 */ 274 mov r0, #(CPU_CONTROL_32BP_ENABLE | CPU_CONTROL_32BD_ENABLE) 275 mcr 15, 0, r0, c1, c0, 0 276 mcrne 15, 0, r2, c8, c7, 0 /* nail I+D TLB on ARMv4 and greater */ 277 mov pc, r4 278 279 /* 280 * _cpu_reset_address contains the address to branch to, to complete 281 * the cpu reset after turning the MMU off 282 * This variable is provided by the hardware specific code 283 */ 284.Lcpu_reset_address: 285 .word _C_LABEL(cpu_reset_address) 286 287 /* 288 * cpu_reset_needs_v4_MMU_disable contains a flag that signals if the 289 * v4 MMU disable instruction needs executing... it is an illegal instruction 290 * on f.e. ARM6/7 that locks up the computer in an endless illegal 291 * instruction / data-abort / reset loop. 292 */ 293.Lcpu_reset_needs_v4_MMU_disable: 294 .word _C_LABEL(cpu_reset_needs_v4_MMU_disable) 295 296 297/* 298 * setjump + longjmp 299 */ 300ENTRY(setjmp) 301 stmia r0, {r4-r14} 302 mov r0, #0x00000000 303 RET 304 305ENTRY(longjmp) 306 ldmia r0, {r4-r14} 307 mov r0, #0x00000001 308 RET 309 310 .data 311 .global _C_LABEL(esym) 312_C_LABEL(esym): .word _C_LABEL(end) 313 314ENTRY_NP(abort) 315 b _C_LABEL(abort) 316 317ENTRY_NP(sigcode) 318 mov r0, sp 319 swi SYS_sigreturn 320 321 /* Well if that failed we better exit quick ! */ 322 323 swi SYS_exit 324 b . - 8 325 326 .align 0 327 .global _C_LABEL(esigcode) 328 _C_LABEL(esigcode): 329 330 .data 331 .global szsigcode 332szsigcode: 333 .long esigcode-sigcode 334/* End of locore.S */ 335