Commit 80409c65 authored by Jan Kara's avatar Jan Kara Committed by Linus Torvalds

mm: migrate: make buffer_migrate_page_norefs() actually succeed

Currently, buffer_migrate_page_norefs() was constantly failing because
buffer_migrate_lock_buffers() grabbed reference on each buffer.  In
fact, there's no reason for buffer_migrate_lock_buffers() to grab any
buffer references as the page is locked during all our operation and
thus nobody can reclaim buffers from the page.

So remove grabbing of buffer references which also makes
buffer_migrate_page_norefs() succeed.

Link: http://lkml.kernel.org/r/20190116131217.7226-1-jack@suse.cz
Fixes: 89cb0888 "mm: migrate: provide buffer_migrate_page_norefs()"
Signed-off-by: default avatarJan Kara <jack@suse.cz>
Cc: Sergey Senozhatsky <sergey.senozhatsky.work@gmail.com>
Cc: Pavel Machek <pavel@ucw.cz>
Cc: Mel Gorman <mgorman@techsingularity.net>
Cc: Vlastimil Babka <vbabka@suse.cz>
Cc: Andrea Arcangeli <aarcange@redhat.com>
Cc: David Rientjes <rientjes@google.com>
Cc: Michal Hocko <mhocko@kernel.org>
Cc: Zi Yan <zi.yan@cs.rutgers.edu>
Cc: Johannes Weiner <hannes@cmpxchg.org>
Cc: <stable@vger.kernel.org>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 8fb335e0
...@@ -709,7 +709,6 @@ static bool buffer_migrate_lock_buffers(struct buffer_head *head, ...@@ -709,7 +709,6 @@ static bool buffer_migrate_lock_buffers(struct buffer_head *head,
/* Simple case, sync compaction */ /* Simple case, sync compaction */
if (mode != MIGRATE_ASYNC) { if (mode != MIGRATE_ASYNC) {
do { do {
get_bh(bh);
lock_buffer(bh); lock_buffer(bh);
bh = bh->b_this_page; bh = bh->b_this_page;
...@@ -720,18 +719,15 @@ static bool buffer_migrate_lock_buffers(struct buffer_head *head, ...@@ -720,18 +719,15 @@ static bool buffer_migrate_lock_buffers(struct buffer_head *head,
/* async case, we cannot block on lock_buffer so use trylock_buffer */ /* async case, we cannot block on lock_buffer so use trylock_buffer */
do { do {
get_bh(bh);
if (!trylock_buffer(bh)) { if (!trylock_buffer(bh)) {
/* /*
* We failed to lock the buffer and cannot stall in * We failed to lock the buffer and cannot stall in
* async migration. Release the taken locks * async migration. Release the taken locks
*/ */
struct buffer_head *failed_bh = bh; struct buffer_head *failed_bh = bh;
put_bh(failed_bh);
bh = head; bh = head;
while (bh != failed_bh) { while (bh != failed_bh) {
unlock_buffer(bh); unlock_buffer(bh);
put_bh(bh);
bh = bh->b_this_page; bh = bh->b_this_page;
} }
return false; return false;
...@@ -818,7 +814,6 @@ static int __buffer_migrate_page(struct address_space *mapping, ...@@ -818,7 +814,6 @@ static int __buffer_migrate_page(struct address_space *mapping,
bh = head; bh = head;
do { do {
unlock_buffer(bh); unlock_buffer(bh);
put_bh(bh);
bh = bh->b_this_page; bh = bh->b_this_page;
} while (bh != head); } while (bh != head);
......
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