blob: b275407d4ac6c607390d4a86d7472b03eb5822c3 [file] [log] [blame]
Tom Rini8b0c8a12018-05-06 18:27:01 -04001// SPDX-License-Identifier: GPL-2.0+ OR BSD-3-Clause
Patrick Delaunay939d5362018-03-12 10:46:11 +01002/*
3 * Copyright (C) 2018, STMicroelectronics - All Rights Reserved
Patrick Delaunay939d5362018-03-12 10:46:11 +01004 */
5
Patrick Delaunay753f5362020-11-06 19:01:36 +01006#define LOG_CATEGORY UCLASS_RAM
7
Patrick Delaunay939d5362018-03-12 10:46:11 +01008#include <clk.h>
Simon Glass0f2af882020-05-10 11:40:05 -06009#include <log.h>
Patrick Delaunay939d5362018-03-12 10:46:11 +010010#include <ram.h>
11#include <reset.h>
12#include <timer.h>
13#include <asm/io.h>
14#include <asm/arch/ddr.h>
Marek Vasut3bb89782025-05-12 19:09:05 +020015#include <dm/device.h>
Simon Glass4dcacfc2020-05-10 11:40:13 -060016#include <linux/bitops.h>
Simon Glassdbd79542020-05-10 11:40:11 -060017#include <linux/delay.h>
Patrick Delaunay939d5362018-03-12 10:46:11 +010018#include <linux/iopoll.h>
Simon Glassbdd5f812023-09-14 18:21:46 -060019#include <linux/printk.h>
Patrick Delaunay939d5362018-03-12 10:46:11 +010020#include "stm32mp1_ddr.h"
21#include "stm32mp1_ddr_regs.h"
22
Marek Vasut3bb89782025-05-12 19:09:05 +020023#define RCC_DDRITFCR_STM32MP13xx 0x5c0
24#define RCC_DDRITFCR_STM32MP15xx 0xd8
Patrick Delaunay939d5362018-03-12 10:46:11 +010025
26#define RCC_DDRITFCR_DDRCAPBRST (BIT(14))
27#define RCC_DDRITFCR_DDRCAXIRST (BIT(15))
28#define RCC_DDRITFCR_DDRCORERST (BIT(16))
29#define RCC_DDRITFCR_DPHYAPBRST (BIT(17))
30#define RCC_DDRITFCR_DPHYRST (BIT(18))
31#define RCC_DDRITFCR_DPHYCTLRST (BIT(19))
Marek Vasut5225bb82022-02-25 02:15:58 +010032#define RCC_DDRITFCR_DDRCKMOD_MASK GENMASK(22, 20)
33#define RCC_DDRITFCR_DDRCKMOD_ASR BIT(20)
Patrick Delaunay939d5362018-03-12 10:46:11 +010034
35struct reg_desc {
36 const char *name;
37 u16 offset; /* offset for base address */
38 u8 par_offset; /* offset for parameter array */
39};
40
41#define INVALID_OFFSET 0xFF
42
43#define DDRCTL_REG(x, y) \
44 {#x,\
45 offsetof(struct stm32mp1_ddrctl, x),\
46 offsetof(struct y, x)}
47
48#define DDRPHY_REG(x, y) \
49 {#x,\
50 offsetof(struct stm32mp1_ddrphy, x),\
51 offsetof(struct y, x)}
52
Patrick Delaunaybbee2702019-04-10 14:09:27 +020053#define DDR_REG_DYN(x) \
54 {#x,\
55 offsetof(struct stm32mp1_ddrctl, x),\
56 INVALID_OFFSET}
57
58#define DDRPHY_REG_DYN(x) \
59 {#x,\
60 offsetof(struct stm32mp1_ddrphy, x),\
61 INVALID_OFFSET}
62
Patrick Delaunayd892d272019-04-10 14:09:25 +020063/***********************************************************
64 * PARAMETERS: value get from device tree :
65 * size / order need to be aligned with binding
66 * modification NOT ALLOWED !!!
67 ***********************************************************/
68#define DDRCTL_REG_REG_SIZE 25 /* st,ctl-reg */
69#define DDRCTL_REG_TIMING_SIZE 12 /* st,ctl-timing */
70#define DDRCTL_REG_MAP_SIZE 9 /* st,ctl-map */
Patrick Delaunayd892d272019-04-10 14:09:25 +020071
Marek Vasut3bb89782025-05-12 19:09:05 +020072#define DDRCTL_REG_PERF_SIZE_STM32MP13xx 11 /* st,ctl-perf */
73#define DDRCTL_REG_PERF_SIZE_STM32MP15xx 17 /* st,ctl-perf */
74#define DDRCTL_REG_PERF_SIZE \
75 (IS_ENABLED(CONFIG_STM32MP15X) ? DDRCTL_REG_PERF_SIZE_STM32MP15xx : \
76 DDRCTL_REG_PERF_SIZE_STM32MP13xx)
77
78#define DDRPHY_REG_REG_SIZE_STM32MP13xx 9 /* st,phy-reg */
79#define DDRPHY_REG_REG_SIZE_STM32MP15xx 11 /* st,phy-reg */
80#define DDRPHY_REG_REG_SIZE \
81 (IS_ENABLED(CONFIG_STM32MP15X) ? DDRPHY_REG_REG_SIZE_STM32MP15xx : \
82 DDRPHY_REG_REG_SIZE_STM32MP13xx)
83
Patrick Delaunayd892d272019-04-10 14:09:25 +020084#define DDRPHY_REG_TIMING_SIZE 10 /* st,phy-timing */
Patrick Delaunayd892d272019-04-10 14:09:25 +020085
Patrick Delaunay939d5362018-03-12 10:46:11 +010086#define DDRCTL_REG_REG(x) DDRCTL_REG(x, stm32mp1_ddrctrl_reg)
Patrick Delaunayd892d272019-04-10 14:09:25 +020087static const struct reg_desc ddr_reg[DDRCTL_REG_REG_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +010088 DDRCTL_REG_REG(mstr),
89 DDRCTL_REG_REG(mrctrl0),
90 DDRCTL_REG_REG(mrctrl1),
91 DDRCTL_REG_REG(derateen),
92 DDRCTL_REG_REG(derateint),
93 DDRCTL_REG_REG(pwrctl),
94 DDRCTL_REG_REG(pwrtmg),
95 DDRCTL_REG_REG(hwlpctl),
96 DDRCTL_REG_REG(rfshctl0),
97 DDRCTL_REG_REG(rfshctl3),
98 DDRCTL_REG_REG(crcparctl0),
99 DDRCTL_REG_REG(zqctl0),
100 DDRCTL_REG_REG(dfitmg0),
101 DDRCTL_REG_REG(dfitmg1),
102 DDRCTL_REG_REG(dfilpcfg0),
103 DDRCTL_REG_REG(dfiupd0),
104 DDRCTL_REG_REG(dfiupd1),
105 DDRCTL_REG_REG(dfiupd2),
106 DDRCTL_REG_REG(dfiphymstr),
107 DDRCTL_REG_REG(odtmap),
108 DDRCTL_REG_REG(dbg0),
109 DDRCTL_REG_REG(dbg1),
110 DDRCTL_REG_REG(dbgcmd),
111 DDRCTL_REG_REG(poisoncfg),
112 DDRCTL_REG_REG(pccfg),
113};
114
115#define DDRCTL_REG_TIMING(x) DDRCTL_REG(x, stm32mp1_ddrctrl_timing)
Patrick Delaunayd892d272019-04-10 14:09:25 +0200116static const struct reg_desc ddr_timing[DDRCTL_REG_TIMING_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100117 DDRCTL_REG_TIMING(rfshtmg),
118 DDRCTL_REG_TIMING(dramtmg0),
119 DDRCTL_REG_TIMING(dramtmg1),
120 DDRCTL_REG_TIMING(dramtmg2),
121 DDRCTL_REG_TIMING(dramtmg3),
122 DDRCTL_REG_TIMING(dramtmg4),
123 DDRCTL_REG_TIMING(dramtmg5),
124 DDRCTL_REG_TIMING(dramtmg6),
125 DDRCTL_REG_TIMING(dramtmg7),
126 DDRCTL_REG_TIMING(dramtmg8),
127 DDRCTL_REG_TIMING(dramtmg14),
128 DDRCTL_REG_TIMING(odtcfg),
129};
130
131#define DDRCTL_REG_MAP(x) DDRCTL_REG(x, stm32mp1_ddrctrl_map)
Patrick Delaunayd892d272019-04-10 14:09:25 +0200132static const struct reg_desc ddr_map[DDRCTL_REG_MAP_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100133 DDRCTL_REG_MAP(addrmap1),
134 DDRCTL_REG_MAP(addrmap2),
135 DDRCTL_REG_MAP(addrmap3),
136 DDRCTL_REG_MAP(addrmap4),
137 DDRCTL_REG_MAP(addrmap5),
138 DDRCTL_REG_MAP(addrmap6),
139 DDRCTL_REG_MAP(addrmap9),
140 DDRCTL_REG_MAP(addrmap10),
141 DDRCTL_REG_MAP(addrmap11),
142};
143
144#define DDRCTL_REG_PERF(x) DDRCTL_REG(x, stm32mp1_ddrctrl_perf)
Patrick Delaunayd892d272019-04-10 14:09:25 +0200145static const struct reg_desc ddr_perf[DDRCTL_REG_PERF_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100146 DDRCTL_REG_PERF(sched),
147 DDRCTL_REG_PERF(sched1),
148 DDRCTL_REG_PERF(perfhpr1),
149 DDRCTL_REG_PERF(perflpr1),
150 DDRCTL_REG_PERF(perfwr1),
151 DDRCTL_REG_PERF(pcfgr_0),
152 DDRCTL_REG_PERF(pcfgw_0),
153 DDRCTL_REG_PERF(pcfgqos0_0),
154 DDRCTL_REG_PERF(pcfgqos1_0),
155 DDRCTL_REG_PERF(pcfgwqos0_0),
156 DDRCTL_REG_PERF(pcfgwqos1_0),
Marek Vasut3bb89782025-05-12 19:09:05 +0200157#if IS_ENABLED(CONFIG_STM32MP15X)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100158 DDRCTL_REG_PERF(pcfgr_1),
159 DDRCTL_REG_PERF(pcfgw_1),
160 DDRCTL_REG_PERF(pcfgqos0_1),
161 DDRCTL_REG_PERF(pcfgqos1_1),
162 DDRCTL_REG_PERF(pcfgwqos0_1),
163 DDRCTL_REG_PERF(pcfgwqos1_1),
Marek Vasut3bb89782025-05-12 19:09:05 +0200164#endif
Patrick Delaunay939d5362018-03-12 10:46:11 +0100165};
166
167#define DDRPHY_REG_REG(x) DDRPHY_REG(x, stm32mp1_ddrphy_reg)
Patrick Delaunayd892d272019-04-10 14:09:25 +0200168static const struct reg_desc ddrphy_reg[DDRPHY_REG_REG_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100169 DDRPHY_REG_REG(pgcr),
170 DDRPHY_REG_REG(aciocr),
171 DDRPHY_REG_REG(dxccr),
172 DDRPHY_REG_REG(dsgcr),
173 DDRPHY_REG_REG(dcr),
174 DDRPHY_REG_REG(odtcr),
175 DDRPHY_REG_REG(zq0cr1),
176 DDRPHY_REG_REG(dx0gcr),
177 DDRPHY_REG_REG(dx1gcr),
Marek Vasut3bb89782025-05-12 19:09:05 +0200178#if IS_ENABLED(CONFIG_STM32MP15X)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100179 DDRPHY_REG_REG(dx2gcr),
180 DDRPHY_REG_REG(dx3gcr),
Marek Vasut3bb89782025-05-12 19:09:05 +0200181#endif
Patrick Delaunay939d5362018-03-12 10:46:11 +0100182};
183
184#define DDRPHY_REG_TIMING(x) DDRPHY_REG(x, stm32mp1_ddrphy_timing)
Patrick Delaunayd892d272019-04-10 14:09:25 +0200185static const struct reg_desc ddrphy_timing[DDRPHY_REG_TIMING_SIZE] = {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100186 DDRPHY_REG_TIMING(ptr0),
187 DDRPHY_REG_TIMING(ptr1),
188 DDRPHY_REG_TIMING(ptr2),
189 DDRPHY_REG_TIMING(dtpr0),
190 DDRPHY_REG_TIMING(dtpr1),
191 DDRPHY_REG_TIMING(dtpr2),
192 DDRPHY_REG_TIMING(mr0),
193 DDRPHY_REG_TIMING(mr1),
194 DDRPHY_REG_TIMING(mr2),
195 DDRPHY_REG_TIMING(mr3),
196};
197
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200198/**************************************************************
199 * DYNAMIC REGISTERS: only used for debug purpose (read/modify)
200 **************************************************************/
201#ifdef CONFIG_STM32MP1_DDR_INTERACTIVE
202static const struct reg_desc ddr_dyn[] = {
203 DDR_REG_DYN(stat),
204 DDR_REG_DYN(init0),
205 DDR_REG_DYN(dfimisc),
206 DDR_REG_DYN(dfistat),
207 DDR_REG_DYN(swctl),
208 DDR_REG_DYN(swstat),
209 DDR_REG_DYN(pctrl_0),
210 DDR_REG_DYN(pctrl_1),
211};
212
213#define DDR_REG_DYN_SIZE ARRAY_SIZE(ddr_dyn)
214
215static const struct reg_desc ddrphy_dyn[] = {
216 DDRPHY_REG_DYN(pir),
217 DDRPHY_REG_DYN(pgsr),
218 DDRPHY_REG_DYN(zq0sr0),
219 DDRPHY_REG_DYN(zq0sr1),
220 DDRPHY_REG_DYN(dx0gsr0),
221 DDRPHY_REG_DYN(dx0gsr1),
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100222 DDRPHY_REG_DYN(dx0dllcr),
223 DDRPHY_REG_DYN(dx0dqtr),
224 DDRPHY_REG_DYN(dx0dqstr),
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200225 DDRPHY_REG_DYN(dx1gsr0),
226 DDRPHY_REG_DYN(dx1gsr1),
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100227 DDRPHY_REG_DYN(dx1dllcr),
228 DDRPHY_REG_DYN(dx1dqtr),
229 DDRPHY_REG_DYN(dx1dqstr),
Marek Vasut3bb89782025-05-12 19:09:05 +0200230#if IS_ENABLED(CONFIG_STM32MP15X)
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200231 DDRPHY_REG_DYN(dx2gsr0),
232 DDRPHY_REG_DYN(dx2gsr1),
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100233 DDRPHY_REG_DYN(dx2dllcr),
234 DDRPHY_REG_DYN(dx2dqtr),
235 DDRPHY_REG_DYN(dx2dqstr),
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200236 DDRPHY_REG_DYN(dx3gsr0),
237 DDRPHY_REG_DYN(dx3gsr1),
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100238 DDRPHY_REG_DYN(dx3dllcr),
239 DDRPHY_REG_DYN(dx3dqtr),
240 DDRPHY_REG_DYN(dx3dqstr),
Marek Vasut3bb89782025-05-12 19:09:05 +0200241#endif
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200242};
243
244#define DDRPHY_REG_DYN_SIZE ARRAY_SIZE(ddrphy_dyn)
245
246#endif
247
Patrick Delaunayd892d272019-04-10 14:09:25 +0200248/*****************************************************************
249 * REGISTERS ARRAY: used to parse device tree and interactive mode
250 *****************************************************************/
Patrick Delaunay939d5362018-03-12 10:46:11 +0100251enum reg_type {
252 REG_REG,
253 REG_TIMING,
254 REG_PERF,
255 REG_MAP,
256 REGPHY_REG,
257 REGPHY_TIMING,
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200258#ifdef CONFIG_STM32MP1_DDR_INTERACTIVE
259/* dynamic registers => managed in driver or not changed,
260 * can be dumped in interactive mode
261 */
262 REG_DYN,
263 REGPHY_DYN,
264#endif
Patrick Delaunay939d5362018-03-12 10:46:11 +0100265 REG_TYPE_NB
266};
267
268enum base_type {
269 DDR_BASE,
270 DDRPHY_BASE,
271 NONE_BASE
272};
273
274struct ddr_reg_info {
275 const char *name;
276 const struct reg_desc *desc;
277 u8 size;
278 enum base_type base;
279};
280
Patrick Delaunay939d5362018-03-12 10:46:11 +0100281const struct ddr_reg_info ddr_registers[REG_TYPE_NB] = {
282[REG_REG] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200283 "static", ddr_reg, DDRCTL_REG_REG_SIZE, DDR_BASE},
Patrick Delaunay939d5362018-03-12 10:46:11 +0100284[REG_TIMING] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200285 "timing", ddr_timing, DDRCTL_REG_TIMING_SIZE, DDR_BASE},
Patrick Delaunay939d5362018-03-12 10:46:11 +0100286[REG_PERF] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200287 "perf", ddr_perf, DDRCTL_REG_PERF_SIZE, DDR_BASE},
Patrick Delaunay939d5362018-03-12 10:46:11 +0100288[REG_MAP] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200289 "map", ddr_map, DDRCTL_REG_MAP_SIZE, DDR_BASE},
Patrick Delaunay939d5362018-03-12 10:46:11 +0100290[REGPHY_REG] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200291 "static", ddrphy_reg, DDRPHY_REG_REG_SIZE, DDRPHY_BASE},
Patrick Delaunay939d5362018-03-12 10:46:11 +0100292[REGPHY_TIMING] = {
Patrick Delaunayd892d272019-04-10 14:09:25 +0200293 "timing", ddrphy_timing, DDRPHY_REG_TIMING_SIZE, DDRPHY_BASE},
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200294#ifdef CONFIG_STM32MP1_DDR_INTERACTIVE
295[REG_DYN] = {
296 "dyn", ddr_dyn, DDR_REG_DYN_SIZE, DDR_BASE},
297[REGPHY_DYN] = {
298 "dyn", ddrphy_dyn, DDRPHY_REG_DYN_SIZE, DDRPHY_BASE},
299#endif
300
Patrick Delaunay939d5362018-03-12 10:46:11 +0100301};
302
303const char *base_name[] = {
304 [DDR_BASE] = "ctl",
305 [DDRPHY_BASE] = "phy",
306};
307
Marek Vasut3bb89782025-05-12 19:09:05 +0200308bool is_stm32mp13_ddrc(const struct ddr_info *priv)
309{
310 if (IS_ENABLED(CONFIG_STM32MP13X) && !IS_ENABLED(CONFIG_STM32MP15X))
311 return true; /* STM32MP13xx only build */
312 else if (!IS_ENABLED(CONFIG_STM32MP13X) && IS_ENABLED(CONFIG_STM32MP15X))
313 return false; /* STM32MP15xx only build */
314
315 /* Combined STM32MP13xx and STM32MP15xx build */
316 return device_is_compatible(priv->dev, "st,stm32mp13-ddr");
317}
318
319static u32 get_rcc_ddritfcr(const struct ddr_info *priv)
320{
321 return priv->rcc + (is_stm32mp13_ddrc(priv) ?
322 RCC_DDRITFCR_STM32MP13xx :
323 RCC_DDRITFCR_STM32MP15xx);
324}
325
Patrick Delaunay939d5362018-03-12 10:46:11 +0100326static u32 get_base_addr(const struct ddr_info *priv, enum base_type base)
327{
328 if (base == DDRPHY_BASE)
329 return (u32)priv->phy;
330 else
331 return (u32)priv->ctl;
332}
333
Marek Vasut3bb89782025-05-12 19:09:05 +0200334static u32 get_type_size(const struct ddr_info *priv, enum reg_type type)
335{
336 bool is_mp13 = is_stm32mp13_ddrc(priv);
337
338 if (type == REG_PERF)
339 return is_mp13 ? DDRCTL_REG_PERF_SIZE_STM32MP13xx :
340 DDRCTL_REG_PERF_SIZE_STM32MP15xx;
341 else if (type == REGPHY_REG)
342 return is_mp13 ? DDRPHY_REG_REG_SIZE_STM32MP13xx :
343 DDRPHY_REG_REG_SIZE_STM32MP15xx;
344
345 /* Everything else is the default size */
346 return ddr_registers[type].size;
347}
348
Patrick Delaunay939d5362018-03-12 10:46:11 +0100349static void set_reg(const struct ddr_info *priv,
350 enum reg_type type,
351 const void *param)
352{
353 unsigned int i;
354 unsigned int *ptr, value;
355 enum base_type base = ddr_registers[type].base;
356 u32 base_addr = get_base_addr(priv, base);
357 const struct reg_desc *desc = ddr_registers[type].desc;
Marek Vasut3bb89782025-05-12 19:09:05 +0200358 u32 size = get_type_size(priv, type);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100359
Patrick Delaunay753f5362020-11-06 19:01:36 +0100360 log_debug("init %s\n", ddr_registers[type].name);
Marek Vasut3bb89782025-05-12 19:09:05 +0200361 for (i = 0; i < size; i++) {
Patrick Delaunay939d5362018-03-12 10:46:11 +0100362 ptr = (unsigned int *)(base_addr + desc[i].offset);
363 if (desc[i].par_offset == INVALID_OFFSET) {
Patrick Delaunay753f5362020-11-06 19:01:36 +0100364 log_err("invalid parameter offset for %s", desc[i].name);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100365 } else {
366 value = *((u32 *)((u32)param +
367 desc[i].par_offset));
368 writel(value, ptr);
Patrick Delaunay753f5362020-11-06 19:01:36 +0100369 log_debug("[0x%x] %s= 0x%08x\n",
370 (u32)ptr, desc[i].name, value);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100371 }
372 }
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200373}
374
375#ifdef CONFIG_STM32MP1_DDR_INTERACTIVE
376static void stm32mp1_dump_reg_desc(u32 base_addr, const struct reg_desc *desc)
377{
378 unsigned int *ptr;
379
380 ptr = (unsigned int *)(base_addr + desc->offset);
381 printf("%s= 0x%08x\n", desc->name, readl(ptr));
382}
383
384static void stm32mp1_dump_param_desc(u32 par_addr, const struct reg_desc *desc)
385{
386 unsigned int *ptr;
387
388 ptr = (unsigned int *)(par_addr + desc->par_offset);
389 printf("%s= 0x%08x\n", desc->name, readl(ptr));
390}
391
392static const struct reg_desc *found_reg(const char *name, enum reg_type *type)
393{
394 unsigned int i, j;
395 const struct reg_desc *desc;
396
397 for (i = 0; i < ARRAY_SIZE(ddr_registers); i++) {
398 desc = ddr_registers[i].desc;
399 for (j = 0; j < ddr_registers[i].size; j++) {
400 if (strcmp(name, desc[j].name) == 0) {
401 *type = i;
402 return &desc[j];
403 }
404 }
405 }
406 *type = REG_TYPE_NB;
407 return NULL;
408}
409
410int stm32mp1_dump_reg(const struct ddr_info *priv,
411 const char *name)
412{
413 unsigned int i, j;
414 const struct reg_desc *desc;
415 u32 base_addr;
416 enum base_type p_base;
417 enum reg_type type;
418 const char *p_name;
419 enum base_type filter = NONE_BASE;
420 int result = -1;
421
422 if (name) {
423 if (strcmp(name, base_name[DDR_BASE]) == 0)
424 filter = DDR_BASE;
425 else if (strcmp(name, base_name[DDRPHY_BASE]) == 0)
426 filter = DDRPHY_BASE;
427 }
428
429 for (i = 0; i < ARRAY_SIZE(ddr_registers); i++) {
430 p_base = ddr_registers[i].base;
431 p_name = ddr_registers[i].name;
432 if (!name || (filter == p_base || !strcmp(name, p_name))) {
433 result = 0;
434 desc = ddr_registers[i].desc;
435 base_addr = get_base_addr(priv, p_base);
436 printf("==%s.%s==\n", base_name[p_base], p_name);
437 for (j = 0; j < ddr_registers[i].size; j++)
438 stm32mp1_dump_reg_desc(base_addr, &desc[j]);
439 }
440 }
441 if (result) {
442 desc = found_reg(name, &type);
443 if (desc) {
444 p_base = ddr_registers[type].base;
445 base_addr = get_base_addr(priv, p_base);
446 stm32mp1_dump_reg_desc(base_addr, desc);
447 result = 0;
448 }
449 }
450 return result;
Patrick Delaunay939d5362018-03-12 10:46:11 +0100451}
452
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200453void stm32mp1_edit_reg(const struct ddr_info *priv,
454 char *name, char *string)
455{
456 unsigned long *ptr, value;
457 enum reg_type type;
458 enum base_type base;
459 const struct reg_desc *desc;
460 u32 base_addr;
461
462 desc = found_reg(name, &type);
463
464 if (!desc) {
465 printf("%s not found\n", name);
466 return;
467 }
468 if (strict_strtoul(string, 16, &value) < 0) {
469 printf("invalid value %s\n", string);
470 return;
471 }
472 base = ddr_registers[type].base;
473 base_addr = get_base_addr(priv, base);
474 ptr = (unsigned long *)(base_addr + desc->offset);
475 writel(value, ptr);
476 printf("%s= 0x%08x\n", desc->name, readl(ptr));
477}
478
479static u32 get_par_addr(const struct stm32mp1_ddr_config *config,
480 enum reg_type type)
481{
482 u32 par_addr = 0x0;
483
484 switch (type) {
485 case REG_REG:
486 par_addr = (u32)&config->c_reg;
487 break;
488 case REG_TIMING:
489 par_addr = (u32)&config->c_timing;
490 break;
491 case REG_PERF:
492 par_addr = (u32)&config->c_perf;
493 break;
494 case REG_MAP:
495 par_addr = (u32)&config->c_map;
496 break;
497 case REGPHY_REG:
498 par_addr = (u32)&config->p_reg;
499 break;
500 case REGPHY_TIMING:
501 par_addr = (u32)&config->p_timing;
502 break;
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200503 case REG_DYN:
504 case REGPHY_DYN:
505 case REG_TYPE_NB:
506 par_addr = (u32)NULL;
507 break;
508 }
509
510 return par_addr;
511}
512
513int stm32mp1_dump_param(const struct stm32mp1_ddr_config *config,
514 const char *name)
515{
516 unsigned int i, j;
517 const struct reg_desc *desc;
518 u32 par_addr;
519 enum base_type p_base;
520 enum reg_type type;
521 const char *p_name;
522 enum base_type filter = NONE_BASE;
523 int result = -EINVAL;
524
525 if (name) {
526 if (strcmp(name, base_name[DDR_BASE]) == 0)
527 filter = DDR_BASE;
528 else if (strcmp(name, base_name[DDRPHY_BASE]) == 0)
529 filter = DDRPHY_BASE;
530 }
531
532 for (i = 0; i < ARRAY_SIZE(ddr_registers); i++) {
533 par_addr = get_par_addr(config, i);
534 if (!par_addr)
535 continue;
536 p_base = ddr_registers[i].base;
537 p_name = ddr_registers[i].name;
538 if (!name || (filter == p_base || !strcmp(name, p_name))) {
539 result = 0;
540 desc = ddr_registers[i].desc;
541 printf("==%s.%s==\n", base_name[p_base], p_name);
542 for (j = 0; j < ddr_registers[i].size; j++)
543 stm32mp1_dump_param_desc(par_addr, &desc[j]);
544 }
545 }
546 if (result) {
547 desc = found_reg(name, &type);
548 if (desc) {
549 par_addr = get_par_addr(config, type);
550 if (par_addr) {
551 stm32mp1_dump_param_desc(par_addr, desc);
552 result = 0;
553 }
554 }
555 }
556 return result;
557}
558
559void stm32mp1_edit_param(const struct stm32mp1_ddr_config *config,
560 char *name, char *string)
561{
562 unsigned long *ptr, value;
563 enum reg_type type;
564 const struct reg_desc *desc;
565 u32 par_addr;
566
567 desc = found_reg(name, &type);
568 if (!desc) {
569 printf("%s not found\n", name);
570 return;
571 }
572 if (strict_strtoul(string, 16, &value) < 0) {
573 printf("invalid value %s\n", string);
574 return;
575 }
576 par_addr = get_par_addr(config, type);
577 if (!par_addr) {
578 printf("no parameter %s\n", name);
579 return;
580 }
581 ptr = (unsigned long *)(par_addr + desc->par_offset);
582 writel(value, ptr);
583 printf("%s= 0x%08x\n", desc->name, readl(ptr));
584}
585#endif
586
587__weak bool stm32mp1_ddr_interactive(void *priv,
588 enum stm32mp1_ddr_interact_step step,
589 const struct stm32mp1_ddr_config *config)
590{
591 return false;
592}
593
594#define INTERACTIVE(step)\
595 stm32mp1_ddr_interactive(priv, step, config)
596
Patrick Delaunay939d5362018-03-12 10:46:11 +0100597static void ddrphy_idone_wait(struct stm32mp1_ddrphy *phy)
598{
599 u32 pgsr;
600 int ret;
601
602 ret = readl_poll_timeout(&phy->pgsr, pgsr,
603 pgsr & (DDRPHYC_PGSR_IDONE |
604 DDRPHYC_PGSR_DTERR |
605 DDRPHYC_PGSR_DTIERR |
606 DDRPHYC_PGSR_DFTERR |
607 DDRPHYC_PGSR_RVERR |
608 DDRPHYC_PGSR_RVEIRR),
609 1000000);
Patrick Delaunay753f5362020-11-06 19:01:36 +0100610 log_debug("\n[0x%08x] pgsr = 0x%08x ret=%d\n",
611 (u32)&phy->pgsr, pgsr, ret);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100612}
613
Patrick Delaunay0c4354e2021-11-15 15:32:30 +0100614static void stm32mp1_ddrphy_init(struct stm32mp1_ddrphy *phy, u32 pir)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100615{
616 pir |= DDRPHYC_PIR_INIT;
617 writel(pir, &phy->pir);
Patrick Delaunay753f5362020-11-06 19:01:36 +0100618 log_debug("[0x%08x] pir = 0x%08x -> 0x%08x\n",
619 (u32)&phy->pir, pir, readl(&phy->pir));
Patrick Delaunay939d5362018-03-12 10:46:11 +0100620
621 /* need to wait 10 configuration clock before start polling */
622 udelay(10);
623
624 /* Wait DRAM initialization and Gate Training Evaluation complete */
625 ddrphy_idone_wait(phy);
626}
627
628/* start quasi dynamic register update */
629static void start_sw_done(struct stm32mp1_ddrctl *ctl)
630{
631 clrbits_le32(&ctl->swctl, DDRCTRL_SWCTL_SW_DONE);
632}
633
634/* wait quasi dynamic register update */
635static void wait_sw_done_ack(struct stm32mp1_ddrctl *ctl)
636{
637 int ret;
638 u32 swstat;
639
640 setbits_le32(&ctl->swctl, DDRCTRL_SWCTL_SW_DONE);
641
642 ret = readl_poll_timeout(&ctl->swstat, swstat,
643 swstat & DDRCTRL_SWSTAT_SW_DONE_ACK,
644 1000000);
645 if (ret)
646 panic("Timeout initialising DRAM : DDR->swstat = %x\n",
647 swstat);
648
Patrick Delaunay753f5362020-11-06 19:01:36 +0100649 log_debug("[0x%08x] swstat = 0x%08x\n", (u32)&ctl->swstat, swstat);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100650}
651
652/* wait quasi dynamic register update */
653static void wait_operating_mode(struct ddr_info *priv, int mode)
654{
655 u32 stat, val, mask, val2 = 0, mask2 = 0;
656 int ret;
657
658 mask = DDRCTRL_STAT_OPERATING_MODE_MASK;
659 val = mode;
660 /* self-refresh due to software => check also STAT.selfref_type */
661 if (mode == DDRCTRL_STAT_OPERATING_MODE_SR) {
662 mask |= DDRCTRL_STAT_SELFREF_TYPE_MASK;
Patrick Delaunay250bf902019-04-10 14:09:21 +0200663 val |= DDRCTRL_STAT_SELFREF_TYPE_SR;
Patrick Delaunay939d5362018-03-12 10:46:11 +0100664 } else if (mode == DDRCTRL_STAT_OPERATING_MODE_NORMAL) {
665 /* normal mode: handle also automatic self refresh */
666 mask2 = DDRCTRL_STAT_OPERATING_MODE_MASK |
667 DDRCTRL_STAT_SELFREF_TYPE_MASK;
668 val2 = DDRCTRL_STAT_OPERATING_MODE_SR |
669 DDRCTRL_STAT_SELFREF_TYPE_ASR;
670 }
671
672 ret = readl_poll_timeout(&priv->ctl->stat, stat,
673 ((stat & mask) == val) ||
674 (mask2 && ((stat & mask2) == val2)),
675 1000000);
676
677 if (ret)
678 panic("Timeout DRAM : DDR->stat = %x\n", stat);
679
Patrick Delaunay753f5362020-11-06 19:01:36 +0100680 log_debug("[0x%08x] stat = 0x%08x\n", (u32)&priv->ctl->stat, stat);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100681}
682
Patrick Delaunay0c4354e2021-11-15 15:32:30 +0100683static void stm32mp1_refresh_disable(struct stm32mp1_ddrctl *ctl)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100684{
685 start_sw_done(ctl);
686 /* quasi-dynamic register update*/
687 setbits_le32(&ctl->rfshctl3, DDRCTRL_RFSHCTL3_DIS_AUTO_REFRESH);
Patrick Delaunayeca9bc02020-03-06 11:14:09 +0100688 clrbits_le32(&ctl->pwrctl, DDRCTRL_PWRCTL_POWERDOWN_EN |
689 DDRCTRL_PWRCTL_SELFREF_EN);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100690 clrbits_le32(&ctl->dfimisc, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
691 wait_sw_done_ack(ctl);
692}
693
Patrick Delaunay0c4354e2021-11-15 15:32:30 +0100694static void stm32mp1_refresh_restore(struct stm32mp1_ddrctl *ctl,
695 u32 rfshctl3, u32 pwrctl)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100696{
697 start_sw_done(ctl);
698 if (!(rfshctl3 & DDRCTRL_RFSHCTL3_DIS_AUTO_REFRESH))
699 clrbits_le32(&ctl->rfshctl3, DDRCTRL_RFSHCTL3_DIS_AUTO_REFRESH);
700 if (pwrctl & DDRCTRL_PWRCTL_POWERDOWN_EN)
701 setbits_le32(&ctl->pwrctl, DDRCTRL_PWRCTL_POWERDOWN_EN);
Patrick Delaunayeca9bc02020-03-06 11:14:09 +0100702 if ((pwrctl & DDRCTRL_PWRCTL_SELFREF_EN))
703 setbits_le32(&ctl->pwrctl, DDRCTRL_PWRCTL_SELFREF_EN);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100704 setbits_le32(&ctl->dfimisc, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
705 wait_sw_done_ack(ctl);
706}
707
Marek Vasut817bc472022-04-26 16:37:41 +0200708static void stm32mp1_asr_enable(struct ddr_info *priv, const u32 pwrctl)
Marek Vasut5225bb82022-02-25 02:15:58 +0100709{
710 struct stm32mp1_ddrctl *ctl = priv->ctl;
Marek Vasut3bb89782025-05-12 19:09:05 +0200711 u32 rcc_ddritfcr = get_rcc_ddritfcr(priv);
Marek Vasut5225bb82022-02-25 02:15:58 +0100712
Marek Vasut817bc472022-04-26 16:37:41 +0200713 /* SSR is the best we can do. */
714 if (!(pwrctl & DDRCTRL_PWRCTL_EN_DFI_DRAM_CLK_DISABLE))
715 return;
716
Marek Vasut3bb89782025-05-12 19:09:05 +0200717 clrsetbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCKMOD_MASK,
Marek Vasut5225bb82022-02-25 02:15:58 +0100718 RCC_DDRITFCR_DDRCKMOD_ASR);
719
720 start_sw_done(ctl);
721
722 setbits_le32(&ctl->hwlpctl, DDRCTRL_HWLPCTL_HW_LP_EN);
723 writel(DDRCTRL_PWRTMG_POWERDOWN_TO_X32(0x10) |
724 DDRCTRL_PWRTMG_SELFREF_TO_X32(0x01),
725 &ctl->pwrtmg);
Marek Vasut817bc472022-04-26 16:37:41 +0200726
727 /* HSR we can do. */
Marek Vasut5225bb82022-02-25 02:15:58 +0100728 setbits_le32(&ctl->pwrctl, DDRCTRL_PWRCTL_EN_DFI_DRAM_CLK_DISABLE);
Marek Vasut817bc472022-04-26 16:37:41 +0200729
730 if (pwrctl & DDRCTRL_PWRCTL_SELFREF_EN) /* ASR we can do. */
731 setbits_le32(&ctl->pwrctl, DDRCTRL_PWRCTL_SELFREF_EN);
Marek Vasut5225bb82022-02-25 02:15:58 +0100732
733 setbits_le32(&ctl->dfimisc, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
734 wait_sw_done_ack(ctl);
735}
736
Patrick Delaunay939d5362018-03-12 10:46:11 +0100737/* board-specific DDR power initializations. */
Patrick Delaunay4175f452019-04-10 14:09:26 +0200738__weak int board_ddr_power_init(enum ddr_type ddr_type)
Patrick Delaunay939d5362018-03-12 10:46:11 +0100739{
740 return 0;
741}
742
743__maybe_unused
744void stm32mp1_ddr_init(struct ddr_info *priv,
745 const struct stm32mp1_ddr_config *config)
746{
Marek Vasut3bb89782025-05-12 19:09:05 +0200747 u32 rcc_ddritfcr = get_rcc_ddritfcr(priv);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100748 u32 pir;
Patrick Delaunay4175f452019-04-10 14:09:26 +0200749 int ret = -EINVAL;
Patrick Delaunay2ebc2112020-03-06 11:14:03 +0100750 char bus_width;
751
752 switch (config->c_reg.mstr & DDRCTRL_MSTR_DATA_BUS_WIDTH_MASK) {
753 case DDRCTRL_MSTR_DATA_BUS_WIDTH_QUARTER:
754 bus_width = 8;
755 break;
756 case DDRCTRL_MSTR_DATA_BUS_WIDTH_HALF:
757 bus_width = 16;
758 break;
759 default:
760 bus_width = 32;
761 break;
762 }
763
Patrick Delaunay4175f452019-04-10 14:09:26 +0200764 if (config->c_reg.mstr & DDRCTRL_MSTR_DDR3)
765 ret = board_ddr_power_init(STM32MP_DDR3);
Patrick Delaunay2ebc2112020-03-06 11:14:03 +0100766 else if (config->c_reg.mstr & DDRCTRL_MSTR_LPDDR2) {
767 if (bus_width == 32)
768 ret = board_ddr_power_init(STM32MP_LPDDR2_32);
769 else
770 ret = board_ddr_power_init(STM32MP_LPDDR2_16);
771 } else if (config->c_reg.mstr & DDRCTRL_MSTR_LPDDR3) {
772 if (bus_width == 32)
773 ret = board_ddr_power_init(STM32MP_LPDDR3_32);
774 else
775 ret = board_ddr_power_init(STM32MP_LPDDR3_16);
776 }
Patrick Delaunay939d5362018-03-12 10:46:11 +0100777 if (ret)
778 panic("ddr power init failed\n");
779
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200780start:
Patrick Delaunay753f5362020-11-06 19:01:36 +0100781 log_debug("name = %s\n", config->info.name);
782 log_debug("speed = %d kHz\n", config->info.speed);
783 log_debug("size = 0x%x\n", config->info.size);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100784/*
785 * 1. Program the DWC_ddr_umctl2 registers
786 * 1.1 RESETS: presetn, core_ddrc_rstn, aresetn
787 */
788 /* Assert All DDR part */
Marek Vasut3bb89782025-05-12 19:09:05 +0200789 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCAPBRST);
790 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCAXIRST);
791 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCORERST);
792 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYAPBRST);
793 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYRST);
794 setbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYCTLRST);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100795
796/* 1.2. start CLOCK */
797 if (stm32mp1_ddr_clk_enable(priv, config->info.speed))
Patrick Delaunay29e1a942019-04-10 14:09:23 +0200798 panic("invalid DRAM clock : %d kHz\n",
Patrick Delaunay939d5362018-03-12 10:46:11 +0100799 config->info.speed);
800
801/* 1.3. deassert reset */
802 /* de-assert PHY rstn and ctl_rstn via DPHYRST and DPHYCTLRST */
Marek Vasut3bb89782025-05-12 19:09:05 +0200803 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYRST);
804 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYCTLRST);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100805 /* De-assert presetn once the clocks are active
806 * and stable via DDRCAPBRST bit
807 */
Marek Vasut3bb89782025-05-12 19:09:05 +0200808 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCAPBRST);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100809
Patrick Delaunay1f987042019-04-10 14:09:22 +0200810/* 1.4. wait 128 cycles to permit initialization of end logic */
811 udelay(2);
812 /* for PCLK = 133MHz => 1 us is enough, 2 to allow lower frequency */
Patrick Delaunay939d5362018-03-12 10:46:11 +0100813
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200814 if (INTERACTIVE(STEP_DDR_RESET))
815 goto start;
816
Patrick Delaunay939d5362018-03-12 10:46:11 +0100817/* 1.5. initialize registers ddr_umctl2 */
818 /* Stop uMCTL2 before PHY is ready */
819 clrbits_le32(&priv->ctl->dfimisc, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
Patrick Delaunay753f5362020-11-06 19:01:36 +0100820 log_debug("[0x%08x] dfimisc = 0x%08x\n",
821 (u32)&priv->ctl->dfimisc, readl(&priv->ctl->dfimisc));
Patrick Delaunay939d5362018-03-12 10:46:11 +0100822
823 set_reg(priv, REG_REG, &config->c_reg);
824 set_reg(priv, REG_TIMING, &config->c_timing);
825 set_reg(priv, REG_MAP, &config->c_map);
826
827 /* skip CTRL init, SDRAM init is done by PHY PUBL */
828 clrsetbits_le32(&priv->ctl->init0,
829 DDRCTRL_INIT0_SKIP_DRAM_INIT_MASK,
830 DDRCTRL_INIT0_SKIP_DRAM_INIT_NORMAL);
831
832 set_reg(priv, REG_PERF, &config->c_perf);
833
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200834 if (INTERACTIVE(STEP_CTL_INIT))
835 goto start;
836
Patrick Delaunay939d5362018-03-12 10:46:11 +0100837/* 2. deassert reset signal core_ddrc_rstn, aresetn and presetn */
Marek Vasut3bb89782025-05-12 19:09:05 +0200838 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCORERST);
839 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DDRCAXIRST);
840 clrbits_le32(rcc_ddritfcr, RCC_DDRITFCR_DPHYAPBRST);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100841
842/* 3. start PHY init by accessing relevant PUBL registers
843 * (DXGCR, DCR, PTR*, MR*, DTPR*)
844 */
845 set_reg(priv, REGPHY_REG, &config->p_reg);
846 set_reg(priv, REGPHY_TIMING, &config->p_timing);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100847
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200848 if (INTERACTIVE(STEP_PHY_INIT))
849 goto start;
850
Patrick Delaunay939d5362018-03-12 10:46:11 +0100851/* 4. Monitor PHY init status by polling PUBL register PGSR.IDONE
852 * Perform DDR PHY DRAM initialization and Gate Training Evaluation
853 */
854 ddrphy_idone_wait(priv->phy);
855
856/* 5. Indicate to PUBL that controller performs SDRAM initialization
857 * by setting PIR.INIT and PIR CTLDINIT and pool PGSR.IDONE
858 * DRAM init is done by PHY, init0.skip_dram.init = 1
859 */
860 pir = DDRPHYC_PIR_DLLSRST | DDRPHYC_PIR_DLLLOCK | DDRPHYC_PIR_ZCAL |
861 DDRPHYC_PIR_ITMSRST | DDRPHYC_PIR_DRAMINIT | DDRPHYC_PIR_ICPC;
862
863 if (config->c_reg.mstr & DDRCTRL_MSTR_DDR3)
864 pir |= DDRPHYC_PIR_DRAMRST; /* only for DDR3 */
865
866 stm32mp1_ddrphy_init(priv->phy, pir);
867
868/* 6. SET DFIMISC.dfi_init_complete_en to 1 */
869 /* Enable quasi-dynamic register programming*/
870 start_sw_done(priv->ctl);
871 setbits_le32(&priv->ctl->dfimisc, DDRCTRL_DFIMISC_DFI_INIT_COMPLETE_EN);
872 wait_sw_done_ack(priv->ctl);
873
874/* 7. Wait for DWC_ddr_umctl2 to move to normal operation mode
875 * by monitoring STAT.operating_mode signal
876 */
877 /* wait uMCTL2 ready */
878
879 wait_operating_mode(priv, DDRCTRL_STAT_OPERATING_MODE_NORMAL);
880
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100881 log_debug("DDR DQS training : ");
Patrick Delaunay939d5362018-03-12 10:46:11 +0100882/* 8. Disable Auto refresh and power down by setting
883 * - RFSHCTL3.dis_au_refresh = 1
884 * - PWRCTL.powerdown_en = 0
885 * - DFIMISC.dfiinit_complete_en = 0
886 */
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100887 stm32mp1_refresh_disable(priv->ctl);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100888
889/* 9. Program PUBL PGCR to enable refresh during training and rank to train
890 * not done => keep the programed value in PGCR
891 */
892
893/* 10. configure PUBL PIR register to specify which training step to run */
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100894 /* RVTRN is excuted only on LPDDR2/LPDDR3 */
895 if (config->c_reg.mstr & DDRCTRL_MSTR_DDR3)
896 pir = DDRPHYC_PIR_QSTRN;
897 else
898 pir = DDRPHYC_PIR_QSTRN | DDRPHYC_PIR_RVTRN;
899 stm32mp1_ddrphy_init(priv->phy, pir);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100900
901/* 11. monitor PUB PGSR.IDONE to poll cpmpletion of training sequence */
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100902 ddrphy_idone_wait(priv->phy);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100903
904/* 12. set back registers in step 8 to the orginal values if desidered */
Patrick Delaunay9e2dd662021-11-15 15:32:29 +0100905 stm32mp1_refresh_restore(priv->ctl, config->c_reg.rfshctl3,
906 config->c_reg.pwrctl);
Patrick Delaunay939d5362018-03-12 10:46:11 +0100907
Marek Vasut5225bb82022-02-25 02:15:58 +0100908/* Enable auto-self-refresh, which saves a bit of power at runtime. */
Marek Vasut817bc472022-04-26 16:37:41 +0200909 stm32mp1_asr_enable(priv, config->c_reg.pwrctl);
Marek Vasut5225bb82022-02-25 02:15:58 +0100910
Marek Vasut3bb89782025-05-12 19:09:05 +0200911 /* enable uMCTL2 AXI port 0 */
Patrick Delaunay939d5362018-03-12 10:46:11 +0100912 setbits_le32(&priv->ctl->pctrl_0, DDRCTRL_PCTRL_N_PORT_EN);
Marek Vasut3bb89782025-05-12 19:09:05 +0200913
914 /* enable uMCTL2 AXI port 1 only on STM32MP15xx with 32bit DRAM bus */
915 if (!is_stm32mp13_ddrc(priv))
916 setbits_le32(&priv->ctl->pctrl_1, DDRCTRL_PCTRL_N_PORT_EN);
Patrick Delaunaybbee2702019-04-10 14:09:27 +0200917
918 if (INTERACTIVE(STEP_DDR_READY))
919 goto start;
Patrick Delaunay939d5362018-03-12 10:46:11 +0100920}