/* SPDX-License-Identifier: GPL-2.0 */ /* * Very simple but very effective user-space memory tester. * Originally by Simon Kirby * Version 2 by Charles Cazabon * Version 3 not publicly released. * Version 4 rewrite: * Copyright (C) 2004-2012 Charles Cazabon * Licensed under the terms of the GNU General Public License version 2 (only). * See the file COPYING for details. * * This file contains the functions for the actual tests, called from the * main routine in memtester.c. See other comments in that file. * */ #include "memtester.h" #include "sizes.h" #include "types.h" #include "../io_map.h" union { unsigned char bytes[UL_LEN / 8]; u32 val; } mword8; union { unsigned short u16s[UL_LEN / 16]; u32 val; } mword16; char progress[] = "-\\|/"; #define PROGRESSLEN 4 #define PROGRESSOFTEN 2500 #define ONE 0x00000001L #define fflush(n) /* Function definitions. */ int compare_regions(u32v *bufa, u32v *bufb, size_t count) { int r = 0; size_t i; u32v *p1 = bufa; u32v *p2 = bufb; off_t physaddr; for (i = 0; i < count; i++, p1++, p2++) { if (*p1 != *p2) { if (use_phys) { physaddr = physaddrbase + (i * sizeof(u32v)); fprintf(stderr, "FAILURE: 0x%08lx != 0x%08lx at physical address " "0x%08lx.\n", (ul)*p1, (ul)*p2, physaddr); } else { fprintf(stderr, "FAILURE: 0x%08lx != 0x%08lx at offset 0x%08lx.\n", (ul)*p1, (ul)*p2, (ul)(i * sizeof(u32v))); } /* printf("Skipping to next test..."); */ r = -1; } } return r; } int test_stuck_address(u32v *bufa, size_t count) { u32v *p1 = bufa; unsigned int j; size_t i; off_t physaddr; printf(" "); fflush(stdout); for (j = 0; j < 16; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); p1 = (u32v *)bufa; printf("setting %3u", j); fflush(stdout); for (i = 0; i < count; i++) { *p1 = ((j + i) % 2) == 0 ? (u32)(ul)p1 : ~((u32)(ul)p1); *p1++; } printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); p1 = (u32v *)bufa; for (i = 0; i < count; i++, p1++) { if (*p1 != (((j + i) % 2) == 0 ? (u32)(ul)p1 : ~((u32)(ul)p1))) { if (use_phys) { physaddr = physaddrbase + (i * sizeof(u32v)); fprintf(stderr, "FAILURE: possible bad address line at physical " "address 0x%08lx.\n", physaddr); } else { fprintf(stderr, "FAILURE: possible bad address line at offset " "0x%08lx.\n", (ul)(i * sizeof(u32v))); } printf("Skipping to next test...\n"); fflush(stdout); return -1; } } } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_random_value(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; ul j = 0; size_t i; putc(' '); fflush(stdout); for (i = 0; i < count; i++) { *p1++ = *p2++ = rand_ul(); if (!(i % PROGRESSOFTEN)) { putc('\b'); putc(progress[++j % PROGRESSLEN]); fflush(stdout); } } printf("\b \b"); fflush(stdout); return compare_regions(bufa, bufb, count); } int test_xor_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { *p1++ ^= q; *p2++ ^= q; } return compare_regions(bufa, bufb, count); } int test_sub_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { *p1++ -= q; *p2++ -= q; } return compare_regions(bufa, bufb, count); } int test_mul_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { *p1++ *= q; *p2++ *= q; } return compare_regions(bufa, bufb, count); } int test_div_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { if (!q) q++; *p1++ /= q; *p2++ /= q; } return compare_regions(bufa, bufb, count); } int test_or_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { *p1++ |= q; *p2++ |= q; } return compare_regions(bufa, bufb, count); } int test_and_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) { *p1++ &= q; *p2++ &= q; } return compare_regions(bufa, bufb, count); } int test_seqinc_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; size_t i; u32 q = rand_ul(); for (i = 0; i < count; i++) *p1++ = *p2++ = (i + q); return compare_regions(bufa, bufb, count); } int test_solidbits_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 q; u32 data[4]; size_t i; printf(" "); fflush(stdout); for (j = 0; j < 64; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); q = (j % 2) == 0 ? UL_ONEBITS : 0; if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = data[2] = q; data[1] = data[3] = ~q; data_cpu_2_io(data, sizeof(data)); printf("setting %3u", j); fflush(stdout); p1 = (u32v *)bufa; p2 = (u32v *)bufb; for (i = 0; i < count; i++) *p1++ = *p2++ = data[i & 3]; printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_checkerboard_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 q; u32 data[4]; size_t i; printf(" "); fflush(stdout); for (j = 0; j < 64; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); q = (j % 2) == 0 ? CHECKERBOARD1 : CHECKERBOARD2; if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = data[2] = q; data[1] = data[3] = ~q; data_cpu_2_io(data, sizeof(data)); printf("setting %3u", j); fflush(stdout); p1 = (u32v *)bufa; p2 = (u32v *)bufb; for (i = 0; i < count; i++) *p1++ = *p2++ = data[i & 3]; printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_blockseq_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 data[4]; u32 q; size_t i; printf(" "); fflush(stdout); for (j = 0; j < 256; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); p1 = (u32v *)bufa; p2 = (u32v *)bufb; printf("setting %3u", j); fflush(stdout); q = (u32)UL_BYTE(j); if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = q; data[1] = q; data[2] = q; data[3] = q; data_cpu_2_io(data, sizeof(data)); for (i = 0; i < count; i++) *p1++ = *p2++ = data[i & 3]; printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_walkbits0_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 data[4]; u32 q; size_t i; printf(" "); fflush(stdout); for (j = 0; j < UL_LEN * 2; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); p1 = (u32v *)bufa; p2 = (u32v *)bufb; printf("setting %3u", j); fflush(stdout); if (j < UL_LEN) q = ONE << j; else q = ONE << (UL_LEN * 2 - j - 1); if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = q; data[1] = q; data[2] = q; data[3] = q; data_cpu_2_io(data, sizeof(data)); for (i = 0; i < count; i++) { *p1++ = *p2++ = data[i & 3]; } printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_walkbits1_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 data[4]; u32 q; size_t i; printf(" "); fflush(stdout); for (j = 0; j < UL_LEN * 2; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); p1 = (u32v *)bufa; p2 = (u32v *)bufb; printf("setting %3u", j); fflush(stdout); if (j < UL_LEN) q = UL_ONEBITS ^ (ONE << j); else q = UL_ONEBITS ^ (ONE << (UL_LEN * 2 - j - 1)); if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = q; data[1] = q; data[2] = q; data[3] = q; data_cpu_2_io(data, sizeof(data)); for (i = 0; i < count; i++) { *p1++ = *p2++ = data[i & 3]; } printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_bitspread_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j; u32 data[4]; size_t i; printf(" "); fflush(stdout); for (j = 0; j < UL_LEN * 2; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); p1 = (u32v *)bufa; p2 = (u32v *)bufb; printf("setting %3u", j); fflush(stdout); if (j < UL_LEN) { data[0] = (ONE << j) | (ONE << (j + 2)); data[1] = UL_ONEBITS ^ ((ONE << j) | (ONE << (j + 2))); } else { data[0] = (ONE << (UL_LEN * 2 - 1 - j)) | (ONE << (UL_LEN * 2 + 1 - j)); data[1] = UL_ONEBITS ^ (ONE << (UL_LEN * 2 - 1 - j) | (ONE << (UL_LEN * 2 + 1 - j))); } if (fix_level) { data[0] |= fix_bit; data[1] |= fix_bit; } else { data[0] &= ~fix_bit; data[1] &= ~fix_bit; } data[2] = data[0]; data[3] = data[1]; data_cpu_2_io(data, sizeof(data)); for (i = 0; i < count; i++) { *p1++ = *p2++ = data[i & 3]; } printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } int test_bitflip_comparison(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u32v *p1 = bufa; u32v *p2 = bufb; unsigned int j, k; u32 q; u32 data[4]; size_t i; printf(" "); fflush(stdout); for (k = 0; k < UL_LEN; k++) { q = ONE << k; for (j = 0; j < 8; j++) { printf("\b\b\b\b\b\b\b\b\b\b\b"); q = ~q; printf("setting %3u", k * 8 + j); fflush(stdout); if (fix_level) q |= fix_bit; else q &= ~fix_bit; data[0] = data[2] = q; data[1] = data[3] = ~q; data_cpu_2_io(data, sizeof(data)); p1 = (u32v *)bufa; p2 = (u32v *)bufb; for (i = 0; i < count; i++) *p1++ = *p2++ = data[i & 3]; printf("\b\b\b\b\b\b\b\b\b\b\b"); printf("testing %3u", k * 8 + j); fflush(stdout); if (compare_regions(bufa, bufb, count)) return -1; } } printf("\b\b\b\b\b\b\b\b\b\b\b \b\b\b\b\b\b\b\b\b\b\b"); fflush(stdout); return 0; } #ifdef TEST_NARROW_WRITES int test_8bit_wide_random(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u8v *p1, *t; u32v *p2; int attempt; unsigned int b, j = 0; size_t i; putc(' '); fflush(stdout); for (attempt = 0; attempt < 2; attempt++) { if (attempt & 1) { p1 = (u8v *)bufa; p2 = bufb; } else { p1 = (u8v *)bufb; p2 = bufa; } for (i = 0; i < count; i++) { t = mword8.bytes; *p2++ = mword8.val = rand_ul(); for (b = 0; b < UL_LEN / 8; b++) *p1++ = *t++; if (!(i % PROGRESSOFTEN)) { putc('\b'); putc(progress[++j % PROGRESSLEN]); fflush(stdout); } } if (compare_regions(bufa, bufb, count)) return -1; } printf("\b \b"); fflush(stdout); return 0; } int test_16bit_wide_random(u32v *bufa, u32v *bufb, size_t count, ul fix_bit, ul fix_level) { u16v *p1, *t; u32v *p2; int attempt; unsigned int b, j = 0; size_t i; putc(' '); fflush(stdout); for (attempt = 0; attempt < 2; attempt++) { if (attempt & 1) { p1 = (u16v *)bufa; p2 = bufb; } else { p1 = (u16v *)bufb; p2 = bufa; } for (i = 0; i < count; i++) { t = mword16.u16s; *p2++ = mword16.val = rand_ul(); for (b = 0; b < UL_LEN / 16; b++) *p1++ = *t++; if (!(i % PROGRESSOFTEN)) { putc('\b'); putc(progress[++j % PROGRESSLEN]); fflush(stdout); } } if (compare_regions(bufa, bufb, count)) return -1; } printf("\b \b"); fflush(stdout); return 0; } #endif