pinebuds/utils/crash_catcher/CrashCatcher_armv7m.S
2022-08-15 17:20:27 +08:00

112 lines
3.7 KiB
ArmAsm

/* Copyright (C) 2018 Adam Green (https://github.com/adamgreen)
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#include "CrashCatcherPriv.h"
/* Implementation of ARMv7-M assembly language code to trap exceptions and call CrashCatcher_Entry(). */
.text
.syntax unified
/* Called on Hard Fault exception. Stacks important registers and calls CrashCatcher_Entry().
extern "C" void HardFault_Handler(void);
*/
.global gdb_fault_handler
.type gdb_fault_handler, %function
.thumb_func
gdb_fault_handler:
/* Push the following onto the stack (see CrashCatcherExceptionRegisters structure). The g_crashCatcherStack buffer
is reserved for use as the stack while CrashCatcher is running.
exceptionPSR
psp
msp
r4
r5
r6
r7
r8
r9
r10
r11
exceptionLR */
mrs r3, xpsr
mrs r2, psp
mrs r1, msp
ldr sp, =(g_crashCatcherStack + 4 * CRASH_CATCHER_STACK_WORD_COUNT)
push.w {r1-r11,lr}
// Call CrashCatcher_Entry with first argument pointing to registers that were just stacked.
mov r0, sp
bl CrashCatcher_Entry
// Only make it back here when CrashCatcher_DumpEnd() returns CRASH_CATCHER_EXIT to indicate that it would like
// execution to be restored back to the faulting code (typically a hard coded breakpoint).
// Restore non-volatile registers and SP to values they had upon entry to fault handler before resuming execution
// at point of fault.
pop.w {r1-r11,lr}
mov sp, r1
bx lr
// Let assembler know that we have hit the end of the HardFault_Handler function.
.pool
.size gdb_fault_handler, .-gdb_fault_handler
/* Called from CrashCatcher core to copy all floating point registers to supplied buffer. The supplied buffer must
be large enough to contain 33 32-bit values (S0-S31 & FPSCR).
void CrashCatcher_CopyAllFloatingPointRegisters(uint32_t* pBuffer);
*/
.global CrashCatcher_CopyAllFloatingPointRegisters
.type CrashCatcher_CopyAllFloatingPointRegisters, %function
.thumb_func
CrashCatcher_CopyAllFloatingPointRegisters:
// Grab a copy of FPSCR before issuing any other FP instructions.
vmrs r1, fpscr
// Move s0 - s31 to pBuffer.
vstmia.32 r0!, {s0 - s31}
// Move fpscr to pBuffer.
str r1, [r0]
// Return to caller.
bx lr
.pool
.size CrashCatcher_CopyAllFloatingPointRegisters, .-CrashCatcher_CopyAllFloatingPointRegisters
/* Called from CrashCatcher core to copy upper 16 floating point registers to supplied buffer. The supplied buffer
must be large enough to contain 16 32-bit values (S16-S31).
void CrashCatcher_CopyUpperFloatingPointRegisters(uint32_t* pBuffer);
*/
.global CrashCatcher_CopyUpperFloatingPointRegisters
.type CrashCatcher_CopyUpperFloatingPointRegisters, %function
.thumb_func
CrashCatcher_CopyUpperFloatingPointRegisters:
// Move s16 - s31 to pBuffer.
vstmia.32 r0!, {s16 - s31}
// Return to caller.
bx lr
.pool
.size CrashCatcher_CopyUpperFloatingPointRegisters, .-CrashCatcher_CopyUpperFloatingPointRegisters
.end