2008-10-21 14:07:00 +01:00
/* linux/arch/arm/plat-s3c64xx/clock.c
*
* Copyright 2008 Openmoko , Inc .
* Copyright 2008 Simtec Electronics
* Ben Dooks < ben @ simtec . co . uk >
* http : //armlinux.simtec.co.uk/
*
* S3C64XX Base clock support
*
* 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/module.h>
# include <linux/interrupt.h>
# include <linux/ioport.h>
2010-01-26 14:53:19 +09:00
# include <linux/clk.h>
# include <linux/err.h>
2008-10-21 14:07:00 +01:00
# include <linux/io.h>
# include <mach/hardware.h>
# include <mach/map.h>
2010-01-26 10:45:40 +09:00
# include <mach/regs-sys.h>
# include <mach/regs-clock.h>
2010-01-26 13:41:30 +09:00
2008-10-21 14:07:00 +01:00
# include <plat/cpu.h>
# include <plat/devs.h>
2010-01-26 14:53:19 +09:00
# include <plat/cpu-freq.h>
2008-10-21 14:07:00 +01:00
# include <plat/clock.h>
2010-01-26 14:53:19 +09:00
# include <plat/clock-clksrc.h>
2011-10-04 19:41:43 +09:00
# include <plat/pll.h>
2010-01-26 14:53:19 +09:00
/* fin_apll, fin_mpll and fin_epll are all the same clock, which we call
* ext_xtal_mux for want of an actual name from the manual .
*/
static struct clk clk_ext_xtal_mux = {
. name = " ext_xtal " ,
} ;
# define clk_fin_apll clk_ext_xtal_mux
# define clk_fin_mpll clk_ext_xtal_mux
# define clk_fin_epll clk_ext_xtal_mux
# define clk_fout_mpll clk_mpll
# define clk_fout_epll clk_epll
2008-10-21 14:07:00 +01:00
2009-03-05 11:43:13 +08:00
struct clk clk_h2 = {
. name = " hclk2 " ,
. rate = 0 ,
} ;
2008-10-21 14:07:00 +01:00
struct clk clk_27m = {
. name = " clk_27m " ,
. rate = 27000000 ,
} ;
2008-10-31 16:14:36 +00:00
static int clk_48m_ctrl ( struct clk * clk , int enable )
{
unsigned long flags ;
u32 val ;
/* can't rely on clock lock, this register has other usages */
local_irq_save ( flags ) ;
val = __raw_readl ( S3C64XX_OTHERS ) ;
if ( enable )
val | = S3C64XX_OTHERS_USBMASK ;
else
val & = ~ S3C64XX_OTHERS_USBMASK ;
__raw_writel ( val , S3C64XX_OTHERS ) ;
local_irq_restore ( flags ) ;
return 0 ;
}
2008-10-21 14:07:00 +01:00
struct clk clk_48m = {
. name = " clk_48m " ,
. rate = 48000000 ,
2008-10-31 16:14:36 +00:00
. enable = clk_48m_ctrl ,
2008-10-21 14:07:00 +01:00
} ;
2010-05-17 20:17:42 +02:00
struct clk clk_xusbxti = {
. name = " xusbxti " ,
. rate = 48000000 ,
} ;
2008-10-21 14:07:00 +01:00
static int inline s3c64xx_gate ( void __iomem * reg ,
struct clk * clk ,
int enable )
{
unsigned int ctrlbit = clk - > ctrlbit ;
u32 con ;
con = __raw_readl ( reg ) ;
if ( enable )
con | = ctrlbit ;
else
con & = ~ ctrlbit ;
__raw_writel ( con , reg ) ;
return 0 ;
}
static int s3c64xx_pclk_ctrl ( struct clk * clk , int enable )
{
return s3c64xx_gate ( S3C_PCLK_GATE , clk , enable ) ;
}
static int s3c64xx_hclk_ctrl ( struct clk * clk , int enable )
{
return s3c64xx_gate ( S3C_HCLK_GATE , clk , enable ) ;
}
2008-10-21 14:07:02 +01:00
int s3c64xx_sclk_ctrl ( struct clk * clk , int enable )
2008-10-21 14:07:00 +01:00
{
return s3c64xx_gate ( S3C_SCLK_GATE , clk , enable ) ;
}
2011-01-04 18:27:18 +09:00
static struct clk init_clocks_off [ ] = {
2008-10-21 14:07:00 +01:00
{
. name = " nand " ,
. parent = & clk_h ,
2010-07-15 11:56:15 +05:30
} , {
. name = " rtc " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_RTC ,
2008-10-21 14:07:00 +01:00
} , {
. name = " adc " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_TSADC ,
} , {
. name = " i2c " ,
2012-01-27 14:43:44 +09:00
# ifdef CONFIG_S3C_DEV_I2C1
. devname = " s3c2440-i2c.0 " ,
# else
. devname = " s3c2440-i2c " ,
# endif
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_IIC ,
2011-03-04 07:55:44 +09:00
} , {
. name = " i2c " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c2440-i2c.1 " ,
2011-03-04 07:55:44 +09:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C6410_CLKCON_PCLK_I2C1 ,
2008-10-21 14:07:00 +01:00
} , {
. name = " iis " ,
2011-06-14 19:12:26 +09:00
. devname = " samsung-i2s.0 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_IIS0 ,
} , {
. name = " iis " ,
2011-06-14 19:12:26 +09:00
. devname = " samsung-i2s.1 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_IIS1 ,
} , {
2010-02-17 19:03:19 +00:00
# ifdef CONFIG_CPU_S3C6410
. name = " iis " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C6410_CLKCON_PCLK_IIS2 ,
} , {
# endif
2010-06-22 07:39:18 +09:00
. name = " keypad " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_KEYPAD ,
} , {
2008-10-21 14:07:00 +01:00
. name = " spi " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.0 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_SPI0 ,
} , {
. name = " spi " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.1 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_SPI1 ,
} , {
. name = " 48m " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c-sdhci.0 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_48m ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_MMC0_48 ,
} , {
. name = " 48m " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c-sdhci.1 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_48m ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_MMC1_48 ,
} , {
. name = " 48m " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c-sdhci.2 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_48m ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_MMC2_48 ,
2012-01-12 10:41:35 +09:00
} , {
. name = " ac97 " ,
. parent = & clk_p ,
. ctrlbit = S3C_CLKCON_PCLK_AC97 ,
} , {
. name = " cfcon " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_IHOST ,
2009-04-28 16:06:24 +01:00
} , {
. name = " dma0 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_DMA0 ,
} , {
. name = " dma1 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_DMA1 ,
2012-01-27 14:42:19 +09:00
} , {
. name = " 3dse " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_3DSE ,
} , {
. name = " hclk_secur " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_SECUR ,
} , {
. name = " sdma1 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_SDMA1 ,
} , {
. name = " sdma0 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_SDMA0 ,
} , {
. name = " hclk_jpeg " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_JPEG ,
} , {
. name = " camif " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_CAMIF ,
} , {
. name = " hclk_scaler " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_SCALER ,
} , {
. name = " 2d " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_2D ,
} , {
. name = " tv " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_TV ,
} , {
. name = " post0 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_POST0 ,
} , {
. name = " rot " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_ROT ,
} , {
. name = " hclk_mfc " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_MFC ,
} , {
. name = " pclk_mfc " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_MFC ,
} , {
. name = " dac27 " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_DAC27 ,
} , {
. name = " tv27 " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_TV27 ,
} , {
. name = " scaler27 " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_SCALER27 ,
} , {
. name = " sclk_scaler " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_SCALER ,
} , {
. name = " post0_27 " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_POST0_27 ,
} , {
. name = " secur " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_SECUR ,
} , {
. name = " sclk_mfc " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_MFC ,
} , {
. name = " cam " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_CAM ,
} , {
. name = " sclk_jpeg " ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_JPEG ,
2008-10-21 14:07:00 +01:00
} ,
} ;
2011-11-02 20:04:08 +09:00
static struct clk clk_48m_spi0 = {
. name = " spi_48m " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.0 " ,
2011-11-02 20:04:08 +09:00
. parent = & clk_48m ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_SPI0_48 ,
} ;
static struct clk clk_48m_spi1 = {
. name = " spi_48m " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.1 " ,
2011-11-02 20:04:08 +09:00
. parent = & clk_48m ,
. enable = s3c64xx_sclk_ctrl ,
. ctrlbit = S3C_CLKCON_SCLK_SPI1_48 ,
} ;
2008-10-21 14:07:00 +01:00
static struct clk init_clocks [ ] = {
{
. name = " lcd " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_LCD ,
} , {
. name = " gpio " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_GPIO ,
} , {
. name = " usb-host " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
2009-06-18 23:54:44 +02:00
. ctrlbit = S3C_CLKCON_HCLK_UHOST ,
2010-05-28 11:41:14 +09:00
} , {
. name = " otg " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_USB ,
2008-10-21 14:07:00 +01:00
} , {
. name = " timers " ,
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_PWM ,
} , {
. name = " uart " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c6400-uart.0 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_UART0 ,
} , {
. name = " uart " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c6400-uart.1 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_UART1 ,
} , {
. name = " uart " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c6400-uart.2 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_UART2 ,
} , {
. name = " uart " ,
2011-06-14 19:12:26 +09:00
. devname = " s3c6400-uart.3 " ,
2008-10-21 14:07:00 +01:00
. parent = & clk_p ,
. enable = s3c64xx_pclk_ctrl ,
. ctrlbit = S3C_CLKCON_PCLK_UART3 ,
} , {
. name = " watchdog " ,
. parent = & clk_p ,
. ctrlbit = S3C_CLKCON_PCLK_WDT ,
2012-01-12 10:41:35 +09:00
} ,
2008-10-21 14:07:00 +01:00
} ;
2011-10-24 17:05:58 +02:00
static struct clk clk_hsmmc0 = {
. name = " hsmmc " ,
. devname = " s3c-sdhci.0 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_HSMMC0 ,
} ;
static struct clk clk_hsmmc1 = {
. name = " hsmmc " ,
. devname = " s3c-sdhci.1 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_HSMMC1 ,
} ;
static struct clk clk_hsmmc2 = {
. name = " hsmmc " ,
. devname = " s3c-sdhci.2 " ,
. parent = & clk_h ,
. enable = s3c64xx_hclk_ctrl ,
. ctrlbit = S3C_CLKCON_HCLK_HSMMC2 ,
} ;
2010-01-26 14:53:19 +09:00
static struct clk clk_fout_apll = {
. name = " fout_apll " ,
} ;
static struct clk * clk_src_apll_list [ ] = {
[ 0 ] = & clk_fin_apll ,
[ 1 ] = & clk_fout_apll ,
} ;
static struct clksrc_sources clk_src_apll = {
. sources = clk_src_apll_list ,
. nr_sources = ARRAY_SIZE ( clk_src_apll_list ) ,
} ;
static struct clksrc_clk clk_mout_apll = {
. clk = {
. name = " mout_apll " ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 0 , . size = 1 } ,
. sources = & clk_src_apll ,
} ;
static struct clk * clk_src_epll_list [ ] = {
[ 0 ] = & clk_fin_epll ,
[ 1 ] = & clk_fout_epll ,
} ;
static struct clksrc_sources clk_src_epll = {
. sources = clk_src_epll_list ,
. nr_sources = ARRAY_SIZE ( clk_src_epll_list ) ,
} ;
static struct clksrc_clk clk_mout_epll = {
. clk = {
. name = " mout_epll " ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 2 , . size = 1 } ,
. sources = & clk_src_epll ,
} ;
static struct clk * clk_src_mpll_list [ ] = {
[ 0 ] = & clk_fin_mpll ,
[ 1 ] = & clk_fout_mpll ,
} ;
static struct clksrc_sources clk_src_mpll = {
. sources = clk_src_mpll_list ,
. nr_sources = ARRAY_SIZE ( clk_src_mpll_list ) ,
} ;
static struct clksrc_clk clk_mout_mpll = {
. clk = {
. name = " mout_mpll " ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 1 , . size = 1 } ,
. sources = & clk_src_mpll ,
} ;
static unsigned int armclk_mask ;
static unsigned long s3c64xx_clk_arm_get_rate ( struct clk * clk )
{
unsigned long rate = clk_get_rate ( clk - > parent ) ;
u32 clkdiv ;
/* divisor mask starts at bit0, so no need to shift */
clkdiv = __raw_readl ( S3C_CLK_DIV0 ) & armclk_mask ;
return rate / ( clkdiv + 1 ) ;
}
static unsigned long s3c64xx_clk_arm_round_rate ( struct clk * clk ,
unsigned long rate )
{
unsigned long parent = clk_get_rate ( clk - > parent ) ;
u32 div ;
if ( parent < rate )
return parent ;
div = ( parent / rate ) - 1 ;
if ( div > armclk_mask )
div = armclk_mask ;
return parent / ( div + 1 ) ;
}
static int s3c64xx_clk_arm_set_rate ( struct clk * clk , unsigned long rate )
{
unsigned long parent = clk_get_rate ( clk - > parent ) ;
u32 div ;
u32 val ;
if ( rate < parent / ( armclk_mask + 1 ) )
return - EINVAL ;
rate = clk_round_rate ( clk , rate ) ;
div = clk_get_rate ( clk - > parent ) / rate ;
val = __raw_readl ( S3C_CLK_DIV0 ) ;
val & = ~ armclk_mask ;
val | = ( div - 1 ) ;
__raw_writel ( val , S3C_CLK_DIV0 ) ;
return 0 ;
}
static struct clk clk_arm = {
. name = " armclk " ,
. parent = & clk_mout_apll . clk ,
. ops = & ( struct clk_ops ) {
. get_rate = s3c64xx_clk_arm_get_rate ,
. set_rate = s3c64xx_clk_arm_set_rate ,
. round_rate = s3c64xx_clk_arm_round_rate ,
} ,
} ;
static unsigned long s3c64xx_clk_doutmpll_get_rate ( struct clk * clk )
{
unsigned long rate = clk_get_rate ( clk - > parent ) ;
printk ( KERN_DEBUG " %s: parent is %ld \n " , __func__ , rate ) ;
if ( __raw_readl ( S3C_CLK_DIV0 ) & S3C6400_CLKDIV0_MPLL_MASK )
rate / = 2 ;
return rate ;
}
static struct clk_ops clk_dout_ops = {
. get_rate = s3c64xx_clk_doutmpll_get_rate ,
} ;
static struct clk clk_dout_mpll = {
. name = " dout_mpll " ,
. parent = & clk_mout_mpll . clk ,
. ops = & clk_dout_ops ,
} ;
static struct clk * clkset_spi_mmc_list [ ] = {
& clk_mout_epll . clk ,
& clk_dout_mpll ,
& clk_fin_epll ,
& clk_27m ,
} ;
static struct clksrc_sources clkset_spi_mmc = {
. sources = clkset_spi_mmc_list ,
. nr_sources = ARRAY_SIZE ( clkset_spi_mmc_list ) ,
} ;
static struct clk * clkset_irda_list [ ] = {
& clk_mout_epll . clk ,
& clk_dout_mpll ,
NULL ,
& clk_27m ,
} ;
static struct clksrc_sources clkset_irda = {
. sources = clkset_irda_list ,
. nr_sources = ARRAY_SIZE ( clkset_irda_list ) ,
} ;
static struct clk * clkset_uart_list [ ] = {
& clk_mout_epll . clk ,
& clk_dout_mpll ,
NULL ,
NULL
} ;
static struct clksrc_sources clkset_uart = {
. sources = clkset_uart_list ,
. nr_sources = ARRAY_SIZE ( clkset_uart_list ) ,
} ;
static struct clk * clkset_uhost_list [ ] = {
& clk_48m ,
& clk_mout_epll . clk ,
& clk_dout_mpll ,
& clk_fin_epll ,
} ;
static struct clksrc_sources clkset_uhost = {
. sources = clkset_uhost_list ,
. nr_sources = ARRAY_SIZE ( clkset_uhost_list ) ,
} ;
/* The peripheral clocks are all controlled via clocksource followed
* by an optional divider and gate stage . We currently roll this into
* one clock which hides the intermediate clock from the mux .
*
* Note , the JPEG clock can only be an even divider . . .
*
* The scaler and LCD clocks depend on the S3C64XX version , and also
* have a common parent divisor so are not included here .
*/
/* clocks that feed other parts of the clock source tree */
static struct clk clk_iis_cd0 = {
. name = " iis_cdclk0 " ,
} ;
static struct clk clk_iis_cd1 = {
. name = " iis_cdclk1 " ,
} ;
2010-03-09 15:10:32 +09:00
static struct clk clk_iisv4_cd = {
. name = " iis_cdclk_v4 " ,
} ;
2010-01-26 14:53:19 +09:00
static struct clk clk_pcm_cd = {
. name = " pcm_cdclk " ,
} ;
static struct clk * clkset_audio0_list [ ] = {
[ 0 ] = & clk_mout_epll . clk ,
[ 1 ] = & clk_dout_mpll ,
[ 2 ] = & clk_fin_epll ,
[ 3 ] = & clk_iis_cd0 ,
[ 4 ] = & clk_pcm_cd ,
} ;
static struct clksrc_sources clkset_audio0 = {
. sources = clkset_audio0_list ,
. nr_sources = ARRAY_SIZE ( clkset_audio0_list ) ,
} ;
static struct clk * clkset_audio1_list [ ] = {
[ 0 ] = & clk_mout_epll . clk ,
[ 1 ] = & clk_dout_mpll ,
[ 2 ] = & clk_fin_epll ,
[ 3 ] = & clk_iis_cd1 ,
[ 4 ] = & clk_pcm_cd ,
} ;
static struct clksrc_sources clkset_audio1 = {
. sources = clkset_audio1_list ,
. nr_sources = ARRAY_SIZE ( clkset_audio1_list ) ,
} ;
2010-03-09 15:10:33 +09:00
static struct clk * clkset_audio2_list [ ] = {
[ 0 ] = & clk_mout_epll . clk ,
[ 1 ] = & clk_dout_mpll ,
[ 2 ] = & clk_fin_epll ,
[ 3 ] = & clk_iisv4_cd ,
[ 4 ] = & clk_pcm_cd ,
} ;
static struct clksrc_sources clkset_audio2 = {
. sources = clkset_audio2_list ,
. nr_sources = ARRAY_SIZE ( clkset_audio2_list ) ,
} ;
2010-01-26 14:53:19 +09:00
static struct clk * clkset_camif_list [ ] = {
& clk_h2 ,
} ;
static struct clksrc_sources clkset_camif = {
. sources = clkset_camif_list ,
. nr_sources = ARRAY_SIZE ( clkset_camif_list ) ,
} ;
static struct clksrc_clk clksrcs [ ] = {
{
. clk = {
. name = " usb-bus-host " ,
. ctrlbit = S3C_CLKCON_SCLK_UHOST ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 5 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV1 , . shift = 20 , . size = 4 } ,
. sources = & clkset_uhost ,
} , {
. clk = {
. name = " audio-bus " ,
2011-06-14 19:12:26 +09:00
. devname = " samsung-i2s.0 " ,
2010-01-26 14:53:19 +09:00
. ctrlbit = S3C_CLKCON_SCLK_AUDIO0 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 7 , . size = 3 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 8 , . size = 4 } ,
. sources = & clkset_audio0 ,
} , {
. clk = {
. name = " audio-bus " ,
2011-06-14 19:12:26 +09:00
. devname = " samsung-i2s.1 " ,
2010-01-26 14:53:19 +09:00
. ctrlbit = S3C_CLKCON_SCLK_AUDIO1 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 10 , . size = 3 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 12 , . size = 4 } ,
. sources = & clkset_audio1 ,
2010-03-09 15:10:34 +09:00
} , {
. clk = {
. name = " audio-bus " ,
2011-06-14 19:12:26 +09:00
. devname = " samsung-i2s.2 " ,
2010-03-09 15:10:34 +09:00
. ctrlbit = S3C6410_CLKCON_SCLK_AUDIO2 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C6410_CLK_SRC2 , . shift = 0 , . size = 3 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 24 , . size = 4 } ,
. sources = & clkset_audio2 ,
2010-01-26 14:53:19 +09:00
} , {
. clk = {
. name = " irda-bus " ,
. ctrlbit = S3C_CLKCON_SCLK_IRDA ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 24 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 20 , . size = 4 } ,
. sources = & clkset_irda ,
} , {
. clk = {
. name = " camera " ,
. ctrlbit = S3C_CLKCON_SCLK_CAM ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_div = { . reg = S3C_CLK_DIV0 , . shift = 20 , . size = 4 } ,
. reg_src = { . reg = NULL , . shift = 0 , . size = 0 } ,
. sources = & clkset_camif ,
} ,
} ;
2011-10-24 12:08:42 +02:00
/* Where does UCLK0 come from? */
static struct clksrc_clk clk_sclk_uclk = {
. clk = {
. name = " uclk1 " ,
. ctrlbit = S3C_CLKCON_SCLK_UART ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 13 , . size = 1 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 16 , . size = 4 } ,
. sources = & clkset_uart ,
} ;
2011-10-24 17:05:58 +02:00
static struct clksrc_clk clk_sclk_mmc0 = {
. clk = {
. name = " mmc_bus " ,
. devname = " s3c-sdhci.0 " ,
. ctrlbit = S3C_CLKCON_SCLK_MMC0 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 18 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV1 , . shift = 0 , . size = 4 } ,
. sources = & clkset_spi_mmc ,
} ;
static struct clksrc_clk clk_sclk_mmc1 = {
. clk = {
. name = " mmc_bus " ,
. devname = " s3c-sdhci.1 " ,
. ctrlbit = S3C_CLKCON_SCLK_MMC1 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 20 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV1 , . shift = 4 , . size = 4 } ,
. sources = & clkset_spi_mmc ,
} ;
static struct clksrc_clk clk_sclk_mmc2 = {
. clk = {
. name = " mmc_bus " ,
. devname = " s3c-sdhci.2 " ,
. ctrlbit = S3C_CLKCON_SCLK_MMC2 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 22 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV1 , . shift = 8 , . size = 4 } ,
. sources = & clkset_spi_mmc ,
} ;
2011-11-02 20:04:08 +09:00
static struct clksrc_clk clk_sclk_spi0 = {
. clk = {
. name = " spi-bus " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.0 " ,
2011-11-02 20:04:08 +09:00
. ctrlbit = S3C_CLKCON_SCLK_SPI0 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 14 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 0 , . size = 4 } ,
. sources = & clkset_spi_mmc ,
} ;
static struct clksrc_clk clk_sclk_spi1 = {
. clk = {
. name = " spi-bus " ,
2012-07-13 07:15:14 +09:00
. devname = " s3c6410-spi.1 " ,
2011-11-02 20:04:08 +09:00
. ctrlbit = S3C_CLKCON_SCLK_SPI1 ,
. enable = s3c64xx_sclk_ctrl ,
} ,
. reg_src = { . reg = S3C_CLK_SRC , . shift = 16 , . size = 2 } ,
. reg_div = { . reg = S3C_CLK_DIV2 , . shift = 4 , . size = 4 } ,
. sources = & clkset_spi_mmc ,
} ;
2010-01-26 14:53:19 +09:00
/* Clock initialisation code */
static struct clksrc_clk * init_parents [ ] = {
& clk_mout_apll ,
& clk_mout_epll ,
& clk_mout_mpll ,
} ;
2011-10-24 12:08:42 +02:00
static struct clksrc_clk * clksrc_cdev [ ] = {
& clk_sclk_uclk ,
2011-10-24 17:05:58 +02:00
& clk_sclk_mmc0 ,
& clk_sclk_mmc1 ,
& clk_sclk_mmc2 ,
2011-11-02 20:04:08 +09:00
& clk_sclk_spi0 ,
& clk_sclk_spi1 ,
2011-10-24 17:05:58 +02:00
} ;
static struct clk * clk_cdev [ ] = {
& clk_hsmmc0 ,
& clk_hsmmc1 ,
& clk_hsmmc2 ,
2011-11-02 20:04:08 +09:00
& clk_48m_spi0 ,
& clk_48m_spi1 ,
2011-10-24 12:08:42 +02:00
} ;
static struct clk_lookup s3c64xx_clk_lookup [ ] = {
CLKDEV_INIT ( NULL , " clk_uart_baud2 " , & clk_p ) ,
CLKDEV_INIT ( NULL , " clk_uart_baud3 " , & clk_sclk_uclk . clk ) ,
2011-10-24 17:05:58 +02:00
CLKDEV_INIT ( " s3c-sdhci.0 " , " mmc_busclk.0 " , & clk_hsmmc0 ) ,
CLKDEV_INIT ( " s3c-sdhci.1 " , " mmc_busclk.0 " , & clk_hsmmc1 ) ,
CLKDEV_INIT ( " s3c-sdhci.2 " , " mmc_busclk.0 " , & clk_hsmmc2 ) ,
CLKDEV_INIT ( " s3c-sdhci.0 " , " mmc_busclk.2 " , & clk_sclk_mmc0 . clk ) ,
CLKDEV_INIT ( " s3c-sdhci.1 " , " mmc_busclk.2 " , & clk_sclk_mmc1 . clk ) ,
CLKDEV_INIT ( " s3c-sdhci.2 " , " mmc_busclk.2 " , & clk_sclk_mmc2 . clk ) ,
2011-11-02 20:04:08 +09:00
CLKDEV_INIT ( NULL , " spi_busclk0 " , & clk_p ) ,
2012-07-13 07:15:14 +09:00
CLKDEV_INIT ( " s3c6410-spi.0 " , " spi_busclk1 " , & clk_sclk_spi0 . clk ) ,
CLKDEV_INIT ( " s3c6410-spi.0 " , " spi_busclk2 " , & clk_48m_spi0 ) ,
CLKDEV_INIT ( " s3c6410-spi.1 " , " spi_busclk1 " , & clk_sclk_spi1 . clk ) ,
CLKDEV_INIT ( " s3c6410-spi.1 " , " spi_busclk2 " , & clk_48m_spi1 ) ,
2011-10-24 12:08:42 +02:00
} ;
2010-01-26 14:53:19 +09:00
# define GET_DIV(clk, field) ((((clk) & field##_MASK) >> field##_SHIFT) + 1)
2011-12-22 23:27:42 +01:00
void __init_or_cpufreq s3c64xx_setup_clocks ( void )
2010-01-26 14:53:19 +09:00
{
struct clk * xtal_clk ;
unsigned long xtal ;
unsigned long fclk ;
unsigned long hclk ;
unsigned long hclk2 ;
unsigned long pclk ;
unsigned long epll ;
unsigned long apll ;
unsigned long mpll ;
unsigned int ptr ;
u32 clkdiv0 ;
printk ( KERN_DEBUG " %s: registering clocks \n " , __func__ ) ;
clkdiv0 = __raw_readl ( S3C_CLK_DIV0 ) ;
printk ( KERN_DEBUG " %s: clkdiv0 = %08x \n " , __func__ , clkdiv0 ) ;
xtal_clk = clk_get ( NULL , " xtal " ) ;
BUG_ON ( IS_ERR ( xtal_clk ) ) ;
xtal = clk_get_rate ( xtal_clk ) ;
clk_put ( xtal_clk ) ;
printk ( KERN_DEBUG " %s: xtal is %ld \n " , __func__ , xtal ) ;
/* For now assume the mux always selects the crystal */
clk_ext_xtal_mux . parent = xtal_clk ;
2011-10-04 19:41:43 +09:00
epll = s3c_get_pll6553x ( xtal , __raw_readl ( S3C_EPLL_CON0 ) ,
__raw_readl ( S3C_EPLL_CON1 ) ) ;
2010-01-26 14:53:19 +09:00
mpll = s3c6400_get_pll ( xtal , __raw_readl ( S3C_MPLL_CON ) ) ;
apll = s3c6400_get_pll ( xtal , __raw_readl ( S3C_APLL_CON ) ) ;
fclk = mpll ;
printk ( KERN_INFO " S3C64XX: PLL settings, A=%ld, M=%ld, E=%ld \n " ,
apll , mpll , epll ) ;
2011-08-19 11:54:31 +02:00
if ( __raw_readl ( S3C64XX_OTHERS ) & S3C64XX_OTHERS_SYNCMUXSEL )
/* Synchronous mode */
hclk2 = apll / GET_DIV ( clkdiv0 , S3C6400_CLKDIV0_HCLK2 ) ;
else
/* Asynchronous mode */
hclk2 = mpll / GET_DIV ( clkdiv0 , S3C6400_CLKDIV0_HCLK2 ) ;
2010-01-26 14:53:19 +09:00
hclk = hclk2 / GET_DIV ( clkdiv0 , S3C6400_CLKDIV0_HCLK ) ;
pclk = hclk2 / GET_DIV ( clkdiv0 , S3C6400_CLKDIV0_PCLK ) ;
printk ( KERN_INFO " S3C64XX: HCLK2=%ld, HCLK=%ld, PCLK=%ld \n " ,
hclk2 , hclk , pclk ) ;
clk_fout_mpll . rate = mpll ;
clk_fout_epll . rate = epll ;
clk_fout_apll . rate = apll ;
clk_h2 . rate = hclk2 ;
clk_h . rate = hclk ;
clk_p . rate = pclk ;
clk_f . rate = fclk ;
for ( ptr = 0 ; ptr < ARRAY_SIZE ( init_parents ) ; ptr + + )
s3c_set_clksrc ( init_parents [ ptr ] , true ) ;
for ( ptr = 0 ; ptr < ARRAY_SIZE ( clksrcs ) ; ptr + + )
s3c_set_clksrc ( & clksrcs [ ptr ] , true ) ;
}
static struct clk * clks1 [ ] __initdata = {
& clk_ext_xtal_mux ,
& clk_iis_cd0 ,
& clk_iis_cd1 ,
2010-03-09 15:10:32 +09:00
& clk_iisv4_cd ,
2010-01-26 14:53:19 +09:00
& clk_pcm_cd ,
& clk_mout_epll . clk ,
& clk_mout_mpll . clk ,
& clk_dout_mpll ,
& clk_arm ,
} ;
2010-01-26 15:10:38 +09:00
static struct clk * clks [ ] __initdata = {
& clk_ext ,
& clk_epll ,
& clk_27m ,
& clk_48m ,
& clk_h2 ,
2010-05-17 20:17:42 +02:00
& clk_xusbxti ,
2010-01-26 15:10:38 +09:00
} ;
2010-01-26 14:53:19 +09:00
/**
2010-01-26 15:10:38 +09:00
* s3c64xx_register_clocks - register clocks for s3c6400 and s3c6410
* @ xtal : The rate for the clock crystal feeding the PLLs .
* @ armclk_divlimit : Divisor mask for ARMCLK .
2010-01-26 14:53:19 +09:00
*
2010-01-26 15:10:38 +09:00
* Register the clocks for the S3C6400 and S3C6410 SoC range , such
* as ARMCLK as well as the necessary parent clocks .
2010-01-26 14:53:19 +09:00
*
* This call does not setup the clocks , which is left to the
2011-12-22 23:27:42 +01:00
* s3c64xx_setup_clocks ( ) call which may be needed by the cpufreq
2010-01-26 14:53:19 +09:00
* or resume code to re - set the clocks if the bootloader has changed
* them .
*/
2010-01-26 15:10:38 +09:00
void __init s3c64xx_register_clocks ( unsigned long xtal ,
unsigned armclk_divlimit )
2010-01-26 14:53:19 +09:00
{
2011-10-24 12:08:42 +02:00
unsigned int cnt ;
2010-01-26 14:53:19 +09:00
armclk_mask = armclk_divlimit ;
2010-01-26 15:10:38 +09:00
s3c24xx_register_baseclocks ( xtal ) ;
2008-10-21 14:07:00 +01:00
s3c24xx_register_clocks ( clks , ARRAY_SIZE ( clks ) ) ;
2010-01-26 15:10:38 +09:00
2010-01-06 01:21:38 +09:00
s3c_register_clocks ( init_clocks , ARRAY_SIZE ( init_clocks ) ) ;
2008-10-21 14:07:00 +01:00
2011-01-04 18:27:18 +09:00
s3c_register_clocks ( init_clocks_off , ARRAY_SIZE ( init_clocks_off ) ) ;
s3c_disable_clocks ( init_clocks_off , ARRAY_SIZE ( init_clocks_off ) ) ;
2008-11-21 10:36:05 +00:00
2011-10-24 17:05:58 +02:00
s3c24xx_register_clocks ( clk_cdev , ARRAY_SIZE ( clk_cdev ) ) ;
for ( cnt = 0 ; cnt < ARRAY_SIZE ( clk_cdev ) ; cnt + + )
s3c_disable_clocks ( clk_cdev [ cnt ] , 1 ) ;
2010-01-26 15:10:38 +09:00
s3c24xx_register_clocks ( clks1 , ARRAY_SIZE ( clks1 ) ) ;
s3c_register_clksrc ( clksrcs , ARRAY_SIZE ( clksrcs ) ) ;
2011-10-24 12:08:42 +02:00
for ( cnt = 0 ; cnt < ARRAY_SIZE ( clksrc_cdev ) ; cnt + + )
s3c_register_clksrc ( clksrc_cdev [ cnt ] , 1 ) ;
clkdev_add_table ( s3c64xx_clk_lookup , ARRAY_SIZE ( s3c64xx_clk_lookup ) ) ;
2008-11-21 10:36:05 +00:00
s3c_pwmclk_init ( ) ;
2008-10-21 14:07:00 +01:00
}