2011-03-30 22:57:33 -03:00
/* Inject a hwpoison memory failure on a arbitrary pfn */
2009-09-16 11:50:17 +02:00
# include <linux/module.h>
# include <linux/debugfs.h>
# include <linux/kernel.h>
# include <linux/mm.h>
2009-12-16 12:19:59 +01:00
# include <linux/swap.h>
# include <linux/pagemap.h>
2010-05-28 09:29:22 +09:00
# include <linux/hugetlb.h>
2009-12-16 12:19:59 +01:00
# include "internal.h"
2009-09-16 11:50:17 +02:00
2009-12-16 12:19:58 +01:00
static struct dentry * hwpoison_dir ;
2009-09-16 11:50:17 +02:00
static int hwpoison_inject ( void * data , u64 val )
{
2009-12-16 12:19:59 +01:00
unsigned long pfn = val ;
struct page * p ;
2010-05-28 09:29:22 +09:00
struct page * hpage ;
2009-12-16 12:19:59 +01:00
int err ;
2009-09-16 11:50:17 +02:00
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EPERM ;
2009-12-16 12:19:59 +01:00
if ( ! pfn_valid ( pfn ) )
return - ENXIO ;
p = pfn_to_page ( pfn ) ;
2010-05-28 09:29:22 +09:00
hpage = compound_head ( p ) ;
2009-12-16 12:19:59 +01:00
/*
* This implies unable to support free buddy pages .
*/
2015-06-24 16:56:48 -07:00
if ( ! get_hwpoison_page ( p ) )
2009-12-16 12:19:59 +01:00
return 0 ;
2013-09-30 13:45:24 -07:00
if ( ! hwpoison_filter_enable )
goto inject ;
2015-05-05 16:23:52 -07:00
if ( ! PageLRU ( hpage ) & & ! PageHuge ( p ) )
shake_page ( hpage , 0 ) ;
2009-12-16 12:19:59 +01:00
/*
* This implies unable to support non - LRU pages .
*/
2015-05-05 16:23:52 -07:00
if ( ! PageLRU ( hpage ) & & ! PageHuge ( p ) )
2015-05-05 16:23:49 -07:00
goto put_out ;
2009-12-16 12:19:59 +01:00
/*
* 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 ) .
2011-12-15 10:48:12 -08:00
* memory_failure ( ) will redo the check reliably inside page lock .
2009-12-16 12:19:59 +01:00
*/
2010-05-28 09:29:22 +09:00
err = hwpoison_filter ( hpage ) ;
2009-12-16 12:19:59 +01:00
if ( err )
2015-05-05 16:23:49 -07:00
goto put_out ;
2009-12-16 12:19:59 +01:00
2009-12-16 12:20:01 +01:00
inject :
2014-01-21 15:50:56 -08:00
pr_info ( " Injecting memory failure at pfn %#lx \n " , pfn ) ;
2011-12-15 10:48:12 -08:00
return memory_failure ( pfn , 18 , MF_COUNT_INCREASED ) ;
2015-05-05 16:23:49 -07:00
put_out :
2015-09-08 15:03:18 -07:00
put_hwpoison_page ( p ) ;
2015-05-05 16:23:49 -07:00
return 0 ;
2009-09-16 11:50:17 +02:00
}
2009-12-16 12:19:58 +01:00
static int hwpoison_unpoison ( void * data , u64 val )
{
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EPERM ;
return unpoison_memory ( val ) ;
}
2009-09-16 11:50:17 +02:00
DEFINE_SIMPLE_ATTRIBUTE ( hwpoison_fops , NULL , hwpoison_inject , " %lli \n " ) ;
2009-12-16 12:19:58 +01:00
DEFINE_SIMPLE_ATTRIBUTE ( unpoison_fops , NULL , hwpoison_unpoison , " %lli \n " ) ;
2009-09-16 11:50:17 +02:00
static void pfn_inject_exit ( void )
{
2014-08-06 16:06:41 -07:00
debugfs_remove_recursive ( hwpoison_dir ) ;
2009-09-16 11:50:17 +02:00
}
static int pfn_inject_init ( void )
{
2009-12-16 12:19:58 +01:00
struct dentry * dentry ;
2009-09-16 11:50:17 +02:00
hwpoison_dir = debugfs_create_dir ( " hwpoison " , NULL ) ;
if ( hwpoison_dir = = NULL )
return - ENOMEM ;
2009-12-16 12:19:58 +01: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 .
*/
2013-09-11 14:23:00 -07:00
dentry = debugfs_create_file ( " corrupt-pfn " , 0200 , hwpoison_dir ,
2009-09-16 11:50:17 +02:00
NULL , & hwpoison_fops ) ;
2009-12-16 12:19:58 +01:00
if ( ! dentry )
goto fail ;
2013-09-11 14:23:00 -07:00
dentry = debugfs_create_file ( " unpoison-pfn " , 0200 , hwpoison_dir ,
2009-12-16 12:19:58 +01:00
NULL , & unpoison_fops ) ;
if ( ! dentry )
goto fail ;
2009-12-16 12:19:59 +01:00
dentry = debugfs_create_u32 ( " corrupt-filter-enable " , 0600 ,
hwpoison_dir , & hwpoison_filter_enable ) ;
if ( ! dentry )
goto fail ;
2009-12-16 12:19:59 +01: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-12-16 12:19:59 +01:00
dentry = debugfs_create_u64 ( " corrupt-filter-flags-mask " , 0600 ,
hwpoison_dir , & hwpoison_filter_flags_mask ) ;
if ( ! dentry )
goto fail ;
dentry = debugfs_create_u64 ( " corrupt-filter-flags-value " , 0600 ,
hwpoison_dir , & hwpoison_filter_flags_value ) ;
if ( ! dentry )
goto fail ;
2015-09-09 15:35:31 -07:00
# ifdef CONFIG_MEMCG
2009-12-16 12:19:59 +01:00
dentry = debugfs_create_u64 ( " corrupt-filter-memcg " , 0600 ,
hwpoison_dir , & hwpoison_filter_memcg ) ;
if ( ! dentry )
goto fail ;
# endif
2009-09-16 11:50:17 +02:00
return 0 ;
2009-12-16 12:19:58 +01:00
fail :
pfn_inject_exit ( ) ;
return - ENOMEM ;
2009-09-16 11:50:17 +02:00
}
module_init ( pfn_inject_init ) ;
module_exit ( pfn_inject_exit ) ;
MODULE_LICENSE ( " GPL " ) ;