a9b3c6998d
In preparation for wider use, move the ENCODE_FRAME_POINTER macros to a common header and provide inline asm versions. These macros are used to encode a pt_regs frame for the unwinder; see unwind_frame.c:decode_frame_pointer(). Signed-off-by: Peter Zijlstra (Intel) <peterz@infradead.org> Reviewed-by: Josh Poimboeuf <jpoimboe@redhat.com> Cc: Linus Torvalds <torvalds@linux-foundation.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Thomas Gleixner <tglx@linutronix.de> Signed-off-by: Ingo Molnar <mingo@kernel.org>
95 lines
2.2 KiB
C
95 lines
2.2 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
#ifndef _ASM_X86_FRAME_H
|
|
#define _ASM_X86_FRAME_H
|
|
|
|
#include <asm/asm.h>
|
|
|
|
/*
|
|
* These are stack frame creation macros. They should be used by every
|
|
* callable non-leaf asm function to make kernel stack traces more reliable.
|
|
*/
|
|
|
|
#ifdef CONFIG_FRAME_POINTER
|
|
|
|
#ifdef __ASSEMBLY__
|
|
|
|
.macro FRAME_BEGIN
|
|
push %_ASM_BP
|
|
_ASM_MOV %_ASM_SP, %_ASM_BP
|
|
.endm
|
|
|
|
.macro FRAME_END
|
|
pop %_ASM_BP
|
|
.endm
|
|
|
|
#ifdef CONFIG_X86_64
|
|
/*
|
|
* This is a sneaky trick to help the unwinder find pt_regs on the stack. The
|
|
* frame pointer is replaced with an encoded pointer to pt_regs. The encoding
|
|
* is just setting the LSB, which makes it an invalid stack address and is also
|
|
* a signal to the unwinder that it's a pt_regs pointer in disguise.
|
|
*
|
|
* NOTE: This macro must be used *after* PUSH_AND_CLEAR_REGS because it corrupts
|
|
* the original rbp.
|
|
*/
|
|
.macro ENCODE_FRAME_POINTER ptregs_offset=0
|
|
leaq 1+\ptregs_offset(%rsp), %rbp
|
|
.endm
|
|
#else /* !CONFIG_X86_64 */
|
|
/*
|
|
* This is a sneaky trick to help the unwinder find pt_regs on the stack. The
|
|
* frame pointer is replaced with an encoded pointer to pt_regs. The encoding
|
|
* is just clearing the MSB, which makes it an invalid stack address and is also
|
|
* a signal to the unwinder that it's a pt_regs pointer in disguise.
|
|
*
|
|
* NOTE: This macro must be used *after* SAVE_ALL because it corrupts the
|
|
* original ebp.
|
|
*/
|
|
.macro ENCODE_FRAME_POINTER
|
|
mov %esp, %ebp
|
|
andl $0x7fffffff, %ebp
|
|
.endm
|
|
#endif /* CONFIG_X86_64 */
|
|
|
|
#else /* !__ASSEMBLY__ */
|
|
|
|
#define FRAME_BEGIN \
|
|
"push %" _ASM_BP "\n" \
|
|
_ASM_MOV "%" _ASM_SP ", %" _ASM_BP "\n"
|
|
|
|
#define FRAME_END "pop %" _ASM_BP "\n"
|
|
|
|
#ifdef CONFIG_X86_64
|
|
#define ENCODE_FRAME_POINTER \
|
|
"lea 1(%rsp), %rbp\n\t"
|
|
#else /* !CONFIG_X86_64 */
|
|
#define ENCODE_FRAME_POINTER \
|
|
"movl %esp, %ebp\n\t" \
|
|
"andl $0x7fffffff, %ebp\n\t"
|
|
#endif /* CONFIG_X86_64 */
|
|
|
|
#endif /* __ASSEMBLY__ */
|
|
|
|
#define FRAME_OFFSET __ASM_SEL(4, 8)
|
|
|
|
#else /* !CONFIG_FRAME_POINTER */
|
|
|
|
#ifdef __ASSEMBLY__
|
|
|
|
.macro ENCODE_FRAME_POINTER ptregs_offset=0
|
|
.endm
|
|
|
|
#else /* !__ASSEMBLY */
|
|
|
|
#define ENCODE_FRAME_POINTER
|
|
|
|
#endif
|
|
|
|
#define FRAME_BEGIN
|
|
#define FRAME_END
|
|
#define FRAME_OFFSET 0
|
|
|
|
#endif /* CONFIG_FRAME_POINTER */
|
|
|
|
#endif /* _ASM_X86_FRAME_H */
|