| #!/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' |
| } |