summaryrefslogtreecommitdiff
path: root/src/pal/src/arch/arm/callsignalhandlerwrapper.S
blob: b9398d6d632c547478b280cd5143e62cf59a17d7 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.

#include "unixasmmacros.inc"
#include "asmconstants.h"

.syntax unified
.thumb

.macro CALL_SIGNAL_HANDLER_WRAPPER Alignment

.globl C_FUNC(SignalHandlerWorkerReturnOffset\Alignment)
C_FUNC(SignalHandlerWorkerReturnOffset\Alignment):
    .int LOCAL_LABEL(SignalHandlerWorkerReturn\Alignment)-C_FUNC(CallSignalHandlerWrapper\Alignment)

// This function is never called, only a fake stack frame will be setup to have a return
// address set to SignalHandlerWorkerReturn during SIGSEGV handling.
// It enables the unwinder to unwind stack from the handling code to the actual failure site.
NESTED_ENTRY CallSignalHandlerWrapper\Alignment, _TEXT, NoHandler
#ifndef __linux__
__StackAllocationSize = (8 + 4 + \Alignment) // red zone + alignment
    alloc_stack __StackAllocationSize
    PROLOG_PUSH "{r7, r11, lr}"
    bl      EXTERNAL_C_FUNC(signal_handler_worker)
LOCAL_LABEL(SignalHandlerWorkerReturn\Alignment):
    EPILOG_POP "{r7, r11, lr}"
    free_stack __StackAllocationSize
    bx      lr
#else
    // This unwind information is needed for lldb gdb doesn't use it and tries
    // to read all registers from $sp + 12
    .save {r0-r15}
    .pad #12
    bl      EXTERNAL_C_FUNC(signal_handler_worker)
LOCAL_LABEL(SignalHandlerWorkerReturn\Alignment):
    // Following instruction are needed to say gdb that this frame is SIGTRAMP_FRAME
    // and it can restore all registers from stack
    mov.w r7, #119
    svc 0
#endif
NESTED_END CallSignalHandlerWrapper\Alignment, _TEXT

.endm

CALL_SIGNAL_HANDLER_WRAPPER 0
CALL_SIGNAL_HANDLER_WRAPPER 4