xref: /openbmc/qemu/tests/tcg/multiarch/system/memory.c (revision c97377652dfeef2e0373faa2cdc05548b1536d77)
1 /*
2  * Memory Test
3  *
4  * This is intended to test the softmmu code and ensure we properly
5  * behave across normal and unaligned accesses across several pages.
6  * We are not replicating memory tests for stuck bits and other
7  * hardware level failures but looking for issues with different size
8  * accesses when access is:
9  *
10  *   - unaligned at various sizes (if -DCHECK_UNALIGNED set)
11  *   - spanning a (softmmu) page
12  *   - sign extension when loading
13  */
14 
15 #include <inttypes.h>
16 #include <stdbool.h>
17 #include <minilib.h>
18 
19 #ifndef CHECK_UNALIGNED
20 # error "Target does not specify CHECK_UNALIGNED"
21 #endif
22 
23 #define MEM_PAGE_SIZE 4096             /* nominal 4k "pages" */
24 #define TEST_SIZE (MEM_PAGE_SIZE * 4)  /* 4 pages */
25 
26 #define ARRAY_SIZE(x) ((sizeof(x) / sizeof((x)[0])))
27 
28 __attribute__((aligned(MEM_PAGE_SIZE)))
29 static uint8_t test_data[TEST_SIZE];
30 
31 typedef void (*init_ufn) (int offset);
32 typedef bool (*read_ufn) (int offset);
33 typedef bool (*read_sfn) (int offset, bool nf);
34 
35 static void pdot(int count)
36 {
37     if (count % 128 == 0) {
38         ml_printf(".");
39     }
40 }
41 
42 /*
43  * Helper macros for shift/extract so we can keep our endian handling
44  * in one place.
45  */
46 #define BYTE_SHIFT(b, pos) ((uint64_t)b << (pos * 8))
47 #define BYTE_EXTRACT(b, pos) ((b >> (pos * 8)) & 0xff)
48 
49 /*
50  * Fill the data with ascending value bytes.
51  *
52  * Currently we only support Little Endian machines so write in
53  * ascending address order. When we read higher address bytes should
54  * either be zero or higher than the lower bytes.
55  */
56 
57 static void init_test_data_u8(int unused_offset)
58 {
59     uint8_t count = 0, *ptr = &test_data[0];
60     int i;
61     (void)(unused_offset);
62 
63     ml_printf("Filling test area with u8:");
64     for (i = 0; i < TEST_SIZE; i++) {
65         *ptr++ = count++;
66         pdot(i);
67     }
68     ml_printf("done\n");
69 }
70 
71 /*
72  * Full the data with alternating positive and negative bytes. This
73  * should mean for reads larger than a byte all subsequent reads will
74  * stay either negative or positive. We never write 0.
75  */
76 
77 static inline uint8_t get_byte(int index, bool neg)
78 {
79     return neg ? (0xff << (index % 7)) : (0xff >> ((index % 6) + 1));
80 }
81 
82 static void init_test_data_s8(bool neg_first)
83 {
84     uint8_t top, bottom, *ptr = &test_data[0];
85     int i;
86 
87     ml_printf("Filling test area with s8 pairs (%s):",
88               neg_first ? "neg first" : "pos first");
89     for (i = 0; i < TEST_SIZE / 2; i++) {
90         *ptr++ = get_byte(i, neg_first);
91         *ptr++ = get_byte(i, !neg_first);
92         pdot(i);
93     }
94     ml_printf("done\n");
95 }
96 
97 /*
98  * Zero the first few bytes of the test data in preparation for
99  * new offset values.
100  */
101 static void reset_start_data(int offset)
102 {
103     uint32_t *ptr = (uint32_t *) &test_data[0];
104     int i;
105     for (i = 0; i < offset; i++) {
106         *ptr++ = 0;
107     }
108 }
109 
110 static void init_test_data_u16(int offset)
111 {
112     uint8_t count = 0;
113     uint16_t word, *ptr = (uint16_t *) &test_data[offset];
114     const int max = (TEST_SIZE - offset) / sizeof(word);
115     int i;
116 
117     ml_printf("Filling test area with u16 (offset %d, %p):", offset, ptr);
118 
119     reset_start_data(offset);
120 
121     for (i = 0; i < max; i++) {
122         uint8_t low = count++, high = count++;
123         word = BYTE_SHIFT(high, 1) | BYTE_SHIFT(low, 0);
124         *ptr++ = word;
125         pdot(i);
126     }
127     ml_printf("done @ %p\n", ptr);
128 }
129 
130 static void init_test_data_u32(int offset)
131 {
132     uint8_t count = 0;
133     uint32_t word, *ptr = (uint32_t *) &test_data[offset];
134     const int max = (TEST_SIZE - offset) / sizeof(word);
135     int i;
136 
137     ml_printf("Filling test area with u32 (offset %d, %p):", offset, ptr);
138 
139     reset_start_data(offset);
140 
141     for (i = 0; i < max; i++) {
142         uint8_t b4 = count++, b3 = count++;
143         uint8_t b2 = count++, b1 = count++;
144         word = BYTE_SHIFT(b1, 3) | BYTE_SHIFT(b2, 2) | BYTE_SHIFT(b3, 1) | b4;
145         *ptr++ = word;
146         pdot(i);
147     }
148     ml_printf("done @ %p\n", ptr);
149 }
150 
151 static void init_test_data_u64(int offset)
152 {
153     uint8_t count = 0;
154     uint64_t word, *ptr = (uint64_t *) &test_data[offset];
155     const int max = (TEST_SIZE - offset) / sizeof(word);
156     int i;
157 
158     ml_printf("Filling test area with u64 (offset %d, %p):", offset, ptr);
159 
160     reset_start_data(offset);
161 
162     for (i = 0; i < max; i++) {
163         uint8_t b8 = count++, b7 = count++;
164         uint8_t b6 = count++, b5 = count++;
165         uint8_t b4 = count++, b3 = count++;
166         uint8_t b2 = count++, b1 = count++;
167         word = BYTE_SHIFT(b1, 7) | BYTE_SHIFT(b2, 6) | BYTE_SHIFT(b3, 5) |
168                BYTE_SHIFT(b4, 4) | BYTE_SHIFT(b5, 3) | BYTE_SHIFT(b6, 2) |
169                BYTE_SHIFT(b7, 1) | b8;
170         *ptr++ = word;
171         pdot(i);
172     }
173     ml_printf("done @ %p\n", ptr);
174 }
175 
176 static bool read_test_data_u16(int offset)
177 {
178     uint16_t word, *ptr = (uint16_t *)&test_data[offset];
179     int i;
180     const int max = (TEST_SIZE - offset) / sizeof(word);
181 
182     ml_printf("Reading u16 from %#lx (offset %d):", ptr, offset);
183 
184     for (i = 0; i < max; i++) {
185         uint8_t high, low;
186         word = *ptr++;
187         high = (word >> 8) & 0xff;
188         low = word & 0xff;
189         if (high < low && high != 0) {
190             ml_printf("Error %d < %d\n", high, low);
191             return false;
192         } else {
193             pdot(i);
194         }
195 
196     }
197     ml_printf("done @ %p\n", ptr);
198     return true;
199 }
200 
201 static bool read_test_data_u32(int offset)
202 {
203     uint32_t word, *ptr = (uint32_t *)&test_data[offset];
204     int i;
205     const int max = (TEST_SIZE - offset) / sizeof(word);
206 
207     ml_printf("Reading u32 from %#lx (offset %d):", ptr, offset);
208 
209     for (i = 0; i < max; i++) {
210         uint8_t b1, b2, b3, b4;
211         int zeros = 0;
212         word = *ptr++;
213 
214         b1 = word >> 24 & 0xff;
215         b2 = word >> 16 & 0xff;
216         b3 = word >> 8 & 0xff;
217         b4 = word & 0xff;
218 
219         zeros += (b1 == 0 ? 1 : 0);
220         zeros += (b2 == 0 ? 1 : 0);
221         zeros += (b3 == 0 ? 1 : 0);
222         zeros += (b4 == 0 ? 1 : 0);
223         if (zeros > 1) {
224             ml_printf("Error @ %p, more zeros than expected: %d, %d, %d, %d",
225                       ptr - 1, b1, b2, b3, b4);
226             return false;
227         }
228 
229         if ((b1 < b2 && b1 != 0) ||
230             (b2 < b3 && b2 != 0) ||
231             (b3 < b4 && b3 != 0)) {
232             ml_printf("Error %d, %d, %d, %d", b1, b2, b3, b4);
233             return false;
234         } else {
235             pdot(i);
236         }
237     }
238     ml_printf("done @ %p\n", ptr);
239     return true;
240 }
241 
242 static bool read_test_data_u64(int offset)
243 {
244     uint64_t word, *ptr = (uint64_t *)&test_data[offset];
245     int i;
246     const int max = (TEST_SIZE - offset) / sizeof(word);
247 
248     ml_printf("Reading u64 from %#lx (offset %d):", ptr, offset);
249 
250     for (i = 0; i < max; i++) {
251         uint8_t b1, b2, b3, b4, b5, b6, b7, b8;
252         int zeros = 0;
253         word = *ptr++;
254 
255         b1 = ((uint64_t) (word >> 56)) & 0xff;
256         b2 = ((uint64_t) (word >> 48)) & 0xff;
257         b3 = ((uint64_t) (word >> 40)) & 0xff;
258         b4 = (word >> 32) & 0xff;
259         b5 = (word >> 24) & 0xff;
260         b6 = (word >> 16) & 0xff;
261         b7 = (word >> 8)  & 0xff;
262         b8 = (word >> 0)  & 0xff;
263 
264         zeros += (b1 == 0 ? 1 : 0);
265         zeros += (b2 == 0 ? 1 : 0);
266         zeros += (b3 == 0 ? 1 : 0);
267         zeros += (b4 == 0 ? 1 : 0);
268         zeros += (b5 == 0 ? 1 : 0);
269         zeros += (b6 == 0 ? 1 : 0);
270         zeros += (b7 == 0 ? 1 : 0);
271         zeros += (b8 == 0 ? 1 : 0);
272         if (zeros > 1) {
273             ml_printf("Error @ %p, more zeros than expected: %d, %d, %d, %d, %d, %d, %d, %d",
274                       ptr - 1, b1, b2, b3, b4, b5, b6, b7, b8);
275             return false;
276         }
277 
278         if ((b1 < b2 && b1 != 0) ||
279             (b2 < b3 && b2 != 0) ||
280             (b3 < b4 && b3 != 0) ||
281             (b4 < b5 && b4 != 0) ||
282             (b5 < b6 && b5 != 0) ||
283             (b6 < b7 && b6 != 0) ||
284             (b7 < b8 && b7 != 0)) {
285             ml_printf("Error %d, %d, %d, %d, %d, %d, %d, %d",
286                       b1, b2, b3, b4, b5, b6, b7, b8);
287             return false;
288         } else {
289             pdot(i);
290         }
291     }
292     ml_printf("done @ %p\n", ptr);
293     return true;
294 }
295 
296 /* Read the test data and verify at various offsets */
297 read_ufn read_ufns[] = { read_test_data_u16,
298                          read_test_data_u32,
299                          read_test_data_u64 };
300 
301 bool do_unsigned_reads(int start_off)
302 {
303     int i;
304     bool ok = true;
305 
306     for (i = 0; i < ARRAY_SIZE(read_ufns) && ok; i++) {
307 #if CHECK_UNALIGNED
308         int off;
309         for (off = start_off; off < 8 && ok; off++) {
310             ok = read_ufns[i](off);
311         }
312 #else
313         ok = read_ufns[i](start_off);
314 #endif
315     }
316 
317     return ok;
318 }
319 
320 static bool do_unsigned_test(init_ufn fn)
321 {
322 #if CHECK_UNALIGNED
323     bool ok = true;
324     int i;
325     for (i = 0; i < 8 && ok; i++) {
326         fn(i);
327         ok = do_unsigned_reads(i);
328     }
329 #else
330     fn(0);
331     return do_unsigned_reads(0);
332 #endif
333 }
334 
335 /*
336  * We need to ensure signed data is read into a larger data type to
337  * ensure that sign extension is working properly.
338  */
339 
340 static bool read_test_data_s8(int offset, bool neg_first)
341 {
342     int8_t *ptr = (int8_t *)&test_data[offset];
343     int i;
344     const int max = (TEST_SIZE - offset) / 2;
345 
346     ml_printf("Reading s8 pairs from %#lx (offset %d):", ptr, offset);
347 
348     for (i = 0; i < max; i++) {
349         int16_t first, second;
350         bool ok;
351         first = *ptr++;
352         second = *ptr++;
353 
354         if (neg_first && first < 0 && second > 0) {
355             pdot(i);
356         } else if (!neg_first && first > 0 && second < 0) {
357             pdot(i);
358         } else {
359             ml_printf("Error %d %c %d\n", first, neg_first ? '<' : '>', second);
360             return false;
361         }
362     }
363     ml_printf("done @ %p\n", ptr);
364     return true;
365 }
366 
367 static bool read_test_data_s16(int offset, bool neg_first)
368 {
369     int16_t *ptr = (int16_t *)&test_data[offset];
370     int i;
371     const int max = (TEST_SIZE - offset) / (sizeof(*ptr));
372 
373     ml_printf("Reading s16 from %#lx (offset %d, %s):", ptr,
374               offset, neg_first ? "neg" : "pos");
375 
376     for (i = 0; i < max; i++) {
377         int32_t data = *ptr++;
378 
379         if (neg_first && data < 0) {
380             pdot(i);
381         } else if (data > 0) {
382             pdot(i);
383         } else {
384             ml_printf("Error %d %c 0\n", data, neg_first ? '<' : '>');
385             return false;
386         }
387     }
388     ml_printf("done @ %p\n", ptr);
389     return true;
390 }
391 
392 static bool read_test_data_s32(int offset, bool neg_first)
393 {
394     int32_t *ptr = (int32_t *)&test_data[offset];
395     int i;
396     const int max = (TEST_SIZE - offset) / (sizeof(int32_t));
397 
398     ml_printf("Reading s32 from %#lx (offset %d, %s):",
399               ptr, offset, neg_first ? "neg" : "pos");
400 
401     for (i = 0; i < max; i++) {
402         int64_t data = *ptr++;
403 
404         if (neg_first && data < 0) {
405             pdot(i);
406         } else if (data > 0) {
407             pdot(i);
408         } else {
409             ml_printf("Error %d %c 0\n", data, neg_first ? '<' : '>');
410             return false;
411         }
412     }
413     ml_printf("done @ %p\n", ptr);
414     return true;
415 }
416 
417 /*
418  * Read the test data and verify at various offsets
419  *
420  * For everything except bytes all our reads should be either positive
421  * or negative depending on what offset we are reading from. Currently
422  * we only handle LE systems.
423  */
424 read_sfn read_sfns[] = { read_test_data_s8,
425                          read_test_data_s16,
426                          read_test_data_s32 };
427 
428 bool do_signed_reads(bool neg_first)
429 {
430     int i;
431     bool ok = true;
432 
433     for (i = 0; i < ARRAY_SIZE(read_sfns) && ok; i++) {
434 #if CHECK_UNALIGNED
435         int off;
436         for (off = 0; off < 8 && ok; off++) {
437             bool nf = i == 0 ? neg_first ^ (off & 1) : !(neg_first ^ (off & 1));
438             ok = read_sfns[i](off, nf);
439         }
440 #else
441         ok = read_sfns[i](0, i == 0 ? neg_first : !neg_first);
442 #endif
443     }
444 
445     return ok;
446 }
447 
448 init_ufn init_ufns[] = { init_test_data_u8,
449                          init_test_data_u16,
450                          init_test_data_u32,
451                          init_test_data_u64 };
452 
453 int main(void)
454 {
455     int i;
456     bool ok = true;
457 
458     /* Run through the unsigned tests first */
459     for (i = 0; i < ARRAY_SIZE(init_ufns) && ok; i++) {
460         ok = do_unsigned_test(init_ufns[i]);
461     }
462 
463     if (ok) {
464         init_test_data_s8(false);
465         ok = do_signed_reads(false);
466     }
467 
468     if (ok) {
469         init_test_data_s8(true);
470         ok = do_signed_reads(true);
471     }
472 
473     ml_printf("Test complete: %s\n", ok ? "PASSED" : "FAILED");
474     return ok ? 0 : -1;
475 }
476