mirror of
https://github.com/AsahiLinux/u-boot
synced 2025-01-19 00:24:04 +00:00
457e2cd665
There are quite some functions in the Allwinner H616 DRAM "driver", some of them actually change the parameters in the structure passed to them, but many are actually not. To increase the optimisation potential for the code, mark those functions that just read members of the passed dram_para struct as "const". This in itself does not decrease the code size, but lays the groundwork for future changes doing so. Signed-off-by: Andre Przywara <andre.przywara@arm.com> Reviewed-by: Jernej Skrabec <jernej.skrabec@gmail.com>
106 lines
3.7 KiB
C
106 lines
3.7 KiB
C
/*
|
|
* sun50i H616 DDR3-1333 timings, as programmed by Allwinner's boot0
|
|
*
|
|
* The chips are probably able to be driven by a faster clock, but boot0
|
|
* uses a more conservative timing (as usual).
|
|
*
|
|
* (C) Copyright 2020 Jernej Skrabec <jernej.skrabec@siol.net>
|
|
* Based on H6 DDR3 timings:
|
|
* (C) Copyright 2018,2019 Arm Ltd.
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0+
|
|
*/
|
|
|
|
#include <common.h>
|
|
#include <asm/arch/dram.h>
|
|
#include <asm/arch/cpu.h>
|
|
|
|
void mctl_set_timing_params(const struct dram_para *para)
|
|
{
|
|
struct sunxi_mctl_ctl_reg * const mctl_ctl =
|
|
(struct sunxi_mctl_ctl_reg *)SUNXI_DRAM_CTL0_BASE;
|
|
|
|
u8 tccd = 2; /* JEDEC: 4nCK */
|
|
u8 tfaw = ns_to_t(50); /* JEDEC: 30 ns w/ 1K pages */
|
|
u8 trrd = max(ns_to_t(6), 4); /* JEDEC: max(6 ns, 4nCK) */
|
|
u8 trcd = ns_to_t(15); /* JEDEC: 13.5 ns */
|
|
u8 trc = ns_to_t(53); /* JEDEC: 49.5 ns */
|
|
u8 txp = max(ns_to_t(6), 3); /* JEDEC: max(6 ns, 3nCK) */
|
|
u8 trtp = max(ns_to_t(8), 2); /* JEDEC: max(7.5 ns, 4nCK) */
|
|
u8 trp = ns_to_t(15); /* JEDEC: >= 13.75 ns */
|
|
u8 tras = ns_to_t(38); /* JEDEC >= 36 ns, <= 9*trefi */
|
|
u16 trefi = ns_to_t(7800) / 32; /* JEDEC: 7.8us@Tcase <= 85C */
|
|
u16 trfc = ns_to_t(350); /* JEDEC: 160 ns for 2Gb */
|
|
u16 txsr = 4; /* ? */
|
|
|
|
u8 tmrw = 0; /* ? */
|
|
u8 tmrd = 4; /* JEDEC: 4nCK */
|
|
u8 tmod = max(ns_to_t(15), 12); /* JEDEC: max(15 ns, 12nCK) */
|
|
u8 tcke = max(ns_to_t(6), 3); /* JEDEC: max(5.625 ns, 3nCK) */
|
|
u8 tcksrx = max(ns_to_t(10), 4); /* JEDEC: max(10 ns, 5nCK) */
|
|
u8 tcksre = max(ns_to_t(10), 4); /* JEDEC: max(10 ns, 5nCK) */
|
|
u8 tckesr = tcke + 1; /* JEDEC: tCKE(min) + 1nCK */
|
|
u8 trasmax = (para->clk / 2) / 15; /* JEDEC: tREFI * 9 */
|
|
u8 txs = ns_to_t(360) / 32; /* JEDEC: max(5nCK,tRFC+10ns) */
|
|
u8 txsdll = 16; /* JEDEC: 512 nCK */
|
|
u8 txsabort = 4; /* ? */
|
|
u8 txsfast = 4; /* ? */
|
|
u8 tcl = 7; /* JEDEC: CL / 2 => 6 */
|
|
u8 tcwl = 5; /* JEDEC: 8 */
|
|
u8 t_rdata_en = 9; /* ? */
|
|
u8 t_wr_lat = 5; /* ? */
|
|
|
|
u8 twtp; /* (WL + BL / 2 + tWR) / 2 */
|
|
u8 twr2rd; /* (WL + BL / 2 + tWTR) / 2 */
|
|
u8 trd2wr; /* (RL + BL / 2 + 2 - WL) / 2 */
|
|
|
|
if (para->tpr2 & 0x100) {
|
|
tcl = 5;
|
|
tcwl = 4;
|
|
t_rdata_en = 5;
|
|
t_wr_lat = 3;
|
|
}
|
|
|
|
twtp = tcl + 2 + tcwl;
|
|
twr2rd = trtp + 2 + tcwl;
|
|
trd2wr = tcl + 3 - tcwl;
|
|
|
|
/* set DRAM timing */
|
|
writel((twtp << 24) | (tfaw << 16) | (trasmax << 8) | tras,
|
|
&mctl_ctl->dramtmg[0]);
|
|
writel((txp << 16) | (trtp << 8) | trc, &mctl_ctl->dramtmg[1]);
|
|
writel((tcwl << 24) | (tcl << 16) | (trd2wr << 8) | twr2rd,
|
|
&mctl_ctl->dramtmg[2]);
|
|
writel((tmrw << 20) | (tmrd << 12) | tmod, &mctl_ctl->dramtmg[3]);
|
|
writel((trcd << 24) | (tccd << 16) | (trrd << 8) | trp,
|
|
&mctl_ctl->dramtmg[4]);
|
|
writel((tcksrx << 24) | (tcksre << 16) | (tckesr << 8) | tcke,
|
|
&mctl_ctl->dramtmg[5]);
|
|
/* Value suggested by ZynqMP manual and used by libdram */
|
|
writel((txp + 2) | 0x02020000, &mctl_ctl->dramtmg[6]);
|
|
writel((txsfast << 24) | (txsabort << 16) | (txsdll << 8) | txs,
|
|
&mctl_ctl->dramtmg[8]);
|
|
writel(0x00020208, &mctl_ctl->dramtmg[9]);
|
|
writel(0xE0C05, &mctl_ctl->dramtmg[10]);
|
|
writel(0x440C021C, &mctl_ctl->dramtmg[11]);
|
|
writel(8, &mctl_ctl->dramtmg[12]);
|
|
writel(0xA100002, &mctl_ctl->dramtmg[13]);
|
|
writel(txsr, &mctl_ctl->dramtmg[14]);
|
|
|
|
clrbits_le32(&mctl_ctl->init[0], 3 << 30);
|
|
writel(0x420000, &mctl_ctl->init[1]);
|
|
writel(5, &mctl_ctl->init[2]);
|
|
writel(0x1f140004, &mctl_ctl->init[3]);
|
|
writel(0x00200000, &mctl_ctl->init[4]);
|
|
|
|
writel(0, &mctl_ctl->dfimisc);
|
|
clrsetbits_le32(&mctl_ctl->rankctl, 0xff0, 0x660);
|
|
|
|
/* Configure DFI timing */
|
|
writel(t_wr_lat | 0x2000000 | (t_rdata_en << 16) | 0x808000,
|
|
&mctl_ctl->dfitmg0);
|
|
writel(0x100202, &mctl_ctl->dfitmg1);
|
|
|
|
/* set refresh timing */
|
|
writel((trefi << 16) | trfc, &mctl_ctl->rfshtmg);
|
|
}
|