* [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation
@ 2026-05-07 11:11 Piotr Kubaj
2026-05-07 11:57 ` [LTP] " linuxtestproject.agent
` (3 more replies)
0 siblings, 4 replies; 5+ messages in thread
From: Piotr Kubaj @ 2026-05-07 11:11 UTC (permalink / raw)
To: ltp; +Cc: helena.anna.dubel, tomasz.ossowski, rafael.j.wysocki,
daniel.niestepski
Runs various sanity checks for intel_pstate cpufreq sysfs interface.
Checks whether intel_pstate driver is used, whether it is possible to
switch scaling_governor to powersave and performance, whether it is
possible to disable turbo frequency and whether it is possible to change
scaling_min_freq and scaling_max_freq to its minimal valid and maximal
valid values.
Requires root to write to /sys/devices/system/cpu/intel_pstate/ (status,
no_turbo) and per-CPU /sys/devices/system/cpu/cpu*\/cpufreq/ attributes
(scaling_governor, scaling_min_freq, scaling_max_freq).
Motivation: LTP has no regression coverage for the intel_pstate cpufreq
sysfs interface, so breakage in any of these attributes (driver
exposure, governor switching, active/passive mode transitions, no_turbo
toggling or scaling_{min,max}_freq writes) can reach users undetected.
This test exercises the interface end-to-end to catch such regressions.
Signed-off-by: Piotr Kubaj <piotr.kubaj@intel.com>
---
Address further review:
1. better commit message.
2. drop SAFE_READ().
3. reset setup_done.
runtest/power_management_tests | 1 +
testcases/kernel/power_management/.gitignore | 1 +
testcases/kernel/power_management/cpufreq.c | 410 +++++++++++++++++++
3 files changed, 412 insertions(+)
create mode 100644 testcases/kernel/power_management/.gitignore
create mode 100644 testcases/kernel/power_management/cpufreq.c
diff --git a/runtest/power_management_tests b/runtest/power_management_tests
index b670da6ec..0bc7dd0c8 100644
--- a/runtest/power_management_tests
+++ b/runtest/power_management_tests
@@ -1,4 +1,5 @@
#POWER_MANAGEMENT
+cpufreq cpufreq
runpwtests03 runpwtests03.sh
runpwtests04 runpwtests04.sh
runpwtests06 runpwtests06.sh
diff --git a/testcases/kernel/power_management/.gitignore b/testcases/kernel/power_management/.gitignore
new file mode 100644
index 000000000..7f34e4c2f
--- /dev/null
+++ b/testcases/kernel/power_management/.gitignore
@@ -0,0 +1 @@
+cpufreq
diff --git a/testcases/kernel/power_management/cpufreq.c b/testcases/kernel/power_management/cpufreq.c
new file mode 100644
index 000000000..d157006ba
--- /dev/null
+++ b/testcases/kernel/power_management/cpufreq.c
@@ -0,0 +1,410 @@
+// SPDX-License-Identifier: GPL-2.0-or-later
+/*
+ * Copyright (c) 2026 Piotr Kubaj <piotr.kubaj@intel.com>
+ */
+
+/*\
+ * Runs various sanity checks for intel_pstate cpufreq sysfs interface.
+ *
+ * Checks whether intel_pstate driver is used, whether it is possible to switch
+ * scaling_governor to powersave and performance, whether it is possible
+ * to disable turbo frequency and whether it is possible to change
+ * scaling_min_freq and scaling_max_freq to its minimal valid and maximal valid
+ * values.
+ *
+ * Requires root to write to /sys/devices/system/cpu/intel_pstate/ (status,
+ * no_turbo) and per-CPU /sys/devices/system/cpu/cpu*\/cpufreq/ attributes
+ * (scaling_governor, scaling_min_freq, scaling_max_freq).
+ *
+ * Motivation: LTP has no regression coverage for the intel_pstate cpufreq
+ * sysfs interface, so breakage in any of these attributes (driver exposure,
+ * governor switching, active/passive mode transitions, no_turbo toggling or
+ * scaling_{min,max}_freq writes) can reach users undetected. This test
+ * exercises the interface end-to-end to catch such regressions.
+ */
+
+#include "tst_test.h"
+
+static bool *online;
+static bool setup_done;
+static char intel_pstate_status[16];
+static char (*previous_scaling_governor)[16];
+static int no_turbo, nproc;
+static long *previous_scaling_max_freq, *previous_scaling_min_freq;
+
+static void cleanup(void)
+{
+ char path[PATH_MAX];
+
+ if (!setup_done) {
+ free(online);
+ free(previous_scaling_max_freq);
+ free(previous_scaling_min_freq);
+ free(previous_scaling_governor);
+ return;
+ }
+
+ SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/status", "%s", intel_pstate_status);
+ SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/no_turbo", "%d", no_turbo);
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i);
+ SAFE_FILE_PRINTF(path, "%s", previous_scaling_governor[i]);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", previous_scaling_max_freq[i]);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", previous_scaling_min_freq[i]);
+ }
+
+ free(online);
+ free(previous_scaling_max_freq);
+ free(previous_scaling_min_freq);
+ free(previous_scaling_governor);
+ setup_done = false;
+}
+
+static void setup(void)
+{
+ char path[PATH_MAX];
+
+ if (access("/sys/devices/system/cpu/intel_pstate/status", F_OK) == -1)
+ tst_brk(TCONF, "intel_pstate driver not active");
+
+ SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/no_turbo", "%d", &no_turbo);
+ SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/status", "%15s", intel_pstate_status);
+
+ nproc = tst_ncpus_conf();
+ online = SAFE_CALLOC(nproc, sizeof(*online));
+ previous_scaling_max_freq = SAFE_CALLOC(nproc, sizeof(*previous_scaling_max_freq));
+ previous_scaling_min_freq = SAFE_CALLOC(nproc, sizeof(*previous_scaling_min_freq));
+ previous_scaling_governor = SAFE_CALLOC(nproc, sizeof(*previous_scaling_governor));
+ online[0] = true;
+ for (int i = 1; i < nproc; i++) {
+ int tmp;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/online", i);
+ SAFE_FILE_SCANF(path, "%d", &tmp);
+ online[i] = (bool)tmp;
+ }
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i);
+ SAFE_FILE_SCANF(path, "%15s", previous_scaling_governor[i]);
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i);
+ SAFE_FILE_SCANF(path, "%ld", &previous_scaling_max_freq[i]);
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i);
+ SAFE_FILE_SCANF(path, "%ld", &previous_scaling_min_freq[i]);
+ }
+
+ SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/status", "active");
+
+ setup_done = true;
+}
+
+static void run(void)
+{
+ const char * const cpufreq_nodes[] = {
+ "affected_cpus",
+ "cpuinfo_max_freq",
+ "cpuinfo_min_freq",
+ "cpuinfo_transition_latency",
+ "related_cpus",
+ "scaling_available_governors",
+ "scaling_cur_freq",
+ "scaling_driver",
+ "scaling_governor",
+ "scaling_max_freq",
+ "scaling_min_freq",
+ "scaling_setspeed",
+ NULL
+ };
+ char contents[256] = {0}, path[PATH_MAX], path_cpuinfo_min_freq[PATH_MAX];
+ int no_turbo_val;
+ long cpuinfo_max_freq = 0, cpuinfo_min_freq = 0, scaling_max_freq, scaling_min_freq;
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_driver", i);
+
+ SAFE_FILE_SCANF(path, "%255s", contents);
+ tst_res(TDEBUG, "Checking whether %s is \"intel_pstate\"", path);
+ if (strstr(contents, "intel_pstate")) {
+ tst_res(TPASS, "%s is intel_pstate", path);
+ } else {
+ tst_res(TINFO, "%s contains: %s", path, contents);
+ tst_res(TFAIL, "%s is not intel_pstate", path);
+ }
+
+ for (int j = 0; cpufreq_nodes[j]; j++) {
+ struct stat stats;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/%s", i, cpufreq_nodes[j]);
+ tst_res(TDEBUG, "Checking whether %s is a regular file", path);
+ SAFE_STAT(path, &stats);
+ if (!S_ISREG(stats.st_mode)) {
+ tst_res(TINFO, "%s mode: %o", path, stats.st_mode);
+ tst_res(TFAIL, "%s is not a regular file", path);
+ } else {
+ tst_res(TPASS, "%s is a regular file", path);
+ }
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_governors", i);
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255[^\n]", contents);
+
+ tst_res(TDEBUG, "Checking whether %s contains \"performance\" and \"powersave\"", path);
+ if (strstr(contents, "performance") && strstr(contents, "powersave")) {
+ tst_res(TPASS, "%s contains: %s", path, contents);
+ } else {
+ tst_res(TINFO, "%s contains: %s", path, contents);
+ tst_res(TFAIL, "%s is not \"performance powersave\"", path);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i);
+ tst_res(TDEBUG, "Checking %s", path);
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+ if (strstr(contents, "performance")) {
+ tst_res(TDEBUG, "%s is \"performance\"", path);
+ tst_res(TDEBUG, "Checking whether %s can be switched to \"powersave\"", path);
+ SAFE_FILE_PRINTF(path, "powersave");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+ if (strstr(contents, "powersave"))
+ tst_res(TPASS, "Changing scaling_governor from performance to powersave succeeded");
+ else
+ tst_res(TFAIL, "%s: failed to change scaling_governor from performance to powersave, current scaling_governor: %s", path, contents);
+
+ } else if (strstr(contents, "powersave")) {
+ tst_res(TDEBUG, "%s is \"powersave\"", path);
+ tst_res(TDEBUG, "Checking whether %s can be switched to \"performance\"", path);
+ SAFE_FILE_PRINTF(path, "performance");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+ if (strstr(contents, "performance"))
+ tst_res(TPASS, "Changing scaling_governor from powersave to performance succeeded");
+ else
+ tst_res(TFAIL, "%s: failed to change scaling_governor from powersave to performance, current scaling_governor: %s", path, contents);
+
+ } else {
+ tst_brk(TBROK, "Unknown scaling_governor: %s", contents);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i);
+ snprintf(path_cpuinfo_min_freq, sizeof(path_cpuinfo_min_freq), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i);
+ tst_res(TDEBUG, "Checking whether %s is bigger than cpuinfo_min_freq", path);
+ SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq);
+
+ SAFE_FILE_SCANF(path_cpuinfo_min_freq, "%ld", &cpuinfo_min_freq);
+
+ if (cpuinfo_min_freq > cpuinfo_max_freq)
+ tst_res(TFAIL, "Value in %s: %ld, should be bigger than cpuinfo_min_freq: %ld", path, cpuinfo_max_freq, cpuinfo_min_freq);
+ else
+ tst_res(TPASS, "Value in %s: %ld, is bigger than cpuinfo_min_freq: %ld", path, cpuinfo_max_freq, cpuinfo_min_freq);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/intel_pstate/status");
+ tst_res(TDEBUG, "Checking whether %s contains \"active\"", path);
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+
+ if (strstr(contents, "active"))
+ tst_res(TPASS, "%s is \"active\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"active\", but %s", path, contents);
+
+ tst_res(TDEBUG, "Checking whether %s can be switched to \"passive\"", path);
+ SAFE_FILE_PRINTF(path, "passive");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+
+ if (strstr(contents, "passive"))
+ tst_res(TPASS, "%s is \"passive\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"passive\", but %s", path, contents);
+
+ tst_res(TDEBUG, "Checking whether %s can be switched back to \"active\"", path);
+ SAFE_FILE_PRINTF(path, "active");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+
+ if (strstr(contents, "active"))
+ tst_res(TPASS, "%s is \"active\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"active\", but %s", path, contents);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/intel_pstate/no_turbo");
+ tst_res(TDEBUG, "Checking whether %s can be switched to 1", path);
+ SAFE_FILE_PRINTF(path, "1");
+ SAFE_FILE_SCANF(path, "%d", &no_turbo_val);
+
+ if (no_turbo_val == 1)
+ tst_res(TPASS, "%s is \"1\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"1\", but %d", path, no_turbo_val);
+
+ tst_res(TDEBUG, "Checking whether %s can be switched back to 0", path);
+ SAFE_FILE_PRINTF(path, "0");
+ SAFE_FILE_SCANF(path, "%d", &no_turbo_val);
+
+ if (no_turbo_val == 0)
+ tst_res(TPASS, "%s is \"0\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"0\", but %d", path, no_turbo_val);
+
+ tst_res(TDEBUG, "Checking whether scaling_available_governors contains \"performance\" and \"schedutil\" after switching /sys/devices/system/cpu/intel_pstate/status to \"passive\"");
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/intel_pstate/status");
+ SAFE_FILE_PRINTF(path, "passive");
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_governors", i);
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255[^\n]", contents);
+
+ tst_res(TDEBUG, "Checking whether %s contains \"performance\" and \"schedutil\"", path);
+ if (strstr(contents, "performance") && strstr(contents, "schedutil")) {
+ tst_res(TPASS, "%s contains: %s", path, contents);
+ } else {
+ tst_res(TINFO, "%s contains: %s", path, contents);
+ tst_res(TFAIL, "%s does not contain \"performance\" and \"schedutil\"", path);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i);
+ tst_res(TDEBUG, "Checking whether %s can be changed to performance", path);
+ SAFE_FILE_PRINTF(path, "performance");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+
+ if (strstr(contents, "performance"))
+ tst_res(TPASS, "%s is \"performance\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"performance\", but %s", path, contents);
+
+ tst_res(TDEBUG, "Checking whether %s can be changed to schedutil", path);
+ SAFE_FILE_PRINTF(path, "schedutil");
+ memset(contents, 0, sizeof(contents));
+ SAFE_FILE_SCANF(path, "%255s", contents);
+
+ if (strstr(contents, "schedutil"))
+ tst_res(TPASS, "%s is \"schedutil\"", path);
+ else
+ tst_res(TFAIL, "%s is not \"schedutil\", but %s", path, contents);
+ }
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i);
+ tst_res(TDEBUG, "Checking whether %s can be assigned to scaling_max_freq and scaling_min_freq", path);
+ SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i);
+ SAFE_FILE_SCANF(path, "%ld", &cpuinfo_min_freq);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", cpuinfo_max_freq);
+ SAFE_FILE_SCANF(path, "%ld", &scaling_max_freq);
+
+ if (cpuinfo_max_freq < scaling_max_freq) {
+ tst_res(TINFO, "cpuinfo_max_freq: %ld", cpuinfo_max_freq);
+ tst_res(TINFO, "scaling_max_freq: %ld", scaling_max_freq);
+ tst_res(TFAIL, "Failure setting %s", path);
+ } else {
+ tst_res(TPASS, "Successfully set up %s", path);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", cpuinfo_max_freq);
+ SAFE_FILE_SCANF(path, "%ld", &scaling_min_freq);
+
+ if (cpuinfo_max_freq < scaling_min_freq) {
+ tst_res(TINFO, "cpuinfo_max_freq: %ld", cpuinfo_max_freq);
+ tst_res(TINFO, "scaling_min_freq: %ld", scaling_min_freq);
+ tst_res(TFAIL, "Failure setting %s", path);
+ } else {
+ tst_res(TPASS, "Successfully set up %s", path);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", previous_scaling_min_freq[i]);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", previous_scaling_max_freq[i]);
+ }
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i);
+ tst_res(TDEBUG, "Checking whether %s can be assigned to scaling_max_freq and scaling_min_freq", path);
+ SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i);
+ SAFE_FILE_SCANF(path, "%ld", &cpuinfo_min_freq);
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", cpuinfo_min_freq);
+ SAFE_FILE_SCANF(path, "%ld", &scaling_min_freq);
+
+ if (cpuinfo_min_freq > scaling_min_freq) {
+ tst_res(TINFO, "cpuinfo_min_freq: %ld", cpuinfo_min_freq);
+ tst_res(TINFO, "scaling_min_freq: %ld", scaling_min_freq);
+ tst_res(TFAIL, "Failure setting %s", path);
+ } else {
+ tst_res(TPASS, "Successfully set up %s", path);
+ }
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i);
+ SAFE_FILE_PRINTF(path, "%ld", cpuinfo_min_freq);
+ SAFE_FILE_SCANF(path, "%ld", &scaling_max_freq);
+
+ if (cpuinfo_min_freq > scaling_max_freq) {
+ tst_res(TINFO, "cpuinfo_min_freq: %ld", cpuinfo_min_freq);
+ tst_res(TINFO, "scaling_max_freq: %ld", scaling_max_freq);
+ tst_res(TFAIL, "Failure setting %s", path);
+ } else {
+ tst_res(TPASS, "Successfully set up %s", path);
+ }
+ }
+
+ SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/status", "active");
+
+ for (int i = 0; i < nproc; i++) {
+ if (!online[i])
+ continue;
+
+ snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i);
+ SAFE_FILE_PRINTF(path, "%s", previous_scaling_governor[i]);
+ }
+}
+
+static struct tst_test test = {
+ .cleanup = cleanup,
+ .needs_kconfigs = (const char *const []) {
+ "CONFIG_X86_INTEL_PSTATE",
+ NULL
+ },
+ .needs_root = 1,
+ .setup = setup,
+ .supported_archs = (const char *const []) {
+ "x86",
+ "x86_64",
+ NULL
+ },
+ .test_all = run
+};
--
2.47.3
---------------------------------------------------------------------
Intel Technology Poland sp. z o.o.
ul. Slowackiego 173 | 80-298 Gdansk | Sad Rejonowy Gdansk Polnoc | VII Wydzial Gospodarczy Krajowego Rejestru Sadowego - KRS 101882 | NIP 957-07-52-316 | Kapital zakladowy 200.000 PLN.
Spolka oswiadcza, ze posiada status duzego przedsiebiorcy w rozumieniu ustawy z dnia 8 marca 2013 r. o przeciwdzialaniu nadmiernym opoznieniom w transakcjach handlowych.
Ta wiadomosc wraz z zalacznikami jest przeznaczona dla okreslonego adresata i moze zawierac informacje poufne. W razie przypadkowego otrzymania tej wiadomosci, prosimy o powiadomienie nadawcy oraz trwale jej usuniecie; jakiekolwiek przegladanie lub rozpowszechnianie jest zabronione.
This e-mail and any attachments may contain confidential material for the sole use of the intended recipient(s). If you are not the intended recipient, please contact the sender and delete all copies; any review or distribution by others is strictly prohibited.
--
Mailing list info: https://lists.linux.it/listinfo/ltp
^ permalink raw reply related [flat|nested] 5+ messages in thread* Re: [LTP] cpufreq.c: add new test for cpufreq sysfs interface validation 2026-05-07 11:11 [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation Piotr Kubaj @ 2026-05-07 11:57 ` linuxtestproject.agent 2026-05-27 7:27 ` [LTP] [PATCH v8] " Kubaj, Piotr ` (2 subsequent siblings) 3 siblings, 0 replies; 5+ messages in thread From: linuxtestproject.agent @ 2026-05-07 11:57 UTC (permalink / raw) To: Piotr Kubaj; +Cc: ltp, LTP AI Reviewer Hi Piotr, On Thu, 7 May 2026, Piotr Kubaj wrote: > cpufreq.c: add new test for cpufreq sysfs interface validation Reviewed-by: LTP AI Reviewer <ltp-ai@noreply.github.com> --- Note: Our agent completed the review of the patch. The full review can be found at: https://patchwork.ozlabs.org/project/ltp/list/?series=503158 The agent can sometimes produce false positives although often its findings are genuine. If you find issues with the review, please comment this email or ignore the suggestions. Regards, LTP AI Reviewer -- Mailing list info: https://lists.linux.it/listinfo/ltp ^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation 2026-05-07 11:11 [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation Piotr Kubaj 2026-05-07 11:57 ` [LTP] " linuxtestproject.agent @ 2026-05-27 7:27 ` Kubaj, Piotr 2026-06-03 12:27 ` Cyril Hrubis 2026-06-11 7:13 ` Andrea Cervesato via ltp 3 siblings, 0 replies; 5+ messages in thread From: Kubaj, Piotr @ 2026-05-27 7:27 UTC (permalink / raw) To: ltp@lists.linux.it Cc: Dubel, Helena Anna, Ossowski, Tomasz, Wysocki, Rafael J, Niestepski, Daniel Ping. AI bot says it's ok, no other review. I guess it's ready to merge? 2026-05-07 (木) の 13:11 +0200 に Piotr Kubaj さんは書きました: > Runs various sanity checks for intel_pstate cpufreq sysfs interface. > > Checks whether intel_pstate driver is used, whether it is possible to > switch scaling_governor to powersave and performance, whether it is > possible to disable turbo frequency and whether it is possible to > change > scaling_min_freq and scaling_max_freq to its minimal valid and > maximal > valid values. > > Requires root to write to /sys/devices/system/cpu/intel_pstate/ > (status, > no_turbo) and per-CPU /sys/devices/system/cpu/cpu*\/cpufreq/ > attributes > (scaling_governor, scaling_min_freq, scaling_max_freq). > > Motivation: LTP has no regression coverage for the intel_pstate > cpufreq > sysfs interface, so breakage in any of these attributes (driver > exposure, governor switching, active/passive mode transitions, > no_turbo > toggling or scaling_{min,max}_freq writes) can reach users > undetected. > This test exercises the interface end-to-end to catch such > regressions. > > Signed-off-by: Piotr Kubaj <piotr.kubaj@intel.com> > --- > Address further review: > 1. better commit message. > 2. drop SAFE_READ(). > 3. reset setup_done. > runtest/power_management_tests | 1 + > testcases/kernel/power_management/.gitignore | 1 + > testcases/kernel/power_management/cpufreq.c | 410 > +++++++++++++++++++ > 3 files changed, 412 insertions(+) > create mode 100644 testcases/kernel/power_management/.gitignore > create mode 100644 testcases/kernel/power_management/cpufreq.c > > diff --git a/runtest/power_management_tests > b/runtest/power_management_tests > index b670da6ec..0bc7dd0c8 100644 > --- a/runtest/power_management_tests > +++ b/runtest/power_management_tests > @@ -1,4 +1,5 @@ > #POWER_MANAGEMENT > +cpufreq cpufreq > runpwtests03 runpwtests03.sh > runpwtests04 runpwtests04.sh > runpwtests06 runpwtests06.sh > diff --git a/testcases/kernel/power_management/.gitignore > b/testcases/kernel/power_management/.gitignore > new file mode 100644 > index 000000000..7f34e4c2f > --- /dev/null > +++ b/testcases/kernel/power_management/.gitignore > @@ -0,0 +1 @@ > +cpufreq > diff --git a/testcases/kernel/power_management/cpufreq.c > b/testcases/kernel/power_management/cpufreq.c > new file mode 100644 > index 000000000..d157006ba > --- /dev/null > +++ b/testcases/kernel/power_management/cpufreq.c > @@ -0,0 +1,410 @@ > +// SPDX-License-Identifier: GPL-2.0-or-later > +/* > + * Copyright (c) 2026 Piotr Kubaj <piotr.kubaj@intel.com> > + */ > + > +/*\ > + * Runs various sanity checks for intel_pstate cpufreq sysfs > interface. > + * > + * Checks whether intel_pstate driver is used, whether it is > possible to switch > + * scaling_governor to powersave and performance, whether it is > possible > + * to disable turbo frequency and whether it is possible to change > + * scaling_min_freq and scaling_max_freq to its minimal valid and > maximal valid > + * values. > + * > + * Requires root to write to /sys/devices/system/cpu/intel_pstate/ > (status, > + * no_turbo) and per-CPU /sys/devices/system/cpu/cpu*\/cpufreq/ > attributes > + * (scaling_governor, scaling_min_freq, scaling_max_freq). > + * > + * Motivation: LTP has no regression coverage for the intel_pstate > cpufreq > + * sysfs interface, so breakage in any of these attributes (driver > exposure, > + * governor switching, active/passive mode transitions, no_turbo > toggling or > + * scaling_{min,max}_freq writes) can reach users undetected. This > test > + * exercises the interface end-to-end to catch such regressions. > + */ > + > +#include "tst_test.h" > + > +static bool *online; > +static bool setup_done; > +static char intel_pstate_status[16]; > +static char (*previous_scaling_governor)[16]; > +static int no_turbo, nproc; > +static long *previous_scaling_max_freq, *previous_scaling_min_freq; > + > +static void cleanup(void) > +{ > + char path[PATH_MAX]; > + > + if (!setup_done) { > + free(online); > + free(previous_scaling_max_freq); > + free(previous_scaling_min_freq); > + free(previous_scaling_governor); > + return; > + } > + > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/statu > s", "%s", intel_pstate_status); > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/no_tu > rbo", "%d", no_turbo); > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i); > + SAFE_FILE_PRINTF(path, "%s", > previous_scaling_governor[i]); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", > previous_scaling_max_freq[i]); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", > previous_scaling_min_freq[i]); > + } > + > + free(online); > + free(previous_scaling_max_freq); > + free(previous_scaling_min_freq); > + free(previous_scaling_governor); > + setup_done = false; > +} > + > +static void setup(void) > +{ > + char path[PATH_MAX]; > + > + if (access("/sys/devices/system/cpu/intel_pstate/status", > F_OK) == -1) > + tst_brk(TCONF, "intel_pstate driver not active"); > + > + SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/no_tur > bo", "%d", &no_turbo); > + SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/status > ", "%15s", intel_pstate_status); > + > + nproc = tst_ncpus_conf(); > + online = SAFE_CALLOC(nproc, sizeof(*online)); > + previous_scaling_max_freq = SAFE_CALLOC(nproc, > sizeof(*previous_scaling_max_freq)); > + previous_scaling_min_freq = SAFE_CALLOC(nproc, > sizeof(*previous_scaling_min_freq)); > + previous_scaling_governor = SAFE_CALLOC(nproc, > sizeof(*previous_scaling_governor)); > + online[0] = true; > + for (int i = 1; i < nproc; i++) { > + int tmp; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/online", i); > + SAFE_FILE_SCANF(path, "%d", &tmp); > + online[i] = (bool)tmp; > + } > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i); > + SAFE_FILE_SCANF(path, "%15s", > previous_scaling_governor[i]); > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i); > + SAFE_FILE_SCANF(path, "%ld", > &previous_scaling_max_freq[i]); > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i); > + SAFE_FILE_SCANF(path, "%ld", > &previous_scaling_min_freq[i]); > + } > + > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/statu > s", "active"); > + > + setup_done = true; > +} > + > +static void run(void) > +{ > + const char * const cpufreq_nodes[] = { > + "affected_cpus", > + "cpuinfo_max_freq", > + "cpuinfo_min_freq", > + "cpuinfo_transition_latency", > + "related_cpus", > + "scaling_available_governors", > + "scaling_cur_freq", > + "scaling_driver", > + "scaling_governor", > + "scaling_max_freq", > + "scaling_min_freq", > + "scaling_setspeed", > + NULL > + }; > + char contents[256] = {0}, path[PATH_MAX], > path_cpuinfo_min_freq[PATH_MAX]; > + int no_turbo_val; > + long cpuinfo_max_freq = 0, cpuinfo_min_freq = 0, > scaling_max_freq, scaling_min_freq; > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_driver", i); > + > + SAFE_FILE_SCANF(path, "%255s", contents); > + tst_res(TDEBUG, "Checking whether %s is > \"intel_pstate\"", path); > + if (strstr(contents, "intel_pstate")) { > + tst_res(TPASS, "%s is intel_pstate", path); > + } else { > + tst_res(TINFO, "%s contains: %s", path, > contents); > + tst_res(TFAIL, "%s is not intel_pstate", > path); > + } > + > + for (int j = 0; cpufreq_nodes[j]; j++) { > + struct stat stats; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/%s", i, cpufreq_nodes[j]); > + tst_res(TDEBUG, "Checking whether %s is a > regular file", path); > + SAFE_STAT(path, &stats); > + if (!S_ISREG(stats.st_mode)) { > + tst_res(TINFO, "%s mode: %o", path, > stats.st_mode); > + tst_res(TFAIL, "%s is not a regular > file", path); > + } else { > + tst_res(TPASS, "%s is a regular > file", path); > + } > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_governors", > i); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255[^\n]", contents); > + > + tst_res(TDEBUG, "Checking whether %s contains > \"performance\" and \"powersave\"", path); > + if (strstr(contents, "performance") && > strstr(contents, "powersave")) { > + tst_res(TPASS, "%s contains: %s", path, > contents); > + } else { > + tst_res(TINFO, "%s contains: %s", path, > contents); > + tst_res(TFAIL, "%s is not \"performance > powersave\"", path); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i); > + tst_res(TDEBUG, "Checking %s", path); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + if (strstr(contents, "performance")) { > + tst_res(TDEBUG, "%s is \"performance\"", > path); > + tst_res(TDEBUG, "Checking whether %s can be > switched to \"powersave\"", path); > + SAFE_FILE_PRINTF(path, "powersave"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + if (strstr(contents, "powersave")) > + tst_res(TPASS, "Changing > scaling_governor from performance to powersave succeeded"); > + else > + tst_res(TFAIL, "%s: failed to change > scaling_governor from performance to powersave, current > scaling_governor: %s", path, contents); > + > + } else if (strstr(contents, "powersave")) { > + tst_res(TDEBUG, "%s is \"powersave\"", > path); > + tst_res(TDEBUG, "Checking whether %s can be > switched to \"performance\"", path); > + SAFE_FILE_PRINTF(path, "performance"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + if (strstr(contents, "performance")) > + tst_res(TPASS, "Changing > scaling_governor from powersave to performance succeeded"); > + else > + tst_res(TFAIL, "%s: failed to change > scaling_governor from powersave to performance, current > scaling_governor: %s", path, contents); > + > + } else { > + tst_brk(TBROK, "Unknown scaling_governor: > %s", contents); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i); > + snprintf(path_cpuinfo_min_freq, > sizeof(path_cpuinfo_min_freq), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i); > + tst_res(TDEBUG, "Checking whether %s is bigger than > cpuinfo_min_freq", path); > + SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq); > + > + SAFE_FILE_SCANF(path_cpuinfo_min_freq, "%ld", > &cpuinfo_min_freq); > + > + if (cpuinfo_min_freq > cpuinfo_max_freq) > + tst_res(TFAIL, "Value in %s: %ld, should be > bigger than cpuinfo_min_freq: %ld", path, cpuinfo_max_freq, > cpuinfo_min_freq); > + else > + tst_res(TPASS, "Value in %s: %ld, is bigger > than cpuinfo_min_freq: %ld", path, cpuinfo_max_freq, > cpuinfo_min_freq); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/intel_pstate/status"); > + tst_res(TDEBUG, "Checking whether %s contains \"active\"", > path); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "active")) > + tst_res(TPASS, "%s is \"active\"", path); > + else > + tst_res(TFAIL, "%s is not \"active\", but %s", path, > contents); > + > + tst_res(TDEBUG, "Checking whether %s can be switched to > \"passive\"", path); > + SAFE_FILE_PRINTF(path, "passive"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "passive")) > + tst_res(TPASS, "%s is \"passive\"", path); > + else > + tst_res(TFAIL, "%s is not \"passive\", but %s", > path, contents); > + > + tst_res(TDEBUG, "Checking whether %s can be switched back to > \"active\"", path); > + SAFE_FILE_PRINTF(path, "active"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "active")) > + tst_res(TPASS, "%s is \"active\"", path); > + else > + tst_res(TFAIL, "%s is not \"active\", but %s", path, > contents); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/intel_pstate/no_turbo"); > + tst_res(TDEBUG, "Checking whether %s can be switched to 1", > path); > + SAFE_FILE_PRINTF(path, "1"); > + SAFE_FILE_SCANF(path, "%d", &no_turbo_val); > + > + if (no_turbo_val == 1) > + tst_res(TPASS, "%s is \"1\"", path); > + else > + tst_res(TFAIL, "%s is not \"1\", but %d", path, > no_turbo_val); > + > + tst_res(TDEBUG, "Checking whether %s can be switched back to > 0", path); > + SAFE_FILE_PRINTF(path, "0"); > + SAFE_FILE_SCANF(path, "%d", &no_turbo_val); > + > + if (no_turbo_val == 0) > + tst_res(TPASS, "%s is \"0\"", path); > + else > + tst_res(TFAIL, "%s is not \"0\", but %d", path, > no_turbo_val); > + > + tst_res(TDEBUG, "Checking whether > scaling_available_governors contains \"performance\" and > \"schedutil\" after switching > /sys/devices/system/cpu/intel_pstate/status to \"passive\""); > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/intel_pstate/status"); > + SAFE_FILE_PRINTF(path, "passive"); > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_governors", > i); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255[^\n]", contents); > + > + tst_res(TDEBUG, "Checking whether %s contains > \"performance\" and \"schedutil\"", path); > + if (strstr(contents, "performance") && > strstr(contents, "schedutil")) { > + tst_res(TPASS, "%s contains: %s", path, > contents); > + } else { > + tst_res(TINFO, "%s contains: %s", path, > contents); > + tst_res(TFAIL, "%s does not contain > \"performance\" and \"schedutil\"", path); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i); > + tst_res(TDEBUG, "Checking whether %s can be changed > to performance", path); > + SAFE_FILE_PRINTF(path, "performance"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "performance")) > + tst_res(TPASS, "%s is \"performance\"", > path); > + else > + tst_res(TFAIL, "%s is not \"performance\", > but %s", path, contents); > + > + tst_res(TDEBUG, "Checking whether %s can be changed > to schedutil", path); > + SAFE_FILE_PRINTF(path, "schedutil"); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "schedutil")) > + tst_res(TPASS, "%s is \"schedutil\"", path); > + else > + tst_res(TFAIL, "%s is not \"schedutil\", but > %s", path, contents); > + } > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i); > + tst_res(TDEBUG, "Checking whether %s can be assigned > to scaling_max_freq and scaling_min_freq", path); > + SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i); > + SAFE_FILE_SCANF(path, "%ld", &cpuinfo_min_freq); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", cpuinfo_max_freq); > + SAFE_FILE_SCANF(path, "%ld", &scaling_max_freq); > + > + if (cpuinfo_max_freq < scaling_max_freq) { > + tst_res(TINFO, "cpuinfo_max_freq: %ld", > cpuinfo_max_freq); > + tst_res(TINFO, "scaling_max_freq: %ld", > scaling_max_freq); > + tst_res(TFAIL, "Failure setting %s", path); > + } else { > + tst_res(TPASS, "Successfully set up %s", > path); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", cpuinfo_max_freq); > + SAFE_FILE_SCANF(path, "%ld", &scaling_min_freq); > + > + if (cpuinfo_max_freq < scaling_min_freq) { > + tst_res(TINFO, "cpuinfo_max_freq: %ld", > cpuinfo_max_freq); > + tst_res(TINFO, "scaling_min_freq: %ld", > scaling_min_freq); > + tst_res(TFAIL, "Failure setting %s", path); > + } else { > + tst_res(TPASS, "Successfully set up %s", > path); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", > previous_scaling_min_freq[i]); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", > previous_scaling_max_freq[i]); > + } > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_max_freq", i); > + tst_res(TDEBUG, "Checking whether %s can be assigned > to scaling_max_freq and scaling_min_freq", path); > + SAFE_FILE_SCANF(path, "%ld", &cpuinfo_max_freq); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/cpuinfo_min_freq", i); > + SAFE_FILE_SCANF(path, "%ld", &cpuinfo_min_freq); > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_min_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", cpuinfo_min_freq); > + SAFE_FILE_SCANF(path, "%ld", &scaling_min_freq); > + > + if (cpuinfo_min_freq > scaling_min_freq) { > + tst_res(TINFO, "cpuinfo_min_freq: %ld", > cpuinfo_min_freq); > + tst_res(TINFO, "scaling_min_freq: %ld", > scaling_min_freq); > + tst_res(TFAIL, "Failure setting %s", path); > + } else { > + tst_res(TPASS, "Successfully set up %s", > path); > + } > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_max_freq", i); > + SAFE_FILE_PRINTF(path, "%ld", cpuinfo_min_freq); > + SAFE_FILE_SCANF(path, "%ld", &scaling_max_freq); > + > + if (cpuinfo_min_freq > scaling_max_freq) { > + tst_res(TINFO, "cpuinfo_min_freq: %ld", > cpuinfo_min_freq); > + tst_res(TINFO, "scaling_max_freq: %ld", > scaling_max_freq); > + tst_res(TFAIL, "Failure setting %s", path); > + } else { > + tst_res(TPASS, "Successfully set up %s", > path); > + } > + } > + > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/statu > s", "active"); > + > + for (int i = 0; i < nproc; i++) { > + if (!online[i]) > + continue; > + > + snprintf(path, sizeof(path), > "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_governor", i); > + SAFE_FILE_PRINTF(path, "%s", > previous_scaling_governor[i]); > + } > +} > + > +static struct tst_test test = { > + .cleanup = cleanup, > + .needs_kconfigs = (const char *const []) { > + "CONFIG_X86_INTEL_PSTATE", > + NULL > + }, > + .needs_root = 1, > + .setup = setup, > + .supported_archs = (const char *const []) { > + "x86", > + "x86_64", > + NULL > + }, > + .test_all = run > +}; --------------------------------------------------------------------- Intel Technology Poland sp. z o.o. ul. Slowackiego 173 | 80-298 Gdansk | Sad Rejonowy Gdansk Polnoc | VII Wydzial Gospodarczy Krajowego Rejestru Sadowego - KRS 101882 | NIP 957-07-52-316 | Kapital zakladowy 200.000 PLN. Spolka oswiadcza, ze posiada status duzego przedsiebiorcy w rozumieniu ustawy z dnia 8 marca 2013 r. o przeciwdzialaniu nadmiernym opoznieniom w transakcjach handlowych. Ta wiadomosc wraz z zalacznikami jest przeznaczona dla okreslonego adresata i moze zawierac informacje poufne. W razie przypadkowego otrzymania tej wiadomosci, prosimy o powiadomienie nadawcy oraz trwale jej usuniecie; jakiekolwiek przegladanie lub rozpowszechnianie jest zabronione. This e-mail and any attachments may contain confidential material for the sole use of the intended recipient(s). If you are not the intended recipient, please contact the sender and delete all copies; any review or distribution by others is strictly prohibited. -- Mailing list info: https://lists.linux.it/listinfo/ltp ^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation 2026-05-07 11:11 [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation Piotr Kubaj 2026-05-07 11:57 ` [LTP] " linuxtestproject.agent 2026-05-27 7:27 ` [LTP] [PATCH v8] " Kubaj, Piotr @ 2026-06-03 12:27 ` Cyril Hrubis 2026-06-11 7:13 ` Andrea Cervesato via ltp 3 siblings, 0 replies; 5+ messages in thread From: Cyril Hrubis @ 2026-06-03 12:27 UTC (permalink / raw) To: Piotr Kubaj Cc: daniel.niestepski, tomasz.ossowski, helena.anna.dubel, rafael.j.wysocki, ltp Hi! > + SAFE_FILE_SCANF(path, "%255s", contents); > + tst_res(TDEBUG, "Checking whether %s is \"intel_pstate\"", path); I slightly prefer single quotes inside double quotes e.g. " ... 'intel_pstate'" since they does not need to be escaped. And maybe the run() could have been split into functions so that it's not that long. But both of these are very minor. Reviewed-by: Cyril Hrubis <chrubis@suse.cz> -- Cyril Hrubis chrubis@suse.cz -- Mailing list info: https://lists.linux.it/listinfo/ltp ^ permalink raw reply [flat|nested] 5+ messages in thread
* Re: [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation 2026-05-07 11:11 [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation Piotr Kubaj ` (2 preceding siblings ...) 2026-06-03 12:27 ` Cyril Hrubis @ 2026-06-11 7:13 ` Andrea Cervesato via ltp 3 siblings, 0 replies; 5+ messages in thread From: Andrea Cervesato via ltp @ 2026-06-11 7:13 UTC (permalink / raw) To: Piotr Kubaj Cc: daniel.niestepski, tomasz.ossowski, helena.anna.dubel, rafael.j.wysocki, ltp Hi Piotr, > +static void cleanup(void) > +{ > + char path[PATH_MAX]; > + > + if (!setup_done) { > + free(online); > + free(previous_scaling_max_freq); > + free(previous_scaling_min_freq); > + free(previous_scaling_governor); Better to use goto at the end of the cleanup() function, instead of defining free() in two different parts and duplicating code. > + return; > + } > + > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/status", "%s", intel_pstate_status); > + SAFE_FILE_PRINTF("/sys/devices/system/cpu/intel_pstate/no_turbo", "%d", no_turbo); For fixed sysfs files that we need to modify and restore, we should use .save_restore instead. > +static void setup(void) > +{ > + char path[PATH_MAX]; > + > + if (access("/sys/devices/system/cpu/intel_pstate/status", F_OK) == -1) > + tst_brk(TCONF, "intel_pstate driver not active"); Are you sure it's not a TBROK/TFAIL? We are checking for the driver configuration inside the kernel at tst_test level. I expect that .../cpu/intel_pstate/status is present at this point. If it's not, we are probably facing a bug in the driver, isnt it? [..] > + > + SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/no_turbo", "%d", &no_turbo); > + SAFE_FILE_SCANF("/sys/devices/system/cpu/intel_pstate/status", "%15s", intel_pstate_status); And since we are scanning the status value later on, we probably don't need the access(.../cpu/intel_pstate/status) verification. The test will TBROK anyway here. We should probably use .save_restore at this point, since we can safely guard the test from having drivers issues at configuration level. In particular, we should use .save_restore with TST_SR_TBROK (or TST_SR_TCONF if you confirm that driver might not initialize the sysfs under certain circumstances). > + snprintf(path, sizeof(path), "/sys/devices/system/cpu/cpu%d/cpufreq/scaling_available_governors", i); > + memset(contents, 0, sizeof(contents)); > + SAFE_FILE_SCANF(path, "%255[^\n]", contents); > + > + tst_res(TDEBUG, "Checking whether %s contains \"performance\" and \"schedutil\"", path); Please change all of these double quotes inside tst_res() message parameter single quotes (as Cyril mentioned already). > + SAFE_FILE_SCANF(path, "%255s", contents); > + > + if (strstr(contents, "schedutil")) Also, I'm a bit concern for the massive use of strstr() patterns. Again, if drivers are bugged and show a string with garbage data matching one of the "supposed" correct string, we pass the test while drivers are still bugged. You are also missing these two configurations which are needed by the test: - CONFIG_CPU_FREQ_GOV_SCHEDUTIL - CONFIG_CPU_FREQ_GOV_PERFORMANCE Otherwise, it's not possible to run: > + SAFE_FILE_PRINTF(path, "schedutil"); or setting CPU governor to performance. Regards, -- Andrea Cervesato SUSE QE Automation Engineer Linux andrea.cervesato@suse.com -- Mailing list info: https://lists.linux.it/listinfo/ltp ^ permalink raw reply [flat|nested] 5+ messages in thread
end of thread, other threads:[~2026-06-11 7:13 UTC | newest] Thread overview: 5+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2026-05-07 11:11 [LTP] [PATCH v8] cpufreq.c: add new test for cpufreq sysfs interface validation Piotr Kubaj 2026-05-07 11:57 ` [LTP] " linuxtestproject.agent 2026-05-27 7:27 ` [LTP] [PATCH v8] " Kubaj, Piotr 2026-06-03 12:27 ` Cyril Hrubis 2026-06-11 7:13 ` Andrea Cervesato via ltp
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox