#include <linux/export.h>
 #include <linux/kallsyms.h>
 #include <linux/module.h>
+#include <linux/nmi.h>
 #include <linux/sched.h>
 #include <linux/sched/debug.h>
 #include <linux/sched/task_stack.h>
 #include <linux/ftrace.h>
 #include <asm/kprobes.h>
 
+#include <asm/paca.h>
+
 /*
  * Save stack-backtrace addresses into a stack_trace buffer.
  */
 }
 EXPORT_SYMBOL_GPL(save_stack_trace_tsk_reliable);
 #endif /* CONFIG_HAVE_RELIABLE_STACKTRACE */
+
+#ifdef CONFIG_PPC_BOOK3S_64
+static void handle_backtrace_ipi(struct pt_regs *regs)
+{
+       nmi_cpu_backtrace(regs);
+}
+
+static void raise_backtrace_ipi(cpumask_t *mask)
+{
+       unsigned int cpu;
+
+       for_each_cpu(cpu, mask) {
+               if (cpu == smp_processor_id())
+                       handle_backtrace_ipi(NULL);
+               else
+                       smp_send_safe_nmi_ipi(cpu, handle_backtrace_ipi, 5 * USEC_PER_SEC);
+       }
+
+       for_each_cpu(cpu, mask) {
+               struct paca_struct *p = paca_ptrs[cpu];
+
+               cpumask_clear_cpu(cpu, mask);
+
+               pr_warn("CPU %d didn't respond to backtrace IPI, inspecting paca.\n", cpu);
+               if (!virt_addr_valid(p)) {
+                       pr_warn("paca pointer appears corrupt? (%px)\n", p);
+                       continue;
+               }
+
+               pr_warn("irq_soft_mask: 0x%02x in_mce: %d in_nmi: %d",
+                       p->irq_soft_mask, p->in_mce, p->in_nmi);
+
+               if (virt_addr_valid(p->__current))
+                       pr_cont(" current: %d (%s)\n", p->__current->pid,
+                               p->__current->comm);
+               else
+                       pr_cont(" current pointer corrupt? (%px)\n", p->__current);
+
+               pr_warn("Back trace of paca->saved_r1 (0x%016llx) (possibly stale):\n", p->saved_r1);
+               show_stack(p->__current, (unsigned long *)p->saved_r1);
+       }
+}
+
+void arch_trigger_cpumask_backtrace(const cpumask_t *mask, bool exclude_self)
+{
+       nmi_trigger_cpumask_backtrace(mask, exclude_self, raise_backtrace_ipi);
+}
+#endif /* CONFIG_PPC64 */