blob: b5a6a4c981bf1912831981afe2356f599deb58a7 [file] [log] [blame]
wdenkc6097192002-11-03 00:24:07 +00001/*
Stefan Roese8d982302007-01-18 10:25:34 +01002 * (C) Copyright 2005-2007
Stefan Roesec443fe92005-11-22 13:20:42 +01003 * Stefan Roese, DENX Software Engineering, sr@denx.de.
4 *
Stefan Roesefd637932006-03-17 10:28:24 +01005 * (C) Copyright 2006
6 * DAVE Srl <www.dave-tech.it>
7 *
stroese55ca7492004-07-15 14:41:13 +00008 * (C) Copyright 2002-2004
wdenkc6097192002-11-03 00:24:07 +00009 * Stefan Roese, esd gmbh germany, stefan.roese@esd-electronics.com
10 *
11 * See file CREDITS for list of people who contributed to this
12 * project.
13 *
14 * This program is free software; you can redistribute it and/or
15 * modify it under the terms of the GNU General Public License as
16 * published by the Free Software Foundation; either version 2 of
17 * the License, or (at your option) any later version.
18 *
19 * This program is distributed in the hope that it will be useful,
20 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020021 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
wdenkc6097192002-11-03 00:24:07 +000022 * GNU General Public License for more details.
23 *
24 * You should have received a copy of the GNU General Public License
25 * along with this program; if not, write to the Free Software
26 * Foundation, Inc., 59 Temple Place, Suite 330, Boston,
27 * MA 02111-1307 USA
28 */
29
30#include <common.h>
31#include <ppc4xx.h>
32#include <asm/processor.h>
Stefan Roesefd637932006-03-17 10:28:24 +010033#include "sdram.h"
Grant Ericksonb6933412008-05-22 14:44:14 -070034#include "ecc.h"
wdenkc6097192002-11-03 00:24:07 +000035
wdenkc6097192002-11-03 00:24:07 +000036#ifdef CONFIG_SDRAM_BANK0
37
Stefan Roese8d982302007-01-18 10:25:34 +010038#ifndef CONFIG_440
wdenkc6097192002-11-03 00:24:07 +000039
Stefan Roesec443fe92005-11-22 13:20:42 +010040#ifndef CFG_SDRAM_TABLE
stroese55ca7492004-07-15 14:41:13 +000041sdram_conf_t mb0cf[] = {
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020042 {(128 << 20), 13, 0x000A4001}, /* (0-128MB) Address Mode 3, 13x10(4) */
43 {(64 << 20), 13, 0x00084001}, /* (0-64MB) Address Mode 3, 13x9(4) */
44 {(32 << 20), 12, 0x00062001}, /* (0-32MB) Address Mode 2, 12x9(4) */
45 {(16 << 20), 12, 0x00046001}, /* (0-16MB) Address Mode 4, 12x8(4) */
46 {(4 << 20), 11, 0x00008001}, /* (0-4MB) Address Mode 5, 11x8(2) */
stroese55ca7492004-07-15 14:41:13 +000047};
Stefan Roesec443fe92005-11-22 13:20:42 +010048#else
49sdram_conf_t mb0cf[] = CFG_SDRAM_TABLE;
50#endif
51
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020052#define N_MB0CF (sizeof(mb0cf) / sizeof(mb0cf[0]))
stroese55ca7492004-07-15 14:41:13 +000053
Stefan Roesefd637932006-03-17 10:28:24 +010054#ifdef CFG_SDRAM_CASL
55static ulong ns2clks(ulong ns)
56{
57 ulong bus_period_x_10 = ONE_BILLION / (get_bus_freq(0) / 10);
58
59 return ((ns * 10) + bus_period_x_10) / bus_period_x_10;
60}
61#endif /* CFG_SDRAM_CASL */
62
63static ulong compute_sdtr1(ulong speed)
64{
65#ifdef CFG_SDRAM_CASL
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020066 ulong tmp;
67 ulong sdtr1 = 0;
Stefan Roesefd637932006-03-17 10:28:24 +010068
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020069 /* CASL */
70 if (CFG_SDRAM_CASL < 2)
71 sdtr1 |= (1 << SDRAM0_TR_CASL);
72 else
73 if (CFG_SDRAM_CASL > 4)
74 sdtr1 |= (3 << SDRAM0_TR_CASL);
75 else
76 sdtr1 |= ((CFG_SDRAM_CASL-1) << SDRAM0_TR_CASL);
Stefan Roesefd637932006-03-17 10:28:24 +010077
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020078 /* PTA */
79 tmp = ns2clks(CFG_SDRAM_PTA);
80 if ((tmp >= 2) && (tmp <= 4))
81 sdtr1 |= ((tmp-1) << SDRAM0_TR_PTA);
82 else
83 sdtr1 |= ((4-1) << SDRAM0_TR_PTA);
Stefan Roesefd637932006-03-17 10:28:24 +010084
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020085 /* CTP */
86 tmp = ns2clks(CFG_SDRAM_CTP);
87 if ((tmp >= 2) && (tmp <= 4))
88 sdtr1 |= ((tmp-1) << SDRAM0_TR_CTP);
89 else
90 sdtr1 |= ((4-1) << SDRAM0_TR_CTP);
Stefan Roesefd637932006-03-17 10:28:24 +010091
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020092 /* LDF */
93 tmp = ns2clks(CFG_SDRAM_LDF);
94 if ((tmp >= 2) && (tmp <= 4))
95 sdtr1 |= ((tmp-1) << SDRAM0_TR_LDF);
96 else
97 sdtr1 |= ((2-1) << SDRAM0_TR_LDF);
Stefan Roesefd637932006-03-17 10:28:24 +010098
Wolfgang Denkebd3deb2006-04-16 10:51:58 +020099 /* RFTA */
100 tmp = ns2clks(CFG_SDRAM_RFTA);
101 if ((tmp >= 4) && (tmp <= 10))
102 sdtr1 |= ((tmp-4) << SDRAM0_TR_RFTA);
103 else
104 sdtr1 |= ((10-4) << SDRAM0_TR_RFTA);
Stefan Roesefd637932006-03-17 10:28:24 +0100105
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200106 /* RCD */
107 tmp = ns2clks(CFG_SDRAM_RCD);
108 if ((tmp >= 2) && (tmp <= 4))
109 sdtr1 |= ((tmp-1) << SDRAM0_TR_RCD);
110 else
111 sdtr1 |= ((4-1) << SDRAM0_TR_RCD);
Stefan Roesefd637932006-03-17 10:28:24 +0100112
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200113 return sdtr1;
Stefan Roesefd637932006-03-17 10:28:24 +0100114#else /* CFG_SDRAM_CASL */
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200115 /*
116 * If no values are configured in the board config file
117 * use the default values, which seem to be ok for most
118 * boards.
119 *
120 * REMARK:
121 * For new board ports we strongly recommend to define the
122 * correct values for the used SDRAM chips in your board
123 * config file (see PPChameleonEVB.h)
124 */
125 if (speed > 100000000) {
126 /*
127 * 133 MHz SDRAM
128 */
129 return 0x01074015;
130 } else {
131 /*
132 * default: 100 MHz SDRAM
133 */
134 return 0x0086400d;
135 }
Stefan Roesefd637932006-03-17 10:28:24 +0100136#endif /* CFG_SDRAM_CASL */
137}
138
139/* refresh is expressed in ms */
140static ulong compute_rtr(ulong speed, ulong rows, ulong refresh)
141{
142#ifdef CFG_SDRAM_CASL
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200143 ulong tmp;
Stefan Roesefd637932006-03-17 10:28:24 +0100144
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200145 tmp = ((refresh*1000*1000) / (1 << rows)) * (speed / 1000);
146 tmp /= 1000000;
Stefan Roesefd637932006-03-17 10:28:24 +0100147
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200148 return ((tmp & 0x00003FF8) << 16);
Stefan Roesefd637932006-03-17 10:28:24 +0100149#else /* CFG_SDRAM_CASL */
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200150 if (speed > 100000000) {
151 /*
152 * 133 MHz SDRAM
153 */
Stefan Roesefd637932006-03-17 10:28:24 +0100154 return 0x07f00000;
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200155 } else {
156 /*
157 * default: 100 MHz SDRAM
158 */
Stefan Roesefd637932006-03-17 10:28:24 +0100159 return 0x05f00000;
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200160 }
Stefan Roesefd637932006-03-17 10:28:24 +0100161#endif /* CFG_SDRAM_CASL */
162}
163
Stefan Roesec443fe92005-11-22 13:20:42 +0100164/*
165 * Autodetect onboard SDRAM on 405 platforms
166 */
Becky Brucebd99ae72008-06-09 16:03:40 -0500167phys_size_t initdram(int board_type)
wdenkc6097192002-11-03 00:24:07 +0000168{
Stefan Roesefd637932006-03-17 10:28:24 +0100169 ulong speed;
wdenkc6097192002-11-03 00:24:07 +0000170 ulong sdtr1;
stroese55ca7492004-07-15 14:41:13 +0000171 int i;
wdenkc6097192002-11-03 00:24:07 +0000172
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200173 /*
174 * Determine SDRAM speed
175 */
176 speed = get_bus_freq(0); /* parameter not used on ppc4xx */
Stefan Roesefd637932006-03-17 10:28:24 +0100177
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200178 /*
179 * sdtr1 (register SDRAM0_TR) must take into account timings listed
180 * in SDRAM chip datasheet. rtr (register SDRAM0_RTR) must take into
181 * account actual SDRAM size. So we can set up sdtr1 according to what
182 * is specified in board configuration file while rtr dependds on SDRAM
183 * size we are assuming before detection.
184 */
185 sdtr1 = compute_sdtr1(speed);
wdenkc6097192002-11-03 00:24:07 +0000186
stroese55ca7492004-07-15 14:41:13 +0000187 for (i=0; i<N_MB0CF; i++) {
stroese51c57b92003-02-10 16:26:37 +0000188 /*
stroese55ca7492004-07-15 14:41:13 +0000189 * Disable memory controller.
stroese51c57b92003-02-10 16:26:37 +0000190 */
Stefan Roesea5d182e2007-08-14 14:44:41 +0200191 mtsdram(mem_mcopt1, 0x00000000);
wdenk41e2e052003-02-11 01:49:43 +0000192
wdenkc6097192002-11-03 00:24:07 +0000193 /*
stroese55ca7492004-07-15 14:41:13 +0000194 * Set MB0CF for bank 0.
wdenkc6097192002-11-03 00:24:07 +0000195 */
Stefan Roesea5d182e2007-08-14 14:44:41 +0200196 mtsdram(mem_mb0cf, mb0cf[i].reg);
197 mtsdram(mem_sdtr1, sdtr1);
198 mtsdram(mem_rtr, compute_rtr(speed, mb0cf[i].rows, 64));
wdenk41e2e052003-02-11 01:49:43 +0000199
stroese55ca7492004-07-15 14:41:13 +0000200 udelay(200);
wdenkc6097192002-11-03 00:24:07 +0000201
wdenkc6097192002-11-03 00:24:07 +0000202 /*
stroese55ca7492004-07-15 14:41:13 +0000203 * Set memory controller options reg, MCOPT1.
204 * Set DC_EN to '1' and BRD_PRF to '01' for 16 byte PLB Burst
205 * read/prefetch.
wdenkc6097192002-11-03 00:24:07 +0000206 */
Stefan Roesea5d182e2007-08-14 14:44:41 +0200207 mtsdram(mem_mcopt1, 0x80800000);
wdenkc6097192002-11-03 00:24:07 +0000208
stroese55ca7492004-07-15 14:41:13 +0000209 udelay(10000);
wdenkc6097192002-11-03 00:24:07 +0000210
stroese55ca7492004-07-15 14:41:13 +0000211 if (get_ram_size(0, mb0cf[i].size) == mb0cf[i].size) {
Stefan Roese58296a12008-09-10 16:53:47 +0200212 phys_size_t size = mb0cf[i].size;
213
stroese55ca7492004-07-15 14:41:13 +0000214 /*
John Otken9aa36772007-07-26 17:49:11 +0200215 * OK, size detected. Enable second bank if
216 * defined (assumes same type as bank 0)
stroese55ca7492004-07-15 14:41:13 +0000217 */
John Otken9aa36772007-07-26 17:49:11 +0200218#ifdef CONFIG_SDRAM_BANK1
Stefan Roesea5d182e2007-08-14 14:44:41 +0200219 mtsdram(mem_mcopt1, 0x00000000);
Stefan Roese58296a12008-09-10 16:53:47 +0200220 mtsdram(mem_mb1cf, mb0cf[i].size | mb0cf[i].reg);
Stefan Roesea5d182e2007-08-14 14:44:41 +0200221 mtsdram(mem_mcopt1, 0x80800000);
John Otken9aa36772007-07-26 17:49:11 +0200222 udelay(10000);
Stefan Roesea5d182e2007-08-14 14:44:41 +0200223
224 /*
225 * Check if 2nd bank is really available.
226 * If the size not equal to the size of the first
227 * bank, then disable the 2nd bank completely.
228 */
229 if (get_ram_size((long *)mb0cf[i].size, mb0cf[i].size) !=
230 mb0cf[i].size) {
231 mtsdram(mem_mb1cf, 0);
232 mtsdram(mem_mcopt1, 0);
Stefan Roese58296a12008-09-10 16:53:47 +0200233 } else {
234 /*
235 * We have two identical banks, so the size
236 * is twice the bank size
237 */
238 size = 2 * size;
Stefan Roesea5d182e2007-08-14 14:44:41 +0200239 }
John Otken9aa36772007-07-26 17:49:11 +0200240#endif
Stefan Roesecdb04702008-06-02 17:37:28 +0200241
242 /*
243 * OK, size detected -> all done
244 */
Stefan Roese58296a12008-09-10 16:53:47 +0200245 return size;
stroese55ca7492004-07-15 14:41:13 +0000246 }
stroese51c57b92003-02-10 16:26:37 +0000247 }
Stefan Roesecdb04702008-06-02 17:37:28 +0200248
249 return 0;
wdenkc6097192002-11-03 00:24:07 +0000250}
251
Stefan Roesec443fe92005-11-22 13:20:42 +0100252#else /* CONFIG_440 */
253
Stefan Roese8d982302007-01-18 10:25:34 +0100254/*
255 * Define some default values. Those can be overwritten in the
256 * board config file.
257 */
258
259#ifndef CFG_SDRAM_TABLE
260sdram_conf_t mb0cf[] = {
261 {(256 << 20), 13, 0x000C4001}, /* 256MB mode 3, 13x10(4) */
262 {(64 << 20), 12, 0x00082001} /* 64MB mode 2, 12x9(4) */
263};
264#else
265sdram_conf_t mb0cf[] = CFG_SDRAM_TABLE;
266#endif
267
268#ifndef CFG_SDRAM0_TR0
269#define CFG_SDRAM0_TR0 0x41094012
270#endif
271
272#define N_MB0CF (sizeof(mb0cf) / sizeof(mb0cf[0]))
273
Stefan Roesefd637932006-03-17 10:28:24 +0100274#define NUM_TRIES 64
275#define NUM_READS 10
276
277static void sdram_tr1_set(int ram_address, int* tr1_value)
278{
279 int i;
280 int j, k;
281 volatile unsigned int* ram_pointer = (unsigned int *)ram_address;
282 int first_good = -1, last_bad = 0x1ff;
283
284 unsigned long test[NUM_TRIES] = {
285 0x00000000, 0x00000000, 0xFFFFFFFF, 0xFFFFFFFF,
286 0x00000000, 0x00000000, 0xFFFFFFFF, 0xFFFFFFFF,
287 0xFFFFFFFF, 0xFFFFFFFF, 0x00000000, 0x00000000,
288 0xFFFFFFFF, 0xFFFFFFFF, 0x00000000, 0x00000000,
289 0xAAAAAAAA, 0xAAAAAAAA, 0x55555555, 0x55555555,
290 0xAAAAAAAA, 0xAAAAAAAA, 0x55555555, 0x55555555,
291 0x55555555, 0x55555555, 0xAAAAAAAA, 0xAAAAAAAA,
292 0x55555555, 0x55555555, 0xAAAAAAAA, 0xAAAAAAAA,
293 0xA5A5A5A5, 0xA5A5A5A5, 0x5A5A5A5A, 0x5A5A5A5A,
294 0xA5A5A5A5, 0xA5A5A5A5, 0x5A5A5A5A, 0x5A5A5A5A,
295 0x5A5A5A5A, 0x5A5A5A5A, 0xA5A5A5A5, 0xA5A5A5A5,
296 0x5A5A5A5A, 0x5A5A5A5A, 0xA5A5A5A5, 0xA5A5A5A5,
297 0xAA55AA55, 0xAA55AA55, 0x55AA55AA, 0x55AA55AA,
298 0xAA55AA55, 0xAA55AA55, 0x55AA55AA, 0x55AA55AA,
299 0x55AA55AA, 0x55AA55AA, 0xAA55AA55, 0xAA55AA55,
300 0x55AA55AA, 0x55AA55AA, 0xAA55AA55, 0xAA55AA55 };
301
302 /* go through all possible SDRAM0_TR1[RDCT] values */
303 for (i=0; i<=0x1ff; i++) {
304 /* set the current value for TR1 */
305 mtsdram(mem_tr1, (0x80800800 | i));
306
307 /* write values */
308 for (j=0; j<NUM_TRIES; j++) {
309 ram_pointer[j] = test[j];
310
311 /* clear any cache at ram location */
312 __asm__("dcbf 0,%0": :"r" (&ram_pointer[j]));
313 }
314
315 /* read values back */
316 for (j=0; j<NUM_TRIES; j++) {
317 for (k=0; k<NUM_READS; k++) {
318 /* clear any cache at ram location */
319 __asm__("dcbf 0,%0": :"r" (&ram_pointer[j]));
320
321 if (ram_pointer[j] != test[j])
322 break;
323 }
324
325 /* read error */
326 if (k != NUM_READS)
327 break;
328 }
329
330 /* we have a SDRAM0_TR1[RDCT] that is part of the window */
331 if (j == NUM_TRIES) {
332 if (first_good == -1)
333 first_good = i; /* found beginning of window */
334 } else { /* bad read */
335 /* if we have not had a good read then don't care */
336 if (first_good != -1) {
337 /* first failure after a good read */
338 last_bad = i-1;
339 break;
340 }
341 }
342 }
343
344 /* return the current value for TR1 */
345 *tr1_value = (first_good + last_bad) / 2;
346}
347
Stefan Roesec443fe92005-11-22 13:20:42 +0100348/*
349 * Autodetect onboard DDR SDRAM on 440 platforms
350 *
351 * NOTE: Some of the hardcoded values are hardware dependant,
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200352 * so this should be extended for other future boards
353 * using this routine!
Stefan Roesec443fe92005-11-22 13:20:42 +0100354 */
Becky Brucebd99ae72008-06-09 16:03:40 -0500355phys_size_t initdram(int board_type)
Stefan Roesec443fe92005-11-22 13:20:42 +0100356{
357 int i;
Stefan Roesefd637932006-03-17 10:28:24 +0100358 int tr1_bank1;
Stefan Roesec443fe92005-11-22 13:20:42 +0100359
Stefan Roese8d982302007-01-18 10:25:34 +0100360#if defined(CONFIG_440GX) || defined(CONFIG_440EP) || \
361 defined(CONFIG_440GR) || defined(CONFIG_440SP)
Stefan Roese49639682006-08-15 14:22:35 +0200362 /*
363 * Soft-reset SDRAM controller.
364 */
365 mtsdr(sdr_srst, SDR0_SRST_DMC);
366 mtsdr(sdr_srst, 0x00000000);
367#endif
368
Stefan Roesec443fe92005-11-22 13:20:42 +0100369 for (i=0; i<N_MB0CF; i++) {
370 /*
371 * Disable memory controller.
372 */
373 mtsdram(mem_cfg0, 0x00000000);
374
375 /*
376 * Setup some default
377 */
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200378 mtsdram(mem_uabba, 0x00000000); /* ubba=0 (default) */
379 mtsdram(mem_slio, 0x00000000); /* rdre=0 wrre=0 rarw=0 */
Stefan Roesec443fe92005-11-22 13:20:42 +0100380 mtsdram(mem_devopt, 0x00000000); /* dll=0 ds=0 (normal) */
381 mtsdram(mem_wddctr, 0x00000000); /* wrcp=0 dcd=0 */
Wolfgang Denkebd3deb2006-04-16 10:51:58 +0200382 mtsdram(mem_clktr, 0x40000000); /* clkp=1 (90 deg wr) dcdt=0 */
Stefan Roesec443fe92005-11-22 13:20:42 +0100383
384 /*
385 * Following for CAS Latency = 2.5 @ 133 MHz PLB
386 */
387 mtsdram(mem_b0cr, mb0cf[i].reg);
Stefan Roese8d982302007-01-18 10:25:34 +0100388 mtsdram(mem_tr0, CFG_SDRAM0_TR0);
Stefan Roesec443fe92005-11-22 13:20:42 +0100389 mtsdram(mem_tr1, 0x80800800); /* SS=T2 SL=STAGE 3 CD=1 CT=0x00*/
Stefan Roese76e3d2a2007-01-06 15:58:09 +0100390 mtsdram(mem_rtr, 0x04100000); /* Interval 7.8µs @ 133MHz PLB */
Stefan Roesec443fe92005-11-22 13:20:42 +0100391 mtsdram(mem_cfg1, 0x00000000); /* Self-refresh exit, disable PM*/
392 udelay(400); /* Delay 200 usecs (min) */
393
394 /*
395 * Enable the controller, then wait for DCEN to complete
396 */
Stefan Roesee335ee02006-07-28 18:34:58 +0200397 mtsdram(mem_cfg0, 0x82000000); /* DCEN=1, PMUD=0, 64-bit */
Stefan Roesec443fe92005-11-22 13:20:42 +0100398 udelay(10000);
399
400 if (get_ram_size(0, mb0cf[i].size) == mb0cf[i].size) {
401 /*
Stefan Roesefd637932006-03-17 10:28:24 +0100402 * Optimize TR1 to current hardware environment
403 */
404 sdram_tr1_set(0x00000000, &tr1_bank1);
405 mtsdram(mem_tr1, (tr1_bank1 | 0x80800800));
406
407#ifdef CONFIG_SDRAM_ECC
408 ecc_init(0, mb0cf[i].size);
409#endif
410
411 /*
Stefan Roesec443fe92005-11-22 13:20:42 +0100412 * OK, size detected -> all done
413 */
414 return mb0cf[i].size;
415 }
416 }
417
418 return 0; /* nothing found ! */
419}
420
421#endif /* CONFIG_440 */
422
wdenkc6097192002-11-03 00:24:07 +0000423#endif /* CONFIG_SDRAM_BANK0 */