diff options
author | Huayang Duan <huayang.duan@mediatek.com> | 2018-09-26 16:10:42 +0800 |
---|---|---|
committer | Patrick Georgi <pgeorgi@google.com> | 2018-11-08 17:20:24 +0000 |
commit | cede791d4f6381032056f54355c621b4e125fdba (patch) | |
tree | d2f5ba2460ff2216372c872cb2f4dac835ea0d75 /src/soc/mediatek/mt8183 | |
parent | dac7f53a8b77b9891b1a9129a48f385063a4c066 (diff) | |
download | coreboot-cede791d4f6381032056f54355c621b4e125fdba.tar.xz |
mediatek/mt8183: Add DDR driver of cmd bus training part
BRANCH=none
TEST=Boots correctly on Kukui, and inits DRAM successfully with related
patches.
Change-Id: Icb281f1b23c637971497eb28ed428235adf42f2d
Signed-off-by: Huayang Duan <huayang.duan@mediatek.com>
Reviewed-on: https://review.coreboot.org/28839
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Hung-Te Lin <hungte@chromium.org>
Diffstat (limited to 'src/soc/mediatek/mt8183')
-rw-r--r-- | src/soc/mediatek/mt8183/dramc_pi_calibration_api.c | 87 | ||||
-rw-r--r-- | src/soc/mediatek/mt8183/emi.c | 1 | ||||
-rw-r--r-- | src/soc/mediatek/mt8183/include/soc/dramc_pi_api.h | 1 |
3 files changed, 89 insertions, 0 deletions
diff --git a/src/soc/mediatek/mt8183/dramc_pi_calibration_api.c b/src/soc/mediatek/mt8183/dramc_pi_calibration_api.c index 7a027b2d29..3f4bd94639 100644 --- a/src/soc/mediatek/mt8183/dramc_pi_calibration_api.c +++ b/src/soc/mediatek/mt8183/dramc_pi_calibration_api.c @@ -20,6 +20,82 @@ #include <soc/dramc_register.h> #include <soc/dramc_pi_api.h> +static void auto_refresh_switch(u8 chn, u8 option) +{ + clrsetbits_le32(&ch[chn].ao.refctrl0, 1 << REFCTRL0_REFDIS_SHIFT, + (option ? 0 : 1) << REFCTRL0_REFDIS_SHIFT); + + if (option == DISABLE) { + /* + * Because HW will actually disable autorefresh after + * refresh_queue empty, we need to wait until queue empty. + */ + udelay(((read32(&ch[chn].nao.misc_statusa) & + MISC_STATUSA_REFRESH_QUEUE_CNT_MASK) >> + MISC_STATUSA_REFRESH_QUEUE_CNT_SHIFT) * 4); + } +} + +static void dramc_cke_fix_onoff(int option, u8 chn) +{ + u8 on = 0, off = 0; + + /* If CKE is dynamic, set both CKE fix On and Off as 0. */ + if (option != CKE_DYNAMIC) { + on = option; + off = (1 - option); + } + + clrsetbits_le32(&ch[chn].ao.ckectrl, + (0x1 << 6) | (0x1 << 7), (on << 6) | (off << 7)); +} + +static void dramc_mode_reg_write(u8 chn, u8 mr_idx, u8 value) +{ + u32 ckectrl_bak = read32(&ch[chn].ao.ckectrl); + + dramc_cke_fix_onoff(CKE_FIXON, chn); + clrsetbits_le32(&ch[chn].ao.mrs, + MRS_MRSMA_MASK, mr_idx << MRS_MRSMA_SHIFT); + clrsetbits_le32(&ch[chn].ao.mrs, + MRS_MRSOP_MASK, value << MRS_MRSOP_SHIFT); + setbits_le32(&ch[chn].ao.spcmd, 1 << SPCMD_MRWEN_SHIFT); + + /* Wait MRW command fired */ + while ((read32(&ch[chn].nao.spcmdresp) & 1) == 0) + ; + + clrbits_le32(&ch[chn].ao.spcmd, 1 << SPCMD_MRWEN_SHIFT); + setbits_le32(&ch[chn].ao.ckectrl, ckectrl_bak); +} + +static void dramc_mode_reg_write_by_rank(u8 chn, u8 rank, + u8 mr_idx, u8 value) +{ + u32 mrs_back = read32(&ch[chn].ao.mrs) & MRS_MRSRK_MASK; + + clrsetbits_le32(&ch[chn].ao.mrs, + MRS_MRSRK_MASK, rank << MRS_MRSRK_SHIFT); + dramc_mode_reg_write(chn, mr_idx, value); + clrsetbits_le32(&ch[chn].ao.mrs, MRS_MRSRK_MASK, mrs_back); +} + +static void cmd_bus_training(u8 chn, u8 rank, + const struct sdram_params *params) +{ + u32 cbt_cs, mr12_value; + + cbt_cs = params->cbt_cs[chn][rank]; + mr12_value = params->cbt_mr12[chn][rank]; + + /* CBT adjust cs */ + clrsetbits_le32(&ch[chn].phy.shu[0].rk[rank].ca_cmd[9], + SHU1_CA_CMD9_RG_RK_ARFINE_TUNE_CS_MASK, cbt_cs << 0); + + /* CBT set vref */ + dramc_mode_reg_write_by_rank(chn, rank, 12, mr12_value); +} + static void dramc_read_dbi_onoff(u8 onoff) { for (u8 chn = 0; chn < CHANNEL_MAX; chn++) @@ -191,3 +267,14 @@ void dramc_apply_pre_calibration_config(void) clrbits_le32(&ch[0].phy.r0_ca_rxdvs[2], 0x3 << 30); } } + +void dramc_calibrate_all_channels(const struct sdram_params *pams) +{ + for (u8 chn = 0; chn < CHANNEL_MAX; chn++) { + for (u8 rk = RANK_0; rk < RANK_MAX; rk++) { + dramc_show("Start K ch:%d, rank:%d\n", chn, rk); + auto_refresh_switch(chn, 0); + cmd_bus_training(chn, rk, pams); + } + } +} diff --git a/src/soc/mediatek/mt8183/emi.c b/src/soc/mediatek/mt8183/emi.c index a65831449f..a4ac4ab4d9 100644 --- a/src/soc/mediatek/mt8183/emi.c +++ b/src/soc/mediatek/mt8183/emi.c @@ -287,6 +287,7 @@ static void init_dram(const struct sdram_params *params) static void do_calib(const struct sdram_params *params) { dramc_apply_pre_calibration_config(); + dramc_calibrate_all_channels(params); } void mt_set_emi(const struct sdram_params *params) diff --git a/src/soc/mediatek/mt8183/include/soc/dramc_pi_api.h b/src/soc/mediatek/mt8183/include/soc/dramc_pi_api.h index 6608ff75e5..6fc3ef71f4 100644 --- a/src/soc/mediatek/mt8183/include/soc/dramc_pi_api.h +++ b/src/soc/mediatek/mt8183/include/soc/dramc_pi_api.h @@ -140,4 +140,5 @@ u32 dramc_get_broadcast(void); void dramc_init(void); void dramc_sw_impedance(const struct sdram_params *params); void dramc_apply_pre_calibration_config(void); +void dramc_calibrate_all_channels(const struct sdram_params *params); #endif /* _DRAMC_PI_API_MT8183_H */ |