ref: f899b2fe3c091ccab33673f9fea9d1d99c0c2940
dir: /sys/src/9/omap/rebootcode.s/
/* * omap3530 reboot code * * must fit in 11K to avoid stepping on PTEs; see mem.h. * * R11 is used by the loader as a temporary, so avoid it. */ #include "arm.s" /* * Turn off MMU, then copy the new kernel to its correct location * in physical memory. Then jump to the start of the kernel. */ /* main(PADDR(entry), PADDR(code), size); */ TEXT main(SB), 1, $-4 MOVW $setR12(SB), R12 MOVW R0, p1+0(FP) /* destination, passed in R0 */ MOVW CPSR, R0 ORR $(PsrDirq|PsrDfiq), R0 MOVW R0, CPSR /* splhi */ BARRIERS WAVE('R') MRC CpSC, 0, R1, C(CpCONTROL), C(0), CpAuxctl BIC $CpACasa, R1 /* no speculative I access forwarding to mem */ /* slow down */ ORR $(CpACcachenopipe|CpACcp15serial|CpACcp15waitidle|CpACcp15pipeflush), R1 MCR CpSC, 0, R1, C(CpCONTROL), C(0), CpAuxctl BARRIERS BL cachesoff(SB) /* now back in 29- or 26-bit addressing, mainly for SB */ /* double mapping of PHYSDRAM & KZERO now in effect */ /* * turn the MMU off */ WAVE('e') /* first switch to PHYSDRAM-based addresses */ DMB MOVW $KSEGM, R7 /* clear segment bits */ MOVW $PHYSDRAM, R0 /* set dram base bits */ BIC R7, R12 /* adjust SB */ ORR R0, R12 BL _r15warp(SB) /* don't care about saving R14; we're not returning */ /* * now running in PHYSDRAM segment, not KZERO. */ WAVE('b') SUB $12, SP /* paranoia */ BL cacheuwbinv(SB) ADD $12, SP /* paranoia */ /* invalidate mmu mappings */ MOVW $KZERO, R0 /* some valid virtual address */ MCR CpSC, 0, R0, C(CpTLB), C(CpTLBinvu), CpTLBinv BARRIERS WAVE('o') MRC CpSC, 0, R0, C(CpCONTROL), C(0) BIC $(CpCmmu|CpCdcache|CpCicache), R0 MCR CpSC, 0, R0, C(CpCONTROL), C(0) /* mmu off */ BARRIERS WAVE('o') /* copy in arguments from stack frame before moving stack */ MOVW p2+4(FP), R4 /* phys source */ MOVW n+8(FP), R5 /* byte count */ MOVW p1+0(FP), R6 /* phys destination */ /* set up a new stack for local vars and memmove args */ MOVW R6, SP /* tiny trampoline stack */ SUB $(0x20 + 4), SP /* back up before a.out header */ // MOVW R14, -48(SP) /* store return addr */ SUB $48, SP /* allocate stack frame */ MOVW R5, 40(SP) /* save count */ MOVW R6, 44(SP) /* save dest/entry */ DELAY(printloop2, 2) WAVE('t') MOVW 40(SP), R5 /* restore count */ MOVW 44(SP), R6 /* restore dest/entry */ MOVW R6, 0(SP) /* normally saved LR goes here */ MOVW R6, 4(SP) /* push dest */ MOVW R6, R0 MOVW R4, 8(SP) /* push src */ MOVW R5, 12(SP) /* push size */ BL memmove(SB) WAVE('-') /* * flush caches */ BL cacheuwbinv(SB) WAVE('>') DELAY(printloopret, 1) WAVE('\r') DELAY(printloopnl, 1) WAVE('\n') /* * jump to kernel entry point. Note the true kernel entry point is * the virtual address KZERO|R6, but this must wait until * the MMU is enabled by the kernel in l.s */ MOVW 44(SP), R6 /* restore R6 (dest/entry) */ ORR R6, R6 /* NOP: avoid link bug */ B (R6) WAVE('?') B 0(PC) /* * turn the caches off, double map PHYSDRAM & KZERO, invalidate TLBs, revert * to tiny addresses. upon return, it will be safe to turn off the mmu. */ TEXT cachesoff(SB), 1, $-4 MOVM.DB.W [R14,R1-R10], (R13) /* save regs on stack */ MOVW CPSR, R0 ORR $(PsrDirq|PsrDfiq), R0 MOVW R0, CPSR BARRIERS SUB $12, SP /* paranoia */ BL cacheuwbinv(SB) ADD $12, SP /* paranoia */ MRC CpSC, 0, R0, C(CpCONTROL), C(0) BIC $(CpCicache|CpCdcache), R0 MCR CpSC, 0, R0, C(CpCONTROL), C(0) /* caches off */ BARRIERS /* * caches are off */ /* invalidate stale TLBs before changing them */ MOVW $KZERO, R0 /* some valid virtual address */ MCR CpSC, 0, R0, C(CpTLB), C(CpTLBinvu), CpTLBinv BARRIERS /* redo double map of PHYSDRAM, KZERO */ MOVW $PHYSDRAM, R3 CMP $KZERO, R3 BEQ noun2map MOVW $(L1+L1X(PHYSDRAM)), R4 /* address of PHYSDRAM's PTE */ MOVW $PTEDRAM, R2 /* PTE bits */ MOVW $DOUBLEMAPMBS, R5 _ptrdbl: ORR R3, R2, R1 /* first identity-map 0 to 0, etc. */ MOVW R1, (R4) ADD $4, R4 /* bump PTE address */ ADD $MiB, R3 /* bump pa */ SUB.S $1, R5 BNE _ptrdbl noun2map: /* * flush stale TLB entries */ BARRIERS MOVW $KZERO, R0 /* some valid virtual address */ MCR CpSC, 0, R0, C(CpTLB), C(CpTLBinvu), CpTLBinv BARRIERS /* switch back to PHYSDRAM addressing, mainly for SB */ MOVW $KSEGM, R7 /* clear segment bits */ MOVW $PHYSDRAM, R0 /* set dram base bits */ BIC R7, R12 /* adjust SB */ ORR R0, R12 BIC R7, SP ORR R0, SP MOVM.IA.W (R13), [R14,R1-R10] /* restore regs from stack */ MOVW $KSEGM, R0 /* clear segment bits */ BIC R0, R14 /* adjust link */ MOVW $PHYSDRAM, R0 /* set dram base bits */ ORR R0, R14 RET TEXT _r15warp(SB), 1, $-4 BIC R7, R14 /* link */ ORR R0, R14 BIC R7, R13 /* SP */ ORR R0, R13 RET TEXT panic(SB), 1, $-4 /* stub */ WAVE('?') RET TEXT pczeroseg(SB), 1, $-4 /* stub */ RET #include "cache.v7.s"