diff options
author | Konstantin Khlebnikov <khlebnikov@openvz.org> | 2012-01-10 18:07:04 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-01-10 19:30:41 -0500 |
commit | cc59850ef940e4ee6a765d28b439b9bafe07cf63 (patch) | |
tree | 03b666986e9cc7dfc113a14721c44aa9e149f871 /mm | |
parent | c909e99364c8b6ca07864d752950b6b4ecf6bef4 (diff) |
mm: add free_hot_cold_page_list() helper
This patch adds helper free_hot_cold_page_list() to free list of 0-order
pages. It frees pages directly from list without temporary page-vector.
It also calls trace_mm_pagevec_free() to simulate pagevec_free()
behaviour.
bloat-o-meter:
add/remove: 1/1 grow/shrink: 1/3 up/down: 267/-295 (-28)
function old new delta
free_hot_cold_page_list - 264 +264
get_page_from_freelist 2129 2132 +3
__pagevec_free 243 239 -4
split_free_page 380 373 -7
release_pages 606 510 -96
free_page_list 188 - -188
Signed-off-by: Konstantin Khlebnikov <khlebnikov@openvz.org>
Cc: Mel Gorman <mel@csn.ul.ie>
Cc: KOSAKI Motohiro <kosaki.motohiro@jp.fujitsu.com>
Acked-by: Minchan Kim <minchan.kim@gmail.com>
Acked-by: Hugh Dickins <hughd@google.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
Diffstat (limited to 'mm')
-rw-r--r-- | mm/page_alloc.c | 13 | ||||
-rw-r--r-- | mm/swap.c | 14 | ||||
-rw-r--r-- | mm/vmscan.c | 20 |
3 files changed, 17 insertions, 30 deletions
diff --git a/mm/page_alloc.c b/mm/page_alloc.c index 7990ca154d1b..cd0c95c6cc9e 100644 --- a/mm/page_alloc.c +++ b/mm/page_alloc.c | |||
@@ -1189,6 +1189,19 @@ out: | |||
1189 | } | 1189 | } |
1190 | 1190 | ||
1191 | /* | 1191 | /* |
1192 | * Free a list of 0-order pages | ||
1193 | */ | ||
1194 | void free_hot_cold_page_list(struct list_head *list, int cold) | ||
1195 | { | ||
1196 | struct page *page, *next; | ||
1197 | |||
1198 | list_for_each_entry_safe(page, next, list, lru) { | ||
1199 | trace_mm_pagevec_free(page, cold); | ||
1200 | free_hot_cold_page(page, cold); | ||
1201 | } | ||
1202 | } | ||
1203 | |||
1204 | /* | ||
1192 | * split_page takes a non-compound higher-order page, and splits it into | 1205 | * split_page takes a non-compound higher-order page, and splits it into |
1193 | * n (1<<order) sub-pages: page[0..n] | 1206 | * n (1<<order) sub-pages: page[0..n] |
1194 | * Each sub-page must be freed individually. | 1207 | * Each sub-page must be freed individually. |
@@ -585,11 +585,10 @@ int lru_add_drain_all(void) | |||
585 | void release_pages(struct page **pages, int nr, int cold) | 585 | void release_pages(struct page **pages, int nr, int cold) |
586 | { | 586 | { |
587 | int i; | 587 | int i; |
588 | struct pagevec pages_to_free; | 588 | LIST_HEAD(pages_to_free); |
589 | struct zone *zone = NULL; | 589 | struct zone *zone = NULL; |
590 | unsigned long uninitialized_var(flags); | 590 | unsigned long uninitialized_var(flags); |
591 | 591 | ||
592 | pagevec_init(&pages_to_free, cold); | ||
593 | for (i = 0; i < nr; i++) { | 592 | for (i = 0; i < nr; i++) { |
594 | struct page *page = pages[i]; | 593 | struct page *page = pages[i]; |
595 | 594 | ||
@@ -620,19 +619,12 @@ void release_pages(struct page **pages, int nr, int cold) | |||
620 | del_page_from_lru(zone, page); | 619 | del_page_from_lru(zone, page); |
621 | } | 620 | } |
622 | 621 | ||
623 | if (!pagevec_add(&pages_to_free, page)) { | 622 | list_add(&page->lru, &pages_to_free); |
624 | if (zone) { | ||
625 | spin_unlock_irqrestore(&zone->lru_lock, flags); | ||
626 | zone = NULL; | ||
627 | } | ||
628 | __pagevec_free(&pages_to_free); | ||
629 | pagevec_reinit(&pages_to_free); | ||
630 | } | ||
631 | } | 623 | } |
632 | if (zone) | 624 | if (zone) |
633 | spin_unlock_irqrestore(&zone->lru_lock, flags); | 625 | spin_unlock_irqrestore(&zone->lru_lock, flags); |
634 | 626 | ||
635 | pagevec_free(&pages_to_free); | 627 | free_hot_cold_page_list(&pages_to_free, cold); |
636 | } | 628 | } |
637 | EXPORT_SYMBOL(release_pages); | 629 | EXPORT_SYMBOL(release_pages); |
638 | 630 | ||
diff --git a/mm/vmscan.c b/mm/vmscan.c index 753a2dc300b9..3d571df41c79 100644 --- a/mm/vmscan.c +++ b/mm/vmscan.c | |||
@@ -734,24 +734,6 @@ static enum page_references page_check_references(struct page *page, | |||
734 | return PAGEREF_RECLAIM; | 734 | return PAGEREF_RECLAIM; |
735 | } | 735 | } |
736 | 736 | ||
737 | static noinline_for_stack void free_page_list(struct list_head *free_pages) | ||
738 | { | ||
739 | struct pagevec freed_pvec; | ||
740 | struct page *page, *tmp; | ||
741 | |||
742 | pagevec_init(&freed_pvec, 1); | ||
743 | |||
744 | list_for_each_entry_safe(page, tmp, free_pages, lru) { | ||
745 | list_del(&page->lru); | ||
746 | if (!pagevec_add(&freed_pvec, page)) { | ||
747 | __pagevec_free(&freed_pvec); | ||
748 | pagevec_reinit(&freed_pvec); | ||
749 | } | ||
750 | } | ||
751 | |||
752 | pagevec_free(&freed_pvec); | ||
753 | } | ||
754 | |||
755 | /* | 737 | /* |
756 | * shrink_page_list() returns the number of reclaimed pages | 738 | * shrink_page_list() returns the number of reclaimed pages |
757 | */ | 739 | */ |
@@ -1015,7 +997,7 @@ keep_lumpy: | |||
1015 | if (nr_dirty && nr_dirty == nr_congested && scanning_global_lru(sc)) | 997 | if (nr_dirty && nr_dirty == nr_congested && scanning_global_lru(sc)) |
1016 | zone_set_flag(zone, ZONE_CONGESTED); | 998 | zone_set_flag(zone, ZONE_CONGESTED); |
1017 | 999 | ||
1018 | free_page_list(&free_pages); | 1000 | free_hot_cold_page_list(&free_pages, 1); |
1019 | 1001 | ||
1020 | list_splice(&ret_pages, page_list); | 1002 | list_splice(&ret_pages, page_list); |
1021 | count_vm_events(PGACTIVATE, pgactivate); | 1003 | count_vm_events(PGACTIVATE, pgactivate); |