mirror of
https://github.com/u-boot/u-boot.git
synced 2025-04-26 15:28:50 +00:00

Kernel stores information to the RTC_SCRATCH0 and RTC_SCRATCH1 registers for wakeup from RTC-only mode with DDR in self-refresh. Parse these registers during SPL boot and jump to the kernel resume vector if the device is waking up from RTC-only modewith DDR in Self-refresh. The RTC scratch register layout used is: SCRATCH0 : bits00-31 : kernel resume address SCRATCH1 : bits00-15 : RTC magic value used to detect valid config SCRATCH1 : bits16-31 : board type information populated by bootloader During the normal boot path the SCRATCH1 : bits16-31 are updated with the eeprom read board type data. In the rtc_only boot path the rtc scratchpad register is read and the board type is determined and correspondingly ddr dpll parameters are set. This is done so as to avoid costly i2c read to eeprom. RTC-only +DRR in self-refresh mode support is currently only enabled for am43xx_evm_rtconly_config. This is not to be used with epos evm builds. Signed-off-by: Tero Kristo <t-kristo@ti.com> [j-keerthy@ti.com Rebased to latest u-boot master branch] Signed-off-by: Keerthy <j-keerthy@ti.com> Reviewed-by: Tom Rini <trini@konsulko.com>
252 lines
5.9 KiB
C
252 lines
5.9 KiB
C
/*
|
|
* clock_am43xx.c
|
|
*
|
|
* clocks for AM43XX based boards
|
|
* Derived from AM33XX based boards
|
|
*
|
|
* Copyright (C) 2013, Texas Instruments, Incorporated - http://www.ti.com/
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <asm/arch/cpu.h>
|
|
#include <asm/arch/clock.h>
|
|
#include <asm/arch/hardware.h>
|
|
#include <asm/arch/sys_proto.h>
|
|
#include <asm/io.h>
|
|
|
|
struct cm_perpll *const cmper = (struct cm_perpll *)CM_PER;
|
|
struct cm_wkuppll *const cmwkup = (struct cm_wkuppll *)CM_WKUP;
|
|
struct cm_dpll *const cmdpll = (struct cm_dpll *)CM_DPLL;
|
|
|
|
const struct dpll_regs dpll_mpu_regs = {
|
|
.cm_clkmode_dpll = CM_WKUP + 0x560,
|
|
.cm_idlest_dpll = CM_WKUP + 0x564,
|
|
.cm_clksel_dpll = CM_WKUP + 0x56c,
|
|
.cm_div_m2_dpll = CM_WKUP + 0x570,
|
|
};
|
|
|
|
const struct dpll_regs dpll_core_regs = {
|
|
.cm_clkmode_dpll = CM_WKUP + 0x520,
|
|
.cm_idlest_dpll = CM_WKUP + 0x524,
|
|
.cm_clksel_dpll = CM_WKUP + 0x52C,
|
|
.cm_div_m4_dpll = CM_WKUP + 0x538,
|
|
.cm_div_m5_dpll = CM_WKUP + 0x53C,
|
|
.cm_div_m6_dpll = CM_WKUP + 0x540,
|
|
};
|
|
|
|
const struct dpll_regs dpll_per_regs = {
|
|
.cm_clkmode_dpll = CM_WKUP + 0x5E0,
|
|
.cm_idlest_dpll = CM_WKUP + 0x5E4,
|
|
.cm_clksel_dpll = CM_WKUP + 0x5EC,
|
|
.cm_div_m2_dpll = CM_WKUP + 0x5F0,
|
|
};
|
|
|
|
const struct dpll_regs dpll_ddr_regs = {
|
|
.cm_clkmode_dpll = CM_WKUP + 0x5A0,
|
|
.cm_idlest_dpll = CM_WKUP + 0x5A4,
|
|
.cm_clksel_dpll = CM_WKUP + 0x5AC,
|
|
.cm_div_m2_dpll = CM_WKUP + 0x5B0,
|
|
.cm_div_m4_dpll = CM_WKUP + 0x5B8,
|
|
};
|
|
|
|
void setup_clocks_for_console(void)
|
|
{
|
|
u32 clkctrl, idlest = MODULE_CLKCTRL_IDLEST_DISABLED;
|
|
|
|
/* Do not add any spl_debug prints in this function */
|
|
clrsetbits_le32(&cmwkup->wkclkstctrl, CD_CLKCTRL_CLKTRCTRL_MASK,
|
|
CD_CLKCTRL_CLKTRCTRL_SW_WKUP <<
|
|
CD_CLKCTRL_CLKTRCTRL_SHIFT);
|
|
|
|
/* Enable UART0 */
|
|
clrsetbits_le32(&cmwkup->wkup_uart0ctrl,
|
|
MODULE_CLKCTRL_MODULEMODE_MASK,
|
|
MODULE_CLKCTRL_MODULEMODE_SW_EXPLICIT_EN <<
|
|
MODULE_CLKCTRL_MODULEMODE_SHIFT);
|
|
|
|
while ((idlest == MODULE_CLKCTRL_IDLEST_DISABLED) ||
|
|
(idlest == MODULE_CLKCTRL_IDLEST_TRANSITIONING)) {
|
|
clkctrl = readl(&cmwkup->wkup_uart0ctrl);
|
|
idlest = (clkctrl & MODULE_CLKCTRL_IDLEST_MASK) >>
|
|
MODULE_CLKCTRL_IDLEST_SHIFT;
|
|
}
|
|
}
|
|
|
|
void enable_basic_clocks(void)
|
|
{
|
|
u32 *const clk_domains[] = {
|
|
&cmper->l3clkstctrl,
|
|
&cmper->l3sclkstctrl,
|
|
&cmper->l4lsclkstctrl,
|
|
&cmwkup->wkclkstctrl,
|
|
&cmper->emifclkstctrl,
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_explicit_en[] = {
|
|
&cmper->l3clkctrl,
|
|
&cmper->l4lsclkctrl,
|
|
&cmper->l4fwclkctrl,
|
|
&cmwkup->wkl4wkclkctrl,
|
|
&cmper->l3instrclkctrl,
|
|
&cmper->l4hsclkctrl,
|
|
&cmwkup->wkgpio0clkctrl,
|
|
&cmwkup->wkctrlclkctrl,
|
|
&cmper->timer2clkctrl,
|
|
&cmper->gpmcclkctrl,
|
|
&cmper->elmclkctrl,
|
|
&cmper->mmc0clkctrl,
|
|
&cmper->mmc1clkctrl,
|
|
&cmwkup->wkup_i2c0ctrl,
|
|
&cmper->gpio1clkctrl,
|
|
&cmper->gpio2clkctrl,
|
|
&cmper->gpio3clkctrl,
|
|
&cmper->gpio4clkctrl,
|
|
&cmper->gpio5clkctrl,
|
|
&cmper->i2c1clkctrl,
|
|
&cmper->cpgmac0clkctrl,
|
|
&cmper->emiffwclkctrl,
|
|
&cmper->emifclkctrl,
|
|
&cmper->otfaemifclkctrl,
|
|
&cmper->qspiclkctrl,
|
|
&cmper->spi0clkctrl,
|
|
0
|
|
};
|
|
|
|
do_enable_clocks(clk_domains, clk_modules_explicit_en, 1);
|
|
|
|
/* Select the Master osc clk as Timer2 clock source */
|
|
writel(0x1, &cmdpll->clktimer2clk);
|
|
|
|
/* For OPP100 the mac clock should be /5. */
|
|
writel(0x4, &cmdpll->clkselmacclk);
|
|
}
|
|
|
|
void rtc_only_enable_basic_clocks(void)
|
|
{
|
|
u32 *const clk_domains[] = {
|
|
&cmper->emifclkstctrl,
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_explicit_en[] = {
|
|
&cmper->gpio5clkctrl,
|
|
&cmper->emiffwclkctrl,
|
|
&cmper->emifclkctrl,
|
|
&cmper->otfaemifclkctrl,
|
|
0
|
|
};
|
|
|
|
do_enable_clocks(clk_domains, clk_modules_explicit_en, 1);
|
|
|
|
/* Select the Master osc clk as Timer2 clock source */
|
|
writel(0x1, &cmdpll->clktimer2clk);
|
|
}
|
|
|
|
#ifdef CONFIG_TI_EDMA3
|
|
void enable_edma3_clocks(void)
|
|
{
|
|
u32 *const clk_domains_edma3[] = {
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_explicit_en_edma3[] = {
|
|
&cmper->tpccclkctrl,
|
|
&cmper->tptc0clkctrl,
|
|
0
|
|
};
|
|
|
|
do_enable_clocks(clk_domains_edma3,
|
|
clk_modules_explicit_en_edma3,
|
|
1);
|
|
}
|
|
|
|
void disable_edma3_clocks(void)
|
|
{
|
|
u32 *const clk_domains_edma3[] = {
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_disable_edma3[] = {
|
|
&cmper->tpccclkctrl,
|
|
&cmper->tptc0clkctrl,
|
|
0
|
|
};
|
|
|
|
do_disable_clocks(clk_domains_edma3,
|
|
clk_modules_disable_edma3,
|
|
1);
|
|
}
|
|
#endif
|
|
|
|
#if defined(CONFIG_USB_DWC3) || defined(CONFIG_USB_XHCI_OMAP)
|
|
void enable_usb_clocks(int index)
|
|
{
|
|
u32 *usbclkctrl = 0;
|
|
u32 *usbphyocp2scpclkctrl = 0;
|
|
|
|
if (index == 0) {
|
|
usbclkctrl = &cmper->usb0clkctrl;
|
|
usbphyocp2scpclkctrl = &cmper->usbphyocp2scp0clkctrl;
|
|
setbits_le32(&cmper->usb0clkctrl,
|
|
USBOTGSSX_CLKCTRL_OPTFCLKEN_REFCLK960);
|
|
setbits_le32(&cmwkup->usbphy0clkctrl,
|
|
USBPHY0_CLKCTRL_OPTFCLKEN_CLK32K);
|
|
} else if (index == 1) {
|
|
usbclkctrl = &cmper->usb1clkctrl;
|
|
usbphyocp2scpclkctrl = &cmper->usbphyocp2scp1clkctrl;
|
|
setbits_le32(&cmper->usb1clkctrl,
|
|
USBOTGSSX_CLKCTRL_OPTFCLKEN_REFCLK960);
|
|
setbits_le32(&cmwkup->usbphy1clkctrl,
|
|
USBPHY0_CLKCTRL_OPTFCLKEN_CLK32K);
|
|
}
|
|
|
|
u32 *const clk_domains_usb[] = {
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_explicit_en_usb[] = {
|
|
usbclkctrl,
|
|
usbphyocp2scpclkctrl,
|
|
0
|
|
};
|
|
|
|
do_enable_clocks(clk_domains_usb, clk_modules_explicit_en_usb, 1);
|
|
}
|
|
|
|
void disable_usb_clocks(int index)
|
|
{
|
|
u32 *usbclkctrl = 0;
|
|
u32 *usbphyocp2scpclkctrl = 0;
|
|
|
|
if (index == 0) {
|
|
usbclkctrl = &cmper->usb0clkctrl;
|
|
usbphyocp2scpclkctrl = &cmper->usbphyocp2scp0clkctrl;
|
|
clrbits_le32(&cmper->usb0clkctrl,
|
|
USBOTGSSX_CLKCTRL_OPTFCLKEN_REFCLK960);
|
|
clrbits_le32(&cmwkup->usbphy0clkctrl,
|
|
USBPHY0_CLKCTRL_OPTFCLKEN_CLK32K);
|
|
} else if (index == 1) {
|
|
usbclkctrl = &cmper->usb1clkctrl;
|
|
usbphyocp2scpclkctrl = &cmper->usbphyocp2scp1clkctrl;
|
|
clrbits_le32(&cmper->usb1clkctrl,
|
|
USBOTGSSX_CLKCTRL_OPTFCLKEN_REFCLK960);
|
|
clrbits_le32(&cmwkup->usbphy1clkctrl,
|
|
USBPHY0_CLKCTRL_OPTFCLKEN_CLK32K);
|
|
}
|
|
|
|
u32 *const clk_domains_usb[] = {
|
|
0
|
|
};
|
|
|
|
u32 *const clk_modules_disable_usb[] = {
|
|
usbclkctrl,
|
|
usbphyocp2scpclkctrl,
|
|
0
|
|
};
|
|
|
|
do_disable_clocks(clk_domains_usb, clk_modules_disable_usb, 1);
|
|
}
|
|
#endif
|