Add unwind_next_frame_sframe() function to unwind by sframe info if
present. Use this method at exception boundaries, falling back to
frame-pointer unwind only on failure. In such failure cases, the
stacktrace is considered unreliable.

During normal unwind, prefer frame pointer unwind (for better
performance) with sframe as a backup.

This change restores the LR behavior originally introduced in commit
c2c6b27b5aa14fa2 ("arm64: stacktrace: unwind exception boundaries"),
But later removed in commit 32ed1205682e ("arm64: stacktrace: Skip
reporting LR at exception boundaries")

This can be done because the sframe data can be used to determine
whether the LR is current for the PC value recovered from pt_regs at the
exception boundary.

Signed-off-by: Weinan Liu <[email protected]>
Reviewed-by: Prasanna Kumar T S M <[email protected]>
Reviewed-by: Jens Remus <[email protected]>
Signed-off-by: Dylan Hatch <[email protected]>
---
 arch/arm64/kernel/stacktrace.c | 222 ++++++++++++++++++++++++++++++---
 1 file changed, 202 insertions(+), 20 deletions(-)

diff --git a/arch/arm64/kernel/stacktrace.c b/arch/arm64/kernel/stacktrace.c
index 3ebcf8c53fb0..cee860ca8ce5 100644
--- a/arch/arm64/kernel/stacktrace.c
+++ b/arch/arm64/kernel/stacktrace.c
@@ -13,6 +13,7 @@
 #include <linux/sched.h>
 #include <linux/sched/debug.h>
 #include <linux/sched/task_stack.h>
+#include <linux/sframe.h>
 #include <linux/stacktrace.h>
 
 #include <asm/efi.h>
@@ -26,6 +27,7 @@ enum kunwind_source {
        KUNWIND_SOURCE_CALLER,
        KUNWIND_SOURCE_TASK,
        KUNWIND_SOURCE_REGS_PC,
+       KUNWIND_SOURCE_REGS_LR,
 };
 
 union unwind_flags {
@@ -45,6 +47,7 @@ union unwind_flags {
  * @kr_cur:      When KRETPROBES is selected, holds the kretprobe instance
  *               associated with the most recently encountered replacement lr
  *               value.
+ * @unreliable:  Stacktrace is unreliable.
  */
 struct kunwind_state {
        struct unwind_state common;
@@ -56,6 +59,7 @@ struct kunwind_state {
        enum kunwind_source source;
        union unwind_flags flags;
        struct pt_regs *regs;
+       bool unreliable;
 };
 
 static __always_inline void
@@ -181,7 +185,6 @@ int kunwind_next_regs_pc(struct kunwind_state *state)
        state->regs = regs;
        state->common.pc = regs->pc;
        state->common.fp = regs->regs[29];
-       state->regs = NULL;
        state->source = KUNWIND_SOURCE_REGS_PC;
        return 0;
 }
@@ -244,6 +247,168 @@ kunwind_next_frame_record(struct kunwind_state *state)
        return 0;
 }
 
+#ifdef CONFIG_HAVE_UNWIND_KERNEL_SFRAME
+
+static __always_inline struct stack_info *
+get_word(struct unwind_state *state, unsigned long *word)
+{
+       unsigned long addr = *word;
+       struct stack_info *info;
+
+       info = unwind_find_stack(state, addr, sizeof(addr));
+       if (!info)
+               return info;
+
+       *word = READ_ONCE(*(unsigned long *)addr);
+
+       return info;
+}
+
+static __always_inline int
+get_consume_word(struct unwind_state *state, unsigned long *word)
+{
+       struct stack_info *info;
+       unsigned long addr = *word;
+
+       info = get_word(state, word);
+       if (!info)
+               return -EINVAL;
+
+       unwind_consume_stack(state, info, addr, sizeof(addr));
+       return 0;
+}
+
+/*
+ * Unwind from a pt_regs according to sframe.
+ */
+static __always_inline int
+kunwind_next_regs_sframe(struct kunwind_state *state)
+{
+       struct unwind_frame frame;
+       unsigned long cfa, fp, ra;
+       enum kunwind_source source = KUNWIND_SOURCE_FRAME;
+       struct pt_regs *regs = state->regs;
+
+       int err;
+
+       if (WARN_ON_ONCE(state->source != KUNWIND_SOURCE_REGS_PC))
+               return -EINVAL;
+       if (WARN_ON_ONCE(!state->regs))
+               return -EINVAL;
+
+       /* FP/SP alignment 8 bytes */
+       if (state->common.fp & 0x7)
+               return -EINVAL;
+
+       err = sframe_find_kernel(state->common.pc, &frame);
+       if (err)
+               return -EINVAL;
+
+       /*
+        * A kernel unwind should always end at a FRAME_META_TYPE_FINAL
+        * frame. There should be no outermost frames within the kernel.
+        */
+       if (frame.outermost)
+               return -EINVAL;
+
+       /* Get the Canonical Frame Address (CFA) */
+       switch (frame.cfa.rule) {
+       case UNWIND_CFA_RULE_SP_OFFSET:
+               cfa = state->regs->sp;
+               break;
+       case UNWIND_CFA_RULE_FP_OFFSET:
+               if (state->common.fp < state->regs->sp)
+                       return -EINVAL;
+               cfa = state->common.fp;
+               break;
+       /*
+        * UNWIND_CFA_RULE_REG_OFFSET and UNWIND_CFA_RULE_REG_OFFSET_DEREF not
+        * implemented -- flexible FDEs are not currently generated by assembler
+        * for arm64.
+        */
+       default:
+               WARN_ON_ONCE(1);
+               return -EINVAL;
+       }
+       cfa += frame.cfa.offset;
+
+       /* CFA alignment 16 bytes */
+       if (cfa & 0x15)
+               return -EINVAL;
+
+       /* Get the Return Address (RA) */
+       switch (frame.ra.rule) {
+       case UNWIND_RULE_RETAIN:
+               ra = regs->regs[30];
+               source = KUNWIND_SOURCE_REGS_LR;
+               break;
+
+       /*
+        * UNWIND_RULE_CFA_OFFSET doesn't make sense for RA.
+        * The return address cannot legitimately be a stack address.
+        */
+       case UNWIND_RULE_CFA_OFFSET_DEREF:
+               ra = cfa + frame.ra.offset;
+               break;
+       /*
+        * UNWIND_RULE_REG_OFFSET and UNWIND_RULE_REG_OFFSET_DEREF not
+        * implemented -- flexible FDEs are not currently generated by assembler
+        * for arm64.
+        */
+       default:
+               WARN_ON_ONCE(1);
+               return -EINVAL;
+       }
+
+       /* Get the Frame Pointer (FP) */
+       switch (frame.fp.rule) {
+       case UNWIND_RULE_RETAIN:
+               fp = state->common.fp;
+               break;
+       /*
+        * UNWIND_RULE_CFA_OFFSET is currently not used for FP
+        * (e.g. SFrame cannot represent this rule).
+        */
+       case UNWIND_RULE_CFA_OFFSET_DEREF:
+               fp = cfa + frame.fp.offset;
+               break;
+       /*
+        * UNWIND_RULE_REG_OFFSET and UNWIND_RULE_REG_OFFSET_DEREF not
+        * implemented -- flexible FDEs are not currently generated by assembler
+        * for arm64.
+        */
+       default:
+               WARN_ON_ONCE(1);
+               return -EINVAL;
+       }
+
+       /*
+        * Consume RA and FP from the stack. The frame record puts FP at a lower
+        * address than RA, so we always read FP first.
+        */
+       if (frame.fp.rule & UNWIND_RULE_DEREF &&
+           !get_word(&state->common, &fp))
+               return -EINVAL;
+
+       if (frame.ra.rule & UNWIND_RULE_DEREF &&
+           get_consume_word(&state->common, &ra))
+               return -EINVAL;
+
+       state->common.pc = ra;
+       state->common.fp = fp;
+
+       state->source = source;
+
+       return 0;
+}
+
+#else /* !CONFIG_HAVE_UNWIND_KERNEL_SFRAME */
+
+static __always_inline int
+unwind_next_frame_sframe(struct kunwind_state *state) { return -EINVAL; }
+
+#endif /* !CONFIG_HAVE_UNWIND_KERNEL_SFRAME*/
+
 /*
  * Unwind from one frame record (A) to the next frame record (B).
  *
@@ -259,10 +424,20 @@ kunwind_next(struct kunwind_state *state)
        state->flags.all = 0;
 
        switch (state->source) {
+       case KUNWIND_SOURCE_REGS_PC:
+               err = kunwind_next_regs_sframe(state);
+
+               if (err && err != -ENOENT) {
+                       /* Fallback to FP based unwinder */
+                       err = kunwind_next_frame_record(state);
+                       state->unreliable = true;
+               }
+               state->regs = NULL;
+               break;
        case KUNWIND_SOURCE_FRAME:
        case KUNWIND_SOURCE_CALLER:
        case KUNWIND_SOURCE_TASK:
-       case KUNWIND_SOURCE_REGS_PC:
+       case KUNWIND_SOURCE_REGS_LR:
                err = kunwind_next_frame_record(state);
                break;
        default:
@@ -390,34 +565,40 @@ noinline noinstr void 
arch_stack_walk(stack_trace_consume_fn consume_entry,
        kunwind_stack_walk(arch_kunwind_consume_entry, &data, task, regs);
 }
 
+struct kunwind_reliable_consume_entry_data {
+       stack_trace_consume_fn consume_entry;
+       void *cookie;
+       bool unreliable;
+};
+
 static __always_inline bool
-arch_reliable_kunwind_consume_entry(const struct kunwind_state *state, void 
*cookie)
+arch_kunwind_reliable_consume_entry(const struct kunwind_state *state, void 
*cookie)
 {
-       /*
-        * At an exception boundary we can reliably consume the saved PC. We do
-        * not know whether the LR was live when the exception was taken, and
-        * so we cannot perform the next unwind step reliably.
-        *
-        * All that matters is whether the *entire* unwind is reliable, so give
-        * up as soon as we hit an exception boundary.
-        */
-       if (state->source == KUNWIND_SOURCE_REGS_PC)
-               return false;
+       struct kunwind_reliable_consume_entry_data *data = cookie;
 
-       return arch_kunwind_consume_entry(state, cookie);
+       if (state->unreliable) {
+               data->unreliable = true;
+               return false;
+       }
+       return data->consume_entry(data->cookie, state->common.pc);
 }
 
-noinline noinstr int arch_stack_walk_reliable(stack_trace_consume_fn 
consume_entry,
-                                             void *cookie,
-                                             struct task_struct *task)
+noinline notrace int arch_stack_walk_reliable(
+                               stack_trace_consume_fn consume_entry,
+                               void *cookie, struct task_struct *task)
 {
-       struct kunwind_consume_entry_data data = {
+       struct kunwind_reliable_consume_entry_data data = {
                .consume_entry = consume_entry,
                .cookie = cookie,
+               .unreliable = false,
        };
 
-       return kunwind_stack_walk(arch_reliable_kunwind_consume_entry, &data,
-                                 task, NULL);
+       kunwind_stack_walk(arch_kunwind_reliable_consume_entry, &data, task, 
NULL);
+
+       if (data.unreliable)
+               return -EINVAL;
+
+       return 0;
 }
 
 struct bpf_unwind_consume_entry_data {
@@ -452,6 +633,7 @@ static const char *state_source_string(const struct 
kunwind_state *state)
        case KUNWIND_SOURCE_CALLER:     return "C";
        case KUNWIND_SOURCE_TASK:       return "T";
        case KUNWIND_SOURCE_REGS_PC:    return "P";
+       case KUNWIND_SOURCE_REGS_LR:    return "L";
        default:                        return "U";
        }
 }
-- 
2.54.0.563.g4f69b47b94-goog


Reply via email to