2008-10-08 17:30:58 +05:30
/*
* linux / arch / arm / mach - omap2 / cpuidle34xx . c
*
* OMAP3 CPU IDLE Routines
*
* Copyright ( C ) 2008 Texas Instruments , Inc .
* Rajendra Nayak < rnayak @ ti . com >
*
* Copyright ( C ) 2007 Texas Instruments , Inc .
* Karthik Dasu < karthik - dp @ ti . com >
*
* Copyright ( C ) 2006 Nokia Corporation
* Tony Lindgren < tony @ atomide . com >
*
* Copyright ( C ) 2005 Texas Instruments , Inc .
* Richard Woodruff < r - woodruff2 @ ti . com >
*
* Based on pm . c for omap2
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*/
2009-03-20 15:21:02 +02:00
# include <linux/sched.h>
2008-10-08 17:30:58 +05:30
# include <linux/cpuidle.h>
2011-11-07 15:58:40 -08:00
# include <linux/export.h>
2011-09-03 22:38:27 +05:30
# include <linux/cpu_pm.h>
2013-04-23 08:54:36 +00:00
# include <asm/cpuidle.h>
2008-10-08 17:30:58 +05:30
2010-12-21 21:05:16 -07:00
# include "powerdomain.h"
2010-12-21 21:05:15 -07:00
# include "clockdomain.h"
2008-10-08 17:30:58 +05:30
2008-10-28 17:30:07 -07:00
# include "pm.h"
2010-10-08 11:40:20 -06:00
# include "control.h"
2011-12-05 09:46:24 +01:00
# include "common.h"
2008-10-28 17:30:07 -07:00
2011-05-09 12:02:14 +02:00
/* Mach specific information to be recorded in the C-state driver_data */
struct omap3_idle_statedata {
2013-01-26 00:58:12 -07:00
u8 mpu_state ;
u8 core_state ;
u8 per_min_state ;
2013-01-26 00:58:13 -07:00
u8 flags ;
2011-05-09 12:02:14 +02:00
} ;
2012-04-24 16:05:33 +02:00
2012-12-15 01:39:19 -07:00
static struct powerdomain * mpu_pd , * core_pd , * per_pd , * cam_pd ;
2013-01-26 00:58:13 -07:00
/*
* Possible flag bits for struct omap3_idle_statedata . flags :
*
* OMAP_CPUIDLE_CX_NO_CLKDM_IDLE : don ' t allow the MPU clockdomain to go
* inactive . This in turn prevents the MPU DPLL from entering autoidle
* mode , so wakeup latency is greatly reduced , at the cost of additional
* energy consumption . This also prevents the CORE clockdomain from
* entering idle .
*/
# define OMAP_CPUIDLE_CX_NO_CLKDM_IDLE BIT(0)
2013-01-26 00:58:12 -07:00
/*
* Prevent PER OFF if CORE is not in RETention or OFF as this would
* disable PER wakeups completely .
*/
2012-04-24 16:05:37 +02:00
static struct omap3_idle_statedata omap3_idle_data [ ] = {
2012-04-24 16:05:34 +02:00
{
. mpu_state = PWRDM_POWER_ON ,
. core_state = PWRDM_POWER_ON ,
2013-01-26 00:58:12 -07:00
/* In C1 do not allow PER state lower than CORE state */
. per_min_state = PWRDM_POWER_ON ,
2013-01-26 00:58:13 -07:00
. flags = OMAP_CPUIDLE_CX_NO_CLKDM_IDLE ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_ON ,
. core_state = PWRDM_POWER_ON ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_RET ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_RET ,
. core_state = PWRDM_POWER_ON ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_RET ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_OFF ,
. core_state = PWRDM_POWER_ON ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_RET ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_RET ,
. core_state = PWRDM_POWER_RET ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_OFF ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_OFF ,
. core_state = PWRDM_POWER_RET ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_OFF ,
2012-04-24 16:05:34 +02:00
} ,
{
. mpu_state = PWRDM_POWER_OFF ,
. core_state = PWRDM_POWER_OFF ,
2013-01-26 00:58:12 -07:00
. per_min_state = PWRDM_POWER_OFF ,
2012-04-24 16:05:34 +02:00
} ,
} ;
2011-05-09 12:02:14 +02:00
2013-04-12 12:35:49 +00:00
/**
* omap3_enter_idle - Programs OMAP3 to enter the specified state
* @ dev : cpuidle device
* @ drv : cpuidle driver
* @ index : the index of state to be entered
*/
static int omap3_enter_idle ( struct cpuidle_device * dev ,
struct cpuidle_driver * drv ,
int index )
2008-10-08 17:30:58 +05:30
{
2012-04-24 16:05:35 +02:00
struct omap3_idle_statedata * cx = & omap3_idle_data [ index ] ;
2008-10-08 17:30:58 +05:30
2009-03-20 15:21:02 +02:00
if ( omap_irq_pending ( ) | | need_resched ( ) )
2008-10-08 17:31:22 +05:30
goto return_sleep_time ;
2008-10-08 17:30:58 +05:30
2011-05-09 12:02:14 +02:00
/* Deny idle for C1 */
2013-01-26 00:58:13 -07:00
if ( cx - > flags & OMAP_CPUIDLE_CX_NO_CLKDM_IDLE ) {
2012-06-01 17:11:08 +02:00
clkdm_deny_idle ( mpu_pd - > pwrdm_clkdms [ 0 ] ) ;
2013-01-26 00:58:13 -07:00
} else {
pwrdm_set_next_pwrst ( mpu_pd , cx - > mpu_state ) ;
pwrdm_set_next_pwrst ( core_pd , cx - > core_state ) ;
2009-03-13 18:19:16 +02:00
}
2011-09-03 22:38:27 +05:30
/*
* Call idle CPU PM enter notifier chain so that
* VFP context is saved .
*/
2013-01-26 00:58:13 -07:00
if ( cx - > mpu_state = = PWRDM_POWER_OFF )
2011-09-03 22:38:27 +05:30
cpu_pm_enter ( ) ;
2008-10-08 17:30:58 +05:30
/* Execute ARM wfi */
omap_sram_idle ( ) ;
2011-09-03 22:38:27 +05:30
/*
* Call idle CPU PM enter notifier chain to restore
* VFP context .
*/
2013-01-26 00:58:13 -07:00
if ( cx - > mpu_state = = PWRDM_POWER_OFF & &
pwrdm_read_prev_pwrst ( mpu_pd ) = = PWRDM_POWER_OFF )
2011-09-03 22:38:27 +05:30
cpu_pm_exit ( ) ;
2011-05-09 12:02:14 +02:00
/* Re-allow idle for C1 */
2013-01-26 00:58:13 -07:00
if ( cx - > flags & OMAP_CPUIDLE_CX_NO_CLKDM_IDLE )
2012-06-01 17:11:08 +02:00
clkdm_allow_idle ( mpu_pd - > pwrdm_clkdms [ 0 ] ) ;
2009-03-13 18:19:16 +02:00
2008-10-08 17:31:22 +05:30
return_sleep_time :
2008-10-08 17:30:58 +05:30
2011-10-28 16:20:09 +05:30
return index ;
2008-10-08 17:30:58 +05:30
}
2010-01-28 23:16:43 +05:30
/**
2011-05-09 12:02:16 +02:00
* next_valid_state - Find next valid C - state
2010-01-28 23:16:43 +05:30
* @ dev : cpuidle device
2011-10-28 16:20:42 +05:30
* @ drv : cpuidle driver
2011-10-28 16:20:09 +05:30
* @ index : Index of currently selected c - state
2010-01-28 23:16:43 +05:30
*
2011-10-28 16:20:09 +05:30
* If the state corresponding to index is valid , index is returned back
* to the caller . Else , this function searches for a lower c - state which is
* still valid ( as defined in omap3_power_states [ ] ) and returns its index .
2011-05-09 12:02:16 +02:00
*
* A state is valid if the ' valid ' field is enabled and
* if it satisfies the enable_off_mode condition .
2010-01-28 23:16:43 +05:30
*/
2011-10-28 16:20:09 +05:30
static int next_valid_state ( struct cpuidle_device * dev ,
2012-04-24 16:05:36 +02:00
struct cpuidle_driver * drv , int index )
2010-01-28 23:16:43 +05:30
{
2012-04-24 16:05:35 +02:00
struct omap3_idle_statedata * cx = & omap3_idle_data [ index ] ;
2011-05-09 12:02:16 +02:00
u32 mpu_deepest_state = PWRDM_POWER_RET ;
u32 core_deepest_state = PWRDM_POWER_RET ;
2012-04-24 16:05:36 +02:00
int idx ;
2012-06-01 17:11:06 +02:00
int next_index = 0 ; /* C1 is the default value */
2011-05-09 12:02:16 +02:00
if ( enable_off_mode ) {
mpu_deepest_state = PWRDM_POWER_OFF ;
/*
* Erratum i583 : valable for ES rev < Es1 .2 on 3630.
* CORE OFF mode is not supported in a stable form , restrict
* instead the CORE state to RET .
*/
if ( ! IS_PM34XX_ERRATUM ( PM_SDRC_WAKEUP_ERRATUM_i583 ) )
core_deepest_state = PWRDM_POWER_OFF ;
}
2010-01-28 23:16:43 +05:30
/* Check if current state is valid */
2012-04-24 16:05:32 +02:00
if ( ( cx - > mpu_state > = mpu_deepest_state ) & &
2012-04-24 16:05:36 +02:00
( cx - > core_state > = core_deepest_state ) )
2011-10-28 16:20:09 +05:30
return index ;
2010-01-28 23:16:43 +05:30
2012-04-24 16:05:36 +02:00
/*
* Drop to next valid state .
* Start search from the next ( lower ) state .
*/
for ( idx = index - 1 ; idx > = 0 ; idx - - ) {
2013-01-26 00:58:13 -07:00
cx = & omap3_idle_data [ idx ] ;
2012-04-24 16:05:36 +02:00
if ( ( cx - > mpu_state > = mpu_deepest_state ) & &
( cx - > core_state > = core_deepest_state ) ) {
next_index = idx ;
break ;
2010-01-28 23:16:43 +05:30
}
}
2011-10-28 16:20:09 +05:30
return next_index ;
2010-01-28 23:16:43 +05:30
}
2008-10-08 17:30:58 +05:30
/**
* omap3_enter_idle_bm - Checks for any bus activity
* @ dev : cpuidle device
2011-10-28 16:20:42 +05:30
* @ drv : cpuidle driver
2011-10-28 16:20:09 +05:30
* @ index : array index of target state to be programmed
2008-10-08 17:30:58 +05:30
*
2011-05-09 12:02:14 +02:00
* This function checks for any pending activity and then programs
* the device to the specified or a safer state .
2008-10-08 17:30:58 +05:30
*/
static int omap3_enter_idle_bm ( struct cpuidle_device * dev ,
2012-06-01 17:11:07 +02:00
struct cpuidle_driver * drv ,
2011-10-28 16:20:09 +05:30
int index )
2008-10-08 17:30:58 +05:30
{
2013-01-26 00:58:12 -07:00
int new_state_idx , ret ;
u8 per_next_state , per_saved_state ;
2011-05-09 12:02:14 +02:00
struct omap3_idle_statedata * cx ;
2008-10-28 17:32:11 -07:00
2010-09-08 16:37:42 -07:00
/*
2012-06-01 17:11:07 +02:00
* Use only C1 if CAM is active .
2010-09-08 16:37:42 -07:00
* CAM does not have wakeup capability in OMAP3 .
*/
2012-06-01 17:11:07 +02:00
if ( pwrdm_read_pwrst ( cam_pd ) = = PWRDM_POWER_ON )
2011-10-28 16:20:42 +05:30
new_state_idx = drv - > safe_state_index ;
2012-06-01 17:11:07 +02:00
else
new_state_idx = next_valid_state ( dev , drv , index ) ;
2010-09-08 16:37:42 -07:00
2011-05-09 12:02:15 +02:00
/*
* FIXME : we currently manage device - specific idle states
* for PER and CORE in combination with CPU - specific
* idle states . This is wrong , and device - specific
* idle management needs to be separated out into
* its own code .
*/
2012-06-01 17:11:07 +02:00
/* Program PER state */
cx = & omap3_idle_data [ new_state_idx ] ;
2008-10-28 17:32:11 -07:00
2013-01-26 00:58:12 -07:00
per_next_state = pwrdm_read_next_pwrst ( per_pd ) ;
per_saved_state = per_next_state ;
if ( per_next_state < cx - > per_min_state ) {
per_next_state = cx - > per_min_state ;
2010-09-08 16:37:42 -07:00
pwrdm_set_next_pwrst ( per_pd , per_next_state ) ;
2013-01-26 00:58:12 -07:00
}
2010-09-08 16:37:42 -07:00
2011-10-28 16:20:42 +05:30
ret = omap3_enter_idle ( dev , drv , new_state_idx ) ;
2010-09-08 16:37:42 -07:00
/* Restore original PER state if it was modified */
if ( per_next_state ! = per_saved_state )
pwrdm_set_next_pwrst ( per_pd , per_saved_state ) ;
return ret ;
2008-10-08 17:30:58 +05:30
}
2012-12-15 01:39:19 -07:00
static struct cpuidle_driver omap3_idle_driver = {
2013-03-29 11:31:35 +01:00
. name = " omap3_idle " ,
. owner = THIS_MODULE ,
2012-04-24 16:05:30 +02:00
. states = {
{
2012-06-01 17:11:07 +02:00
. enter = omap3_enter_idle_bm ,
2012-04-24 16:05:30 +02:00
. exit_latency = 2 + 2 ,
. target_residency = 5 ,
. name = " C1 " ,
. desc = " MPU ON + CORE ON " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 10 + 10 ,
. target_residency = 30 ,
. name = " C2 " ,
. desc = " MPU ON + CORE ON " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 50 + 50 ,
. target_residency = 300 ,
. name = " C3 " ,
. desc = " MPU RET + CORE ON " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 1500 + 1800 ,
. target_residency = 4000 ,
. name = " C4 " ,
. desc = " MPU OFF + CORE ON " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 2500 + 7500 ,
. target_residency = 12000 ,
. name = " C5 " ,
. desc = " MPU RET + CORE RET " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 3000 + 8500 ,
. target_residency = 15000 ,
. name = " C6 " ,
. desc = " MPU OFF + CORE RET " ,
} ,
{
. enter = omap3_enter_idle_bm ,
. exit_latency = 10000 + 30000 ,
. target_residency = 30000 ,
. name = " C7 " ,
. desc = " MPU OFF + CORE OFF " ,
} ,
} ,
2012-04-24 16:05:34 +02:00
. state_count = ARRAY_SIZE ( omap3_idle_data ) ,
2012-04-24 16:05:30 +02:00
. safe_state_index = 0 ,
2008-10-08 17:30:58 +05:30
} ;
2012-12-15 01:39:19 -07:00
/* Public functions */
2008-10-08 17:30:58 +05:30
/**
* omap3_idle_init - Init routine for OMAP3 idle
*
2011-05-09 12:02:14 +02:00
* Registers the OMAP3 specific cpuidle driver to the cpuidle
2008-10-08 17:30:58 +05:30
* framework with the valid set of states .
*/
2008-09-26 11:04:20 +03:00
int __init omap3_idle_init ( void )
2008-10-08 17:30:58 +05:30
{
mpu_pd = pwrdm_lookup ( " mpu_pwrdm " ) ;
2008-10-08 17:31:22 +05:30
core_pd = pwrdm_lookup ( " core_pwrdm " ) ;
2010-09-08 16:37:42 -07:00
per_pd = pwrdm_lookup ( " per_pwrdm " ) ;
cam_pd = pwrdm_lookup ( " cam_pwrdm " ) ;
2008-10-08 17:30:58 +05:30
2012-05-04 19:18:40 +02:00
if ( ! mpu_pd | | ! core_pd | | ! per_pd | | ! cam_pd )
return - ENODEV ;
2013-04-23 08:54:36 +00:00
return cpuidle_register ( & omap3_idle_driver , NULL ) ;
2008-10-08 17:30:58 +05:30
}