xref: /openbmc/linux/arch/arm64/include/asm/stacktrace/common.h (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
16bf212c8SKalesh Singh /* SPDX-License-Identifier: GPL-2.0-only */
26bf212c8SKalesh Singh /*
36bf212c8SKalesh Singh  * Common arm64 stack unwinder code.
46bf212c8SKalesh Singh  *
5a4c750e2SMarc Zyngier  * See: arch/arm64/kernel/stacktrace.c for the reference implementation.
6051ece67SKalesh Singh  *
76bf212c8SKalesh Singh  * Copyright (C) 2012 ARM Ltd.
86bf212c8SKalesh Singh  */
96bf212c8SKalesh Singh #ifndef __ASM_STACKTRACE_COMMON_H
106bf212c8SKalesh Singh #define __ASM_STACKTRACE_COMMON_H
116bf212c8SKalesh Singh 
12f51e7146SKalesh Singh #include <linux/kprobes.h>
136bf212c8SKalesh Singh #include <linux/types.h>
146bf212c8SKalesh Singh 
156bf212c8SKalesh Singh struct stack_info {
166bf212c8SKalesh Singh 	unsigned long low;
176bf212c8SKalesh Singh 	unsigned long high;
186bf212c8SKalesh Singh };
196bf212c8SKalesh Singh 
2016283c54SMark Rutland /**
2116283c54SMark Rutland  * struct unwind_state - state used for robust unwinding.
226bf212c8SKalesh Singh  *
236bf212c8SKalesh Singh  * @fp:          The fp value in the frame record (or the real fp)
246bf212c8SKalesh Singh  * @pc:          The lr value in the frame record (or the real lr)
256bf212c8SKalesh Singh  *
266bf212c8SKalesh Singh  * @kr_cur:      When KRETPROBES is selected, holds the kretprobe instance
276bf212c8SKalesh Singh  *               associated with the most recently encountered replacement lr
286bf212c8SKalesh Singh  *               value.
296bf212c8SKalesh Singh  *
306bf212c8SKalesh Singh  * @task:        The task being unwound.
318df13730SMark Rutland  *
328df13730SMark Rutland  * @stack:       The stack currently being unwound.
338df13730SMark Rutland  * @stacks:      An array of stacks which can be unwound.
348df13730SMark Rutland  * @nr_stacks:   The number of stacks in @stacks.
356bf212c8SKalesh Singh  */
366bf212c8SKalesh Singh struct unwind_state {
376bf212c8SKalesh Singh 	unsigned long fp;
386bf212c8SKalesh Singh 	unsigned long pc;
396bf212c8SKalesh Singh #ifdef CONFIG_KRETPROBES
406bf212c8SKalesh Singh 	struct llist_node *kr_cur;
416bf212c8SKalesh Singh #endif
426bf212c8SKalesh Singh 	struct task_struct *task;
438df13730SMark Rutland 
448df13730SMark Rutland 	struct stack_info stack;
458df13730SMark Rutland 	struct stack_info *stacks;
468df13730SMark Rutland 	int nr_stacks;
476bf212c8SKalesh Singh };
486bf212c8SKalesh Singh 
stackinfo_get_unknown(void)49d1f684e4SMark Rutland static inline struct stack_info stackinfo_get_unknown(void)
50d1f684e4SMark Rutland {
51d1f684e4SMark Rutland 	return (struct stack_info) {
52d1f684e4SMark Rutland 		.low = 0,
53d1f684e4SMark Rutland 		.high = 0,
54d1f684e4SMark Rutland 	};
55d1f684e4SMark Rutland }
56d1f684e4SMark Rutland 
stackinfo_on_stack(const struct stack_info * info,unsigned long sp,unsigned long size)5736f9a879SMark Rutland static inline bool stackinfo_on_stack(const struct stack_info *info,
5836f9a879SMark Rutland 				      unsigned long sp, unsigned long size)
5936f9a879SMark Rutland {
6036f9a879SMark Rutland 	if (!info->low)
6136f9a879SMark Rutland 		return false;
6236f9a879SMark Rutland 
6336f9a879SMark Rutland 	if (sp < info->low || sp + size < sp || sp + size > info->high)
6436f9a879SMark Rutland 		return false;
6536f9a879SMark Rutland 
6636f9a879SMark Rutland 	return true;
6736f9a879SMark Rutland }
6836f9a879SMark Rutland 
unwind_init_common(struct unwind_state * state,struct task_struct * task)696bf212c8SKalesh Singh static inline void unwind_init_common(struct unwind_state *state,
706bf212c8SKalesh Singh 				      struct task_struct *task)
716bf212c8SKalesh Singh {
726bf212c8SKalesh Singh 	state->task = task;
736bf212c8SKalesh Singh #ifdef CONFIG_KRETPROBES
746bf212c8SKalesh Singh 	state->kr_cur = NULL;
756bf212c8SKalesh Singh #endif
766bf212c8SKalesh Singh 
778df13730SMark Rutland 	state->stack = stackinfo_get_unknown();
786bf212c8SKalesh Singh }
796bf212c8SKalesh Singh 
unwind_find_next_stack(const struct unwind_state * state,unsigned long sp,unsigned long size)808df13730SMark Rutland static struct stack_info *unwind_find_next_stack(const struct unwind_state *state,
818df13730SMark Rutland 						 unsigned long sp,
828df13730SMark Rutland 						 unsigned long size)
838df13730SMark Rutland {
848df13730SMark Rutland 	for (int i = 0; i < state->nr_stacks; i++) {
858df13730SMark Rutland 		struct stack_info *info = &state->stacks[i];
868df13730SMark Rutland 
878df13730SMark Rutland 		if (stackinfo_on_stack(info, sp, size))
888df13730SMark Rutland 			return info;
898df13730SMark Rutland 	}
908df13730SMark Rutland 
918df13730SMark Rutland 	return NULL;
928df13730SMark Rutland }
938df13730SMark Rutland 
9416283c54SMark Rutland /**
958df13730SMark Rutland  * unwind_consume_stack() - Check if an object is on an accessible stack,
968df13730SMark Rutland  * updating stack boundaries so that future unwind steps cannot consume this
978df13730SMark Rutland  * object again.
984e00532fSMarc Zyngier  *
998df13730SMark Rutland  * @state: the current unwind state.
1008df13730SMark Rutland  * @sp:    the base address of the object.
1018df13730SMark Rutland  * @size:  the size of the object.
10216283c54SMark Rutland  *
1038df13730SMark Rutland  * Return: 0 upon success, an error code otherwise.
1044e00532fSMarc Zyngier  */
unwind_consume_stack(struct unwind_state * state,unsigned long sp,unsigned long size)1058df13730SMark Rutland static inline int unwind_consume_stack(struct unwind_state *state,
1068df13730SMark Rutland 				       unsigned long sp,
1078df13730SMark Rutland 				       unsigned long size)
1088df13730SMark Rutland {
1098df13730SMark Rutland 	struct stack_info *next;
1108df13730SMark Rutland 
1118df13730SMark Rutland 	if (stackinfo_on_stack(&state->stack, sp, size))
1128df13730SMark Rutland 		goto found;
1138df13730SMark Rutland 
1148df13730SMark Rutland 	next = unwind_find_next_stack(state, sp, size);
1158df13730SMark Rutland 	if (!next)
1168df13730SMark Rutland 		return -EINVAL;
1178df13730SMark Rutland 
1188df13730SMark Rutland 	/*
1198df13730SMark Rutland 	 * Stack transitions are strictly one-way, and once we've
1208df13730SMark Rutland 	 * transitioned from one stack to another, it's never valid to
1218df13730SMark Rutland 	 * unwind back to the old stack.
1228df13730SMark Rutland 	 *
1238df13730SMark Rutland 	 * Remove the current stack from the list of stacks so that it cannot
1248df13730SMark Rutland 	 * be found on a subsequent transition.
1258df13730SMark Rutland 	 *
1268df13730SMark Rutland 	 * Note that stacks can nest in several valid orders, e.g.
1278df13730SMark Rutland 	 *
1288df13730SMark Rutland 	 *   TASK -> IRQ -> OVERFLOW -> SDEI_NORMAL
1298df13730SMark Rutland 	 *   TASK -> SDEI_NORMAL -> SDEI_CRITICAL -> OVERFLOW
1308df13730SMark Rutland 	 *   HYP -> OVERFLOW
1318df13730SMark Rutland 	 *
1328df13730SMark Rutland 	 * ... so we do not check the specific order of stack
1338df13730SMark Rutland 	 * transitions.
1348df13730SMark Rutland 	 */
1358df13730SMark Rutland 	state->stack = *next;
1368df13730SMark Rutland 	*next = stackinfo_get_unknown();
1378df13730SMark Rutland 
1388df13730SMark Rutland found:
1398df13730SMark Rutland 	/*
1408df13730SMark Rutland 	 * Future unwind steps can only consume stack above this frame record.
1418df13730SMark Rutland 	 * Update the current stack to start immediately above it.
1428df13730SMark Rutland 	 */
1438df13730SMark Rutland 	state->stack.low = sp + size;
1448df13730SMark Rutland 	return 0;
1458df13730SMark Rutland }
1464e00532fSMarc Zyngier 
147b532ab5fSMark Rutland /**
148b532ab5fSMark Rutland  * unwind_next_frame_record() - Unwind to the next frame record.
149b532ab5fSMark Rutland  *
150b532ab5fSMark Rutland  * @state:        the current unwind state.
151b532ab5fSMark Rutland  *
152b532ab5fSMark Rutland  * Return: 0 upon success, an error code otherwise.
153b532ab5fSMark Rutland  */
154b532ab5fSMark Rutland static inline int
unwind_next_frame_record(struct unwind_state * state)155*4b5e694eSMark Rutland unwind_next_frame_record(struct unwind_state *state)
156be63c647SKalesh Singh {
157*4b5e694eSMark Rutland 	unsigned long fp = state->fp;
1588df13730SMark Rutland 	int err;
159be63c647SKalesh Singh 
160be63c647SKalesh Singh 	if (fp & 0x7)
161be63c647SKalesh Singh 		return -EINVAL;
162be63c647SKalesh Singh 
1638df13730SMark Rutland 	err = unwind_consume_stack(state, fp, 16);
1648df13730SMark Rutland 	if (err)
1658df13730SMark Rutland 		return err;
166be63c647SKalesh Singh 
167be63c647SKalesh Singh 	/*
1688df13730SMark Rutland 	 * Record this frame record's values.
169be63c647SKalesh Singh 	 */
170*4b5e694eSMark Rutland 	state->fp = READ_ONCE(*(unsigned long *)(fp));
171*4b5e694eSMark Rutland 	state->pc = READ_ONCE(*(unsigned long *)(fp + 8));
172be63c647SKalesh Singh 
173be63c647SKalesh Singh 	return 0;
174be63c647SKalesh Singh }
175f51e7146SKalesh Singh 
1766bf212c8SKalesh Singh #endif	/* __ASM_STACKTRACE_COMMON_H */
177