diff options
author | Nick Piggin <npiggin@suse.de> | 2006-03-22 03:08:05 -0500 |
---|---|---|
committer | Linus Torvalds <torvalds@g5.osdl.org> | 2006-03-22 10:53:57 -0500 |
commit | 8dfcc9ba27e2ed257e5de9539f7f03e57c2c0e33 (patch) | |
tree | aecaeb6a0b33c23f79dfcd2418e4a3881a29f2e2 /mm | |
parent | 8e7a9aae91101916b86de07fafe3272ea8dc1f10 (diff) |
[PATCH] mm: split highorder pages
Have an explicit mm call to split higher order pages into individual pages.
Should help to avoid bugs and be more explicit about the code's intention.
Signed-off-by: Nick Piggin <npiggin@suse.de>
Cc: Russell King <rmk@arm.linux.org.uk>
Cc: David Howells <dhowells@redhat.com>
Cc: Ralf Baechle <ralf@linux-mips.org>
Cc: Benjamin Herrenschmidt <benh@kernel.crashing.org>
Cc: Paul Mundt <lethal@linux-sh.org>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Chris Zankel <chris@zankel.net>
Signed-off-by: Yoichi Yuasa <yoichi_yuasa@tripeaks.co.jp>
Signed-off-by: Andrew Morton <akpm@osdl.org>
Signed-off-by: Linus Torvalds <torvalds@osdl.org>
Diffstat (limited to 'mm')
-rw-r--r-- | mm/memory.c | 4 | ||||
-rw-r--r-- | mm/page_alloc.c | 22 |
2 files changed, 23 insertions, 3 deletions
diff --git a/mm/memory.c b/mm/memory.c index 85e80a57db29..6af555c1c42a 100644 --- a/mm/memory.c +++ b/mm/memory.c | |||
@@ -1221,9 +1221,7 @@ out: | |||
1221 | * The page has to be a nice clean _individual_ kernel allocation. | 1221 | * The page has to be a nice clean _individual_ kernel allocation. |
1222 | * If you allocate a compound page, you need to have marked it as | 1222 | * If you allocate a compound page, you need to have marked it as |
1223 | * such (__GFP_COMP), or manually just split the page up yourself | 1223 | * such (__GFP_COMP), or manually just split the page up yourself |
1224 | * (which is mainly an issue of doing "set_page_count(page, 1)" for | 1224 | * (see split_page()). |
1225 | * each sub-page, and then freeing them one by one when you free | ||
1226 | * them rather than freeing it as a compound page). | ||
1227 | * | 1225 | * |
1228 | * NOTE! Traditionally this was done with "remap_pfn_range()" which | 1226 | * NOTE! Traditionally this was done with "remap_pfn_range()" which |
1229 | * took an arbitrary page protection parameter. This doesn't allow | 1227 | * took an arbitrary page protection parameter. This doesn't allow |
diff --git a/mm/page_alloc.c b/mm/page_alloc.c index 102919851353..fc65e87368b3 100644 --- a/mm/page_alloc.c +++ b/mm/page_alloc.c | |||
@@ -752,6 +752,28 @@ static inline void prep_zero_page(struct page *page, int order, gfp_t gfp_flags) | |||
752 | clear_highpage(page + i); | 752 | clear_highpage(page + i); |
753 | } | 753 | } |
754 | 754 | ||
755 | #ifdef CONFIG_MMU | ||
756 | /* | ||
757 | * split_page takes a non-compound higher-order page, and splits it into | ||
758 | * n (1<<order) sub-pages: page[0..n] | ||
759 | * Each sub-page must be freed individually. | ||
760 | * | ||
761 | * Note: this is probably too low level an operation for use in drivers. | ||
762 | * Please consult with lkml before using this in your driver. | ||
763 | */ | ||
764 | void split_page(struct page *page, unsigned int order) | ||
765 | { | ||
766 | int i; | ||
767 | |||
768 | BUG_ON(PageCompound(page)); | ||
769 | BUG_ON(!page_count(page)); | ||
770 | for (i = 1; i < (1 << order); i++) { | ||
771 | BUG_ON(page_count(page + i)); | ||
772 | set_page_count(page + i, 1); | ||
773 | } | ||
774 | } | ||
775 | #endif | ||
776 | |||
755 | /* | 777 | /* |
756 | * Really, prep_compound_page() should be called from __rmqueue_bulk(). But | 778 | * Really, prep_compound_page() should be called from __rmqueue_bulk(). But |
757 | * we cheat by calling it from here, in the order > 0 path. Saves a branch | 779 | * we cheat by calling it from here, in the order > 0 path. Saves a branch |