2009-09-16 13:50:17 +04:00
/* Inject a hwpoison memory failure on a arbitary pfn */
# include <linux/module.h>
# include <linux/debugfs.h>
# include <linux/kernel.h>
# include <linux/mm.h>
2009-12-16 14:19:59 +03:00
# include <linux/swap.h>
# include <linux/pagemap.h>
2009-12-16 14:19:59 +03:00
# include "internal.h"
2009-09-16 13:50:17 +04:00
2009-12-16 14:19:58 +03:00
static struct dentry * hwpoison_dir ;
2009-09-16 13:50:17 +04:00
static int hwpoison_inject ( void * data , u64 val )
{
2009-12-16 14:19:59 +03:00
unsigned long pfn = val ;
struct page * p ;
int err ;
2009-09-16 13:50:17 +04:00
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EPERM ;
2009-12-16 14:19:59 +03:00
if ( ! pfn_valid ( pfn ) )
return - ENXIO ;
p = pfn_to_page ( pfn ) ;
/*
* This implies unable to support free buddy pages .
*/
if ( ! get_page_unless_zero ( p ) )
return 0 ;
if ( ! PageLRU ( p ) )
shake_page ( p ) ;
/*
* This implies unable to support non - LRU pages .
*/
if ( ! PageLRU ( p ) )
return 0 ;
/*
* do a racy check with elevated page count , to make sure PG_hwpoison
* will only be set for the targeted owner ( or on a free page ) .
* We temporarily take page lock for try_get_mem_cgroup_from_page ( ) .
* __memory_failure ( ) will redo the check reliably inside page lock .
*/
lock_page ( p ) ;
err = hwpoison_filter ( p ) ;
unlock_page ( p ) ;
if ( err )
return 0 ;
printk ( KERN_INFO " Injecting memory failure at pfn %lx \n " , pfn ) ;
return __memory_failure ( pfn , 18 , MF_COUNT_INCREASED ) ;
2009-09-16 13:50:17 +04:00
}
2009-12-16 14:19:58 +03:00
static int hwpoison_unpoison ( void * data , u64 val )
{
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EPERM ;
return unpoison_memory ( val ) ;
}
2009-09-16 13:50:17 +04:00
DEFINE_SIMPLE_ATTRIBUTE ( hwpoison_fops , NULL , hwpoison_inject , " %lli \n " ) ;
2009-12-16 14:19:58 +03:00
DEFINE_SIMPLE_ATTRIBUTE ( unpoison_fops , NULL , hwpoison_unpoison , " %lli \n " ) ;
2009-09-16 13:50:17 +04:00
static void pfn_inject_exit ( void )
{
if ( hwpoison_dir )
debugfs_remove_recursive ( hwpoison_dir ) ;
}
static int pfn_inject_init ( void )
{
2009-12-16 14:19:58 +03:00
struct dentry * dentry ;
2009-09-16 13:50:17 +04:00
hwpoison_dir = debugfs_create_dir ( " hwpoison " , NULL ) ;
if ( hwpoison_dir = = NULL )
return - ENOMEM ;
2009-12-16 14:19:58 +03:00
/*
* Note that the below poison / unpoison interfaces do not involve
* hardware status change , hence do not require hardware support .
* They are mainly for testing hwpoison in software level .
*/
dentry = debugfs_create_file ( " corrupt-pfn " , 0600 , hwpoison_dir ,
2009-09-16 13:50:17 +04:00
NULL , & hwpoison_fops ) ;
2009-12-16 14:19:58 +03:00
if ( ! dentry )
goto fail ;
dentry = debugfs_create_file ( " unpoison-pfn " , 0600 , hwpoison_dir ,
NULL , & unpoison_fops ) ;
if ( ! dentry )
goto fail ;
2009-12-16 14:19:59 +03:00
dentry = debugfs_create_u32 ( " corrupt-filter-dev-major " , 0600 ,
hwpoison_dir , & hwpoison_filter_dev_major ) ;
if ( ! dentry )
goto fail ;
dentry = debugfs_create_u32 ( " corrupt-filter-dev-minor " , 0600 ,
hwpoison_dir , & hwpoison_filter_dev_minor ) ;
if ( ! dentry )
goto fail ;
2009-09-16 13:50:17 +04:00
return 0 ;
2009-12-16 14:19:58 +03:00
fail :
pfn_inject_exit ( ) ;
return - ENOMEM ;
2009-09-16 13:50:17 +04:00
}
module_init ( pfn_inject_init ) ;
module_exit ( pfn_inject_exit ) ;
MODULE_LICENSE ( " GPL " ) ;