Message ID | 1516721671-16360-6-git-send-email-daniel.lezcano@linaro.org |
---|---|
State | New |
Headers | show |
Series | CPU cooling device new strategies | expand |
On 31 January 2018 at 10:33, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: > On 31/01/2018 10:01, Vincent Guittot wrote: >> Hi Daniel, >> >> On 23 January 2018 at 16:34, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: > > [ ... ] (please trim :) > >>> + /* >>> + * Each cooling device is per package. Each package >>> + * has a set of cpus where the physical number is >>> + * duplicate in the kernel namespace. We need a way to >>> + * address the waitq[] and tsk[] arrays with index >>> + * which are not Linux cpu numbered. >>> + * >>> + * One solution is to use the >>> + * topology_core_id(cpu). Other solution is to use the >>> + * modulo. >>> + * >>> + * eg. 2 x cluster - 4 cores. >>> + * >>> + * Physical numbering -> Linux numbering -> % nr_cpus >>> + * >>> + * Pkg0 - Cpu0 -> 0 -> 0 >>> + * Pkg0 - Cpu1 -> 1 -> 1 >>> + * Pkg0 - Cpu2 -> 2 -> 2 >>> + * Pkg0 - Cpu3 -> 3 -> 3 >>> + * >>> + * Pkg1 - Cpu0 -> 4 -> 0 >>> + * Pkg1 - Cpu1 -> 5 -> 1 >>> + * Pkg1 - Cpu2 -> 6 -> 2 >>> + * Pkg1 - Cpu3 -> 7 -> 3 >> >> >> I'm not sure that the assumption above for the CPU numbering is safe. >> Can't you use a per cpu structure to point to resources that are per >> cpu instead ? so you will not have to rely on CPU ordering > > Can you elaborate ? I don't get the part with the percpu structure. Something like: struct cpuidle_cooling_cpu { struct task_struct *tsk; wait_queue_head_t waitq; }; DECLARE_PER_CPU(struct cpuidle_cooling_cpu *, cpu_data); > > > -- > <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs > > Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | > <http://twitter.com/#!/linaroorg> Twitter | > <http://www.linaro.org/linaro-blog/> Blog >
On 31/01/2018 10:46, Vincent Guittot wrote: > On 31 January 2018 at 10:33, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >> On 31/01/2018 10:01, Vincent Guittot wrote: >>> Hi Daniel, >>> >>> On 23 January 2018 at 16:34, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >> >> [ ... ] (please trim :) >> >>>> + /* >>>> + * Each cooling device is per package. Each package >>>> + * has a set of cpus where the physical number is >>>> + * duplicate in the kernel namespace. We need a way to >>>> + * address the waitq[] and tsk[] arrays with index >>>> + * which are not Linux cpu numbered. >>>> + * >>>> + * One solution is to use the >>>> + * topology_core_id(cpu). Other solution is to use the >>>> + * modulo. >>>> + * >>>> + * eg. 2 x cluster - 4 cores. >>>> + * >>>> + * Physical numbering -> Linux numbering -> % nr_cpus >>>> + * >>>> + * Pkg0 - Cpu0 -> 0 -> 0 >>>> + * Pkg0 - Cpu1 -> 1 -> 1 >>>> + * Pkg0 - Cpu2 -> 2 -> 2 >>>> + * Pkg0 - Cpu3 -> 3 -> 3 >>>> + * >>>> + * Pkg1 - Cpu0 -> 4 -> 0 >>>> + * Pkg1 - Cpu1 -> 5 -> 1 >>>> + * Pkg1 - Cpu2 -> 6 -> 2 >>>> + * Pkg1 - Cpu3 -> 7 -> 3 >>> >>> >>> I'm not sure that the assumption above for the CPU numbering is safe. >>> Can't you use a per cpu structure to point to resources that are per >>> cpu instead ? so you will not have to rely on CPU ordering >> >> Can you elaborate ? I don't get the part with the percpu structure. > > Something like: > > struct cpuidle_cooling_cpu { > struct task_struct *tsk; > wait_queue_head_t waitq; > }; > > DECLARE_PER_CPU(struct cpuidle_cooling_cpu *, cpu_data); I got this part but I don't get how that fixes the ordering thing. -- <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | <http://twitter.com/#!/linaroorg> Twitter | <http://www.linaro.org/linaro-blog/> Blog
On 31 January 2018 at 16:27, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: > On 31/01/2018 10:56, Vincent Guittot wrote: >> On 31 January 2018 at 10:50, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>> On 31/01/2018 10:46, Vincent Guittot wrote: >>>> On 31 January 2018 at 10:33, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>>>> On 31/01/2018 10:01, Vincent Guittot wrote: >>>>>> Hi Daniel, >>>>>> >>>>>> On 23 January 2018 at 16:34, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>>>> >>>>> [ ... ] (please trim :) >>>>> >>>>>>> + /* >>>>>>> + * Each cooling device is per package. Each package >>>>>>> + * has a set of cpus where the physical number is >>>>>>> + * duplicate in the kernel namespace. We need a way to >>>>>>> + * address the waitq[] and tsk[] arrays with index >>>>>>> + * which are not Linux cpu numbered. >>>>>>> + * >>>>>>> + * One solution is to use the >>>>>>> + * topology_core_id(cpu). Other solution is to use the >>>>>>> + * modulo. >>>>>>> + * >>>>>>> + * eg. 2 x cluster - 4 cores. >>>>>>> + * >>>>>>> + * Physical numbering -> Linux numbering -> % nr_cpus >>>>>>> + * >>>>>>> + * Pkg0 - Cpu0 -> 0 -> 0 >>>>>>> + * Pkg0 - Cpu1 -> 1 -> 1 >>>>>>> + * Pkg0 - Cpu2 -> 2 -> 2 >>>>>>> + * Pkg0 - Cpu3 -> 3 -> 3 >>>>>>> + * >>>>>>> + * Pkg1 - Cpu0 -> 4 -> 0 >>>>>>> + * Pkg1 - Cpu1 -> 5 -> 1 >>>>>>> + * Pkg1 - Cpu2 -> 6 -> 2 >>>>>>> + * Pkg1 - Cpu3 -> 7 -> 3 >>>>>> >>>>>> >>>>>> I'm not sure that the assumption above for the CPU numbering is safe. >>>>>> Can't you use a per cpu structure to point to resources that are per >>>>>> cpu instead ? so you will not have to rely on CPU ordering >>>>> >>>>> Can you elaborate ? I don't get the part with the percpu structure. >>>> >>>> Something like: >>>> >>>> struct cpuidle_cooling_cpu { >>>> struct task_struct *tsk; >>>> wait_queue_head_t waitq; >>>> }; >>>> >>>> DECLARE_PER_CPU(struct cpuidle_cooling_cpu *, cpu_data); >>> >>> I got this part but I don't get how that fixes the ordering thing. >> >> Because you don't care of the CPU ordering to retrieve the data as >> they are stored per cpu directly > > That's what I did initially, but for consistency reasons with the > cpufreq cpu cooling device which is stored in a list and the combo cpu > cooling device, the cpuidle cooling device must be per cluster and > stored in a list. I'm not sure to catch your problem. You can still have cpuidle cooling device per cluster and stored in the list but keep per cpu data in a AFAICT, you will not have more than one cpu cooling device registered per CPU so one per cpu variable that will gathers cpu private data should be enough ? > > Alternatively I can do: > > struct cpuidle_cooling_device { > struct thermal_cooling_device *cdev; > - struct task_struct **tsk; > + struct task_struct __percpu *tsk; > struct cpumask *cpumask; > struct list_head node; > struct hrtimer timer; > struct kref kref; > - wait_queue_head_t *waitq; > + wait_queue_head_t __percpu waitq; > atomic_t count; > unsigned int idle_cycle; > unsigned int state; > }; struct cpuidle_cooling_device { struct thermal_cooling_device *cdev; struct cpumask *cpumask; struct list_head node; struct hrtimer timer; struct kref kref; atomic_t count; unsigned int idle_cycle; unsigned int state; }; struct cpuidle_cooling_cpu { struct task_struct *tsk; wait_queue_head_t waitq; }; DECLARE_PER_CPU(struct cpuidle_cooling_cpu *, cpu_data); You continue to have cpuidle_cooling_device allocated dynamically per cluster and added in the list but task and waitq are stored per cpu > > > -- > <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs > > Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | > <http://twitter.com/#!/linaroorg> Twitter | > <http://www.linaro.org/linaro-blog/> Blog >
On 01/02/2018 08:57, Vincent Guittot wrote: > On 31 January 2018 at 16:27, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >> On 31/01/2018 10:56, Vincent Guittot wrote: >>> On 31 January 2018 at 10:50, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>>> On 31/01/2018 10:46, Vincent Guittot wrote: >>>>> On 31 January 2018 at 10:33, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>>>>> On 31/01/2018 10:01, Vincent Guittot wrote: >>>>>>> Hi Daniel, >>>>>>> >>>>>>> On 23 January 2018 at 16:34, Daniel Lezcano <daniel.lezcano@linaro.org> wrote: >>>>>> >>>>>> [ ... ] (please trim :) >>>>>> [ ... ] > struct cpuidle_cooling_device { > struct thermal_cooling_device *cdev; > struct cpumask *cpumask; > struct list_head node; > struct hrtimer timer; > struct kref kref; > atomic_t count; > unsigned int idle_cycle; > unsigned int state; > }; > > struct cpuidle_cooling_cpu { > struct task_struct *tsk; > wait_queue_head_t waitq; > }; > DECLARE_PER_CPU(struct cpuidle_cooling_cpu *, cpu_data); > > You continue to have cpuidle_cooling_device allocated dynamically per > cluster and added in the list but task and waitq are stored per cpu Ok. I will try that. -- <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | <http://twitter.com/#!/linaroorg> Twitter | <http://www.linaro.org/linaro-blog/> Blog
Hi Viresh, thanks for reviewing. On 07/02/2018 10:12, Viresh Kumar wrote: > On 23-01-18, 16:34, Daniel Lezcano wrote: >> diff --git a/drivers/thermal/cpu_cooling.c b/drivers/thermal/cpu_cooling.c > >> +/** >> + * cpuidle_cooling_ops - thermal cooling device ops >> + */ >> +static struct thermal_cooling_device_ops cpuidle_cooling_ops = { >> + .get_max_state = cpuidle_cooling_get_max_state, >> + .get_cur_state = cpuidle_cooling_get_cur_state, >> + .set_cur_state = cpuidle_cooling_set_cur_state, >> +}; >> + >> +/** >> + * cpuidle_cooling_release - Kref based release helper >> + * @kref: a pointer to the kref structure >> + * >> + * This function is automatically called by the kref_put function when >> + * the idle cooling device refcount reaches zero. At this point, we >> + * have the guarantee the structure is no longer in use and we can >> + * safely release all the ressources. >> + */ > > Don't really need doc style comments for internal routines. From Documentation/doc-guide/kernel-doc.rst: "We also recommend providing kernel-doc formatted documentation for private (file "static") routines, for consistency of kernel source code layout. But this is lower priority and at the discretion of the MAINTAINER of that kernel source file." Vote is open :) >> +static void __init cpuidle_cooling_release(struct kref *kref) >> +{ >> + struct cpuidle_cooling_device *idle_cdev = >> + container_of(kref, struct cpuidle_cooling_device, kref); >> + >> + thermal_cooling_device_unregister(idle_cdev->cdev); >> + kfree(idle_cdev->waitq); >> + kfree(idle_cdev->tsk); >> + kfree(idle_cdev); > > What about list-del here (cpuidle_cdev_list) ? Yes, thanks for pointing this. I have to convert the calling loop with the 'safe' list variant. >> +} >> + >> +/** >> + * cpuidle_cooling_register - Idle cooling device initialization function >> + * >> + * This function is in charge of creating a cooling device per cluster >> + * and register it to thermal framework. For this we rely on the >> + * topology as there is nothing yet describing better the idle state >> + * power domains. >> + * >> + * For each first CPU of the cluster's cpumask, we allocate the idle >> + * cooling device, initialize the general fields and then we initialze >> + * the rest in a per cpu basis. >> + * >> + * Returns zero on success, < 0 otherwise. >> + */ >> +int cpuidle_cooling_register(void) >> +{ >> + struct cpuidle_cooling_device *idle_cdev = NULL; >> + struct thermal_cooling_device *cdev; >> + struct task_struct *tsk; >> + struct device_node *np; >> + cpumask_t *cpumask; >> + char dev_name[THERMAL_NAME_LENGTH]; >> + int weight; >> + int ret = -ENOMEM, cpu; >> + int index = 0; >> + >> + for_each_possible_cpu(cpu) { >> + > > Perhaps this is coding choice, but just for the sake of consistency in > this driver should we remove such empty lines at the beginning of a > blocks ? Yes, it is coding choice. I'm in favor of separated blocks. I can remove the lines if that hurts. >> + cpumask = topology_core_cpumask(cpu); >> + weight = cpumask_weight(cpumask); >> + >> + /* >> + * This condition makes the first cpu belonging to the >> + * cluster to create a cooling device and allocates >> + * the structure. Others CPUs belonging to the same >> + * cluster will just increment the refcount on the >> + * cooling device structure and initialize it. >> + */ >> + if (cpu == cpumask_first(cpumask)) { >> + > > Like here as well. Ok. [ ... ] >> + pr_err("Failed to create idle cooling device (%d)\n", ret); >> + >> + return ret; >> +} > > What about cpuidle_cooling_unregister() ? The unregister function is not needed because cpuidle can't be unloaded. The cpuidle cooling device is registered after the cpuidle successfully initialized itself, there is no error path. >> +#endif >> diff --git a/include/linux/cpu_cooling.h b/include/linux/cpu_cooling.h >> index d4292eb..2b5950b 100644 >> --- a/include/linux/cpu_cooling.h >> +++ b/include/linux/cpu_cooling.h >> @@ -45,6 +45,7 @@ struct thermal_cooling_device * >> cpufreq_power_cooling_register(struct cpufreq_policy *policy, >> u32 capacitance, get_static_t plat_static_func); >> >> +extern int cpuidle_cooling_register(void); >> /** >> * of_cpufreq_cooling_register - create cpufreq cooling device based on DT. >> * @np: a valid struct device_node to the cooling device device tree node. >> @@ -118,6 +119,11 @@ void cpufreq_cooling_unregister(struct thermal_cooling_device *cdev) >> { >> return; >> } >> + >> +static inline int cpuidle_cooling_register(void) > > You need to use the new macros of cpufreq and cpuidle here as well, > else you will see compile time errors with some configurations. Ok, I thought I tried the different combinations but I will double check. Thanks -- Daniel >> +{ >> + return 0; >> +} >> #endif /* CONFIG_CPU_THERMAL */ >> >> #endif /* __CPU_COOLING_H__ */ >> -- >> 2.7.4 > -- <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | <http://twitter.com/#!/linaroorg> Twitter | <http://www.linaro.org/linaro-blog/> Blog
On 07-02-18, 11:34, Daniel Lezcano wrote: > On 07/02/2018 10:12, Viresh Kumar wrote: > > What about cpuidle_cooling_unregister() ? > > The unregister function is not needed because cpuidle can't be unloaded. > The cpuidle cooling device is registered after the cpuidle successfully > initialized itself, there is no error path. Okay, then there are two more things here. First, you don't need a kref in your patch and simple counter should be used instead, as kref is obviously more heavy to be used for the single error path here. Secondly, what about CPU hotplug ? For example, the cpu-freq cooling device gets removed currently if all CPUs of a cluster are hotplugged-out. But with your code, even if the CPUs are gone, their cpu-idle cooling device will stay. -- viresh
Hi Viresh, sorry for the late reply. On 09/02/2018 10:41, Viresh Kumar wrote: > On 07-02-18, 11:34, Daniel Lezcano wrote: >> On 07/02/2018 10:12, Viresh Kumar wrote: >>> What about cpuidle_cooling_unregister() ? >> >> The unregister function is not needed because cpuidle can't be unloaded. >> The cpuidle cooling device is registered after the cpuidle successfully >> initialized itself, there is no error path. > > Okay, then there are two more things here. > > First, you don't need a kref in your patch and simple counter should > be used instead, as kref is obviously more heavy to be used for the > single error path here. I prefer to keep the kref for its API. And I disagree about the heavy aspect :) struct kref { refcount_t refcount; }; > Secondly, what about CPU hotplug ? For example, the cpu-freq cooling > device gets removed currently if all CPUs of a cluster are > hotplugged-out. But with your code, even if the CPUs are gone, their > cpu-idle cooling device will stay. Yes and it will continue to compute the state, so if new CPUs are inserted the cooling device automatically uses the cooling state. I don't see a problem with that. -- <http://www.linaro.org/> Linaro.org │ Open source software for ARM SoCs Follow Linaro: <http://www.facebook.com/pages/Linaro> Facebook | <http://twitter.com/#!/linaroorg> Twitter | <http://www.linaro.org/linaro-blog/> Blog
diff --git a/drivers/thermal/Kconfig b/drivers/thermal/Kconfig index 925e73b..4bd4be7 100644 --- a/drivers/thermal/Kconfig +++ b/drivers/thermal/Kconfig @@ -166,6 +166,16 @@ config CPU_FREQ_THERMAL This will be useful for platforms using the generic thermal interface and not the ACPI interface. +config CPU_IDLE_THERMAL + bool "CPU idle cooling strategy" + depends on CPU_IDLE + help + This implements the generic CPU cooling mechanism through + idle injection. This will throttle the CPU by injecting + fixed idle cycle. All CPUs belonging to the same cluster + will enter idle synchronously to reach the deepest idle + state. + endchoice config CLOCK_THERMAL diff --git a/drivers/thermal/cpu_cooling.c b/drivers/thermal/cpu_cooling.c index d05bb73..916a627 100644 --- a/drivers/thermal/cpu_cooling.c +++ b/drivers/thermal/cpu_cooling.c @@ -10,18 +10,33 @@ * Viresh Kumar <viresh.kumar@linaro.org> * */ +#undef DEBUG +#define pr_fmt(fmt) "CPU cooling: " fmt + #include <linux/module.h> #include <linux/thermal.h> #include <linux/cpufreq.h> +#include <linux/cpuidle.h> #include <linux/err.h> +#include <linux/freezer.h> #include <linux/idr.h> +#include <linux/kthread.h> #include <linux/pm_opp.h> #include <linux/slab.h> +#include <linux/sched/prio.h> +#include <linux/sched/rt.h> #include <linux/cpu.h> #include <linux/cpu_cooling.h> +#include <linux/wait.h> + +#include <linux/platform_device.h> +#include <linux/of_platform.h> #include <trace/events/thermal.h> +#include <uapi/linux/sched/types.h> + +#ifdef CONFIG_CPU_FREQ_THERMAL /* * Cooling state <-> CPUFreq frequency * @@ -926,3 +941,459 @@ void cpufreq_cooling_unregister(struct thermal_cooling_device *cdev) kfree(cpufreq_cdev); } EXPORT_SYMBOL_GPL(cpufreq_cooling_unregister); + +#endif /* CPU_FREQ_THERMAL */ + +#ifdef CONFIG_CPU_IDLE_THERMAL +/* + * The idle duration injection. As we don't have yet a way to specify + * from the DT configuration, let's default to a tick duration. + */ +#define DEFAULT_IDLE_TIME_US TICK_USEC + +/** + * struct cpuidle_cooling_device - data for the idle cooling device + * @cdev: a pointer to a struct thermal_cooling_device + * @tsk: an array of pointer to the idle injection tasks + * @cpumask: a cpumask containing the CPU managed by the cooling device + * @timer: a hrtimer giving the tempo for the idle injection cycles + * @kref: a kernel refcount on this structure + * @waitq: the waiq for the idle injection tasks + * @count: an atomic to keep track of the last task exiting the idle cycle + * @idle_cycle: an integer defining the duration of the idle injection + * @state: an normalized integer giving the state of the cooling device + */ +struct cpuidle_cooling_device { + struct thermal_cooling_device *cdev; + struct task_struct **tsk; + struct cpumask *cpumask; + struct list_head node; + struct hrtimer timer; + struct kref kref; + wait_queue_head_t *waitq; + atomic_t count; + unsigned int idle_cycle; + unsigned int state; +}; + +static LIST_HEAD(cpuidle_cdev_list); + +/** + * cpuidle_cooling_wakeup - Wake up all idle injection threads + * @idle_cdev: the idle cooling device + * + * Every idle injection task belonging to the idle cooling device and + * running on an online cpu will be wake up by this call. + */ +static void cpuidle_cooling_wakeup(struct cpuidle_cooling_device *idle_cdev) +{ + int cpu; + int weight = cpumask_weight(idle_cdev->cpumask); + + for_each_cpu_and(cpu, idle_cdev->cpumask, cpu_online_mask) + wake_up_process(idle_cdev->tsk[cpu % weight]); +} + +/** + * cpuidle_cooling_wakeup_fn - Running cycle timer callback + * @timer: a hrtimer structure + * + * When the mitigation is acting, the CPU is allowed to run an amount + * of time, then the idle injection happens for the specified delay + * and the idle task injection schedules itself until the timer event + * wakes the idle injection tasks again for a new idle injection + * cycle. The time between the end of the idle injection and the timer + * expiration is the allocated running time for the CPU. + * + * Returns always HRTIMER_NORESTART + */ +static enum hrtimer_restart cpuidle_cooling_wakeup_fn(struct hrtimer *timer) +{ + struct cpuidle_cooling_device *idle_cdev = + container_of(timer, struct cpuidle_cooling_device, timer); + + cpuidle_cooling_wakeup(idle_cdev); + + return HRTIMER_NORESTART; +} + +/** + * cpuidle_cooling_runtime - Running time computation + * @idle_cdev: the idle cooling device + * + * The running duration is computed from the idle injection duration + * which is fixed. If we reach 100% of idle injection ratio, that + * means the running duration is zero. If we have a 50% ratio + * injection, that means we have equal duration for idle and for + * running duration. + * + * The formula is deduced as the following: + * + * running = idle x ((100 / ratio) - 1) + * + * For precision purpose for integer math, we use the following: + * + * running = (idle x 100) / ratio - idle + * + * For example, if we have an injected duration of 50%, then we end up + * with 10ms of idle injection and 10ms of running duration. + * + * Returns a s64 nanosecond based + */ +static s64 cpuidle_cooling_runtime(struct cpuidle_cooling_device *idle_cdev) +{ + s64 next_wakeup; + int state = idle_cdev->state; + + /* + * The function must never be called when there is no + * mitigation because: + * - that does not make sense + * - we end up with a division by zero + */ + BUG_ON(!state); + + next_wakeup = (s64)((idle_cdev->idle_cycle * 100) / state) - + idle_cdev->idle_cycle; + + return next_wakeup * NSEC_PER_USEC; +} + +/** + * cpuidle_cooling_injection_thread - Idle injection mainloop thread function + * @arg: a void pointer containing the idle cooling device address + * + * This main function does basically two operations: + * + * - Goes idle for a specific amount of time + * + * - Sets a timer to wake up all the idle injection threads after a + * running period + * + * That happens only when the mitigation is enabled, otherwise the + * task is scheduled out. + * + * In order to keep the tasks synchronized together, it is the last + * task exiting the idle period which is in charge of setting the + * timer. + * + * This function never returns. + */ +static int cpuidle_cooling_injection_thread(void *arg) +{ + struct sched_param param = { .sched_priority = MAX_USER_RT_PRIO/2 }; + struct cpuidle_cooling_device *idle_cdev = arg; + int index = smp_processor_id() % cpumask_weight(idle_cdev->cpumask); + DEFINE_WAIT(wait); + + set_freezable(); + + sched_setscheduler(current, SCHED_FIFO, ¶m); + + while (1) { + + s64 next_wakeup; + + prepare_to_wait(&idle_cdev->waitq[index], + &wait, TASK_INTERRUPTIBLE); + + schedule(); + + atomic_inc(&idle_cdev->count); + + play_idle(idle_cdev->idle_cycle / USEC_PER_MSEC); + + /* + * The last CPU waking up is in charge of setting the + * timer. If the CPU is hotplugged, the timer will + * move to another CPU (which may not belong to the + * same cluster) but that is not a problem as the + * timer will be set again by another CPU belonging to + * the cluster, so this mechanism is self adaptive and + * does not require any hotplugging dance. + */ + if (!atomic_dec_and_test(&idle_cdev->count)) + continue; + + if (!idle_cdev->state) + continue; + + next_wakeup = cpuidle_cooling_runtime(idle_cdev); + + hrtimer_start(&idle_cdev->timer, ns_to_ktime(next_wakeup), + HRTIMER_MODE_REL_PINNED); + } + + finish_wait(&idle_cdev->waitq[index], &wait); + + return 0; +} + +/** + * cpuidle_cooling_get_max_state - Get the maximum state + * @cdev : the thermal cooling device + * @state : a pointer to the state variable to be filled + * + * The function gives always 100 as the injection ratio is percentile + * based for consistency accros different platforms. + * + * The function can not fail, it returns always zero. + */ +static int cpuidle_cooling_get_max_state(struct thermal_cooling_device *cdev, + unsigned long *state) +{ + /* + * Depending on the configuration or the hardware, the running + * cycle and the idle cycle could be different. We want unify + * that to an 0..100 interval, so the set state interface will + * be the same whatever the platform is. + * + * The state 100% will make the cluster 100% ... idle. A 0% + * injection ratio means no idle injection at all and 50% + * means for 10ms of idle injection, we have 10ms of running + * time. + */ + *state = 100; + + return 0; +} + +/** + * cpuidle_cooling_get_cur_state - Get the current cooling state + * @cdev: the thermal cooling device + * @state: a pointer to the state + * + * The function just copy the state value from the private thermal + * cooling device structure, the mapping is 1 <-> 1. + * + * The function can not fail, it returns always zero. + */ +static int cpuidle_cooling_get_cur_state(struct thermal_cooling_device *cdev, + unsigned long *state) +{ + struct cpuidle_cooling_device *idle_cdev = cdev->devdata; + + *state = idle_cdev->state; + + return 0; +} + +/** + * cpuidle_cooling_set_cur_state - Set the current cooling state + * @cdev: the thermal cooling device + * @state: the target state + * + * The function checks first if we are initiating the mitigation which + * in turn wakes up all the idle injection tasks belonging to the idle + * cooling device. In any case, it updates the internal state for the + * cooling device. + * + * The function can not fail, it returns always zero. + */ +static int cpuidle_cooling_set_cur_state(struct thermal_cooling_device *cdev, + unsigned long state) +{ + struct cpuidle_cooling_device *idle_cdev = cdev->devdata; + unsigned long current_state = idle_cdev->state; + + idle_cdev->state = state; + + if (current_state == 0 && state > 0) { + pr_debug("Starting cooling cpus '%*pbl'\n", + cpumask_pr_args(idle_cdev->cpumask)); + cpuidle_cooling_wakeup(idle_cdev); + } else if (current_state > 0 && !state) { + pr_debug("Stopping cooling cpus '%*pbl'\n", + cpumask_pr_args(idle_cdev->cpumask)); + } + + return 0; +} + +/** + * cpuidle_cooling_ops - thermal cooling device ops + */ +static struct thermal_cooling_device_ops cpuidle_cooling_ops = { + .get_max_state = cpuidle_cooling_get_max_state, + .get_cur_state = cpuidle_cooling_get_cur_state, + .set_cur_state = cpuidle_cooling_set_cur_state, +}; + +/** + * cpuidle_cooling_release - Kref based release helper + * @kref: a pointer to the kref structure + * + * This function is automatically called by the kref_put function when + * the idle cooling device refcount reaches zero. At this point, we + * have the guarantee the structure is no longer in use and we can + * safely release all the ressources. + */ +static void __init cpuidle_cooling_release(struct kref *kref) +{ + struct cpuidle_cooling_device *idle_cdev = + container_of(kref, struct cpuidle_cooling_device, kref); + + thermal_cooling_device_unregister(idle_cdev->cdev); + kfree(idle_cdev->waitq); + kfree(idle_cdev->tsk); + kfree(idle_cdev); +} + +/** + * cpuidle_cooling_register - Idle cooling device initialization function + * + * This function is in charge of creating a cooling device per cluster + * and register it to thermal framework. For this we rely on the + * topology as there is nothing yet describing better the idle state + * power domains. + * + * For each first CPU of the cluster's cpumask, we allocate the idle + * cooling device, initialize the general fields and then we initialze + * the rest in a per cpu basis. + * + * Returns zero on success, < 0 otherwise. + */ +int cpuidle_cooling_register(void) +{ + struct cpuidle_cooling_device *idle_cdev = NULL; + struct thermal_cooling_device *cdev; + struct task_struct *tsk; + struct device_node *np; + cpumask_t *cpumask; + char dev_name[THERMAL_NAME_LENGTH]; + int weight; + int ret = -ENOMEM, cpu; + int index = 0; + + for_each_possible_cpu(cpu) { + + cpumask = topology_core_cpumask(cpu); + weight = cpumask_weight(cpumask); + + /* + * This condition makes the first cpu belonging to the + * cluster to create a cooling device and allocates + * the structure. Others CPUs belonging to the same + * cluster will just increment the refcount on the + * cooling device structure and initialize it. + */ + if (cpu == cpumask_first(cpumask)) { + + np = of_cpu_device_node_get(cpu); + + idle_cdev = kzalloc(sizeof(*idle_cdev), GFP_KERNEL); + if (!idle_cdev) + goto out_fail; + + idle_cdev->tsk = kzalloc(sizeof(*idle_cdev->tsk) * + weight, GFP_KERNEL); + if (!idle_cdev->tsk) + goto out_fail; + + idle_cdev->waitq = kzalloc(sizeof(*idle_cdev->waitq) * + weight, GFP_KERNEL); + if (!idle_cdev->waitq) + goto out_fail; + + idle_cdev->idle_cycle = DEFAULT_IDLE_TIME_US; + + atomic_set(&idle_cdev->count, 0); + + kref_init(&idle_cdev->kref); + + /* + * Initialize the timer to wakeup all the idle + * injection tasks + */ + hrtimer_init(&idle_cdev->timer, + CLOCK_MONOTONIC, HRTIMER_MODE_REL); + + /* + * The wakeup function callback which is in + * charge of waking up all CPUs belonging to + * the same cluster + */ + idle_cdev->timer.function = cpuidle_cooling_wakeup_fn; + + /* + * The thermal cooling device name + */ + snprintf(dev_name, sizeof(dev_name), "thermal-idle-%d", index++); + cdev = thermal_of_cooling_device_register(np, dev_name, + idle_cdev, + &cpuidle_cooling_ops); + if (IS_ERR(cdev)) { + ret = PTR_ERR(cdev); + goto out_fail; + } + + idle_cdev->cdev = cdev; + + idle_cdev->cpumask = cpumask; + + list_add(&idle_cdev->node, &cpuidle_cdev_list); + + pr_info("Created idle cooling device for cpus '%*pbl'\n", + cpumask_pr_args(cpumask)); + } + + kref_get(&idle_cdev->kref); + + /* + * Each cooling device is per package. Each package + * has a set of cpus where the physical number is + * duplicate in the kernel namespace. We need a way to + * address the waitq[] and tsk[] arrays with index + * which are not Linux cpu numbered. + * + * One solution is to use the + * topology_core_id(cpu). Other solution is to use the + * modulo. + * + * eg. 2 x cluster - 4 cores. + * + * Physical numbering -> Linux numbering -> % nr_cpus + * + * Pkg0 - Cpu0 -> 0 -> 0 + * Pkg0 - Cpu1 -> 1 -> 1 + * Pkg0 - Cpu2 -> 2 -> 2 + * Pkg0 - Cpu3 -> 3 -> 3 + * + * Pkg1 - Cpu0 -> 4 -> 0 + * Pkg1 - Cpu1 -> 5 -> 1 + * Pkg1 - Cpu2 -> 6 -> 2 + * Pkg1 - Cpu3 -> 7 -> 3 + */ + init_waitqueue_head(&idle_cdev->waitq[cpu % weight]); + + tsk = kthread_create_on_cpu(cpuidle_cooling_injection_thread, + idle_cdev, cpu, "kidle_inject/%u"); + if (IS_ERR(tsk)) { + ret = PTR_ERR(tsk); + goto out_fail; + } + + idle_cdev->tsk[cpu % weight] = tsk; + + wake_up_process(tsk); + } + + return 0; + +out_fail: + list_for_each_entry(idle_cdev, &cpuidle_cdev_list, node) { + + for_each_cpu(cpu, idle_cdev->cpumask) { + + if (idle_cdev->tsk[cpu]) + kthread_stop(idle_cdev->tsk[cpu]); + + kref_put(&idle_cdev->kref, cpuidle_cooling_release); + } + } + + pr_err("Failed to create idle cooling device (%d)\n", ret); + + return ret; +} +#endif diff --git a/include/linux/cpu_cooling.h b/include/linux/cpu_cooling.h index d4292eb..2b5950b 100644 --- a/include/linux/cpu_cooling.h +++ b/include/linux/cpu_cooling.h @@ -45,6 +45,7 @@ struct thermal_cooling_device * cpufreq_power_cooling_register(struct cpufreq_policy *policy, u32 capacitance, get_static_t plat_static_func); +extern int cpuidle_cooling_register(void); /** * of_cpufreq_cooling_register - create cpufreq cooling device based on DT. * @np: a valid struct device_node to the cooling device device tree node. @@ -118,6 +119,11 @@ void cpufreq_cooling_unregister(struct thermal_cooling_device *cdev) { return; } + +static inline int cpuidle_cooling_register(void) +{ + return 0; +} #endif /* CONFIG_CPU_THERMAL */ #endif /* __CPU_COOLING_H__ */