Skip to content

Commit d3c7443

Browse files
Correctly update isSignalFrame when unwinding the stack via dwarf.
A "signal frame" is a function or block of code where execution arrives via a signal or interrupt, rather than via a normal call instruction. In fact, a particular instruction is interrupted by the signal and needs to be restarted. Therefore, when the signal handler is complete, execution needs to return to the interrupted instruction, rather than the instruction immediately following the call instruction, as in a normal call. Stack unwinders need to know this to correctly unwind signal frames. Dwarf handily provides an "S" in the CIE augmentation string to describe this case, and the libunwind API provides various functions to for unwinders to determine it,. The llvm libunwind implementation correctly sets it's internal variable "isSignalFrame" when initializing an unwind context. However, upon stepping up the stack, the current implementation correctly reads the augmentation string and sets it in the CIE info (which it then discards), libunwind doesn't update it's internal unwind context data structure. This change fixes that, and provides compatibility with both the canonical libunwind and the libgcc implementation. Reviewers: jfb Subscribers: christof, libcxx-commits Tags: #libc Differential Revision: https://reviews.llvm.org/D69677
1 parent b4237db commit d3c7443

File tree

4 files changed

+39
-4
lines changed

4 files changed

+39
-4
lines changed

libunwind/src/DwarfInstructions.hpp

+5-2
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ class DwarfInstructions {
3434
typedef typename A::sint_t sint_t;
3535

3636
static int stepWithDwarf(A &addressSpace, pint_t pc, pint_t fdeStart,
37-
R &registers);
37+
R &registers, bool &isSignalFrame);
3838

3939
private:
4040

@@ -150,7 +150,8 @@ v128 DwarfInstructions<A, R>::getSavedVectorRegister(
150150

151151
template <typename A, typename R>
152152
int DwarfInstructions<A, R>::stepWithDwarf(A &addressSpace, pint_t pc,
153-
pint_t fdeStart, R &registers) {
153+
pint_t fdeStart, R &registers,
154+
bool &isSignalFrame) {
154155
FDE_Info fdeInfo;
155156
CIE_Info cieInfo;
156157
if (CFI_Parser<A>::decodeFDE(addressSpace, fdeStart, &fdeInfo,
@@ -196,6 +197,8 @@ int DwarfInstructions<A, R>::stepWithDwarf(A &addressSpace, pint_t pc,
196197
// restoring SP means setting it to CFA.
197198
newRegisters.setSP(cfa);
198199

200+
isSignalFrame = cieInfo.isSignalFrame;
201+
199202
#if defined(_LIBUNWIND_TARGET_AARCH64)
200203
// If the target is aarch64 then the return address may have been signed
201204
// using the v8.3 pointer authentication extensions. The original

libunwind/src/UnwindCursor.hpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -929,7 +929,7 @@ class UnwindCursor : public AbstractUnwindCursor{
929929
return DwarfInstructions<A, R>::stepWithDwarf(_addressSpace,
930930
(pint_t)this->getReg(UNW_REG_IP),
931931
(pint_t)_info.unwind_info,
932-
_registers);
932+
_registers, _isSignalFrame);
933933
}
934934
#endif
935935

libunwind/src/UnwindLevel1-gcc-ext.c

+8-1
Original file line numberDiff line numberDiff line change
@@ -221,7 +221,14 @@ _LIBUNWIND_EXPORT uintptr_t _Unwind_GetCFA(struct _Unwind_Context *context) {
221221
_LIBUNWIND_EXPORT uintptr_t _Unwind_GetIPInfo(struct _Unwind_Context *context,
222222
int *ipBefore) {
223223
_LIBUNWIND_TRACE_API("_Unwind_GetIPInfo(context=%p)", (void *)context);
224-
*ipBefore = 0;
224+
int isSignalFrame = __unw_is_signal_frame((unw_cursor_t *)context);
225+
// Negative means some kind of error (probably UNW_ENOINFO), but we have no
226+
// good way to report that, and this maintains backward compatibility with the
227+
// implementation that hard-coded zero in every case, even signal frames.
228+
if (isSignalFrame <= 0)
229+
*ipBefore = 0;
230+
else
231+
*ipBefore = 1;
225232
return _Unwind_GetIP(context);
226233
}
227234

libunwind/test/signal_frame.pass.cpp

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
// -*- C++ -*-
2+
//===----------------------------------------------------------------------===//
3+
//
4+
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
5+
// See https://llvm.org/LICENSE.txt for license information.
6+
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7+
//
8+
//===----------------------------------------------------------------------===//
9+
10+
// Ensure that functions marked as signal frames are reported as such.
11+
12+
#include <assert.h>
13+
#include <stdlib.h>
14+
#include <libunwind.h>
15+
16+
int main(void) {
17+
asm(".cfi_signal_frame");
18+
unw_cursor_t cursor;
19+
unw_context_t uc;
20+
unw_getcontext(&uc);
21+
unw_init_local(&cursor, &uc);
22+
assert(unw_step(&cursor) > 0);
23+
assert(unw_is_signal_frame(&cursor));
24+
return 0;
25+
}

0 commit comments

Comments
 (0)