2019-05-19 13:08:20 +01:00
// SPDX-License-Identifier: GPL-2.0-only
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
2013-09-30 13:45:24 -07:00
if ( ! hwpoison_filter_enable )
goto inject ;
2017-05-03 14:56:19 -07:00
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 ) )
2020-10-15 20:06:46 -07:00
return 0 ;
2009-12-16 12:19:59 +01:00
/*
2020-10-15 20:06:46 -07:00
* do a racy check 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 )
2020-10-15 20:06:46 -07:00
return 0 ;
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 ) ;
2020-10-15 20:06:46 -07:00
return memory_failure ( pfn , 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 ) ;
}
2019-11-30 17:57:35 -08:00
DEFINE_DEBUGFS_ATTRIBUTE ( hwpoison_fops , NULL , hwpoison_inject , " %lli \n " ) ;
DEFINE_DEBUGFS_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 )
{
hwpoison_dir = debugfs_create_dir ( " hwpoison " , NULL ) ;
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 .
*/
2019-01-22 16:21:10 +01:00
debugfs_create_file ( " corrupt-pfn " , 0200 , hwpoison_dir , NULL ,
& hwpoison_fops ) ;
debugfs_create_file ( " unpoison-pfn " , 0200 , hwpoison_dir , NULL ,
& unpoison_fops ) ;
debugfs_create_u32 ( " corrupt-filter-enable " , 0600 , hwpoison_dir ,
& hwpoison_filter_enable ) ;
debugfs_create_u32 ( " corrupt-filter-dev-major " , 0600 , hwpoison_dir ,
& hwpoison_filter_dev_major ) ;
debugfs_create_u32 ( " corrupt-filter-dev-minor " , 0600 , hwpoison_dir ,
& hwpoison_filter_dev_minor ) ;
debugfs_create_u64 ( " corrupt-filter-flags-mask " , 0600 , hwpoison_dir ,
& hwpoison_filter_flags_mask ) ;
debugfs_create_u64 ( " corrupt-filter-flags-value " , 0600 , hwpoison_dir ,
& hwpoison_filter_flags_value ) ;
2009-12-16 12:19:59 +01:00
2015-09-09 15:35:31 -07:00
# ifdef CONFIG_MEMCG
2019-01-22 16:21:10 +01:00
debugfs_create_u64 ( " corrupt-filter-memcg " , 0600 , hwpoison_dir ,
& hwpoison_filter_memcg ) ;
2009-12-16 12:19:59 +01:00
# endif
2009-09-16 11:50:17 +02:00
return 0 ;
}
module_init ( pfn_inject_init ) ;
module_exit ( pfn_inject_exit ) ;
MODULE_LICENSE ( " GPL " ) ;