2011-03-10 13:33:59 +09:00
/* linux/arch/arm/mach-exynos4/pm.c
*
* Copyright ( c ) 2011 Samsung Electronics Co . , Ltd .
* http : //www.samsung.com
*
* EXYNOS4210 - Power Management support
*
* Based on arch / arm / mach - s3c2410 / pm . c
* Copyright ( c ) 2006 Simtec Electronics
* Ben Dooks < ben @ simtec . co . uk >
*
* 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 .
*/
# include <linux/init.h>
# include <linux/suspend.h>
2011-04-22 22:03:21 +02:00
# include <linux/syscore_ops.h>
2011-03-10 13:33:59 +09:00
# include <linux/io.h>
2011-07-18 19:25:13 +09:00
# include <linux/err.h>
# include <linux/clk.h>
2011-03-10 13:33:59 +09:00
# include <asm/cacheflush.h>
# include <asm/hardware/cache-l2x0.h>
2011-11-17 01:19:11 +09:00
# include <asm/smp_scu.h>
2011-03-10 13:33:59 +09:00
# include <plat/cpu.h>
# include <plat/pm.h>
2011-07-18 19:25:13 +09:00
# include <plat/pll.h>
2011-07-21 11:25:23 +09:00
# include <plat/regs-srom.h>
2011-03-10 13:33:59 +09:00
# include <mach/regs-irq.h>
# include <mach/regs-gpio.h>
# include <mach/regs-clock.h>
# include <mach/regs-pmu.h>
# include <mach/pm-core.h>
2011-07-18 19:21:27 +09:00
# include <mach/pmu.h>
2011-03-10 13:33:59 +09:00
static struct sleep_save exynos4_set_clksrc [ ] = {
2012-03-09 14:19:10 -08:00
{ . reg = EXYNOS4_CLKSRC_MASK_TOP , . val = 0x00000001 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_CAM , . val = 0x11111111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_TV , . val = 0x00000111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_LCD0 , . val = 0x00001111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_MAUDIO , . val = 0x00000001 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_FSYS , . val = 0x01011111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_PERIL0 , . val = 0x01111111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_PERIL1 , . val = 0x01110111 , } ,
{ . reg = EXYNOS4_CLKSRC_MASK_DMC , . val = 0x00010000 , } ,
2011-03-10 13:33:59 +09:00
} ;
2011-08-24 21:52:45 +09:00
static struct sleep_save exynos4210_set_clksrc [ ] = {
2012-03-09 14:19:10 -08:00
{ . reg = EXYNOS4210_CLKSRC_MASK_LCD1 , . val = 0x00001111 , } ,
2011-08-24 21:52:45 +09:00
} ;
2011-07-18 19:25:13 +09:00
static struct sleep_save exynos4_epll_save [ ] = {
2012-03-09 14:19:10 -08:00
SAVE_ITEM ( EXYNOS4_EPLL_CON0 ) ,
SAVE_ITEM ( EXYNOS4_EPLL_CON1 ) ,
2011-07-18 19:25:13 +09:00
} ;
static struct sleep_save exynos4_vpll_save [ ] = {
2012-03-09 14:19:10 -08:00
SAVE_ITEM ( EXYNOS4_VPLL_CON0 ) ,
SAVE_ITEM ( EXYNOS4_VPLL_CON1 ) ,
2011-07-18 19:25:13 +09:00
} ;
2011-03-10 13:33:59 +09:00
static struct sleep_save exynos4_core_save [ ] = {
2011-07-21 11:25:23 +09:00
/* SROM side */
SAVE_ITEM ( S5P_SROM_BW ) ,
SAVE_ITEM ( S5P_SROM_BC0 ) ,
SAVE_ITEM ( S5P_SROM_BC1 ) ,
SAVE_ITEM ( S5P_SROM_BC2 ) ,
SAVE_ITEM ( S5P_SROM_BC3 ) ,
2011-03-10 13:33:59 +09:00
} ;
2011-07-18 19:25:03 +09:00
/* For Cortex-A9 Diagnostic and Power control register */
static unsigned int save_arm_register [ 2 ] ;
2011-07-02 09:54:01 +01:00
static int exynos4_cpu_suspend ( unsigned long arg )
2011-03-10 13:33:59 +09:00
{
outer_flush_all ( ) ;
/* issue the standby signal into the pm unit. */
cpu_do_idle ( ) ;
/* we should never get past here */
panic ( " sleep resumed to originator? " ) ;
}
static void exynos4_pm_prepare ( void )
{
u32 tmp ;
s3c_pm_do_save ( exynos4_core_save , ARRAY_SIZE ( exynos4_core_save ) ) ;
2011-07-18 19:25:13 +09:00
s3c_pm_do_save ( exynos4_epll_save , ARRAY_SIZE ( exynos4_epll_save ) ) ;
s3c_pm_do_save ( exynos4_vpll_save , ARRAY_SIZE ( exynos4_vpll_save ) ) ;
2011-03-10 13:33:59 +09:00
tmp = __raw_readl ( S5P_INFORM1 ) ;
/* Set value of power down register for sleep mode */
2011-07-18 19:21:27 +09:00
exynos4_sys_powerdown_conf ( SYS_SLEEP ) ;
2011-03-10 13:33:59 +09:00
__raw_writel ( S5P_CHECK_SLEEP , S5P_INFORM1 ) ;
/* ensure at least INFORM0 has the resume address */
__raw_writel ( virt_to_phys ( s3c_cpu_resume ) , S5P_INFORM0 ) ;
/* Before enter central sequence mode, clock src register have to set */
s3c_pm_do_restore_core ( exynos4_set_clksrc , ARRAY_SIZE ( exynos4_set_clksrc ) ) ;
2011-08-24 21:52:45 +09:00
if ( soc_is_exynos4210 ( ) )
s3c_pm_do_restore_core ( exynos4210_set_clksrc , ARRAY_SIZE ( exynos4210_set_clksrc ) ) ;
2011-03-10 13:33:59 +09:00
}
2012-01-27 15:30:48 +09:00
static int exynos4_pm_add ( struct device * dev , struct subsys_interface * sif )
2011-03-10 13:33:59 +09:00
{
pm_cpu_prep = exynos4_pm_prepare ;
pm_cpu_sleep = exynos4_cpu_suspend ;
return 0 ;
}
2011-07-18 19:25:13 +09:00
static unsigned long pll_base_rate ;
static void exynos4_restore_pll ( void )
{
unsigned long pll_con , locktime , lockcnt ;
unsigned long pll_in_rate ;
unsigned int p_div , epll_wait = 0 , vpll_wait = 0 ;
if ( pll_base_rate = = 0 )
return ;
pll_in_rate = pll_base_rate ;
/* EPLL */
pll_con = exynos4_epll_save [ 0 ] . val ;
if ( pll_con & ( 1 < < 31 ) ) {
pll_con & = ( PLL46XX_PDIV_MASK < < PLL46XX_PDIV_SHIFT ) ;
p_div = ( pll_con > > PLL46XX_PDIV_SHIFT ) ;
pll_in_rate / = 1000000 ;
locktime = ( 3000 / pll_in_rate ) * p_div ;
lockcnt = locktime * 10000 / ( 10000 / pll_in_rate ) ;
2012-03-09 14:19:10 -08:00
__raw_writel ( lockcnt , EXYNOS4_EPLL_LOCK ) ;
2011-07-18 19:25:13 +09:00
s3c_pm_do_restore_core ( exynos4_epll_save ,
ARRAY_SIZE ( exynos4_epll_save ) ) ;
epll_wait = 1 ;
}
pll_in_rate = pll_base_rate ;
/* VPLL */
pll_con = exynos4_vpll_save [ 0 ] . val ;
if ( pll_con & ( 1 < < 31 ) ) {
pll_in_rate / = 1000000 ;
/* 750us */
locktime = 750 ;
lockcnt = locktime * 10000 / ( 10000 / pll_in_rate ) ;
2012-03-09 14:19:10 -08:00
__raw_writel ( lockcnt , EXYNOS4_VPLL_LOCK ) ;
2011-07-18 19:25:13 +09:00
s3c_pm_do_restore_core ( exynos4_vpll_save ,
ARRAY_SIZE ( exynos4_vpll_save ) ) ;
vpll_wait = 1 ;
}
/* Wait PLL locking */
do {
if ( epll_wait ) {
2012-03-09 14:19:10 -08:00
pll_con = __raw_readl ( EXYNOS4_EPLL_CON0 ) ;
if ( pll_con & ( 1 < < EXYNOS4_EPLLCON0_LOCKED_SHIFT ) )
2011-07-18 19:25:13 +09:00
epll_wait = 0 ;
}
if ( vpll_wait ) {
2012-03-09 14:19:10 -08:00
pll_con = __raw_readl ( EXYNOS4_VPLL_CON0 ) ;
if ( pll_con & ( 1 < < EXYNOS4_VPLLCON0_LOCKED_SHIFT ) )
2011-07-18 19:25:13 +09:00
vpll_wait = 0 ;
}
} while ( epll_wait | | vpll_wait ) ;
}
2011-12-21 16:01:38 -08:00
static struct subsys_interface exynos4_pm_interface = {
. name = " exynos4_pm " ,
2012-05-15 15:47:40 +09:00
. subsys = & exynos_subsys ,
2011-12-21 16:01:38 -08:00
. add_dev = exynos4_pm_add ,
2011-04-22 22:03:21 +02:00
} ;
static __init int exynos4_pm_drvinit ( void )
{
2011-07-18 19:25:13 +09:00
struct clk * pll_base ;
2011-04-22 22:03:21 +02:00
unsigned int tmp ;
s3c_pm_init ( ) ;
/* All wakeup disable */
tmp = __raw_readl ( S5P_WAKEUP_MASK ) ;
tmp | = ( ( 0xFF < < 8 ) | ( 0x1F < < 1 ) ) ;
__raw_writel ( tmp , S5P_WAKEUP_MASK ) ;
2011-07-18 19:25:13 +09:00
pll_base = clk_get ( NULL , " xtal " ) ;
if ( ! IS_ERR ( pll_base ) ) {
pll_base_rate = clk_get_rate ( pll_base ) ;
clk_put ( pll_base ) ;
}
2011-12-21 16:01:38 -08:00
return subsys_interface_register ( & exynos4_pm_interface ) ;
2011-04-22 22:03:21 +02:00
}
arch_initcall ( exynos4_pm_drvinit ) ;
2011-07-18 19:21:41 +09:00
static int exynos4_pm_suspend ( void )
{
unsigned long tmp ;
/* Setting Central Sequence Register for power down mode */
tmp = __raw_readl ( S5P_CENTRAL_SEQ_CONFIGURATION ) ;
tmp & = ~ S5P_CENTRAL_LOWPWR_CFG ;
__raw_writel ( tmp , S5P_CENTRAL_SEQ_CONFIGURATION ) ;
2012-05-15 00:20:09 +09:00
if ( soc_is_exynos4212 ( ) | | soc_is_exynos4412 ( ) ) {
2011-09-27 07:26:04 +09:00
tmp = __raw_readl ( S5P_CENTRAL_SEQ_OPTION ) ;
tmp & = ~ ( S5P_USE_STANDBYWFI_ISP_ARM |
S5P_USE_STANDBYWFE_ISP_ARM ) ;
__raw_writel ( tmp , S5P_CENTRAL_SEQ_OPTION ) ;
}
2011-07-18 19:25:03 +09:00
/* Save Power control register */
asm ( " mrc p15, 0, %0, c15, c0, 0 "
: " =r " ( tmp ) : : " cc " ) ;
save_arm_register [ 0 ] = tmp ;
/* Save Diagnostic register */
asm ( " mrc p15, 0, %0, c15, c0, 1 "
: " =r " ( tmp ) : : " cc " ) ;
save_arm_register [ 1 ] = tmp ;
2011-07-18 19:21:41 +09:00
return 0 ;
}
2011-04-22 22:03:21 +02:00
static void exynos4_pm_resume ( void )
2011-03-10 13:33:59 +09:00
{
2011-07-18 19:21:34 +09:00
unsigned long tmp ;
/*
* If PMU failed while entering sleep mode , WFI will be
* ignored by PMU and then exiting cpu_do_idle ( ) .
* S5P_CENTRAL_LOWPWR_CFG bit will not be set automatically
* in this situation .
*/
tmp = __raw_readl ( S5P_CENTRAL_SEQ_CONFIGURATION ) ;
if ( ! ( tmp & S5P_CENTRAL_LOWPWR_CFG ) ) {
tmp | = S5P_CENTRAL_LOWPWR_CFG ;
__raw_writel ( tmp , S5P_CENTRAL_SEQ_CONFIGURATION ) ;
/* No need to perform below restore code */
goto early_wakeup ;
}
2011-07-18 19:25:03 +09:00
/* Restore Power control register */
tmp = save_arm_register [ 0 ] ;
asm volatile ( " mcr p15, 0, %0, c15, c0, 0 "
: : " r " ( tmp )
: " cc " ) ;
/* Restore Diagnostic register */
tmp = save_arm_register [ 1 ] ;
asm volatile ( " mcr p15, 0, %0, c15, c0, 1 "
: : " r " ( tmp )
: " cc " ) ;
2011-07-18 19:21:34 +09:00
2011-03-10 13:33:59 +09:00
/* For release retention */
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_MAUDIO_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_GPIO_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_UART_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_MMCA_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_MMCB_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_EBIA_OPTION ) ;
__raw_writel ( ( 1 < < 28 ) , S5P_PAD_RET_EBIB_OPTION ) ;
s3c_pm_do_restore_core ( exynos4_core_save , ARRAY_SIZE ( exynos4_core_save ) ) ;
2011-07-18 19:25:13 +09:00
exynos4_restore_pll ( ) ;
2012-01-27 14:47:45 +09:00
# ifdef CONFIG_SMP
2011-11-17 01:19:11 +09:00
scu_enable ( S5P_VA_SCU ) ;
2012-01-27 14:47:45 +09:00
# endif
2011-03-10 13:33:59 +09:00
2011-07-18 19:21:34 +09:00
early_wakeup :
return ;
2011-03-10 13:33:59 +09:00
}
2011-04-22 22:03:21 +02:00
static struct syscore_ops exynos4_pm_syscore_ops = {
2011-07-18 19:21:41 +09:00
. suspend = exynos4_pm_suspend ,
2011-03-10 13:33:59 +09:00
. resume = exynos4_pm_resume ,
} ;
2011-04-22 22:03:21 +02:00
static __init int exynos4_pm_syscore_init ( void )
2011-03-10 13:33:59 +09:00
{
2011-04-22 22:03:21 +02:00
register_syscore_ops ( & exynos4_pm_syscore_ops ) ;
return 0 ;
2011-03-10 13:33:59 +09:00
}
2011-04-22 22:03:21 +02:00
arch_initcall ( exynos4_pm_syscore_init ) ;