mirror of
https://github.com/rust-embedded/rust-raspberrypi-OS-tutorials.git
synced 2024-11-11 07:10:59 +00:00
202 lines
5.2 KiB
ArmAsm
202 lines
5.2 KiB
ArmAsm
/*
|
|
* Copyright (C) 2018 bzt (bztsrc@github)
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person
|
|
* obtaining a copy of this software and associated documentation
|
|
* files (the "Software"), to deal in the Software without
|
|
* restriction, including without limitation the rights to use, copy,
|
|
* modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
* of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be
|
|
* included in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
|
* DEALINGS IN THE SOFTWARE.
|
|
*
|
|
*/
|
|
|
|
.section ".text.boot"
|
|
|
|
.global _start
|
|
|
|
_start:
|
|
// read cpu id, stop slave cores
|
|
mrs x1, mpidr_el1
|
|
and x1, x1, #3
|
|
cbz x1, 2f
|
|
// cpu id > 0, stop
|
|
1: wfe
|
|
b 1b
|
|
2: // cpu id == 0
|
|
|
|
// set stack before our code
|
|
ldr x1, =_start
|
|
|
|
// set up EL1
|
|
mrs x0, CurrentEL
|
|
and x0, x0, #12 // clear reserved bits
|
|
|
|
// running at EL3?
|
|
cmp x0, #12
|
|
bne 5f
|
|
// should never be executed, just for completeness
|
|
mov x2, #0x5b1
|
|
msr scr_el3, x2
|
|
mov x2, #0x3c9
|
|
msr spsr_el3, x2
|
|
adr x2, 5f
|
|
msr elr_el3, x2
|
|
eret
|
|
|
|
// running at EL2?
|
|
5: cmp x0, #4
|
|
beq 5f
|
|
msr sp_el1, x1
|
|
// enable CNTP for EL1
|
|
mrs x0, cnthctl_el2
|
|
orr x0, x0, #3
|
|
msr cnthctl_el2, x0
|
|
msr cntvoff_el2, xzr
|
|
// disable coprocessor traps
|
|
mov x0, #0x33FF
|
|
msr cptr_el2, x0
|
|
msr hstr_el2, xzr
|
|
mov x0, #(3 << 20)
|
|
msr cpacr_el1, x0
|
|
// enable AArch64 in EL1
|
|
mov x0, #(1 << 31) // AArch64
|
|
orr x0, x0, #(1 << 1) // SWIO hardwired on Pi3
|
|
msr hcr_el2, x0
|
|
mrs x0, hcr_el2
|
|
// Setup SCTLR access
|
|
mov x2, #0x0800
|
|
movk x2, #0x30d0, lsl #16
|
|
msr sctlr_el1, x2
|
|
// set up exception handlers
|
|
ldr x2, =_vectors
|
|
msr vbar_el1, x2
|
|
// change execution level to EL1
|
|
mov x2, #0x3c4
|
|
msr spsr_el2, x2
|
|
adr x2, 5f
|
|
msr elr_el2, x2
|
|
// clear EL1 system registers
|
|
msr elr_el1, xzr
|
|
msr far_el1, xzr
|
|
eret
|
|
|
|
5: mov sp, x1
|
|
|
|
// clear bss
|
|
ldr x1, =__bss_start
|
|
ldr w2, =__bss_size
|
|
3: cbz w2, 4f
|
|
str xzr, [x1], #8
|
|
sub w2, w2, #1
|
|
cbnz w2, 3b
|
|
|
|
// jump to C code, should not return
|
|
4: bl main
|
|
// for failsafe, halt this core too
|
|
b 1b
|
|
|
|
// save registers before we call any C code
|
|
dbg_saveregs:
|
|
str x0, [sp, #-16]! // push x0
|
|
ldr x0, =dbg_regs+8
|
|
str x1, [x0], #8 // dbg_regs[1]=x1
|
|
ldr x1, [sp, #16] // pop x1
|
|
str x1, [x0, #-16]! // dbg_regs[0]=x1 (x0)
|
|
add x0, x0, #16
|
|
str x2, [x0], #8 // dbg_regs[2]=x2
|
|
str x3, [x0], #8 // ...etc.
|
|
str x4, [x0], #8
|
|
str x5, [x0], #8
|
|
str x6, [x0], #8
|
|
str x7, [x0], #8
|
|
str x8, [x0], #8
|
|
str x9, [x0], #8
|
|
str x10, [x0], #8
|
|
str x11, [x0], #8
|
|
str x12, [x0], #8
|
|
str x13, [x0], #8
|
|
str x14, [x0], #8
|
|
str x15, [x0], #8
|
|
str x16, [x0], #8
|
|
str x17, [x0], #8
|
|
str x18, [x0], #8
|
|
str x19, [x0], #8
|
|
str x20, [x0], #8
|
|
str x21, [x0], #8
|
|
str x22, [x0], #8
|
|
str x23, [x0], #8
|
|
str x24, [x0], #8
|
|
str x25, [x0], #8
|
|
str x26, [x0], #8
|
|
str x27, [x0], #8
|
|
str x28, [x0], #8
|
|
str x29, [x0], #8
|
|
ldr x1, [sp, #16] // pop x30
|
|
str x1, [x0], #8
|
|
// also read and store some system registers
|
|
mrs x1, elr_el1
|
|
str x1, [x0], #8
|
|
mrs x1, spsr_el1
|
|
str x1, [x0], #8
|
|
mrs x1, esr_el1
|
|
str x1, [x0], #8
|
|
mrs x1, far_el1
|
|
str x1, [x0], #8
|
|
mrs x1, sctlr_el1
|
|
str x1, [x0], #8
|
|
mrs x1, tcr_el1
|
|
str x1, [x0], #8
|
|
ret
|
|
|
|
// important, code has to be properly aligned
|
|
.align 11
|
|
_vectors:
|
|
// synchronous
|
|
.align 7
|
|
str x30, [sp, #-16]! // push x30
|
|
bl dbg_saveregs
|
|
mov x0, #0
|
|
bl dbg_decodeexc
|
|
bl dbg_main
|
|
eret
|
|
|
|
// IRQ
|
|
.align 7
|
|
str x30, [sp, #-16]! // push x30
|
|
bl dbg_saveregs
|
|
mov x0, #1
|
|
bl dbg_decodeexc
|
|
bl dbg_main
|
|
eret
|
|
|
|
// FIQ
|
|
.align 7
|
|
str x30, [sp, #-16]! // push x30
|
|
bl dbg_saveregs
|
|
mov x0, #2
|
|
bl dbg_decodeexc
|
|
bl dbg_main
|
|
eret
|
|
|
|
// SError
|
|
.align 7
|
|
str x30, [sp, #-16]! // push x30
|
|
bl dbg_saveregs
|
|
mov x0, #3
|
|
bl dbg_decodeexc
|
|
bl dbg_main
|
|
eret
|