arm64: add on_accessible_stack()
authorMark Rutland <mark.rutland@arm.com>
Tue, 1 Aug 2017 17:51:15 +0000 (18:51 +0100)
committerMark Rutland <mark.rutland@arm.com>
Tue, 15 Aug 2017 17:36:12 +0000 (18:36 +0100)
Both unwind_frame() and dump_backtrace() try to check whether a stack
address is sane to access, with very similar logic. Both will need
updating in order to handle overflow stacks.

Factor out this logic into a helper, so that we can avoid further
duplication when we add overflow stacks.

Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Reviewed-by: Will Deacon <will.deacon@arm.com>
Tested-by: Laura Abbott <labbott@redhat.com>
Cc: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Cc: Catalin Marinas <catalin.marinas@arm.com>
Cc: James Morse <james.morse@arm.com>
arch/arm64/include/asm/stacktrace.h
arch/arm64/kernel/stacktrace.c
arch/arm64/kernel/traps.c

index 4c68d8a81988f576f3eab0bf058e3b3277aaa6ce..92ddb6d25cf3235f2c385aefbe6de61e3ce0aee9 100644 (file)
@@ -57,4 +57,20 @@ static inline bool on_task_stack(struct task_struct *tsk, unsigned long sp)
        return (low <= sp && sp < high);
 }
 
+/*
+ * We can only safely access per-cpu stacks from current in a non-preemptible
+ * context.
+ */
+static inline bool on_accessible_stack(struct task_struct *tsk, unsigned long sp)
+{
+       if (on_task_stack(tsk, sp))
+               return true;
+       if (tsk != current || preemptible())
+               return false;
+       if (on_irq_stack(sp))
+               return true;
+
+       return false;
+}
+
 #endif /* __ASM_STACKTRACE_H */
index 35588caad9d05c0a1ad889e58d993f5c1881a010..3144584617e7a474cc9c15e829d294bb4315ebb0 100644 (file)
@@ -50,12 +50,7 @@ int notrace unwind_frame(struct task_struct *tsk, struct stackframe *frame)
        if (!tsk)
                tsk = current;
 
-       /*
-        * Switching between stacks is valid when tracing current and in
-        * non-preemptible context.
-        */
-       if (!(tsk == current && !preemptible() && on_irq_stack(fp)) &&
-           !on_task_stack(tsk, fp))
+       if (!on_accessible_stack(tsk, fp))
                return -EINVAL;
 
        frame->fp = READ_ONCE_NOCHECK(*(unsigned long *)(fp));
index 9633773ca42c455ccb02b53fffe461c337984814..d01c5988354b1e178834c054e0e2339edad4d7ff 100644 (file)
@@ -193,8 +193,7 @@ void dump_backtrace(struct pt_regs *regs, struct task_struct *tsk)
                if (in_entry_text(frame.pc)) {
                        stack = frame.fp - offsetof(struct pt_regs, stackframe);
 
-                       if (on_task_stack(tsk, stack) ||
-                           (tsk == current && !preemptible() && on_irq_stack(stack)))
+                       if (on_accessible_stack(tsk, stack))
                                dump_mem("", "Exception stack", stack,
                                         stack + sizeof(struct pt_regs));
                }