2014-05-09 06:43:26 +09:00
/* linux/arch/arm/mach-exynos/cpuidle.c
2011-03-16 07:28:23 +09:00
*
* Copyright ( c ) 2011 Samsung Electronics Co . , Ltd .
* http : //www.samsung.com
*
* 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/cpuidle.h>
2012-03-08 02:07:27 -08:00
# include <linux/cpu_pm.h>
2011-11-08 19:57:59 +09:00
# include <linux/export.h>
2014-03-19 18:29:36 +01:00
# include <linux/module.h>
2013-08-30 12:15:04 +02:00
# include <linux/platform_device.h>
2011-03-16 07:28:23 +09:00
# include <asm/proc-fns.h>
2012-03-08 02:07:27 -08:00
# include <asm/suspend.h>
2012-05-12 16:29:21 +09:00
# include <asm/cpuidle.h>
2012-03-08 02:07:27 -08:00
2014-05-09 06:56:29 +09:00
static void ( * exynos_enter_aftr ) ( void ) ;
2012-12-31 10:06:48 -08:00
2014-05-09 06:43:26 +09:00
static int exynos_enter_lowpower ( struct cpuidle_device * dev ,
2012-03-08 02:07:27 -08:00
struct cpuidle_driver * drv ,
int index )
{
int new_index = index ;
2013-12-20 19:47:23 +01:00
/* AFTR can only be entered when cores other than CPU0 are offline */
if ( num_online_cpus ( ) > 1 | | dev - > cpu ! = 0 )
2012-03-08 02:07:27 -08:00
new_index = drv - > safe_state_index ;
if ( new_index = = 0 )
2012-05-12 16:29:21 +09:00
return arm_cpuidle_simple_enter ( dev , drv , new_index ) ;
2014-08-05 14:43:10 +02:00
exynos_enter_aftr ( ) ;
return new_index ;
2012-03-08 02:07:27 -08:00
}
2014-05-09 06:43:26 +09:00
static struct cpuidle_driver exynos_idle_driver = {
. name = " exynos_idle " ,
2014-05-09 06:43:26 +09:00
. owner = THIS_MODULE ,
. states = {
[ 0 ] = ARM_CPUIDLE_WFI_STATE ,
[ 1 ] = {
2014-05-09 06:43:26 +09:00
. enter = exynos_enter_lowpower ,
2014-05-09 06:43:26 +09:00
. exit_latency = 300 ,
. target_residency = 100000 ,
. name = " C1 " ,
. desc = " ARM power down " ,
} ,
} ,
. state_count = 2 ,
. safe_state_index = 0 ,
} ;
2013-10-21 10:53:03 +09:00
static int exynos_cpuidle_probe ( struct platform_device * pdev )
2011-03-16 07:28:23 +09:00
{
2014-05-09 06:43:26 +09:00
int ret ;
2011-10-28 16:20:42 +05:30
2014-05-09 06:56:29 +09:00
exynos_enter_aftr = ( void * ) ( pdev - > dev . platform_data ) ;
2014-05-09 06:43:26 +09:00
ret = cpuidle_register ( & exynos_idle_driver , NULL ) ;
2013-01-18 21:57:58 -08:00
if ( ret ) {
2013-10-21 10:52:15 +09:00
dev_err ( & pdev - > dev , " failed to register cpuidle driver \n " ) ;
2013-01-18 21:57:58 -08:00
return ret ;
2011-10-28 16:20:42 +05:30
}
2011-03-16 07:28:23 +09:00
return 0 ;
}
2013-08-30 12:15:04 +02:00
static struct platform_driver exynos_cpuidle_driver = {
. probe = exynos_cpuidle_probe ,
. driver = {
. name = " exynos_cpuidle " ,
} ,
} ;
module_platform_driver ( exynos_cpuidle_driver ) ;