diff options
Diffstat (limited to 'mm/page_alloc.c')
-rw-r--r-- | mm/page_alloc.c | 22 |
1 files changed, 22 insertions, 0 deletions
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 |