Commit e1eb4779 authored by Stanislaw Gruszka's avatar Stanislaw Gruszka Committed by Len Brown

ACPI: Avoid wiping out pr->performance during preregistering

When cpufreq driver call acpi_processor_preregister_performance() , function
will clean up pr->performance even if there is possibly already registered
other cpufreq driver. The patch fix this potential problem. It also remove
double checks in P domain basic validity code and move these checks to function
where _PSD data is captured.
Signed-off-by: default avatarStanislaw Gruszka <sgruszka@redhat.com>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent 8e0ee43b
...@@ -479,6 +479,13 @@ static int acpi_processor_get_psd(struct acpi_processor *pr) ...@@ -479,6 +479,13 @@ static int acpi_processor_get_psd(struct acpi_processor *pr)
goto end; goto end;
} }
if (pdomain->coord_type != DOMAIN_COORD_TYPE_SW_ALL &&
pdomain->coord_type != DOMAIN_COORD_TYPE_SW_ANY &&
pdomain->coord_type != DOMAIN_COORD_TYPE_HW_ALL) {
printk(KERN_ERR PREFIX "Invalid _PSD:coord_type\n");
result = -EFAULT;
goto end;
}
end: end:
kfree(buffer.pointer); kfree(buffer.pointer);
return result; return result;
...@@ -501,9 +508,10 @@ int acpi_processor_preregister_performance( ...@@ -501,9 +508,10 @@ int acpi_processor_preregister_performance(
mutex_lock(&performance_mutex); mutex_lock(&performance_mutex);
retval = 0; /*
* Check if another driver has already registered, and abort before
/* Call _PSD for all CPUs */ * changing pr->performance if it has. Check input data as well.
*/
for_each_possible_cpu(i) { for_each_possible_cpu(i) {
pr = per_cpu(processors, i); pr = per_cpu(processors, i);
if (!pr) { if (!pr) {
...@@ -513,13 +521,20 @@ int acpi_processor_preregister_performance( ...@@ -513,13 +521,20 @@ int acpi_processor_preregister_performance(
if (pr->performance) { if (pr->performance) {
retval = -EBUSY; retval = -EBUSY;
continue; goto err_out;
} }
if (!performance || !percpu_ptr(performance, i)) { if (!performance || !percpu_ptr(performance, i)) {
retval = -EINVAL; retval = -EINVAL;
continue; goto err_out;
} }
}
/* Call _PSD for all CPUs */
for_each_possible_cpu(i) {
pr = per_cpu(processors, i);
if (!pr)
continue;
pr->performance = percpu_ptr(performance, i); pr->performance = percpu_ptr(performance, i);
cpumask_set_cpu(i, pr->performance->shared_cpu_map); cpumask_set_cpu(i, pr->performance->shared_cpu_map);
...@@ -535,26 +550,6 @@ int acpi_processor_preregister_performance( ...@@ -535,26 +550,6 @@ int acpi_processor_preregister_performance(
* Now that we have _PSD data from all CPUs, lets setup P-state * Now that we have _PSD data from all CPUs, lets setup P-state
* domain info. * domain info.
*/ */
for_each_possible_cpu(i) {
pr = per_cpu(processors, i);
if (!pr)
continue;
/* Basic validity check for domain info */
pdomain = &(pr->performance->domain_info);
if ((pdomain->revision != ACPI_PSD_REV0_REVISION) ||
(pdomain->num_entries != ACPI_PSD_REV0_ENTRIES)) {
retval = -EINVAL;
goto err_ret;
}
if (pdomain->coord_type != DOMAIN_COORD_TYPE_SW_ALL &&
pdomain->coord_type != DOMAIN_COORD_TYPE_SW_ANY &&
pdomain->coord_type != DOMAIN_COORD_TYPE_HW_ALL) {
retval = -EINVAL;
goto err_ret;
}
}
cpumask_clear(covered_cpus); cpumask_clear(covered_cpus);
for_each_possible_cpu(i) { for_each_possible_cpu(i) {
pr = per_cpu(processors, i); pr = per_cpu(processors, i);
...@@ -643,6 +638,7 @@ int acpi_processor_preregister_performance( ...@@ -643,6 +638,7 @@ int acpi_processor_preregister_performance(
pr->performance = NULL; /* Will be set for real in register */ pr->performance = NULL; /* Will be set for real in register */
} }
err_out:
mutex_unlock(&performance_mutex); mutex_unlock(&performance_mutex);
free_cpumask_var(covered_cpus); free_cpumask_var(covered_cpus);
return retval; return retval;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment