X-Git-Url: https://git.saurik.com/redis.git/blobdiff_plain/0d83011a11b42fe1b0d1bebb7b9a3318fd44f0e5..9c00f078978e452d541ddc8a9a2b7704db2cb7f3:/src/memtest.c diff --git a/src/memtest.c b/src/memtest.c index c8b47d48..82da27c8 100644 --- a/src/memtest.c +++ b/src/memtest.c @@ -1,3 +1,32 @@ +/* + * Copyright (c) 2009-2012, Salvatore Sanfilippo + * All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions are met: + * + * * Redistributions of source code must retain the above copyright notice, + * this list of conditions and the following disclaimer. + * * Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * * Neither the name of Redis nor the names of its contributors may be used + * to endorse or promote products derived from this software without + * specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" + * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE + * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE + * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE + * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR + * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF + * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS + * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN + * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) + * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE + * POSSIBILITY OF SUCH DAMAGE. + */ + #include #include #include @@ -6,6 +35,7 @@ #include #include #include +#include "config.h" #if (ULONG_MAX == 4294967295UL) #define MEMTEST_32BIT @@ -47,7 +77,7 @@ void memtest_progress_end(void) { } void memtest_progress_step(size_t curr, size_t size, char c) { - size_t chars = (curr*progress_full)/size, j; + size_t chars = ((unsigned long long)curr*progress_full)/size, j; for (j = 0; j < chars-progress_printed; j++) { printf("%c",c); @@ -132,13 +162,13 @@ void memtest_fill_value(unsigned long *l, size_t bytes, unsigned long v1, v = (off & 1) ? v2 : v1; for (w = 0; w < iwords; w++) { #ifdef MEMTEST_32BIT - *l1 = *l2 = ((unsigned long) (rand()&0xffff)) | - (((unsigned long) (rand()&0xffff)) << 16); + *l1 = *l2 = ((unsigned long) v) | + (((unsigned long) v) << 16); #else - *l1 = *l2 = ((unsigned long) (rand()&0xffff)) | - (((unsigned long) (rand()&0xffff)) << 16) | - (((unsigned long) (rand()&0xffff)) << 32) | - (((unsigned long) (rand()&0xffff)) << 48); + *l1 = *l2 = ((unsigned long) v) | + (((unsigned long) v) << 16) | + (((unsigned long) v) << 32) | + (((unsigned long) v) << 48); #endif l1 += step; l2 += step; @@ -211,6 +241,36 @@ void memtest_test(size_t megabytes, int passes) { } } +/* This is a fast O(N) best effort memory test, only ZERO-ONE tests and + * checkerboard tests are performed, without pauses between setting and + * reading the value, so this can only detect a subclass of permanent errors. + * + * However the function does not destroy the content of the memory tested that + * is left unmodified. + * + * If a memory error is detected, 1 is returned. Otherwise 0 is returned. */ +int memtest_non_destructive(void *addr, size_t size) { + volatile unsigned long *p = addr; + unsigned long val; + size_t j; + + size /= sizeof(unsigned long); + for (j = 0; j < size; j++) { + val = p[j]; + + p[j] = 0; if (p[j] != 0) goto err; + p[j] = (unsigned long)-1; if (p[j] != (unsigned long)-1) goto err; + p[j] = ULONG_ONEZERO; if (p[j] != ULONG_ONEZERO) goto err; + p[j] = ULONG_ZEROONE; if (p[j] != ULONG_ZEROONE) goto err; + p[j] = val; /* restore the original value. */ + } + return 0; + +err: /* memory error detected. */ + p[j] = val; + return 1; +} + void memtest(size_t megabytes, int passes) { if (ioctl(1, TIOCGWINSZ, &ws) == -1) { ws.ws_col = 80;