crashinfo: Print stack traces for all on-CPU threads

Add a python script which implements the bulk of this functionality.
Over time, this would ideally evolve into a library of python routines
which can be used to inspect kernel data structures and automate some
debugging tasks, similar to jhb's out-of-tree scripts, but written in a
somewhat nicer language and with better integration into the kgdb
command prompt.

Note that kgdb currently won't auto-load scripts in this directory.
This should perhaps change in the future.  It probably also makes more
sense to have a crashinfo.py which provides all the kgdb output that we
want to include in core.txt, rather than having crashinfo.sh pipe in
several commands.

Reviewed by:	avg, imp
Discussed with:	jhb
MFC after:	3 weeks
Sponsored by:	The FreeBSD Foundation
Differential Revision:	https://reviews.freebsd.org/D33817
This commit is contained in:
Mark Johnston 2024-01-15 15:39:26 -05:00
parent b9e8ae1d8a
commit 2524b7dfb0
5 changed files with 76 additions and 4 deletions

View File

@ -161,6 +161,8 @@
..
hyperv
..
kgdb
..
lpr
ru
..

View File

@ -1,4 +1,3 @@
.include <src.opts.mk>
.include <bsd.compat.pre.mk>
@ -11,6 +10,7 @@ SUBDIR= ${_atf} \
flua \
getty \
${_hyperv} \
kgdb \
${_mail.local} \
${_makewhatis.local} \
${_mknetid} \

5
libexec/kgdb/Makefile Normal file
View File

@ -0,0 +1,5 @@
FILESDIR?= /usr/libexec/kgdb
FILES= acttrace.py
.include <bsd.prog.mk>

63
libexec/kgdb/acttrace.py Normal file
View File

@ -0,0 +1,63 @@
#-
# Copyright (c) 2022 The FreeBSD Foundation
#
# This software was developed by Mark Johnston under sponsorship from the
# FreeBSD Foundation.
#
import gdb
def symval(name):
return gdb.lookup_global_symbol(name).value()
def tid_to_gdb_thread(tid):
for thread in gdb.inferiors()[0].threads():
if thread.ptid[2] == tid:
return thread
else:
return None
def all_pcpus():
mp_maxid = symval("mp_maxid")
cpuid_to_pcpu = symval("cpuid_to_pcpu")
cpu = 0
while cpu <= mp_maxid:
pcpu = cpuid_to_pcpu[cpu]
if pcpu:
yield pcpu
cpu = cpu + 1
class acttrace(gdb.Command):
def __init__(self):
super(acttrace, self).__init__("acttrace", gdb.COMMAND_USER)
def invoke(self, arg, from_tty):
# Save the current thread so that we can switch back after.
curthread = gdb.selected_thread()
for pcpu in all_pcpus():
td = pcpu['pc_curthread']
tid = td['td_tid']
gdb_thread = tid_to_gdb_thread(tid)
if gdb_thread is None:
print("failed to find GDB thread with TID {}".format(tid))
else:
gdb_thread.switch()
p = td['td_proc']
print("Tracing command {} pid {} tid {} (CPU {})".format(
p['p_comm'], p['p_pid'], td['td_tid'], pcpu['pc_cpuid']))
gdb.execute("bt")
print()
curthread.switch()
# Registers the command with gdb, doesn't do anything.
acttrace()

View File

@ -215,13 +215,15 @@ echo
sed -ne '/^ Panic String: /{s//panic: /;p;}' $INFO
echo
# XXX: /bin/sh on 7.0+ is broken so we can't simply pipe the commands to
# kgdb via stdin and have to use a temporary file instead.
file=`mktemp /tmp/crashinfo.XXXXXX`
if [ $? -eq 0 ]; then
scriptdir=/usr/libexec/kgdb
echo "bt -full" >> $file
echo "source ${scriptdir}/acttrace.py" >> $file
echo "acttrace" >> $file
echo "quit" >> $file
${GDB%gdb}kgdb $KERNEL $VMCORE < $file
${GDB%gdb}kgdb -q $KERNEL $VMCORE < $file
rm -f $file
echo
fi