/*
|
* (C) Copyright 2007-2011
|
* Allwinner Technology Co., Ltd. <www.allwinnertech.com>
|
*
|
* SPDX-License-Identifier: GPL-2.0+
|
*/
|
#include <common.h>
|
#include <asm/io.h>
|
#include <asm/arch/cpu.h>
|
#include <asm/arch/clock.h>
|
#include <asm/arch/timer.h>
|
|
|
void clock_init_uart(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
|
/* uart clock source is apb2 */
|
writel(APB2_CLK_SRC_OSC24M|
|
APB2_CLK_RATE_N_1|
|
APB2_CLK_RATE_M(1),
|
&ccm->apb2_cfg);
|
|
/* open the clock for uart */
|
setbits_le32(&ccm->uart_gate_reset,
|
1 << (CONFIG_CONS_INDEX - 1));
|
|
/* deassert uart reset */
|
setbits_le32(&ccm->uart_gate_reset,
|
1 << (RESET_SHIFT + CONFIG_CONS_INDEX - 1));
|
}
|
|
uint clock_get_pll6(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
uint reg_val;
|
uint factor_n, factor_m0,factor_m1, pll6;
|
|
reg_val = readl(&ccm->pll6_cfg);
|
|
factor_n = ((reg_val >> 8) & 0xff) + 1;
|
factor_m0 = ((reg_val >> 0) & 0x01) + 1;
|
factor_m1 = ((reg_val >> 1) & 0x01) + 1;
|
/*default is 1200M, 1X is 600M */
|
pll6 = (24 * factor_n /factor_m0/factor_m1)>>1;
|
|
|
return pll6;
|
}
|
|
static int clk_get_pll_para(struct core_pll_freq_tbl *factor, int pll_clk)
|
{
|
int index;
|
|
index = pll_clk / 24;
|
factor->FactorP = 0;
|
factor->FactorN = (index - 1);
|
factor->FactorM = 0;
|
|
return 0;
|
}
|
|
int clock_set_corepll(int frequency)
|
{
|
|
unsigned int reg_val = 0;
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
struct core_pll_freq_tbl pll_factor;
|
|
if (frequency == clock_get_corepll())
|
return 0;
|
else if (frequency >= 1008)
|
frequency = 1008;
|
|
/* switch to 24M*/
|
reg_val = readl(&ccm->cpu_axi_cfg);
|
reg_val &= ~(0x03 << 24);
|
reg_val |= (0x00 << 24);
|
writel(reg_val, &ccm->cpu_axi_cfg);
|
__udelay(20);
|
|
/*get config para form freq table */
|
clk_get_pll_para(&pll_factor, frequency);
|
|
/* 24M*(N+1)/(M+1)/(P+1)*/
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val &= ~((1 << 31) | (0x03 << 16) | (0xff << 8) | (0x03 << 0));
|
reg_val |= (pll_factor.FactorP << 16) | \
|
(pll_factor.FactorN << 8) | \
|
(pll_factor.FactorM << 0) ;
|
writel(reg_val, &ccm->pll1_cfg);
|
__udelay(20);
|
/* lock enable */
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val |= (1 << 29);
|
writel(reg_val, &ccm->pll1_cfg);
|
|
/* enable pll */
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val |= (1 << 31);
|
writel(reg_val, &ccm->pll1_cfg);
|
|
/*wait PLL_CPUX stable */
|
#ifndef FPGA_PLATFORM
|
while (!(readl(&ccm->pll1_cfg) & (0x1 << 28)))
|
;
|
__usdelay(1);
|
#endif
|
/* lock disable */
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val &= ~(1 << 29);
|
writel(reg_val, &ccm->pll1_cfg);
|
|
/*set and change cpu clk src to PLL_CPUX, PLL_CPUX:AXI0 = 408M:136M */
|
reg_val = readl(&ccm->cpu_axi_cfg);
|
reg_val &= ~(0x03 << 24);
|
reg_val |= (0x03 << 24);
|
writel(reg_val, &ccm->cpu_axi_cfg);
|
|
return 0;
|
}
|
|
uint clock_get_corepll(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val;
|
int div_m, div_p;
|
int factor_n;
|
int clock,clock_src;
|
|
reg_val = readl(&ccm->cpu_axi_cfg);
|
clock_src = (reg_val >> 24) & 0x03;
|
|
switch(clock_src)
|
{
|
case 0://OSC24M
|
clock = 24;
|
break;
|
case 1://RTC32K
|
clock = 32/1000 ;
|
break;
|
case 2://RC16M
|
clock = 16;
|
break;
|
case 3://PLL_CPUX
|
reg_val = readl(&ccm->pll1_cfg);
|
div_p = 1<<((reg_val >>16) & 0x3);
|
factor_n = ((reg_val >> 8) & 0xff) + 1;
|
div_m = ((reg_val >> 0) & 0x3) + 1;
|
|
clock = 24*factor_n/div_m/div_p;
|
break;
|
default:
|
return 0;
|
}
|
return clock;
|
}
|
|
|
uint clock_get_axi(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val = 0;
|
int factor = 0;
|
int clock = 0;
|
|
reg_val = readl(&ccm->cpu_axi_cfg);
|
factor = ((reg_val >> 0) & 0x03) + 1;
|
clock = clock_get_corepll()/factor;
|
|
return clock;
|
}
|
|
|
uint clock_get_ahb(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val = 0;
|
int factor_m = 0,factor_n = 0;
|
int clock = 0;
|
int src = 0,src_clock = 0;
|
|
reg_val = readl(&ccm->psi_ahb1_ahb2_cfg);
|
src = (reg_val >> 24)&0x3;
|
factor_m = ((reg_val >> 0) & 0x03) + 1;
|
factor_n = 1<< ((reg_val >> 8) & 0x03);
|
|
switch(src)
|
{
|
case 0://OSC24M
|
src_clock = 24;
|
break;
|
case 1://CCMU_32K
|
src_clock =32/1000;
|
break;
|
case 2: //RC16M
|
src_clock =16;
|
break;
|
case 3://PLL_PERI0(1X)
|
src_clock = clock_get_pll6();
|
break;
|
default:
|
return 0;
|
}
|
|
clock = src_clock/factor_m/factor_n;
|
|
return clock;
|
}
|
|
|
uint clock_get_apb1(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val = 0;
|
int src = 0,src_clock = 0;
|
int clock = 0, factor_m = 0,factor_n = 0;
|
|
reg_val = readl(&ccm->apb1_cfg);
|
factor_m = ((reg_val >> 0) & 0x03);
|
factor_n = 1<<((reg_val >> 8) & 0x03);
|
src = (reg_val >> 24)&0x3;
|
|
switch(src)
|
{
|
case 0://OSC24M
|
src_clock = 24;
|
break;
|
case 1://CCMU_32K
|
src_clock =32/1000;
|
break;
|
case 2: //PSI
|
src_clock = clock_get_ahb();
|
break;
|
case 3://PLL_PERI0(1X)
|
src_clock = clock_get_pll6();
|
break;
|
default:
|
return 0;
|
}
|
|
clock = src_clock/factor_m/factor_n;
|
|
return clock;
|
}
|
|
uint clock_get_apb2(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val = 0;
|
int clock = 0, factor_m = 0,factor_n = 0;
|
int src = 0,src_clock = 0;
|
|
reg_val = readl(&ccm->apb2_cfg);
|
src = (reg_val >> 24)&0x3;
|
factor_m = ((reg_val >> 0) & 0x03) + 1;
|
factor_n = ((reg_val >> 8) & 0x03) + 1;
|
|
switch(src)
|
{
|
case 0://OSC24M
|
src_clock = 24;
|
break;
|
case 1://CCMU_32K
|
src_clock =32/1000;
|
break;
|
case 2: //PSI
|
src_clock =clock_get_ahb();
|
break;
|
case 3: //PSI
|
src_clock =clock_get_pll6();
|
break;
|
default:
|
return 0;
|
}
|
|
clock = src_clock/factor_m/factor_n;
|
|
return clock;
|
|
}
|
|
|
uint clock_get_mbus(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
unsigned int reg_val;
|
unsigned int src = 0,clock=0, div = 0;
|
reg_val = readl(&ccm->mbus_cfg);
|
|
//get src
|
src = (reg_val >> 24)&0x3;
|
//get div M, the divided clock is divided by M+1
|
div = (reg_val&0x3) + 1;
|
|
switch(src)
|
{
|
case 0://src is OSC24M
|
clock = 24;
|
break;
|
case 1://src is pll_periph0(1x)/2
|
clock = clock_get_pll6()*2;
|
break;
|
case 2://src is pll_ddr0 --not set in boot
|
clock = 0;
|
break;
|
case 3://src is pll_ddr1 --not set in boot
|
clock = 0;
|
break;
|
}
|
|
clock = clock/div;
|
|
return clock;
|
}
|
|
|
|
int usb_open_clock(void)
|
{
|
u32 reg_value = 0;
|
|
//USB0 Clock Reg
|
//bit30: USB PHY0 reset
|
//Bit29: Gating Special Clk for USB PHY0
|
reg_value = readl(SUNXI_CCM_BASE + 0xA70);
|
reg_value |= (1 << 29) | (1 << 30);
|
writel(reg_value, (SUNXI_CCM_BASE + 0xA70));
|
//delay some time
|
__msdelay(1);
|
|
//USB BUS Gating Reset Reg
|
//bit8:USB_OTG Gating
|
reg_value = readl(SUNXI_CCM_BASE + 0xA8C);
|
reg_value |= (1 << 8);
|
writel(reg_value, (SUNXI_CCM_BASE + 0xA8C));
|
|
//delay to wati SIE stable
|
__msdelay(1);
|
|
//USB BUS Gating Reset Reg: USB_OTG reset
|
reg_value = readl(SUNXI_CCM_BASE + 0xA8C);
|
reg_value |= (1 << 24);
|
writel(reg_value, (SUNXI_CCM_BASE + 0xA8C));
|
__msdelay(1);
|
|
reg_value = readl(SUNXI_USBOTG_BASE + 0x420);
|
reg_value |= (0x01 << 0);
|
writel(reg_value, (SUNXI_USBOTG_BASE + 0x420));
|
__msdelay(1);
|
|
return 0;
|
}
|
|
int usb_close_clock(void)
|
{
|
u32 reg_value = 0;
|
|
/* AHB reset */
|
reg_value = readl(SUNXI_CCM_BASE + 0xA8C);
|
reg_value &= ~(1 << 24);
|
writel(reg_value, (SUNXI_CCM_BASE + 0xA8C));
|
__msdelay(1);
|
|
reg_value = readl(SUNXI_CCM_BASE + 0xA8C);
|
reg_value &= ~(1 << 8);
|
writel(reg_value, (SUNXI_CCM_BASE + 0xA8C));
|
__msdelay(1);
|
|
reg_value = readl(SUNXI_CCM_BASE + 0xcc);
|
reg_value &= ~((1 << 29) | (1 << 30));
|
writel(reg_value, (SUNXI_CCM_BASE + 0xcc));
|
__msdelay(1);
|
|
return 0;
|
}
|