1 /* 2 * linux/arch/arm/mach-omap2/cpuidle34xx.c 3 * 4 * OMAP3 CPU IDLE Routines 5 * 6 * Copyright (C) 2008 Texas Instruments, Inc. 7 * Rajendra Nayak <rnayak@ti.com> 8 * 9 * Copyright (C) 2007 Texas Instruments, Inc. 10 * Karthik Dasu <karthik-dp@ti.com> 11 * 12 * Copyright (C) 2006 Nokia Corporation 13 * Tony Lindgren <tony@atomide.com> 14 * 15 * Copyright (C) 2005 Texas Instruments, Inc. 16 * Richard Woodruff <r-woodruff2@ti.com> 17 * 18 * Based on pm.c for omap2 19 * 20 * This program is free software; you can redistribute it and/or modify 21 * it under the terms of the GNU General Public License version 2 as 22 * published by the Free Software Foundation. 23 */ 24 25 #include <linux/sched.h> 26 #include <linux/cpuidle.h> 27 #include <linux/export.h> 28 #include <linux/cpu_pm.h> 29 30 #include "powerdomain.h" 31 #include "clockdomain.h" 32 33 #include "pm.h" 34 #include "control.h" 35 #include "common.h" 36 37 /* Mach specific information to be recorded in the C-state driver_data */ 38 struct omap3_idle_statedata { 39 u32 mpu_state; 40 u32 core_state; 41 }; 42 43 static struct powerdomain *mpu_pd, *core_pd, *per_pd, *cam_pd; 44 45 static struct omap3_idle_statedata omap3_idle_data[] = { 46 { 47 .mpu_state = PWRDM_POWER_ON, 48 .core_state = PWRDM_POWER_ON, 49 }, 50 { 51 .mpu_state = PWRDM_POWER_ON, 52 .core_state = PWRDM_POWER_ON, 53 }, 54 { 55 .mpu_state = PWRDM_POWER_RET, 56 .core_state = PWRDM_POWER_ON, 57 }, 58 { 59 .mpu_state = PWRDM_POWER_OFF, 60 .core_state = PWRDM_POWER_ON, 61 }, 62 { 63 .mpu_state = PWRDM_POWER_RET, 64 .core_state = PWRDM_POWER_RET, 65 }, 66 { 67 .mpu_state = PWRDM_POWER_OFF, 68 .core_state = PWRDM_POWER_RET, 69 }, 70 { 71 .mpu_state = PWRDM_POWER_OFF, 72 .core_state = PWRDM_POWER_OFF, 73 }, 74 }; 75 76 /* Private functions */ 77 78 static int __omap3_enter_idle(struct cpuidle_device *dev, 79 struct cpuidle_driver *drv, 80 int index) 81 { 82 struct omap3_idle_statedata *cx = &omap3_idle_data[index]; 83 u32 mpu_state = cx->mpu_state, core_state = cx->core_state; 84 85 local_fiq_disable(); 86 87 pwrdm_set_next_pwrst(mpu_pd, mpu_state); 88 pwrdm_set_next_pwrst(core_pd, core_state); 89 90 if (omap_irq_pending() || need_resched()) 91 goto return_sleep_time; 92 93 /* Deny idle for C1 */ 94 if (index == 0) { 95 clkdm_deny_idle(mpu_pd->pwrdm_clkdms[0]); 96 clkdm_deny_idle(core_pd->pwrdm_clkdms[0]); 97 } 98 99 /* 100 * Call idle CPU PM enter notifier chain so that 101 * VFP context is saved. 102 */ 103 if (mpu_state == PWRDM_POWER_OFF) 104 cpu_pm_enter(); 105 106 /* Execute ARM wfi */ 107 omap_sram_idle(); 108 109 /* 110 * Call idle CPU PM enter notifier chain to restore 111 * VFP context. 112 */ 113 if (pwrdm_read_prev_pwrst(mpu_pd) == PWRDM_POWER_OFF) 114 cpu_pm_exit(); 115 116 /* Re-allow idle for C1 */ 117 if (index == 0) { 118 clkdm_allow_idle(mpu_pd->pwrdm_clkdms[0]); 119 clkdm_allow_idle(core_pd->pwrdm_clkdms[0]); 120 } 121 122 return_sleep_time: 123 124 local_fiq_enable(); 125 126 return index; 127 } 128 129 /** 130 * omap3_enter_idle - Programs OMAP3 to enter the specified state 131 * @dev: cpuidle device 132 * @drv: cpuidle driver 133 * @index: the index of state to be entered 134 * 135 * Called from the CPUidle framework to program the device to the 136 * specified target state selected by the governor. 137 */ 138 static inline int omap3_enter_idle(struct cpuidle_device *dev, 139 struct cpuidle_driver *drv, 140 int index) 141 { 142 return cpuidle_wrap_enter(dev, drv, index, __omap3_enter_idle); 143 } 144 145 /** 146 * next_valid_state - Find next valid C-state 147 * @dev: cpuidle device 148 * @drv: cpuidle driver 149 * @index: Index of currently selected c-state 150 * 151 * If the state corresponding to index is valid, index is returned back 152 * to the caller. Else, this function searches for a lower c-state which is 153 * still valid (as defined in omap3_power_states[]) and returns its index. 154 * 155 * A state is valid if the 'valid' field is enabled and 156 * if it satisfies the enable_off_mode condition. 157 */ 158 static int next_valid_state(struct cpuidle_device *dev, 159 struct cpuidle_driver *drv, int index) 160 { 161 struct omap3_idle_statedata *cx = &omap3_idle_data[index]; 162 u32 mpu_deepest_state = PWRDM_POWER_RET; 163 u32 core_deepest_state = PWRDM_POWER_RET; 164 int idx; 165 int next_index = 0; /* C1 is the default value */ 166 167 if (enable_off_mode) { 168 mpu_deepest_state = PWRDM_POWER_OFF; 169 /* 170 * Erratum i583: valable for ES rev < Es1.2 on 3630. 171 * CORE OFF mode is not supported in a stable form, restrict 172 * instead the CORE state to RET. 173 */ 174 if (!IS_PM34XX_ERRATUM(PM_SDRC_WAKEUP_ERRATUM_i583)) 175 core_deepest_state = PWRDM_POWER_OFF; 176 } 177 178 /* Check if current state is valid */ 179 if ((cx->mpu_state >= mpu_deepest_state) && 180 (cx->core_state >= core_deepest_state)) 181 return index; 182 183 /* 184 * Drop to next valid state. 185 * Start search from the next (lower) state. 186 */ 187 for (idx = index - 1; idx >= 0; idx--) { 188 cx = &omap3_idle_data[idx]; 189 if ((cx->mpu_state >= mpu_deepest_state) && 190 (cx->core_state >= core_deepest_state)) { 191 next_index = idx; 192 break; 193 } 194 } 195 196 return next_index; 197 } 198 199 /** 200 * omap3_enter_idle_bm - Checks for any bus activity 201 * @dev: cpuidle device 202 * @drv: cpuidle driver 203 * @index: array index of target state to be programmed 204 * 205 * This function checks for any pending activity and then programs 206 * the device to the specified or a safer state. 207 */ 208 static int omap3_enter_idle_bm(struct cpuidle_device *dev, 209 struct cpuidle_driver *drv, 210 int index) 211 { 212 int new_state_idx; 213 u32 core_next_state, per_next_state = 0, per_saved_state = 0; 214 struct omap3_idle_statedata *cx; 215 int ret; 216 217 /* 218 * Use only C1 if CAM is active. 219 * CAM does not have wakeup capability in OMAP3. 220 */ 221 if (pwrdm_read_pwrst(cam_pd) == PWRDM_POWER_ON) 222 new_state_idx = drv->safe_state_index; 223 else 224 new_state_idx = next_valid_state(dev, drv, index); 225 226 /* 227 * FIXME: we currently manage device-specific idle states 228 * for PER and CORE in combination with CPU-specific 229 * idle states. This is wrong, and device-specific 230 * idle management needs to be separated out into 231 * its own code. 232 */ 233 234 /* Program PER state */ 235 cx = &omap3_idle_data[new_state_idx]; 236 core_next_state = cx->core_state; 237 per_next_state = per_saved_state = pwrdm_read_next_pwrst(per_pd); 238 if (new_state_idx == 0) { 239 /* In C1 do not allow PER state lower than CORE state */ 240 if (per_next_state < core_next_state) 241 per_next_state = core_next_state; 242 } else { 243 /* 244 * Prevent PER OFF if CORE is not in RETention or OFF as this 245 * would disable PER wakeups completely. 246 */ 247 if ((per_next_state == PWRDM_POWER_OFF) && 248 (core_next_state > PWRDM_POWER_RET)) 249 per_next_state = PWRDM_POWER_RET; 250 } 251 252 /* Are we changing PER target state? */ 253 if (per_next_state != per_saved_state) 254 pwrdm_set_next_pwrst(per_pd, per_next_state); 255 256 ret = omap3_enter_idle(dev, drv, new_state_idx); 257 258 /* Restore original PER state if it was modified */ 259 if (per_next_state != per_saved_state) 260 pwrdm_set_next_pwrst(per_pd, per_saved_state); 261 262 return ret; 263 } 264 265 static DEFINE_PER_CPU(struct cpuidle_device, omap3_idle_dev); 266 267 static struct cpuidle_driver omap3_idle_driver = { 268 .name = "omap3_idle", 269 .owner = THIS_MODULE, 270 .states = { 271 { 272 .enter = omap3_enter_idle_bm, 273 .exit_latency = 2 + 2, 274 .target_residency = 5, 275 .flags = CPUIDLE_FLAG_TIME_VALID, 276 .name = "C1", 277 .desc = "MPU ON + CORE ON", 278 }, 279 { 280 .enter = omap3_enter_idle_bm, 281 .exit_latency = 10 + 10, 282 .target_residency = 30, 283 .flags = CPUIDLE_FLAG_TIME_VALID, 284 .name = "C2", 285 .desc = "MPU ON + CORE ON", 286 }, 287 { 288 .enter = omap3_enter_idle_bm, 289 .exit_latency = 50 + 50, 290 .target_residency = 300, 291 .flags = CPUIDLE_FLAG_TIME_VALID, 292 .name = "C3", 293 .desc = "MPU RET + CORE ON", 294 }, 295 { 296 .enter = omap3_enter_idle_bm, 297 .exit_latency = 1500 + 1800, 298 .target_residency = 4000, 299 .flags = CPUIDLE_FLAG_TIME_VALID, 300 .name = "C4", 301 .desc = "MPU OFF + CORE ON", 302 }, 303 { 304 .enter = omap3_enter_idle_bm, 305 .exit_latency = 2500 + 7500, 306 .target_residency = 12000, 307 .flags = CPUIDLE_FLAG_TIME_VALID, 308 .name = "C5", 309 .desc = "MPU RET + CORE RET", 310 }, 311 { 312 .enter = omap3_enter_idle_bm, 313 .exit_latency = 3000 + 8500, 314 .target_residency = 15000, 315 .flags = CPUIDLE_FLAG_TIME_VALID, 316 .name = "C6", 317 .desc = "MPU OFF + CORE RET", 318 }, 319 { 320 .enter = omap3_enter_idle_bm, 321 .exit_latency = 10000 + 30000, 322 .target_residency = 30000, 323 .flags = CPUIDLE_FLAG_TIME_VALID, 324 .name = "C7", 325 .desc = "MPU OFF + CORE OFF", 326 }, 327 }, 328 .state_count = ARRAY_SIZE(omap3_idle_data), 329 .safe_state_index = 0, 330 }; 331 332 /* Public functions */ 333 334 /** 335 * omap3_idle_init - Init routine for OMAP3 idle 336 * 337 * Registers the OMAP3 specific cpuidle driver to the cpuidle 338 * framework with the valid set of states. 339 */ 340 int __init omap3_idle_init(void) 341 { 342 struct cpuidle_device *dev; 343 344 mpu_pd = pwrdm_lookup("mpu_pwrdm"); 345 core_pd = pwrdm_lookup("core_pwrdm"); 346 per_pd = pwrdm_lookup("per_pwrdm"); 347 cam_pd = pwrdm_lookup("cam_pwrdm"); 348 349 if (!mpu_pd || !core_pd || !per_pd || !cam_pd) 350 return -ENODEV; 351 352 cpuidle_register_driver(&omap3_idle_driver); 353 354 dev = &per_cpu(omap3_idle_dev, smp_processor_id()); 355 dev->cpu = 0; 356 357 if (cpuidle_register_device(dev)) { 358 printk(KERN_ERR "%s: CPUidle register device failed\n", 359 __func__); 360 return -EIO; 361 } 362 363 return 0; 364 } 365