/* SPDX-License-Identifier: GPL-2.0+ */
|
/*
|
* (C) Copyright 2015 Hans de Goede <hdegoede@redhat.com>
|
*
|
* Configuration settings for the Allwinner A80 (sun9i) CPU
|
*/
|
|
#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_pll5(void)
|
{
|
struct sunxi_ccm_reg *const ccm =
|
(struct sunxi_ccm_reg *)SUNXI_CCM_BASE;
|
uint reg_val;
|
uint factor_n, factor_m0, factor_m1, pll5;
|
|
reg_val = readl(&ccm->pll5_cfg);
|
|
factor_n = ((reg_val >> 8) & 0xff) + 1;
|
factor_m0 = ((reg_val >> 0) & 0x01) + 1;
|
factor_m1 = ((reg_val >> 1) & 0x01) + 1;
|
/*default DDR0 is 432M */
|
pll5 = (24 * factor_n /factor_m0/factor_m1)>>1;
|
|
|
return pll5;
|
}
|
|
|
|
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;
|
}
|
|
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;
|
}
|
|
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);
|
writel(reg_val, &ccm->cpu_axi_cfg);
|
__udelay(20);
|
|
/*pll output disable*/
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val &= ~(0x01 << 27);
|
writel(reg_val, &ccm->pll1_cfg);
|
|
/*get config para form freq table*/
|
clk_get_pll_para(&pll_factor, frequency);
|
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val &= ~((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);
|
|
/*enable lock*/
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val |= (0x1 << 29);
|
writel(reg_val, &ccm->pll1_cfg);
|
#ifndef FPGA_PLATFORM
|
do {
|
reg_val = readl(&ccm->pll1_cfg);
|
} while (!(reg_val & (0x1 << 28)));
|
#endif
|
|
/*enable pll output*/
|
reg_val = readl(&ccm->pll1_cfg);
|
reg_val |= (0x1 << 27);
|
writel(reg_val, &ccm->pll1_cfg);
|
|
/* switch clk src to COREPLL*/
|
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_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->dram_clk_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 DDR0*/
|
clock = clock_get_pll5();
|
break;
|
case 1: /*src is pll_periph0(1x)/2*/
|
clock = clock_get_pll6() * 2;
|
break;
|
}
|
|
/*mbus clk=dram clk/4*/
|
clock = (clock/div)/4;
|
|
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;
|
}
|