On Fri, May 16, 2014 at 12:36 AM, Eduardo Valentin <edubezval@xxxxxxxxx> wrote: > Hello Amit, > > On Thu, May 08, 2014 at 08:08:00PM +0530, Amit Daniel Kachhap wrote: >> This patch upgrades the ACPI cpufreq cooling portions to use the generic >> cpufreq cooling infrastructure. There should not be any functionality >> related changes as the same behaviour is provided by the generic >> cpufreq APIs with the notifier mechanism. >> > > This is a very good move as we are converging towards a more and more > common infrastructure. How did you test this code? Thanks. Actually with linaro acpi kernel I was able to enable acpi processor in my arndale(exynos5250) board. After that in some hacky way I enabled acpi __TMP method to read its tmu temperatures and able to test thermal cpufreq scaling. Ofcourse i am not able to test cputhrottling as it is not supported. > >> Signed-off-by: Amit Daniel Kachhap <amit.daniel@xxxxxxxxxxx> >> --- >> drivers/acpi/processor_driver.c | 6 +- >> drivers/acpi/processor_thermal.c | 210 +++++++++++++++++--------------------- >> 2 files changed, 99 insertions(+), 117 deletions(-) >> >> diff --git a/drivers/acpi/processor_driver.c b/drivers/acpi/processor_driver.c >> index 7f70f31..10aba4a 100644 >> --- a/drivers/acpi/processor_driver.c >> +++ b/drivers/acpi/processor_driver.c >> @@ -36,6 +36,7 @@ >> #include <linux/cpuidle.h> >> #include <linux/slab.h> >> #include <linux/acpi.h> >> +#include <linux/cpu_cooling.h> >> >> #include <acpi/processor.h> >> >> @@ -178,8 +179,7 @@ static int __acpi_processor_start(struct acpi_device *device) >> if (!cpuidle_get_driver() || cpuidle_get_driver() == &acpi_idle_driver) >> acpi_processor_power_init(pr); >> >> - pr->cdev = thermal_cooling_device_register("Processor", device, >> - &processor_cooling_ops); >> + pr->cdev = acpi_processor_cooling_register(device); >> if (IS_ERR(pr->cdev)) { >> result = PTR_ERR(pr->cdev); >> goto err_power_exit; >> @@ -250,7 +250,7 @@ static int acpi_processor_stop(struct device *dev) >> if (pr->cdev) { >> sysfs_remove_link(&device->dev.kobj, "thermal_cooling"); >> sysfs_remove_link(&pr->cdev->device.kobj, "device"); >> - thermal_cooling_device_unregister(pr->cdev); >> + cpufreq_cooling_unregister(pr->cdev); >> pr->cdev = NULL; >> } >> return 0; >> diff --git a/drivers/acpi/processor_thermal.c b/drivers/acpi/processor_thermal.c >> index e003663..c442a58 100644 >> --- a/drivers/acpi/processor_thermal.c >> +++ b/drivers/acpi/processor_thermal.c >> @@ -31,6 +31,7 @@ >> #include <linux/init.h> >> #include <linux/cpufreq.h> >> #include <linux/acpi.h> >> +#include <linux/cpu_cooling.h> >> #include <acpi/processor.h> >> #include <asm/uaccess.h> >> >> @@ -53,28 +54,11 @@ ACPI_MODULE_NAME("processor_thermal"); >> >> static DEFINE_PER_CPU(unsigned int, cpufreq_thermal_reduction_pctg); >> static unsigned int acpi_thermal_cpufreq_is_init = 0; >> +static struct notifier_block cpufreq_cooling_notifier_block; >> >> #define reduction_pctg(cpu) \ >> per_cpu(cpufreq_thermal_reduction_pctg, phys_package_first_cpu(cpu)) >> >> -/* >> - * Emulate "per package data" using per cpu data (which should really be >> - * provided elsewhere) >> - * >> - * Note we can lose a CPU on cpu hotunplug, in this case we forget the state >> - * temporarily. Fortunately that's not a big issue here (I hope) >> - */ >> -static int phys_package_first_cpu(int cpu) >> -{ >> - int i; >> - int id = topology_physical_package_id(cpu); >> - >> - for_each_online_cpu(i) >> - if (topology_physical_package_id(i) == id) >> - return i; >> - return 0; >> -} >> - >> static int cpu_has_cpufreq(unsigned int cpu) >> { >> struct cpufreq_policy policy; >> @@ -83,30 +67,6 @@ static int cpu_has_cpufreq(unsigned int cpu) >> return 1; >> } >> >> -static int acpi_thermal_cpufreq_notifier(struct notifier_block *nb, >> - unsigned long event, void *data) >> -{ >> - struct cpufreq_policy *policy = data; >> - unsigned long max_freq = 0; >> - >> - if (event != CPUFREQ_ADJUST) >> - goto out; >> - >> - max_freq = ( >> - policy->cpuinfo.max_freq * >> - (100 - reduction_pctg(policy->cpu) * 20) >> - ) / 100; >> - >> - cpufreq_verify_within_limits(policy, 0, max_freq); >> - >> - out: >> - return 0; >> -} >> - >> -static struct notifier_block acpi_thermal_cpufreq_notifier_block = { >> - .notifier_call = acpi_thermal_cpufreq_notifier, >> -}; >> - >> static int cpufreq_get_max_state(unsigned int cpu) >> { >> if (!cpu_has_cpufreq(cpu)) >> @@ -123,34 +83,32 @@ static int cpufreq_get_cur_state(unsigned int cpu) >> return reduction_pctg(cpu); >> } >> >> -static int cpufreq_set_cur_state(unsigned int cpu, int state) >> +static int acpi_processor_freq_level(unsigned int cpu, int state) >> { >> - int i; >> + struct cpufreq_policy policy; >> + unsigned long max_freq = 0; >> + int level = 0; >> >> - if (!cpu_has_cpufreq(cpu)) >> + if (!acpi_thermal_cpufreq_is_init || cpufreq_get_policy(&policy, cpu)) >> return 0; >> >> reduction_pctg(cpu) = state; >> + max_freq = ( >> + policy.cpuinfo.max_freq * >> + (100 - reduction_pctg(cpu) * 20) >> + ) / 100; >> >> - /* >> - * Update all the CPUs in the same package because they all >> - * contribute to the temperature and often share the same >> - * frequency. >> - */ >> - for_each_online_cpu(i) { >> - if (topology_physical_package_id(i) == >> - topology_physical_package_id(cpu)) >> - cpufreq_update_policy(i); >> - } >> - return 0; >> + level = cpufreq_cooling_get_level(phys_package_first_cpu(cpu), >> + max_freq, GET_LEVEL_FLOOR); >> + return level; >> } >> >> void acpi_thermal_cpufreq_init(void) >> { >> int i; >> >> - i = cpufreq_register_notifier(&acpi_thermal_cpufreq_notifier_block, >> - CPUFREQ_POLICY_NOTIFIER); >> + i = cpufreq_cooling_register_notifier(&cpufreq_cooling_notifier_block, >> + CPU_COOLING_STATE_NOTIFIER); >> if (!i) >> acpi_thermal_cpufreq_is_init = 1; >> } >> @@ -158,9 +116,9 @@ void acpi_thermal_cpufreq_init(void) >> void acpi_thermal_cpufreq_exit(void) >> { >> if (acpi_thermal_cpufreq_is_init) >> - cpufreq_unregister_notifier >> - (&acpi_thermal_cpufreq_notifier_block, >> - CPUFREQ_POLICY_NOTIFIER); >> + cpufreq_cooling_unregister_notifier( >> + &cpufreq_cooling_notifier_block, >> + CPU_COOLING_STATE_NOTIFIER); >> >> acpi_thermal_cpufreq_is_init = 0; >> } >> @@ -176,13 +134,31 @@ static int cpufreq_get_cur_state(unsigned int cpu) >> return 0; >> } >> >> -static int cpufreq_set_cur_state(unsigned int cpu, int state) >> +static int acpi_processor_freq_level(unsigned int cpu, int state) >> { >> return 0; >> } >> >> #endif >> >> +/* >> + * Emulate "per package data" using per cpu data (which should really be >> + * provided elsewhere) >> + * >> + * Note we can lose a CPU on cpu hotunplug, in this case we forget the state >> + * temporarily. Fortunately that's not a big issue here (I hope) >> + */ >> +static int phys_package_first_cpu(int cpu) >> +{ >> + int i; >> + int id = topology_physical_package_id(cpu); >> + >> + for_each_online_cpu(i) >> + if (topology_physical_package_id(i) == id) >> + return i; >> + return 0; >> +} >> + >> /* thermal cooling device callbacks */ >> static int acpi_processor_max_state(struct acpi_processor *pr) >> { >> @@ -198,57 +174,22 @@ static int acpi_processor_max_state(struct acpi_processor *pr) >> >> return max_state; >> } >> -static int >> -processor_get_max_state(struct thermal_cooling_device *cdev, >> - unsigned long *state) >> -{ >> - struct acpi_device *device = cdev->devdata; >> - struct acpi_processor *pr; >> - >> - if (!device) >> - return -EINVAL; >> - >> - pr = acpi_driver_data(device); >> - if (!pr) >> - return -EINVAL; >> >> - *state = acpi_processor_max_state(pr); >> - return 0; >> -} >> - >> -static int >> -processor_get_cur_state(struct thermal_cooling_device *cdev, >> - unsigned long *cur_state) >> +static int acpi_processor_cur_state(struct acpi_processor *pr) >> { >> - struct acpi_device *device = cdev->devdata; >> - struct acpi_processor *pr; >> - >> - if (!device) >> - return -EINVAL; >> - >> - pr = acpi_driver_data(device); >> - if (!pr) >> - return -EINVAL; >> - >> - *cur_state = cpufreq_get_cur_state(pr->id); >> + int cur_state = 0; >> + cur_state = cpufreq_get_cur_state(pr->id); >> if (pr->flags.throttling) >> - *cur_state += pr->throttling.state; >> - return 0; >> + cur_state += pr->throttling.state; >> + return cur_state; >> } >> >> -static int >> -processor_set_cur_state(struct thermal_cooling_device *cdev, >> - unsigned long state) >> +static int acpi_processor_set_cur_state(struct acpi_processor *pr, >> + struct cpufreq_cooling_status *cooling, unsigned long event) >> { >> - struct acpi_device *device = cdev->devdata; >> - struct acpi_processor *pr; >> - int result = 0; >> - int max_pstate; >> - >> - if (!device) >> - return -EINVAL; >> + int result = 0, level = 0; >> + int max_pstate, state = cooling->new_state; >> >> - pr = acpi_driver_data(device); >> if (!pr) >> return -EINVAL; >> >> @@ -257,20 +198,61 @@ processor_set_cur_state(struct thermal_cooling_device *cdev, >> if (state > acpi_processor_max_state(pr)) >> return -EINVAL; >> >> - if (state <= max_pstate) { >> + if (state <= max_pstate && event == CPU_COOLING_SET_STATE_PRE) { >> if (pr->flags.throttling && pr->throttling.state) >> result = acpi_processor_set_throttling(pr, 0, false); >> - cpufreq_set_cur_state(pr->id, state); >> - } else { >> - cpufreq_set_cur_state(pr->id, max_pstate); >> + } else if (state > max_pstate && event == CPU_COOLING_SET_STATE_POST) { >> result = acpi_processor_set_throttling(pr, >> state - max_pstate, false); >> } >> + >> + level = acpi_processor_freq_level(pr->id, state); >> + cooling->new_state = level; >> + >> return result; >> } >> >> -const struct thermal_cooling_device_ops processor_cooling_ops = { >> - .get_max_state = processor_get_max_state, >> - .get_cur_state = processor_get_cur_state, >> - .set_cur_state = processor_set_cur_state, >> +static int acpi_cpufreq_cooling_notifier(struct notifier_block *nb, >> + unsigned long event, void *data) >> +{ >> + struct cpufreq_cooling_status *cooling = data; >> + struct acpi_device *device = cooling->devdata; >> + struct acpi_processor *pr = acpi_driver_data(device); >> + switch (event) { >> + case CPU_COOLING_SET_STATE_PRE: >> + case CPU_COOLING_SET_STATE_POST: >> + acpi_processor_set_cur_state(pr, cooling, event); >> + break; >> + case CPU_COOLING_GET_MAX_STATE: >> + cooling->max_state = acpi_processor_max_state(pr); >> + break; >> + case CPU_COOLING_GET_CUR_STATE: >> + cooling->cur_state = acpi_processor_cur_state(pr); >> + break; >> + default: >> + return -EINVAL; >> + } >> + return 0; >> +} >> + >> +static struct notifier_block cpufreq_cooling_notifier_block = { >> + .notifier_call = acpi_cpufreq_cooling_notifier, >> }; >> + >> +struct thermal_cooling_device * >> +acpi_processor_cooling_register(struct acpi_device *device) >> +{ >> + struct thermal_cooling_device *cdev; >> + struct acpi_processor *pr = acpi_driver_data(device); >> + int cpu = phys_package_first_cpu(pr->id); >> + int i; >> + int id = topology_physical_package_id(cpu); >> + struct cpumask cpus; >> + >> + for_each_online_cpu(i) >> + if (topology_physical_package_id(i) == id) >> + cpumask_set_cpu(i, &cpus); >> + >> + cdev = cpufreq_cooling_register(&cpus, (void *)device); >> + return cdev; >> +} >> -- >> 1.7.1 >> >> -- >> To unsubscribe from this list: send the line "unsubscribe linux-kernel" in >> the body of a message to majordomo@xxxxxxxxxxxxxxx >> More majordomo info at http://vger.kernel.org/majordomo-info.html >> Please read the FAQ at http://www.tux.org/lkml/ -- To unsubscribe from this list: send the line "unsubscribe linux-acpi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html