Commit a527e8d4 authored by David Rientjes's avatar David Rientjes Committed by Greg Kroah-Hartman

mm, migration: add destination page freeing callback

commit 68711a74 upstream.

Memory migration uses a callback defined by the caller to determine how to
allocate destination pages.  When migration fails for a source page,
however, it frees the destination page back to the system.

This patch adds a memory migration callback defined by the caller to
determine how to free destination pages.  If a caller, such as memory
compaction, builds its own freelist for migration targets, this can reuse
already freed memory instead of scanning additional memory.

If the caller provides a function to handle freeing of destination pages,
it is called when page migration fails.  If the caller passes NULL then
freeing back to the system will be handled as usual.  This patch
introduces no functional change.
Signed-off-by: default avatarDavid Rientjes <rientjes@google.com>
Reviewed-by: default avatarNaoya Horiguchi <n-horiguchi@ah.jp.nec.com>
Acked-by: default avatarMel Gorman <mgorman@suse.de>
Acked-by: default avatarVlastimil Babka <vbabka@suse.cz>
Cc: Greg Thelen <gthelen@google.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: default avatarMel Gorman <mgorman@suse.de>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 5721949c
...@@ -5,7 +5,9 @@ ...@@ -5,7 +5,9 @@
#include <linux/mempolicy.h> #include <linux/mempolicy.h>
#include <linux/migrate_mode.h> #include <linux/migrate_mode.h>
typedef struct page *new_page_t(struct page *, unsigned long private, int **); typedef struct page *new_page_t(struct page *page, unsigned long private,
int **reason);
typedef void free_page_t(struct page *page, unsigned long private);
/* /*
* Return values from addresss_space_operations.migratepage(): * Return values from addresss_space_operations.migratepage():
...@@ -38,7 +40,7 @@ enum migrate_reason { ...@@ -38,7 +40,7 @@ enum migrate_reason {
extern void putback_movable_pages(struct list_head *l); extern void putback_movable_pages(struct list_head *l);
extern int migrate_page(struct address_space *, extern int migrate_page(struct address_space *,
struct page *, struct page *, enum migrate_mode); struct page *, struct page *, enum migrate_mode);
extern int migrate_pages(struct list_head *l, new_page_t x, extern int migrate_pages(struct list_head *l, new_page_t new, free_page_t free,
unsigned long private, enum migrate_mode mode, int reason); unsigned long private, enum migrate_mode mode, int reason);
extern int migrate_prep(void); extern int migrate_prep(void);
...@@ -56,8 +58,9 @@ extern int migrate_page_move_mapping(struct address_space *mapping, ...@@ -56,8 +58,9 @@ extern int migrate_page_move_mapping(struct address_space *mapping,
#else #else
static inline void putback_movable_pages(struct list_head *l) {} static inline void putback_movable_pages(struct list_head *l) {}
static inline int migrate_pages(struct list_head *l, new_page_t x, static inline int migrate_pages(struct list_head *l, new_page_t new,
unsigned long private, enum migrate_mode mode, int reason) free_page_t free, unsigned long private, enum migrate_mode mode,
int reason)
{ return -ENOSYS; } { return -ENOSYS; }
static inline int migrate_prep(void) { return -ENOSYS; } static inline int migrate_prep(void) { return -ENOSYS; }
......
...@@ -1016,7 +1016,7 @@ static int compact_zone(struct zone *zone, struct compact_control *cc) ...@@ -1016,7 +1016,7 @@ static int compact_zone(struct zone *zone, struct compact_control *cc)
} }
nr_migrate = cc->nr_migratepages; nr_migrate = cc->nr_migratepages;
err = migrate_pages(&cc->migratepages, compaction_alloc, err = migrate_pages(&cc->migratepages, compaction_alloc, NULL,
(unsigned long)cc, (unsigned long)cc,
cc->sync ? MIGRATE_SYNC_LIGHT : MIGRATE_ASYNC, cc->sync ? MIGRATE_SYNC_LIGHT : MIGRATE_ASYNC,
MR_COMPACTION); MR_COMPACTION);
......
...@@ -1540,7 +1540,7 @@ static int soft_offline_huge_page(struct page *page, int flags) ...@@ -1540,7 +1540,7 @@ static int soft_offline_huge_page(struct page *page, int flags)
/* Keep page count to indicate a given hugepage is isolated. */ /* Keep page count to indicate a given hugepage is isolated. */
list_move(&hpage->lru, &pagelist); list_move(&hpage->lru, &pagelist);
ret = migrate_pages(&pagelist, new_page, MPOL_MF_MOVE_ALL, ret = migrate_pages(&pagelist, new_page, NULL, MPOL_MF_MOVE_ALL,
MIGRATE_SYNC, MR_MEMORY_FAILURE); MIGRATE_SYNC, MR_MEMORY_FAILURE);
if (ret) { if (ret) {
pr_info("soft offline: %#lx: migration failed %d, type %lx\n", pr_info("soft offline: %#lx: migration failed %d, type %lx\n",
...@@ -1621,7 +1621,7 @@ static int __soft_offline_page(struct page *page, int flags) ...@@ -1621,7 +1621,7 @@ static int __soft_offline_page(struct page *page, int flags)
inc_zone_page_state(page, NR_ISOLATED_ANON + inc_zone_page_state(page, NR_ISOLATED_ANON +
page_is_file_cache(page)); page_is_file_cache(page));
list_add(&page->lru, &pagelist); list_add(&page->lru, &pagelist);
ret = migrate_pages(&pagelist, new_page, MPOL_MF_MOVE_ALL, ret = migrate_pages(&pagelist, new_page, NULL, MPOL_MF_MOVE_ALL,
MIGRATE_SYNC, MR_MEMORY_FAILURE); MIGRATE_SYNC, MR_MEMORY_FAILURE);
if (ret) { if (ret) {
if (!list_empty(&pagelist)) { if (!list_empty(&pagelist)) {
......
...@@ -1332,7 +1332,7 @@ do_migrate_range(unsigned long start_pfn, unsigned long end_pfn) ...@@ -1332,7 +1332,7 @@ do_migrate_range(unsigned long start_pfn, unsigned long end_pfn)
* alloc_migrate_target should be improooooved!! * alloc_migrate_target should be improooooved!!
* migrate_pages returns # of failed pages. * migrate_pages returns # of failed pages.
*/ */
ret = migrate_pages(&source, alloc_migrate_target, 0, ret = migrate_pages(&source, alloc_migrate_target, NULL, 0,
MIGRATE_SYNC, MR_MEMORY_HOTPLUG); MIGRATE_SYNC, MR_MEMORY_HOTPLUG);
if (ret) if (ret)
putback_movable_pages(&source); putback_movable_pages(&source);
......
...@@ -1060,7 +1060,7 @@ static int migrate_to_node(struct mm_struct *mm, int source, int dest, ...@@ -1060,7 +1060,7 @@ static int migrate_to_node(struct mm_struct *mm, int source, int dest,
flags | MPOL_MF_DISCONTIG_OK, &pagelist); flags | MPOL_MF_DISCONTIG_OK, &pagelist);
if (!list_empty(&pagelist)) { if (!list_empty(&pagelist)) {
err = migrate_pages(&pagelist, new_node_page, dest, err = migrate_pages(&pagelist, new_node_page, NULL, dest,
MIGRATE_SYNC, MR_SYSCALL); MIGRATE_SYNC, MR_SYSCALL);
if (err) if (err)
putback_movable_pages(&pagelist); putback_movable_pages(&pagelist);
...@@ -1306,7 +1306,7 @@ static long do_mbind(unsigned long start, unsigned long len, ...@@ -1306,7 +1306,7 @@ static long do_mbind(unsigned long start, unsigned long len,
if (!list_empty(&pagelist)) { if (!list_empty(&pagelist)) {
WARN_ON_ONCE(flags & MPOL_MF_LAZY); WARN_ON_ONCE(flags & MPOL_MF_LAZY);
nr_failed = migrate_pages(&pagelist, new_page, nr_failed = migrate_pages(&pagelist, new_page, NULL,
start, MIGRATE_SYNC, MR_MEMPOLICY_MBIND); start, MIGRATE_SYNC, MR_MEMPOLICY_MBIND);
if (nr_failed) if (nr_failed)
putback_movable_pages(&pagelist); putback_movable_pages(&pagelist);
......
...@@ -941,8 +941,9 @@ static int __unmap_and_move(struct page *page, struct page *newpage, ...@@ -941,8 +941,9 @@ static int __unmap_and_move(struct page *page, struct page *newpage,
* Obtain the lock on page, remove all ptes and migrate the page * Obtain the lock on page, remove all ptes and migrate the page
* to the newly allocated page in newpage. * to the newly allocated page in newpage.
*/ */
static int unmap_and_move(new_page_t get_new_page, unsigned long private, static int unmap_and_move(new_page_t get_new_page, free_page_t put_new_page,
struct page *page, int force, enum migrate_mode mode) unsigned long private, struct page *page, int force,
enum migrate_mode mode)
{ {
int rc = 0; int rc = 0;
int *result = NULL; int *result = NULL;
...@@ -986,11 +987,17 @@ static int unmap_and_move(new_page_t get_new_page, unsigned long private, ...@@ -986,11 +987,17 @@ static int unmap_and_move(new_page_t get_new_page, unsigned long private,
page_is_file_cache(page)); page_is_file_cache(page));
putback_lru_page(page); putback_lru_page(page);
} }
/* /*
* Move the new page to the LRU. If migration was not successful * If migration was not successful and there's a freeing callback, use
* then this will free the page. * it. Otherwise, putback_lru_page() will drop the reference grabbed
* during isolation.
*/ */
putback_lru_page(newpage); if (rc != MIGRATEPAGE_SUCCESS && put_new_page)
put_new_page(newpage, private);
else
putback_lru_page(newpage);
if (result) { if (result) {
if (rc) if (rc)
*result = rc; *result = rc;
...@@ -1019,8 +1026,9 @@ static int unmap_and_move(new_page_t get_new_page, unsigned long private, ...@@ -1019,8 +1026,9 @@ static int unmap_and_move(new_page_t get_new_page, unsigned long private,
* will wait in the page fault for migration to complete. * will wait in the page fault for migration to complete.
*/ */
static int unmap_and_move_huge_page(new_page_t get_new_page, static int unmap_and_move_huge_page(new_page_t get_new_page,
unsigned long private, struct page *hpage, free_page_t put_new_page, unsigned long private,
int force, enum migrate_mode mode) struct page *hpage, int force,
enum migrate_mode mode)
{ {
int rc = 0; int rc = 0;
int *result = NULL; int *result = NULL;
...@@ -1059,20 +1067,30 @@ static int unmap_and_move_huge_page(new_page_t get_new_page, ...@@ -1059,20 +1067,30 @@ static int unmap_and_move_huge_page(new_page_t get_new_page,
if (!page_mapped(hpage)) if (!page_mapped(hpage))
rc = move_to_new_page(new_hpage, hpage, 1, mode); rc = move_to_new_page(new_hpage, hpage, 1, mode);
if (rc) if (rc != MIGRATEPAGE_SUCCESS)
remove_migration_ptes(hpage, hpage); remove_migration_ptes(hpage, hpage);
if (anon_vma) if (anon_vma)
put_anon_vma(anon_vma); put_anon_vma(anon_vma);
if (!rc) if (rc == MIGRATEPAGE_SUCCESS)
hugetlb_cgroup_migrate(hpage, new_hpage); hugetlb_cgroup_migrate(hpage, new_hpage);
unlock_page(hpage); unlock_page(hpage);
out: out:
if (rc != -EAGAIN) if (rc != -EAGAIN)
putback_active_hugepage(hpage); putback_active_hugepage(hpage);
put_page(new_hpage);
/*
* If migration was not successful and there's a freeing callback, use
* it. Otherwise, put_page() will drop the reference grabbed during
* isolation.
*/
if (rc != MIGRATEPAGE_SUCCESS && put_new_page)
put_new_page(new_hpage, private);
else
put_page(new_hpage);
if (result) { if (result) {
if (rc) if (rc)
*result = rc; *result = rc;
...@@ -1089,6 +1107,8 @@ static int unmap_and_move_huge_page(new_page_t get_new_page, ...@@ -1089,6 +1107,8 @@ static int unmap_and_move_huge_page(new_page_t get_new_page,
* @from: The list of pages to be migrated. * @from: The list of pages to be migrated.
* @get_new_page: The function used to allocate free pages to be used * @get_new_page: The function used to allocate free pages to be used
* as the target of the page migration. * as the target of the page migration.
* @put_new_page: The function used to free target pages if migration
* fails, or NULL if no special handling is necessary.
* @private: Private data to be passed on to get_new_page() * @private: Private data to be passed on to get_new_page()
* @mode: The migration mode that specifies the constraints for * @mode: The migration mode that specifies the constraints for
* page migration, if any. * page migration, if any.
...@@ -1102,7 +1122,8 @@ static int unmap_and_move_huge_page(new_page_t get_new_page, ...@@ -1102,7 +1122,8 @@ static int unmap_and_move_huge_page(new_page_t get_new_page,
* Returns the number of pages that were not migrated, or an error code. * Returns the number of pages that were not migrated, or an error code.
*/ */
int migrate_pages(struct list_head *from, new_page_t get_new_page, int migrate_pages(struct list_head *from, new_page_t get_new_page,
unsigned long private, enum migrate_mode mode, int reason) free_page_t put_new_page, unsigned long private,
enum migrate_mode mode, int reason)
{ {
int retry = 1; int retry = 1;
int nr_failed = 0; int nr_failed = 0;
...@@ -1124,10 +1145,11 @@ int migrate_pages(struct list_head *from, new_page_t get_new_page, ...@@ -1124,10 +1145,11 @@ int migrate_pages(struct list_head *from, new_page_t get_new_page,
if (PageHuge(page)) if (PageHuge(page))
rc = unmap_and_move_huge_page(get_new_page, rc = unmap_and_move_huge_page(get_new_page,
private, page, pass > 2, mode); put_new_page, private, page,
pass > 2, mode);
else else
rc = unmap_and_move(get_new_page, private, rc = unmap_and_move(get_new_page, put_new_page,
page, pass > 2, mode); private, page, pass > 2, mode);
switch(rc) { switch(rc) {
case -ENOMEM: case -ENOMEM:
...@@ -1276,7 +1298,7 @@ static int do_move_page_to_node_array(struct mm_struct *mm, ...@@ -1276,7 +1298,7 @@ static int do_move_page_to_node_array(struct mm_struct *mm,
err = 0; err = 0;
if (!list_empty(&pagelist)) { if (!list_empty(&pagelist)) {
err = migrate_pages(&pagelist, new_page_node, err = migrate_pages(&pagelist, new_page_node, NULL,
(unsigned long)pm, MIGRATE_SYNC, MR_SYSCALL); (unsigned long)pm, MIGRATE_SYNC, MR_SYSCALL);
if (err) if (err)
putback_movable_pages(&pagelist); putback_movable_pages(&pagelist);
...@@ -1732,7 +1754,8 @@ int migrate_misplaced_page(struct page *page, struct vm_area_struct *vma, ...@@ -1732,7 +1754,8 @@ int migrate_misplaced_page(struct page *page, struct vm_area_struct *vma,
list_add(&page->lru, &migratepages); list_add(&page->lru, &migratepages);
nr_remaining = migrate_pages(&migratepages, alloc_misplaced_dst_page, nr_remaining = migrate_pages(&migratepages, alloc_misplaced_dst_page,
node, MIGRATE_ASYNC, MR_NUMA_MISPLACED); NULL, node, MIGRATE_ASYNC,
MR_NUMA_MISPLACED);
if (nr_remaining) { if (nr_remaining) {
if (!list_empty(&migratepages)) { if (!list_empty(&migratepages)) {
list_del(&page->lru); list_del(&page->lru);
......
...@@ -6261,7 +6261,7 @@ static int __alloc_contig_migrate_range(struct compact_control *cc, ...@@ -6261,7 +6261,7 @@ static int __alloc_contig_migrate_range(struct compact_control *cc,
cc->nr_migratepages -= nr_reclaimed; cc->nr_migratepages -= nr_reclaimed;
ret = migrate_pages(&cc->migratepages, alloc_migrate_target, ret = migrate_pages(&cc->migratepages, alloc_migrate_target,
0, MIGRATE_SYNC, MR_CMA); NULL, 0, MIGRATE_SYNC, MR_CMA);
} }
if (ret < 0) { if (ret < 0) {
putback_movable_pages(&cc->migratepages); putback_movable_pages(&cc->migratepages);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment