aboutsummaryrefslogtreecommitdiff
path: root/arch/x86/entry/vdso/vdso32/sigreturn.S
blob: b33fcc501ba30cc906a8806d434fbb4f05a47911 (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
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
/* SPDX-License-Identifier: GPL-2.0 */
#include <linux/linkage.h>
#include <asm/unistd_32.h>
#include <asm/dwarf2.h>
#include <asm/asm-offsets.h>

.macro STARTPROC_SIGNAL_FRAME sc
	CFI_STARTPROC	simple
	CFI_SIGNAL_FRAME
	/* -4 as pretcode has already been popped */
	CFI_DEF_CFA	esp,	\sc - 4
	CFI_OFFSET	eip,    IA32_SIGCONTEXT_ip
	CFI_OFFSET	eax,    IA32_SIGCONTEXT_ax
	CFI_OFFSET	ebx,    IA32_SIGCONTEXT_bx
	CFI_OFFSET	ecx,    IA32_SIGCONTEXT_cx
	CFI_OFFSET	edx,    IA32_SIGCONTEXT_dx
	CFI_OFFSET	esp,    IA32_SIGCONTEXT_sp
	CFI_OFFSET	ebp,    IA32_SIGCONTEXT_bp
	CFI_OFFSET	esi,    IA32_SIGCONTEXT_si
	CFI_OFFSET	edi,    IA32_SIGCONTEXT_di
	CFI_OFFSET	es,     IA32_SIGCONTEXT_es
	CFI_OFFSET	cs,     IA32_SIGCONTEXT_cs
	CFI_OFFSET	ss,     IA32_SIGCONTEXT_ss
	CFI_OFFSET	ds,     IA32_SIGCONTEXT_ds
/*
 * .cfi_offset eflags requires LLVM 16 or newer:
 *
 *   https://github.com/llvm/llvm-project/commit/67bd3c58c0c7389e39c5a2f4d3b1a30459ccf5b7
 *
 * Check for 16.0.1 to ensure the support is present, as 16.0.0 may be a
 * prerelease version.
 */
#if defined(CONFIG_AS_IS_GNU) || (defined(CONFIG_AS_IS_LLVM) && CONFIG_AS_VERSION >= 160001)
	CFI_OFFSET	eflags, IA32_SIGCONTEXT_flags
#endif
.endm

/*
 * WARNING:
 *
 * A bug in the libgcc unwinder as of at least gcc 15.2 (2026) means that
 * the unwinder fails to recognize the signal frame flag.
 *
 * There is a hacky legacy fallback path in libgcc which ends up
 * getting invoked instead. It happens to work as long as BOTH of the
 * following conditions are true:
 *
 * 1. There is at least one byte before the each of the sigreturn
 *    functions which falls outside any function. This is enforced by
 *    an explicit nop instruction before the ALIGN.
 * 2. The code sequences between the entry point up to and including
 *    the int $0x80 below need to match EXACTLY. Do not change them
 *    in any way. The exact byte sequences are:
 *
 *    __kernel_sigreturn:
 *        0:   58                      pop    %eax
 *        1:   b8 77 00 00 00          mov    $0x77,%eax
 *        6:   cd 80                   int    $0x80
 *
 *    __kernel_rt_sigreturn:
 *        0:   b8 ad 00 00 00          mov    $0xad,%eax
 *        5:   cd 80                   int    $0x80
 *
 * For details, see: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=124050
 */
	.text
	.globl __kernel_sigreturn
	.type __kernel_sigreturn,@function
	nop			/* libgcc hack: see comment above */
	ALIGN
__kernel_sigreturn:
	STARTPROC_SIGNAL_FRAME IA32_SIGFRAME_sigcontext
	popl %eax
	CFI_ADJUST_CFA_OFFSET -4
	movl $__NR_sigreturn, %eax
	int $0x80
SYM_INNER_LABEL(vdso32_sigreturn_landing_pad, SYM_L_GLOBAL)
	ud2a
	CFI_ENDPROC
	.size __kernel_sigreturn,.-__kernel_sigreturn

	.globl __kernel_rt_sigreturn
	.type __kernel_rt_sigreturn,@function
	nop			/* libgcc hack: see comment above */
	ALIGN
__kernel_rt_sigreturn:
	STARTPROC_SIGNAL_FRAME IA32_RT_SIGFRAME_sigcontext
	movl $__NR_rt_sigreturn, %eax
	int $0x80
SYM_INNER_LABEL(vdso32_rt_sigreturn_landing_pad, SYM_L_GLOBAL)
	ud2a
	CFI_ENDPROC
	.size __kernel_rt_sigreturn,.-__kernel_rt_sigreturn
	.previous