#!/bin/bash # SPDX-License-Identifier: GPL-2.0 SYSFS= prerequisite() { msg="skip all tests:" if [ $UID != 0 ]; then echo $msg must be run as root >&2 exit 0 fi taskset -p 01 $$ SYSFS=`mount -t sysfs | head -1 | awk '{ print $3 }'` if [ ! -d "$SYSFS" ]; then echo $msg sysfs is not mounted >&2 exit 0 fi if ! ls $SYSFS/devices/system/cpu/cpu* > /dev/null 2>&1; then echo $msg cpu hotplug is not supported >&2 exit 0 fi echo "CPU online/offline summary:" online_cpus=`cat $SYSFS/devices/system/cpu/online` online_max=${online_cpus##*-} if [[ "$online_cpus" = "$online_max" ]]; then echo "$msg: since there is only one cpu: $online_cpus" exit 0 fi present_cpus=`cat $SYSFS/devices/system/cpu/present` present_max=${present_cpus##*-} echo "present_cpus = $present_cpus present_max = $present_max" echo -e "\t Cpus in online state: $online_cpus" offline_cpus=`cat $SYSFS/devices/system/cpu/offline` if [[ "a$offline_cpus" = "a" ]]; then offline_cpus=0 else offline_max=${offline_cpus##*-} fi echo -e "\t Cpus in offline state: $offline_cpus" } # # list all hot-pluggable CPUs # hotpluggable_cpus() { local state=${1:-.\*} for cpu in $SYSFS/devices/system/cpu/cpu*; do if [ -f $cpu/online ] && grep -q $state $cpu/online; then echo ${cpu##/*/cpu} fi done } hotplaggable_offline_cpus() { hotpluggable_cpus 0 } hotpluggable_online_cpus() { hotpluggable_cpus 1 } cpu_is_online() { grep -q 1 $SYSFS/devices/system/cpu/cpu$1/online } cpu_is_offline() { grep -q 0 $SYSFS/devices/system/cpu/cpu$1/online } online_cpu() { echo 1 > $SYSFS/devices/system/cpu/cpu$1/online } offline_cpu() { echo 0 > $SYSFS/devices/system/cpu/cpu$1/online } online_cpu_expect_success() { local cpu=$1 if ! online_cpu $cpu; then echo $FUNCNAME $cpu: unexpected fail >&2 exit 1 elif ! cpu_is_online $cpu; then echo $FUNCNAME $cpu: unexpected offline >&2 exit 1 fi } online_cpu_expect_fail() { local cpu=$1 if online_cpu $cpu 2> /dev/null; then echo $FUNCNAME $cpu: unexpected success >&2 exit 1 elif ! cpu_is_offline $cpu; then echo $FUNCNAME $cpu: unexpected online >&2 exit 1 fi } offline_cpu_expect_success() { local cpu=$1 if ! offline_cpu $cpu; then echo $FUNCNAME $cpu: unexpected fail >&2 exit 1 elif ! cpu_is_offline $cpu; then echo $FUNCNAME $cpu: unexpected offline >&2 exit 1 fi } offline_cpu_expect_fail() { local cpu=$1 if offline_cpu $cpu 2> /dev/null; then echo $FUNCNAME $cpu: unexpected success >&2 exit 1 elif ! cpu_is_online $cpu; then echo $FUNCNAME $cpu: unexpected offline >&2 exit 1 fi } error=-12 allcpus=0 priority=0 online_cpus=0 online_max=0 offline_cpus=0 offline_max=0 present_cpus=0 present_max=0 while getopts e:ahp: opt; do case $opt in e) error=$OPTARG ;; a) allcpus=1 ;; h) echo "Usage $0 [ -a ] [ -e errno ] [ -p notifier-priority ]" echo -e "\t default offline one cpu" echo -e "\t run with -a option to offline all cpus" exit ;; p) priority=$OPTARG ;; esac done if ! [ "$error" -ge -4095 -a "$error" -lt 0 ]; then echo "error code must be -4095 <= errno < 0" >&2 exit 1 fi prerequisite # # Safe test (default) - offline and online one cpu # if [ $allcpus -eq 0 ]; then echo "Limited scope test: one hotplug cpu" echo -e "\t (leaves cpu in the original state):" echo -e "\t online to offline to online: cpu $online_max" offline_cpu_expect_success $online_max online_cpu_expect_success $online_max if [[ $offline_cpus -gt 0 ]]; then echo -e "\t offline to online to offline: cpu $present_max" online_cpu_expect_success $present_max offline_cpu_expect_success $present_max online_cpu $present_max fi exit 0 else echo "Full scope test: all hotplug cpus" echo -e "\t online all offline cpus" echo -e "\t offline all online cpus" echo -e "\t online all offline cpus" fi # # Online all hot-pluggable CPUs # for cpu in `hotplaggable_offline_cpus`; do online_cpu_expect_success $cpu done # # Offline all hot-pluggable CPUs # for cpu in `hotpluggable_online_cpus`; do offline_cpu_expect_success $cpu done # # Online all hot-pluggable CPUs again # for cpu in `hotplaggable_offline_cpus`; do online_cpu_expect_success $cpu done # # Test with cpu notifier error injection # DEBUGFS=`mount -t debugfs | head -1 | awk '{ print $3 }'` NOTIFIER_ERR_INJECT_DIR=$DEBUGFS/notifier-error-inject/cpu prerequisite_extra() { msg="skip extra tests:" /sbin/modprobe -q -r cpu-notifier-error-inject /sbin/modprobe -q cpu-notifier-error-inject priority=$priority if [ ! -d "$DEBUGFS" ]; then echo $msg debugfs is not mounted >&2 exit 0 fi if [ ! -d $NOTIFIER_ERR_INJECT_DIR ]; then echo $msg cpu-notifier-error-inject module is not available >&2 exit 0 fi } prerequisite_extra # # Offline all hot-pluggable CPUs # echo 0 > $NOTIFIER_ERR_INJECT_DIR/actions/CPU_DOWN_PREPARE/error for cpu in `hotpluggable_online_cpus`; do offline_cpu_expect_success $cpu done # # Test CPU hot-add error handling (offline => online) # echo $error > $NOTIFIER_ERR_INJECT_DIR/actions/CPU_UP_PREPARE/error for cpu in `hotplaggable_offline_cpus`; do online_cpu_expect_fail $cpu done # # Online all hot-pluggable CPUs # echo 0 > $NOTIFIER_ERR_INJECT_DIR/actions/CPU_UP_PREPARE/error for cpu in `hotplaggable_offline_cpus`; do online_cpu_expect_success $cpu done # # Test CPU hot-remove error handling (online => offline) # echo $error > $NOTIFIER_ERR_INJECT_DIR/actions/CPU_DOWN_PREPARE/error for cpu in `hotpluggable_online_cpus`; do offline_cpu_expect_fail $cpu done echo 0 > $NOTIFIER_ERR_INJECT_DIR/actions/CPU_DOWN_PREPARE/error /sbin/modprobe -q -r cpu-notifier-error-inject