mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-01 10:45:49 +00:00
mm/debug-pagealloc: prepare boottime configurable on/off
Until now, debug-pagealloc needs extra flags in struct page, so we need to recompile whole source code when we decide to use it. This is really painful, because it takes some time to recompile and sometimes rebuild is not possible due to third party module depending on struct page. So, we can't use this good feature in many cases. Now, we have the page extension feature that allows us to insert extra flags to outside of struct page. This gets rid of third party module issue mentioned above. And, this allows us to determine if we need extra memory for this page extension in boottime. With these property, we can avoid using debug-pagealloc in boottime with low computational overhead in the kernel built with CONFIG_DEBUG_PAGEALLOC. This will help our development process greatly. This patch is the preparation step to achive above goal. debug-pagealloc originally uses extra field of struct page, but, after this patch, it will use field of struct page_ext. Because memory for page_ext is allocated later than initialization of page allocator in CONFIG_SPARSEMEM, we should disable debug-pagealloc feature temporarily until initialization of page_ext. This patch implements this. Signed-off-by: Joonsoo Kim <iamjoonsoo.kim@lge.com> Cc: Mel Gorman <mgorman@suse.de> Cc: Johannes Weiner <hannes@cmpxchg.org> Cc: Minchan Kim <minchan@kernel.org> Cc: Dave Hansen <dave@sr71.net> Cc: Michal Nazarewicz <mina86@mina86.com> Cc: Jungsoo Son <jungsoo.son@lge.com> Cc: Ingo Molnar <mingo@redhat.com> Cc: Joonsoo Kim <iamjoonsoo.kim@lge.com> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
This commit is contained in:
parent
eefa864b70
commit
e30825f186
@ -19,6 +19,7 @@
|
|||||||
#include <linux/bit_spinlock.h>
|
#include <linux/bit_spinlock.h>
|
||||||
#include <linux/shrinker.h>
|
#include <linux/shrinker.h>
|
||||||
#include <linux/resource.h>
|
#include <linux/resource.h>
|
||||||
|
#include <linux/page_ext.h>
|
||||||
|
|
||||||
struct mempolicy;
|
struct mempolicy;
|
||||||
struct anon_vma;
|
struct anon_vma;
|
||||||
@ -2155,20 +2156,36 @@ extern void copy_user_huge_page(struct page *dst, struct page *src,
|
|||||||
unsigned int pages_per_huge_page);
|
unsigned int pages_per_huge_page);
|
||||||
#endif /* CONFIG_TRANSPARENT_HUGEPAGE || CONFIG_HUGETLBFS */
|
#endif /* CONFIG_TRANSPARENT_HUGEPAGE || CONFIG_HUGETLBFS */
|
||||||
|
|
||||||
|
extern struct page_ext_operations debug_guardpage_ops;
|
||||||
|
extern struct page_ext_operations page_poisoning_ops;
|
||||||
|
|
||||||
#ifdef CONFIG_DEBUG_PAGEALLOC
|
#ifdef CONFIG_DEBUG_PAGEALLOC
|
||||||
extern unsigned int _debug_guardpage_minorder;
|
extern unsigned int _debug_guardpage_minorder;
|
||||||
|
extern bool _debug_guardpage_enabled;
|
||||||
|
|
||||||
static inline unsigned int debug_guardpage_minorder(void)
|
static inline unsigned int debug_guardpage_minorder(void)
|
||||||
{
|
{
|
||||||
return _debug_guardpage_minorder;
|
return _debug_guardpage_minorder;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline bool debug_guardpage_enabled(void)
|
||||||
|
{
|
||||||
|
return _debug_guardpage_enabled;
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool page_is_guard(struct page *page)
|
static inline bool page_is_guard(struct page *page)
|
||||||
{
|
{
|
||||||
return test_bit(PAGE_DEBUG_FLAG_GUARD, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
if (!debug_guardpage_enabled())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
return test_bit(PAGE_EXT_DEBUG_GUARD, &page_ext->flags);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
static inline unsigned int debug_guardpage_minorder(void) { return 0; }
|
static inline unsigned int debug_guardpage_minorder(void) { return 0; }
|
||||||
|
static inline bool debug_guardpage_enabled(void) { return false; }
|
||||||
static inline bool page_is_guard(struct page *page) { return false; }
|
static inline bool page_is_guard(struct page *page) { return false; }
|
||||||
#endif /* CONFIG_DEBUG_PAGEALLOC */
|
#endif /* CONFIG_DEBUG_PAGEALLOC */
|
||||||
|
|
||||||
|
@ -10,7 +10,6 @@
|
|||||||
#include <linux/rwsem.h>
|
#include <linux/rwsem.h>
|
||||||
#include <linux/completion.h>
|
#include <linux/completion.h>
|
||||||
#include <linux/cpumask.h>
|
#include <linux/cpumask.h>
|
||||||
#include <linux/page-debug-flags.h>
|
|
||||||
#include <linux/uprobes.h>
|
#include <linux/uprobes.h>
|
||||||
#include <linux/page-flags-layout.h>
|
#include <linux/page-flags-layout.h>
|
||||||
#include <asm/page.h>
|
#include <asm/page.h>
|
||||||
@ -186,9 +185,6 @@ struct page {
|
|||||||
void *virtual; /* Kernel virtual address (NULL if
|
void *virtual; /* Kernel virtual address (NULL if
|
||||||
not kmapped, ie. highmem) */
|
not kmapped, ie. highmem) */
|
||||||
#endif /* WANT_PAGE_VIRTUAL */
|
#endif /* WANT_PAGE_VIRTUAL */
|
||||||
#ifdef CONFIG_WANT_PAGE_DEBUG_FLAGS
|
|
||||||
unsigned long debug_flags; /* Use atomic bitops on this */
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef CONFIG_KMEMCHECK
|
#ifdef CONFIG_KMEMCHECK
|
||||||
/*
|
/*
|
||||||
|
@ -1,32 +0,0 @@
|
|||||||
#ifndef LINUX_PAGE_DEBUG_FLAGS_H
|
|
||||||
#define LINUX_PAGE_DEBUG_FLAGS_H
|
|
||||||
|
|
||||||
/*
|
|
||||||
* page->debug_flags bits:
|
|
||||||
*
|
|
||||||
* PAGE_DEBUG_FLAG_POISON is set for poisoned pages. This is used to
|
|
||||||
* implement generic debug pagealloc feature. The pages are filled with
|
|
||||||
* poison patterns and set this flag after free_pages(). The poisoned
|
|
||||||
* pages are verified whether the patterns are not corrupted and clear
|
|
||||||
* the flag before alloc_pages().
|
|
||||||
*/
|
|
||||||
|
|
||||||
enum page_debug_flags {
|
|
||||||
PAGE_DEBUG_FLAG_POISON, /* Page is poisoned */
|
|
||||||
PAGE_DEBUG_FLAG_GUARD,
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Ensure that CONFIG_WANT_PAGE_DEBUG_FLAGS reliably
|
|
||||||
* gets turned off when no debug features are enabling it!
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifdef CONFIG_WANT_PAGE_DEBUG_FLAGS
|
|
||||||
#if !defined(CONFIG_PAGE_POISONING) && \
|
|
||||||
!defined(CONFIG_PAGE_GUARD) \
|
|
||||||
/* && !defined(CONFIG_PAGE_DEBUG_SOMETHING_ELSE) && ... */
|
|
||||||
#error WANT_PAGE_DEBUG_FLAGS is turned on with no debug features!
|
|
||||||
#endif
|
|
||||||
#endif /* CONFIG_WANT_PAGE_DEBUG_FLAGS */
|
|
||||||
|
|
||||||
#endif /* LINUX_PAGE_DEBUG_FLAGS_H */
|
|
@ -9,6 +9,21 @@ struct page_ext_operations {
|
|||||||
|
|
||||||
#ifdef CONFIG_PAGE_EXTENSION
|
#ifdef CONFIG_PAGE_EXTENSION
|
||||||
|
|
||||||
|
/*
|
||||||
|
* page_ext->flags bits:
|
||||||
|
*
|
||||||
|
* PAGE_EXT_DEBUG_POISON is set for poisoned pages. This is used to
|
||||||
|
* implement generic debug pagealloc feature. The pages are filled with
|
||||||
|
* poison patterns and set this flag after free_pages(). The poisoned
|
||||||
|
* pages are verified whether the patterns are not corrupted and clear
|
||||||
|
* the flag before alloc_pages().
|
||||||
|
*/
|
||||||
|
|
||||||
|
enum page_ext_flags {
|
||||||
|
PAGE_EXT_DEBUG_POISON, /* Page is poisoned */
|
||||||
|
PAGE_EXT_DEBUG_GUARD,
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Page Extension can be considered as an extended mem_map.
|
* Page Extension can be considered as an extended mem_map.
|
||||||
* A page_ext page is associated with every page descriptor. The
|
* A page_ext page is associated with every page descriptor. The
|
||||||
|
@ -12,6 +12,7 @@ config DEBUG_PAGEALLOC
|
|||||||
depends on DEBUG_KERNEL
|
depends on DEBUG_KERNEL
|
||||||
depends on !HIBERNATION || ARCH_SUPPORTS_DEBUG_PAGEALLOC && !PPC && !SPARC
|
depends on !HIBERNATION || ARCH_SUPPORTS_DEBUG_PAGEALLOC && !PPC && !SPARC
|
||||||
depends on !KMEMCHECK
|
depends on !KMEMCHECK
|
||||||
|
select PAGE_EXTENSION
|
||||||
select PAGE_POISONING if !ARCH_SUPPORTS_DEBUG_PAGEALLOC
|
select PAGE_POISONING if !ARCH_SUPPORTS_DEBUG_PAGEALLOC
|
||||||
select PAGE_GUARD if ARCH_SUPPORTS_DEBUG_PAGEALLOC
|
select PAGE_GUARD if ARCH_SUPPORTS_DEBUG_PAGEALLOC
|
||||||
---help---
|
---help---
|
||||||
|
@ -2,23 +2,49 @@
|
|||||||
#include <linux/string.h>
|
#include <linux/string.h>
|
||||||
#include <linux/mm.h>
|
#include <linux/mm.h>
|
||||||
#include <linux/highmem.h>
|
#include <linux/highmem.h>
|
||||||
#include <linux/page-debug-flags.h>
|
#include <linux/page_ext.h>
|
||||||
#include <linux/poison.h>
|
#include <linux/poison.h>
|
||||||
#include <linux/ratelimit.h>
|
#include <linux/ratelimit.h>
|
||||||
|
|
||||||
|
static bool page_poisoning_enabled __read_mostly;
|
||||||
|
|
||||||
|
static bool need_page_poisoning(void)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void init_page_poisoning(void)
|
||||||
|
{
|
||||||
|
page_poisoning_enabled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct page_ext_operations page_poisoning_ops = {
|
||||||
|
.need = need_page_poisoning,
|
||||||
|
.init = init_page_poisoning,
|
||||||
|
};
|
||||||
|
|
||||||
static inline void set_page_poison(struct page *page)
|
static inline void set_page_poison(struct page *page)
|
||||||
{
|
{
|
||||||
__set_bit(PAGE_DEBUG_FLAG_POISON, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
__set_bit(PAGE_EXT_DEBUG_POISON, &page_ext->flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void clear_page_poison(struct page *page)
|
static inline void clear_page_poison(struct page *page)
|
||||||
{
|
{
|
||||||
__clear_bit(PAGE_DEBUG_FLAG_POISON, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
__clear_bit(PAGE_EXT_DEBUG_POISON, &page_ext->flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool page_poison(struct page *page)
|
static inline bool page_poison(struct page *page)
|
||||||
{
|
{
|
||||||
return test_bit(PAGE_DEBUG_FLAG_POISON, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
return test_bit(PAGE_EXT_DEBUG_POISON, &page_ext->flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void poison_page(struct page *page)
|
static void poison_page(struct page *page)
|
||||||
@ -95,6 +121,9 @@ static void unpoison_pages(struct page *page, int n)
|
|||||||
|
|
||||||
void kernel_map_pages(struct page *page, int numpages, int enable)
|
void kernel_map_pages(struct page *page, int numpages, int enable)
|
||||||
{
|
{
|
||||||
|
if (!page_poisoning_enabled)
|
||||||
|
return;
|
||||||
|
|
||||||
if (enable)
|
if (enable)
|
||||||
unpoison_pages(page, numpages);
|
unpoison_pages(page, numpages);
|
||||||
else
|
else
|
||||||
|
@ -56,7 +56,7 @@
|
|||||||
#include <linux/prefetch.h>
|
#include <linux/prefetch.h>
|
||||||
#include <linux/mm_inline.h>
|
#include <linux/mm_inline.h>
|
||||||
#include <linux/migrate.h>
|
#include <linux/migrate.h>
|
||||||
#include <linux/page-debug-flags.h>
|
#include <linux/page_ext.h>
|
||||||
#include <linux/hugetlb.h>
|
#include <linux/hugetlb.h>
|
||||||
#include <linux/sched/rt.h>
|
#include <linux/sched/rt.h>
|
||||||
|
|
||||||
@ -425,6 +425,22 @@ static inline void prep_zero_page(struct page *page, unsigned int order,
|
|||||||
|
|
||||||
#ifdef CONFIG_DEBUG_PAGEALLOC
|
#ifdef CONFIG_DEBUG_PAGEALLOC
|
||||||
unsigned int _debug_guardpage_minorder;
|
unsigned int _debug_guardpage_minorder;
|
||||||
|
bool _debug_guardpage_enabled __read_mostly;
|
||||||
|
|
||||||
|
static bool need_debug_guardpage(void)
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void init_debug_guardpage(void)
|
||||||
|
{
|
||||||
|
_debug_guardpage_enabled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct page_ext_operations debug_guardpage_ops = {
|
||||||
|
.need = need_debug_guardpage,
|
||||||
|
.init = init_debug_guardpage,
|
||||||
|
};
|
||||||
|
|
||||||
static int __init debug_guardpage_minorder_setup(char *buf)
|
static int __init debug_guardpage_minorder_setup(char *buf)
|
||||||
{
|
{
|
||||||
@ -443,7 +459,14 @@ __setup("debug_guardpage_minorder=", debug_guardpage_minorder_setup);
|
|||||||
static inline void set_page_guard(struct zone *zone, struct page *page,
|
static inline void set_page_guard(struct zone *zone, struct page *page,
|
||||||
unsigned int order, int migratetype)
|
unsigned int order, int migratetype)
|
||||||
{
|
{
|
||||||
__set_bit(PAGE_DEBUG_FLAG_GUARD, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
if (!debug_guardpage_enabled())
|
||||||
|
return;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
__set_bit(PAGE_EXT_DEBUG_GUARD, &page_ext->flags);
|
||||||
|
|
||||||
INIT_LIST_HEAD(&page->lru);
|
INIT_LIST_HEAD(&page->lru);
|
||||||
set_page_private(page, order);
|
set_page_private(page, order);
|
||||||
/* Guard pages are not available for any usage */
|
/* Guard pages are not available for any usage */
|
||||||
@ -453,12 +476,20 @@ static inline void set_page_guard(struct zone *zone, struct page *page,
|
|||||||
static inline void clear_page_guard(struct zone *zone, struct page *page,
|
static inline void clear_page_guard(struct zone *zone, struct page *page,
|
||||||
unsigned int order, int migratetype)
|
unsigned int order, int migratetype)
|
||||||
{
|
{
|
||||||
__clear_bit(PAGE_DEBUG_FLAG_GUARD, &page->debug_flags);
|
struct page_ext *page_ext;
|
||||||
|
|
||||||
|
if (!debug_guardpage_enabled())
|
||||||
|
return;
|
||||||
|
|
||||||
|
page_ext = lookup_page_ext(page);
|
||||||
|
__clear_bit(PAGE_EXT_DEBUG_GUARD, &page_ext->flags);
|
||||||
|
|
||||||
set_page_private(page, 0);
|
set_page_private(page, 0);
|
||||||
if (!is_migrate_isolate(migratetype))
|
if (!is_migrate_isolate(migratetype))
|
||||||
__mod_zone_freepage_state(zone, (1 << order), migratetype);
|
__mod_zone_freepage_state(zone, (1 << order), migratetype);
|
||||||
}
|
}
|
||||||
#else
|
#else
|
||||||
|
struct page_ext_operations debug_guardpage_ops = { NULL, };
|
||||||
static inline void set_page_guard(struct zone *zone, struct page *page,
|
static inline void set_page_guard(struct zone *zone, struct page *page,
|
||||||
unsigned int order, int migratetype) {}
|
unsigned int order, int migratetype) {}
|
||||||
static inline void clear_page_guard(struct zone *zone, struct page *page,
|
static inline void clear_page_guard(struct zone *zone, struct page *page,
|
||||||
@ -869,6 +900,7 @@ static inline void expand(struct zone *zone, struct page *page,
|
|||||||
VM_BUG_ON_PAGE(bad_range(zone, &page[size]), &page[size]);
|
VM_BUG_ON_PAGE(bad_range(zone, &page[size]), &page[size]);
|
||||||
|
|
||||||
if (IS_ENABLED(CONFIG_DEBUG_PAGEALLOC) &&
|
if (IS_ENABLED(CONFIG_DEBUG_PAGEALLOC) &&
|
||||||
|
debug_guardpage_enabled() &&
|
||||||
high < debug_guardpage_minorder()) {
|
high < debug_guardpage_minorder()) {
|
||||||
/*
|
/*
|
||||||
* Mark as guard pages (or page), that will allow to
|
* Mark as guard pages (or page), that will allow to
|
||||||
|
@ -51,6 +51,10 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
static struct page_ext_operations *page_ext_ops[] = {
|
static struct page_ext_operations *page_ext_ops[] = {
|
||||||
|
&debug_guardpage_ops,
|
||||||
|
#ifdef CONFIG_PAGE_POISONING
|
||||||
|
&page_poisoning_ops,
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
static unsigned long total_usage;
|
static unsigned long total_usage;
|
||||||
|
Loading…
Reference in New Issue
Block a user