2011-06-01 14:18:22 -07:00
/*
2011-02-14 16:33:10 +09:00
* Copyright ( c ) 2010 - 2011 Samsung Electronics Co . , Ltd .
2010-09-16 17:59:21 +09:00
* http : //www.samsung.com
*
2012-01-07 20:18:35 +09:00
* EXYNOS4210 - CPU frequency scaling support
2010-09-16 17:59:21 +09:00
*
* 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 .
*/
2012-01-07 20:18:39 +09:00
# include <linux/module.h>
2010-09-16 17:59:21 +09:00
# include <linux/kernel.h>
# include <linux/err.h>
# include <linux/clk.h>
# include <linux/io.h>
# include <linux/slab.h>
# include <linux/cpufreq.h>
2012-12-28 16:29:10 -08:00
# include "exynos-cpufreq.h"
2010-09-16 17:59:21 +09:00
static struct clk * cpu_clk ;
static struct clk * moutcore ;
static struct clk * mout_mpll ;
static struct clk * mout_apll ;
2012-12-23 15:57:42 -08:00
static unsigned int exynos4210_volt_table [ ] = {
2012-01-07 20:18:35 +09:00
1250000 , 1150000 , 1050000 , 975000 , 950000 ,
2010-09-16 17:59:21 +09:00
} ;
2012-01-07 20:18:35 +09:00
static struct cpufreq_frequency_table exynos4210_freq_table [ ] = {
2012-12-23 15:57:42 -08:00
{ L0 , 1200 * 1000 } ,
{ L1 , 1000 * 1000 } ,
{ L2 , 800 * 1000 } ,
{ L3 , 500 * 1000 } ,
{ L4 , 200 * 1000 } ,
2010-09-16 17:59:21 +09:00
{ 0 , CPUFREQ_TABLE_END } ,
} ;
2012-12-23 15:57:42 -08:00
static struct apll_freq apll_freq_4210 [ ] = {
2010-09-16 17:59:21 +09:00
/*
2012-12-23 15:57:42 -08:00
* values :
* freq
* clock divider for CORE , COREM0 , COREM1 , PERIPH , ATB , PCLK_DBG , APLL , RESERVED
* clock divider for COPY , HPM , RESERVED
* PLL M , P , S
2010-09-16 17:59:21 +09:00
*/
2012-12-23 15:57:42 -08:00
APLL_FREQ ( 1200 , 0 , 3 , 7 , 3 , 4 , 1 , 7 , 0 , 5 , 0 , 0 , 150 , 3 , 1 ) ,
APLL_FREQ ( 1000 , 0 , 3 , 7 , 3 , 4 , 1 , 7 , 0 , 4 , 0 , 0 , 250 , 6 , 1 ) ,
APLL_FREQ ( 800 , 0 , 3 , 7 , 3 , 3 , 1 , 7 , 0 , 3 , 0 , 0 , 200 , 6 , 1 ) ,
APLL_FREQ ( 500 , 0 , 3 , 7 , 3 , 3 , 1 , 7 , 0 , 3 , 0 , 0 , 250 , 6 , 2 ) ,
APLL_FREQ ( 200 , 0 , 1 , 3 , 1 , 3 , 1 , 0 , 0 , 3 , 0 , 0 , 200 , 6 , 3 ) ,
2010-12-22 16:49:32 +09:00
} ;
2012-01-07 20:18:35 +09:00
static void exynos4210_set_clkdiv ( unsigned int div_index )
2010-09-16 17:59:21 +09:00
{
unsigned int tmp ;
/* Change Divider - CPU0 */
2012-12-23 15:57:42 -08:00
tmp = apll_freq_4210 [ div_index ] . clk_div_cpu0 ;
2010-09-16 17:59:21 +09:00
2012-01-31 13:49:24 +09:00
__raw_writel ( tmp , EXYNOS4_CLKDIV_CPU ) ;
2010-09-16 17:59:21 +09:00
do {
2012-01-31 13:49:24 +09:00
tmp = __raw_readl ( EXYNOS4_CLKDIV_STATCPU ) ;
2010-09-16 17:59:21 +09:00
} while ( tmp & 0x1111111 ) ;
2010-12-22 16:49:32 +09:00
/* Change Divider - CPU1 */
2012-12-23 15:57:42 -08:00
tmp = apll_freq_4210 [ div_index ] . clk_div_cpu1 ;
2010-12-22 16:49:32 +09:00
2012-01-31 13:49:24 +09:00
__raw_writel ( tmp , EXYNOS4_CLKDIV_CPU1 ) ;
2010-12-22 16:49:32 +09:00
do {
2012-01-31 13:49:24 +09:00
tmp = __raw_readl ( EXYNOS4_CLKDIV_STATCPU1 ) ;
2010-12-22 16:49:32 +09:00
} while ( tmp & 0x11 ) ;
2010-09-16 17:59:21 +09:00
}
2012-01-07 20:18:35 +09:00
static void exynos4210_set_apll ( unsigned int index )
2010-12-22 16:49:32 +09:00
{
2013-10-09 14:08:43 +02:00
unsigned int tmp , freq = apll_freq_4210 [ index ] . freq ;
2010-12-22 16:49:32 +09:00
2013-10-09 14:08:43 +02:00
/* MUX_CORE_SEL = MPLL, ARMCLK uses MPLL for lock time */
2010-12-22 16:49:32 +09:00
clk_set_parent ( moutcore , mout_mpll ) ;
do {
2012-01-31 13:49:24 +09:00
tmp = ( __raw_readl ( EXYNOS4_CLKMUX_STATCPU )
> > EXYNOS4_CLKSRC_CPU_MUXCORE_SHIFT ) ;
2010-12-22 16:49:32 +09:00
tmp & = 0x7 ;
} while ( tmp ! = 0x2 ) ;
2013-10-09 14:08:43 +02:00
clk_set_rate ( mout_apll , freq * 1000 ) ;
2010-12-22 16:49:32 +09:00
2013-10-09 14:08:43 +02:00
/* MUX_CORE_SEL = APLL */
2010-12-22 16:49:32 +09:00
clk_set_parent ( moutcore , mout_apll ) ;
do {
2012-01-31 13:49:24 +09:00
tmp = __raw_readl ( EXYNOS4_CLKMUX_STATCPU ) ;
tmp & = EXYNOS4_CLKMUX_STATCPU_MUXCORE_MASK ;
} while ( tmp ! = ( 0x1 < < EXYNOS4_CLKSRC_CPU_MUXCORE_SHIFT ) ) ;
2010-12-22 16:49:32 +09:00
}
2012-01-07 20:18:35 +09:00
static void exynos4210_set_frequency ( unsigned int old_index ,
unsigned int new_index )
2010-12-22 16:49:32 +09:00
{
if ( old_index > new_index ) {
2013-10-09 14:08:43 +02:00
exynos4210_set_clkdiv ( new_index ) ;
exynos4210_set_apll ( new_index ) ;
2011-12-07 11:44:09 +09:00
} else if ( old_index < new_index ) {
2013-10-09 14:08:43 +02:00
exynos4210_set_apll ( new_index ) ;
exynos4210_set_clkdiv ( new_index ) ;
2010-12-22 16:49:32 +09:00
}
}
2012-01-07 20:18:35 +09:00
int exynos4210_cpufreq_init ( struct exynos_dvfs_info * info )
2010-09-16 17:59:21 +09:00
{
2012-01-07 20:18:35 +09:00
unsigned long rate ;
2011-12-07 11:44:09 +09:00
2010-09-16 17:59:21 +09:00
cpu_clk = clk_get ( NULL , " armclk " ) ;
if ( IS_ERR ( cpu_clk ) )
return PTR_ERR ( cpu_clk ) ;
moutcore = clk_get ( NULL , " moutcore " ) ;
if ( IS_ERR ( moutcore ) )
2012-01-07 20:18:35 +09:00
goto err_moutcore ;
2010-09-16 17:59:21 +09:00
mout_mpll = clk_get ( NULL , " mout_mpll " ) ;
if ( IS_ERR ( mout_mpll ) )
2012-01-07 20:18:35 +09:00
goto err_mout_mpll ;
rate = clk_get_rate ( mout_mpll ) / 1000 ;
2010-09-16 17:59:21 +09:00
mout_apll = clk_get ( NULL , " mout_apll " ) ;
if ( IS_ERR ( mout_apll ) )
2012-01-07 20:18:35 +09:00
goto err_mout_apll ;
2011-08-18 19:45:16 +09:00
2012-01-07 20:18:35 +09:00
info - > mpll_freq_khz = rate ;
2012-12-23 15:57:42 -08:00
/* 800Mhz */
2012-01-07 20:18:35 +09:00
info - > pll_safe_idx = L2 ;
info - > cpu_clk = cpu_clk ;
info - > volt_table = exynos4210_volt_table ;
info - > freq_table = exynos4210_freq_table ;
info - > set_freq = exynos4210_set_frequency ;
2010-09-16 17:59:21 +09:00
2012-01-07 20:18:35 +09:00
return 0 ;
2010-09-16 17:59:21 +09:00
2012-01-07 20:18:35 +09:00
err_mout_apll :
2012-12-23 15:51:40 -08:00
clk_put ( mout_mpll ) ;
2012-01-07 20:18:35 +09:00
err_mout_mpll :
2012-12-23 15:51:40 -08:00
clk_put ( moutcore ) ;
2012-01-07 20:18:35 +09:00
err_moutcore :
2012-12-23 15:51:40 -08:00
clk_put ( cpu_clk ) ;
2010-09-16 17:59:21 +09:00
2012-01-07 20:18:35 +09:00
pr_debug ( " %s: failed initialization \n " , __func__ ) ;
2010-09-16 17:59:21 +09:00
return - EINVAL ;
}