2009-04-01 02:23:17 +04:00
# include <linux/kernel.h>
2011-11-01 04:08:10 +04:00
# include <linux/string.h>
2009-04-01 02:23:17 +04:00
# include <linux/mm.h>
2011-11-01 04:08:38 +04:00
# include <linux/highmem.h>
2009-04-01 02:23:17 +04:00
# include <linux/page-debug-flags.h>
# include <linux/poison.h>
2011-11-01 04:08:05 +04:00
# include <linux/ratelimit.h>
2009-04-01 02:23:17 +04:00
static inline void set_page_poison ( struct page * page )
{
__set_bit ( PAGE_DEBUG_FLAG_POISON , & page - > debug_flags ) ;
}
static inline void clear_page_poison ( struct page * page )
{
__clear_bit ( PAGE_DEBUG_FLAG_POISON , & page - > debug_flags ) ;
}
static inline bool page_poison ( struct page * page )
{
return test_bit ( PAGE_DEBUG_FLAG_POISON , & page - > debug_flags ) ;
}
static void poison_page ( struct page * page )
{
2011-11-01 04:08:38 +04:00
void * addr = kmap_atomic ( page ) ;
2009-04-01 02:23:17 +04:00
set_page_poison ( page ) ;
memset ( addr , PAGE_POISON , PAGE_SIZE ) ;
2011-11-01 04:08:38 +04:00
kunmap_atomic ( addr ) ;
2009-04-01 02:23:17 +04:00
}
static void poison_pages ( struct page * page , int n )
{
int i ;
for ( i = 0 ; i < n ; i + + )
poison_page ( page + i ) ;
}
static bool single_bit_flip ( unsigned char a , unsigned char b )
{
unsigned char error = a ^ b ;
return error & & ! ( error & ( error - 1 ) ) ;
}
static void check_poison_mem ( unsigned char * mem , size_t bytes )
{
2011-11-01 04:08:05 +04:00
static DEFINE_RATELIMIT_STATE ( ratelimit , 5 * HZ , 10 ) ;
2009-04-01 02:23:17 +04:00
unsigned char * start ;
unsigned char * end ;
2011-11-01 04:08:10 +04:00
start = memchr_inv ( mem , PAGE_POISON , bytes ) ;
if ( ! start )
2009-04-01 02:23:17 +04:00
return ;
for ( end = mem + bytes - 1 ; end > start ; end - - ) {
if ( * end ! = PAGE_POISON )
break ;
}
2011-11-01 04:08:05 +04:00
if ( ! __ratelimit ( & ratelimit ) )
2009-04-01 02:23:17 +04:00
return ;
else if ( start = = end & & single_bit_flip ( * start , PAGE_POISON ) )
printk ( KERN_ERR " pagealloc: single bit error \n " ) ;
else
printk ( KERN_ERR " pagealloc: memory corruption \n " ) ;
print_hex_dump ( KERN_ERR , " " , DUMP_PREFIX_ADDRESS , 16 , 1 , start ,
end - start + 1 , 1 ) ;
dump_stack ( ) ;
}
static void unpoison_page ( struct page * page )
{
2011-11-01 04:08:38 +04:00
void * addr ;
if ( ! page_poison ( page ) )
2009-04-01 02:23:17 +04:00
return ;
2011-11-01 04:08:38 +04:00
addr = kmap_atomic ( page ) ;
check_poison_mem ( addr , PAGE_SIZE ) ;
clear_page_poison ( page ) ;
kunmap_atomic ( addr ) ;
2009-04-01 02:23:17 +04:00
}
static void unpoison_pages ( struct page * page , int n )
{
int i ;
for ( i = 0 ; i < n ; i + + )
unpoison_page ( page + i ) ;
}
void kernel_map_pages ( struct page * page , int numpages , int enable )
{
if ( ! debug_pagealloc_enabled )
return ;
if ( enable )
unpoison_pages ( page , numpages ) ;
else
poison_pages ( page , numpages ) ;
}