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