diff options
author | KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com> | 2007-07-26 13:41:08 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@woody.linux-foundation.org> | 2007-07-26 14:35:17 -0400 |
commit | 3dd9fe8c397df68086e6a1b2160573abbe944813 (patch) | |
tree | 86dda41ade146e2a403151c7ba2f7e32a930bc63 /mm/migrate.c | |
parent | dc386d4d1e98bb39fb967ee156cd456c802fc692 (diff) |
memory unplug: isolate_lru_page fix
release_pages() in mm/swap.c changes page_count() to be 0 without removing
PageLRU flag...
This means isolate_lru_page() can see a page, PageLRU() &&
page_count(page)==0.. This is BUG. (get_page() will be called against
count=0 page.)
Signed-off-by: KAMEZAWA Hiroyuki <kamezawa.hiroyu@jp.fujitsu.com>
Acked-by: Christoph Lameter <clameter@sgi.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm/migrate.c')
-rw-r--r-- | mm/migrate.c | 3 |
1 files changed, 1 insertions, 2 deletions
diff --git a/mm/migrate.c b/mm/migrate.c index c8d87221f368..37c73b902008 100644 --- a/mm/migrate.c +++ b/mm/migrate.c | |||
@@ -49,9 +49,8 @@ int isolate_lru_page(struct page *page, struct list_head *pagelist) | |||
49 | struct zone *zone = page_zone(page); | 49 | struct zone *zone = page_zone(page); |
50 | 50 | ||
51 | spin_lock_irq(&zone->lru_lock); | 51 | spin_lock_irq(&zone->lru_lock); |
52 | if (PageLRU(page)) { | 52 | if (PageLRU(page) && get_page_unless_zero(page)) { |
53 | ret = 0; | 53 | ret = 0; |
54 | get_page(page); | ||
55 | ClearPageLRU(page); | 54 | ClearPageLRU(page); |
56 | if (PageActive(page)) | 55 | if (PageActive(page)) |
57 | del_page_from_active_list(zone, page); | 56 | del_page_from_active_list(zone, page); |