/*
|
* Copyright 2000-2009
|
* Wolfgang Denk, DENX Software Engineering, wd@denx.de.
|
*
|
* 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;
|
pll6 = (24 * factor_n /factor_m0/factor_m1)>>1;
|
|
|
return pll6;
|
}
|
|
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) + 1;
|
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;
|
}
|
|
|
int sunxi_set_sramc_mode(void)
|
{
|
u32 reg_val;
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
/* SRAM Area C 128K Bytes Configuration by AHB ,default map to VE*/
|
reg_val = readl(SUNXI_SRAMC_BASE);
|
reg_val &= ~(0x7FFFFFFF);
|
writel(reg_val, SUNXI_SRAMC_BASE);
|
|
/* VE SRAM:set sram to normal mode, default boot mode */
|
reg_val = readl(SUNXI_SRAMC_BASE + 0X0004);
|
reg_val &= ~(0x1 << 24);
|
writel(reg_val, SUNXI_SRAMC_BASE + 0X0004);
|
|
/* VE gating&VE Bus Reset :brom set them, but not require now */
|
reg_val = readl(&ccm->ve_clk_cfg);
|
reg_val &= ~(0x1 << 0);
|
reg_val &= ~(0x1 << 16);
|
writel(reg_val, &ccm->ve_clk_cfg);
|
|
/* DE gating&DE Bus Reset :brom set them, but not require now */
|
reg_val = readl(&ccm->de_clk_cfg);
|
reg_val &= ~(0x1 << 0);
|
reg_val &= ~(0x1 << 16);
|
writel(reg_val, &ccm->de_clk_cfg);
|
return 0;
|
}
|