blob: e1fe7cf5bddf0935a0c9eaab5273b230c4917d1b [file] [log] [blame]
Joonsoo Kimeefa864b2014-12-12 16:55:46 -08001#ifndef __LINUX_PAGE_EXT_H
2#define __LINUX_PAGE_EXT_H
3
Joonsoo Kim48c96a32014-12-12 16:56:01 -08004#include <linux/types.h>
5#include <linux/stacktrace.h>
6
Joonsoo Kimeefa864b2014-12-12 16:55:46 -08007struct pglist_data;
8struct page_ext_operations {
9 bool (*need)(void);
10 void (*init)(void);
11};
12
13#ifdef CONFIG_PAGE_EXTENSION
14
15/*
Joonsoo Kime30825f2014-12-12 16:55:49 -080016 * page_ext->flags bits:
17 *
18 * PAGE_EXT_DEBUG_POISON is set for poisoned pages. This is used to
19 * implement generic debug pagealloc feature. The pages are filled with
20 * poison patterns and set this flag after free_pages(). The poisoned
21 * pages are verified whether the patterns are not corrupted and clear
22 * the flag before alloc_pages().
23 */
24
25enum page_ext_flags {
26 PAGE_EXT_DEBUG_POISON, /* Page is poisoned */
27 PAGE_EXT_DEBUG_GUARD,
Joonsoo Kim48c96a32014-12-12 16:56:01 -080028 PAGE_EXT_OWNER,
Vladimir Davydov33c3fc72015-09-09 15:35:45 -070029#if defined(CONFIG_IDLE_PAGE_TRACKING) && !defined(CONFIG_64BIT)
30 PAGE_EXT_YOUNG,
31 PAGE_EXT_IDLE,
32#endif
Joonsoo Kime30825f2014-12-12 16:55:49 -080033};
34
35/*
Joonsoo Kimeefa864b2014-12-12 16:55:46 -080036 * Page Extension can be considered as an extended mem_map.
37 * A page_ext page is associated with every page descriptor. The
38 * page_ext helps us add more information about the page.
39 * All page_ext are allocated at boot or memory hotplug event,
40 * then the page_ext for pfn always exists.
41 */
42struct page_ext {
43 unsigned long flags;
Joonsoo Kim48c96a32014-12-12 16:56:01 -080044#ifdef CONFIG_PAGE_OWNER
45 unsigned int order;
46 gfp_t gfp_mask;
Sergei Rogachev94f759d62015-02-11 15:28:34 -080047 unsigned int nr_entries;
Vlastimil Babka7cd12b42016-03-15 14:56:18 -070048 int last_migrate_reason;
Joonsoo Kim48c96a32014-12-12 16:56:01 -080049 unsigned long trace_entries[8];
50#endif
Joonsoo Kimeefa864b2014-12-12 16:55:46 -080051};
52
53extern void pgdat_page_ext_init(struct pglist_data *pgdat);
54
55#ifdef CONFIG_SPARSEMEM
56static inline void page_ext_init_flatmem(void)
57{
58}
59extern void page_ext_init(void);
60#else
61extern void page_ext_init_flatmem(void);
62static inline void page_ext_init(void)
63{
64}
65#endif
66
67struct page_ext *lookup_page_ext(struct page *page);
68
69#else /* !CONFIG_PAGE_EXTENSION */
70struct page_ext;
71
72static inline void pgdat_page_ext_init(struct pglist_data *pgdat)
73{
74}
75
76static inline struct page_ext *lookup_page_ext(struct page *page)
77{
78 return NULL;
79}
80
81static inline void page_ext_init(void)
82{
83}
84
85static inline void page_ext_init_flatmem(void)
86{
87}
88#endif /* CONFIG_PAGE_EXTENSION */
89#endif /* __LINUX_PAGE_EXT_H */