blob: 6fb89fbe3d65770fea2757dbdc7dc60d85d6c6a0 [file] [log] [blame]
Vikas Manochabf1ae442017-04-10 15:02:51 -07001/*
2 * (C) Copyright 2017
3 * Vikas Manocha, <vikas.manocha@st.com>
4 *
5 * SPDX-License-Identifier: GPL-2.0+
6 */
7
8#include <common.h>
Vikas Manochad0b24c12017-04-10 15:02:55 -07009#include <clk.h>
Vikas Manocha910a52e2017-04-10 15:02:52 -070010#include <dm.h>
11#include <ram.h>
Vikas Manochabf1ae442017-04-10 15:02:51 -070012#include <asm/io.h>
Vikas Manochabf1ae442017-04-10 15:02:51 -070013
Vikas Manocha6c9a1002017-04-10 15:02:56 -070014DECLARE_GLOBAL_DATA_PTR;
15
Patrice Chotard9242ece2017-07-18 17:37:24 +020016struct stm32_fmc_regs {
Patrice Chotard1421e0a2017-07-18 17:37:25 +020017 /* 0x0 */
18 u32 bcr1; /* NOR/PSRAM Chip select control register 1 */
19 u32 btr1; /* SRAM/NOR-Flash Chip select timing register 1 */
20 u32 bcr2; /* NOR/PSRAM Chip select Control register 2 */
21 u32 btr2; /* SRAM/NOR-Flash Chip select timing register 2 */
22 u32 bcr3; /* NOR/PSRAMChip select Control register 3 */
23 u32 btr3; /* SRAM/NOR-Flash Chip select timing register 3 */
24 u32 bcr4; /* NOR/PSRAM Chip select Control register 4 */
25 u32 btr4; /* SRAM/NOR-Flash Chip select timing register 4 */
26 u32 reserved1[24];
Patrice Chotard9242ece2017-07-18 17:37:24 +020027
Patrice Chotard1421e0a2017-07-18 17:37:25 +020028 /* 0x80 */
29 u32 pcr; /* NAND Flash control register */
30 u32 sr; /* FIFO status and interrupt register */
31 u32 pmem; /* Common memory space timing register */
32 u32 patt; /* Attribute memory space timing registers */
33 u32 reserved2[1];
34 u32 eccr; /* ECC result registers */
35 u32 reserved3[27];
36
37 /* 0x104 */
38 u32 bwtr1; /* SRAM/NOR-Flash write timing register 1 */
39 u32 reserved4[1];
40 u32 bwtr2; /* SRAM/NOR-Flash write timing register 2 */
41 u32 reserved5[1];
42 u32 bwtr3; /* SRAM/NOR-Flash write timing register 3 */
43 u32 reserved6[1];
44 u32 bwtr4; /* SRAM/NOR-Flash write timing register 4 */
45 u32 reserved7[8];
46
47 /* 0x140 */
48 u32 sdcr1; /* SDRAM Control register 1 */
49 u32 sdcr2; /* SDRAM Control register 2 */
50 u32 sdtr1; /* SDRAM Timing register 1 */
51 u32 sdtr2; /* SDRAM Timing register 2 */
52 u32 sdcmr; /* SDRAM Mode register */
53 u32 sdrtr; /* SDRAM Refresh timing register */
54 u32 sdsr; /* SDRAM Status register */
55};
Patrice Chotard9242ece2017-07-18 17:37:24 +020056
57/* Control register SDCR */
58#define FMC_SDCR_RPIPE_SHIFT 13 /* RPIPE bit shift */
59#define FMC_SDCR_RBURST_SHIFT 12 /* RBURST bit shift */
60#define FMC_SDCR_SDCLK_SHIFT 10 /* SDRAM clock divisor shift */
61#define FMC_SDCR_WP_SHIFT 9 /* Write protection shift */
62#define FMC_SDCR_CAS_SHIFT 7 /* CAS latency shift */
63#define FMC_SDCR_NB_SHIFT 6 /* Number of banks shift */
64#define FMC_SDCR_MWID_SHIFT 4 /* Memory width shift */
65#define FMC_SDCR_NR_SHIFT 2 /* Number of row address bits shift */
66#define FMC_SDCR_NC_SHIFT 0 /* Number of col address bits shift */
67
68/* Timings register SDTR */
69#define FMC_SDTR_TMRD_SHIFT 0 /* Load mode register to active */
70#define FMC_SDTR_TXSR_SHIFT 4 /* Exit self-refresh time */
71#define FMC_SDTR_TRAS_SHIFT 8 /* Self-refresh time */
72#define FMC_SDTR_TRC_SHIFT 12 /* Row cycle delay */
73#define FMC_SDTR_TWR_SHIFT 16 /* Recovery delay */
74#define FMC_SDTR_TRP_SHIFT 20 /* Row precharge delay */
75#define FMC_SDTR_TRCD_SHIFT 24 /* Row-to-column delay */
76
77#define FMC_SDCMR_NRFS_SHIFT 5
78
79#define FMC_SDCMR_MODE_NORMAL 0
80#define FMC_SDCMR_MODE_START_CLOCK 1
81#define FMC_SDCMR_MODE_PRECHARGE 2
82#define FMC_SDCMR_MODE_AUTOREFRESH 3
83#define FMC_SDCMR_MODE_WRITE_MODE 4
84#define FMC_SDCMR_MODE_SELFREFRESH 5
85#define FMC_SDCMR_MODE_POWERDOWN 6
86
87#define FMC_SDCMR_BANK_1 BIT(4)
88#define FMC_SDCMR_BANK_2 BIT(3)
89
90#define FMC_SDCMR_MODE_REGISTER_SHIFT 9
91
92#define FMC_SDSR_BUSY BIT(5)
93
Patrice Chotard1421e0a2017-07-18 17:37:25 +020094#define FMC_BUSY_WAIT(regs) do { \
Patrice Chotard9242ece2017-07-18 17:37:24 +020095 __asm__ __volatile__ ("dsb" : : : "memory"); \
Patrice Chotard1421e0a2017-07-18 17:37:25 +020096 while (regs->sdsr & FMC_SDSR_BUSY) \
Patrice Chotard9242ece2017-07-18 17:37:24 +020097 ; \
98 } while (0)
99
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700100struct stm32_sdram_control {
101 u8 no_columns;
102 u8 no_rows;
103 u8 memory_width;
104 u8 no_banks;
105 u8 cas_latency;
Vikas Manochabfea69a2017-04-10 15:03:03 -0700106 u8 sdclk;
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700107 u8 rd_burst;
108 u8 rd_pipe_delay;
109};
110
111struct stm32_sdram_timing {
112 u8 tmrd;
113 u8 txsr;
114 u8 tras;
115 u8 trc;
116 u8 trp;
Vikas Manochabfea69a2017-04-10 15:03:03 -0700117 u8 twr;
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700118 u8 trcd;
119};
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200120enum stm32_fmc_bank {
121 SDRAM_BANK1,
122 SDRAM_BANK2,
123 MAX_SDRAM_BANK,
124};
125
126struct bank_params {
Patrice Chotardf39b90d2017-07-18 17:37:26 +0200127 struct stm32_sdram_control *sdram_control;
128 struct stm32_sdram_timing *sdram_timing;
Vikas Manochabfea69a2017-04-10 15:03:03 -0700129 u32 sdram_ref_count;
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200130 enum stm32_fmc_bank target_bank;
131};
132
133struct stm32_sdram_params {
134 struct stm32_fmc_regs *base;
135 u8 no_sdram_banks;
136 struct bank_params bank_params[MAX_SDRAM_BANK];
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700137};
Vikas Manochabf1ae442017-04-10 15:02:51 -0700138
139#define SDRAM_MODE_BL_SHIFT 0
140#define SDRAM_MODE_CAS_SHIFT 4
141#define SDRAM_MODE_BL 0
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700142
143int stm32_sdram_init(struct udevice *dev)
Vikas Manochabf1ae442017-04-10 15:02:51 -0700144{
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700145 struct stm32_sdram_params *params = dev_get_platdata(dev);
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200146 struct stm32_sdram_control *control;
147 struct stm32_sdram_timing *timing;
Patrice Chotard1421e0a2017-07-18 17:37:25 +0200148 struct stm32_fmc_regs *regs = params->base;
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200149 enum stm32_fmc_bank target_bank;
150 u32 ctb; /* SDCMR register: Command Target Bank */
151 u32 ref_count;
152 u8 i;
Vikas Manochabf1ae442017-04-10 15:02:51 -0700153
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200154 for (i = 0; i < params->no_sdram_banks; i++) {
155 control = params->bank_params[i].sdram_control;
156 timing = params->bank_params[i].sdram_timing;
157 target_bank = params->bank_params[i].target_bank;
158 ref_count = params->bank_params[i].sdram_ref_count;
Vikas Manochabf1ae442017-04-10 15:02:51 -0700159
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200160 writel(control->sdclk << FMC_SDCR_SDCLK_SHIFT
161 | control->cas_latency << FMC_SDCR_CAS_SHIFT
162 | control->no_banks << FMC_SDCR_NB_SHIFT
163 | control->memory_width << FMC_SDCR_MWID_SHIFT
164 | control->no_rows << FMC_SDCR_NR_SHIFT
165 | control->no_columns << FMC_SDCR_NC_SHIFT
166 | control->rd_pipe_delay << FMC_SDCR_RPIPE_SHIFT
167 | control->rd_burst << FMC_SDCR_RBURST_SHIFT,
168 &regs->sdcr1);
Vikas Manochabf1ae442017-04-10 15:02:51 -0700169
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200170 if (target_bank == SDRAM_BANK2)
171 writel(control->cas_latency << FMC_SDCR_CAS_SHIFT
172 | control->no_banks << FMC_SDCR_NB_SHIFT
173 | control->memory_width << FMC_SDCR_MWID_SHIFT
174 | control->no_rows << FMC_SDCR_NR_SHIFT
175 | control->no_columns << FMC_SDCR_NC_SHIFT,
176 &regs->sdcr2);
Vikas Manochabf1ae442017-04-10 15:02:51 -0700177
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200178 writel(timing->trcd << FMC_SDTR_TRCD_SHIFT
179 | timing->trp << FMC_SDTR_TRP_SHIFT
180 | timing->twr << FMC_SDTR_TWR_SHIFT
181 | timing->trc << FMC_SDTR_TRC_SHIFT
182 | timing->tras << FMC_SDTR_TRAS_SHIFT
183 | timing->txsr << FMC_SDTR_TXSR_SHIFT
184 | timing->tmrd << FMC_SDTR_TMRD_SHIFT,
185 &regs->sdtr1);
Vikas Manochabf1ae442017-04-10 15:02:51 -0700186
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200187 if (target_bank == SDRAM_BANK2)
188 writel(timing->trcd << FMC_SDTR_TRCD_SHIFT
189 | timing->trp << FMC_SDTR_TRP_SHIFT
190 | timing->twr << FMC_SDTR_TWR_SHIFT
191 | timing->trc << FMC_SDTR_TRC_SHIFT
192 | timing->tras << FMC_SDTR_TRAS_SHIFT
193 | timing->txsr << FMC_SDTR_TXSR_SHIFT
194 | timing->tmrd << FMC_SDTR_TMRD_SHIFT,
195 &regs->sdtr2);
196 if (target_bank == SDRAM_BANK1)
197 ctb = FMC_SDCMR_BANK_1;
198 else
199 ctb = FMC_SDCMR_BANK_2;
Vikas Manochabf1ae442017-04-10 15:02:51 -0700200
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200201 writel(ctb | FMC_SDCMR_MODE_START_CLOCK, &regs->sdcmr);
202 udelay(200); /* 200 us delay, page 10, "Power-Up" */
203 FMC_BUSY_WAIT(regs);
Vikas Manochabf1ae442017-04-10 15:02:51 -0700204
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200205 writel(ctb | FMC_SDCMR_MODE_PRECHARGE, &regs->sdcmr);
206 udelay(100);
207 FMC_BUSY_WAIT(regs);
Vikas Manochabf1ae442017-04-10 15:02:51 -0700208
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200209 writel((ctb | FMC_SDCMR_MODE_AUTOREFRESH | 7 << FMC_SDCMR_NRFS_SHIFT),
210 &regs->sdcmr);
211 udelay(100);
212 FMC_BUSY_WAIT(regs);
213
214 writel(ctb | (SDRAM_MODE_BL << SDRAM_MODE_BL_SHIFT
215 | control->cas_latency << SDRAM_MODE_CAS_SHIFT)
216 << FMC_SDCMR_MODE_REGISTER_SHIFT | FMC_SDCMR_MODE_WRITE_MODE,
217 &regs->sdcmr);
218 udelay(100);
219 FMC_BUSY_WAIT(regs);
220
221 writel(ctb | FMC_SDCMR_MODE_NORMAL, &regs->sdcmr);
222 FMC_BUSY_WAIT(regs);
223
224 /* Refresh timer */
225 writel(ref_count << 1, &regs->sdrtr);
226 }
Vikas Manochabf1ae442017-04-10 15:02:51 -0700227
228 return 0;
229}
Vikas Manocha910a52e2017-04-10 15:02:52 -0700230
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700231static int stm32_fmc_ofdata_to_platdata(struct udevice *dev)
232{
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700233 struct stm32_sdram_params *params = dev_get_platdata(dev);
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200234 struct bank_params *bank_params;
235 ofnode bank_node;
236 char *bank_name;
237 u8 bank = 0;
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700238
Patrice Chotardf39b90d2017-07-18 17:37:26 +0200239 dev_for_each_subnode(bank_node, dev) {
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200240 /* extract the bank index from DT */
241 bank_name = (char *)ofnode_get_name(bank_node);
242 strsep(&bank_name, "@");
243 if (!bank_name) {
244 error("missing sdram bank index");
Patrice Chotardf39b90d2017-07-18 17:37:26 +0200245 return -EINVAL;
246 }
247
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200248 bank_params = &params->bank_params[bank];
249 strict_strtoul(bank_name, 10,
250 (long unsigned int *)&bank_params->target_bank);
Patrice Chotardf39b90d2017-07-18 17:37:26 +0200251
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200252 if (bank_params->target_bank >= MAX_SDRAM_BANK) {
253 error("Found bank %d , but only bank 0 and 1 are supported",
254 bank_params->target_bank);
Patrice Chotardf39b90d2017-07-18 17:37:26 +0200255 return -EINVAL;
256 }
257
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200258 debug("Find bank %s %u\n", bank_name, bank_params->target_bank);
259
260 params->bank_params[bank].sdram_control =
261 (struct stm32_sdram_control *)
262 ofnode_read_u8_array_ptr(bank_node,
263 "st,sdram-control",
264 sizeof(struct stm32_sdram_control));
265
266 if (!params->bank_params[bank].sdram_control) {
267 error("st,sdram-control not found for %s",
268 ofnode_get_name(bank_node));
269 return -EINVAL;
270 }
271
272
273 params->bank_params[bank].sdram_timing =
274 (struct stm32_sdram_timing *)
275 ofnode_read_u8_array_ptr(bank_node,
276 "st,sdram-timing",
277 sizeof(struct stm32_sdram_timing));
278
279 if (!params->bank_params[bank].sdram_timing) {
280 error("st,sdram-timing not found for %s",
281 ofnode_get_name(bank_node));
282 return -EINVAL;
283 }
284
285
286 bank_params->sdram_ref_count = ofnode_read_u32_default(bank_node,
Vikas Manochabfea69a2017-04-10 15:03:03 -0700287 "st,sdram-refcount", 8196);
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200288 bank++;
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700289 }
290
Patrice Chotardf303aaf2017-07-18 17:37:27 +0200291 params->no_sdram_banks = bank;
292 debug("%s, no of banks = %d\n", __func__, params->no_sdram_banks);
293
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700294 return 0;
295}
296
Vikas Manocha910a52e2017-04-10 15:02:52 -0700297static int stm32_fmc_probe(struct udevice *dev)
298{
Patrice Chotard1421e0a2017-07-18 17:37:25 +0200299 struct stm32_sdram_params *params = dev_get_platdata(dev);
Vikas Manochad0b24c12017-04-10 15:02:55 -0700300 int ret;
Patrice Chotard1421e0a2017-07-18 17:37:25 +0200301 fdt_addr_t addr;
302
303 addr = dev_read_addr(dev);
304 if (addr == FDT_ADDR_T_NONE)
305 return -EINVAL;
306
307 params->base = (struct stm32_fmc_regs *)addr;
308
Patrice Chotard14a50e32017-05-30 15:06:31 +0200309#ifdef CONFIG_CLK
Vikas Manochad0b24c12017-04-10 15:02:55 -0700310 struct clk clk;
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700311
Vikas Manochad0b24c12017-04-10 15:02:55 -0700312 ret = clk_get_by_index(dev, 0, &clk);
313 if (ret < 0)
314 return ret;
315
316 ret = clk_enable(&clk);
317
318 if (ret) {
319 dev_err(dev, "failed to enable clock\n");
320 return ret;
321 }
322#endif
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700323 ret = stm32_sdram_init(dev);
324 if (ret)
325 return ret;
326
Vikas Manocha910a52e2017-04-10 15:02:52 -0700327 return 0;
328}
329
330static int stm32_fmc_get_info(struct udevice *dev, struct ram_info *info)
331{
Vikas Manocha910a52e2017-04-10 15:02:52 -0700332 return 0;
333}
334
335static struct ram_ops stm32_fmc_ops = {
336 .get_info = stm32_fmc_get_info,
337};
338
339static const struct udevice_id stm32_fmc_ids[] = {
340 { .compatible = "st,stm32-fmc" },
341 { }
342};
343
344U_BOOT_DRIVER(stm32_fmc) = {
345 .name = "stm32_fmc",
346 .id = UCLASS_RAM,
347 .of_match = stm32_fmc_ids,
348 .ops = &stm32_fmc_ops,
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700349 .ofdata_to_platdata = stm32_fmc_ofdata_to_platdata,
Vikas Manocha910a52e2017-04-10 15:02:52 -0700350 .probe = stm32_fmc_probe,
Vikas Manocha6c9a1002017-04-10 15:02:56 -0700351 .platdata_auto_alloc_size = sizeof(struct stm32_sdram_params),
Vikas Manocha910a52e2017-04-10 15:02:52 -0700352};