blob: 83d6cad9737ee1f7fd8b32fa7be271b35d9c3ef2 [file] [log] [blame]
/*
* Copyright (c) 2024, Rockchip, Inc. All rights reserved.
*
* SPDX-License-Identifier: BSD-3-Clause
*/
#include <assert.h>
#include <errno.h>
#include <arch_helpers.h>
#include <bl31/bl31.h>
#include <common/debug.h>
#include <drivers/arm/gicv3.h>
#include <drivers/console.h>
#include <drivers/delay_timer.h>
#include <drivers/ti/uart/uart_16550.h>
#include <lib/mmio.h>
#include <plat/common/platform.h>
#include <platform_def.h>
#include <pmu.h>
#include <cpus_on_fixed_addr.h>
#include <plat_pm_helpers.h>
#include <plat_private.h>
#include <pm_pd_regs.h>
#include <rk3588_clk.h>
#include <rockchip_sip_svc.h>
#include <secure.h>
#include <soc.h>
#define PSRAM_SP_TOP ((PMUSRAM_BASE + PMUSRAM_RSIZE) & ~0xf)
#define NONBOOT_CPUS_OFF_LOOP (500000)
#define DSUGRF_REG_CNT (0x78 / 4 + 1)
#define BCORE_GRF_REG_CNT (0x30 / 4 + 1)
#define LCORE_GRF_REG_CNT (0x30 / 4 + 1)
#define CENTER_GRF_REG_CNT (0x20 / 4 + 1)
static struct psram_data_t *psram_sleep_cfg =
(struct psram_data_t *)&sys_sleep_flag_sram;
static int8_t pd_repair_map[] = {
[PD_GPU] = PD_RPR_GPU,
[PD_NPU] = -1,
[PD_VCODEC] = -1,
[PD_NPUTOP] = PD_RPR_NPUTOP,
[PD_NPU1] = PD_RPR_NPU1,
[PD_NPU2] = PD_RPR_NPU2,
[PD_VENC0] = PD_RPR_VENC0,
[PD_VENC1] = PD_RPR_VENC1,
[PD_RKVDEC0] = PD_RPR_RKVDEC0,
[PD_RKVDEC1] = PD_RPR_RKVDEC1,
[PD_VDPU] = PD_RPR_VDPU,
[PD_RGA30] = PD_RPR_RGA30,
[PD_AV1] = PD_RPR_AV1,
[PD_VI] = PD_RPR_VI,
[PD_FEC] = PD_RPR_FEC,
[PD_ISP1] = PD_RPR_ISP1,
[PD_RGA31] = PD_RPR_RGA31,
[PD_VOP] = PD_RPR_VOP,
[PD_VO0] = PD_RPR_VO0,
[PD_VO1] = PD_RPR_VO1,
[PD_AUDIO] = PD_RPR_AUDIO,
[PD_PHP] = PD_RPR_PHP,
[PD_GMAC] = PD_RPR_GMAC,
[PD_PCIE] = PD_RPR_PCIE,
[PD_NVM] = -1,
[PD_NVM0] = PD_RPR_NVM0,
[PD_SDIO] = PD_RPR_SDIO,
[PD_USB] = PD_RPR_USB,
[PD_SECURE] = -1,
[PD_SDMMC] = PD_RPR_SDMMC,
[PD_CRYPTO] = PD_RPR_CRYPTO,
[PD_CENTER] = PD_RPR_CENTER,
[PD_DDR01] = PD_RPR_DDR01,
[PD_DDR23] = PD_RPR_DDR23,
};
struct rk3588_sleep_ddr_data {
uint32_t gpio0a_iomux_l, gpio0a_iomux_h, gpio0b_iomux_l;
uint32_t pmu_pd_st0, bus_idle_st0, qch_pwr_st;
uint32_t pmu2_vol_gate_con[3], pmu2_submem_gate_sft_con0;
uint32_t pmu2_bisr_con0;
uint32_t cpll_con0;
uint32_t cru_mode_con, busscru_mode_con;
uint32_t bussgrf_soc_con7;
uint32_t pmu0grf_soc_con0, pmu0grf_soc_con1, pmu0grf_soc_con3;
uint32_t pmu1grf_soc_con2, pmu1grf_soc_con7, pmu1grf_soc_con8, pmu1grf_soc_con9;
uint32_t pmu0sgrf_soc_con1;
uint32_t pmu1sgrf_soc_con14;
uint32_t ddrgrf_chn_con0[4], ddrgrf_chn_con1[4],
ddrgrf_chn_con2[4], pmu1_ddr_pwr_sft_con[4];
uint32_t pmu1cru_clksel_con1;
};
static struct rk3588_sleep_ddr_data ddr_data;
struct rk3588_sleep_pmusram_data {
uint32_t dsusgrf_soc_con[DSUSGRF_SOC_CON_CNT],
dsusgrf_ddr_hash_con[DSUSGRF_DDR_HASH_CON_CNT];
uint32_t dsu_ddr_fw_rgn_reg[FIREWALL_DSU_RGN_CNT],
dsu_ddr_fw_mst_reg[FIREWALL_DSU_MST_CNT],
dsu_ddr_fw_con_reg[FIREWALL_DSU_CON_CNT];
uint32_t busioc_gpio0b_iomux_h;
};
static __pmusramdata struct rk3588_sleep_pmusram_data pmusram_data;
static __pmusramfunc void dsu_restore_early(void)
{
int i;
/* dsusgrf */
for (i = 0; i < DSUSGRF_SOC_CON_CNT; i++)
mmio_write_32(DSUSGRF_BASE + DSUSGRF_SOC_CON(i),
WITH_16BITS_WMSK(pmusram_data.dsusgrf_soc_con[i]));
for (i = 0; i < DSUSGRF_DDR_HASH_CON_CNT; i++)
mmio_write_32(DSUSGRF_BASE + DSUSGRF_DDR_HASH_CON(i),
pmusram_data.dsusgrf_ddr_hash_con[i]);
/* dsu ddr firewall */
for (i = 0; i < FIREWALL_DSU_RGN_CNT; i++)
mmio_write_32(FIREWALL_DSU_BASE + FIREWALL_DSU_RGN(i),
pmusram_data.dsu_ddr_fw_rgn_reg[i]);
for (i = 0; i < FIREWALL_DSU_MST_CNT; i++)
mmio_write_32(FIREWALL_DSU_BASE + FIREWALL_DSU_MST(i),
pmusram_data.dsu_ddr_fw_mst_reg[i]);
for (i = 0; i < FIREWALL_DSU_CON_CNT; i++)
mmio_write_32(FIREWALL_DSU_BASE + FIREWALL_DSU_CON(i),
pmusram_data.dsu_ddr_fw_con_reg[i]);
}
static __pmusramfunc void ddr_resume(void)
{
dsu_restore_early();
}
static void dsu_core_save(void)
{
int i;
/* dsusgrf */
for (i = 0; i < DSUSGRF_SOC_CON_CNT; i++)
pmusram_data.dsusgrf_soc_con[i] =
mmio_read_32(DSUSGRF_BASE + DSUSGRF_SOC_CON(i));
for (i = 0; i < DSUSGRF_DDR_HASH_CON_CNT; i++)
pmusram_data.dsusgrf_ddr_hash_con[i] =
mmio_read_32(DSUSGRF_BASE + DSUSGRF_DDR_HASH_CON(i));
/* dsu ddr firewall */
for (i = 0; i < FIREWALL_DSU_RGN_CNT; i++)
pmusram_data.dsu_ddr_fw_rgn_reg[i] =
mmio_read_32(FIREWALL_DSU_BASE + FIREWALL_DSU_RGN(i));
for (i = 0; i < FIREWALL_DSU_MST_CNT; i++)
pmusram_data.dsu_ddr_fw_mst_reg[i] =
mmio_read_32(FIREWALL_DSU_BASE + FIREWALL_DSU_MST(i));
for (i = 0; i < FIREWALL_DSU_CON_CNT; i++)
pmusram_data.dsu_ddr_fw_con_reg[i] =
mmio_read_32(FIREWALL_DSU_BASE + FIREWALL_DSU_CON(i));
pvtplls_suspend();
pd_dsu_core_save();
}
static void dsu_core_restore(void)
{
pd_dsu_core_restore();
pvtplls_resume();
}
static uint32_t clk_save[CRU_CLKGATE_CON_CNT + PHPCRU_CLKGATE_CON_CNT +
SECURECRU_CLKGATE_CON_CNT + PMU1CRU_CLKGATE_CON_CNT];
void clk_gate_con_save(void)
{
int i, j = 0;
for (i = 0; i < CRU_CLKGATE_CON_CNT; i++, j++)
clk_save[j] = mmio_read_32(CRU_BASE + CRU_CLKGATE_CON(i));
clk_save[j] = mmio_read_32(PHP_CRU_BASE + PHPCRU_CLKGATE_CON);
for (i = 0; i < SECURECRU_CLKGATE_CON_CNT; i++, j++)
clk_save[j] = mmio_read_32(SCRU_BASE + SECURECRU_CLKGATE_CON(i));
for (i = 0; i < PMU1CRU_CLKGATE_CON_CNT; i++, j++)
clk_save[j] = mmio_read_32(PMU1CRU_BASE + CRU_CLKGATE_CON(i));
}
void clk_gate_con_disable(void)
{
int i;
for (i = 0; i < CRU_CLKGATE_CON_CNT; i++)
mmio_write_32(CRU_BASE + CRU_CLKGATE_CON(i), 0xffff0000);
mmio_write_32(PHP_CRU_BASE + PHPCRU_CLKGATE_CON, 0xffff0000);
for (i = 0; i < SECURECRU_CLKGATE_CON_CNT; i++)
mmio_write_32(SCRU_BASE + SECURECRU_CLKGATE_CON(i), 0xffff0000);
for (i = 0; i < PMU1CRU_CLKGATE_CON_CNT; i++)
mmio_write_32(PMU1CRU_BASE + CRU_CLKGATE_CON(i), 0xffff0000);
}
void clk_gate_con_restore(void)
{
int i, j = 0;
for (i = 0; i < CRU_CLKGATE_CON_CNT; i++, j++)
mmio_write_32(CRU_BASE + CRU_CLKGATE_CON(i),
WITH_16BITS_WMSK(clk_save[j]));
mmio_write_32(PHP_CRU_BASE + PHPCRU_CLKGATE_CON,
WITH_16BITS_WMSK(clk_save[j]));
for (i = 0; i < SECURECRU_CLKGATE_CON_CNT; i++, j++)
mmio_write_32(SCRU_BASE + SECURECRU_CLKGATE_CON(i),
WITH_16BITS_WMSK(clk_save[j]));
for (i = 0; i < PMU1CRU_CLKGATE_CON_CNT; i++, j++)
mmio_write_32(PMU1CRU_BASE + CRU_CLKGATE_CON(i),
WITH_16BITS_WMSK(clk_save[j]));
}
static void pmu_bus_idle_req(uint32_t bus, uint32_t state)
{
uint32_t wait_cnt = 0;
mmio_write_32(PMU_BASE + PMU2_BUS_IDLE_SFTCON(bus / 16),
BITS_WITH_WMASK(state, 0x1, bus % 16));
while (pmu_bus_idle_st(bus) != state ||
pmu_bus_idle_ack(bus) != state) {
if (++wait_cnt > BUS_IDLE_LOOP)
break;
udelay(1);
}
if (wait_cnt > BUS_IDLE_LOOP)
WARN("%s: can't wait state %d for bus %d (0x%x)\n",
__func__, state, bus,
mmio_read_32(PMU_BASE + PMU2_BUS_IDLE_ST(bus / 32)));
}
static void pmu_qch_pwr_ctlr(uint32_t msk, uint32_t state)
{
uint32_t wait_cnt = 0;
if (state != 0)
state = msk;
mmio_write_32(PMU_BASE + PMU2_QCHANNEL_PWR_SFTCON,
BITS_WITH_WMASK(state, msk, 0));
while ((mmio_read_32(PMU_BASE + PMU2_QCHANNEL_STATUS) & msk) != state) {
if (++wait_cnt > QCH_PWR_LOOP)
break;
udelay(1);
}
if (wait_cnt > BUS_IDLE_LOOP)
WARN("%s: can't wait qch:0x%x to state:0x%x (0x%x)\n",
__func__, msk, state,
mmio_read_32(PMU_BASE + PMU2_QCHANNEL_STATUS));
}
static inline uint32_t pmu_power_domain_chain_st(uint32_t pd)
{
return mmio_read_32(PMU_BASE + PMU2_PWR_CHAIN1_ST(pd / 32)) & BIT(pd % 32) ?
pmu_pd_on :
pmu_pd_off;
}
static inline uint32_t pmu_power_domain_mem_st(uint32_t pd)
{
return mmio_read_32(PMU_BASE + PMU2_PWR_MEM_ST(pd / 32)) & BIT(pd % 32) ?
pmu_pd_off :
pmu_pd_on;
}
static inline uint32_t pmu_power_domain_st(uint32_t pd)
{
int8_t pd_repair = pd_repair_map[pd];
if (pd_repair >= 0)
return mmio_read_32(PMU_BASE + PMU2_BISR_STATUS(4)) & BIT(pd_repair) ?
pmu_pd_on :
pmu_pd_off;
else
return mmio_read_32(PMU_BASE + PMU2_PWR_GATE_ST(pd / 32)) & BIT(pd % 32) ?
pmu_pd_off :
pmu_pd_on;
}
static int pmu_power_domain_pd_to_mem_st(uint32_t pd, uint32_t *pd_mem_st)
{
uint32_t mem_st;
switch (pd) {
case PD_NPUTOP:
mem_st = PD_NPU_TOP_MEM_ST;
break;
case PD_NPU1:
mem_st = PD_NPU1_MEM_ST;
break;
case PD_NPU2:
mem_st = PD_NPU2_MEM_ST;
break;
case PD_VENC0:
mem_st = PD_VENC0_MEM_ST;
break;
case PD_VENC1:
mem_st = PD_VENC1_MEM_ST;
break;
case PD_RKVDEC0:
mem_st = PD_RKVDEC0_MEM_ST;
break;
case PD_RKVDEC1:
mem_st = PD_RKVDEC1_MEM_ST;
break;
case PD_RGA30:
mem_st = PD_RGA30_MEM_ST;
break;
case PD_AV1:
mem_st = PD_AV1_MEM_ST;
break;
case PD_VI:
mem_st = PD_VI_MEM_ST;
break;
case PD_FEC:
mem_st = PD_FEC_MEM_ST;
break;
case PD_ISP1:
mem_st = PD_ISP1_MEM_ST;
break;
case PD_RGA31:
mem_st = PD_RGA31_MEM_ST;
break;
case PD_VOP:
mem_st = PD_VOP_MEM_ST;
break;
case PD_VO0:
mem_st = PD_VO0_MEM_ST;
break;
case PD_VO1:
mem_st = PD_VO1_MEM_ST;
break;
case PD_AUDIO:
mem_st = PD_AUDIO_MEM_ST;
break;
case PD_PHP:
mem_st = PD_PHP_MEM_ST;
break;
case PD_GMAC:
mem_st = PD_GMAC_MEM_ST;
break;
case PD_PCIE:
mem_st = PD_PCIE_MEM_ST;
break;
case PD_NVM0:
mem_st = PD_NVM0_MEM_ST;
break;
case PD_SDIO:
mem_st = PD_SDIO_MEM_ST;
break;
case PD_USB:
mem_st = PD_USB_MEM_ST;
break;
case PD_SDMMC:
mem_st = PD_SDMMC_MEM_ST;
break;
default:
return -EINVAL;
}
*pd_mem_st = mem_st;
return 0;
}
static int pmu_power_domain_reset_mem(uint32_t pd, uint32_t pd_mem_st)
{
uint32_t loop = 0;
int ret = 0;
while (pmu_power_domain_chain_st(pd_mem_st) != pmu_pd_on) {
udelay(1);
loop++;
if (loop >= PD_CTR_LOOP) {
WARN("%s: %d chain up time out\n", __func__, pd);
ret = -EINVAL;
goto error;
}
}
udelay(60);
mmio_write_32(PMU_BASE + PMU2_MEMPWR_GATE_SFTCON(pd / 16),
BITS_WITH_WMASK(pmu_pd_off, 0x1, pd % 16));
dsb();
loop = 0;
while (pmu_power_domain_mem_st(pd_mem_st) != pmu_pd_off) {
udelay(1);
loop++;
if (loop >= PD_CTR_LOOP) {
WARN("%s: %d mem down time out\n", __func__, pd);
ret = -EINVAL;
goto error;
}
}
mmio_write_32(PMU_BASE + PMU2_MEMPWR_GATE_SFTCON(pd / 16),
BITS_WITH_WMASK(pmu_pd_on, 0x1, pd % 16));
dsb();
loop = 0;
while (pmu_power_domain_mem_st(pd_mem_st) != pmu_pd_on) {
udelay(1);
loop++;
if (loop >= PD_CTR_LOOP) {
WARN("%s: %d mem up time out\n", __func__, pd);
ret = -EINVAL;
goto error;
}
}
return 0;
error:
return ret;
}
static int pmu_power_domain_ctr(uint32_t pd, uint32_t pd_state)
{
uint32_t loop = 0;
uint32_t is_mem_on = pmu_pd_off;
uint32_t pd_mem_st;
int ret = 0;
if (pd_state == pmu_pd_on) {
ret = pmu_power_domain_pd_to_mem_st(pd, &pd_mem_st);
if (ret == 0) {
is_mem_on = pmu_power_domain_mem_st(pd_mem_st);
if (is_mem_on == pmu_pd_on)
WARN("%s: %d mem is up\n", __func__, pd);
}
}
mmio_write_32(PMU_BASE + PMU2_PWR_GATE_SFTCON(pd / 16),
BITS_WITH_WMASK(pd_state, 0x1, pd % 16));
dsb();
if (is_mem_on == pmu_pd_on) {
ret = pmu_power_domain_reset_mem(pd, pd_mem_st);
if (ret != 0)
goto out;
WARN("%s: %d mem reset ok\n", __func__, pd);
}
while ((pmu_power_domain_st(pd) != pd_state) && (loop < PD_CTR_LOOP)) {
udelay(1);
loop++;
}
if (pmu_power_domain_st(pd) != pd_state) {
WARN("%s: %d, %d, (0x%x, 0x%x) error!\n", __func__, pd, pd_state,
mmio_read_32(PMU_BASE + PMU2_PWR_GATE_ST(0)),
mmio_read_32(PMU_BASE + PMU2_BISR_STATUS(4)));
ret = -EINVAL;
}
out:
return ret;
}
static int pmu_set_power_domain(uint32_t pd_id, uint32_t pd_state)
{
uint32_t state;
if (pmu_power_domain_st(pd_id) == pd_state)
goto out;
if (pd_state == pmu_pd_on)
pmu_power_domain_ctr(pd_id, pd_state);
state = (pd_state == pmu_pd_off) ? bus_idle : bus_active;
switch (pd_id) {
case PD_GPU:
pmu_bus_idle_req(BUS_ID_GPU, state);
break;
case PD_NPUTOP:
pmu_bus_idle_req(BUS_ID_NPUTOP, state);
break;
case PD_NPU1:
pmu_bus_idle_req(BUS_ID_NPU1, state);
break;
case PD_NPU2:
pmu_bus_idle_req(BUS_ID_NPU2, state);
break;
case PD_VENC0:
pmu_bus_idle_req(BUS_ID_RKVENC0, state);
break;
case PD_VENC1:
pmu_bus_idle_req(BUS_ID_RKVENC1, state);
break;
case PD_RKVDEC0:
pmu_bus_idle_req(BUS_ID_RKVDEC0, state);
break;
case PD_RKVDEC1:
pmu_bus_idle_req(BUS_ID_RKVDEC1, state);
break;
case PD_VDPU:
pmu_bus_idle_req(BUS_ID_VDPU, state);
break;
case PD_AV1:
pmu_bus_idle_req(BUS_ID_AV1, state);
break;
case PD_VI:
pmu_bus_idle_req(BUS_ID_VI, state);
break;
case PD_ISP1:
pmu_bus_idle_req(BUS_ID_ISP, state);
break;
case PD_RGA31:
pmu_bus_idle_req(BUS_ID_RGA31, state);
break;
case PD_VOP:
pmu_bus_idle_req(BUS_ID_VOP_CHANNEL, state);
pmu_bus_idle_req(BUS_ID_VOP, state);
break;
case PD_VO0:
pmu_bus_idle_req(BUS_ID_VO0, state);
break;
case PD_VO1:
pmu_bus_idle_req(BUS_ID_VO1, state);
break;
case PD_AUDIO:
pmu_bus_idle_req(BUS_ID_AUDIO, state);
break;
case PD_PHP:
pmu_bus_idle_req(BUS_ID_PHP, state);
break;
case PD_NVM:
pmu_bus_idle_req(BUS_ID_NVM, state);
break;
case PD_SDIO:
pmu_bus_idle_req(BUS_ID_SDIO, state);
break;
case PD_USB:
pmu_bus_idle_req(BUS_ID_USB, state);
break;
case PD_SECURE:
pmu_bus_idle_req(BUS_ID_SECURE, state);
break;
default:
break;
}
if (pd_state == pmu_pd_off)
pmu_power_domain_ctr(pd_id, pd_state);
out:
return 0;
}
static void pmu_power_domains_suspend(void)
{
ddr_data.qch_pwr_st =
mmio_read_32(PMU_BASE + PMU2_QCHANNEL_STATUS) & PMU2_QCH_PWR_MSK;
ddr_data.pmu_pd_st0 = mmio_read_32(PMU_BASE + PMU2_PWR_GATE_ST(0));
ddr_data.bus_idle_st0 = mmio_read_32(PMU_BASE + PMU2_BUS_IDLE_ST(0));
qos_save();
if ((ddr_data.pmu_pd_st0 & BIT(PD_PHP)) == 0)
pd_php_save();
if ((ddr_data.pmu_pd_st0 & BIT(PD_CRYPTO)) == 0)
pd_crypto_save();
pmu_qch_pwr_ctlr(0x20, 1);
pmu_qch_pwr_ctlr(0x40, 1);
pmu_qch_pwr_ctlr(0x1, 1);
pmu_qch_pwr_ctlr(0x2, 1);
pmu_qch_pwr_ctlr(0x4, 1);
pmu_qch_pwr_ctlr(0x8, 1);
pmu_qch_pwr_ctlr(0x10, 1);
pmu_bus_idle_req(BUS_ID_VO1USBTOP, bus_idle);
pmu_bus_idle_req(BUS_ID_SECURE_VO1USB_CHANNEL, bus_idle);
pmu_bus_idle_req(BUS_ID_USB, bus_idle);
pmu_set_power_domain(PD_GPU, pmu_pd_off);
pmu_set_power_domain(PD_NPU1, pmu_pd_off);
pmu_set_power_domain(PD_NPU2, pmu_pd_off);
pmu_set_power_domain(PD_NPUTOP, pmu_pd_off);
pmu_set_power_domain(PD_NPU, pmu_pd_off);
pmu_set_power_domain(PD_RKVDEC1, pmu_pd_off);
pmu_set_power_domain(PD_RKVDEC0, pmu_pd_off);
pmu_set_power_domain(PD_VENC1, pmu_pd_off);
pmu_set_power_domain(PD_VENC0, pmu_pd_off);
pmu_set_power_domain(PD_VCODEC, pmu_pd_off);
pmu_set_power_domain(PD_RGA30, pmu_pd_off);
pmu_set_power_domain(PD_AV1, pmu_pd_off);
pmu_set_power_domain(PD_VDPU, pmu_pd_off);
pmu_set_power_domain(PD_VO0, pmu_pd_off);
pmu_set_power_domain(PD_VO1, pmu_pd_off);
pmu_set_power_domain(PD_VOP, pmu_pd_off);
pmu_set_power_domain(PD_FEC, pmu_pd_off);
pmu_set_power_domain(PD_ISP1, pmu_pd_off);
pmu_set_power_domain(PD_VI, pmu_pd_off);
pmu_set_power_domain(PD_RGA31, pmu_pd_off);
pmu_set_power_domain(PD_AUDIO, pmu_pd_off);
pmu_set_power_domain(PD_GMAC, pmu_pd_off);
pmu_set_power_domain(PD_PCIE, pmu_pd_off);
pmu_set_power_domain(PD_PHP, pmu_pd_off);
pmu_set_power_domain(PD_SDIO, pmu_pd_off);
pmu_set_power_domain(PD_NVM0, pmu_pd_off);
pmu_set_power_domain(PD_NVM, pmu_pd_off);
pmu_set_power_domain(PD_SDMMC, pmu_pd_off);
pmu_set_power_domain(PD_CRYPTO, pmu_pd_off);
}
static void pmu_power_domains_resume(void)
{
int i;
pmu_set_power_domain(PD_CRYPTO, !!(ddr_data.pmu_pd_st0 & BIT(PD_CRYPTO)));
pmu_set_power_domain(PD_SDMMC, !!(ddr_data.pmu_pd_st0 & BIT(PD_SDMMC)));
pmu_set_power_domain(PD_NVM, !!(ddr_data.pmu_pd_st0 & BIT(PD_NVM)));
pmu_set_power_domain(PD_NVM0, !!(ddr_data.pmu_pd_st0 & BIT(PD_NVM0)));
pmu_set_power_domain(PD_SDIO, !!(ddr_data.pmu_pd_st0 & BIT(PD_SDIO)));
pmu_set_power_domain(PD_PHP, !!(ddr_data.pmu_pd_st0 & BIT(PD_PHP)));
pmu_set_power_domain(PD_PCIE, !!(ddr_data.pmu_pd_st0 & BIT(PD_PCIE)));
pmu_set_power_domain(PD_GMAC, !!(ddr_data.pmu_pd_st0 & BIT(PD_GMAC)));
pmu_set_power_domain(PD_AUDIO, !!(ddr_data.pmu_pd_st0 & BIT(PD_AUDIO)));
pmu_set_power_domain(PD_USB, !!(ddr_data.pmu_pd_st0 & BIT(PD_USB)));
pmu_set_power_domain(PD_RGA31, !!(ddr_data.pmu_pd_st0 & BIT(PD_RGA31)));
pmu_set_power_domain(PD_VI, !!(ddr_data.pmu_pd_st0 & BIT(PD_VI)));
pmu_set_power_domain(PD_ISP1, !!(ddr_data.pmu_pd_st0 & BIT(PD_ISP1)));
pmu_set_power_domain(PD_FEC, !!(ddr_data.pmu_pd_st0 & BIT(PD_FEC)));
pmu_set_power_domain(PD_VOP, !!(ddr_data.pmu_pd_st0 & BIT(PD_VOP)));
pmu_set_power_domain(PD_VO1, !!(ddr_data.pmu_pd_st0 & BIT(PD_VO1)));
pmu_set_power_domain(PD_VO0, !!(ddr_data.pmu_pd_st0 & BIT(PD_VO0)));
pmu_set_power_domain(PD_VDPU, !!(ddr_data.pmu_pd_st0 & BIT(PD_VDPU)));
pmu_set_power_domain(PD_AV1, !!(ddr_data.pmu_pd_st0 & BIT(PD_AV1)));
pmu_set_power_domain(PD_RGA30, !!(ddr_data.pmu_pd_st0 & BIT(PD_RGA30)));
pmu_set_power_domain(PD_VCODEC, !!(ddr_data.pmu_pd_st0 & BIT(PD_VCODEC)));
pmu_set_power_domain(PD_VENC0, !!(ddr_data.pmu_pd_st0 & BIT(PD_VENC0)));
pmu_set_power_domain(PD_VENC1, !!(ddr_data.pmu_pd_st0 & BIT(PD_VENC1)));
pmu_set_power_domain(PD_RKVDEC0, !!(ddr_data.pmu_pd_st0 & BIT(PD_RKVDEC0)));
pmu_set_power_domain(PD_RKVDEC1, !!(ddr_data.pmu_pd_st0 & BIT(PD_RKVDEC1)));
pmu_set_power_domain(PD_NPU, !!(ddr_data.pmu_pd_st0 & BIT(PD_NPU)));
pmu_set_power_domain(PD_NPUTOP, !!(ddr_data.pmu_pd_st0 & BIT(PD_NPUTOP)));
pmu_set_power_domain(PD_NPU2, !!(ddr_data.pmu_pd_st0 & BIT(PD_NPU2)));
pmu_set_power_domain(PD_NPU1, !!(ddr_data.pmu_pd_st0 & BIT(PD_NPU1)));
pmu_set_power_domain(PD_GPU, !!(ddr_data.pmu_pd_st0 & BIT(PD_GPU)));
for (i = 0; i < 32; i++)
pmu_bus_idle_req(i, !!(ddr_data.bus_idle_st0 & BIT(i)));
pmu_qch_pwr_ctlr(0x10, !!(ddr_data.qch_pwr_st & 0x10));
pmu_qch_pwr_ctlr(0x8, !!(ddr_data.qch_pwr_st & 0x8));
pmu_qch_pwr_ctlr(0x4, !!(ddr_data.qch_pwr_st & 0x4));
pmu_qch_pwr_ctlr(0x2, !!(ddr_data.qch_pwr_st & 0x2));
pmu_qch_pwr_ctlr(0x1, !!(ddr_data.qch_pwr_st & 0x1));
pmu_qch_pwr_ctlr(0x40, !!(ddr_data.qch_pwr_st & 0x40));
pmu_qch_pwr_ctlr(0x20, !!(ddr_data.qch_pwr_st & 0x20));
if ((ddr_data.pmu_pd_st0 & BIT(PD_CRYPTO)) == 0)
pd_crypto_restore();
if ((ddr_data.pmu_pd_st0 & BIT(PD_PHP)) == 0)
pd_php_restore();
qos_restore();
}
static int cpus_power_domain_on(uint32_t cpu_id)
{
mmio_write_32(PMU_BASE + PMU2_CPU_AUTO_PWR_CON(cpu_id),
BITS_WITH_WMASK(0, 0x1, core_pm_en));
mmio_write_32(PMU_BASE + PMU2_CPU_AUTO_PWR_CON(cpu_id),
BITS_WITH_WMASK(1, 0x1, core_pm_sft_wakeup_en));
dsb();
return 0;
}
static int cpus_power_domain_off(uint32_t cpu_id, uint32_t pd_cfg)
{
uint32_t apm_value = BIT(core_pm_en);
if (pd_cfg == core_pwr_wfi_int)
apm_value |= BIT(core_pm_int_wakeup_en);
mmio_write_32(PMU_BASE + PMU2_CPU_AUTO_PWR_CON(cpu_id),
BITS_WITH_WMASK(apm_value, 0x3, 0));
dsb();
return 0;
}
static inline void cpus_pd_req_enter_wfi(void)
{
/* CORTEX_A55_CPUACTLR_EL1 */
__asm__ volatile ("msr DBGPRCR_EL1, xzr\n"
"mrs x0, S3_0_C15_C2_7\n"
"orr x0, x0, #0x1\n"
"msr S3_0_C15_C2_7, x0\n"
"wfi_loop:\n"
"isb\n"
"wfi\n"
"b wfi_loop\n");
}
static void nonboot_cpus_off(void)
{
uint32_t boot_cpu, cpu, tmp;
uint32_t exp_st;
uint32_t bcore0_rst_msk = 0, bcore1_rst_msk = 0;
int wait_cnt;
bcore0_rst_msk = CRU_BIGCPU02_RST_MSK | CRU_BIGCPU13_RST_MSK;
bcore1_rst_msk = CRU_BIGCPU02_RST_MSK | CRU_BIGCPU13_RST_MSK;
mmio_write_32(BIGCORE0CRU_BASE + 0xa00, BITS_WITH_WMASK(0, bcore0_rst_msk, 0));
mmio_write_32(BIGCORE1CRU_BASE + 0xa00, BITS_WITH_WMASK(0, bcore1_rst_msk, 0));
wait_cnt = NONBOOT_CPUS_OFF_LOOP;
exp_st = SYS_GRF_BIG_CPUS_WFE;
do {
wait_cnt--;
tmp = mmio_read_32(SYSGRF_BASE + SYS_GRF_SOC_STATUS(3));
tmp &= SYS_GRF_BIG_CPUS_WFE;
} while (tmp != exp_st && wait_cnt);
boot_cpu = plat_my_core_pos();
/* turn off noboot cpus */
for (cpu = 0; cpu < PLATFORM_CORE_COUNT; cpu++) {
if (cpu == boot_cpu)
continue;
cpus_power_domain_off(cpu, core_pwr_wfi);
}
mmio_write_32(SRAM_BASE + 0x08, (uintptr_t)&cpus_pd_req_enter_wfi);
mmio_write_32(SRAM_BASE + 0x04, 0xdeadbeaf);
dsb();
isb();
sev();
wait_cnt = NONBOOT_CPUS_OFF_LOOP;
do {
wait_cnt--;
tmp = mmio_read_32(PMU_BASE + PMU2_CLUSTER_ST);
tmp &= CLUSTER_STS_NONBOOT_CPUS_DWN;
} while (tmp != CLUSTER_STS_NONBOOT_CPUS_DWN && wait_cnt);
if (tmp != CLUSTER_STS_NONBOOT_CPUS_DWN)
ERROR("nonboot cpus status(%x) error!\n", tmp);
}
int rockchip_soc_cores_pwr_dm_on(unsigned long mpidr,
uint64_t entrypoint)
{
uint32_t cpu_id = plat_core_pos_by_mpidr(mpidr);
assert(cpu_id < PLATFORM_CORE_COUNT);
assert(cpuson_flags[cpu_id] == 0);
cpuson_flags[cpu_id] = PMU_CPU_HOTPLUG;
cpuson_entry_point[cpu_id] = entrypoint;
dsb();
flush_dcache_range((uintptr_t)cpuson_flags, sizeof(cpuson_flags));
flush_dcache_range((uintptr_t)cpuson_entry_point,
sizeof(cpuson_entry_point));
dsb();
isb();
cpus_power_domain_on(cpu_id);
return PSCI_E_SUCCESS;
}
int rockchip_soc_cores_pwr_dm_on_finish(void)
{
uint32_t cpu_id = plat_my_core_pos();
mmio_write_32(PMU_BASE + PMU2_CPU_AUTO_PWR_CON(cpu_id),
BITS_WITH_WMASK(0, 0xf, 0));
return PSCI_E_SUCCESS;
}
int rockchip_soc_cores_pwr_dm_off(void)
{
uint32_t cpu_id = plat_my_core_pos();
cpus_power_domain_off(cpu_id, core_pwr_wfi);
return PSCI_E_SUCCESS;
}
int rockchip_soc_cores_pwr_dm_suspend(void)
{
uint32_t cpu_id = plat_my_core_pos();
assert(cpu_id < PLATFORM_CORE_COUNT);
cpuson_flags[cpu_id] = PMU_CPU_AUTO_PWRDN;
cpuson_entry_point[cpu_id] = plat_get_sec_entrypoint();
dsb();
flush_dcache_range((uintptr_t)cpuson_flags, sizeof(cpuson_flags));
flush_dcache_range((uintptr_t)cpuson_entry_point,
sizeof(cpuson_entry_point));
dsb();
isb();
cpus_power_domain_off(cpu_id, core_pwr_wfi_int);
__asm__ volatile ("msr DBGPRCR_EL1, xzr\n"
"mrs x0, S3_0_C15_C2_7\n"
"orr x0, x0, #0x1\n"
"msr S3_0_C15_C2_7, x0\n");
return PSCI_E_SUCCESS;
}
int rockchip_soc_cores_pwr_dm_resume(void)
{
uint32_t cpu_id = plat_my_core_pos();
mmio_write_32(PMU_BASE + PMU2_CPU_AUTO_PWR_CON(cpu_id),
BITS_WITH_WMASK(0, 0x3, 0));
dsb();
return PSCI_E_SUCCESS;
}
static void ddr_sleep_config(void)
{
int i;
if (pmu_power_domain_st(PD_DDR01) == 0) {
ddr_data.ddrgrf_chn_con0[0] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHA_CON(0));
ddr_data.ddrgrf_chn_con0[1] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHB_CON(0));
ddr_data.ddrgrf_chn_con1[0] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHA_CON(1));
ddr_data.ddrgrf_chn_con1[1] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHB_CON(1));
ddr_data.ddrgrf_chn_con2[0] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHA_CON(2));
ddr_data.ddrgrf_chn_con2[1] =
mmio_read_32(DDR01GRF_BASE + DDRGRF_CHB_CON(2));
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(2), 0x20002000);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(2), 0x20002000);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(2), 0x08000000);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(2), 0x08000000);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(0), 0x00200020);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(0), 0x00200020);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(1), 0x00400040);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(1), 0x00400040);
}
if (pmu_power_domain_st(PD_DDR23) == 0) {
ddr_data.ddrgrf_chn_con0[2] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHA_CON(0));
ddr_data.ddrgrf_chn_con0[3] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHB_CON(0));
ddr_data.ddrgrf_chn_con1[2] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHA_CON(1));
ddr_data.ddrgrf_chn_con1[3] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHB_CON(1));
ddr_data.ddrgrf_chn_con2[2] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHA_CON(2));
ddr_data.ddrgrf_chn_con2[3] =
mmio_read_32(DDR23GRF_BASE + DDRGRF_CHB_CON(2));
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(2), 0x20002000);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(2), 0x20002000);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(2), 0x08000000);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(2), 0x08000000);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(0), 0x00200020);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(0), 0x00200020);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(1), 0x00400040);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(1), 0x00400040);
}
for (i = 0; i < DDR_CHN_CNT; i++) {
ddr_data.pmu1_ddr_pwr_sft_con[i] =
mmio_read_32(PMU_BASE + PMU1_DDR_PWR_SFTCON(i));
mmio_write_32(PMU_BASE + PMU1_DDR_PWR_SFTCON(i), 0x0fff0900);
}
}
static void ddr_sleep_config_restore(void)
{
int i;
for (i = 0; i < DDR_CHN_CNT; i++) {
mmio_write_32(PMU_BASE + PMU1_DDR_PWR_SFTCON(i),
0x0fff0000 | ddr_data.pmu1_ddr_pwr_sft_con[i]);
}
if (pmu_power_domain_st(PD_DDR01) == 0) {
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(1),
0x00400000 | ddr_data.ddrgrf_chn_con1[0]);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(1),
0x00400000 | ddr_data.ddrgrf_chn_con1[1]);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(0),
0x00200000 | ddr_data.ddrgrf_chn_con0[0]);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(0),
0x00200000 | ddr_data.ddrgrf_chn_con0[1]);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHA_CON(2),
0x28000000 | ddr_data.ddrgrf_chn_con2[0]);
mmio_write_32(DDR01GRF_BASE + DDRGRF_CHB_CON(2),
0x28000000 | ddr_data.ddrgrf_chn_con2[1]);
}
if (pmu_power_domain_st(PD_DDR23) == 0) {
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(1),
0x00400000 | ddr_data.ddrgrf_chn_con1[2]);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(1),
0x00400000 | ddr_data.ddrgrf_chn_con1[3]);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(0),
0x00200000 | ddr_data.ddrgrf_chn_con0[2]);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(0),
0x00200000 | ddr_data.ddrgrf_chn_con0[3]);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHA_CON(2),
0x28000000 | ddr_data.ddrgrf_chn_con2[2]);
mmio_write_32(DDR23GRF_BASE + DDRGRF_CHB_CON(2),
0x28000000 | ddr_data.ddrgrf_chn_con2[3]);
}
}
static void pmu_sleep_config(void)
{
uint32_t pmu1_pwr_con, pmu1_wkup_int_con, pmu1_cru_pwr_con;
uint32_t pmu1_ddr_pwr_con, pmu1_pll_pd_con[2] = {0};
uint32_t pmu2_dsu_pwr_con, pmu2_core_pwr_con, pmu2_clst_idle_con;
uint32_t pmu2_bus_idle_con[3] = {0}, pmu2_pwr_gate_con[3] = {0};
uint32_t pmu2_vol_gate_con[3] = {0}, pmu2_qch_pwr_con = 0;
int i;
ddr_data.pmu1grf_soc_con7 = mmio_read_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(7));
ddr_data.pmu1grf_soc_con8 = mmio_read_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(8));
ddr_data.pmu1grf_soc_con9 = mmio_read_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(9));
ddr_data.pmu1sgrf_soc_con14 = mmio_read_32(PMU1SGRF_BASE + PMU1_SGRF_SOC_CON(14));
ddr_data.pmu0sgrf_soc_con1 = mmio_read_32(PMU0SGRF_BASE + PMU0_SGRF_SOC_CON(1));
ddr_data.pmu0grf_soc_con1 = mmio_read_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(1));
ddr_data.pmu2_vol_gate_con[0] = mmio_read_32(PMU_BASE + PMU2_VOL_GATE_CON(0));
ddr_data.pmu2_vol_gate_con[1] = mmio_read_32(PMU_BASE + PMU2_VOL_GATE_CON(1));
ddr_data.pmu2_vol_gate_con[2] = mmio_read_32(PMU_BASE + PMU2_VOL_GATE_CON(2));
ddr_data.pmu2_submem_gate_sft_con0 =
mmio_read_32(PMU_BASE + PMU2_MEMPWR_MD_GATE_SFTCON(0));
/* save pmic_sleep iomux gpio0_a4 */
ddr_data.gpio0a_iomux_l = mmio_read_32(PMU0IOC_BASE + 0);
ddr_data.gpio0a_iomux_h = mmio_read_32(PMU0IOC_BASE + 4);
ddr_data.pmu0grf_soc_con3 = mmio_read_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(3));
/* PMU1 repair disable */
mmio_write_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(0), 0x00010000);
/* set pmic_sleep iomux */
mmio_write_32(PMU0IOC_BASE + 0,
BITS_WITH_WMASK(1, 0xf, 8) |
BITS_WITH_WMASK(1, 0xfu, 12));
/* set tsadc_shut_m0 pin iomux to gpio */
mmio_write_32(PMU0IOC_BASE + 0,
BITS_WITH_WMASK(0, 0xf, 4));
/* set spi2_cs0/1 pin iomux to gpio */
mmio_write_32(PMU0IOC_BASE + 8,
BITS_WITH_WMASK(0, 0xff, 0));
/* sleep 1~2 src select */
mmio_write_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(3),
BITS_WITH_WMASK(0x8, 0xf, 0) |
BITS_WITH_WMASK(0x8, 0xf, 4) |
BITS_WITH_WMASK(0x0, 0x3, 8));
pmu1_wkup_int_con = BIT(WAKEUP_GPIO0_INT_EN) |
BIT(WAKEUP_CPU0_INT_EN);
pmu1_pwr_con = BIT(powermode_en);
pmu1_cru_pwr_con =
BIT(alive_osc_mode_en) |
BIT(power_off_en) |
BIT(pd_clk_src_gate_en);
pmu1_ddr_pwr_con = 0;
pmu2_dsu_pwr_con =
BIT(DSU_PWRDN_EN) |
BIT(DSU_PWROFF_EN);
pmu2_core_pwr_con = BIT(CORE_PWRDN_EN);
pmu2_clst_idle_con =
BIT(IDLE_REQ_BIGCORE0_EN) |
BIT(IDLE_REQ_BIGCORE1_EN) |
BIT(IDLE_REQ_DSU_EN) |
BIT(IDLE_REQ_LITDSU_EN) |
BIT(IDLE_REQ_ADB400_CORE_QCH_EN);
pmu1_pll_pd_con[0] =
BIT(B0PLL_PD_EN) |
BIT(B1PLL_PD_EN) |
BIT(LPLL_PD_EN) |
BIT(V0PLL_PD_EN) |
BIT(AUPLL_PD_EN) |
BIT(GPLL_PD_EN) |
BIT(CPLL_PD_EN) |
BIT(NPLL_PD_EN);
pmu1_pll_pd_con[1] =
BIT(PPLL_PD_EN) |
BIT(SPLL_PD_EN);
pmu2_bus_idle_con[0] = 0;
pmu2_bus_idle_con[1] =
BIT(BUS_ID_SECURE - 16) |
BIT(BUS_ID_SECURE_CENTER_CHANNEL - 16) |
BIT(BUS_ID_CENTER_CHANNEL - 16);
pmu2_bus_idle_con[2] =
BIT(BUS_ID_MSCH - 32) |
BIT(BUS_ID_BUS - 32) |
BIT(BUS_ID_TOP - 32);
pmu2_pwr_gate_con[0] = 0;
pmu2_pwr_gate_con[1] = BIT(PD_SECURE - 16);
pmu2_pwr_gate_con[2] = 0;
pmu2_qch_pwr_con = 0;
pmu2_vol_gate_con[0] = 0x7;
pmu2_vol_gate_con[2] = 0;
mmio_write_32(PMU_BASE + PMU2_CORE_AUTO_PWR_CON(0), 0x00030000);
mmio_write_32(PMU_BASE + PMU2_CORE_AUTO_PWR_CON(1), 0x00030000);
mmio_write_32(PMU_BASE + PMU2_CORE_PWR_CON(0),
WITH_16BITS_WMSK(pmu2_core_pwr_con));
mmio_write_32(PMU_BASE + PMU2_CORE_PWR_CON(1),
WITH_16BITS_WMSK(pmu2_core_pwr_con));
mmio_write_32(PMU_BASE + PMU2_CLUSTER_IDLE_CON,
WITH_16BITS_WMSK(pmu2_clst_idle_con));
mmio_write_32(PMU_BASE + PMU2_DSU_AUTO_PWR_CON, 0x00030000);
mmio_write_32(PMU_BASE + PMU2_DSU_PWR_CON,
WITH_16BITS_WMSK(pmu2_dsu_pwr_con));
mmio_write_32(PMU_BASE + PMU1_OSC_STABLE_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU1_STABLE_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU1_WAKEUP_RST_CLR_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU1_PLL_LOCK_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU1_PWM_SWITCH_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE0_STABLE_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE0_PWRUP_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE0_PWRDN_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE1_STABLE_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE1_PWRUP_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_CORE1_PWRDN_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_DSU_STABLE_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_DSU_PWRUP_CNT_THRESH, 24000);
mmio_write_32(PMU_BASE + PMU2_DSU_PWRDN_CNT_THRESH, 24000);
/* Config pmu power mode and pmu wakeup source */
mmio_write_32(PMU_BASE + PMU1_INT_MASK_CON,
BITS_WITH_WMASK(1, 0x1, 0));
/* pmu1_pwr_con */
mmio_write_32(PMU_BASE + PMU1_PWR_CON,
WITH_16BITS_WMSK(pmu1_pwr_con));
/* cru_pwr_con */
mmio_write_32(PMU_BASE + PMU1_CRU_PWR_CON,
WITH_16BITS_WMSK(pmu1_cru_pwr_con));
/* wakeup source */
mmio_write_32(PMU_BASE + PMU1_WAKEUP_INT_CON, pmu1_wkup_int_con);
/* ddr pwr con */
for (i = 0; i < DDR_CHN_CNT; i++) {
mmio_write_32(PMU_BASE + PMU1_DDR_PWR_CON(i),
WITH_16BITS_WMSK(pmu1_ddr_pwr_con));
pmu2_bus_idle_con[1] |=
BIT(BUS_ID_MSCH0 - 16 + i);
}
/* pll_pd */
mmio_write_32(PMU_BASE + PMU1_PLLPD_CON(0),
WITH_16BITS_WMSK(pmu1_pll_pd_con[0]));
mmio_write_32(PMU_BASE + PMU1_PLLPD_CON(1),
WITH_16BITS_WMSK(pmu1_pll_pd_con[1]));
/* bypass cpu1~7*/
mmio_write_32(PMU_BASE + PMU2_PWR_CON1, 0x00ff00fe);
/* bus idle */
mmio_write_32(PMU_BASE + PMU2_BUS_IDLE_CON(0),
WITH_16BITS_WMSK(pmu2_bus_idle_con[0]));
mmio_write_32(PMU_BASE + PMU2_BUS_IDLE_CON(1),
WITH_16BITS_WMSK(pmu2_bus_idle_con[1]));
mmio_write_32(PMU_BASE + PMU2_BUS_IDLE_CON(2),
WITH_16BITS_WMSK(pmu2_bus_idle_con[2]));
mmio_write_32(PMU_BASE + PMU2_BUS_IDLE_CON(2),
0xf000f000);
/* power gate */
mmio_write_32(PMU_BASE + PMU2_PWR_GATE_CON(0),
WITH_16BITS_WMSK(pmu2_pwr_gate_con[0]));
mmio_write_32(PMU_BASE + PMU2_PWR_GATE_CON(1),
WITH_16BITS_WMSK(pmu2_pwr_gate_con[1]));
mmio_write_32(PMU_BASE + PMU2_PWR_GATE_CON(2),
WITH_16BITS_WMSK(pmu2_pwr_gate_con[2]));
/* vol gate */
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(0),
BITS_WITH_WMASK(pmu2_vol_gate_con[0], 0x7, 0));
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(1), 0);
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(2),
BITS_WITH_WMASK(pmu2_vol_gate_con[2], 0x3, 0));
/* qch */
mmio_write_32(PMU_BASE + PMU2_QCHANNEL_PWR_CON,
BITS_WITH_WMASK(pmu2_qch_pwr_con, 0x7f, 0));
mmio_write_32(PMU_BASE + PMU2_MEMPWR_MD_GATE_SFTCON(0),
0x000f000f);
}
static void pmu_sleep_restore(void)
{
mmio_write_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(7),
WITH_16BITS_WMSK(ddr_data.pmu1grf_soc_con7));
mmio_write_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(8),
WITH_16BITS_WMSK(ddr_data.pmu1grf_soc_con8));
mmio_write_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(9),
WITH_16BITS_WMSK(ddr_data.pmu1grf_soc_con9));
mmio_write_32(PMU1SGRF_BASE + PMU1_SGRF_SOC_CON(14),
WITH_16BITS_WMSK(ddr_data.pmu1sgrf_soc_con14));
mmio_write_32(PMU0SGRF_BASE + PMU0_SGRF_SOC_CON(1),
WITH_16BITS_WMSK(ddr_data.pmu0sgrf_soc_con1));
mmio_write_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(1),
WITH_16BITS_WMSK(ddr_data.pmu0grf_soc_con1));
mmio_write_32(PMU_BASE + PMU2_CORE_PWR_CON(0), 0xffff0000);
mmio_write_32(PMU_BASE + PMU2_CORE_PWR_CON(1), 0xffff0000);
mmio_write_32(PMU_BASE + PMU2_CLUSTER_IDLE_CON, 0xffff0000);
mmio_write_32(PMU_BASE + PMU2_DSU_PWR_CON, 0xffff0000);
mmio_write_32(PMU_BASE + PMU2_PWR_CON1, 0xffff0000);
/* Must clear PMU1_WAKEUP_INT_CON because the wakeup source
* in PMU1_WAKEUP_INT_CON will wakeup cpus in cpu_auto_pd state.
*/
mmio_write_32(PMU_BASE + PMU1_WAKEUP_INT_CON, 0);
mmio_write_32(PMU_BASE + PMU1_PWR_CON, 0xffff0000);
mmio_write_32(PMU_BASE + PMU1_INT_MASK_CON, 0x00010000);
mmio_write_32(PMU_BASE + PMU0_WAKEUP_INT_CON, 0x00010000);
mmio_write_32(PMU_BASE + PMU0_PWR_CON, 0xffff0000);
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(0),
WITH_16BITS_WMSK(ddr_data.pmu2_vol_gate_con[0]));
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(1),
WITH_16BITS_WMSK(ddr_data.pmu2_vol_gate_con[1]));
mmio_write_32(PMU_BASE + PMU2_VOL_GATE_CON(2),
WITH_16BITS_WMSK(ddr_data.pmu2_vol_gate_con[2]));
mmio_write_32(PMU_BASE + PMU2_MEMPWR_MD_GATE_SFTCON(0),
WITH_16BITS_WMSK(ddr_data.pmu2_submem_gate_sft_con0));
mmio_write_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(3),
WITH_16BITS_WMSK(ddr_data.pmu0grf_soc_con3));
mmio_write_32(PMU1GRF_BASE + PMU1_GRF_SOC_CON(2),
WITH_16BITS_WMSK(ddr_data.pmu1grf_soc_con2));
mmio_write_32(PMU0IOC_BASE + 0x4,
WITH_16BITS_WMSK(ddr_data.gpio0a_iomux_h));
mmio_write_32(PMU0IOC_BASE + 0,
WITH_16BITS_WMSK(ddr_data.gpio0a_iomux_l));
}
static void soc_sleep_config(void)
{
ddr_data.gpio0b_iomux_l = mmio_read_32(PMU0IOC_BASE + 0x8);
pmu_sleep_config();
ddr_sleep_config();
}
static void soc_sleep_restore(void)
{
ddr_sleep_config_restore();
pmu_sleep_restore();
mmio_write_32(PMU0IOC_BASE + 0x8, WITH_16BITS_WMSK(ddr_data.gpio0b_iomux_l));
}
static void pm_pll_suspend(void)
{
ddr_data.cru_mode_con = mmio_read_32(CRU_BASE + 0x280);
ddr_data.busscru_mode_con = mmio_read_32(BUSSCRU_BASE + 0x280);
ddr_data.pmu2_bisr_con0 = mmio_read_32(PMU_BASE + PMU2_BISR_CON(0));
ddr_data.cpll_con0 = mmio_read_32(CRU_BASE + CRU_PLLS_CON(2, 0));
ddr_data.pmu1cru_clksel_con1 = mmio_read_32(PMU1CRU_BASE + CRU_CLKSEL_CON(1));
/* disable bisr_init */
mmio_write_32(PMU_BASE + PMU2_BISR_CON(0), BITS_WITH_WMASK(0, 0x1, 0));
/* cpll bypass */
mmio_write_32(CRU_BASE + CRU_PLLS_CON(2, 0), BITS_WITH_WMASK(1u, 1u, 15));
}
static void pm_pll_restore(void)
{
pm_pll_wait_lock(CRU_BASE + CRU_PLLS_CON(2, 0));
mmio_write_32(CRU_BASE + 0x280, WITH_16BITS_WMSK(ddr_data.cru_mode_con));
mmio_write_32(BUSSCRU_BASE + 0x280, WITH_16BITS_WMSK(ddr_data.busscru_mode_con));
mmio_write_32(CRU_BASE + CRU_PLLS_CON(2, 0), WITH_16BITS_WMSK(ddr_data.cpll_con0));
dsb();
isb();
mmio_write_32(PMU_BASE + PMU2_BISR_CON(0), WITH_16BITS_WMSK(ddr_data.pmu2_bisr_con0));
}
int rockchip_soc_sys_pwr_dm_suspend(void)
{
clk_gate_con_save();
clk_gate_con_disable();
psram_sleep_cfg->pm_flag &= ~PM_WARM_BOOT_BIT;
pmu_power_domains_suspend();
soc_sleep_config();
dsu_core_save();
pm_pll_suspend();
return 0;
}
int rockchip_soc_sys_pwr_dm_resume(void)
{
pm_pll_restore();
dsu_core_restore();
soc_sleep_restore();
pmu_power_domains_resume();
plat_rockchip_gic_cpuif_enable();
psram_sleep_cfg->pm_flag |= PM_WARM_BOOT_BIT;
clk_gate_con_restore();
return 0;
}
void __dead2 rockchip_soc_cores_pd_pwr_dn_wfi(const
psci_power_state_t *target_state)
{
psci_power_down_wfi();
}
void __dead2 rockchip_soc_sys_pd_pwr_dn_wfi(void)
{
cpus_pd_req_enter_wfi();
psci_power_down_wfi();
}
void __dead2 rockchip_soc_soft_reset(void)
{
/* pll slow mode */
mmio_write_32(CRU_BASE + 0x280, 0x03ff0000);
mmio_write_32(BIGCORE0CRU_BASE + 0x280, 0x00030000);
mmio_write_32(BIGCORE0CRU_BASE + 0x300, 0x60000000);
mmio_write_32(BIGCORE0CRU_BASE + 0x304, 0x00600000);
mmio_write_32(BIGCORE1CRU_BASE + 0x280, 0x00030000);
mmio_write_32(BIGCORE1CRU_BASE + 0x300, 0x60000000);
mmio_write_32(BIGCORE1CRU_BASE + 0x304, 0x00600000);
mmio_write_32(DSUCRU_BASE + 0x280, 0x00030000);
mmio_write_32(DSUCRU_BASE + 0x318, 0x30600000);
mmio_write_32(DSUCRU_BASE + 0x31c, 0x30600000);
mmio_write_32(DSUCRU_BASE + 0x304, 0x00010000);
mmio_write_32(BUSSCRU_BASE + 0x280, 0x0003000);
dsb();
isb();
mmio_write_32(CRU_BASE + CRU_GLB_SRST_FST, GLB_SRST_FST_CFG_VAL);
/*
* Maybe the HW needs some times to reset the system,
* so we do not hope the core to execute valid codes.
*/
psci_power_down_wfi();
}
void __dead2 rockchip_soc_system_off(void)
{
/* set pmic_sleep pin(gpio0_a2) to gpio mode */
mmio_write_32(PMU0IOC_BASE + 0, BITS_WITH_WMASK(0, 0xf, 8));
/* config output */
mmio_write_32(GPIO0_BASE + GPIO_SWPORT_DDR_L,
BITS_WITH_WMASK(1, 0x1, 2));
/* config output high level */
mmio_write_32(GPIO0_BASE + GPIO_SWPORT_DR_L,
BITS_WITH_WMASK(1, 0x1, 2));
dsb();
/*
* Maybe the HW needs some times to reset the system,
* so we do not hope the core to execute valid codes.
*/
psci_power_down_wfi();
}
static void rockchip_pmu_pd_init(void)
{
mmio_write_32(PMU_BASE + PMU2_BISR_CON(1), 0xffffffff);
mmio_write_32(PMU_BASE + PMU2_BISR_CON(2), 0xffffffff);
mmio_write_32(PMU_BASE + PMU2_BISR_CON(3), 0xffffffff);
pmu_set_power_domain(PD_PHP, pmu_pd_on);
pmu_set_power_domain(PD_PCIE, pmu_pd_on);
pmu_set_power_domain(PD_GMAC, pmu_pd_on);
pmu_set_power_domain(PD_SECURE, pmu_pd_on);
pmu_set_power_domain(PD_VOP, pmu_pd_on);
pmu_set_power_domain(PD_VO0, pmu_pd_on);
pmu_set_power_domain(PD_VO1, pmu_pd_on);
}
#define PLL_LOCKED_TIMEOUT 600000U
void pm_pll_wait_lock(uint32_t pll_base)
{
int delay = PLL_LOCKED_TIMEOUT;
if ((mmio_read_32(pll_base + CRU_PLL_CON(1)) & CRU_PLLCON1_PWRDOWN) != 0)
return;
while (delay-- >= 0) {
if (mmio_read_32(pll_base + CRU_PLL_CON(6)) &
CRU_PLLCON6_LOCK_STATUS)
break;
udelay(1);
}
if (delay <= 0)
ERROR("Can't wait pll(0x%x) lock\n", pll_base);
}
void rockchip_plat_mmu_el3(void)
{
/* Nothing todo */
}
void plat_rockchip_pmu_init(void)
{
int cpu;
for (cpu = 0; cpu < PLATFORM_CORE_COUNT; cpu++)
cpuson_flags[cpu] = 0;
psram_sleep_cfg->sp = PSRAM_SP_TOP;
psram_sleep_cfg->ddr_func = (uint64_t)ddr_resume;
psram_sleep_cfg->ddr_data = 0;
psram_sleep_cfg->ddr_flag = 0;
psram_sleep_cfg->boot_mpidr = read_mpidr_el1() & 0xffff;
psram_sleep_cfg->pm_flag = PM_WARM_BOOT_BIT;
nonboot_cpus_off();
/*
* When perform idle operation, corresponding clock can be
* opened or gated automatically.
*/
mmio_write_32(PMU_BASE + PMU2_BIU_AUTO_CON(0), 0xffffffff);
mmio_write_32(PMU_BASE + PMU2_BIU_AUTO_CON(1), 0xffffffff);
mmio_write_32(PMU_BASE + PMU2_BIU_AUTO_CON(2), 0x00070007);
rockchip_pmu_pd_init();
/* grf_con_pmic_sleep_sel
* pmic sleep function selection
* 1'b0: From reset pulse generator, can reset external PMIC
* 1'b1: From pmu block, only support sleep function for external PMIC
*/
mmio_write_32(PMU0GRF_BASE + PMU0_GRF_SOC_CON(3), 0x03ff0000);
/* pmusram remap to 0xffff0000 */
mmio_write_32(PMU0SGRF_BASE + PMU0_SGRF_SOC_CON(2), 0x00030001);
pm_reg_rgns_init();
}