mirror of
https://github.com/Atmosphere-NX/Atmosphere
synced 2024-11-14 17:16:36 +00:00
611 lines
20 KiB
ArmAsm
611 lines
20 KiB
ArmAsm
|
/*
|
||
|
* Copyright (c) 2018-2020 Atmosphère-NX
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify it
|
||
|
* under the terms and conditions of the GNU General Public License,
|
||
|
* version 2, as published by the Free Software Foundation.
|
||
|
*
|
||
|
* This program is distributed in the hope it will be useful, but WITHOUT
|
||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
|
||
|
* more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||
|
*/
|
||
|
|
||
|
/* ams::kern::arm64::EL1IrqExceptionHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6422EL1IrqExceptionHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6422EL1IrqExceptionHandlerEv
|
||
|
.type _ZN3ams4kern5arm6422EL1IrqExceptionHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6422EL1IrqExceptionHandlerEv:
|
||
|
/* Save registers that need saving. */
|
||
|
sub sp, sp, #(8 * 24)
|
||
|
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x30, [sp, #(8 * 22)]
|
||
|
|
||
|
mrs x19, sp_el0
|
||
|
mrs x20, elr_el1
|
||
|
mrs x21, spsr_el1
|
||
|
mov w21, w21
|
||
|
|
||
|
/* Invoke KInterruptManager::HandleInterrupt(bool user_mode). */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, #0
|
||
|
bl _ZN3ams4kern5arm6417KInterruptManager15HandleInterruptEb
|
||
|
|
||
|
/* Restore registers that we saved. */
|
||
|
msr sp_el0, x19
|
||
|
msr elr_el1, x20
|
||
|
msr spsr_el1, x21
|
||
|
|
||
|
ldp x0, x1, [sp, #(8 * 0)]
|
||
|
ldp x2, x3, [sp, #(8 * 2)]
|
||
|
ldp x4, x5, [sp, #(8 * 4)]
|
||
|
ldp x6, x7, [sp, #(8 * 6)]
|
||
|
ldp x8, x9, [sp, #(8 * 8)]
|
||
|
ldp x10, x11, [sp, #(8 * 10)]
|
||
|
ldp x12, x13, [sp, #(8 * 12)]
|
||
|
ldp x14, x15, [sp, #(8 * 14)]
|
||
|
ldp x16, x17, [sp, #(8 * 16)]
|
||
|
ldp x18, x19, [sp, #(8 * 18)]
|
||
|
ldp x20, x21, [sp, #(8 * 20)]
|
||
|
ldp x22, x30, [sp, #(8 * 22)]
|
||
|
|
||
|
add sp, sp, #(8 * 24)
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
/* ams::kern::arm64::EL0IrqExceptionHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6422EL0IrqExceptionHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6422EL0IrqExceptionHandlerEv
|
||
|
.type _ZN3ams4kern5arm6422EL0IrqExceptionHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6422EL0IrqExceptionHandlerEv:
|
||
|
/* Save registers that need saving. */
|
||
|
sub sp, sp, #(8 * 36)
|
||
|
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x23, [sp, #(8 * 22)]
|
||
|
stp x24, x25, [sp, #(8 * 24)]
|
||
|
stp x26, x27, [sp, #(8 * 26)]
|
||
|
stp x28, x29, [sp, #(8 * 28)]
|
||
|
|
||
|
mrs x20, sp_el0
|
||
|
mrs x21, elr_el1
|
||
|
mrs x22, spsr_el1
|
||
|
mrs x23, tpidr_el0
|
||
|
mov w22, w22
|
||
|
stp x30, x20, [sp, #(8 * 30)]
|
||
|
stp x21, x22, [sp, #(8 * 32)]
|
||
|
str x23, [sp, #(8 * 34)]
|
||
|
|
||
|
/* Invoke KInterruptManager::HandleInterrupt(bool user_mode). */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, #1
|
||
|
bl _ZN3ams4kern5arm6417KInterruptManager15HandleInterruptEb
|
||
|
|
||
|
/* Restore state from the context. */
|
||
|
ldp x30, x20, [sp, #(8 * 30)]
|
||
|
ldp x21, x22, [sp, #(8 * 32)]
|
||
|
ldr x23, [sp, #(8 * 34)]
|
||
|
msr sp_el0, x20
|
||
|
msr elr_el1, x21
|
||
|
msr spsr_el1, x22
|
||
|
msr tpidr_el0, x23
|
||
|
ldp x0, x1, [sp, #(8 * 0)]
|
||
|
ldp x2, x3, [sp, #(8 * 2)]
|
||
|
ldp x4, x5, [sp, #(8 * 4)]
|
||
|
ldp x6, x7, [sp, #(8 * 6)]
|
||
|
ldp x8, x9, [sp, #(8 * 8)]
|
||
|
ldp x10, x11, [sp, #(8 * 10)]
|
||
|
ldp x12, x13, [sp, #(8 * 12)]
|
||
|
ldp x14, x15, [sp, #(8 * 14)]
|
||
|
ldp x16, x17, [sp, #(8 * 16)]
|
||
|
ldp x18, x19, [sp, #(8 * 18)]
|
||
|
ldp x20, x21, [sp, #(8 * 20)]
|
||
|
ldp x22, x23, [sp, #(8 * 22)]
|
||
|
ldp x24, x25, [sp, #(8 * 24)]
|
||
|
ldp x26, x27, [sp, #(8 * 26)]
|
||
|
ldp x28, x29, [sp, #(8 * 28)]
|
||
|
add sp, sp, #0x120
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
/* ams::kern::arm64::EL0SynchronousExceptionHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6430EL0SynchronousExceptionHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6430EL0SynchronousExceptionHandlerEv
|
||
|
.type _ZN3ams4kern5arm6430EL0SynchronousExceptionHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6430EL0SynchronousExceptionHandlerEv:
|
||
|
/* Save x16 and x17, so that we can use them as scratch. */
|
||
|
stp x16, x17, [sp, #-16]!
|
||
|
|
||
|
/* Get and parse the exception syndrome register. */
|
||
|
mrs x16, esr_el1
|
||
|
lsr x17, x16, #0x1a
|
||
|
|
||
|
/* Is this an aarch32 SVC? */
|
||
|
cmp x17, #0x11
|
||
|
b.eq 2f
|
||
|
|
||
|
/* Is this an aarch64 SVC? */
|
||
|
cmp x17, #0x15
|
||
|
b.eq 3f
|
||
|
|
||
|
/* Is this an FPU error? */
|
||
|
cmp x17, #0x7
|
||
|
b.eq 4f
|
||
|
|
||
|
/* Is this an instruction abort? */
|
||
|
cmp x17, #0x21
|
||
|
b.eq 5f
|
||
|
|
||
|
/* Is this a data abort? */
|
||
|
cmp x17, #0x25
|
||
|
b.eq 5f
|
||
|
|
||
|
1: /* The exception is not a data abort or instruction abort caused by a TLB conflict. */
|
||
|
/* It is also not an SVC or an FPU exception. Handle it generically! */
|
||
|
|
||
|
/* Restore x16 and x17. */
|
||
|
ldp x16, x17, [sp], 16
|
||
|
|
||
|
/* Create a KExceptionContext to pass to HandleException. */
|
||
|
sub sp, sp, #0x120
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x23, [sp, #(8 * 22)]
|
||
|
stp x24, x25, [sp, #(8 * 24)]
|
||
|
stp x26, x27, [sp, #(8 * 26)]
|
||
|
stp x28, x29, [sp, #(8 * 28)]
|
||
|
mrs x20, sp_el0
|
||
|
mrs x21, elr_el1
|
||
|
mrs x22, spsr_el1
|
||
|
mrs x23, tpidr_el0
|
||
|
mov w22, w22
|
||
|
stp x30, x20, [sp, #(8 * 30)]
|
||
|
stp x21, x22, [sp, #(8 * 32)]
|
||
|
str x23, [sp, #(8 * 34)]
|
||
|
|
||
|
/* Call ams::kern::arm64::HandleException(ams::kern::arm64::KExceptionContext *) */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, sp
|
||
|
bl _ZN3ams4kern5arm6415HandleExceptionEPNS1_17KExceptionContextE
|
||
|
|
||
|
/* Restore state from the context. */
|
||
|
ldp x30, x20, [sp, #(8 * 30)]
|
||
|
ldp x21, x22, [sp, #(8 * 32)]
|
||
|
ldr x23, [sp, #(8 * 34)]
|
||
|
msr sp_el0, x20
|
||
|
msr elr_el1, x21
|
||
|
msr spsr_el1, x22
|
||
|
msr tpidr_el0, x23
|
||
|
ldp x0, x1, [sp, #(8 * 0)]
|
||
|
ldp x2, x3, [sp, #(8 * 2)]
|
||
|
ldp x4, x5, [sp, #(8 * 4)]
|
||
|
ldp x6, x7, [sp, #(8 * 6)]
|
||
|
ldp x8, x9, [sp, #(8 * 8)]
|
||
|
ldp x10, x11, [sp, #(8 * 10)]
|
||
|
ldp x12, x13, [sp, #(8 * 12)]
|
||
|
ldp x14, x15, [sp, #(8 * 14)]
|
||
|
ldp x16, x17, [sp, #(8 * 16)]
|
||
|
ldp x18, x19, [sp, #(8 * 18)]
|
||
|
ldp x20, x21, [sp, #(8 * 20)]
|
||
|
ldp x22, x23, [sp, #(8 * 22)]
|
||
|
ldp x24, x25, [sp, #(8 * 24)]
|
||
|
ldp x26, x27, [sp, #(8 * 26)]
|
||
|
ldp x28, x29, [sp, #(8 * 28)]
|
||
|
add sp, sp, #0x120
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
2: /* SVC from aarch32. */
|
||
|
ldp x16, x17, [sp], 16
|
||
|
b _ZN3ams4kern5arm6412SvcHandler32Ev
|
||
|
|
||
|
3: /* SVC from aarch64. */
|
||
|
ldp x16, x17, [sp], 16
|
||
|
b _ZN3ams4kern5arm6412SvcHandler64Ev
|
||
|
|
||
|
4: /* FPU exception. */
|
||
|
ldp x16, x17, [sp], 16
|
||
|
b _ZN3ams4kern5arm6425FpuAccessExceptionHandlerEv
|
||
|
|
||
|
5: /* Check if there's a TLB conflict that caused the abort. */
|
||
|
and x17, x16, #0x3F
|
||
|
cmp x17, #0x30
|
||
|
b.ne 1b
|
||
|
|
||
|
/* Get the ASID in x17. */
|
||
|
mrs x17, ttbr0_el1
|
||
|
and x17, x17, #(0xFFFF << 48)
|
||
|
|
||
|
/* Check if FAR is valid by examining the FnV bit. */
|
||
|
tbnz x16, #10, 6f
|
||
|
|
||
|
/* FAR is valid, so we can invalidate the address it holds. */
|
||
|
mrs x16, far_el1
|
||
|
lsr x16, x16, #12
|
||
|
orr x17, x16, x17
|
||
|
tlbi vaae1, x17
|
||
|
b 7f
|
||
|
|
||
|
6: /* There's a TLB conflict and FAR isn't valid. */
|
||
|
/* Invalidate the entire TLB. */
|
||
|
tlbi vmalle1
|
||
|
|
||
|
7: /* Return from a TLB conflict. */
|
||
|
/* Ensure instruction consistency. */
|
||
|
dsb ish
|
||
|
isb
|
||
|
|
||
|
/* Restore x16 and x17. */
|
||
|
ldp x16, x17, [sp], 16
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
|
||
|
/* ams::kern::arm64::EL1SynchronousExceptionHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6430EL1SynchronousExceptionHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6430EL1SynchronousExceptionHandlerEv
|
||
|
.type _ZN3ams4kern5arm6430EL1SynchronousExceptionHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6430EL1SynchronousExceptionHandlerEv:
|
||
|
/* Nintendo uses the "unused" virtual timer compare value as a scratch register. */
|
||
|
msr cntv_cval_el0, x0
|
||
|
|
||
|
/* Get and parse the exception syndrome register. */
|
||
|
mrs x0, esr_el1
|
||
|
lsr x0, x0, #0x1a
|
||
|
|
||
|
/* Is this an instruction abort? */
|
||
|
cmp x0, #0x21
|
||
|
b.eq 5f
|
||
|
|
||
|
/* Is this a data abort? */
|
||
|
cmp x0, #0x25
|
||
|
b.eq 5f
|
||
|
|
||
|
1: /* The exception is not a data abort or instruction abort caused by a TLB conflict. */
|
||
|
/* Load the CoreLocalContext into x0. */
|
||
|
mrs x0, tpidr_el1
|
||
|
cbz x0, 2f
|
||
|
|
||
|
/* Load the exception stack top from the context. */
|
||
|
ldr x0, [x0, #0x28]
|
||
|
|
||
|
/* Setup the stack for a generic exception handle */
|
||
|
sub x0, x0, #0x20
|
||
|
str x1, [x0, #16]
|
||
|
mov x1, sp
|
||
|
str x1, [x0]
|
||
|
mov sp, x0
|
||
|
ldr x1, [x0, #16]
|
||
|
mrs x0, cntv_cval_el0
|
||
|
str x0, [sp, #8]
|
||
|
|
||
|
/* Check again if this is a data abort from EL1. */
|
||
|
mrs x0, esr_el1
|
||
|
lsr x1, x0, #0x1a
|
||
|
cmp x1, #0x25
|
||
|
b.ne 3f
|
||
|
|
||
|
/* Data abort. Check if it was from trying to access userspace memory. */
|
||
|
mrs x1, elr_el1
|
||
|
adr x0, _ZN3ams4kern5arm6438UserspaceMemoryAccessFunctionAreaBeginEv
|
||
|
cmp x1, x0
|
||
|
b.lo 3f
|
||
|
adr x0, _ZN3ams4kern5arm6436UserspaceMemoryAccessFunctionAreaEndEv
|
||
|
cmp x1, x0
|
||
|
b.hs 3f
|
||
|
|
||
|
/* We aborted trying to access userspace memory. */
|
||
|
/* All functions that access user memory return a boolean for whether they succeeded. */
|
||
|
/* With that in mind, we can simply restore the stack pointer and return false directly. */
|
||
|
ldr x0, [sp]
|
||
|
mov sp, x0
|
||
|
|
||
|
/* Return false. */
|
||
|
mov x0, #0x0
|
||
|
msr elr_el1, x30
|
||
|
eret
|
||
|
|
||
|
2: /* The CoreLocalContext is nullptr. */
|
||
|
/* Setup the stack for a generic exception handle. */
|
||
|
/* NOTE: Nintendo does not restore X0 here, and thus saves nullptr. */
|
||
|
/* This is probably not their intention, so we'll fix it. */
|
||
|
/* NOTE: Nintendo also does not really save SP correctly, and so we */
|
||
|
/* will also fix that. */
|
||
|
mov x0, sp
|
||
|
sub x0, x0, #0x20
|
||
|
str x1, [x0, #16]
|
||
|
mov x1, sp
|
||
|
str x1, [x0]
|
||
|
mov sp, x0
|
||
|
mrs x0, cntv_cval_el0
|
||
|
str x0, [sp, #8]
|
||
|
|
||
|
3: /* The exception wasn't an triggered by copying memory from userspace. */
|
||
|
ldr x0, [sp, #8]
|
||
|
ldr x1, [sp, #16]
|
||
|
|
||
|
/* Create a KExceptionContext to pass to HandleException. */
|
||
|
sub sp, sp, #0x120
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x23, [sp, #(8 * 22)]
|
||
|
stp x24, x25, [sp, #(8 * 24)]
|
||
|
stp x26, x27, [sp, #(8 * 26)]
|
||
|
stp x28, x29, [sp, #(8 * 28)]
|
||
|
mrs x20, sp_el0
|
||
|
mrs x21, elr_el1
|
||
|
mrs x22, spsr_el1
|
||
|
mrs x23, tpidr_el0
|
||
|
mov w22, w22
|
||
|
stp x30, x20, [sp, #(8 * 30)]
|
||
|
stp x21, x22, [sp, #(8 * 32)]
|
||
|
str x23, [sp, #(8 * 34)]
|
||
|
|
||
|
/* Call ams::kern::arm64::HandleException(ams::kern::arm64::KExceptionContext *) */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, sp
|
||
|
bl _ZN3ams4kern5arm6415HandleExceptionEPNS1_17KExceptionContextE
|
||
|
|
||
|
4: /* HandleException should never return. The best we can do is infinite loop. */
|
||
|
b 4b
|
||
|
|
||
|
5: /* Check if there's a TLB conflict that caused the abort. */
|
||
|
/* NOTE: There is a Nintendo bug in this code that we correct. */
|
||
|
/* Nintendo compares the low 6 bits of x0 without restoring the value. */
|
||
|
/* They intend to check the DFSC/IFSC bits of esr_el1, but because they */
|
||
|
/* shifted esr earlier, the check is invalid and always fails. */
|
||
|
mrs x0, esr_el1
|
||
|
and x0, x0, #0x3F
|
||
|
cmp x0, #0x30
|
||
|
b.ne 1b
|
||
|
|
||
|
/* Check if FAR is valid by examining the FnV bit. */
|
||
|
/* NOTE: Nintendo again has a bug here, the same as above. */
|
||
|
/* They do not refresh the value of x0, and again compare with */
|
||
|
/* the relevant bit already masked out of x0. */
|
||
|
mrs x0, esr_el1
|
||
|
tbnz x0, #10, 6f
|
||
|
|
||
|
/* FAR is valid, so we can invalidate the address it holds. */
|
||
|
mrs x0, far_el1
|
||
|
lsr x0, x0, #12
|
||
|
tlbi vaae1, x0
|
||
|
b 7f
|
||
|
|
||
|
6: /* There's a TLB conflict and FAR isn't valid. */
|
||
|
/* Invalidate the entire TLB. */
|
||
|
tlbi vmalle1
|
||
|
|
||
|
7: /* Return from a TLB conflict. */
|
||
|
/* Ensure instruction consistency. */
|
||
|
dsb ish
|
||
|
isb
|
||
|
|
||
|
/* Restore x0 from scratch. */
|
||
|
mrs x0, cntv_cval_el0
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
|
||
|
/* ams::kern::arm64::FpuAccessExceptionHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6425FpuAccessExceptionHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6425FpuAccessExceptionHandlerEv
|
||
|
.type _ZN3ams4kern5arm6425FpuAccessExceptionHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6425FpuAccessExceptionHandlerEv:
|
||
|
/* Save registers that need saving. */
|
||
|
sub sp, sp, #(8 * 24)
|
||
|
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x30, [sp, #(8 * 22)]
|
||
|
|
||
|
mrs x18, tpidr_el1
|
||
|
mrs x19, sp_el0
|
||
|
mrs x20, elr_el1
|
||
|
mrs x21, spsr_el1
|
||
|
|
||
|
/* Invoke the FPU context switch handler. */
|
||
|
bl _ZN3ams4kern5arm6423FpuContextSwitchHandlerEv
|
||
|
|
||
|
/* Restore registers that we saved. */
|
||
|
msr sp_el0, x19
|
||
|
msr elr_el1, x20
|
||
|
msr spsr_el1, x21
|
||
|
|
||
|
ldp x0, x1, [sp, #(8 * 0)]
|
||
|
ldp x2, x3, [sp, #(8 * 2)]
|
||
|
ldp x4, x5, [sp, #(8 * 4)]
|
||
|
ldp x6, x7, [sp, #(8 * 6)]
|
||
|
ldp x8, x9, [sp, #(8 * 8)]
|
||
|
ldp x10, x11, [sp, #(8 * 10)]
|
||
|
ldp x12, x13, [sp, #(8 * 12)]
|
||
|
ldp x14, x15, [sp, #(8 * 14)]
|
||
|
ldp x16, x17, [sp, #(8 * 16)]
|
||
|
ldp x18, x19, [sp, #(8 * 18)]
|
||
|
ldp x20, x21, [sp, #(8 * 20)]
|
||
|
ldp x22, x30, [sp, #(8 * 22)]
|
||
|
|
||
|
add sp, sp, #(8 * 24)
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
/* ams::kern::arm64::EL1SystemErrorHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6421EL1SystemErrorHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6421EL1SystemErrorHandlerEv
|
||
|
.type _ZN3ams4kern5arm6421EL1SystemErrorHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6421EL1SystemErrorHandlerEv:
|
||
|
/* Nintendo uses the "unused" virtual timer compare value as a scratch register. */
|
||
|
msr cntv_cval_el0, x0
|
||
|
|
||
|
/* Load the exception stack top from the context. */
|
||
|
ldr x0, [x0, #0x28]
|
||
|
|
||
|
/* Setup the stack for a generic exception handle */
|
||
|
sub x0, x0, #0x20
|
||
|
str x1, [x0, #16]
|
||
|
mov x1, sp
|
||
|
str x1, [x0]
|
||
|
mov sp, x0
|
||
|
ldr x1, [x0, #16]
|
||
|
mrs x0, cntv_cval_el0
|
||
|
str x0, [sp, #8]
|
||
|
|
||
|
/* Create a KExceptionContext to pass to HandleException. */
|
||
|
sub sp, sp, #0x120
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x23, [sp, #(8 * 22)]
|
||
|
stp x24, x25, [sp, #(8 * 24)]
|
||
|
stp x26, x27, [sp, #(8 * 26)]
|
||
|
stp x28, x29, [sp, #(8 * 28)]
|
||
|
mrs x20, sp_el0
|
||
|
mrs x21, elr_el1
|
||
|
mrs x22, spsr_el1
|
||
|
mrs x23, tpidr_el0
|
||
|
mov w22, w22
|
||
|
stp x30, x20, [sp, #(8 * 30)]
|
||
|
stp x21, x22, [sp, #(8 * 32)]
|
||
|
str x23, [sp, #(8 * 34)]
|
||
|
|
||
|
/* Invoke ams::kern::arm64::HandleException(ams::kern::arm64::KExceptionContext *). */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, sp
|
||
|
bl _ZN3ams4kern5arm6415HandleExceptionEPNS1_17KExceptionContextE
|
||
|
|
||
|
1: /* HandleException should never return. The best we can do is infinite loop. */
|
||
|
b 1b
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|
||
|
/* ams::kern::arm64::EL0SystemErrorHandler() */
|
||
|
.section .text._ZN3ams4kern5arm6421EL0SystemErrorHandlerEv, "ax", %progbits
|
||
|
.global _ZN3ams4kern5arm6421EL0SystemErrorHandlerEv
|
||
|
.type _ZN3ams4kern5arm6421EL0SystemErrorHandlerEv, %function
|
||
|
_ZN3ams4kern5arm6421EL0SystemErrorHandlerEv:
|
||
|
/* Create a KExceptionContext to pass to HandleException. */
|
||
|
sub sp, sp, #0x120
|
||
|
stp x0, x1, [sp, #(8 * 0)]
|
||
|
stp x2, x3, [sp, #(8 * 2)]
|
||
|
stp x4, x5, [sp, #(8 * 4)]
|
||
|
stp x6, x7, [sp, #(8 * 6)]
|
||
|
stp x8, x9, [sp, #(8 * 8)]
|
||
|
stp x10, x11, [sp, #(8 * 10)]
|
||
|
stp x12, x13, [sp, #(8 * 12)]
|
||
|
stp x14, x15, [sp, #(8 * 14)]
|
||
|
stp x16, x17, [sp, #(8 * 16)]
|
||
|
stp x18, x19, [sp, #(8 * 18)]
|
||
|
stp x20, x21, [sp, #(8 * 20)]
|
||
|
stp x22, x23, [sp, #(8 * 22)]
|
||
|
stp x24, x25, [sp, #(8 * 24)]
|
||
|
stp x26, x27, [sp, #(8 * 26)]
|
||
|
stp x28, x29, [sp, #(8 * 28)]
|
||
|
mrs x20, sp_el0
|
||
|
mrs x21, elr_el1
|
||
|
mrs x22, spsr_el1
|
||
|
mrs x23, tpidr_el0
|
||
|
mov w22, w22
|
||
|
stp x30, x20, [sp, #(8 * 30)]
|
||
|
stp x21, x22, [sp, #(8 * 32)]
|
||
|
str x23, [sp, #(8 * 34)]
|
||
|
|
||
|
/* Invoke ams::kern::arm64::HandleException(ams::kern::arm64::KExceptionContext *). */
|
||
|
mrs x18, tpidr_el1
|
||
|
mov x0, sp
|
||
|
bl _ZN3ams4kern5arm6415HandleExceptionEPNS1_17KExceptionContextE
|
||
|
|
||
|
/* Restore state from the context. */
|
||
|
ldp x30, x20, [sp, #(8 * 30)]
|
||
|
ldp x21, x22, [sp, #(8 * 32)]
|
||
|
ldr x23, [sp, #(8 * 34)]
|
||
|
msr sp_el0, x20
|
||
|
msr elr_el1, x21
|
||
|
msr spsr_el1, x22
|
||
|
msr tpidr_el0, x23
|
||
|
ldp x0, x1, [sp, #(8 * 0)]
|
||
|
ldp x2, x3, [sp, #(8 * 2)]
|
||
|
ldp x4, x5, [sp, #(8 * 4)]
|
||
|
ldp x6, x7, [sp, #(8 * 6)]
|
||
|
ldp x8, x9, [sp, #(8 * 8)]
|
||
|
ldp x10, x11, [sp, #(8 * 10)]
|
||
|
ldp x12, x13, [sp, #(8 * 12)]
|
||
|
ldp x14, x15, [sp, #(8 * 14)]
|
||
|
ldp x16, x17, [sp, #(8 * 16)]
|
||
|
ldp x18, x19, [sp, #(8 * 18)]
|
||
|
ldp x20, x21, [sp, #(8 * 20)]
|
||
|
ldp x22, x23, [sp, #(8 * 22)]
|
||
|
ldp x24, x25, [sp, #(8 * 24)]
|
||
|
ldp x26, x27, [sp, #(8 * 26)]
|
||
|
ldp x28, x29, [sp, #(8 * 28)]
|
||
|
add sp, sp, #0x120
|
||
|
|
||
|
/* Return from the exception. */
|
||
|
eret
|
||
|
|