Source-Changes-HG archive

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]

[src/trunk]: src/sys/arch/aarch64 fix to do backtrace properly for running LW...



details:   https://anonhg.NetBSD.org/src/rev/8d5c272a2c1c
branches:  trunk
changeset: 972276:8d5c272a2c1c
user:      ryo <ryo%NetBSD.org@localhost>
date:      Fri May 22 19:29:26 2020 +0000

description:
fix to do backtrace properly for running LWPs and cpu_lwp_fork().
when dump of pcb_tf, only the switchframe part is now displayed instead of the whole trapframe.

diffstat:

 sys/arch/aarch64/aarch64/cpuswitch.S  |  14 ++++++++-
 sys/arch/aarch64/aarch64/db_machdep.c |  40 +++++++++++++++++++++++-----
 sys/arch/aarch64/aarch64/db_trace.c   |  47 +++++++++++++++++++++++++++++-----
 sys/arch/aarch64/aarch64/vm_machdep.c |  11 ++++++-
 sys/arch/aarch64/include/db_machdep.h |   3 +-
 5 files changed, 94 insertions(+), 21 deletions(-)

diffs (278 lines):

diff -r 9e7a76fe6cab -r 8d5c272a2c1c sys/arch/aarch64/aarch64/cpuswitch.S
--- a/sys/arch/aarch64/aarch64/cpuswitch.S      Fri May 22 19:02:59 2020 +0000
+++ b/sys/arch/aarch64/aarch64/cpuswitch.S      Fri May 22 19:29:26 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: cpuswitch.S,v 1.19 2020/05/15 09:08:10 ryo Exp $ */
+/* $NetBSD: cpuswitch.S,v 1.20 2020/05/22 19:29:26 ryo Exp $ */
 
 /*-
  * Copyright (c) 2014 The NetBSD Foundation, Inc.
@@ -38,7 +38,7 @@
 #include "opt_ddb.h"
 #include "opt_kasan.h"
 
-RCSID("$NetBSD: cpuswitch.S,v 1.19 2020/05/15 09:08:10 ryo Exp $")
+RCSID("$NetBSD: cpuswitch.S,v 1.20 2020/05/22 19:29:26 ryo Exp $")
 
        ARMV8_DEFINE_OPTIONS
 
@@ -60,6 +60,10 @@
        stp     x25, x26, [sp, #TF_X25]
        stp     x27, x28, [sp, #TF_X27]
        stp     x29, x30, [sp, #TF_X29]
+#ifdef DDB
+       str     lr, [sp, #TF_PC]                /* for backtrace */
+       str     xzr, [sp, #TF_SP]               /* mark as switchframe */
+#endif
 
        /*
         * Save the current stack pointer and the CPACR and save them in
@@ -76,6 +80,9 @@
        DISABLE_INTERRUPT
        ldr     x6, [x1, #L_PCB]        /* x6 = lwp_getpcb(newlwp) */
        ldr     x4, [x6, #PCB_TF]       /* get trapframe ptr (aka SP) */
+#ifdef DDB
+       str     xzr, [x6, #PCB_TF]      /* clear l->l_addr->pcb_tf */
+#endif
        ldr     x5, [x1, #L_MD_CPACR]   /* get cpacr_el1 */
        mov     sp, x4                  /* restore stack pointer */
        msr     cpacr_el1, x5           /* restore cpacr_el1 */
@@ -190,6 +197,9 @@
        str     x19, [x3, #CI_CURLWP]   /* curcpu()->ci_curlwp := x19 */
        ldr     x6, [x19, #L_PCB]       /* x6 = lwp_getpcb(curlwp) */
        ldr     x4, [x6, #PCB_TF]       /* x4 := pinned_lwp->l_addr->pcb_tf */
+#ifdef DDB
+       str     xzr, [x6, #PCB_TF]      /* clear l->l_addr->pcb_tf */
+#endif
        ldr     x5, [x19, #L_MD_CPACR]  /* x5 := pinned_lwp->l_md_cpacr */
        mov     sp, x4                  /* restore pinned_lwp sp */
        msr     cpacr_el1, x5           /* restore pinned_lwp cpacr */
diff -r 9e7a76fe6cab -r 8d5c272a2c1c sys/arch/aarch64/aarch64/db_machdep.c
--- a/sys/arch/aarch64/aarch64/db_machdep.c     Fri May 22 19:02:59 2020 +0000
+++ b/sys/arch/aarch64/aarch64/db_machdep.c     Fri May 22 19:29:26 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: db_machdep.c,v 1.23 2020/05/22 04:46:26 ryo Exp $ */
+/* $NetBSD: db_machdep.c,v 1.24 2020/05/22 19:29:26 ryo Exp $ */
 
 /*-
  * Copyright (c) 2014 The NetBSD Foundation, Inc.
@@ -30,7 +30,7 @@
  */
 
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: db_machdep.c,v 1.23 2020/05/22 04:46:26 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: db_machdep.c,v 1.24 2020/05/22 19:29:26 ryo Exp $");
 
 #ifdef _KERNEL_OPT
 #include "opt_compat_netbsd32.h"
@@ -264,6 +264,29 @@
            tf->tf_reg[30],  tf->tf_sp);
 }
 
+void
+dump_switchframe(struct trapframe *tf, void (*pr)(const char *, ...))
+{
+       struct trapframe tf_buf;
+
+       db_read_bytes((db_addr_t)tf, sizeof(tf_buf), (char *)&tf_buf);
+       tf = &tf_buf;
+
+       (*pr)("   x19=%016"PRIxREGISTER",    x20=%016"PRIxREGISTER"\n",
+           tf->tf_reg[19], tf->tf_reg[20]);
+       (*pr)("   x21=%016"PRIxREGISTER",    x22=%016"PRIxREGISTER"\n",
+           tf->tf_reg[21], tf->tf_reg[22]);
+       (*pr)("   x23=%016"PRIxREGISTER",    x24=%016"PRIxREGISTER"\n",
+           tf->tf_reg[23], tf->tf_reg[24]);
+       (*pr)("   x25=%016"PRIxREGISTER",    x26=%016"PRIxREGISTER"\n",
+           tf->tf_reg[25], tf->tf_reg[26]);
+       (*pr)("   x27=%016"PRIxREGISTER",    x28=%016"PRIxREGISTER"\n",
+           tf->tf_reg[27], tf->tf_reg[28]);
+       (*pr)("fp=x29=%016"PRIxREGISTER", lr=x30=%016"PRIxREGISTER"\n",
+           tf->tf_reg[29], tf->tf_reg[30]);
+}
+
+
 #if defined(_KERNEL)
 static void
 show_cpuinfo(struct cpu_info *ci)
@@ -373,7 +396,7 @@
 
        db_printf("\tl->l_addr.pcb_tf    =%p\n", pcb->pcb_tf);
        if (pcb->pcb_tf != l->l_md.md_utf)
-               dump_trapframe(pcb->pcb_tf, db_printf);
+               dump_switchframe(pcb->pcb_tf, db_printf);
        db_printf("\tl->l_md.md_cpacr  =%016" PRIx64 "\n", l->l_md.md_cpacr);
        db_printf("\tl->l_md.md_flags  =%08x\n", l->l_md.md_flags);
 
@@ -881,7 +904,7 @@
 volatile struct cpu_info *db_trigger;
 volatile struct cpu_info *db_onproc;
 volatile struct cpu_info *db_newcpu;
-volatile int db_readytoswitch[MAXCPUS];
+volatile struct trapframe *db_readytoswitch[MAXCPUS];
 
 #ifdef _KERNEL
 void
@@ -896,8 +919,9 @@
 
        if (!have_addr) {
                for (i = 0; i < ncpu; i++) {
-                       if (db_readytoswitch[i] != 0)
-                               db_printf("cpu%d: ready\n", i);
+                       if (db_readytoswitch[i] != NULL)
+                               db_printf("cpu%d: ready. tf=%p\n", i,
+                                   db_readytoswitch[i]);
                        else
                                db_printf("cpu%d: not responding\n", i);
                }
@@ -969,7 +993,7 @@
                db_trigger = ci;
                membar_producer();
        }
-       db_readytoswitch[ci->ci_index] = 1;
+       db_readytoswitch[ci->ci_index] = tf;
        membar_producer();
 #endif
 
@@ -1024,7 +1048,7 @@
                __asm __volatile ("sev; sev; sev");
        }
        db_trigger = NULL;
-       db_readytoswitch[ci->ci_index] = 0;
+       db_readytoswitch[ci->ci_index] = NULL;
        membar_producer();
 #endif
 
diff -r 9e7a76fe6cab -r 8d5c272a2c1c sys/arch/aarch64/aarch64/db_trace.c
--- a/sys/arch/aarch64/aarch64/db_trace.c       Fri May 22 19:02:59 2020 +0000
+++ b/sys/arch/aarch64/aarch64/db_trace.c       Fri May 22 19:29:26 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: db_trace.c,v 1.10 2020/05/13 06:08:51 ryo Exp $ */
+/* $NetBSD: db_trace.c,v 1.11 2020/05/22 19:29:26 ryo Exp $ */
 
 /*
  * Copyright (c) 2017 Ryo Shimizu <ryo%nerv.org@localhost>
@@ -28,7 +28,7 @@
 
 #include <sys/cdefs.h>
 
-__KERNEL_RCSID(0, "$NetBSD: db_trace.c,v 1.10 2020/05/13 06:08:51 ryo Exp $");
+__KERNEL_RCSID(0, "$NetBSD: db_trace.c,v 1.11 2020/05/22 19:29:26 ryo Exp $");
 
 #include <sys/param.h>
 #include <sys/proc.h>
@@ -208,12 +208,37 @@
                } else
 #endif
                {
+                       struct pcb pcb_buf;
                        struct pcb *pcb = lwp_getpcb(&l);
 
-                       tf = pcb->pcb_tf;
-                       db_read_bytes((db_addr_t)&tf->tf_reg[29], sizeof(fp), (char *)&fp);
-                       (*pr)("trace: pid %d lid %d at tf %p\n",
-                           p.p_pid, l.l_lid, tf);
+                       db_read_bytes((db_addr_t)pcb, sizeof(pcb_buf),
+                           (char *)&pcb_buf);
+                       tf = pcb_buf.pcb_tf;
+                       if (tf != 0) {
+                               db_read_bytes((db_addr_t)&tf->tf_reg[29],
+                                   sizeof(fp), (char *)&fp);
+                               (*pr)("trace: pid %d lid %d at tf %p (in pcb)\n",
+                                   p.p_pid, l.l_lid, tf);
+                       }
+#if defined(MULTIPROCESSOR) && defined(_KERNEL)
+                       else if (l.l_stat == LSONPROC ||
+                           (l.l_pflag & LP_RUNNING) != 0) {
+
+                               /* running lwp on other cpus */
+                               extern struct trapframe *db_readytoswitch[];
+                               struct cpu_info cpuinfobuf;
+
+                               db_read_bytes((db_addr_t)l.l_cpu,
+                                   sizeof(cpuinfobuf), (char *)&cpuinfobuf);
+                               tf = db_readytoswitch[cpuinfobuf.ci_index];
+
+                               (*pr)("trace: pid %d lid %d at tf %p (in kdb_trap)\n",
+                                   p.p_pid, l.l_lid, tf);
+                       }
+#endif
+                       else {
+                               (*pr)("trace: no trapframe found for lwp: %p\n", (void *)addr);
+                       }
                }
        } else if (tf == NULL) {
                fp = addr;
@@ -227,14 +252,20 @@
 
        if (tf != NULL) {
 #if defined(_KERNEL)
-               (*pr)("---- trapframe %p (%zu bytes) ----\n",
+               bool is_switchframe = (tf->tf_sp == 0);
+               (*pr)("---- %s %p (%zu bytes) ----\n",
+                   is_switchframe ? "switchframe" : "trapframe",
                    tf, sizeof(*tf));
-               dump_trapframe(tf, pr);
+               if (is_switchframe)
+                       dump_switchframe(tf, pr);
+               else
+                       dump_trapframe(tf, pr);
                (*pr)("------------------------"
                      "------------------------\n");
 
 #endif
                lastfp = lastlr = lr = fp = 0;
+
                db_read_bytes((db_addr_t)&tf->tf_pc, sizeof(lr), (char *)&lr);
                db_read_bytes((db_addr_t)&tf->tf_reg[29], sizeof(fp), (char *)&fp);
                lr = aarch64_strip_pac(lr);
diff -r 9e7a76fe6cab -r 8d5c272a2c1c sys/arch/aarch64/aarch64/vm_machdep.c
--- a/sys/arch/aarch64/aarch64/vm_machdep.c     Fri May 22 19:02:59 2020 +0000
+++ b/sys/arch/aarch64/aarch64/vm_machdep.c     Fri May 22 19:29:26 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: vm_machdep.c,v 1.6 2020/04/12 07:49:58 maxv Exp $ */
+/* $NetBSD: vm_machdep.c,v 1.7 2020/05/22 19:29:26 ryo Exp $ */
 
 /*-
  * Copyright (c) 2014 The NetBSD Foundation, Inc.
@@ -29,8 +29,10 @@
  * POSSIBILITY OF SUCH DAMAGE.
  */
 
+#include "opt_ddb.h"
+
 #include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: vm_machdep.c,v 1.6 2020/04/12 07:49:58 maxv Exp $");
+__KERNEL_RCSID(0, "$NetBSD: vm_machdep.c,v 1.7 2020/05/22 19:29:26 ryo Exp $");
 
 #include <sys/param.h>
 #include <sys/systm.h>
@@ -140,6 +142,11 @@
        ktf->tf_reg[29] = 0;
        KASSERT(reg_daif_read() == 0);
        ktf->tf_lr = (uintptr_t)lwp_trampoline;
+#ifdef DDB
+       ktf->tf_pc = (uint64_t)&&backtrace_here;
+       ktf->tf_sp = 0; /* mark as switchframe */
+ backtrace_here:
+#endif
 
        pcb2->pcb_tf = ktf;
 }
diff -r 9e7a76fe6cab -r 8d5c272a2c1c sys/arch/aarch64/include/db_machdep.h
--- a/sys/arch/aarch64/include/db_machdep.h     Fri May 22 19:02:59 2020 +0000
+++ b/sys/arch/aarch64/include/db_machdep.h     Fri May 22 19:29:26 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: db_machdep.h,v 1.8 2018/10/12 01:28:58 ryo Exp $ */
+/* $NetBSD: db_machdep.h,v 1.9 2020/05/22 19:29:26 ryo Exp $ */
 
 /*-
  * Copyright (c) 2014 The NetBSD Foundation, Inc.
@@ -202,6 +202,7 @@
 
 #define DB_MACHINE_COMMANDS
 void dump_trapframe(struct trapframe *, void (*)(const char *, ...));
+void dump_switchframe(struct trapframe *, void (*)(const char *, ...));
 const char *strdisasm(vaddr_t);
 const char *strdisasm_aarch32(vaddr_t);
 void db_machdep_init(void);



Home | Main Index | Thread Index | Old Index