2011-03-22 00:30:37 +03:00
/*
2011-11-07 15:36:48 +04:00
* Copyright ( C ) 2011 Freescale Semiconductor , Inc . All Rights Reserved .
*
2011-03-22 00:30:37 +03:00
* 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
*/
2011-11-07 15:36:48 +04:00
# include <linux/suspend.h>
# include <linux/clk.h>
2011-03-22 00:30:37 +03:00
# include <linux/io.h>
2011-11-07 15:36:48 +04:00
# include <linux/err.h>
2012-05-22 02:50:29 +04:00
# include <linux/export.h>
2011-11-07 15:36:48 +04:00
# include <asm/cacheflush.h>
2012-05-22 02:50:26 +04:00
# include <asm/system_misc.h>
2011-11-07 15:36:48 +04:00
# include <asm/tlbflush.h>
2012-09-13 17:01:00 +04:00
# include "common.h"
2012-09-13 17:12:50 +04:00
# include "cpuidle.h"
2012-09-14 10:14:45 +04:00
# include "hardware.h"
2011-11-07 15:36:48 +04:00
2014-05-20 09:41:36 +04:00
# define MXC_CCM_CLPCR 0x54
2014-05-20 06:23:50 +04:00
# define MXC_CCM_CLPCR_LPM_OFFSET 0
# define MXC_CCM_CLPCR_LPM_MASK 0x3
# define MXC_CCM_CLPCR_STBY_COUNT_OFFSET 9
# define MXC_CCM_CLPCR_VSTBY (0x1 << 8)
# define MXC_CCM_CLPCR_SBYOS (0x1 << 6)
2014-05-20 10:55:15 +04:00
# define MXC_CORTEXA8_PLAT_LPC 0xc
2014-05-20 06:23:50 +04:00
# define MXC_CORTEXA8_PLAT_LPC_DSM (1 << 0)
# define MXC_CORTEXA8_PLAT_LPC_DBG_DSM (1 << 1)
2014-05-20 10:55:15 +04:00
# define MXC_SRPG_NEON_SRPGCR 0x280
# define MXC_SRPG_ARM_SRPGCR 0x2a0
# define MXC_SRPG_EMPGC0_SRPGCR 0x2c0
# define MXC_SRPG_EMPGC1_SRPGCR 0x2d0
2014-05-20 06:23:50 +04:00
# define MXC_SRPGCR_PCR 1
2012-05-22 02:50:26 +04:00
/*
* The WAIT_UNCLOCKED_POWER_OFF state only requires < = 500 ns to exit .
* This is also the lowest power state possible without affecting
* non - cpu parts of the system . For these reasons , imx5 should default
* to always using this state for cpu idling . The PM_SUSPEND_STANDBY also
* uses this state and needs to take no action when registers remain confgiured
* for this state .
*/
# define IMX5_DEFAULT_CPU_IDLE_STATE WAIT_UNCLOCKED_POWER_OFF
2011-03-22 00:30:37 +03:00
2014-05-20 10:55:15 +04:00
struct imx5_pm_data {
2015-04-25 17:38:19 +03:00
phys_addr_t ccm_addr ;
2014-05-20 10:55:15 +04:00
phys_addr_t cortex_addr ;
phys_addr_t gpc_addr ;
} ;
static const struct imx5_pm_data imx51_pm_data __initconst = {
2015-04-25 17:38:19 +03:00
. ccm_addr = 0x73fd4000 ,
2014-05-20 10:55:15 +04:00
. cortex_addr = 0x83fa0000 ,
. gpc_addr = 0x73fd8000 ,
} ;
static const struct imx5_pm_data imx53_pm_data __initconst = {
2015-04-25 17:38:19 +03:00
. ccm_addr = 0x53fd4000 ,
2014-05-20 10:55:15 +04:00
. cortex_addr = 0x63fa0000 ,
. gpc_addr = 0x53fd8000 ,
} ;
2014-05-20 09:41:36 +04:00
static void __iomem * ccm_base ;
2014-05-20 10:55:15 +04:00
static void __iomem * cortex_base ;
static void __iomem * gpc_base ;
2014-05-20 09:41:36 +04:00
2011-11-07 15:36:48 +04:00
/*
* set cpu low power mode before WFI instruction . This function is called
2013-01-22 16:40:55 +04:00
* mx5 because it can be used for mx51 , and mx53 .
2011-11-07 15:36:48 +04:00
*/
2012-05-22 02:50:26 +04:00
static void mx5_cpu_lp_set ( enum mxc_cpu_pwr_mode mode )
2011-03-22 00:30:37 +03:00
{
u32 plat_lpc , arm_srpgcr , ccm_clpcr ;
u32 empgc0 , empgc1 ;
int stop_mode = 0 ;
/* always allow platform to issue a deep sleep mode request */
2014-05-20 10:55:15 +04:00
plat_lpc = __raw_readl ( cortex_base + MXC_CORTEXA8_PLAT_LPC ) &
2011-03-22 00:30:37 +03:00
~ ( MXC_CORTEXA8_PLAT_LPC_DSM ) ;
2014-05-20 09:41:36 +04:00
ccm_clpcr = __raw_readl ( ccm_base + MXC_CCM_CLPCR ) &
~ ( MXC_CCM_CLPCR_LPM_MASK ) ;
2014-05-20 10:55:15 +04:00
arm_srpgcr = __raw_readl ( gpc_base + MXC_SRPG_ARM_SRPGCR ) &
~ ( MXC_SRPGCR_PCR ) ;
empgc0 = __raw_readl ( gpc_base + MXC_SRPG_EMPGC0_SRPGCR ) &
~ ( MXC_SRPGCR_PCR ) ;
empgc1 = __raw_readl ( gpc_base + MXC_SRPG_EMPGC1_SRPGCR ) &
~ ( MXC_SRPGCR_PCR ) ;
2011-03-22 00:30:37 +03:00
switch ( mode ) {
case WAIT_CLOCKED :
break ;
case WAIT_UNCLOCKED :
ccm_clpcr | = 0x1 < < MXC_CCM_CLPCR_LPM_OFFSET ;
break ;
case WAIT_UNCLOCKED_POWER_OFF :
case STOP_POWER_OFF :
plat_lpc | = MXC_CORTEXA8_PLAT_LPC_DSM
| MXC_CORTEXA8_PLAT_LPC_DBG_DSM ;
if ( mode = = WAIT_UNCLOCKED_POWER_OFF ) {
ccm_clpcr | = 0x1 < < MXC_CCM_CLPCR_LPM_OFFSET ;
ccm_clpcr & = ~ MXC_CCM_CLPCR_VSTBY ;
ccm_clpcr & = ~ MXC_CCM_CLPCR_SBYOS ;
stop_mode = 0 ;
} else {
ccm_clpcr | = 0x2 < < MXC_CCM_CLPCR_LPM_OFFSET ;
ccm_clpcr | = 0x3 < < MXC_CCM_CLPCR_STBY_COUNT_OFFSET ;
ccm_clpcr | = MXC_CCM_CLPCR_VSTBY ;
ccm_clpcr | = MXC_CCM_CLPCR_SBYOS ;
stop_mode = 1 ;
}
arm_srpgcr | = MXC_SRPGCR_PCR ;
break ;
case STOP_POWER_ON :
ccm_clpcr | = 0x2 < < MXC_CCM_CLPCR_LPM_OFFSET ;
break ;
default :
printk ( KERN_WARNING " UNKNOWN cpu power mode: %d \n " , mode ) ;
return ;
}
2014-05-20 10:55:15 +04:00
__raw_writel ( plat_lpc , cortex_base + MXC_CORTEXA8_PLAT_LPC ) ;
2014-05-20 09:41:36 +04:00
__raw_writel ( ccm_clpcr , ccm_base + MXC_CCM_CLPCR ) ;
2014-05-20 10:55:15 +04:00
__raw_writel ( arm_srpgcr , gpc_base + MXC_SRPG_ARM_SRPGCR ) ;
__raw_writel ( arm_srpgcr , gpc_base + MXC_SRPG_NEON_SRPGCR ) ;
2011-03-22 00:30:37 +03:00
if ( stop_mode ) {
empgc0 | = MXC_SRPGCR_PCR ;
empgc1 | = MXC_SRPGCR_PCR ;
2014-05-20 10:55:15 +04:00
__raw_writel ( empgc0 , gpc_base + MXC_SRPG_EMPGC0_SRPGCR ) ;
__raw_writel ( empgc1 , gpc_base + MXC_SRPG_EMPGC1_SRPGCR ) ;
2011-03-22 00:30:37 +03:00
}
}
2011-11-07 15:36:48 +04:00
static int mx5_suspend_enter ( suspend_state_t state )
{
switch ( state ) {
case PM_SUSPEND_MEM :
mx5_cpu_lp_set ( STOP_POWER_OFF ) ;
break ;
case PM_SUSPEND_STANDBY :
2012-05-22 02:50:26 +04:00
/* DEFAULT_IDLE_STATE already configured */
2011-11-07 15:36:48 +04:00
break ;
default :
return - EINVAL ;
}
if ( state = = PM_SUSPEND_MEM ) {
local_flush_tlb_all ( ) ;
flush_cache_all ( ) ;
/*clear the EMPGC0/1 bits */
2014-05-20 10:55:15 +04:00
__raw_writel ( 0 , gpc_base + MXC_SRPG_EMPGC0_SRPGCR ) ;
__raw_writel ( 0 , gpc_base + MXC_SRPG_EMPGC1_SRPGCR ) ;
2011-11-07 15:36:48 +04:00
}
cpu_do_idle ( ) ;
2012-05-22 02:50:26 +04:00
/* return registers to default idle state */
mx5_cpu_lp_set ( IMX5_DEFAULT_CPU_IDLE_STATE ) ;
return 0 ;
2011-11-07 15:36:48 +04:00
}
static int mx5_pm_valid ( suspend_state_t state )
{
return ( state > PM_SUSPEND_ON & & state < = PM_SUSPEND_MAX ) ;
}
static const struct platform_suspend_ops mx5_suspend_ops = {
. valid = mx5_pm_valid ,
. enter = mx5_suspend_enter ,
} ;
2012-05-22 02:50:29 +04:00
static inline int imx5_cpu_do_idle ( void )
2011-11-07 15:36:48 +04:00
{
2012-05-22 02:50:29 +04:00
int ret = tzic_enable_wake ( ) ;
if ( likely ( ! ret ) )
2012-05-22 02:50:26 +04:00
cpu_do_idle ( ) ;
2012-05-22 02:50:29 +04:00
return ret ;
}
static void imx5_pm_idle ( void )
{
imx5_cpu_do_idle ( ) ;
}
2014-05-20 10:55:15 +04:00
static int __init imx5_pm_common_init ( const struct imx5_pm_data * data )
2012-05-22 02:50:26 +04:00
{
int ret ;
struct clk * gpc_dvfs_clk = clk_get ( NULL , " gpc_dvfs " ) ;
if ( IS_ERR ( gpc_dvfs_clk ) )
return PTR_ERR ( gpc_dvfs_clk ) ;
2011-11-07 15:36:48 +04:00
2012-05-22 02:50:26 +04:00
ret = clk_prepare_enable ( gpc_dvfs_clk ) ;
if ( ret )
return ret ;
2011-11-07 15:36:48 +04:00
2012-05-22 02:50:26 +04:00
arm_pm_idle = imx5_pm_idle ;
2015-04-25 17:38:19 +03:00
ccm_base = ioremap ( data - > ccm_addr , SZ_16K ) ;
2014-05-20 10:55:15 +04:00
cortex_base = ioremap ( data - > cortex_addr , SZ_16K ) ;
gpc_base = ioremap ( data - > gpc_addr , SZ_16K ) ;
WARN_ON ( ! ccm_base | | ! cortex_base | | ! gpc_base ) ;
2014-05-20 09:41:36 +04:00
2012-05-22 02:50:26 +04:00
/* Set the registers to the default cpu idle state. */
mx5_cpu_lp_set ( IMX5_DEFAULT_CPU_IDLE_STATE ) ;
2011-11-07 15:36:48 +04:00
2014-05-20 10:55:15 +04:00
ret = imx5_cpuidle_init ( ) ;
if ( ret )
pr_warn ( " %s: cpuidle init failed %d \n " , __func__ , ret ) ;
suspend_set_ops ( & mx5_suspend_ops ) ;
return 0 ;
}
void __init imx51_pm_init ( void )
{
imx5_pm_common_init ( & imx51_pm_data ) ;
2011-11-07 15:36:48 +04:00
}
2012-05-22 02:50:26 +04:00
2014-05-20 10:55:15 +04:00
void __init imx53_pm_init ( void )
2012-05-22 02:50:26 +04:00
{
2014-05-20 10:55:15 +04:00
imx5_pm_common_init ( & imx53_pm_data ) ;
2012-05-22 02:50:26 +04:00
}