|
@@ -72,12 +72,94 @@ static const struct of_device_id arm_idle_state_match[] __initconst = {
|
|
|
};
|
|
|
|
|
|
/*
|
|
|
- * arm_idle_init
|
|
|
+ * arm_idle_init_cpu
|
|
|
*
|
|
|
* Registers the arm specific cpuidle driver with the cpuidle
|
|
|
* framework. It relies on core code to parse the idle states
|
|
|
* and initialize them using driver data structures accordingly.
|
|
|
*/
|
|
|
+static int __init arm_idle_init_cpu(int cpu)
|
|
|
+{
|
|
|
+ int ret;
|
|
|
+ struct cpuidle_driver *drv;
|
|
|
+ struct cpuidle_device *dev;
|
|
|
+
|
|
|
+ drv = kmemdup(&arm_idle_driver, sizeof(*drv), GFP_KERNEL);
|
|
|
+ if (!drv)
|
|
|
+ return -ENOMEM;
|
|
|
+
|
|
|
+ drv->cpumask = (struct cpumask *)cpumask_of(cpu);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Initialize idle states data, starting at index 1. This
|
|
|
+ * driver is DT only, if no DT idle states are detected (ret
|
|
|
+ * == 0) let the driver initialization fail accordingly since
|
|
|
+ * there is no reason to initialize the idle driver if only
|
|
|
+ * wfi is supported.
|
|
|
+ */
|
|
|
+ ret = dt_init_idle_driver(drv, arm_idle_state_match, 1);
|
|
|
+ if (ret <= 0) {
|
|
|
+ ret = ret ? : -ENODEV;
|
|
|
+ goto out_kfree_drv;
|
|
|
+ }
|
|
|
+
|
|
|
+ ret = cpuidle_register_driver(drv);
|
|
|
+ if (ret) {
|
|
|
+ pr_err("Failed to register cpuidle driver\n");
|
|
|
+ goto out_kfree_drv;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Call arch CPU operations in order to initialize
|
|
|
+ * idle states suspend back-end specific data
|
|
|
+ */
|
|
|
+ ret = arm_cpuidle_init(cpu);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Skip the cpuidle device initialization if the reported
|
|
|
+ * failure is a HW misconfiguration/breakage (-ENXIO).
|
|
|
+ */
|
|
|
+ if (ret == -ENXIO)
|
|
|
+ return 0;
|
|
|
+
|
|
|
+ if (ret) {
|
|
|
+ pr_err("CPU %d failed to init idle CPU ops\n", cpu);
|
|
|
+ goto out_unregister_drv;
|
|
|
+ }
|
|
|
+
|
|
|
+ dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
|
|
+ if (!dev) {
|
|
|
+ pr_err("Failed to allocate cpuidle device\n");
|
|
|
+ ret = -ENOMEM;
|
|
|
+ goto out_unregister_drv;
|
|
|
+ }
|
|
|
+ dev->cpu = cpu;
|
|
|
+
|
|
|
+ ret = cpuidle_register_device(dev);
|
|
|
+ if (ret) {
|
|
|
+ pr_err("Failed to register cpuidle device for CPU %d\n",
|
|
|
+ cpu);
|
|
|
+ goto out_kfree_dev;
|
|
|
+ }
|
|
|
+
|
|
|
+ return 0;
|
|
|
+
|
|
|
+out_kfree_dev:
|
|
|
+ kfree(dev);
|
|
|
+out_unregister_drv:
|
|
|
+ cpuidle_unregister_driver(drv);
|
|
|
+out_kfree_drv:
|
|
|
+ kfree(drv);
|
|
|
+ return ret;
|
|
|
+}
|
|
|
+
|
|
|
+/*
|
|
|
+ * arm_idle_init - Initializes arm cpuidle driver
|
|
|
+ *
|
|
|
+ * Initializes arm cpuidle driver for all CPUs, if any CPU fails
|
|
|
+ * to register cpuidle driver then rollback to cancel all CPUs
|
|
|
+ * registeration.
|
|
|
+ */
|
|
|
static int __init arm_idle_init(void)
|
|
|
{
|
|
|
int cpu, ret;
|
|
@@ -85,79 +167,20 @@ static int __init arm_idle_init(void)
|
|
|
struct cpuidle_device *dev;
|
|
|
|
|
|
for_each_possible_cpu(cpu) {
|
|
|
-
|
|
|
- drv = kmemdup(&arm_idle_driver, sizeof(*drv), GFP_KERNEL);
|
|
|
- if (!drv) {
|
|
|
- ret = -ENOMEM;
|
|
|
- goto out_fail;
|
|
|
- }
|
|
|
-
|
|
|
- drv->cpumask = (struct cpumask *)cpumask_of(cpu);
|
|
|
-
|
|
|
- /*
|
|
|
- * Initialize idle states data, starting at index 1. This
|
|
|
- * driver is DT only, if no DT idle states are detected (ret
|
|
|
- * == 0) let the driver initialization fail accordingly since
|
|
|
- * there is no reason to initialize the idle driver if only
|
|
|
- * wfi is supported.
|
|
|
- */
|
|
|
- ret = dt_init_idle_driver(drv, arm_idle_state_match, 1);
|
|
|
- if (ret <= 0) {
|
|
|
- ret = ret ? : -ENODEV;
|
|
|
- goto init_fail;
|
|
|
- }
|
|
|
-
|
|
|
- ret = cpuidle_register_driver(drv);
|
|
|
- if (ret) {
|
|
|
- pr_err("Failed to register cpuidle driver\n");
|
|
|
- goto init_fail;
|
|
|
- }
|
|
|
-
|
|
|
- /*
|
|
|
- * Call arch CPU operations in order to initialize
|
|
|
- * idle states suspend back-end specific data
|
|
|
- */
|
|
|
- ret = arm_cpuidle_init(cpu);
|
|
|
-
|
|
|
- /*
|
|
|
- * Skip the cpuidle device initialization if the reported
|
|
|
- * failure is a HW misconfiguration/breakage (-ENXIO).
|
|
|
- */
|
|
|
- if (ret == -ENXIO)
|
|
|
- continue;
|
|
|
-
|
|
|
- if (ret) {
|
|
|
- pr_err("CPU %d failed to init idle CPU ops\n", cpu);
|
|
|
- goto out_fail;
|
|
|
- }
|
|
|
-
|
|
|
- dev = kzalloc(sizeof(*dev), GFP_KERNEL);
|
|
|
- if (!dev) {
|
|
|
- pr_err("Failed to allocate cpuidle device\n");
|
|
|
- ret = -ENOMEM;
|
|
|
+ ret = arm_idle_init_cpu(cpu);
|
|
|
+ if (ret)
|
|
|
goto out_fail;
|
|
|
- }
|
|
|
- dev->cpu = cpu;
|
|
|
-
|
|
|
- ret = cpuidle_register_device(dev);
|
|
|
- if (ret) {
|
|
|
- pr_err("Failed to register cpuidle device for CPU %d\n",
|
|
|
- cpu);
|
|
|
- kfree(dev);
|
|
|
- goto out_fail;
|
|
|
- }
|
|
|
}
|
|
|
|
|
|
return 0;
|
|
|
-init_fail:
|
|
|
- kfree(drv);
|
|
|
+
|
|
|
out_fail:
|
|
|
while (--cpu >= 0) {
|
|
|
dev = per_cpu(cpuidle_devices, cpu);
|
|
|
+ drv = cpuidle_get_cpu_driver(dev);
|
|
|
cpuidle_unregister_device(dev);
|
|
|
- kfree(dev);
|
|
|
- drv = cpuidle_get_driver();
|
|
|
cpuidle_unregister_driver(drv);
|
|
|
+ kfree(dev);
|
|
|
kfree(drv);
|
|
|
}
|
|
|
|