sunxi: add R528/T113-s3/D1(s) DRAM initialisation code

The Allwinner R528/T113-s/D1/D1s SoCs all share the same die, so use the
same DRAM initialisation code.
Make use of prior art here and lift some code from awboot[1], which
carried init code based on earlier decompilation efforts, but with a
GPL2 license tag.
This code has been heavily reworked and cleaned up, to match previous
DRAM routines for other SoCs, and also to be closer to U-Boot's coding
style and support routines.
The actual DRAM chip timing parameters are included in the main file,
since they cover all DRAM types, and are protected by a new Kconfig
CONFIG_SUNXI_DRAM_TYPE symbol, which allows the compiler to pick only
the relevant settings, at build time.

The relevant DRAM chips/board specific configuration parameters are
delivered via Kconfig, so this code here should work for all supported
SoCs and DRAM chips combinations.

Signed-off-by: Andre Przywara <andre.przywara@arm.com>
Tested-by: Sam Edwards <CFSworks@gmail.com>
This commit is contained in:
Andre Przywara 2022-12-31 18:38:21 +00:00
parent 8bddb9742b
commit 124289bd56
6 changed files with 1575 additions and 0 deletions

View file

@ -57,6 +57,7 @@ obj-$(CONFIG_$(SPL_)ALTERA_SDRAM) += ddr/altera/
obj-$(CONFIG_ARCH_IMX8M) += ddr/imx/imx8m/
obj-$(CONFIG_IMX8ULP_DRAM) += ddr/imx/imx8ulp/
obj-$(CONFIG_ARCH_IMX9) += ddr/imx/imx9/
obj-$(CONFIG_DRAM_SUN20I_D1) += ram/
obj-$(CONFIG_SPL_DM_RESET) += reset/
obj-$(CONFIG_SPL_MUSB_NEW) += usb/musb-new/
obj-$(CONFIG_SPL_USB_GADGET) += usb/gadget/

View file

@ -23,6 +23,9 @@ obj-$(CONFIG_RAM_SIFIVE) += sifive/
ifdef CONFIG_SPL_BUILD
obj-$(CONFIG_SPL_STARFIVE_DDR) += starfive/
endif
obj-$(CONFIG_DRAM_SUN20I_D1) += sunxi/
obj-$(CONFIG_ARCH_OCTEON) += octeon/
obj-$(CONFIG_ARCH_RMOBILE) += renesas/

View file

@ -4,3 +4,57 @@ config DRAM_SUN20I_D1
help
This enables support for the DRAM controller driver covering
the Allwinner D1/R528/T113s SoCs.
if DRAM_SUN20I_D1
config DRAM_SUNXI_ODT_EN
hex "DRAM ODT EN parameter"
help
ODT EN value from vendor DRAM settings.
config DRAM_SUNXI_TPR0
hex "DRAM TPR0 parameter"
help
TPR0 value from vendor DRAM settings.
config DRAM_SUNXI_TPR11
hex "DRAM TPR11 parameter"
help
TPR11 value from vendor DRAM settings.
config DRAM_SUNXI_TPR12
hex "DRAM TPR12 parameter"
help
TPR12 value from vendor DRAM settings.
config DRAM_SUNXI_TPR13
hex "DRAM TPR13 parameter"
help
TPR13 value from vendor DRAM settings. It tells which features
should be configured.
choice
prompt "DRAM chip type"
default SUNXI_DRAM_TYPE_DDR3 if DRAM_SUN20I_D1
config SUNXI_DRAM_TYPE_DDR2
bool "DDR2 chips"
config SUNXI_DRAM_TYPE_DDR3
bool "DDR3 chips"
config SUNXI_DRAM_TYPE_LPDDR2
bool "LPDDR2 chips"
config SUNXI_DRAM_TYPE_LPDDR3
bool "LPDDR3 chips"
endchoice
config SUNXI_DRAM_TYPE
int
default 2 if SUNXI_DRAM_TYPE_DDR2
default 3 if SUNXI_DRAM_TYPE_DDR3
default 6 if SUNXI_DRAM_TYPE_LPDDR2
default 7 if SUNXI_DRAM_TYPE_LPDDR3
endif

View file

@ -0,0 +1,3 @@
# SPDX-License-Identifier: GPL-2.0+
obj-$(CONFIG_DRAM_SUN20I_D1) += dram_sun20i_d1.o

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,73 @@
// SPDX-License-Identifier: GPL-2.0+
/*
* D1/R528/T113 DRAM controller register and constant defines
*
* (C) Copyright 2022 Arm Ltd.
* Based on H6 and H616 header, which are:
* (C) Copyright 2017 Icenowy Zheng <icenowy@aosc.io>
* (C) Copyright 2020 Jernej Skrabec <jernej.skrabec@siol.net>
*
*/
#ifndef _SUNXI_DRAM_SUN20I_D1_H
#define _SUNXI_DRAM_SUN20I_D1_H
enum sunxi_dram_type {
SUNXI_DRAM_TYPE_DDR2 = 2,
SUNXI_DRAM_TYPE_DDR3 = 3,
SUNXI_DRAM_TYPE_LPDDR2 = 6,
SUNXI_DRAM_TYPE_LPDDR3 = 7,
};
/*
* This structure contains a mixture of fixed configuration settings,
* variables that are used at runtime to communicate settings between
* different stages and functions, and unused values.
* This is copied from Allwinner's boot0 data structure, which can be
* found at offset 0x38 in any boot0 binary. To allow matching up some
* board specific settings, this struct is kept compatible, even though
* we don't need all members in our code.
*/
typedef struct dram_para {
/* normal configuration */
const u32 dram_clk;
const u32 dram_type;
const u32 dram_zq;
const u32 dram_odt_en;
/* timing configuration */
const u32 dram_mr0;
const u32 dram_mr1;
const u32 dram_mr2;
const u32 dram_mr3;
const u32 dram_tpr0; //DRAMTMG0
const u32 dram_tpr1; //DRAMTMG1
const u32 dram_tpr2; //DRAMTMG2
const u32 dram_tpr3; //DRAMTMG3
const u32 dram_tpr4; //DRAMTMG4
const u32 dram_tpr5; //DRAMTMG5
const u32 dram_tpr6; //DRAMTMG8
const u32 dram_tpr7;
const u32 dram_tpr8;
const u32 dram_tpr9;
const u32 dram_tpr10;
const u32 dram_tpr11;
const u32 dram_tpr12;
} dram_para_t;
typedef struct dram_config {
/* control configuration */
u32 dram_para1;
u32 dram_para2;
/* contains a bitfield of DRAM setup settings */
u32 dram_tpr13;
} dram_config_t;
static inline int ns_to_t(int nanoseconds)
{
const unsigned int ctrl_freq = CONFIG_DRAM_CLK / 2;
return DIV_ROUND_UP(ctrl_freq * nanoseconds, 1000);
}
#endif /* _SUNXI_DRAM_SUN20I_D1_H */