1 |
786 |
skrzyp |
//==========================================================================
|
2 |
|
|
//
|
3 |
|
|
// sdram0.cxx
|
4 |
|
|
//
|
5 |
|
|
// SDRAM function test 0
|
6 |
|
|
//
|
7 |
|
|
//==========================================================================
|
8 |
|
|
// ####ECOSGPLCOPYRIGHTBEGIN####
|
9 |
|
|
// -------------------------------------------
|
10 |
|
|
// This file is part of eCos, the Embedded Configurable Operating System.
|
11 |
|
|
// Copyright (C) 1998, 1999, 2000, 2001, 2002 Free Software Foundation, Inc.
|
12 |
|
|
//
|
13 |
|
|
// eCos is free software; you can redistribute it and/or modify it under
|
14 |
|
|
// the terms of the GNU General Public License as published by the Free
|
15 |
|
|
// Software Foundation; either version 2 or (at your option) any later
|
16 |
|
|
// version.
|
17 |
|
|
//
|
18 |
|
|
// eCos is distributed in the hope that it will be useful, but WITHOUT
|
19 |
|
|
// ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
20 |
|
|
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
21 |
|
|
// for more details.
|
22 |
|
|
//
|
23 |
|
|
// You should have received a copy of the GNU General Public License
|
24 |
|
|
// along with eCos; if not, write to the Free Software Foundation, Inc.,
|
25 |
|
|
// 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
26 |
|
|
//
|
27 |
|
|
// As a special exception, if other files instantiate templates or use
|
28 |
|
|
// macros or inline functions from this file, or you compile this file
|
29 |
|
|
// and link it with other works to produce a work based on this file,
|
30 |
|
|
// this file does not by itself cause the resulting work to be covered by
|
31 |
|
|
// the GNU General Public License. However the source code for this file
|
32 |
|
|
// must still be made available in accordance with section (3) of the GNU
|
33 |
|
|
// General Public License v2.
|
34 |
|
|
//
|
35 |
|
|
// This exception does not invalidate any other reasons why a work based
|
36 |
|
|
// on this file might be covered by the GNU General Public License.
|
37 |
|
|
// -------------------------------------------
|
38 |
|
|
// ####ECOSGPLCOPYRIGHTEND####
|
39 |
|
|
//==========================================================================
|
40 |
|
|
//#####DESCRIPTIONBEGIN####
|
41 |
|
|
//
|
42 |
|
|
// Author(s): hmt
|
43 |
|
|
// Contributors: hmt
|
44 |
|
|
// Date: 1999-11-11
|
45 |
|
|
// Description: Basic memory test, knowledgeable of the EBSA285's
|
46 |
|
|
// memory size.
|
47 |
|
|
//####DESCRIPTIONEND####
|
48 |
|
|
|
49 |
|
|
#include <cyg/infra/testcase.h>
|
50 |
|
|
|
51 |
|
|
#include <cyg/infra/diag.h>
|
52 |
|
|
|
53 |
|
|
#include <cyg/hal/hal_arch.h>
|
54 |
|
|
#include <cyg/hal/hal_intr.h>
|
55 |
|
|
|
56 |
|
|
#define ONE_MEG (0x100000)
|
57 |
|
|
#define ONE_MEG_IN_WORDS (ONE_MEG/4)
|
58 |
|
|
|
59 |
|
|
#define MEGS 9
|
60 |
|
|
#define WORDS ONE_MEG_IN_WORDS
|
61 |
|
|
#define START 0x400000
|
62 |
|
|
|
63 |
|
|
#define INNERLOOPS 10
|
64 |
|
|
|
65 |
|
|
#define NUMTESTS 1
|
66 |
|
|
|
67 |
|
|
#if WORDS > ONE_MEG_IN_WORDS
|
68 |
|
|
# error "Too many WORDS in a block - they'll overlap!"
|
69 |
|
|
#endif
|
70 |
|
|
|
71 |
|
|
#include <cyg/hal/hal_ebsa285.h>
|
72 |
|
|
|
73 |
|
|
|
74 |
|
|
void
|
75 |
|
|
check_addrsize_setup( void )
|
76 |
|
|
{
|
77 |
|
|
cyg_uint32 sizes[4] = { 0, };
|
78 |
|
|
cyg_uint32 bases[4] = { 0, };
|
79 |
|
|
cyg_uint32 codes[4] = { 0, };
|
80 |
|
|
cyg_uint32 muxes[4] = { 0, };
|
81 |
|
|
cyg_uint32 i;
|
82 |
|
|
|
83 |
|
|
#define MBytes <<20
|
84 |
|
|
|
85 |
|
|
static cyg_uint32 lookup[] =
|
86 |
|
|
{ 0, 1 MBytes, 2 MBytes, 4 MBytes, 8 MBytes, 16 MBytes, 32 MBytes, 64 MBytes };
|
87 |
|
|
|
88 |
|
|
static cyg_uint32 maxsizes[] =
|
89 |
|
|
{ 2 MBytes, 16 MBytes, 64 MBytes, 8 MBytes, 64 MBytes, 0, 0, 0 };
|
90 |
|
|
|
91 |
|
|
codes[0] = *SA110_SDRAM_ADDRESS_SIZE_ARRAY_0;
|
92 |
|
|
codes[1] = *SA110_SDRAM_ADDRESS_SIZE_ARRAY_1;
|
93 |
|
|
codes[2] = *SA110_SDRAM_ADDRESS_SIZE_ARRAY_2;
|
94 |
|
|
codes[3] = *SA110_SDRAM_ADDRESS_SIZE_ARRAY_3;
|
95 |
|
|
|
96 |
|
|
// Print all the info for the benefit of humans:
|
97 |
|
|
for ( i = 0; i < 4; i++ ) {
|
98 |
|
|
bases[i] = 0x0ff00000 & codes[i];
|
99 |
|
|
sizes[i] = lookup[ 7 & codes[i] ];
|
100 |
|
|
muxes[i] = 7 & (codes[i] >> 4);
|
101 |
|
|
diag_printf( "Bank %d: [%08x]: base %08x, size %08x; mux mode %d\n",
|
102 |
|
|
i, codes[i], bases[i] , sizes[i] , muxes[i] );
|
103 |
|
|
}
|
104 |
|
|
// THEN check individual entries for sanity
|
105 |
|
|
for ( i = 0; i < 4; i++ ) {
|
106 |
|
|
if ( 0 == sizes[i] ) {
|
107 |
|
|
// then the bank is not in use
|
108 |
|
|
CYG_TEST_CHECK( 0 == bases[i], "Unused bank nonzero address" );
|
109 |
|
|
CYG_TEST_CHECK( 0 == muxes[i], "Unused bank nonzero mux mode" );
|
110 |
|
|
} else {
|
111 |
|
|
CYG_TEST_CHECK( muxes[i] <= 4, "Mux mode overflow" );
|
112 |
|
|
if ( (muxes[i] == 3) && (8 != sizes[i]) )
|
113 |
|
|
CYG_TEST_FAIL( "Mux mode 3 and size not 8Mb" );
|
114 |
|
|
CYG_TEST_CHECK( maxsizes[ muxes[i] ] >= sizes[i],
|
115 |
|
|
"Size too larget for mux mode" );
|
116 |
|
|
}
|
117 |
|
|
}
|
118 |
|
|
|
119 |
|
|
// NEXT check that addresses are singly mapped IYSWIM:
|
120 |
|
|
// Easiest way is, foreach megabyte, check it is mapped exactly once;
|
121 |
|
|
// shouldn't take too long.
|
122 |
|
|
for ( i = 0; i < hal_dram_size; i += ONE_MEG ) {
|
123 |
|
|
int j = 0;
|
124 |
|
|
int k;
|
125 |
|
|
for ( k = 0; k < 4; k++ )
|
126 |
|
|
// this test works OK for an unused slot because i is +ve:
|
127 |
|
|
if ( (bases[k] <= i) && (i < (bases[k] + sizes[k])) )
|
128 |
|
|
j++;
|
129 |
|
|
CYG_TEST_CHECK( 2 > j, "Good memory is multiply mapped" );
|
130 |
|
|
CYG_TEST_CHECK( 0 < j, "Good memory is not mapped" );
|
131 |
|
|
}
|
132 |
|
|
for ( /* i */ ; i < 256 MBytes; i += ONE_MEG ) {
|
133 |
|
|
int j = 0;
|
134 |
|
|
int k;
|
135 |
|
|
for ( k = 0; k < 4; k++ )
|
136 |
|
|
// this test works OK for an unused slot because i is +ve:
|
137 |
|
|
if ( (bases[k] <= i) && (i < (bases[k] + sizes[k])) )
|
138 |
|
|
j++;
|
139 |
|
|
CYG_TEST_CHECK( 2 > j, "Non-existent memory is multiply mapped" );
|
140 |
|
|
CYG_TEST_CHECK( 0 == j, "Non-existent memory is mapped" );
|
141 |
|
|
}
|
142 |
|
|
CYG_TEST_PASS( "Memory controller setup self-consistent" );
|
143 |
|
|
}
|
144 |
|
|
|
145 |
|
|
|
146 |
|
|
void mymain( void )
|
147 |
|
|
{
|
148 |
|
|
cyg_uint32 h, i, j, k;
|
149 |
|
|
cyg_uint32 *pbase[ MEGS ] = { 0, };
|
150 |
|
|
int totaltests = 0;
|
151 |
|
|
int ptotalerrors[ MEGS ] = { 0, };
|
152 |
|
|
|
153 |
|
|
CYG_TEST_INIT();
|
154 |
|
|
|
155 |
|
|
|
156 |
|
|
check_addrsize_setup();
|
157 |
|
|
|
158 |
|
|
|
159 |
|
|
h = hal_dram_size - ONE_MEG;
|
160 |
|
|
i = MEGS - 1;
|
161 |
|
|
j = START;
|
162 |
|
|
k = 0;
|
163 |
|
|
while ( (i > k) && (h > j) ) {
|
164 |
|
|
pbase[i] = (cyg_uint32 *)h;
|
165 |
|
|
pbase[k] = (cyg_uint32 *)j;
|
166 |
|
|
i--;
|
167 |
|
|
k++;
|
168 |
|
|
h -= ONE_MEG;
|
169 |
|
|
j += ONE_MEG;
|
170 |
|
|
}
|
171 |
|
|
if ( (i == k) && (h > j) )
|
172 |
|
|
pbase[ i ] = (cyg_uint32 *)((h+j)/2);
|
173 |
|
|
|
174 |
|
|
for ( h = 0; h < NUMTESTS; h++ ) {
|
175 |
|
|
int perrors[ MEGS ] = { 0, };
|
176 |
|
|
cyg_uint32 pbadbits[ MEGS ] = { 0, };
|
177 |
|
|
for ( i = 0 ; i < INNERLOOPS; i++ ) {
|
178 |
|
|
cyg_uint32 d = 0xdeadbeef ^ ((cyg_uint32)i * 0x10001);
|
179 |
|
|
for ( k = 0; k < MEGS; k++ ) {
|
180 |
|
|
cyg_uint32 *p = pbase[k];
|
181 |
|
|
cyg_uint32 dp = d ^ (cyg_uint32)p;
|
182 |
|
|
if ( ! p ) continue;
|
183 |
|
|
for ( j = 0; j < WORDS; j++ )
|
184 |
|
|
p[j] = dp ^ j ^ (j << 19) ;
|
185 |
|
|
}
|
186 |
|
|
for ( k = 0; k < MEGS; k++ ) {
|
187 |
|
|
cyg_uint32 *p = pbase[k];
|
188 |
|
|
cyg_uint32 dp = d ^ (cyg_uint32)p;
|
189 |
|
|
if ( ! p ) continue;
|
190 |
|
|
for ( j = 0; j < WORDS; j++ )
|
191 |
|
|
if ( p[j] != (dp ^ j ^ (j << 19)) ) {
|
192 |
|
|
perrors[k]++;
|
193 |
|
|
pbadbits[k] |= (p[j] ^ dp ^ j ^ (j << 19));
|
194 |
|
|
}
|
195 |
|
|
}
|
196 |
|
|
}
|
197 |
|
|
totaltests += i * j;
|
198 |
|
|
for ( k = 0; k < MEGS; k++ ) {
|
199 |
|
|
if ( ! pbase[k] ) continue;
|
200 |
|
|
ptotalerrors[k] += perrors[k];
|
201 |
|
|
diag_printf(
|
202 |
|
|
"p %x: %d tests of %d words: %d errors, badbits %x ...totals %d tests %d errors\n",
|
203 |
|
|
pbase[k], i, j, perrors[k], pbadbits[k], totaltests, ptotalerrors[k] );
|
204 |
|
|
if ( 0 != perrors[k] )
|
205 |
|
|
CYG_TEST_FAIL( "Errors in memory test" );
|
206 |
|
|
}
|
207 |
|
|
}
|
208 |
|
|
|
209 |
|
|
h = j = 0;
|
210 |
|
|
for ( k = 0; k < MEGS; k++ ) {
|
211 |
|
|
if ( ! pbase[k] ) continue;
|
212 |
|
|
h += ptotalerrors[k] ;
|
213 |
|
|
j += totaltests;
|
214 |
|
|
}
|
215 |
|
|
diag_printf( "Total tests %d, total errors %d\n", j, h );
|
216 |
|
|
if ( 0 == h )
|
217 |
|
|
CYG_TEST_PASS( "Memory test all OK" );
|
218 |
|
|
|
219 |
|
|
CYG_TEST_EXIT("End of mem test");
|
220 |
|
|
|
221 |
|
|
}
|
222 |
|
|
|
223 |
|
|
externC void
|
224 |
|
|
cyg_start( void )
|
225 |
|
|
{
|
226 |
|
|
HAL_ENABLE_INTERRUPTS();
|
227 |
|
|
|
228 |
|
|
#ifdef CYGPKG_HAL_ARM_EBSA285
|
229 |
|
|
cyg_uint32 i;
|
230 |
|
|
i = *(cyg_uint32 *)(0x42000000 + 0x10c);
|
231 |
|
|
diag_printf( "SDRAM timing %08x\n", i );
|
232 |
|
|
for ( i = 0; i < 4; i++ ) {
|
233 |
|
|
diag_printf( "Bank %d: addrsize %08x\n", i, *(cyg_uint32 *)(0x42000000 + 0x110 + i * 4 ));
|
234 |
|
|
}
|
235 |
|
|
diag_printf( "Mem size: %08x == %d\n", hal_dram_size, hal_dram_size );
|
236 |
|
|
#endif
|
237 |
|
|
|
238 |
|
|
mymain();
|
239 |
|
|
}
|
240 |
|
|
// EOF sdram0.cxx
|