2011-01-04 21:28:14 +01:00
/*
* Atheros AR71XX / AR724X / AR913X common routines
*
2012-03-14 10:45:22 +01:00
* Copyright ( C ) 2010 - 2011 Jaiganesh Narayanan < jnarayanan @ atheros . com >
2011-01-04 21:28:14 +01:00
* Copyright ( C ) 2011 Gabor Juhos < juhosg @ openwrt . org >
*
2012-03-14 10:45:22 +01:00
* Parts of this file are based on Atheros ' 2.6 .15 / 2.6 .31 BSP
*
2011-01-04 21:28:14 +01: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 .
*/
# include <linux/kernel.h>
# include <linux/init.h>
# include <linux/err.h>
# include <linux/clk.h>
2013-08-28 10:41:47 +02:00
# include <linux/clkdev.h>
2015-04-19 14:30:04 +02:00
# include <linux/clk-provider.h>
2016-03-17 06:34:15 +03:00
# include <linux/of.h>
# include <linux/of_address.h>
2016-03-17 06:34:14 +03:00
# include <dt-bindings/clock/ath79-clk.h>
2011-01-04 21:28:14 +01:00
2012-09-08 14:02:21 +02:00
# include <asm/div64.h>
2011-01-04 21:28:14 +01:00
# include <asm/mach-ath79/ath79.h>
# include <asm/mach-ath79/ar71xx_regs.h>
# include "common.h"
2016-03-17 06:34:15 +03:00
# include "machtypes.h"
2011-01-04 21:28:14 +01:00
# define AR71XX_BASE_FREQ 40000000
2016-03-17 06:34:09 +03:00
# define AR724X_BASE_FREQ 40000000
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:14 +03:00
static struct clk * clks [ ATH79_CLK_END ] ;
2015-05-31 02:18:22 +02:00
static struct clk_onecell_data clk_data = {
. clks = clks ,
. clk_num = ARRAY_SIZE ( clks ) ,
} ;
static struct clk * __init ath79_add_sys_clkdev (
const char * id , unsigned long rate )
2013-08-28 10:41:47 +02:00
{
struct clk * clk ;
int err ;
2016-04-19 18:33:21 -07:00
clk = clk_register_fixed_rate ( NULL , id , NULL , 0 , rate ) ;
2016-10-30 09:25:46 +01:00
if ( IS_ERR ( clk ) )
2013-08-28 10:41:47 +02:00
panic ( " failed to allocate %s clock structure " , id ) ;
err = clk_register_clkdev ( clk , id , NULL ) ;
if ( err )
panic ( " unable to register %s clock device " , id ) ;
2015-05-31 02:18:22 +02:00
return clk ;
2013-08-28 10:41:47 +02:00
}
2011-01-04 21:28:14 +01:00
static void __init ar71xx_clocks_init ( void )
{
2013-08-28 10:41:46 +02:00
unsigned long ref_rate ;
unsigned long cpu_rate ;
unsigned long ddr_rate ;
unsigned long ahb_rate ;
2011-01-04 21:28:14 +01:00
u32 pll ;
u32 freq ;
u32 div ;
2013-08-28 10:41:46 +02:00
ref_rate = AR71XX_BASE_FREQ ;
2011-01-04 21:28:14 +01:00
pll = ath79_pll_rr ( AR71XX_PLL_REG_CPU_CONFIG ) ;
2015-04-19 14:30:02 +02:00
div = ( ( pll > > AR71XX_PLL_FB_SHIFT ) & AR71XX_PLL_FB_MASK ) + 1 ;
2013-08-28 10:41:46 +02:00
freq = div * ref_rate ;
2011-01-04 21:28:14 +01:00
div = ( ( pll > > AR71XX_CPU_DIV_SHIFT ) & AR71XX_CPU_DIV_MASK ) + 1 ;
2013-08-28 10:41:46 +02:00
cpu_rate = freq / div ;
2011-01-04 21:28:14 +01:00
div = ( ( pll > > AR71XX_DDR_DIV_SHIFT ) & AR71XX_DDR_DIV_MASK ) + 1 ;
2013-08-28 10:41:46 +02:00
ddr_rate = freq / div ;
2011-01-04 21:28:14 +01:00
div = ( ( ( pll > > AR71XX_AHB_DIV_SHIFT ) & AR71XX_AHB_DIV_MASK ) + 1 ) * 2 ;
2013-08-28 10:41:46 +02:00
ahb_rate = cpu_rate / div ;
2013-08-28 10:41:47 +02:00
ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
2016-03-17 06:34:14 +03:00
clks [ ATH79_CLK_CPU ] = ath79_add_sys_clkdev ( " cpu " , cpu_rate ) ;
clks [ ATH79_CLK_DDR ] = ath79_add_sys_clkdev ( " ddr " , ddr_rate ) ;
clks [ ATH79_CLK_AHB ] = ath79_add_sys_clkdev ( " ahb " , ahb_rate ) ;
2011-01-04 21:28:14 +01:00
2013-08-28 10:41:47 +02:00
clk_add_alias ( " wdt " , NULL , " ahb " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ahb " , NULL ) ;
2011-01-04 21:28:14 +01:00
}
2016-03-17 06:34:15 +03:00
static struct clk * __init ath79_reg_ffclk ( const char * name ,
const char * parent_name , unsigned int mult , unsigned int div )
2011-01-04 21:28:14 +01:00
{
2016-03-17 06:34:15 +03:00
struct clk * clk ;
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
clk = clk_register_fixed_factor ( NULL , name , parent_name , 0 , mult , div ) ;
2016-08-12 08:36:54 +05:30
if ( IS_ERR ( clk ) )
2016-03-17 06:34:15 +03:00
panic ( " failed to allocate %s clock structure " , name ) ;
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
return clk ;
}
static void __init ar724x_clk_init ( struct clk * ref_clk , void __iomem * pll_base )
{
u32 pll ;
u32 mult , div , ddr_div , ahb_div ;
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
pll = __raw_readl ( pll_base + AR724X_PLL_REG_CPU_CONFIG ) ;
mult = ( ( pll > > AR724X_PLL_FB_SHIFT ) & AR724X_PLL_FB_MASK ) ;
2016-03-17 06:34:09 +03:00
div = ( ( pll > > AR724X_PLL_REF_DIV_SHIFT ) & AR724X_PLL_REF_DIV_MASK ) * 2 ;
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
ddr_div = ( ( pll > > AR724X_DDR_DIV_SHIFT ) & AR724X_DDR_DIV_MASK ) + 1 ;
ahb_div = ( ( ( pll > > AR724X_AHB_DIV_SHIFT ) & AR724X_AHB_DIV_MASK ) + 1 ) * 2 ;
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
clks [ ATH79_CLK_CPU ] = ath79_reg_ffclk ( " cpu " , " ref " , mult , div ) ;
clks [ ATH79_CLK_DDR ] = ath79_reg_ffclk ( " ddr " , " ref " , mult , div * ddr_div ) ;
clks [ ATH79_CLK_AHB ] = ath79_reg_ffclk ( " ahb " , " ref " , mult , div * ahb_div ) ;
}
2011-01-04 21:28:14 +01:00
2016-03-17 06:34:15 +03:00
static void __init ar724x_clocks_init ( void )
{
struct clk * ref_clk ;
2013-08-28 10:41:46 +02:00
2016-03-17 06:34:15 +03:00
ref_clk = ath79_add_sys_clkdev ( " ref " , AR724X_BASE_FREQ ) ;
ar724x_clk_init ( ref_clk , ath79_pll_base ) ;
/* just make happy plat_time_init() from arch/mips/ath79/setup.c */
clk_register_clkdev ( clks [ ATH79_CLK_CPU ] , " cpu " , NULL ) ;
clk_register_clkdev ( clks [ ATH79_CLK_DDR ] , " ddr " , NULL ) ;
clk_register_clkdev ( clks [ ATH79_CLK_AHB ] , " ahb " , NULL ) ;
2011-01-04 21:28:14 +01:00
2013-08-28 10:41:47 +02:00
clk_add_alias ( " wdt " , NULL , " ahb " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ahb " , NULL ) ;
2011-01-04 21:28:14 +01:00
}
2016-03-17 06:34:18 +03:00
static void __init ar9330_clk_init ( struct clk * ref_clk , void __iomem * pll_base )
2011-06-20 21:26:04 +02:00
{
u32 clock_ctrl ;
2016-03-17 06:34:18 +03:00
u32 ref_div ;
u32 ninit_mul ;
u32 out_div ;
2011-06-20 21:26:04 +02:00
2016-03-17 06:34:18 +03:00
u32 cpu_div ;
u32 ddr_div ;
u32 ahb_div ;
2011-06-20 21:26:04 +02:00
2016-03-17 06:34:18 +03:00
clock_ctrl = __raw_readl ( pll_base + AR933X_PLL_CLOCK_CTRL_REG ) ;
2011-06-20 21:26:04 +02:00
if ( clock_ctrl & AR933X_PLL_CLOCK_CTRL_BYPASS ) {
2016-03-17 06:34:18 +03:00
ref_div = 1 ;
ninit_mul = 1 ;
out_div = 1 ;
cpu_div = 1 ;
ddr_div = 1 ;
ahb_div = 1 ;
2011-06-20 21:26:04 +02:00
} else {
2016-03-17 06:34:18 +03:00
u32 cpu_config ;
u32 t ;
cpu_config = __raw_readl ( pll_base + AR933X_PLL_CPU_CONFIG_REG ) ;
2011-06-20 21:26:04 +02:00
t = ( cpu_config > > AR933X_PLL_CPU_CONFIG_REFDIV_SHIFT ) &
AR933X_PLL_CPU_CONFIG_REFDIV_MASK ;
2016-03-17 06:34:18 +03:00
ref_div = t ;
2011-06-20 21:26:04 +02:00
2016-03-17 06:34:18 +03:00
ninit_mul = ( cpu_config > > AR933X_PLL_CPU_CONFIG_NINT_SHIFT ) &
2011-06-20 21:26:04 +02:00
AR933X_PLL_CPU_CONFIG_NINT_MASK ;
t = ( cpu_config > > AR933X_PLL_CPU_CONFIG_OUTDIV_SHIFT ) &
AR933X_PLL_CPU_CONFIG_OUTDIV_MASK ;
if ( t = = 0 )
t = 1 ;
2016-03-17 06:34:18 +03:00
out_div = ( 1 < < t ) ;
2011-06-20 21:26:04 +02:00
2016-03-17 06:34:18 +03:00
cpu_div = ( ( clock_ctrl > > AR933X_PLL_CLOCK_CTRL_CPU_DIV_SHIFT ) &
2011-06-20 21:26:04 +02:00
AR933X_PLL_CLOCK_CTRL_CPU_DIV_MASK ) + 1 ;
2016-03-17 06:34:18 +03:00
ddr_div = ( ( clock_ctrl > > AR933X_PLL_CLOCK_CTRL_DDR_DIV_SHIFT ) &
2011-06-20 21:26:04 +02:00
AR933X_PLL_CLOCK_CTRL_DDR_DIV_MASK ) + 1 ;
2016-03-17 06:34:18 +03:00
ahb_div = ( ( clock_ctrl > > AR933X_PLL_CLOCK_CTRL_AHB_DIV_SHIFT ) &
2011-06-20 21:26:04 +02:00
AR933X_PLL_CLOCK_CTRL_AHB_DIV_MASK ) + 1 ;
}
2016-03-17 06:34:18 +03:00
clks [ ATH79_CLK_CPU ] = ath79_reg_ffclk ( " cpu " , " ref " ,
ninit_mul , ref_div * out_div * cpu_div ) ;
clks [ ATH79_CLK_DDR ] = ath79_reg_ffclk ( " ddr " , " ref " ,
ninit_mul , ref_div * out_div * ddr_div ) ;
clks [ ATH79_CLK_AHB ] = ath79_reg_ffclk ( " ahb " , " ref " ,
ninit_mul , ref_div * out_div * ahb_div ) ;
}
static void __init ar933x_clocks_init ( void )
{
struct clk * ref_clk ;
unsigned long ref_rate ;
u32 t ;
t = ath79_reset_rr ( AR933X_RESET_REG_BOOTSTRAP ) ;
if ( t & AR933X_BOOTSTRAP_REF_CLK_40 )
ref_rate = ( 40 * 1000 * 1000 ) ;
else
ref_rate = ( 25 * 1000 * 1000 ) ;
ref_clk = ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
ar9330_clk_init ( ref_clk , ath79_pll_base ) ;
/* just make happy plat_time_init() from arch/mips/ath79/setup.c */
clk_register_clkdev ( clks [ ATH79_CLK_CPU ] , " cpu " , NULL ) ;
clk_register_clkdev ( clks [ ATH79_CLK_DDR ] , " ddr " , NULL ) ;
clk_register_clkdev ( clks [ ATH79_CLK_AHB ] , " ahb " , NULL ) ;
2013-08-28 10:41:46 +02:00
2013-08-28 10:41:47 +02:00
clk_add_alias ( " wdt " , NULL , " ahb " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ref " , NULL ) ;
2011-06-20 21:26:04 +02:00
}
2012-09-08 14:02:21 +02:00
static u32 __init ar934x_get_pll_freq ( u32 ref , u32 ref_div , u32 nint , u32 nfrac ,
u32 frac , u32 out_div )
{
u64 t ;
u32 ret ;
2013-08-28 10:41:43 +02:00
t = ref ;
2012-09-08 14:02:21 +02:00
t * = nint ;
do_div ( t , ref_div ) ;
ret = t ;
2013-08-28 10:41:43 +02:00
t = ref ;
2012-09-08 14:02:21 +02:00
t * = nfrac ;
do_div ( t , ref_div * frac ) ;
ret + = t ;
ret / = ( 1 < < out_div ) ;
return ret ;
}
2012-03-14 10:45:22 +01:00
static void __init ar934x_clocks_init ( void )
{
2013-08-28 10:41:46 +02:00
unsigned long ref_rate ;
unsigned long cpu_rate ;
unsigned long ddr_rate ;
unsigned long ahb_rate ;
2012-09-08 14:02:21 +02:00
u32 pll , out_div , ref_div , nint , nfrac , frac , clk_ctrl , postdiv ;
2012-03-14 10:45:22 +01:00
u32 cpu_pll , ddr_pll ;
u32 bootstrap ;
2012-09-08 14:02:21 +02:00
void __iomem * dpll_base ;
dpll_base = ioremap ( AR934X_SRIF_BASE , AR934X_SRIF_SIZE ) ;
2012-03-14 10:45:22 +01:00
bootstrap = ath79_reset_rr ( AR934X_RESET_REG_BOOTSTRAP ) ;
2013-01-22 12:59:30 +01:00
if ( bootstrap & AR934X_BOOTSTRAP_REF_CLK_40 )
2013-08-28 10:41:46 +02:00
ref_rate = 40 * 1000 * 1000 ;
2012-03-14 10:45:22 +01:00
else
2013-08-28 10:41:46 +02:00
ref_rate = 25 * 1000 * 1000 ;
2012-03-14 10:45:22 +01:00
2012-09-08 14:02:21 +02:00
pll = __raw_readl ( dpll_base + AR934X_SRIF_CPU_DPLL2_REG ) ;
if ( pll & AR934X_SRIF_DPLL2_LOCAL_PLL ) {
out_div = ( pll > > AR934X_SRIF_DPLL2_OUTDIV_SHIFT ) &
AR934X_SRIF_DPLL2_OUTDIV_MASK ;
pll = __raw_readl ( dpll_base + AR934X_SRIF_CPU_DPLL1_REG ) ;
nint = ( pll > > AR934X_SRIF_DPLL1_NINT_SHIFT ) &
AR934X_SRIF_DPLL1_NINT_MASK ;
nfrac = pll & AR934X_SRIF_DPLL1_NFRAC_MASK ;
ref_div = ( pll > > AR934X_SRIF_DPLL1_REFDIV_SHIFT ) &
AR934X_SRIF_DPLL1_REFDIV_MASK ;
frac = 1 < < 18 ;
} else {
pll = ath79_pll_rr ( AR934X_PLL_CPU_CONFIG_REG ) ;
out_div = ( pll > > AR934X_PLL_CPU_CONFIG_OUTDIV_SHIFT ) &
AR934X_PLL_CPU_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > AR934X_PLL_CPU_CONFIG_REFDIV_SHIFT ) &
AR934X_PLL_CPU_CONFIG_REFDIV_MASK ;
nint = ( pll > > AR934X_PLL_CPU_CONFIG_NINT_SHIFT ) &
AR934X_PLL_CPU_CONFIG_NINT_MASK ;
nfrac = ( pll > > AR934X_PLL_CPU_CONFIG_NFRAC_SHIFT ) &
AR934X_PLL_CPU_CONFIG_NFRAC_MASK ;
frac = 1 < < 6 ;
}
2013-08-28 10:41:46 +02:00
cpu_pll = ar934x_get_pll_freq ( ref_rate , ref_div , nint ,
2012-09-08 14:02:21 +02:00
nfrac , frac , out_div ) ;
pll = __raw_readl ( dpll_base + AR934X_SRIF_DDR_DPLL2_REG ) ;
if ( pll & AR934X_SRIF_DPLL2_LOCAL_PLL ) {
out_div = ( pll > > AR934X_SRIF_DPLL2_OUTDIV_SHIFT ) &
AR934X_SRIF_DPLL2_OUTDIV_MASK ;
pll = __raw_readl ( dpll_base + AR934X_SRIF_DDR_DPLL1_REG ) ;
nint = ( pll > > AR934X_SRIF_DPLL1_NINT_SHIFT ) &
AR934X_SRIF_DPLL1_NINT_MASK ;
nfrac = pll & AR934X_SRIF_DPLL1_NFRAC_MASK ;
ref_div = ( pll > > AR934X_SRIF_DPLL1_REFDIV_SHIFT ) &
AR934X_SRIF_DPLL1_REFDIV_MASK ;
frac = 1 < < 18 ;
} else {
pll = ath79_pll_rr ( AR934X_PLL_DDR_CONFIG_REG ) ;
out_div = ( pll > > AR934X_PLL_DDR_CONFIG_OUTDIV_SHIFT ) &
AR934X_PLL_DDR_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > AR934X_PLL_DDR_CONFIG_REFDIV_SHIFT ) &
AR934X_PLL_DDR_CONFIG_REFDIV_MASK ;
nint = ( pll > > AR934X_PLL_DDR_CONFIG_NINT_SHIFT ) &
AR934X_PLL_DDR_CONFIG_NINT_MASK ;
nfrac = ( pll > > AR934X_PLL_DDR_CONFIG_NFRAC_SHIFT ) &
AR934X_PLL_DDR_CONFIG_NFRAC_MASK ;
frac = 1 < < 10 ;
}
2013-08-28 10:41:46 +02:00
ddr_pll = ar934x_get_pll_freq ( ref_rate , ref_div , nint ,
2012-09-08 14:02:21 +02:00
nfrac , frac , out_div ) ;
2012-03-14 10:45:22 +01:00
clk_ctrl = ath79_pll_rr ( AR934X_PLL_CPU_DDR_CLK_CTRL_REG ) ;
postdiv = ( clk_ctrl > > AR934X_PLL_CPU_DDR_CLK_CTRL_CPU_POST_DIV_SHIFT ) &
AR934X_PLL_CPU_DDR_CLK_CTRL_CPU_POST_DIV_MASK ;
if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_CPU_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
cpu_rate = ref_rate ;
2012-03-14 10:45:22 +01:00
else if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_CPUCLK_FROM_CPUPLL )
2013-08-28 10:41:46 +02:00
cpu_rate = cpu_pll / ( postdiv + 1 ) ;
2012-03-14 10:45:22 +01:00
else
2013-08-28 10:41:46 +02:00
cpu_rate = ddr_pll / ( postdiv + 1 ) ;
2012-03-14 10:45:22 +01:00
postdiv = ( clk_ctrl > > AR934X_PLL_CPU_DDR_CLK_CTRL_DDR_POST_DIV_SHIFT ) &
AR934X_PLL_CPU_DDR_CLK_CTRL_DDR_POST_DIV_MASK ;
if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_DDR_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
ddr_rate = ref_rate ;
2012-03-14 10:45:22 +01:00
else if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_DDRCLK_FROM_DDRPLL )
2013-08-28 10:41:46 +02:00
ddr_rate = ddr_pll / ( postdiv + 1 ) ;
2012-03-14 10:45:22 +01:00
else
2013-08-28 10:41:46 +02:00
ddr_rate = cpu_pll / ( postdiv + 1 ) ;
2012-03-14 10:45:22 +01:00
postdiv = ( clk_ctrl > > AR934X_PLL_CPU_DDR_CLK_CTRL_AHB_POST_DIV_SHIFT ) &
AR934X_PLL_CPU_DDR_CLK_CTRL_AHB_POST_DIV_MASK ;
if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_AHB_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
ahb_rate = ref_rate ;
2012-03-14 10:45:22 +01:00
else if ( clk_ctrl & AR934X_PLL_CPU_DDR_CLK_CTRL_AHBCLK_FROM_DDRPLL )
2013-08-28 10:41:46 +02:00
ahb_rate = ddr_pll / ( postdiv + 1 ) ;
2012-03-14 10:45:22 +01:00
else
2013-08-28 10:41:46 +02:00
ahb_rate = cpu_pll / ( postdiv + 1 ) ;
2013-08-28 10:41:47 +02:00
ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
2016-03-17 06:34:14 +03:00
clks [ ATH79_CLK_CPU ] = ath79_add_sys_clkdev ( " cpu " , cpu_rate ) ;
clks [ ATH79_CLK_DDR ] = ath79_add_sys_clkdev ( " ddr " , ddr_rate ) ;
clks [ ATH79_CLK_AHB ] = ath79_add_sys_clkdev ( " ahb " , ahb_rate ) ;
2012-03-14 10:45:22 +01:00
2013-08-28 10:41:47 +02:00
clk_add_alias ( " wdt " , NULL , " ref " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ref " , NULL ) ;
2012-09-08 14:02:21 +02:00
iounmap ( dpll_base ) ;
2012-03-14 10:45:22 +01:00
}
2018-07-20 13:58:19 +02:00
static void __init qca953x_clocks_init ( void )
{
unsigned long ref_rate ;
unsigned long cpu_rate ;
unsigned long ddr_rate ;
unsigned long ahb_rate ;
u32 pll , out_div , ref_div , nint , frac , clk_ctrl , postdiv ;
u32 cpu_pll , ddr_pll ;
u32 bootstrap ;
bootstrap = ath79_reset_rr ( QCA953X_RESET_REG_BOOTSTRAP ) ;
if ( bootstrap & QCA953X_BOOTSTRAP_REF_CLK_40 )
ref_rate = 40 * 1000 * 1000 ;
else
ref_rate = 25 * 1000 * 1000 ;
pll = ath79_pll_rr ( QCA953X_PLL_CPU_CONFIG_REG ) ;
out_div = ( pll > > QCA953X_PLL_CPU_CONFIG_OUTDIV_SHIFT ) &
QCA953X_PLL_CPU_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA953X_PLL_CPU_CONFIG_REFDIV_SHIFT ) &
QCA953X_PLL_CPU_CONFIG_REFDIV_MASK ;
nint = ( pll > > QCA953X_PLL_CPU_CONFIG_NINT_SHIFT ) &
QCA953X_PLL_CPU_CONFIG_NINT_MASK ;
frac = ( pll > > QCA953X_PLL_CPU_CONFIG_NFRAC_SHIFT ) &
QCA953X_PLL_CPU_CONFIG_NFRAC_MASK ;
cpu_pll = nint * ref_rate / ref_div ;
cpu_pll + = frac * ( ref_rate > > 6 ) / ref_div ;
cpu_pll / = ( 1 < < out_div ) ;
pll = ath79_pll_rr ( QCA953X_PLL_DDR_CONFIG_REG ) ;
out_div = ( pll > > QCA953X_PLL_DDR_CONFIG_OUTDIV_SHIFT ) &
QCA953X_PLL_DDR_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA953X_PLL_DDR_CONFIG_REFDIV_SHIFT ) &
QCA953X_PLL_DDR_CONFIG_REFDIV_MASK ;
nint = ( pll > > QCA953X_PLL_DDR_CONFIG_NINT_SHIFT ) &
QCA953X_PLL_DDR_CONFIG_NINT_MASK ;
frac = ( pll > > QCA953X_PLL_DDR_CONFIG_NFRAC_SHIFT ) &
QCA953X_PLL_DDR_CONFIG_NFRAC_MASK ;
ddr_pll = nint * ref_rate / ref_div ;
ddr_pll + = frac * ( ref_rate > > 6 ) / ( ref_div < < 4 ) ;
ddr_pll / = ( 1 < < out_div ) ;
clk_ctrl = ath79_pll_rr ( QCA953X_PLL_CLK_CTRL_REG ) ;
postdiv = ( clk_ctrl > > QCA953X_PLL_CLK_CTRL_CPU_POST_DIV_SHIFT ) &
QCA953X_PLL_CLK_CTRL_CPU_POST_DIV_MASK ;
if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_CPU_PLL_BYPASS )
cpu_rate = ref_rate ;
else if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_CPUCLK_FROM_CPUPLL )
cpu_rate = cpu_pll / ( postdiv + 1 ) ;
else
cpu_rate = ddr_pll / ( postdiv + 1 ) ;
postdiv = ( clk_ctrl > > QCA953X_PLL_CLK_CTRL_DDR_POST_DIV_SHIFT ) &
QCA953X_PLL_CLK_CTRL_DDR_POST_DIV_MASK ;
if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_DDR_PLL_BYPASS )
ddr_rate = ref_rate ;
else if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_DDRCLK_FROM_DDRPLL )
ddr_rate = ddr_pll / ( postdiv + 1 ) ;
else
ddr_rate = cpu_pll / ( postdiv + 1 ) ;
postdiv = ( clk_ctrl > > QCA953X_PLL_CLK_CTRL_AHB_POST_DIV_SHIFT ) &
QCA953X_PLL_CLK_CTRL_AHB_POST_DIV_MASK ;
if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_AHB_PLL_BYPASS )
ahb_rate = ref_rate ;
else if ( clk_ctrl & QCA953X_PLL_CLK_CTRL_AHBCLK_FROM_DDRPLL )
ahb_rate = ddr_pll / ( postdiv + 1 ) ;
else
ahb_rate = cpu_pll / ( postdiv + 1 ) ;
ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
ath79_add_sys_clkdev ( " cpu " , cpu_rate ) ;
ath79_add_sys_clkdev ( " ddr " , ddr_rate ) ;
ath79_add_sys_clkdev ( " ahb " , ahb_rate ) ;
clk_add_alias ( " wdt " , NULL , " ref " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ref " , NULL ) ;
}
2013-02-15 13:38:17 +00:00
static void __init qca955x_clocks_init ( void )
{
2013-08-28 10:41:46 +02:00
unsigned long ref_rate ;
unsigned long cpu_rate ;
unsigned long ddr_rate ;
unsigned long ahb_rate ;
2013-02-15 13:38:17 +00:00
u32 pll , out_div , ref_div , nint , frac , clk_ctrl , postdiv ;
u32 cpu_pll , ddr_pll ;
u32 bootstrap ;
bootstrap = ath79_reset_rr ( QCA955X_RESET_REG_BOOTSTRAP ) ;
if ( bootstrap & QCA955X_BOOTSTRAP_REF_CLK_40 )
2013-08-28 10:41:46 +02:00
ref_rate = 40 * 1000 * 1000 ;
2013-02-15 13:38:17 +00:00
else
2013-08-28 10:41:46 +02:00
ref_rate = 25 * 1000 * 1000 ;
2013-02-15 13:38:17 +00:00
pll = ath79_pll_rr ( QCA955X_PLL_CPU_CONFIG_REG ) ;
out_div = ( pll > > QCA955X_PLL_CPU_CONFIG_OUTDIV_SHIFT ) &
QCA955X_PLL_CPU_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA955X_PLL_CPU_CONFIG_REFDIV_SHIFT ) &
QCA955X_PLL_CPU_CONFIG_REFDIV_MASK ;
nint = ( pll > > QCA955X_PLL_CPU_CONFIG_NINT_SHIFT ) &
QCA955X_PLL_CPU_CONFIG_NINT_MASK ;
frac = ( pll > > QCA955X_PLL_CPU_CONFIG_NFRAC_SHIFT ) &
QCA955X_PLL_CPU_CONFIG_NFRAC_MASK ;
2013-08-28 10:41:46 +02:00
cpu_pll = nint * ref_rate / ref_div ;
cpu_pll + = frac * ref_rate / ( ref_div * ( 1 < < 6 ) ) ;
2013-02-15 13:38:17 +00:00
cpu_pll / = ( 1 < < out_div ) ;
pll = ath79_pll_rr ( QCA955X_PLL_DDR_CONFIG_REG ) ;
out_div = ( pll > > QCA955X_PLL_DDR_CONFIG_OUTDIV_SHIFT ) &
QCA955X_PLL_DDR_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA955X_PLL_DDR_CONFIG_REFDIV_SHIFT ) &
QCA955X_PLL_DDR_CONFIG_REFDIV_MASK ;
nint = ( pll > > QCA955X_PLL_DDR_CONFIG_NINT_SHIFT ) &
QCA955X_PLL_DDR_CONFIG_NINT_MASK ;
frac = ( pll > > QCA955X_PLL_DDR_CONFIG_NFRAC_SHIFT ) &
QCA955X_PLL_DDR_CONFIG_NFRAC_MASK ;
2013-08-28 10:41:46 +02:00
ddr_pll = nint * ref_rate / ref_div ;
ddr_pll + = frac * ref_rate / ( ref_div * ( 1 < < 10 ) ) ;
2013-02-15 13:38:17 +00:00
ddr_pll / = ( 1 < < out_div ) ;
clk_ctrl = ath79_pll_rr ( QCA955X_PLL_CLK_CTRL_REG ) ;
postdiv = ( clk_ctrl > > QCA955X_PLL_CLK_CTRL_CPU_POST_DIV_SHIFT ) &
QCA955X_PLL_CLK_CTRL_CPU_POST_DIV_MASK ;
if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_CPU_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
cpu_rate = ref_rate ;
2013-02-15 13:38:17 +00:00
else if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_CPUCLK_FROM_CPUPLL )
2013-08-28 10:41:46 +02:00
cpu_rate = ddr_pll / ( postdiv + 1 ) ;
2013-02-15 13:38:17 +00:00
else
2013-08-28 10:41:46 +02:00
cpu_rate = cpu_pll / ( postdiv + 1 ) ;
2013-02-15 13:38:17 +00:00
postdiv = ( clk_ctrl > > QCA955X_PLL_CLK_CTRL_DDR_POST_DIV_SHIFT ) &
QCA955X_PLL_CLK_CTRL_DDR_POST_DIV_MASK ;
if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_DDR_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
ddr_rate = ref_rate ;
2013-02-15 13:38:17 +00:00
else if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_DDRCLK_FROM_DDRPLL )
2013-08-28 10:41:46 +02:00
ddr_rate = cpu_pll / ( postdiv + 1 ) ;
2013-02-15 13:38:17 +00:00
else
2013-08-28 10:41:46 +02:00
ddr_rate = ddr_pll / ( postdiv + 1 ) ;
2013-02-15 13:38:17 +00:00
postdiv = ( clk_ctrl > > QCA955X_PLL_CLK_CTRL_AHB_POST_DIV_SHIFT ) &
QCA955X_PLL_CLK_CTRL_AHB_POST_DIV_MASK ;
if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_AHB_PLL_BYPASS )
2013-08-28 10:41:46 +02:00
ahb_rate = ref_rate ;
2013-02-15 13:38:17 +00:00
else if ( clk_ctrl & QCA955X_PLL_CLK_CTRL_AHBCLK_FROM_DDRPLL )
2013-08-28 10:41:46 +02:00
ahb_rate = ddr_pll / ( postdiv + 1 ) ;
2013-02-15 13:38:17 +00:00
else
2013-08-28 10:41:46 +02:00
ahb_rate = cpu_pll / ( postdiv + 1 ) ;
2013-08-28 10:41:47 +02:00
ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
2016-03-17 06:34:14 +03:00
clks [ ATH79_CLK_CPU ] = ath79_add_sys_clkdev ( " cpu " , cpu_rate ) ;
clks [ ATH79_CLK_DDR ] = ath79_add_sys_clkdev ( " ddr " , ddr_rate ) ;
clks [ ATH79_CLK_AHB ] = ath79_add_sys_clkdev ( " ahb " , ahb_rate ) ;
2013-02-15 13:38:17 +00:00
2013-08-28 10:41:47 +02:00
clk_add_alias ( " wdt " , NULL , " ref " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ref " , NULL ) ;
2013-02-15 13:38:17 +00:00
}
2018-07-20 13:58:19 +02:00
static void __init qca956x_clocks_init ( void )
{
unsigned long ref_rate ;
unsigned long cpu_rate ;
unsigned long ddr_rate ;
unsigned long ahb_rate ;
u32 pll , out_div , ref_div , nint , hfrac , lfrac , clk_ctrl , postdiv ;
u32 cpu_pll , ddr_pll ;
u32 bootstrap ;
/*
* QCA956x timer init workaround has to be applied right before setting
* up the clock . Else , there will be no jiffies
*/
u32 misc ;
misc = ath79_reset_rr ( AR71XX_RESET_REG_MISC_INT_ENABLE ) ;
misc | = MISC_INT_MIPS_SI_TIMERINT_MASK ;
ath79_reset_wr ( AR71XX_RESET_REG_MISC_INT_ENABLE , misc ) ;
bootstrap = ath79_reset_rr ( QCA956X_RESET_REG_BOOTSTRAP ) ;
if ( bootstrap & QCA956X_BOOTSTRAP_REF_CLK_40 )
ref_rate = 40 * 1000 * 1000 ;
else
ref_rate = 25 * 1000 * 1000 ;
pll = ath79_pll_rr ( QCA956X_PLL_CPU_CONFIG_REG ) ;
out_div = ( pll > > QCA956X_PLL_CPU_CONFIG_OUTDIV_SHIFT ) &
QCA956X_PLL_CPU_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA956X_PLL_CPU_CONFIG_REFDIV_SHIFT ) &
QCA956X_PLL_CPU_CONFIG_REFDIV_MASK ;
pll = ath79_pll_rr ( QCA956X_PLL_CPU_CONFIG1_REG ) ;
nint = ( pll > > QCA956X_PLL_CPU_CONFIG1_NINT_SHIFT ) &
QCA956X_PLL_CPU_CONFIG1_NINT_MASK ;
hfrac = ( pll > > QCA956X_PLL_CPU_CONFIG1_NFRAC_H_SHIFT ) &
QCA956X_PLL_CPU_CONFIG1_NFRAC_H_MASK ;
lfrac = ( pll > > QCA956X_PLL_CPU_CONFIG1_NFRAC_L_SHIFT ) &
QCA956X_PLL_CPU_CONFIG1_NFRAC_L_MASK ;
cpu_pll = nint * ref_rate / ref_div ;
cpu_pll + = ( lfrac * ref_rate ) / ( ( ref_div * 25 ) < < 13 ) ;
cpu_pll + = ( hfrac > > 13 ) * ref_rate / ref_div ;
cpu_pll / = ( 1 < < out_div ) ;
pll = ath79_pll_rr ( QCA956X_PLL_DDR_CONFIG_REG ) ;
out_div = ( pll > > QCA956X_PLL_DDR_CONFIG_OUTDIV_SHIFT ) &
QCA956X_PLL_DDR_CONFIG_OUTDIV_MASK ;
ref_div = ( pll > > QCA956X_PLL_DDR_CONFIG_REFDIV_SHIFT ) &
QCA956X_PLL_DDR_CONFIG_REFDIV_MASK ;
pll = ath79_pll_rr ( QCA956X_PLL_DDR_CONFIG1_REG ) ;
nint = ( pll > > QCA956X_PLL_DDR_CONFIG1_NINT_SHIFT ) &
QCA956X_PLL_DDR_CONFIG1_NINT_MASK ;
hfrac = ( pll > > QCA956X_PLL_DDR_CONFIG1_NFRAC_H_SHIFT ) &
QCA956X_PLL_DDR_CONFIG1_NFRAC_H_MASK ;
lfrac = ( pll > > QCA956X_PLL_DDR_CONFIG1_NFRAC_L_SHIFT ) &
QCA956X_PLL_DDR_CONFIG1_NFRAC_L_MASK ;
ddr_pll = nint * ref_rate / ref_div ;
ddr_pll + = ( lfrac * ref_rate ) / ( ( ref_div * 25 ) < < 13 ) ;
ddr_pll + = ( hfrac > > 13 ) * ref_rate / ref_div ;
ddr_pll / = ( 1 < < out_div ) ;
clk_ctrl = ath79_pll_rr ( QCA956X_PLL_CLK_CTRL_REG ) ;
postdiv = ( clk_ctrl > > QCA956X_PLL_CLK_CTRL_CPU_POST_DIV_SHIFT ) &
QCA956X_PLL_CLK_CTRL_CPU_POST_DIV_MASK ;
if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_CPU_PLL_BYPASS )
cpu_rate = ref_rate ;
else if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_CPU_DDRCLK_FROM_CPUPLL )
cpu_rate = ddr_pll / ( postdiv + 1 ) ;
else
cpu_rate = cpu_pll / ( postdiv + 1 ) ;
postdiv = ( clk_ctrl > > QCA956X_PLL_CLK_CTRL_DDR_POST_DIV_SHIFT ) &
QCA956X_PLL_CLK_CTRL_DDR_POST_DIV_MASK ;
if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_DDR_PLL_BYPASS )
ddr_rate = ref_rate ;
else if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_CPU_DDRCLK_FROM_DDRPLL )
ddr_rate = cpu_pll / ( postdiv + 1 ) ;
else
ddr_rate = ddr_pll / ( postdiv + 1 ) ;
postdiv = ( clk_ctrl > > QCA956X_PLL_CLK_CTRL_AHB_POST_DIV_SHIFT ) &
QCA956X_PLL_CLK_CTRL_AHB_POST_DIV_MASK ;
if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_AHB_PLL_BYPASS )
ahb_rate = ref_rate ;
else if ( clk_ctrl & QCA956X_PLL_CLK_CTRL_AHBCLK_FROM_DDRPLL )
ahb_rate = ddr_pll / ( postdiv + 1 ) ;
else
ahb_rate = cpu_pll / ( postdiv + 1 ) ;
ath79_add_sys_clkdev ( " ref " , ref_rate ) ;
ath79_add_sys_clkdev ( " cpu " , cpu_rate ) ;
ath79_add_sys_clkdev ( " ddr " , ddr_rate ) ;
ath79_add_sys_clkdev ( " ahb " , ahb_rate ) ;
clk_add_alias ( " wdt " , NULL , " ref " , NULL ) ;
clk_add_alias ( " uart " , NULL , " ref " , NULL ) ;
}
2011-01-04 21:28:14 +01:00
void __init ath79_clocks_init ( void )
{
if ( soc_is_ar71xx ( ) )
ar71xx_clocks_init ( ) ;
2016-03-17 06:34:10 +03:00
else if ( soc_is_ar724x ( ) | | soc_is_ar913x ( ) )
2011-01-04 21:28:14 +01:00
ar724x_clocks_init ( ) ;
2011-06-20 21:26:04 +02:00
else if ( soc_is_ar933x ( ) )
ar933x_clocks_init ( ) ;
2012-03-14 10:45:22 +01:00
else if ( soc_is_ar934x ( ) )
ar934x_clocks_init ( ) ;
2018-07-20 13:58:19 +02:00
else if ( soc_is_qca953x ( ) )
qca953x_clocks_init ( ) ;
2013-02-15 13:38:17 +00:00
else if ( soc_is_qca955x ( ) )
qca955x_clocks_init ( ) ;
2018-07-20 13:58:19 +02:00
else if ( soc_is_qca956x ( ) | | soc_is_tp9343 ( ) )
qca956x_clocks_init ( ) ;
2011-01-04 21:28:14 +01:00
else
BUG ( ) ;
}
2013-08-28 10:41:44 +02:00
unsigned long __init
ath79_get_sys_clk_rate ( const char * id )
{
struct clk * clk ;
unsigned long rate ;
clk = clk_get ( NULL , id ) ;
if ( IS_ERR ( clk ) )
panic ( " unable to get %s clock, err=%d " , id , ( int ) PTR_ERR ( clk ) ) ;
rate = clk_get_rate ( clk ) ;
clk_put ( clk ) ;
return rate ;
}
2015-05-31 02:18:22 +02:00
# ifdef CONFIG_OF
static void __init ath79_clocks_init_dt ( struct device_node * np )
{
of_clk_add_provider ( np , of_clk_src_onecell_get , & clk_data ) ;
}
CLK_OF_DECLARE ( ar7100 , " qca,ar7100-pll " , ath79_clocks_init_dt ) ;
CLK_OF_DECLARE ( ar7240 , " qca,ar7240-pll " , ath79_clocks_init_dt ) ;
CLK_OF_DECLARE ( ar9340 , " qca,ar9340-pll " , ath79_clocks_init_dt ) ;
CLK_OF_DECLARE ( ar9550 , " qca,qca9550-pll " , ath79_clocks_init_dt ) ;
2016-03-17 06:34:15 +03:00
static void __init ath79_clocks_init_dt_ng ( struct device_node * np )
{
struct clk * ref_clk ;
void __iomem * pll_base ;
ref_clk = of_clk_get ( np , 0 ) ;
if ( IS_ERR ( ref_clk ) ) {
2017-07-18 16:42:45 -05:00
pr_err ( " %pOF: of_clk_get failed \n " , np ) ;
2016-03-17 06:34:15 +03:00
goto err ;
}
pll_base = of_iomap ( np , 0 ) ;
if ( ! pll_base ) {
2017-07-18 16:42:45 -05:00
pr_err ( " %pOF: can't map pll registers \n " , np ) ;
2016-03-17 06:34:15 +03:00
goto err_clk ;
}
2016-03-17 06:34:18 +03:00
if ( of_device_is_compatible ( np , " qca,ar9130-pll " ) )
ar724x_clk_init ( ref_clk , pll_base ) ;
else if ( of_device_is_compatible ( np , " qca,ar9330-pll " ) )
ar9330_clk_init ( ref_clk , pll_base ) ;
else {
2017-07-18 16:42:45 -05:00
pr_err ( " %pOF: could not find any appropriate clk_init() \n " , np ) ;
2017-01-02 15:18:21 +05:30
goto err_iounmap ;
2016-03-17 06:34:18 +03:00
}
2016-03-17 06:34:15 +03:00
if ( of_clk_add_provider ( np , of_clk_src_onecell_get , & clk_data ) ) {
2017-07-18 16:42:45 -05:00
pr_err ( " %pOF: could not register clk provider \n " , np ) ;
2017-01-02 15:18:21 +05:30
goto err_iounmap ;
2016-03-17 06:34:15 +03:00
}
return ;
2017-01-02 15:18:21 +05:30
err_iounmap :
iounmap ( pll_base ) ;
2016-03-17 06:34:15 +03:00
err_clk :
clk_put ( ref_clk ) ;
err :
return ;
}
CLK_OF_DECLARE ( ar9130_clk , " qca,ar9130-pll " , ath79_clocks_init_dt_ng ) ;
2016-03-17 06:34:18 +03:00
CLK_OF_DECLARE ( ar9330_clk , " qca,ar9330-pll " , ath79_clocks_init_dt_ng ) ;
2015-05-31 02:18:22 +02:00
# endif