2008-11-14 11:01:38 +01:00
/*
* Copyright ( C ) 2008 Sascha Hauer < s . hauer @ pengutronix . de > , Pengutronix
*
* This program is free software ; you can redistribute it and / or modify
2010-06-10 15:59:54 +02:00
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
2008-11-14 11:01:38 +01:00
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
2010-06-10 15:59:54 +02:00
* You should have received a copy of the GNU General Public License along
* with this program ; if not , write to the Free Software Foundation , Inc . ,
* 51 Franklin St , Fifth Floor , Boston , MA 02110 - 1301 , USA .
2008-11-14 11:01:38 +01:00
*/
# include <linux/kernel.h>
# include <linux/init.h>
2009-02-07 13:34:01 +01:00
# include <linux/list.h>
2008-11-14 11:01:38 +01:00
# include <linux/math64.h>
# include <linux/err.h>
# include <linux/clk.h>
# include <linux/io.h>
2009-02-07 13:34:01 +01:00
# include <asm/clkdev.h>
2008-11-14 11:01:38 +01:00
# include <mach/clock.h>
# include <mach/hardware.h>
2009-02-16 14:36:49 +01:00
# include <mach/common.h>
2010-06-10 15:59:54 +02:00
# define IO_ADDR_CCM(off) (MX1_IO_ADDRESS(MX1_CCM_BASE_ADDR + (off)))
/* CCM register addresses */
# define CCM_CSCR IO_ADDR_CCM(0x0)
# define CCM_MPCTL0 IO_ADDR_CCM(0x4)
# define CCM_SPCTL0 IO_ADDR_CCM(0xc)
# define CCM_PCDR IO_ADDR_CCM(0x20)
# define CCM_CSCR_CLKO_OFFSET 29
# define CCM_CSCR_CLKO_MASK (0x7 << 29)
# define CCM_CSCR_USB_OFFSET 26
# define CCM_CSCR_USB_MASK (0x7 << 26)
# define CCM_CSCR_OSC_EN_SHIFT 17
# define CCM_CSCR_SYSTEM_SEL (1 << 16)
# define CCM_CSCR_BCLK_OFFSET 10
# define CCM_CSCR_BCLK_MASK (0xf << 10)
# define CCM_CSCR_PRESC (1 << 15)
# define CCM_PCDR_PCLK3_OFFSET 16
# define CCM_PCDR_PCLK3_MASK (0x7f << 16)
# define CCM_PCDR_PCLK2_OFFSET 4
# define CCM_PCDR_PCLK2_MASK (0xf << 4)
# define CCM_PCDR_PCLK1_OFFSET 0
# define CCM_PCDR_PCLK1_MASK 0xf
# define IO_ADDR_SCM(off) (MX1_IO_ADDRESS(MX1_SCM_BASE_ADDR + (off)))
/* SCM register addresses */
# define SCM_GCCR IO_ADDR_SCM(0xc)
# define SCM_GCCR_DMA_CLK_EN_OFFSET 3
# define SCM_GCCR_CSI_CLK_EN_OFFSET 2
# define SCM_GCCR_MMA_CLK_EN_OFFSET 1
# define SCM_GCCR_USBD_CLK_EN_OFFSET 0
2008-11-14 11:01:38 +01:00
static int _clk_enable ( struct clk * clk )
{
unsigned int reg ;
reg = __raw_readl ( clk - > enable_reg ) ;
reg | = 1 < < clk - > enable_shift ;
__raw_writel ( reg , clk - > enable_reg ) ;
return 0 ;
}
static void _clk_disable ( struct clk * clk )
{
unsigned int reg ;
reg = __raw_readl ( clk - > enable_reg ) ;
reg & = ~ ( 1 < < clk - > enable_shift ) ;
__raw_writel ( reg , clk - > enable_reg ) ;
}
static int _clk_can_use_parent ( const struct clk * clk_arr [ ] , unsigned int size ,
struct clk * parent )
{
int i ;
for ( i = 0 ; i < size ; i + + )
if ( parent = = clk_arr [ i ] )
return i ;
return - EINVAL ;
}
static unsigned long
_clk_simple_round_rate ( struct clk * clk , unsigned long rate , unsigned int limit )
{
int div ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( parent_rate % rate )
div + + ;
if ( div > limit )
div = limit ;
return parent_rate / div ;
}
static unsigned long _clk_parent_round_rate ( struct clk * clk , unsigned long rate )
{
return clk - > parent - > round_rate ( clk - > parent , rate ) ;
}
static int _clk_parent_set_rate ( struct clk * clk , unsigned long rate )
{
return clk - > parent - > set_rate ( clk - > parent , rate ) ;
}
static unsigned long clk16m_get_rate ( struct clk * clk )
{
return 16000000 ;
}
static struct clk clk16m = {
. get_rate = clk16m_get_rate ,
. enable = _clk_enable ,
. enable_reg = CCM_CSCR ,
. enable_shift = CCM_CSCR_OSC_EN_SHIFT ,
. disable = _clk_disable ,
} ;
/* in Hz */
static unsigned long clk32_rate ;
static unsigned long clk32_get_rate ( struct clk * clk )
{
return clk32_rate ;
}
static struct clk clk32 = {
. get_rate = clk32_get_rate ,
} ;
static unsigned long clk32_premult_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) * 512 ;
}
static struct clk clk32_premult = {
. parent = & clk32 ,
. get_rate = clk32_premult_get_rate ,
} ;
static const struct clk * prem_clk_clocks [ ] = {
& clk32_premult ,
& clk16m ,
} ;
static int prem_clk_set_parent ( struct clk * clk , struct clk * parent )
{
int i ;
unsigned int reg = __raw_readl ( CCM_CSCR ) ;
i = _clk_can_use_parent ( prem_clk_clocks , ARRAY_SIZE ( prem_clk_clocks ) ,
parent ) ;
switch ( i ) {
case 0 :
reg & = ~ CCM_CSCR_SYSTEM_SEL ;
break ;
case 1 :
reg | = CCM_CSCR_SYSTEM_SEL ;
break ;
default :
return i ;
}
__raw_writel ( reg , CCM_CSCR ) ;
return 0 ;
}
static struct clk prem_clk = {
. set_parent = prem_clk_set_parent ,
} ;
static unsigned long system_clk_get_rate ( struct clk * clk )
{
2009-01-26 15:41:16 +01:00
return mxc_decode_pll ( __raw_readl ( CCM_SPCTL0 ) ,
2008-11-14 11:01:38 +01:00
clk_get_rate ( clk - > parent ) ) ;
}
static struct clk system_clk = {
. parent = & prem_clk ,
. get_rate = system_clk_get_rate ,
} ;
static unsigned long mcu_clk_get_rate ( struct clk * clk )
{
2009-01-26 15:41:16 +01:00
return mxc_decode_pll ( __raw_readl ( CCM_MPCTL0 ) ,
2008-11-14 11:01:38 +01:00
clk_get_rate ( clk - > parent ) ) ;
}
static struct clk mcu_clk = {
. parent = & clk32_premult ,
. get_rate = mcu_clk_get_rate ,
} ;
static unsigned long fclk_get_rate ( struct clk * clk )
{
unsigned long fclk = clk_get_rate ( clk - > parent ) ;
if ( __raw_readl ( CCM_CSCR ) & CCM_CSCR_PRESC )
fclk / = 2 ;
return fclk ;
}
static struct clk fclk = {
. parent = & mcu_clk ,
. get_rate = fclk_get_rate ,
} ;
/*
* get hclk ( SDRAM , CSI , Memory Stick , I2C , DMA )
*/
static unsigned long hclk_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) / ( ( ( __raw_readl ( CCM_CSCR ) &
CCM_CSCR_BCLK_MASK ) > > CCM_CSCR_BCLK_OFFSET ) + 1 ) ;
}
static unsigned long hclk_round_rate ( struct clk * clk , unsigned long rate )
{
return _clk_simple_round_rate ( clk , rate , 16 ) ;
}
static int hclk_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned int div ;
unsigned int reg ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( div > 16 | | div < 1 | | ( ( parent_rate / div ) ! = rate ) )
return - EINVAL ;
div - - ;
reg = __raw_readl ( CCM_CSCR ) ;
reg & = ~ CCM_CSCR_BCLK_MASK ;
reg | = div < < CCM_CSCR_BCLK_OFFSET ;
__raw_writel ( reg , CCM_CSCR ) ;
return 0 ;
}
static struct clk hclk = {
. parent = & system_clk ,
. get_rate = hclk_get_rate ,
. round_rate = hclk_round_rate ,
. set_rate = hclk_set_rate ,
} ;
static unsigned long clk48m_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) / ( ( ( __raw_readl ( CCM_CSCR ) &
CCM_CSCR_USB_MASK ) > > CCM_CSCR_USB_OFFSET ) + 1 ) ;
}
static unsigned long clk48m_round_rate ( struct clk * clk , unsigned long rate )
{
return _clk_simple_round_rate ( clk , rate , 8 ) ;
}
static int clk48m_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned int div ;
unsigned int reg ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( div > 8 | | div < 1 | | ( ( parent_rate / div ) ! = rate ) )
return - EINVAL ;
div - - ;
reg = __raw_readl ( CCM_CSCR ) ;
reg & = ~ CCM_CSCR_USB_MASK ;
reg | = div < < CCM_CSCR_USB_OFFSET ;
__raw_writel ( reg , CCM_CSCR ) ;
return 0 ;
}
static struct clk clk48m = {
. parent = & system_clk ,
. get_rate = clk48m_get_rate ,
. round_rate = clk48m_round_rate ,
. set_rate = clk48m_set_rate ,
} ;
/*
* get peripheral clock 1 ( UART [ 12 ] , Timer [ 12 ] , PWM )
*/
static unsigned long perclk1_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) / ( ( ( __raw_readl ( CCM_PCDR ) &
CCM_PCDR_PCLK1_MASK ) > > CCM_PCDR_PCLK1_OFFSET ) + 1 ) ;
}
static unsigned long perclk1_round_rate ( struct clk * clk , unsigned long rate )
{
return _clk_simple_round_rate ( clk , rate , 16 ) ;
}
static int perclk1_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned int div ;
unsigned int reg ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( div > 16 | | div < 1 | | ( ( parent_rate / div ) ! = rate ) )
return - EINVAL ;
div - - ;
reg = __raw_readl ( CCM_PCDR ) ;
reg & = ~ CCM_PCDR_PCLK1_MASK ;
reg | = div < < CCM_PCDR_PCLK1_OFFSET ;
__raw_writel ( reg , CCM_PCDR ) ;
return 0 ;
}
/*
* get peripheral clock 2 ( LCD , SD , SPI [ 12 ] )
*/
static unsigned long perclk2_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) / ( ( ( __raw_readl ( CCM_PCDR ) &
CCM_PCDR_PCLK2_MASK ) > > CCM_PCDR_PCLK2_OFFSET ) + 1 ) ;
}
static unsigned long perclk2_round_rate ( struct clk * clk , unsigned long rate )
{
return _clk_simple_round_rate ( clk , rate , 16 ) ;
}
static int perclk2_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned int div ;
unsigned int reg ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( div > 16 | | div < 1 | | ( ( parent_rate / div ) ! = rate ) )
return - EINVAL ;
div - - ;
reg = __raw_readl ( CCM_PCDR ) ;
reg & = ~ CCM_PCDR_PCLK2_MASK ;
reg | = div < < CCM_PCDR_PCLK2_OFFSET ;
__raw_writel ( reg , CCM_PCDR ) ;
return 0 ;
}
/*
* get peripheral clock 3 ( SSI )
*/
static unsigned long perclk3_get_rate ( struct clk * clk )
{
return clk_get_rate ( clk - > parent ) / ( ( ( __raw_readl ( CCM_PCDR ) &
CCM_PCDR_PCLK3_MASK ) > > CCM_PCDR_PCLK3_OFFSET ) + 1 ) ;
}
static unsigned long perclk3_round_rate ( struct clk * clk , unsigned long rate )
{
return _clk_simple_round_rate ( clk , rate , 128 ) ;
}
static int perclk3_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned int div ;
unsigned int reg ;
unsigned long parent_rate ;
parent_rate = clk_get_rate ( clk - > parent ) ;
div = parent_rate / rate ;
if ( div > 128 | | div < 1 | | ( ( parent_rate / div ) ! = rate ) )
return - EINVAL ;
div - - ;
reg = __raw_readl ( CCM_PCDR ) ;
reg & = ~ CCM_PCDR_PCLK3_MASK ;
reg | = div < < CCM_PCDR_PCLK3_OFFSET ;
__raw_writel ( reg , CCM_PCDR ) ;
return 0 ;
}
static struct clk perclk [ ] = {
{
. id = 0 ,
. parent = & system_clk ,
. get_rate = perclk1_get_rate ,
. round_rate = perclk1_round_rate ,
. set_rate = perclk1_set_rate ,
} , {
. id = 1 ,
. parent = & system_clk ,
. get_rate = perclk2_get_rate ,
. round_rate = perclk2_round_rate ,
. set_rate = perclk2_set_rate ,
} , {
. id = 2 ,
. parent = & system_clk ,
. get_rate = perclk3_get_rate ,
. round_rate = perclk3_round_rate ,
. set_rate = perclk3_set_rate ,
}
} ;
static const struct clk * clko_clocks [ ] = {
& perclk [ 0 ] ,
& hclk ,
& clk48m ,
& clk16m ,
& prem_clk ,
& fclk ,
} ;
static int clko_set_parent ( struct clk * clk , struct clk * parent )
{
int i ;
unsigned int reg ;
i = _clk_can_use_parent ( clko_clocks , ARRAY_SIZE ( clko_clocks ) , parent ) ;
if ( i < 0 )
return i ;
reg = __raw_readl ( CCM_CSCR ) & ~ CCM_CSCR_CLKO_MASK ;
reg | = i < < CCM_CSCR_CLKO_OFFSET ;
__raw_writel ( reg , CCM_CSCR ) ;
if ( clko_clocks [ i ] - > set_rate & & clko_clocks [ i ] - > round_rate ) {
clk - > set_rate = _clk_parent_set_rate ;
clk - > round_rate = _clk_parent_round_rate ;
} else {
clk - > set_rate = NULL ;
clk - > round_rate = NULL ;
}
return 0 ;
}
static struct clk clko_clk = {
. set_parent = clko_set_parent ,
} ;
static struct clk dma_clk = {
. parent = & hclk ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
. enable = _clk_enable ,
. enable_reg = SCM_GCCR ,
. enable_shift = SCM_GCCR_DMA_CLK_EN_OFFSET ,
. disable = _clk_disable ,
} ;
static struct clk csi_clk = {
. parent = & hclk ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
. enable = _clk_enable ,
. enable_reg = SCM_GCCR ,
. enable_shift = SCM_GCCR_CSI_CLK_EN_OFFSET ,
. disable = _clk_disable ,
} ;
static struct clk mma_clk = {
. parent = & hclk ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
. enable = _clk_enable ,
. enable_reg = SCM_GCCR ,
. enable_shift = SCM_GCCR_MMA_CLK_EN_OFFSET ,
. disable = _clk_disable ,
} ;
static struct clk usbd_clk = {
. parent = & clk48m ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
. enable = _clk_enable ,
. enable_reg = SCM_GCCR ,
. enable_shift = SCM_GCCR_USBD_CLK_EN_OFFSET ,
. disable = _clk_disable ,
} ;
static struct clk gpt_clk = {
. parent = & perclk [ 0 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk uart_clk = {
. parent = & perclk [ 0 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk i2c_clk = {
. parent = & hclk ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk spi_clk = {
. parent = & perclk [ 1 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk sdhc_clk = {
. parent = & perclk [ 1 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk lcdc_clk = {
. parent = & perclk [ 1 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk mshc_clk = {
. parent = & hclk ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk ssi_clk = {
. parent = & perclk [ 2 ] ,
. round_rate = _clk_parent_round_rate ,
. set_rate = _clk_parent_set_rate ,
} ;
static struct clk rtc_clk = {
. parent = & clk32 ,
} ;
2009-02-07 13:34:01 +01:00
# define _REGISTER_CLOCK(d, n, c) \
{ \
. dev_id = d , \
. con_id = n , \
. clk = & c , \
} ,
static struct clk_lookup lookups [ ] __initdata = {
_REGISTER_CLOCK ( NULL , " dma " , dma_clk )
_REGISTER_CLOCK ( " mx1-camera.0 " , NULL , csi_clk )
_REGISTER_CLOCK ( NULL , " mma " , mma_clk )
_REGISTER_CLOCK ( " imx_udc.0 " , NULL , usbd_clk )
_REGISTER_CLOCK ( NULL , " gpt " , gpt_clk )
_REGISTER_CLOCK ( " imx-uart.0 " , NULL , uart_clk )
_REGISTER_CLOCK ( " imx-uart.1 " , NULL , uart_clk )
_REGISTER_CLOCK ( " imx-uart.2 " , NULL , uart_clk )
_REGISTER_CLOCK ( " imx-i2c.0 " , NULL , i2c_clk )
_REGISTER_CLOCK ( " spi_imx.0 " , NULL , spi_clk )
_REGISTER_CLOCK ( " imx-mmc.0 " , NULL , sdhc_clk )
_REGISTER_CLOCK ( " imx-fb.0 " , NULL , lcdc_clk )
_REGISTER_CLOCK ( NULL , " mshc " , mshc_clk )
_REGISTER_CLOCK ( NULL , " ssi " , ssi_clk )
_REGISTER_CLOCK ( " mxc_rtc.0 " , NULL , rtc_clk )
2008-11-14 11:01:38 +01:00
} ;
2009-02-16 14:36:49 +01:00
int __init mx1_clocks_init ( unsigned long fref )
2008-11-14 11:01:38 +01:00
{
unsigned int reg ;
/* disable clocks we are able to */
__raw_writel ( 0 , SCM_GCCR ) ;
clk32_rate = fref ;
reg = __raw_readl ( CCM_CSCR ) ;
/* detect clock reference for system PLL */
if ( reg & CCM_CSCR_SYSTEM_SEL ) {
prem_clk . parent = & clk16m ;
} else {
/* ensure that oscillator is disabled */
reg & = ~ ( 1 < < CCM_CSCR_OSC_EN_SHIFT ) ;
__raw_writel ( reg , CCM_CSCR ) ;
prem_clk . parent = & clk32_premult ;
}
/* detect reference for CLKO */
reg = ( reg & CCM_CSCR_CLKO_MASK ) > > CCM_CSCR_CLKO_OFFSET ;
clko_clk . parent = ( struct clk * ) clko_clocks [ reg ] ;
2010-01-12 12:28:00 +00:00
clkdev_add_table ( lookups , ARRAY_SIZE ( lookups ) ) ;
2008-11-14 11:01:38 +01:00
clk_enable ( & hclk ) ;
clk_enable ( & fclk ) ;
2010-03-04 21:02:41 +01:00
mxc_timer_init ( & gpt_clk , MX1_IO_ADDRESS ( MX1_TIM1_BASE_ADDR ) ,
MX1_TIM1_INT ) ;
2009-02-16 14:36:49 +01:00
2008-11-14 11:01:38 +01:00
return 0 ;
}