diff options
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 |