xref: /openbmc/u-boot/cmd/aspeed/dramtest.c (revision b8233483f822dc441692cc7df659e402e7edcd1b)
1 /*
2  *  This program is distributed in the hope that it will be useful,
3  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
4  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
5  *  GNU General Public License for more details.
6  *
7  *  You should have received a copy of the GNU General Public License
8  *  along with this program; if not, write to the Free Software
9  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
10  */
11 
12 #include <common.h>
13 #include <console.h>
14 #include <bootretry.h>
15 #include <cli.h>
16 #include <command.h>
17 #include <console.h>
18 
19 #include <inttypes.h>
20 #include <mapmem.h>
21 #include <asm/io.h>
22 #include <linux/compiler.h>
23 
24 DECLARE_GLOBAL_DATA_PTR;
25 
26 #define DRAM_BASE	0x80000000
27 #define TIMEOUT_DRAM	5000000
28 
29 #ifdef CONFIG_ASPEED_AST2600
30 #define BLK_SIZE_MB			64
31 #define N_BLK				32
32 #define MCR74_BLK_OFFSET		26
33 #define MCR74_LEN_OFFSET		4
34 #define MCR74_BLK_LEN_MASK		GENMASK(30, 4)
35 #else
36 #define BLK_SIZE_MB			16
37 #define N_BLK				64
38 #define MCR74_BLK_OFFSET		24
39 #define MCR74_LEN_OFFSET		4
40 #define MCR74_BLK_LEN_MASK		GENMASK(29, 4)
41 #endif
42 #define BLK_SIZE			(BLK_SIZE_MB * 1024 * 1024)
43 #define N_16B_IN_BLK			(BLK_SIZE / 16)
44 
45 /* ------------------------------------------------------------------------- */
46 int MMCTestBurst(unsigned int datagen)
47 {
48 	unsigned int data;
49 	unsigned int timeout = 0;
50 
51 	writel(0x00000000, 0x1E6E0070);
52 	writel((0x000000C1 | (datagen << 3)), 0x1E6E0070);
53 
54 	do {
55 		data = readl(0x1E6E0070) & 0x3000;
56 
57 		if( data & 0x2000 )
58 			return(0);
59 
60 		if( ++timeout > TIMEOUT_DRAM ) {
61 			printf("Timeout!!\n");
62 			writel(0x00000000, 0x1E6E0070);
63 			return(0);
64 		}
65 	} while (!data);
66 
67 	writel(0x00000000, 0x1E6E0070);
68 
69 	return(1);
70 }
71 
72 /* ------------------------------------------------------------------------- */
73 int MMCTestSingle(unsigned int datagen)
74 {
75 	unsigned int data;
76 	unsigned int timeout = 0;
77 
78 	writel(0x00000000, 0x1E6E0070);
79 	writel((0x00000085 | (datagen << 3)), 0x1E6E0070);
80 
81 	do {
82 		data = readl(0x1E6E0070) & 0x3000;
83 
84 		if( data & 0x2000 )
85 			return(0);
86 
87 		if( ++timeout > TIMEOUT_DRAM ){
88 			printf("Timeout!!\n");
89 			writel(0x00000000, 0x1E6E0070);
90 
91 			return(0);
92 		}
93 	} while ( !data );
94 
95 	writel(0x00000000, 0x1E6E0070);
96 
97 	return(1);
98 }
99 
100 /* ------------------------------------------------------------------------- */
101 int MMCTest(void)
102 {
103 	unsigned int pattern;
104 
105 	pattern = rand();
106 	writel(pattern, 0x1E6E007C);
107 	printf("Pattern = %08X : ",pattern);
108 
109 	if(!MMCTestBurst(0))    return(0);
110 	if(!MMCTestBurst(1))    return(0);
111 	if(!MMCTestBurst(2))    return(0);
112 	if(!MMCTestBurst(3))    return(0);
113 	if(!MMCTestBurst(4))    return(0);
114 	if(!MMCTestBurst(5))    return(0);
115 	if(!MMCTestBurst(6))    return(0);
116 	if(!MMCTestBurst(7))    return(0);
117 	if(!MMCTestSingle(0))   return(0);
118 	if(!MMCTestSingle(1))   return(0);
119 	if(!MMCTestSingle(2))   return(0);
120 	if(!MMCTestSingle(3))   return(0);
121 	if(!MMCTestSingle(4))   return(0);
122 	if(!MMCTestSingle(5))   return(0);
123 	if(!MMCTestSingle(6))   return(0);
124 	if(!MMCTestSingle(7))   return(0);
125 
126 	return(1);
127 }
128 
129 /* ------------------------------------------------------------------------- */
130 static void print_usage(void)
131 {
132 	printf("\nASPEED DRAM BIST\n\n");
133 	printf("Usage: dramtest <count> <block> <length>\n\n");
134 	printf("count:  how many iterations to run (mandatory, in decimal)\n");
135 	printf("        0: infinite loop.\n");
136 	printf("block:  index of the address block to test "
137 			"(optional, in decimal, default: 0)\n");
138 	printf("        0: [%08x, %08x)\n", DRAM_BASE + BLK_SIZE * 0, DRAM_BASE + BLK_SIZE * 1);
139 	printf("        1: [%08x, %08x)\n", DRAM_BASE + BLK_SIZE * 1, DRAM_BASE + BLK_SIZE * 2);
140 	printf("        2: [%08x, %08x)\n", DRAM_BASE + BLK_SIZE * 2, DRAM_BASE + BLK_SIZE * 3);
141 	printf("        n: [80000000 + n*%dMB, 80000000 + (n+1)*%dMB)\n", BLK_SIZE_MB, BLK_SIZE_MB);
142 	printf("           where n = [0, %d]\n", N_BLK - 1);
143 	printf("length: size to test (optional, in hex, default: 0x10000)\n");
144 	printf("        0x0: test the whole memory block %dMB\n", BLK_SIZE_MB);
145 	printf("        0x1: test the first 16 Bytes of the memory block\n");
146 	printf("        0x2: test the first 2*16 Bytes of the memory block\n");
147 	printf("        n  : test the first n*16 Bytes of the memory block\n");
148 	printf("             where n = [0x00000001, 0x%08x]\n", N_16B_IN_BLK);
149 	printf("\n\n");
150 }
151 
152 static int
153 do_ast_dramtest(cmd_tbl_t *cmdtp, int flag, int argc, char *const argv[])
154 {
155 	u32 PassCnt = 0;
156 	unsigned long Testcounter = 0;
157 	unsigned long block = 0;
158 	unsigned long length = 0x10000;
159 	int ret;
160 
161 	if (argc < 2) {
162 		ret = 0;
163 		goto cmd_err;
164 	}
165 
166 	ret = CMD_RET_USAGE;
167 	switch (argc) {
168 	case 4:
169 		if (strict_strtoul(argv[3], 16, &length) < 0)
170 			goto cmd_err;
171 	case 3:
172 		if (strict_strtoul(argv[2], 10, &block) < 0)
173 			goto cmd_err;
174 	case 2:
175 		if (strict_strtoul(argv[1], 10, &Testcounter) < 0)
176 			goto cmd_err;
177 		break;
178 	}
179 
180 	if (length == 0)
181 		length = N_16B_IN_BLK;
182 
183 	printf("Test range: 0x%08lx - 0x%08lx\n", DRAM_BASE + (block << MCR74_BLK_OFFSET),
184 	       DRAM_BASE | (block << MCR74_BLK_OFFSET) | ((length - 1) << MCR74_LEN_OFFSET) | 0xf);
185 
186 	ret = 1;
187 	writel(0xFC600309, 0x1E6E0000);
188 	while ((Testcounter > PassCnt) || (Testcounter == 0)) {
189 		clrsetbits_le32(0x1E6E0074, MCR74_BLK_LEN_MASK,
190 				(block << MCR74_BLK_OFFSET) | ((length - 1) << MCR74_LEN_OFFSET));
191 
192 		if (!MMCTest()) {
193 			printf("FAIL %d/%ld (fail DQ 0x%08x)\n", PassCnt,
194 			       Testcounter, readl(0x1E6E0078));
195 			ret = 0;
196 			break;
197 		} else {
198 			PassCnt++;
199 			printf("Pass %d/%ld\n", PassCnt, Testcounter);
200 		}
201 	}
202 
203 	return (ret);
204 
205 cmd_err:
206 	print_usage();
207 	return (ret);
208 }
209 
210 U_BOOT_CMD(
211 	dramtest,   5, 0,  do_ast_dramtest,
212 	"ASPEED DRAM BIST",
213 	""
214 );
215