They have exactly the same conditions, no need to keep it separate for now. Signed-off-by: Michal Berger <michal.berger@intel.com> Change-Id: I6826ffe7b70e3d502f483cf1966930586a6f9f0b Reviewed-on: https://review.spdk.io/gerrit/c/spdk/spdk/+/16934 Reviewed-by: Jim Harris <james.r.harris@intel.com> Reviewed-by: Konrad Sztyber <konrad.sztyber@intel.com> Reviewed-by: Richael <richael.zhuang@arm.com> Tested-by: SPDK CI Jenkins <sys_sgci@intel.com>
162 lines
5.5 KiB
Bash
Executable File
162 lines
5.5 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
# SPDX-License-Identifier: BSD-3-Clause
|
|
# Copyright (C) 2020 Intel Corporation
|
|
# All rights reserved.
|
|
#
|
|
testdir=$(readlink -f "$(dirname "$0")")
|
|
rootdir=$(readlink -f "$testdir/../../")
|
|
|
|
source "$rootdir/test/common/autotest_common.sh"
|
|
source "$testdir/common.sh"
|
|
|
|
trap 'killprocess "$spdk_pid" || :; restore_cpufreq' EXIT
|
|
|
|
restore_cpufreq() {
|
|
local cpu
|
|
|
|
for cpu in "$spdk_main_core" "${cpus[@]}"; do
|
|
set_cpufreq "$cpu" "$main_core_min_freq" "$main_core_max_freq"
|
|
set_cpufreq_governor "$cpu" "$initial_main_core_governor"
|
|
done
|
|
}
|
|
|
|
update_main_core_cpufreq() {
|
|
map_cpufreq
|
|
|
|
main_core_driver=${cpufreq_drivers[spdk_main_core]}
|
|
main_core_governor=${cpufreq_governors[spdk_main_core]}
|
|
main_core_set_min_freq=${cpufreq_min_freqs[spdk_main_core]}
|
|
main_core_set_cur_freq=${cpufreq_cur_freqs[spdk_main_core]}
|
|
main_core_set_max_freq=${cpufreq_max_freqs[spdk_main_core]}
|
|
|
|
if ((${#main_core_freqs[@]} == 0)); then
|
|
main_core_freqs=("${!cpufreq_available_freqs[spdk_main_core]}")
|
|
main_core_max_freq=${main_core_freqs[0]}
|
|
main_core_min_freq=${main_core_freqs[-1]}
|
|
fi
|
|
if ((${#main_core_freqs_map[@]} == 0)); then
|
|
fold_list_onto_array main_core_freqs_map "${main_core_freqs[@]}"
|
|
fi
|
|
|
|
case "$main_core_driver" in
|
|
acpi-cpufreq) main_core_setspeed=${cpufreq_setspeed[spdk_main_core]} ;;
|
|
intel_pstate | intel_cpufreq) main_core_setspeed=$main_core_set_max_freq ;;
|
|
cppc_cpufreq) main_core_setspeed=${cpufreq_setspeed[spdk_main_core]} ;;
|
|
esac
|
|
|
|
local thread
|
|
for thread in "${!cpu_siblings[spdk_main_core]}"; do
|
|
((thread == spdk_main_core)) && continue # handled by DPDK/scheduler
|
|
# While assigning cpus to SPDK, we took every thread from given core,
|
|
# hence the cpufreq governor should already be properly set by the
|
|
# DPDK. So the only thing we need to take care of is {max,min} freq.
|
|
# The expectation here is to see actual freq drop on the main cpu in
|
|
# next iterations. Both, max and min, should be set to the same value.
|
|
set_cpufreq "$thread" "$main_core_set_min_freq" "$main_core_set_max_freq"
|
|
done
|
|
}
|
|
|
|
verify_dpdk_governor() {
|
|
xtrace_disable
|
|
|
|
map_cpus
|
|
# Run the app and see if DPDK's PM subsystem is doing the proper thing. Aim at the main core.
|
|
# What we expect to happen is based mainly on what cpufreq driver is in use. The main assumption
|
|
# is that with all SPDK threads being idle the governor should start lowering the frequency for
|
|
# the main core.
|
|
# - acpi-cpufreq:
|
|
# - governor set to userspace
|
|
# - lowering setspeed for the main core
|
|
# - having setspeed at lowest supported frequency
|
|
# - intel_pstate (active or passive)
|
|
# - governor set to performance
|
|
# - lowering max_freq and min_freq for the main core
|
|
# - having max_freq and min_freq at lowest supported frequency
|
|
# - cppc_cpufreq:
|
|
# - governor set to userspace
|
|
# - lowering setspeed for the main core
|
|
# - having setspeed at lowest supported frequency
|
|
|
|
local -g cpus
|
|
|
|
local dir_map
|
|
dir_map[0]="<"
|
|
dir_map[1]=">"
|
|
dir_map[2]="=="
|
|
|
|
fold_list_onto_array cpus $(parse_cpu_list <(echo "$spdk_cpus_csv"))
|
|
# Get rid of the main core
|
|
unset -v "cpus[spdk_main_core]"
|
|
|
|
local samples=0 all_set=0 dir=-1 old_main_core_setspeed=0
|
|
local old_main_core_set_cur_freq=0 first_main_core_set_cur_freq=0
|
|
|
|
exec_under_dynamic_scheduler "${SPDK_APP[@]}" -m "$spdk_cpumask" --main-core "$spdk_main_core"
|
|
|
|
while ((all_set == 0 && samples++ <= 50)); do
|
|
update_main_core_cpufreq
|
|
|
|
if [[ $main_core_setspeed == "<unsupported>" ]]; then
|
|
# governor hasn't taken over yet, skip this sample
|
|
printf 'Waiting for DPDK governor to take over...\n'
|
|
continue
|
|
fi
|
|
|
|
if ((main_core_setspeed > old_main_core_setspeed)); then
|
|
dir=1
|
|
elif ((main_core_setspeed < old_main_core_setspeed)); then
|
|
dir=0
|
|
elif ((main_core_setspeed == old_main_core_setspeed)); then
|
|
# Frequency didn't change, wait for a bit, but then fall to the main check to
|
|
# see if cur freq actually changed or not.
|
|
sleep 0.5s
|
|
fi
|
|
|
|
if ((first_main_core_set_cur_freq == 0)); then
|
|
first_main_core_set_cur_freq=$main_core_set_cur_freq
|
|
fi
|
|
|
|
case "$main_core_driver" in
|
|
acpi-cpufreq | cppc_cpufreq)
|
|
[[ $main_core_governor == userspace ]] \
|
|
&& [[ -n ${main_core_freqs_map[main_core_setspeed]} ]] \
|
|
&& ((main_core_setspeed == main_core_freqs[-1])) \
|
|
&& ((dir == 0))
|
|
;;
|
|
intel_pstate | intel_cpufreq)
|
|
[[ $main_core_governor == performance ]] \
|
|
&& [[ -n ${main_core_freqs_map[main_core_setspeed]} ]] \
|
|
&& ((main_core_setspeed == main_core_freqs[-1])) \
|
|
&& ((main_core_set_max_freq == main_core_set_min_freq)) \
|
|
&& ((dir == 0))
|
|
;;
|
|
esac && ((main_core_set_cur_freq < old_main_core_set_cur_freq)) && all_set=1
|
|
|
|
# Print stats after first sane sample was taken
|
|
if ((old_main_core_setspeed != 0 && dir != -1)); then
|
|
printf 'MAIN DPDK cpu%u current frequency at %u KHz (%u-%u KHz), set frequency %u KHz %s %u KHz\n' \
|
|
"$spdk_main_core" "$main_core_set_cur_freq" "$main_core_min_freq" "$main_core_max_freq" \
|
|
"$main_core_setspeed" "${dir_map[dir]}" "$old_main_core_setspeed"
|
|
else
|
|
printf 'Waiting for samples...\n'
|
|
fi
|
|
|
|
old_main_core_setspeed=$main_core_setspeed
|
|
old_main_core_set_cur_freq=$main_core_set_cur_freq
|
|
done
|
|
|
|
((all_set == 1))
|
|
|
|
printf 'Main cpu%u frequency dropped by %u%%\n' \
|
|
"$spdk_main_core" \
|
|
$(((first_main_core_set_cur_freq - main_core_set_cur_freq) * 100 / (first_main_core_set_cur_freq - main_core_min_freq)))
|
|
|
|
xtrace_restore
|
|
}
|
|
|
|
map_cpufreq
|
|
# Save initial scaling governor to restore it later on
|
|
initial_main_core_governor=${cpufreq_governors[spdk_main_core]}
|
|
|
|
verify_dpdk_governor
|