mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
synced 2025-09-01 23:46:45 +00:00

When debugging, it can be difficult to quickly find the ftrace dump within the console log, which in turn makes it difficult to process it independent of the rest of the console output. This commit therefore copies the contents of the buffers into its own file to make it easier to locate and process the ftrace dump. The original ftrace dump is still available in the console log in cases because it can be more convenient to process it in situ, for example, for scripts that process console output as well as ftrace-dump data. Also handle the case of multiple ftrace dumps potentially showing up in the log. Example for a file like [1], it will extract as [2]. [1]: foo foo Dumping ftrace buffer: --------------------------------- blah blah --------------------------------- more bar baz Dumping ftrace buffer: --------------------------------- blah2 blah2 --------------------------------- bleh bleh [2]: Ftrace dump 1: blah blah Ftrace dump 2: blah2 blah2 [ paulmck: Fixed awk indentation, input up front. ] Signed-off-by: Joel Fernandes (Google) <joel@joelfernandes.org> Signed-off-by: Paul E. McKenney <paulmck@kernel.org> Signed-off-by: Frederic Weisbecker <frederic@kernel.org>
363 lines
8.0 KiB
Bash
Executable File
363 lines
8.0 KiB
Bash
Executable File
#!/bin/bash
|
|
# SPDX-License-Identifier: GPL-2.0+
|
|
#
|
|
# Shell functions for the rest of the scripts.
|
|
#
|
|
# Copyright (C) IBM Corporation, 2013
|
|
#
|
|
# Authors: Paul E. McKenney <paulmck@linux.ibm.com>
|
|
|
|
# bootparam_hotplug_cpu bootparam-string
|
|
#
|
|
# Returns 1 if the specified boot-parameter string tells rcutorture to
|
|
# test CPU-hotplug operations.
|
|
bootparam_hotplug_cpu () {
|
|
echo "$1" | grep -q "torture\.onoff_"
|
|
}
|
|
|
|
# checkarg --argname argtype $# arg mustmatch cannotmatch
|
|
#
|
|
# Checks the specified argument "arg" against the mustmatch and cannotmatch
|
|
# patterns.
|
|
checkarg () {
|
|
if test $3 -le 1
|
|
then
|
|
echo $1 needs argument $2 matching \"$5\"
|
|
usage
|
|
fi
|
|
if echo "$4" | grep -q -e "$5"
|
|
then
|
|
:
|
|
else
|
|
echo $1 $2 \"$4\" must match \"$5\"
|
|
usage
|
|
fi
|
|
if echo "$4" | grep -q -e "$6"
|
|
then
|
|
echo $1 $2 \"$4\" must not match \"$6\"
|
|
usage
|
|
fi
|
|
}
|
|
|
|
# configfrag_boot_params bootparam-string config-fragment-file
|
|
#
|
|
# Adds boot parameters from the .boot file, if any.
|
|
configfrag_boot_params () {
|
|
if test -r "$2.boot"
|
|
then
|
|
echo `grep -v '^#' "$2.boot" | tr '\012' ' '` $1
|
|
else
|
|
echo $1
|
|
fi
|
|
}
|
|
|
|
# configfrag_boot_cpus bootparam-string config-fragment-file config-cpus
|
|
#
|
|
# Decreases number of CPUs based on any nr_cpus= boot parameters specified.
|
|
configfrag_boot_cpus () {
|
|
local bootargs="`configfrag_boot_params "$1" "$2"`"
|
|
local nr_cpus
|
|
if echo "${bootargs}" | grep -q 'nr_cpus=[0-9]'
|
|
then
|
|
nr_cpus="`echo "${bootargs}" | sed -e 's/^.*nr_cpus=\([0-9]*\).*$/\1/'`"
|
|
if test "$3" -gt "$nr_cpus"
|
|
then
|
|
echo $nr_cpus
|
|
else
|
|
echo $3
|
|
fi
|
|
else
|
|
echo $3
|
|
fi
|
|
}
|
|
|
|
# configfrag_boot_maxcpus bootparam-string config-fragment-file config-cpus
|
|
#
|
|
# Decreases number of CPUs based on any maxcpus= boot parameters specified.
|
|
# This allows tests where additional CPUs come online later during the
|
|
# test run. However, the torture parameters will be set based on the
|
|
# number of CPUs initially present, so the scripting should schedule
|
|
# test runs based on the maxcpus= boot parameter controlling the initial
|
|
# number of CPUs instead of on the ultimate number of CPUs.
|
|
configfrag_boot_maxcpus () {
|
|
local bootargs="`configfrag_boot_params "$1" "$2"`"
|
|
local maxcpus
|
|
if echo "${bootargs}" | grep -q 'maxcpus=[0-9]'
|
|
then
|
|
maxcpus="`echo "${bootargs}" | sed -e 's/^.*maxcpus=\([0-9]*\).*$/\1/'`"
|
|
if test "$3" -gt "$maxcpus"
|
|
then
|
|
echo $maxcpus
|
|
else
|
|
echo $3
|
|
fi
|
|
else
|
|
echo $3
|
|
fi
|
|
}
|
|
|
|
# configfrag_hotplug_cpu config-fragment-file
|
|
#
|
|
# Returns 1 if the config fragment specifies hotplug CPU.
|
|
configfrag_hotplug_cpu () {
|
|
if test ! -r "$1"
|
|
then
|
|
echo Unreadable config fragment "$1" 1>&2
|
|
exit -1
|
|
fi
|
|
grep -q '^CONFIG_HOTPLUG_CPU=y$' "$1"
|
|
}
|
|
|
|
# get_starttime
|
|
#
|
|
# Returns a cookie identifying the current time.
|
|
get_starttime () {
|
|
awk 'BEGIN { print systime() }' < /dev/null
|
|
}
|
|
|
|
# get_starttime_duration starttime
|
|
#
|
|
# Given the return value from get_starttime, compute a human-readable
|
|
# string denoting the time since get_starttime.
|
|
get_starttime_duration () {
|
|
awk -v starttime=$1 '
|
|
BEGIN {
|
|
ts = systime() - starttime;
|
|
tm = int(ts / 60);
|
|
th = int(ts / 3600);
|
|
td = int(ts / 86400);
|
|
d = td;
|
|
h = th - td * 24;
|
|
m = tm - th * 60;
|
|
s = ts - tm * 60;
|
|
if (d >= 1)
|
|
printf "%dd %d:%02d:%02d\n", d, h, m, s
|
|
else if (h >= 1)
|
|
printf "%d:%02d:%02d\n", h, m, s
|
|
else if (m >= 1)
|
|
printf "%d:%02d.0\n", m, s
|
|
else
|
|
print s " seconds"
|
|
}' < /dev/null
|
|
}
|
|
|
|
# identify_boot_image qemu-cmd
|
|
#
|
|
# Returns the relative path to the kernel build image. This will be
|
|
# arch/<arch>/boot/bzImage or vmlinux if bzImage is not a target for the
|
|
# architecture, unless overridden with the TORTURE_BOOT_IMAGE environment
|
|
# variable.
|
|
identify_boot_image () {
|
|
if test -n "$TORTURE_BOOT_IMAGE"
|
|
then
|
|
echo $TORTURE_BOOT_IMAGE
|
|
else
|
|
case "$1" in
|
|
qemu-system-x86_64|qemu-system-i386)
|
|
echo arch/x86/boot/bzImage
|
|
;;
|
|
qemu-system-aarch64)
|
|
echo arch/arm64/boot/Image
|
|
;;
|
|
qemu-system-s390x)
|
|
echo arch/s390/boot/bzImage
|
|
;;
|
|
*)
|
|
echo vmlinux
|
|
;;
|
|
esac
|
|
fi
|
|
}
|
|
|
|
# identify_qemu builddir
|
|
#
|
|
# Returns our best guess as to which qemu command is appropriate for
|
|
# the kernel at hand. Override with the TORTURE_QEMU_CMD environment variable.
|
|
identify_qemu () {
|
|
local u="`file "$1"`"
|
|
if test -n "$TORTURE_QEMU_CMD"
|
|
then
|
|
echo $TORTURE_QEMU_CMD
|
|
elif echo $u | grep -q x86-64
|
|
then
|
|
echo qemu-system-x86_64
|
|
elif echo $u | grep -q "Intel 80386"
|
|
then
|
|
echo qemu-system-i386
|
|
elif echo $u | grep -q aarch64
|
|
then
|
|
echo qemu-system-aarch64
|
|
elif echo $u | grep -q 'IBM S/390'
|
|
then
|
|
echo qemu-system-s390x
|
|
elif uname -a | grep -q ppc64
|
|
then
|
|
echo qemu-system-ppc64
|
|
else
|
|
echo Cannot figure out what qemu command to use! 1>&2
|
|
echo file $1 output: $u
|
|
# Usually this will be one of /usr/bin/qemu-system-*
|
|
# Use TORTURE_QEMU_CMD environment variable or appropriate
|
|
# argument to top-level script.
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
# identify_qemu_append qemu-cmd
|
|
#
|
|
# Output arguments for the qemu "-append" string based on CPU type
|
|
# and the TORTURE_QEMU_INTERACTIVE environment variable.
|
|
identify_qemu_append () {
|
|
echo debug_boot_weak_hash
|
|
echo panic=-1
|
|
local console=ttyS0
|
|
case "$1" in
|
|
qemu-system-x86_64|qemu-system-i386)
|
|
echo selinux=0 initcall_debug debug
|
|
;;
|
|
qemu-system-aarch64)
|
|
console=ttyAMA0
|
|
;;
|
|
esac
|
|
if test -n "$TORTURE_QEMU_INTERACTIVE"
|
|
then
|
|
echo root=/dev/sda
|
|
else
|
|
echo console=$console
|
|
fi
|
|
}
|
|
|
|
# identify_qemu_args qemu-cmd serial-file
|
|
#
|
|
# Output arguments for qemu arguments based on the TORTURE_QEMU_MAC
|
|
# and TORTURE_QEMU_INTERACTIVE environment variables.
|
|
identify_qemu_args () {
|
|
local KVM_CPU=""
|
|
case "$1" in
|
|
qemu-system-x86_64)
|
|
KVM_CPU=kvm64
|
|
;;
|
|
qemu-system-i386)
|
|
KVM_CPU=kvm32
|
|
;;
|
|
esac
|
|
case "$1" in
|
|
qemu-system-x86_64|qemu-system-i386)
|
|
echo -machine q35,accel=kvm
|
|
echo -cpu ${KVM_CPU}
|
|
;;
|
|
qemu-system-aarch64)
|
|
echo -machine virt,gic-version=host -cpu host
|
|
;;
|
|
qemu-system-ppc64)
|
|
echo -M pseries -nodefaults
|
|
echo -device spapr-vscsi
|
|
if test -n "$TORTURE_QEMU_INTERACTIVE" -a -n "$TORTURE_QEMU_MAC"
|
|
then
|
|
echo -device spapr-vlan,netdev=net0,mac=$TORTURE_QEMU_MAC
|
|
echo -netdev bridge,br=br0,id=net0
|
|
fi
|
|
;;
|
|
esac
|
|
if test -n "$TORTURE_QEMU_INTERACTIVE"
|
|
then
|
|
echo -monitor stdio -serial pty -S
|
|
else
|
|
echo -serial file:$2
|
|
fi
|
|
}
|
|
|
|
# identify_qemu_vcpus
|
|
#
|
|
# Returns the number of virtual CPUs available to the aggregate of the
|
|
# guest OSes.
|
|
identify_qemu_vcpus () {
|
|
getconf _NPROCESSORS_ONLN
|
|
}
|
|
|
|
# print_bug
|
|
#
|
|
# Prints "BUG: " in red followed by remaining arguments
|
|
print_bug () {
|
|
printf '\033[031mBUG: \033[m'
|
|
echo $*
|
|
}
|
|
|
|
# print_warning
|
|
#
|
|
# Prints "WARNING: " in yellow followed by remaining arguments
|
|
print_warning () {
|
|
printf '\033[033mWARNING: \033[m'
|
|
echo $*
|
|
}
|
|
|
|
# specify_qemu_cpus qemu-cmd qemu-args #cpus
|
|
#
|
|
# Appends a string containing "-smp XXX" to qemu-args, unless the incoming
|
|
# qemu-args already contains "-smp".
|
|
specify_qemu_cpus () {
|
|
local nt;
|
|
|
|
if echo $2 | grep -q -e -smp
|
|
then
|
|
echo $2
|
|
else
|
|
case "$1" in
|
|
qemu-system-x86_64|qemu-system-i386|qemu-system-aarch64)
|
|
echo $2 -smp $3
|
|
;;
|
|
qemu-system-ppc64)
|
|
nt="`lscpu | sed -n 's/^Thread(s) per core:\s*//p'`"
|
|
echo $2 -smp cores=`expr \( $3 + $nt - 1 \) / $nt`,threads=$nt
|
|
;;
|
|
esac
|
|
fi
|
|
}
|
|
|
|
# specify_qemu_net qemu-args
|
|
#
|
|
# Appends a string containing "-net none" to qemu-args, unless the incoming
|
|
# qemu-args already contains "-smp" or unless the TORTURE_QEMU_INTERACTIVE
|
|
# environment variable is set, in which case the string that is be added is
|
|
# instead "-net nic -net user".
|
|
specify_qemu_net () {
|
|
if echo $1 | grep -q -e -net
|
|
then
|
|
echo $1
|
|
elif test -n "$TORTURE_QEMU_INTERACTIVE"
|
|
then
|
|
echo $1 -net nic -net user
|
|
else
|
|
echo $1 -net none
|
|
fi
|
|
}
|
|
|
|
# Extract the ftrace output from the console log output
|
|
# The ftrace output in the original logs look like:
|
|
# Dumping ftrace buffer:
|
|
# ---------------------------------
|
|
# [...]
|
|
# ---------------------------------
|
|
extract_ftrace_from_console() {
|
|
awk < "$1" '
|
|
|
|
/Dumping ftrace buffer:/ {
|
|
buffer_count++
|
|
print "Ftrace dump " buffer_count ":"
|
|
capture = 1
|
|
next
|
|
}
|
|
|
|
/---------------------------------/ {
|
|
if(capture == 1) {
|
|
capture = 2
|
|
next
|
|
} else if(capture == 2) {
|
|
capture = 0
|
|
print ""
|
|
}
|
|
}
|
|
|
|
capture == 2'
|
|
}
|