cpufreq: CPPC: fix perf_to_khz/khz_to_perf conversion exception
[ Upstream commit d93df29bdab133b85e94b3c328e7fe26a0ebd56c ] When the nominal_freq recorded by the kernel is equal to the lowest_freq, and the frequency adjustment operation is triggered externally, there is a logic error in cppc_perf_to_khz()/cppc_khz_to_perf(), resulting in perf and khz conversion errors. Fix this by adding a branch processing logic when nominal_freq is equal to lowest_freq. Fixes: ec1c7ad47664 ("cpufreq: CPPC: Fix performance/frequency conversion") Signed-off-by: liwei <liwei728@huawei.com> Acked-by: Viresh Kumar <viresh.kumar@linaro.org> Link: https://patch.msgid.link/20241024022952.2627694-1-liwei728@huawei.com [ rjw: Subject and changelog edits ] Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com> Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:
parent
ff2a9c4029
commit
bc5085816e
@ -1718,9 +1718,15 @@ unsigned int cppc_perf_to_khz(struct cppc_perf_caps *caps, unsigned int perf)
|
|||||||
u64 mul, div;
|
u64 mul, div;
|
||||||
|
|
||||||
if (caps->lowest_freq && caps->nominal_freq) {
|
if (caps->lowest_freq && caps->nominal_freq) {
|
||||||
mul = caps->nominal_freq - caps->lowest_freq;
|
/* Avoid special case when nominal_freq is equal to lowest_freq */
|
||||||
|
if (caps->lowest_freq == caps->nominal_freq) {
|
||||||
|
mul = caps->nominal_freq;
|
||||||
|
div = caps->nominal_perf;
|
||||||
|
} else {
|
||||||
|
mul = caps->nominal_freq - caps->lowest_freq;
|
||||||
|
div = caps->nominal_perf - caps->lowest_perf;
|
||||||
|
}
|
||||||
mul *= KHZ_PER_MHZ;
|
mul *= KHZ_PER_MHZ;
|
||||||
div = caps->nominal_perf - caps->lowest_perf;
|
|
||||||
offset = caps->nominal_freq * KHZ_PER_MHZ -
|
offset = caps->nominal_freq * KHZ_PER_MHZ -
|
||||||
div64_u64(caps->nominal_perf * mul, div);
|
div64_u64(caps->nominal_perf * mul, div);
|
||||||
} else {
|
} else {
|
||||||
@ -1741,11 +1747,17 @@ unsigned int cppc_khz_to_perf(struct cppc_perf_caps *caps, unsigned int freq)
|
|||||||
{
|
{
|
||||||
s64 retval, offset = 0;
|
s64 retval, offset = 0;
|
||||||
static u64 max_khz;
|
static u64 max_khz;
|
||||||
u64 mul, div;
|
u64 mul, div;
|
||||||
|
|
||||||
if (caps->lowest_freq && caps->nominal_freq) {
|
if (caps->lowest_freq && caps->nominal_freq) {
|
||||||
mul = caps->nominal_perf - caps->lowest_perf;
|
/* Avoid special case when nominal_freq is equal to lowest_freq */
|
||||||
div = caps->nominal_freq - caps->lowest_freq;
|
if (caps->lowest_freq == caps->nominal_freq) {
|
||||||
|
mul = caps->nominal_perf;
|
||||||
|
div = caps->nominal_freq;
|
||||||
|
} else {
|
||||||
|
mul = caps->nominal_perf - caps->lowest_perf;
|
||||||
|
div = caps->nominal_freq - caps->lowest_freq;
|
||||||
|
}
|
||||||
/*
|
/*
|
||||||
* We don't need to convert to kHz for computing offset and can
|
* We don't need to convert to kHz for computing offset and can
|
||||||
* directly use nominal_freq and lowest_freq as the div64_u64
|
* directly use nominal_freq and lowest_freq as the div64_u64
|
||||||
|
Loading…
x
Reference in New Issue
Block a user