diff options
Diffstat (limited to 'mm/memory-failure.c')
| -rw-r--r-- | mm/memory-failure.c | 122 | 
1 files changed, 25 insertions, 97 deletions
| diff --git a/mm/memory-failure.c b/mm/memory-failure.c index df6ee59527dd..3edebb0cda30 100644 --- a/mm/memory-failure.c +++ b/mm/memory-failure.c @@ -212,106 +212,34 @@ static bool page_handle_poison(struct page *page, bool hugepage_or_freepage, boo  	return true;  } -#if IS_ENABLED(CONFIG_HWPOISON_INJECT) +static hwpoison_filter_func_t __rcu *hwpoison_filter_func __read_mostly; -u32 hwpoison_filter_enable = 0; -u32 hwpoison_filter_dev_major = ~0U; -u32 hwpoison_filter_dev_minor = ~0U; -u64 hwpoison_filter_flags_mask; -u64 hwpoison_filter_flags_value; -EXPORT_SYMBOL_GPL(hwpoison_filter_enable); -EXPORT_SYMBOL_GPL(hwpoison_filter_dev_major); -EXPORT_SYMBOL_GPL(hwpoison_filter_dev_minor); -EXPORT_SYMBOL_GPL(hwpoison_filter_flags_mask); -EXPORT_SYMBOL_GPL(hwpoison_filter_flags_value); - -static int hwpoison_filter_dev(struct page *p) +void hwpoison_filter_register(hwpoison_filter_func_t *filter)  { -	struct folio *folio = page_folio(p); -	struct address_space *mapping; -	dev_t dev; - -	if (hwpoison_filter_dev_major == ~0U && -	    hwpoison_filter_dev_minor == ~0U) -		return 0; - -	mapping = folio_mapping(folio); -	if (mapping == NULL || mapping->host == NULL) -		return -EINVAL; - -	dev = mapping->host->i_sb->s_dev; -	if (hwpoison_filter_dev_major != ~0U && -	    hwpoison_filter_dev_major != MAJOR(dev)) -		return -EINVAL; -	if (hwpoison_filter_dev_minor != ~0U && -	    hwpoison_filter_dev_minor != MINOR(dev)) -		return -EINVAL; - -	return 0; +	rcu_assign_pointer(hwpoison_filter_func, filter);  } +EXPORT_SYMBOL_GPL(hwpoison_filter_register); -static int hwpoison_filter_flags(struct page *p) -{ -	if (!hwpoison_filter_flags_mask) -		return 0; - -	if ((stable_page_flags(p) & hwpoison_filter_flags_mask) == -				    hwpoison_filter_flags_value) -		return 0; -	else -		return -EINVAL; -} - -/* - * This allows stress tests to limit test scope to a collection of tasks - * by putting them under some memcg. This prevents killing unrelated/important - * processes such as /sbin/init. Note that the target task may share clean - * pages with init (eg. libc text), which is harmless. If the target task - * share _dirty_ pages with another task B, the test scheme must make sure B - * is also included in the memcg. At last, due to race conditions this filter - * can only guarantee that the page either belongs to the memcg tasks, or is - * a freed page. - */ -#ifdef CONFIG_MEMCG -u64 hwpoison_filter_memcg; -EXPORT_SYMBOL_GPL(hwpoison_filter_memcg); -static int hwpoison_filter_task(struct page *p) +void hwpoison_filter_unregister(void)  { -	if (!hwpoison_filter_memcg) -		return 0; - -	if (page_cgroup_ino(p) != hwpoison_filter_memcg) -		return -EINVAL; - -	return 0; +	RCU_INIT_POINTER(hwpoison_filter_func, NULL); +	synchronize_rcu();  } -#else -static int hwpoison_filter_task(struct page *p) { return 0; } -#endif +EXPORT_SYMBOL_GPL(hwpoison_filter_unregister); -int hwpoison_filter(struct page *p) +static int hwpoison_filter(struct page *p)  { -	if (!hwpoison_filter_enable) -		return 0; - -	if (hwpoison_filter_dev(p)) -		return -EINVAL; - -	if (hwpoison_filter_flags(p)) -		return -EINVAL; +	int ret = 0; +	hwpoison_filter_func_t *filter; -	if (hwpoison_filter_task(p)) -		return -EINVAL; +	rcu_read_lock(); +	filter = rcu_dereference(hwpoison_filter_func); +	if (filter) +		ret = filter(p); +	rcu_read_unlock(); -	return 0; -} -EXPORT_SYMBOL_GPL(hwpoison_filter); -#else -int hwpoison_filter(struct page *p) -{ -	return 0; +	return ret;  } -#endif  /*   * Kill all processes that have a poisoned page mapped and then isolate @@ -1199,7 +1127,7 @@ static int me_swapcache_clean(struct page_state *ps, struct page *p)  	struct folio *folio = page_folio(p);  	int ret; -	delete_from_swap_cache(folio); +	swap_cache_del_folio(folio);  	ret = delete_from_lru_cache(folio) ? MF_FAILED : MF_RECOVERED;  	folio_unlock(folio); @@ -1708,10 +1636,10 @@ static int identify_page_state(unsigned long pfn, struct page *p,  	 * carried out only if the first check can't determine the page status.  	 */  	for (ps = error_states;; ps++) -		if ((p->flags & ps->mask) == ps->res) +		if ((p->flags.f & ps->mask) == ps->res)  			break; -	page_flags |= (p->flags & (1UL << PG_dirty)); +	page_flags |= (p->flags.f & (1UL << PG_dirty));  	if (!ps->mask)  		for (ps = error_states;; ps++) @@ -2137,7 +2065,7 @@ retry:  		return action_result(pfn, MF_MSG_FREE_HUGE, res);  	} -	page_flags = folio->flags; +	page_flags = folio->flags.f;  	if (!hwpoison_user_mappings(folio, p, pfn, flags)) {  		folio_unlock(folio); @@ -2266,7 +2194,7 @@ int memory_failure(unsigned long pfn, int flags)  			goto unlock_mutex;  		if (pfn_valid(pfn)) { -			pgmap = get_dev_pagemap(pfn, NULL); +			pgmap = get_dev_pagemap(pfn);  			put_ref_page(pfn, flags);  			if (pgmap) {  				res = memory_failure_dev_pagemap(pfn, flags, @@ -2397,7 +2325,7 @@ try_again:  	 * folio_remove_rmap_*() in try_to_unmap_one(). So to determine page  	 * status correctly, we save a copy of the page flags at this time.  	 */ -	page_flags = folio->flags; +	page_flags = folio->flags.f;  	/*  	 * __munlock_folio() may clear a writeback folio's LRU flag without @@ -2742,13 +2670,13 @@ static int soft_offline_in_use_page(struct page *page)  				putback_movable_pages(&pagelist);  			pr_info("%#lx: %s migration failed %ld, type %pGp\n", -				pfn, msg_page[huge], ret, &page->flags); +				pfn, msg_page[huge], ret, &page->flags.f);  			if (ret > 0)  				ret = -EBUSY;  		}  	} else {  		pr_info("%#lx: %s isolation failed, page count %d, type %pGp\n", -			pfn, msg_page[huge], page_count(page), &page->flags); +			pfn, msg_page[huge], page_count(page), &page->flags.f);  		ret = -EBUSY;  	}  	return ret; | 
