|  | // SPDX-License-Identifier: GPL-2.0-or-later | 
|  | /* | 
|  | * Copyright (C) 2014 Imagination Technologies | 
|  | * Author: Paul Burton <paul.burton@mips.com> | 
|  | */ | 
|  |  | 
|  | #include <linux/cpu_pm.h> | 
|  | #include <linux/cpuidle.h> | 
|  | #include <linux/init.h> | 
|  |  | 
|  | #include <asm/idle.h> | 
|  | #include <asm/pm-cps.h> | 
|  |  | 
|  | /* Enumeration of the various idle states this driver may enter */ | 
|  | enum cps_idle_state { | 
|  | STATE_WAIT = 0,		/* MIPS wait instruction, coherent */ | 
|  | STATE_NC_WAIT,		/* MIPS wait instruction, non-coherent */ | 
|  | STATE_CLOCK_GATED,	/* Core clock gated */ | 
|  | STATE_POWER_GATED,	/* Core power gated */ | 
|  | STATE_COUNT | 
|  | }; | 
|  |  | 
|  | static int cps_nc_enter(struct cpuidle_device *dev, | 
|  | struct cpuidle_driver *drv, int index) | 
|  | { | 
|  | enum cps_pm_state pm_state; | 
|  | int err; | 
|  |  | 
|  | /* | 
|  | * At least one core must remain powered up & clocked in order for the | 
|  | * system to have any hope of functioning. | 
|  | * | 
|  | * TODO: don't treat core 0 specially, just prevent the final core | 
|  | * TODO: remap interrupt affinity temporarily | 
|  | */ | 
|  | if (cpus_are_siblings(0, dev->cpu) && (index > STATE_NC_WAIT)) | 
|  | index = STATE_NC_WAIT; | 
|  |  | 
|  | /* Select the appropriate cps_pm_state */ | 
|  | switch (index) { | 
|  | case STATE_NC_WAIT: | 
|  | pm_state = CPS_PM_NC_WAIT; | 
|  | break; | 
|  | case STATE_CLOCK_GATED: | 
|  | pm_state = CPS_PM_CLOCK_GATED; | 
|  | break; | 
|  | case STATE_POWER_GATED: | 
|  | pm_state = CPS_PM_POWER_GATED; | 
|  | break; | 
|  | default: | 
|  | BUG(); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | /* Notify listeners the CPU is about to power down */ | 
|  | if ((pm_state == CPS_PM_POWER_GATED) && cpu_pm_enter()) | 
|  | return -EINTR; | 
|  |  | 
|  | /* Enter that state */ | 
|  | err = cps_pm_enter_state(pm_state); | 
|  |  | 
|  | /* Notify listeners the CPU is back up */ | 
|  | if (pm_state == CPS_PM_POWER_GATED) | 
|  | cpu_pm_exit(); | 
|  |  | 
|  | return err ?: index; | 
|  | } | 
|  |  | 
|  | static struct cpuidle_driver cps_driver = { | 
|  | .name			= "cpc_cpuidle", | 
|  | .owner			= THIS_MODULE, | 
|  | .states = { | 
|  | [STATE_WAIT] = MIPS_CPUIDLE_WAIT_STATE, | 
|  | [STATE_NC_WAIT] = { | 
|  | .enter	= cps_nc_enter, | 
|  | .exit_latency		= 200, | 
|  | .target_residency	= 450, | 
|  | .name	= "nc-wait", | 
|  | .desc	= "non-coherent MIPS wait", | 
|  | }, | 
|  | [STATE_CLOCK_GATED] = { | 
|  | .enter	= cps_nc_enter, | 
|  | .exit_latency		= 300, | 
|  | .target_residency	= 700, | 
|  | .flags	= CPUIDLE_FLAG_TIMER_STOP, | 
|  | .name	= "clock-gated", | 
|  | .desc	= "core clock gated", | 
|  | }, | 
|  | [STATE_POWER_GATED] = { | 
|  | .enter	= cps_nc_enter, | 
|  | .exit_latency		= 600, | 
|  | .target_residency	= 1000, | 
|  | .flags	= CPUIDLE_FLAG_TIMER_STOP, | 
|  | .name	= "power-gated", | 
|  | .desc	= "core power gated", | 
|  | }, | 
|  | }, | 
|  | .state_count		= STATE_COUNT, | 
|  | .safe_state_index	= 0, | 
|  | }; | 
|  |  | 
|  | static void __init cps_cpuidle_unregister(void) | 
|  | { | 
|  | int cpu; | 
|  | struct cpuidle_device *device; | 
|  |  | 
|  | for_each_possible_cpu(cpu) { | 
|  | device = &per_cpu(cpuidle_dev, cpu); | 
|  | cpuidle_unregister_device(device); | 
|  | } | 
|  |  | 
|  | cpuidle_unregister_driver(&cps_driver); | 
|  | } | 
|  |  | 
|  | static int __init cps_cpuidle_init(void) | 
|  | { | 
|  | int err, cpu, i; | 
|  | struct cpuidle_device *device; | 
|  |  | 
|  | /* Detect supported states */ | 
|  | if (!cps_pm_support_state(CPS_PM_POWER_GATED)) | 
|  | cps_driver.state_count = STATE_CLOCK_GATED + 1; | 
|  | if (!cps_pm_support_state(CPS_PM_CLOCK_GATED)) | 
|  | cps_driver.state_count = STATE_NC_WAIT + 1; | 
|  | if (!cps_pm_support_state(CPS_PM_NC_WAIT)) | 
|  | cps_driver.state_count = STATE_WAIT + 1; | 
|  |  | 
|  | /* Inform the user if some states are unavailable */ | 
|  | if (cps_driver.state_count < STATE_COUNT) { | 
|  | pr_info("cpuidle-cps: limited to "); | 
|  | switch (cps_driver.state_count - 1) { | 
|  | case STATE_WAIT: | 
|  | pr_cont("coherent wait\n"); | 
|  | break; | 
|  | case STATE_NC_WAIT: | 
|  | pr_cont("non-coherent wait\n"); | 
|  | break; | 
|  | case STATE_CLOCK_GATED: | 
|  | pr_cont("clock gating\n"); | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Set the coupled flag on the appropriate states if this system | 
|  | * requires it. | 
|  | */ | 
|  | if (coupled_coherence) | 
|  | for (i = STATE_NC_WAIT; i < cps_driver.state_count; i++) | 
|  | cps_driver.states[i].flags |= CPUIDLE_FLAG_COUPLED; | 
|  |  | 
|  | err = cpuidle_register_driver(&cps_driver); | 
|  | if (err) { | 
|  | pr_err("Failed to register CPS cpuidle driver\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | for_each_possible_cpu(cpu) { | 
|  | device = &per_cpu(cpuidle_dev, cpu); | 
|  | device->cpu = cpu; | 
|  | #ifdef CONFIG_ARCH_NEEDS_CPU_IDLE_COUPLED | 
|  | cpumask_copy(&device->coupled_cpus, &cpu_sibling_map[cpu]); | 
|  | #endif | 
|  |  | 
|  | err = cpuidle_register_device(device); | 
|  | if (err) { | 
|  | pr_err("Failed to register CPU%d cpuidle device\n", | 
|  | cpu); | 
|  | goto err_out; | 
|  | } | 
|  | } | 
|  |  | 
|  | return 0; | 
|  | err_out: | 
|  | cps_cpuidle_unregister(); | 
|  | return err; | 
|  | } | 
|  | device_initcall(cps_cpuidle_init); |