From: Mario Limonciello <superm1@kernel.org>
To: K Prateek Nayak <kprateek.nayak@amd.com>,
"Rafael J. Wysocki" <rafael@kernel.org>,
Viresh Kumar <viresh.kumar@linaro.org>,
Huang Rui <ray.huang@amd.com>
Cc: Perry Yuan <perry.yuan@amd.com>,
linux-pm@vger.kernel.org, linux-doc@vger.kernel.org,
linux-kernel@vger.kernel.org
Subject: Re: [RFC PATCH 2/6] cpufreq/amd-pstate: Add dynamic EPP as an "energy_performance_preference" mode
Date: Wed, 1 Jul 2026 16:33:54 -0500 [thread overview]
Message-ID: <01c027bf-0360-49c1-9d11-141f7dafde26@kernel.org> (raw)
In-Reply-To: <20260630185904.5602-3-kprateek.nayak@amd.com>
On 6/30/26 13:59, K Prateek Nayak wrote:
> Conver the global "dynamic_epp" toggle into a per-CPU
Convert
> "energy_performance_preference" mode "dynamic" that allows toggling the
> functionality of "dynamic_epp" at a per-CPU level.
>
> Instead of being a system-wide toggle, users can opt into the
> functionality of dynamic EPP on a per-CPU basis by switching to the
> powersave governor and selecting the "dynamic" mode from the available
> performance preferences.
>
> Unlike the previous implementation that had to check for driver mode
> before toggling on the functionality, block writes to certain sysfs
> files, potentially disallow policy change, etc. the per-CPU toggle fits
> naturally into the intended design and provides more granular control to
> the user.
No real concerns here, just a few small nits.
>
> Signed-off-by: K Prateek Nayak <kprateek.nayak@amd.com>
> ---
> drivers/cpufreq/amd-pstate.c | 90 +++++++++++++++++++++++++++++++-----
> 1 file changed, 79 insertions(+), 11 deletions(-)
>
> diff --git a/drivers/cpufreq/amd-pstate.c b/drivers/cpufreq/amd-pstate.c
> index 1893b0054a6a..c7eec6b96dcc 100644
> --- a/drivers/cpufreq/amd-pstate.c
> +++ b/drivers/cpufreq/amd-pstate.c
> @@ -106,6 +106,7 @@ static struct quirk_entry *quirks;
> * 3 balance_power
> * 4 power
> * 5 custom (for raw EPP values)
> + * 6 dynamic (platform profile driven selection)
I would just say "kernel driven decision". Right now this was driven by
platform profile, but the vision for dynamic EPP was that the kernel
could potentially change EPP values for individual cores on more factors.
> */
> enum energy_perf_value_index {
> EPP_INDEX_DEFAULT = 0,
> @@ -114,6 +115,7 @@ enum energy_perf_value_index {
> EPP_INDEX_BALANCE_POWERSAVE,
> EPP_INDEX_POWERSAVE,
> EPP_INDEX_CUSTOM,
> + EPP_INDEX_DYNAMIC,
> EPP_INDEX_MAX,
> };
>
> @@ -124,6 +126,7 @@ static const char * const energy_perf_strings[] = {
> [EPP_INDEX_BALANCE_POWERSAVE] = "balance_power",
> [EPP_INDEX_POWERSAVE] = "power",
> [EPP_INDEX_CUSTOM] = "custom",
> + [EPP_INDEX_DYNAMIC] = "dynamic",
> };
> static_assert(ARRAY_SIZE(energy_perf_strings) == EPP_INDEX_MAX);
>
> @@ -134,7 +137,7 @@ static unsigned int epp_values[] = {
> [EPP_INDEX_BALANCE_POWERSAVE] = AMD_CPPC_EPP_BALANCE_POWERSAVE,
> [EPP_INDEX_POWERSAVE] = AMD_CPPC_EPP_POWERSAVE,
> };
> -static_assert(ARRAY_SIZE(epp_values) == EPP_INDEX_MAX - 1);
> +static_assert(ARRAY_SIZE(epp_values) == EPP_INDEX_MAX - 2);
>
> typedef int (*cppc_mode_transition_fn)(int);
>
> @@ -1262,6 +1265,7 @@ EXPORT_SYMBOL_GPL(amd_pstate_clear_dynamic_epp);
> static int amd_pstate_set_dynamic_epp(struct cpufreq_policy *policy)
> {
> struct amd_cpudata *cpudata = policy->driver_data;
> + u64 prev = READ_ONCE(cpudata->cppc_req_cached);
> int ret;
> u8 epp;
>
> @@ -1302,6 +1306,9 @@ static int amd_pstate_set_dynamic_epp(struct cpufreq_policy *policy)
> cleanup:
> amd_pstate_clear_dynamic_epp(policy);
>
> + epp = FIELD_GET(AMD_CPPC_EPP_PERF_MASK, prev);
> + /* Restore previous EPP if toggling Dynamic EPP failed. */
> + amd_pstate_set_epp(policy, epp);
> return ret;
> }
>
> @@ -1372,14 +1379,22 @@ static ssize_t show_amd_pstate_hw_prefcore(struct cpufreq_policy *policy,
> static ssize_t show_energy_performance_available_preferences(
> struct cpufreq_policy *policy, char *buf)
> {
> - int offset = 0, i;
> struct amd_cpudata *cpudata = policy->driver_data;
> + int i, max = EPP_INDEX_MAX, offset = 0;
>
> if (cpudata->policy == CPUFREQ_POLICY_PERFORMANCE)
> return sysfs_emit_at(buf, offset, "%s\n",
> energy_perf_strings[EPP_INDEX_PERFORMANCE]);
>
> - for (i = 0; i < ARRAY_SIZE(energy_perf_strings); i++)
> + /*
> + * Do not enumerate "dynamic" option only if disabled during boot.
> + * Users can still opt in to dynamic EPP if platform (server) has
> + * decided to keep it disabled by default.
> + */
As we're keeping it disabled for everyone right now, I think you can
avoid disambiguating server for now.
> + if (!dynamic_epp)
> + max = EPP_INDEX_DYNAMIC;
> +
> + for (i = 0; i < max; i++)
> offset += sysfs_emit_at(buf, offset, "%s ", energy_perf_strings[i]);
>
> offset += sysfs_emit_at(buf, offset, "\n");
> @@ -1395,11 +1410,6 @@ ssize_t store_energy_performance_preference(struct cpufreq_policy *policy,
> bool raw_epp = false;
> u8 epp;
>
> - if (cpudata->dynamic_epp) {
> - pr_debug("EPP cannot be set when dynamic EPP is enabled\n");
> - return -EBUSY;
> - }
> -
> /*
> * if the value matches a number, use that, otherwise see if
> * matches an index in the energy_perf_strings array
> @@ -1410,6 +1420,26 @@ ssize_t store_energy_performance_preference(struct cpufreq_policy *policy,
> ret = sysfs_match_string(energy_perf_strings, buf);
> if (ret < 0 || ret == EPP_INDEX_CUSTOM)
> return -EINVAL;
> +
> + if (ret == EPP_INDEX_DYNAMIC) {
> + /* Dynamic EPP disabled at boot or by platform. */
> + if (!dynamic_epp)
> + return -EINVAL;
> + /*
> + * Dynamic EPP was already enabled for this CPU.
> + * Nothing to do.
> + */
> + if (cpudata->dynamic_epp)
> + return count;
> +
> + cpudata->current_profile = PLATFORM_PROFILE_BALANCED;
> + ret = amd_pstate_set_dynamic_epp(policy);
> + if (ret)
> + return ret;
> +
> + return count;
> + }
> +
> if (ret)
> epp = epp_values[ret];
> else
> @@ -1421,6 +1451,13 @@ ssize_t store_energy_performance_preference(struct cpufreq_policy *policy,
> return -EBUSY;
> }
>
> + /*
> + * Dynamic EPP was enabled previously!
> + * Switch back to the static EPP mode.
> + */
> + if (cpudata->dynamic_epp)
> + amd_pstate_clear_dynamic_epp(policy);
> +
> ret = amd_pstate_set_epp(policy, epp);
> if (ret)
> return ret;
> @@ -1438,7 +1475,7 @@ ssize_t show_energy_performance_preference(struct cpufreq_policy *policy, char *
>
> epp = FIELD_GET(AMD_CPPC_EPP_PERF_MASK, cpudata->cppc_req_cached);
>
> - if (cpudata->raw_epp)
> + if (!cpudata->dynamic_epp && cpudata->raw_epp)
> return sysfs_emit(buf, "%u\n", epp);
>
> switch (epp) {
> @@ -1458,6 +1495,9 @@ ssize_t show_energy_performance_preference(struct cpufreq_policy *policy, char *
> return -EINVAL;
> }
>
> + if (cpudata->dynamic_epp)
> + return sysfs_emit(buf, "dynamic(profile:%s)\n", energy_perf_strings[preference]);
> +
> return sysfs_emit(buf, "%s\n", energy_perf_strings[preference]);
> }
> EXPORT_SYMBOL_GPL(show_energy_performance_preference);
> @@ -1943,10 +1983,14 @@ static int amd_pstate_epp_cpu_init(struct cpufreq_policy *policy)
> cpudata->current_profile = PLATFORM_PROFILE_BALANCED;
> }
>
> - if (dynamic_epp)
> + if (dynamic_epp) {
> + /* Dynamic EPP is only available with the POWERSAVE policy. */
> + policy->policy = CPUFREQ_POLICY_POWERSAVE;
> ret = amd_pstate_set_dynamic_epp(policy);
> - else
> + } else {
> ret = amd_pstate_set_epp(policy, cpudata->epp_default_dc);
> + }
> +
> if (ret)
> goto free_cpudata1;
>
> @@ -2018,6 +2062,30 @@ static int amd_pstate_epp_set_policy(struct cpufreq_policy *policy)
> if (!policy->cpuinfo.max_freq)
> return -ENODEV;
>
> + /* Must be a switch between PERFORMANCE and POWERSAVE */
> + if (cpudata->policy != policy->policy) {
> + /*
> + * Disable dynamic_epp when switching
> + * out of CPUFREQ_POLICY_POWERSAVE.
> + */
> + if (cpudata->dynamic_epp) {
> + WARN_ON_ONCE(cpudata->policy != CPUFREQ_POLICY_POWERSAVE);
> + amd_pstate_clear_dynamic_epp(policy);
> + }
> + /*
> + * If dynamic_epp is enabled by default, toggle it on
> + * when switching to CPUFREQ_POLICY_POWERSAVE.
> + */
> + if (dynamic_epp) {
> + WARN_ON_ONCE(policy->policy != CPUFREQ_POLICY_POWERSAVE);
> +
> + cpudata->current_profile = PLATFORM_PROFILE_BALANCED;
> + ret = amd_pstate_set_dynamic_epp(policy);
> + if (ret)
> + return ret;
> + }
> + }
> +
> cpudata->policy = policy->policy;
>
> ret = amd_pstate_epp_update_limit(policy, true);
next prev parent reply other threads:[~2026-07-01 21:33 UTC|newest]
Thread overview: 12+ messages / expand[flat|nested] mbox.gz Atom feed top
2026-06-30 18:58 [RFC PATCH 0/6] cpufreq/amd-pstate: Rework dynamic_epp as an energy_performance_preference mode K Prateek Nayak
2026-06-30 18:58 ` [RFC PATCH 1/6] cpufreq/amd-pstate: Extract platform profile to EPP conversion into a helper K Prateek Nayak
2026-07-01 21:31 ` Mario Limonciello
2026-06-30 18:59 ` [RFC PATCH 2/6] cpufreq/amd-pstate: Add dynamic EPP as an "energy_performance_preference" mode K Prateek Nayak
2026-07-01 21:33 ` Mario Limonciello [this message]
2026-06-30 18:59 ` [RFC PATCH 3/6] cpufreq/amd-pstate: Repurpose "amd_dynamic_epp" cmdline and corresponding sysfs K Prateek Nayak
2026-07-01 21:40 ` Mario Limonciello
2026-06-30 18:59 ` [RFC PATCH 4/6] Documentation/amd-pstate: Update dynamic_epp documentation with new behavior K Prateek Nayak
2026-06-30 19:03 ` K Prateek Nayak
2026-06-30 19:03 ` [RFC PATCH 5/6] cpufreq/amd-pstate: Reduce the scope of exported symbols K Prateek Nayak
2026-07-01 21:38 ` Mario Limonciello
2026-06-30 19:03 ` [RFC PATCH 6/6] cpufreq/amd-pstate-ut: Add unit test for "dynamic" EPP mode K Prateek Nayak
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=01c027bf-0360-49c1-9d11-141f7dafde26@kernel.org \
--to=superm1@kernel.org \
--cc=kprateek.nayak@amd.com \
--cc=linux-doc@vger.kernel.org \
--cc=linux-kernel@vger.kernel.org \
--cc=linux-pm@vger.kernel.org \
--cc=perry.yuan@amd.com \
--cc=rafael@kernel.org \
--cc=ray.huang@amd.com \
--cc=viresh.kumar@linaro.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox