/* ------------------------------------------------------------------------- */
-#define MAXSIZE (256<<20)
-long int initdram(int board_type)
+/* Try SDRAM initialization with P/LSDMR=sdmr and ORx=orx
+ *
+ * This routine performs standard 8260 initialization sequence
+ * and calculates the available memory size. It may be called
+ * several times to try different SDRAM configurations on both
+ * 60x and local buses.
+ */
+static long int try_init(volatile memctl8260_t *memctl, ulong sdmr, ulong orx,
+ volatile uchar * base)
{
- volatile immap_t * immap = (immap_t *)CFG_IMMR;
- volatile memctl8260_t * memctl = &immap->im_memctl;
-#ifndef CFG_RAMBOOT
- volatile uint * psdmr_ptr = &memctl->memc_psdmr;
- volatile uchar * base = CFG_SDRAM_BASE;
- volatile ulong * addr;
- volatile char c = 0xff;
- ulong save[32];
- ulong val;
- int i,cnt;
-#endif
+ volatile uchar c = 0xff;
+ volatile ulong cnt, val;
+ volatile ulong * addr;
+ volatile uint * sdmr_ptr;
+ volatile uint * orx_ptr;
+ int i;
+ ulong save[32]; /* to make test non-destructive */
+ ulong maxsize;
- memctl->memc_psrt = CFG_PSRT;
- memctl->memc_mptpr = CFG_MPTPR;
+ /* We must be able to test a location outsize the maximum legal size
+ * to find out THAT we are outside; but this address still has to be
+ * mapped by the controller. That means, that the initial mapping has
+ * to be (at least) twice as large as the maximum expected size.
+ */
+ maxsize = (1 + (~orx | 0x7fff)) / 2;
+
+ sdmr_ptr = &memctl->memc_psdmr;
+ orx_ptr = &memctl->memc_or2;
+
+ *orx_ptr = orx;
-#ifndef CFG_RAMBOOT
/*
* Quote from 8260 UM (10.4.2 SDRAM Power-On Initialization, 10-35):
*
* get here. The SDRAM can be accessed at the address CFG_SDRAM_BASE.
*/
- *psdmr_ptr = CFG_PSDMR | PSDMR_OP_PREA;
+ *sdmr_ptr = sdmr | PSDMR_OP_PREA;
*base = c;
- *psdmr_ptr = CFG_PSDMR | PSDMR_OP_CBRR;
+ *sdmr_ptr = sdmr | PSDMR_OP_CBRR;
for (i = 0; i < 8; i++)
*base = c;
- *psdmr_ptr = CFG_PSDMR | PSDMR_OP_MRW;
+ *sdmr_ptr = sdmr | PSDMR_OP_MRW;
*(base + CFG_MRS_OFFS) = c; /* setting MR on address lines */
- *psdmr_ptr = CFG_PSDMR | PSDMR_OP_NORM | PSDMR_RFEN;
+ *sdmr_ptr = sdmr | PSDMR_OP_NORM | PSDMR_RFEN;
*base = c;
/*
* - short between data lines
*/
i = 0;
- for (cnt = MAXSIZE/sizeof(long); cnt > 0; cnt >>= 1) {
+ for (cnt = maxsize/sizeof(long); cnt > 0; cnt >>= 1) {
addr = (volatile ulong *)base + cnt; /* pointer arith! */
save[i++] = *addr;
*addr = ~cnt;
return (0);
}
- for (cnt = 1; cnt <= MAXSIZE/sizeof(long); cnt <<= 1) {
+ for (cnt = 1; cnt <= maxsize/sizeof(long); cnt <<= 1) {
addr = (volatile ulong *)base + cnt; /* pointer arith! */
val = *addr;
*addr = save[--i];
if (val != ~cnt) {
/* Write the actual size to ORx
*/
- memctl->memc_or2 = CFG_OR2_PRELIM | ~(cnt * sizeof(long) - 1);
+ *orx_ptr = orx | ~(cnt * sizeof(long) - 1);
return (cnt * sizeof(long));
}
}
+ return (maxsize);
+}
-#else
- return (32<<20);
+
+long int initdram(int board_type)
+{
+ volatile immap_t * immap = (immap_t *)CFG_IMMR;
+ volatile memctl8260_t * memctl = &immap->im_memctl;
+#ifndef CFG_RAMBOOT
+ ulong size8, size9, psize = 32 * 1024 * 1024;
+#endif
+
+ memctl->memc_psrt = CFG_PSRT;
+ memctl->memc_mptpr = CFG_MPTPR;
+
+#ifndef CFG_RAMBOOT
+ size8 = try_init(memctl, CFG_PSDMR_8COL, CFG_OR2_8COL,
+ (uchar *) CFG_SDRAM_BASE);
+ size9 = try_init(memctl, CFG_PSDMR_9COL, CFG_OR2_9COL,
+ (uchar *) CFG_SDRAM_BASE);
+
+ if (size8 < size9) {
+ psize = size9;
+ printf("(60x:9COL) ");
+ } else {
+ psize = try_init(memctl, CFG_PSDMR_8COL, CFG_OR2_8COL,
+ (uchar *) CFG_SDRAM_BASE);
+ printf("(60x:8COL) ");
+ }
#endif
- return (MAXSIZE);
+ return (psize);
}