mirror of
https://mirrors.bfsu.edu.cn/git/linux.git
synced 2024-11-11 12:28:41 +08:00
cpufreq: Preserve policy structure across suspend/resume
To perform light-weight cpu-init and teardown in the cpufreq subsystem during suspend/resume, we need to separate out the 2 main functionalities of the cpufreq CPU hotplug callbacks, as outlined below: 1. Init/tear-down of core cpufreq and CPU-specific components, which are critical to the correct functioning of the cpufreq subsystem. 2. Init/tear-down of cpufreq sysfs files during suspend/resume. The first part requires accurate updates to the policy structure such as its ->cpus and ->related_cpus masks, whereas the second part requires that the policy->kobj structure is not released or re-initialized during suspend/resume. To handle both these requirements, we need to allow updates to the policy structure throughout suspend/resume, but prevent the structure from getting freed up. Also, we must have a mechanism by which the cpu-up callbacks can restore the policy structure, without allocating things afresh. (That also helps avoid memory leaks). To achieve this, we use 2 schemes: a. Use a fallback per-cpu storage area for preserving the policy structures during suspend, so that they can be restored during resume appropriately. b. Use the 'frozen' flag to determine when to free or allocate the policy structure vs when to restore the policy from the saved fallback storage. Thus we can successfully preserve the structure across suspend/resume. Effectively, this helps us complete the separation of the 'light-weight' and the 'full' init/tear-down sequences in the cpufreq subsystem, so that this can be made use of in the suspend/resume scenario. Signed-off-by: Srivatsa S. Bhat <srivatsa.bhat@linux.vnet.ibm.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
a82fab2928
commit
8414809c6a
@ -44,6 +44,7 @@
|
|||||||
*/
|
*/
|
||||||
static struct cpufreq_driver *cpufreq_driver;
|
static struct cpufreq_driver *cpufreq_driver;
|
||||||
static DEFINE_PER_CPU(struct cpufreq_policy *, cpufreq_cpu_data);
|
static DEFINE_PER_CPU(struct cpufreq_policy *, cpufreq_cpu_data);
|
||||||
|
static DEFINE_PER_CPU(struct cpufreq_policy *, cpufreq_cpu_data_fallback);
|
||||||
static DEFINE_RWLOCK(cpufreq_driver_lock);
|
static DEFINE_RWLOCK(cpufreq_driver_lock);
|
||||||
static DEFINE_MUTEX(cpufreq_governor_lock);
|
static DEFINE_MUTEX(cpufreq_governor_lock);
|
||||||
|
|
||||||
@ -946,6 +947,20 @@ static int cpufreq_add_policy_cpu(unsigned int cpu, unsigned int sibling,
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
static struct cpufreq_policy *cpufreq_policy_restore(unsigned int cpu)
|
||||||
|
{
|
||||||
|
struct cpufreq_policy *policy;
|
||||||
|
unsigned long flags;
|
||||||
|
|
||||||
|
write_lock_irqsave(&cpufreq_driver_lock, flags);
|
||||||
|
|
||||||
|
policy = per_cpu(cpufreq_cpu_data_fallback, cpu);
|
||||||
|
|
||||||
|
write_unlock_irqrestore(&cpufreq_driver_lock, flags);
|
||||||
|
|
||||||
|
return policy;
|
||||||
|
}
|
||||||
|
|
||||||
static struct cpufreq_policy *cpufreq_policy_alloc(void)
|
static struct cpufreq_policy *cpufreq_policy_alloc(void)
|
||||||
{
|
{
|
||||||
struct cpufreq_policy *policy;
|
struct cpufreq_policy *policy;
|
||||||
@ -1023,7 +1038,12 @@ static int __cpufreq_add_dev(struct device *dev, struct subsys_interface *sif,
|
|||||||
goto module_out;
|
goto module_out;
|
||||||
}
|
}
|
||||||
|
|
||||||
policy = cpufreq_policy_alloc();
|
if (frozen)
|
||||||
|
/* Restore the saved policy when doing light-weight init */
|
||||||
|
policy = cpufreq_policy_restore(cpu);
|
||||||
|
else
|
||||||
|
policy = cpufreq_policy_alloc();
|
||||||
|
|
||||||
if (!policy)
|
if (!policy)
|
||||||
goto nomem_out;
|
goto nomem_out;
|
||||||
|
|
||||||
@ -1204,6 +1224,10 @@ static int __cpufreq_remove_dev(struct device *dev,
|
|||||||
data = per_cpu(cpufreq_cpu_data, cpu);
|
data = per_cpu(cpufreq_cpu_data, cpu);
|
||||||
per_cpu(cpufreq_cpu_data, cpu) = NULL;
|
per_cpu(cpufreq_cpu_data, cpu) = NULL;
|
||||||
|
|
||||||
|
/* Save the policy somewhere when doing a light-weight tear-down */
|
||||||
|
if (frozen)
|
||||||
|
per_cpu(cpufreq_cpu_data_fallback, cpu) = data;
|
||||||
|
|
||||||
write_unlock_irqrestore(&cpufreq_driver_lock, flags);
|
write_unlock_irqrestore(&cpufreq_driver_lock, flags);
|
||||||
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
@ -1249,27 +1273,40 @@ static int __cpufreq_remove_dev(struct device *dev,
|
|||||||
if (cpufreq_driver->target)
|
if (cpufreq_driver->target)
|
||||||
__cpufreq_governor(data, CPUFREQ_GOV_POLICY_EXIT);
|
__cpufreq_governor(data, CPUFREQ_GOV_POLICY_EXIT);
|
||||||
|
|
||||||
lock_policy_rwsem_read(cpu);
|
if (!frozen) {
|
||||||
kobj = &data->kobj;
|
lock_policy_rwsem_read(cpu);
|
||||||
cmp = &data->kobj_unregister;
|
kobj = &data->kobj;
|
||||||
unlock_policy_rwsem_read(cpu);
|
cmp = &data->kobj_unregister;
|
||||||
kobject_put(kobj);
|
unlock_policy_rwsem_read(cpu);
|
||||||
|
kobject_put(kobj);
|
||||||
|
|
||||||
/* we need to make sure that the underlying kobj is actually
|
/*
|
||||||
* not referenced anymore by anybody before we proceed with
|
* We need to make sure that the underlying kobj is
|
||||||
* unloading.
|
* actually not referenced anymore by anybody before we
|
||||||
|
* proceed with unloading.
|
||||||
|
*/
|
||||||
|
pr_debug("waiting for dropping of refcount\n");
|
||||||
|
wait_for_completion(cmp);
|
||||||
|
pr_debug("wait complete\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Perform the ->exit() even during light-weight tear-down,
|
||||||
|
* since this is a core component, and is essential for the
|
||||||
|
* subsequent light-weight ->init() to succeed.
|
||||||
*/
|
*/
|
||||||
pr_debug("waiting for dropping of refcount\n");
|
|
||||||
wait_for_completion(cmp);
|
|
||||||
pr_debug("wait complete\n");
|
|
||||||
|
|
||||||
if (cpufreq_driver->exit)
|
if (cpufreq_driver->exit)
|
||||||
cpufreq_driver->exit(data);
|
cpufreq_driver->exit(data);
|
||||||
|
|
||||||
cpufreq_policy_free(data);
|
if (!frozen)
|
||||||
|
cpufreq_policy_free(data);
|
||||||
} else {
|
} else {
|
||||||
pr_debug("%s: removing link, cpu: %d\n", __func__, cpu);
|
|
||||||
cpufreq_cpu_put(data);
|
if (!frozen) {
|
||||||
|
pr_debug("%s: removing link, cpu: %d\n", __func__, cpu);
|
||||||
|
cpufreq_cpu_put(data);
|
||||||
|
}
|
||||||
|
|
||||||
if (cpufreq_driver->target) {
|
if (cpufreq_driver->target) {
|
||||||
__cpufreq_governor(data, CPUFREQ_GOV_START);
|
__cpufreq_governor(data, CPUFREQ_GOV_START);
|
||||||
__cpufreq_governor(data, CPUFREQ_GOV_LIMITS);
|
__cpufreq_governor(data, CPUFREQ_GOV_LIMITS);
|
||||||
|
Loading…
Reference in New Issue
Block a user