2008-04-25 00:58:29 +04:00
/*
2009-09-24 18:11:24 +04:00
* Blackfin core clock scaling
2008-04-25 00:58:29 +04:00
*
2011-03-01 00:23:36 +03:00
* Copyright 2008 - 2011 Analog Devices Inc .
2008-04-25 00:58:29 +04:00
*
2009-09-24 18:11:24 +04:00
* Licensed under the GPL - 2 or later .
2008-04-25 00:58:29 +04:00
*/
# include <linux/kernel.h>
2011-08-10 00:54:30 +04:00
# include <linux/module.h>
2008-04-25 00:58:29 +04:00
# include <linux/types.h>
# include <linux/init.h>
2012-05-16 13:49:52 +04:00
# include <linux/clk.h>
2008-04-25 00:58:29 +04:00
# include <linux/cpufreq.h>
# include <linux/fs.h>
2010-03-08 06:01:35 +03:00
# include <linux/delay.h>
2008-04-25 00:58:29 +04:00
# include <asm/blackfin.h>
# include <asm/time.h>
2009-10-15 21:12:05 +04:00
# include <asm/dpmc.h>
2008-04-25 00:58:29 +04:00
2012-05-16 13:49:52 +04:00
2008-04-25 00:58:29 +04:00
/* this is the table of CCLK frequencies, in Hz */
2013-03-30 14:55:15 +04:00
/* .driver_data is the entry in the auxiliary dpm_state_table[] */
2008-04-25 00:58:29 +04:00
static struct cpufreq_frequency_table bfin_freq_table [ ] = {
{
. frequency = CPUFREQ_TABLE_END ,
2013-03-30 14:55:15 +04:00
. driver_data = 0 ,
2008-04-25 00:58:29 +04:00
} ,
{
. frequency = CPUFREQ_TABLE_END ,
2013-03-30 14:55:15 +04:00
. driver_data = 1 ,
2008-04-25 00:58:29 +04:00
} ,
{
. frequency = CPUFREQ_TABLE_END ,
2013-03-30 14:55:15 +04:00
. driver_data = 2 ,
2008-04-25 00:58:29 +04:00
} ,
{
. frequency = CPUFREQ_TABLE_END ,
2013-03-30 14:55:15 +04:00
. driver_data = 0 ,
2008-04-25 00:58:29 +04:00
} ,
} ;
static struct bfin_dpm_state {
unsigned int csel ; /* system clock divider */
unsigned int tscale ; /* change the divider on the core timer interrupt */
} dpm_state_table [ 3 ] ;
2010-01-27 14:16:32 +03:00
# if defined(CONFIG_CYCLES_CLOCKSOURCE)
2008-05-07 07:41:26 +04:00
/*
2011-03-01 00:23:36 +03:00
* normalized to maximum frequency offset for CYCLES ,
2010-01-27 14:16:32 +03:00
* used in time - ts cycles clock source , but could be used
* somewhere also .
2008-05-07 07:41:26 +04:00
*/
unsigned long long __bfin_cycles_off ;
unsigned int __bfin_cycles_mod ;
2010-01-27 14:16:32 +03:00
# endif
2008-05-07 07:41:26 +04:00
2008-04-25 00:58:29 +04:00
/**************************************************************************/
2010-01-27 14:16:32 +03:00
static void __init bfin_init_tables ( unsigned long cclk , unsigned long sclk )
{
2008-04-25 00:58:29 +04:00
2010-01-27 14:16:32 +03:00
unsigned long csel , min_cclk ;
int index ;
/* Anomaly 273 seems to still exist on non-BF54x w/dcache turned on */
# if ANOMALY_05000273 || ANOMALY_05000274 || \
2012-07-17 09:40:15 +04:00
( ! ( defined ( CONFIG_BF54x ) | | defined ( CONFIG_BF60x ) ) \
& & defined ( CONFIG_BFIN_EXTMEM_DCACHEABLE ) )
2010-01-27 14:16:32 +03:00
min_cclk = sclk * 2 ;
# else
min_cclk = sclk ;
# endif
2012-05-16 13:49:52 +04:00
# ifndef CONFIG_BF60x
2010-01-27 14:16:32 +03:00
csel = ( ( bfin_read_PLL_DIV ( ) & CSEL ) > > 4 ) ;
2012-05-16 13:49:52 +04:00
# else
csel = bfin_read32 ( CGU0_DIV ) & 0x1F ;
# endif
2010-01-27 14:16:32 +03:00
2012-08-20 07:55:36 +04:00
for ( index = 0 ; ( cclk > > index ) > = min_cclk & & csel < = 3 & & index < 3 ; index + + , csel + + ) {
2010-01-27 14:16:32 +03:00
bfin_freq_table [ index ] . frequency = cclk > > index ;
2012-05-16 13:49:52 +04:00
# ifndef CONFIG_BF60x
2010-01-27 14:16:32 +03:00
dpm_state_table [ index ] . csel = csel < < 4 ; /* Shift now into PLL_DIV bitpos */
2012-05-16 13:49:52 +04:00
# else
dpm_state_table [ index ] . csel = csel ;
# endif
2012-08-20 07:55:36 +04:00
dpm_state_table [ index ] . tscale = ( TIME_SCALE > > index ) - 1 ;
2010-01-27 14:16:32 +03:00
pr_debug ( " cpufreq: freq:%d csel:0x%x tscale:%d \n " ,
bfin_freq_table [ index ] . frequency ,
dpm_state_table [ index ] . csel ,
dpm_state_table [ index ] . tscale ) ;
}
return ;
}
static void bfin_adjust_core_timer ( void * info )
2008-04-25 00:58:29 +04:00
{
2010-01-27 14:16:32 +03:00
unsigned int tscale ;
unsigned int index = * ( unsigned int * ) info ;
/* we have to adjust the core timer, because it is using cclk */
tscale = dpm_state_table [ index ] . tscale ;
bfin_write_TSCALE ( tscale ) ;
return ;
}
2008-04-25 00:58:29 +04:00
2010-01-27 14:16:32 +03:00
static unsigned int bfin_getfreq_khz ( unsigned int cpu )
{
/* Both CoreA/B have the same core clock */
2008-10-28 09:18:29 +03:00
return get_cclk ( ) / 1000 ;
2008-04-25 00:58:29 +04:00
}
2012-05-17 10:21:22 +04:00
# ifdef CONFIG_BF60x
2015-12-22 00:12:26 +03:00
static int cpu_set_cclk ( int cpu , unsigned long new )
2012-05-16 13:49:52 +04:00
{
struct clk * clk ;
int ret ;
clk = clk_get ( NULL , " CCLK " ) ;
if ( IS_ERR ( clk ) )
return - ENODEV ;
ret = clk_set_rate ( clk , new ) ;
clk_put ( clk ) ;
return ret ;
}
2012-05-17 10:21:22 +04:00
# endif
2012-05-16 13:49:52 +04:00
2013-10-25 18:15:48 +04:00
static int bfin_target ( struct cpufreq_policy * policy , unsigned int index )
2008-04-25 00:58:29 +04:00
{
2012-05-17 10:21:22 +04:00
# ifndef CONFIG_BF60x
unsigned int plldiv ;
# endif
2010-03-08 06:01:35 +03:00
static unsigned long lpj_ref ;
static unsigned int lpj_ref_freq ;
2013-08-14 18:08:24 +04:00
unsigned int old_freq , new_freq ;
2012-05-16 13:49:52 +04:00
int ret = 0 ;
2010-03-08 06:01:35 +03:00
2010-01-27 14:16:32 +03:00
# if defined(CONFIG_CYCLES_CLOCKSOURCE)
2008-05-07 07:41:26 +04:00
cycles_t cycles ;
2010-01-27 14:16:32 +03:00
# endif
2008-04-25 00:58:29 +04:00
2013-08-14 18:08:24 +04:00
old_freq = bfin_getfreq_khz ( 0 ) ;
new_freq = bfin_freq_table [ index ] . frequency ;
2010-01-27 14:16:32 +03:00
2012-05-16 13:49:52 +04:00
# ifndef CONFIG_BF60x
2013-03-24 10:26:43 +04:00
plldiv = ( bfin_read_PLL_DIV ( ) & SSEL ) | dpm_state_table [ index ] . csel ;
bfin_write_PLL_DIV ( plldiv ) ;
2012-05-16 13:49:52 +04:00
# else
2013-08-14 18:08:24 +04:00
ret = cpu_set_cclk ( policy - > cpu , new_freq * 1000 ) ;
2013-03-24 10:26:43 +04:00
if ( ret ! = 0 ) {
WARN_ONCE ( ret , " cpufreq set freq failed %d \n " , ret ) ;
return ret ;
}
2012-05-16 13:49:52 +04:00
# endif
2013-03-24 10:26:43 +04:00
on_each_cpu ( bfin_adjust_core_timer , & index , 1 ) ;
2010-01-27 14:16:32 +03:00
# if defined(CONFIG_CYCLES_CLOCKSOURCE)
2013-03-24 10:26:43 +04:00
cycles = get_cycles ( ) ;
SSYNC ( ) ;
cycles + = 10 ; /* ~10 cycles we lose after get_cycles() */
__bfin_cycles_off + = ( cycles < < __bfin_cycles_mod ) - ( cycles < < index ) ;
__bfin_cycles_mod = index ;
2010-01-27 14:16:32 +03:00
# endif
2013-03-24 10:26:43 +04:00
if ( ! lpj_ref_freq ) {
lpj_ref = loops_per_jiffy ;
2013-08-14 18:08:24 +04:00
lpj_ref_freq = old_freq ;
2010-01-27 14:16:32 +03:00
}
2013-08-14 18:08:24 +04:00
if ( new_freq ! = old_freq ) {
2013-03-24 10:26:43 +04:00
loops_per_jiffy = cpufreq_scale ( lpj_ref ,
2013-08-14 18:08:24 +04:00
lpj_ref_freq , new_freq ) ;
2013-03-24 10:26:43 +04:00
}
2012-05-16 13:49:52 +04:00
return ret ;
2008-04-25 00:58:29 +04:00
}
2012-05-16 13:49:52 +04:00
static int __bfin_cpu_init ( struct cpufreq_policy * policy )
2008-04-25 00:58:29 +04:00
{
2010-01-27 14:16:32 +03:00
unsigned long cclk , sclk ;
2008-04-25 00:58:29 +04:00
2008-10-28 09:18:29 +03:00
cclk = get_cclk ( ) / 1000 ;
sclk = get_sclk ( ) / 1000 ;
2008-04-25 00:58:29 +04:00
2010-01-27 14:16:32 +03:00
if ( policy - > cpu = = CPUFREQ_CPU )
bfin_init_tables ( cclk , sclk ) ;
2008-04-25 00:58:29 +04:00
2009-09-25 13:03:21 +04:00
policy - > cpuinfo . transition_latency = 50000 ; /* 50us assumed */
2013-09-16 17:26:09 +04:00
return cpufreq_table_validate_and_show ( policy , bfin_freq_table ) ;
2008-04-25 00:58:29 +04:00
}
static struct cpufreq_driver bfin_driver = {
2013-10-03 18:57:59 +04:00
. verify = cpufreq_generic_frequency_table_verify ,
2013-10-25 18:15:48 +04:00
. target_index = bfin_target ,
2008-10-28 09:18:29 +03:00
. get = bfin_getfreq_khz ,
2008-04-25 00:58:29 +04:00
. init = __bfin_cpu_init ,
. name = " bfin cpufreq " ,
2013-10-03 18:57:59 +04:00
. attr = cpufreq_generic_attr ,
2008-04-25 00:58:29 +04:00
} ;
static int __init bfin_cpu_init ( void )
{
return cpufreq_register_driver ( & bfin_driver ) ;
}
static void __exit bfin_cpu_exit ( void )
{
cpufreq_unregister_driver ( & bfin_driver ) ;
}
MODULE_AUTHOR ( " Michael Hennerich <hennerich@blackfin.uclinux.org> " ) ;
MODULE_DESCRIPTION ( " cpufreq driver for Blackfin " ) ;
MODULE_LICENSE ( " GPL " ) ;
module_init ( bfin_cpu_init ) ;
module_exit ( bfin_cpu_exit ) ;