2008-07-15 00:02:28 -07:00
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/string.h>
# include <linux/types.h>
# include <linux/mm.h>
# include <linux/smp.h>
# include <linux/init.h>
# include <linux/pfn.h>
2010-08-25 13:39:17 -07:00
# include <linux/memblock.h>
2008-07-15 00:02:28 -07:00
2009-02-25 11:27:27 +01:00
static u64 patterns [ ] __initdata = {
0 ,
0xffffffffffffffffULL ,
0x5555555555555555ULL ,
0xaaaaaaaaaaaaaaaaULL ,
2009-02-25 11:31:49 +01:00
0x1111111111111111ULL ,
0x2222222222222222ULL ,
0x4444444444444444ULL ,
0x8888888888888888ULL ,
0x3333333333333333ULL ,
0x6666666666666666ULL ,
0x9999999999999999ULL ,
0xccccccccccccccccULL ,
0x7777777777777777ULL ,
0xbbbbbbbbbbbbbbbbULL ,
0xddddddddddddddddULL ,
0xeeeeeeeeeeeeeeeeULL ,
0x7a6c7258554e494cULL , /* yeah ;-) */
2009-02-25 11:27:27 +01:00
} ;
2009-02-25 11:26:18 +01:00
2009-02-25 11:28:58 +01:00
static void __init reserve_bad_mem ( u64 pattern , u64 start_bad , u64 end_bad )
2009-02-25 11:28:07 +01:00
{
2009-02-25 11:28:58 +01:00
printk ( KERN_INFO " %016llx bad mem addr %010llx - %010llx reserved \n " ,
( unsigned long long ) pattern ,
( unsigned long long ) start_bad ,
( unsigned long long ) end_bad ) ;
2010-08-25 13:39:17 -07:00
memblock_x86_reserve_range ( start_bad , end_bad , " BAD RAM " ) ;
2009-02-25 11:28:07 +01:00
}
2009-02-25 11:28:58 +01:00
static void __init memtest ( u64 pattern , u64 start_phys , u64 size )
2008-07-15 00:02:28 -07:00
{
2009-06-11 16:25:01 +02:00
u64 * p , * start , * end ;
2009-02-25 11:28:58 +01:00
u64 start_bad , last_bad ;
u64 start_phys_aligned ;
2009-06-11 16:25:01 +02:00
const size_t incr = sizeof ( pattern ) ;
2008-07-15 00:02:28 -07:00
start_phys_aligned = ALIGN ( start_phys , incr ) ;
start = __va ( start_phys_aligned ) ;
2009-06-11 16:25:01 +02:00
end = start + ( size - ( start_phys_aligned - start_phys ) ) / incr ;
2008-07-15 00:02:28 -07:00
start_bad = 0 ;
last_bad = 0 ;
2009-06-08 19:09:39 +02:00
for ( p = start ; p < end ; p + + )
* p = pattern ;
2009-06-11 16:25:01 +02:00
2009-06-08 19:09:39 +02:00
for ( p = start ; p < end ; p + + , start_phys_aligned + = incr ) {
if ( * p = = pattern )
2009-02-25 11:28:07 +01:00
continue ;
if ( start_phys_aligned = = last_bad + incr ) {
last_bad + = incr ;
continue ;
2008-07-15 00:02:28 -07:00
}
2009-02-25 11:28:07 +01:00
if ( start_bad )
reserve_bad_mem ( pattern , start_bad , last_bad + incr ) ;
start_bad = last_bad = start_phys_aligned ;
2008-07-15 00:02:28 -07:00
}
2009-02-25 11:28:07 +01:00
if ( start_bad )
reserve_bad_mem ( pattern , start_bad , last_bad + incr ) ;
2008-07-15 00:02:28 -07:00
}
2009-02-25 11:30:04 +01:00
static void __init do_one_pass ( u64 pattern , u64 start , u64 end )
{
u64 size = 0 ;
while ( start < end ) {
2010-08-25 13:39:17 -07:00
start = memblock_x86_find_in_range_size ( start , & size , 1 ) ;
2009-02-25 11:30:04 +01:00
/* done ? */
if ( start > = end )
break ;
if ( start + size > end )
size = end - start ;
printk ( KERN_INFO " %010llx - %010llx pattern %016llx \n " ,
( unsigned long long ) start ,
( unsigned long long ) start + size ,
( unsigned long long ) cpu_to_be64 ( pattern ) ) ;
memtest ( pattern , start , size ) ;
start + = size ;
}
}
2008-07-15 00:02:28 -07:00
/* default is disabled */
static int memtest_pattern __initdata ;
static int __init parse_memtest ( char * arg )
{
if ( arg )
memtest_pattern = simple_strtoul ( arg , NULL , 0 ) ;
2009-03-06 03:12:50 -08:00
else
memtest_pattern = ARRAY_SIZE ( patterns ) ;
2008-07-15 00:02:28 -07:00
return 0 ;
}
early_param ( " memtest " , parse_memtest ) ;
void __init early_memtest ( unsigned long start , unsigned long end )
{
2009-02-25 11:27:27 +01:00
unsigned int i ;
2009-02-25 11:30:04 +01:00
unsigned int idx = 0 ;
2008-07-15 00:02:28 -07:00
if ( ! memtest_pattern )
return ;
2009-02-25 11:28:58 +01:00
printk ( KERN_INFO " early_memtest: # of tests: %d \n " , memtest_pattern ) ;
2009-02-25 11:27:27 +01:00
for ( i = 0 ; i < memtest_pattern ; i + + ) {
2009-02-25 11:30:04 +01:00
idx = i % ARRAY_SIZE ( patterns ) ;
do_one_pass ( patterns [ idx ] , start , end ) ;
}
if ( idx > 0 ) {
printk ( KERN_INFO " early_memtest: wipe out "
" test pattern from memory \n " ) ;
/* additional test with pattern 0 will do this */
do_one_pass ( 0 , start , end ) ;
2008-07-15 00:02:28 -07:00
}
}