cpufreq: CPPC: Enable fast_switch
The communication mean of the _CPC desired performance can be
PCC, System Memory, System IO, or Functional Fixed Hardware.
commit b7898fda5b
("cpufreq: Support for fast frequency switching")
fast_switching is 'for switching CPU frequencies from interrupt
context'.
Writes to SystemMemory and SystemIo are fast and suitable this.
This is not the case for PCC and might not be the case for FFH.
Enable fast_switching for the cppc_cpufreq driver in above cases.
Add cppc_allow_fast_switch() to check the desired performance
register address space and set fast_switching accordingly.
Signed-off-by: Pierre Gondois <pierre.gondois@arm.com>
Reviewed-by: Sudeep Holla <sudeep.holla@arm.com>
Acked-by: Viresh Kumar <viresh.kumar@linaro.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
This commit is contained in:
parent
6380b7b2b2
commit
3cc30dd00a
|
@ -434,6 +434,24 @@ bool acpi_cpc_valid(void)
|
||||||
}
|
}
|
||||||
EXPORT_SYMBOL_GPL(acpi_cpc_valid);
|
EXPORT_SYMBOL_GPL(acpi_cpc_valid);
|
||||||
|
|
||||||
|
bool cppc_allow_fast_switch(void)
|
||||||
|
{
|
||||||
|
struct cpc_register_resource *desired_reg;
|
||||||
|
struct cpc_desc *cpc_ptr;
|
||||||
|
int cpu;
|
||||||
|
|
||||||
|
for_each_possible_cpu(cpu) {
|
||||||
|
cpc_ptr = per_cpu(cpc_desc_ptr, cpu);
|
||||||
|
desired_reg = &cpc_ptr->cpc_regs[DESIRED_PERF];
|
||||||
|
if (!CPC_IN_SYSTEM_MEMORY(desired_reg) &&
|
||||||
|
!CPC_IN_SYSTEM_IO(desired_reg))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
EXPORT_SYMBOL_GPL(cppc_allow_fast_switch);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* acpi_get_psd_map - Map the CPUs in the freq domain of a given cpu
|
* acpi_get_psd_map - Map the CPUs in the freq domain of a given cpu
|
||||||
* @cpu: Find all CPUs that share a domain with cpu.
|
* @cpu: Find all CPUs that share a domain with cpu.
|
||||||
|
|
|
@ -389,6 +389,27 @@ static int cppc_cpufreq_set_target(struct cpufreq_policy *policy,
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static unsigned int cppc_cpufreq_fast_switch(struct cpufreq_policy *policy,
|
||||||
|
unsigned int target_freq)
|
||||||
|
{
|
||||||
|
struct cppc_cpudata *cpu_data = policy->driver_data;
|
||||||
|
unsigned int cpu = policy->cpu;
|
||||||
|
u32 desired_perf;
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
desired_perf = cppc_cpufreq_khz_to_perf(cpu_data, target_freq);
|
||||||
|
cpu_data->perf_ctrls.desired_perf = desired_perf;
|
||||||
|
ret = cppc_set_perf(cpu, &cpu_data->perf_ctrls);
|
||||||
|
|
||||||
|
if (ret) {
|
||||||
|
pr_debug("Failed to set target on CPU:%d. ret:%d\n",
|
||||||
|
cpu, ret);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return target_freq;
|
||||||
|
}
|
||||||
|
|
||||||
static int cppc_verify_policy(struct cpufreq_policy_data *policy)
|
static int cppc_verify_policy(struct cpufreq_policy_data *policy)
|
||||||
{
|
{
|
||||||
cpufreq_verify_within_cpu_limits(policy);
|
cpufreq_verify_within_cpu_limits(policy);
|
||||||
|
@ -721,6 +742,8 @@ static int cppc_cpufreq_cpu_init(struct cpufreq_policy *policy)
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
policy->fast_switch_possible = cppc_allow_fast_switch();
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If 'highest_perf' is greater than 'nominal_perf', we assume CPU Boost
|
* If 'highest_perf' is greater than 'nominal_perf', we assume CPU Boost
|
||||||
* is supported.
|
* is supported.
|
||||||
|
@ -866,6 +889,7 @@ static struct cpufreq_driver cppc_cpufreq_driver = {
|
||||||
.verify = cppc_verify_policy,
|
.verify = cppc_verify_policy,
|
||||||
.target = cppc_cpufreq_set_target,
|
.target = cppc_cpufreq_set_target,
|
||||||
.get = cppc_cpufreq_get_rate,
|
.get = cppc_cpufreq_get_rate,
|
||||||
|
.fast_switch = cppc_cpufreq_fast_switch,
|
||||||
.init = cppc_cpufreq_cpu_init,
|
.init = cppc_cpufreq_cpu_init,
|
||||||
.exit = cppc_cpufreq_cpu_exit,
|
.exit = cppc_cpufreq_cpu_exit,
|
||||||
.set_boost = cppc_cpufreq_set_boost,
|
.set_boost = cppc_cpufreq_set_boost,
|
||||||
|
|
|
@ -141,6 +141,7 @@ extern int cppc_set_perf(int cpu, struct cppc_perf_ctrls *perf_ctrls);
|
||||||
extern int cppc_set_enable(int cpu, bool enable);
|
extern int cppc_set_enable(int cpu, bool enable);
|
||||||
extern int cppc_get_perf_caps(int cpu, struct cppc_perf_caps *caps);
|
extern int cppc_get_perf_caps(int cpu, struct cppc_perf_caps *caps);
|
||||||
extern bool acpi_cpc_valid(void);
|
extern bool acpi_cpc_valid(void);
|
||||||
|
extern bool cppc_allow_fast_switch(void);
|
||||||
extern int acpi_get_psd_map(unsigned int cpu, struct cppc_cpudata *cpu_data);
|
extern int acpi_get_psd_map(unsigned int cpu, struct cppc_cpudata *cpu_data);
|
||||||
extern unsigned int cppc_get_transition_latency(int cpu);
|
extern unsigned int cppc_get_transition_latency(int cpu);
|
||||||
extern bool cpc_ffh_supported(void);
|
extern bool cpc_ffh_supported(void);
|
||||||
|
@ -175,6 +176,10 @@ static inline bool acpi_cpc_valid(void)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
static inline bool cppc_allow_fast_switch(void)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
static inline unsigned int cppc_get_transition_latency(int cpu)
|
static inline unsigned int cppc_get_transition_latency(int cpu)
|
||||||
{
|
{
|
||||||
return CPUFREQ_ETERNAL;
|
return CPUFREQ_ETERNAL;
|
||||||
|
|
Loading…
Reference in New Issue