Blame view
mm/balloon_compaction.c
5.48 KB
18468d93e mm: introduce a c... |
1 2 3 4 5 6 7 8 9 10 11 12 13 |
/* * mm/balloon_compaction.c * * Common interface for making balloon pages movable by compaction. * * Copyright (C) 2012, Red Hat, Inc. Rafael Aquini <aquini@redhat.com> */ #include <linux/mm.h> #include <linux/slab.h> #include <linux/export.h> #include <linux/balloon_compaction.h> /* |
c7cdff0e8 virtio_balloon: f... |
14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
* balloon_page_alloc - allocates a new page for insertion into the balloon * page list. * * Driver must call it to properly allocate a new enlisted balloon page. * Driver must call balloon_page_enqueue before definitively removing it from * the guest system. This function returns the page address for the recently * allocated page or NULL in the case we fail to allocate a new page this turn. */ struct page *balloon_page_alloc(void) { struct page *page = alloc_page(balloon_mapping_gfp_mask() | __GFP_NOMEMALLOC | __GFP_NORETRY); return page; } EXPORT_SYMBOL_GPL(balloon_page_alloc); /* |
18468d93e mm: introduce a c... |
31 32 |
* balloon_page_enqueue - allocates a new page and inserts it into the balloon * page list. |
bdb428c82 lib+mm: fix few s... |
33 |
* @b_dev_info: balloon device descriptor where we will insert a new page to |
c7cdff0e8 virtio_balloon: f... |
34 |
* @page: new page to enqueue - allocated using balloon_page_alloc. |
18468d93e mm: introduce a c... |
35 |
* |
c7cdff0e8 virtio_balloon: f... |
36 |
* Driver must call it to properly enqueue a new allocated balloon page |
bdb428c82 lib+mm: fix few s... |
37 |
* before definitively removing it from the guest system. |
18468d93e mm: introduce a c... |
38 39 40 |
* This function returns the page address for the recently enqueued page or * NULL in the case we fail to allocate a new page this turn. */ |
c7cdff0e8 virtio_balloon: f... |
41 42 |
void balloon_page_enqueue(struct balloon_dev_info *b_dev_info, struct page *page) |
18468d93e mm: introduce a c... |
43 44 |
{ unsigned long flags; |
18468d93e mm: introduce a c... |
45 46 47 48 49 50 51 52 |
/* * Block others from accessing the 'page' when we get around to * establishing additional references. We should be the only one * holding a reference to the 'page' at this point. */ BUG_ON(!trylock_page(page)); spin_lock_irqsave(&b_dev_info->pages_lock, flags); |
9d1ba8056 mm/balloon_compac... |
53 |
balloon_page_insert(b_dev_info, page); |
09316c09d mm/balloon_compac... |
54 |
__count_vm_event(BALLOON_INFLATE); |
18468d93e mm: introduce a c... |
55 56 |
spin_unlock_irqrestore(&b_dev_info->pages_lock, flags); unlock_page(page); |
18468d93e mm: introduce a c... |
57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 |
} EXPORT_SYMBOL_GPL(balloon_page_enqueue); /* * balloon_page_dequeue - removes a page from balloon's page list and returns * the its address to allow the driver release the page. * @b_dev_info: balloon device decriptor where we will grab a page from. * * Driver must call it to properly de-allocate a previous enlisted balloon page * before definetively releasing it back to the guest system. * This function returns the page address for the recently dequeued page or * NULL in the case we find balloon's page list temporarily empty due to * compaction isolated pages. */ struct page *balloon_page_dequeue(struct balloon_dev_info *b_dev_info) { struct page *page, *tmp; unsigned long flags; bool dequeued_page; dequeued_page = false; |
21ea9fb69 virtio_balloon: f... |
78 |
spin_lock_irqsave(&b_dev_info->pages_lock, flags); |
18468d93e mm: introduce a c... |
79 80 81 82 83 84 85 |
list_for_each_entry_safe(page, tmp, &b_dev_info->pages, lru) { /* * Block others from accessing the 'page' while we get around * establishing additional references and preparing the 'page' * to be released by the balloon driver. */ if (trylock_page(page)) { |
4d88e6f7d mm/balloon_compac... |
86 |
#ifdef CONFIG_BALLOON_COMPACTION |
b1123ea6d mm: balloon: use ... |
87 |
if (PageIsolated(page)) { |
d6d86c0a7 mm/balloon_compac... |
88 89 90 91 |
/* raced with isolation */ unlock_page(page); continue; } |
4d88e6f7d mm/balloon_compac... |
92 |
#endif |
18468d93e mm: introduce a c... |
93 |
balloon_page_delete(page); |
09316c09d mm/balloon_compac... |
94 |
__count_vm_event(BALLOON_DEFLATE); |
18468d93e mm: introduce a c... |
95 96 97 98 99 |
unlock_page(page); dequeued_page = true; break; } } |
21ea9fb69 virtio_balloon: f... |
100 |
spin_unlock_irqrestore(&b_dev_info->pages_lock, flags); |
18468d93e mm: introduce a c... |
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 |
if (!dequeued_page) { /* * If we are unable to dequeue a balloon page because the page * list is empty and there is no isolated pages, then something * went out of track and some balloon pages are lost. * BUG() here, otherwise the balloon driver may get stuck into * an infinite loop while attempting to release all its pages. */ spin_lock_irqsave(&b_dev_info->pages_lock, flags); if (unlikely(list_empty(&b_dev_info->pages) && !b_dev_info->isolated_pages)) BUG(); spin_unlock_irqrestore(&b_dev_info->pages_lock, flags); page = NULL; } return page; } EXPORT_SYMBOL_GPL(balloon_page_dequeue); #ifdef CONFIG_BALLOON_COMPACTION |
18468d93e mm: introduce a c... |
122 |
|
b1123ea6d mm: balloon: use ... |
123 |
bool balloon_page_isolate(struct page *page, isolate_mode_t mode) |
18468d93e mm: introduce a c... |
124 |
{ |
9d1ba8056 mm/balloon_compac... |
125 |
struct balloon_dev_info *b_dev_info = balloon_page_device(page); |
18468d93e mm: introduce a c... |
126 |
unsigned long flags; |
d6d86c0a7 mm/balloon_compac... |
127 |
|
18468d93e mm: introduce a c... |
128 129 130 131 |
spin_lock_irqsave(&b_dev_info->pages_lock, flags); list_del(&page->lru); b_dev_info->isolated_pages++; spin_unlock_irqrestore(&b_dev_info->pages_lock, flags); |
b1123ea6d mm: balloon: use ... |
132 133 |
return true; |
18468d93e mm: introduce a c... |
134 |
} |
b1123ea6d mm: balloon: use ... |
135 |
void balloon_page_putback(struct page *page) |
18468d93e mm: introduce a c... |
136 |
{ |
9d1ba8056 mm/balloon_compac... |
137 |
struct balloon_dev_info *b_dev_info = balloon_page_device(page); |
18468d93e mm: introduce a c... |
138 |
unsigned long flags; |
d6d86c0a7 mm/balloon_compac... |
139 |
|
18468d93e mm: introduce a c... |
140 141 142 143 144 |
spin_lock_irqsave(&b_dev_info->pages_lock, flags); list_add(&page->lru, &b_dev_info->pages); b_dev_info->isolated_pages--; spin_unlock_irqrestore(&b_dev_info->pages_lock, flags); } |
18468d93e mm: introduce a c... |
145 146 |
/* move_to_new_page() counterpart for a ballooned page */ |
b1123ea6d mm: balloon: use ... |
147 148 149 |
int balloon_page_migrate(struct address_space *mapping, struct page *newpage, struct page *page, enum migrate_mode mode) |
18468d93e mm: introduce a c... |
150 |
{ |
9d1ba8056 mm/balloon_compac... |
151 |
struct balloon_dev_info *balloon = balloon_page_device(page); |
18468d93e mm: introduce a c... |
152 |
|
2916ecc0f mm/migrate: new m... |
153 154 155 156 157 158 159 |
/* * We can not easily support the no copy case here so ignore it as it * is unlikely to be use with ballon pages. See include/linux/hmm.h for * user of the MIGRATE_SYNC_NO_COPY mode. */ if (mode == MIGRATE_SYNC_NO_COPY) return -EINVAL; |
7db7671f8 mm: page migratio... |
160 161 |
VM_BUG_ON_PAGE(!PageLocked(page), page); VM_BUG_ON_PAGE(!PageLocked(newpage), newpage); |
18468d93e mm: introduce a c... |
162 |
|
b1123ea6d mm: balloon: use ... |
163 164 |
return balloon->migratepage(balloon, newpage, page, mode); } |
18468d93e mm: introduce a c... |
165 |
|
b1123ea6d mm: balloon: use ... |
166 167 168 169 170 171 |
const struct address_space_operations balloon_aops = { .migratepage = balloon_page_migrate, .isolate_page = balloon_page_isolate, .putback_page = balloon_page_putback, }; EXPORT_SYMBOL_GPL(balloon_aops); |
18468d93e mm: introduce a c... |
172 |
|
18468d93e mm: introduce a c... |
173 |
#endif /* CONFIG_BALLOON_COMPACTION */ |