[PATCH v3 1/7] cpuidle: Add common init interface and idle functionality
Turquette, Mike
mturquette at ti.com
Tue Jan 24 18:46:55 EST 2012
On Mon, Jan 23, 2012 at 8:37 PM, Robert Lee <rob.lee at linaro.org> wrote:
> diff --git a/drivers/cpuidle/common.c b/drivers/cpuidle/common.c
> new file mode 100644
> index 0000000..dafa758
> --- /dev/null
> +++ b/drivers/cpuidle/common.c
> @@ -0,0 +1,152 @@
> +/*
> + * Copyright 2011 Freescale Semiconductor, Inc.
> + * Copyright 2011 Linaro Ltd.
> + *
> + * The code contained herein is licensed under the GNU General Public
> + * License. You may obtain a copy of the GNU General Public License
> + * Version 2 or later at the following locations:
> + *
> + * http://www.opensource.org/licenses/gpl-license.html
> + * http://www.gnu.org/copyleft/gpl.html
> + */
> +
> +/*
> + * This code performs provides some commonly used cpuidle setup functionality
s/performs //
> +static struct cpuidle_device __percpu * common_cpuidle_devices;
You can use DECLARE_PER_CPU here.
Is there any particular reason to allocate these dynamically? You can
replace the code above with,
static DEFINE_PER_CPU(struct cpuidle_device, common_cpuidle_devices);
I might change the variable name to "cpu_cpuidle_device" in that case
since you are addressing a single CPU when using the per cpu accessor
functions and "common_cpuidle_devices" sounds like an array or a list
or something. No big deal to keep the current name though.
> +static int simple_enter(struct cpuidle_device *dev,
> + struct cpuidle_driver *drv, int index)
> +{
> + ktime_t time_start, time_end;
> +
> + local_irq_disable();
> +
> + time_start = ktime_get();
> +
> + index = do_idle[index](dev, drv, index);
> +
> + time_end = ktime_get();
> +
> + local_irq_enable();
> +
> + dev->last_residency =
> + (int)ktime_to_us(ktime_sub(time_end, time_start));
Sometimes an attempt to enter some C-state fails and the do_idle will
return immediately. What do you think about having do_idle return
-EERROR in this case and the conditionally setting last_residency to
zero in those cases? The point is that a C-state's total residency
time should not increase in the case where the hardware did not
successfully transition into that C-state. I've observed many times
where a specific low power state was actually achieved in the hardware
but /sys/devices/system/cpu/cpuN/cpuidle/stateM/time keeps
incrementing (albeit in very tiny increments). Something like,
if (IS_ERR(index))
dev->last_residency = 0;
else
...
Note: I haven't been through the CPUidle core in a while so maybe the
above suggestion violates some other requirements/assumptions...
> +
> + return index;
> +}
> +
> +void common_cpuidle_devices_uninit(void)
> +{
> + int cpu_id;
> + struct cpuidle_device *dev;
> +
> + for_each_possible_cpu(cpu_id) {
> + dev = per_cpu_ptr(common_cpuidle_devices, cpu_id);
> + cpuidle_unregister_device(dev);
> + }
> +
> + free_percpu(common_cpuidle_devices);
See note above about statically allocating the per-cpu variables.
> +}
> +
> +/**
> + * common_cpuidle_init() - Provides some commonly used init functionality.
> + * @pdrv Pointer to your cpuidle_driver object.
> + * @simple Use the common simple_enter wrapper?
> + * @driver_data_init Pointer to your platform function to initialize your
> + * platform specific driver data. Use NULL if platform
> + * specific data is not needed.
> + *
> + * Common cpuidle init interface to provide common cpuidle functionality
> + * used by many platforms.
> + */
> +int __init common_cpuidle_init(struct cpuidle_driver *pdrv, bool simple,
> + void (*driver_data_init)(struct cpuidle_device *dev))
> +{
> + struct cpuidle_device *dev;
> + struct cpuidle_driver *drv;
> + int i, cpu_id, ret;
> +
> + if (!pdrv || pdrv->state_count > CPUIDLE_STATE_MAX) {
> + pr_err("%s: Invalid Input\n", __func__);
> + return -EINVAL;
> + }
> +
> + drv = kmalloc(sizeof(struct cpuidle_driver), GFP_KERNEL);
> + if (!drv) {
> + pr_err("%s: no memory for cpuidle driver\n", __func__);
> + return -ENOMEM;
> + }
> +
> + *drv = *pdrv;
> +
> + for (i = 0; simple && (i < drv->state_count); i++) {
> + do_idle[i] = drv->states[i].enter;
> + drv->states[i].enter = simple_enter;
> + }
> +
> + ret = cpuidle_register_driver(drv);
> + if (ret) {
> + pr_err("%s: Failed to register cpuidle driver\n", __func__);
> + goto free_drv;
> + }
> +
> + common_cpuidle_devices = alloc_percpu(struct cpuidle_device);
> + if (common_cpuidle_devices == NULL) {
> + ret = -ENOMEM;
> + goto unregister_drv;
> + }
See note above about statically allocating these.
Regards,
Mike
More information about the linux-arm-kernel
mailing list