summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAlexander Duyck <alexander.h.duyck@intel.com>2016-12-14 18:05:26 -0500
committerLinus Torvalds <torvalds@linux-foundation.org>2016-12-14 19:04:08 -0500
commit44fdffd70504c15b617686753dfdf9eb0ddf3729 (patch)
tree932083d4e0fd88a393878418d88d56156de9abf3
parent0495c3d367944e4af053983ff3cdf256b567b053 (diff)
mm: add support for releasing multiple instances of a page
Add a function that allows us to batch free a page that has multiple references outstanding. Specifically this function can be used to drop a page being used in the page frag alloc cache. With this drivers can make use of functionality similar to the page frag alloc cache without having to do any workarounds for the fact that there is no function that frees multiple references. Link: http://lkml.kernel.org/r/20161110113606.76501.70752.stgit@ahduyck-blue-test.jf.intel.com Signed-off-by: Alexander Duyck <alexander.h.duyck@intel.com> Cc: "David S. Miller" <davem@davemloft.net> Cc: "James E.J. Bottomley" <jejb@parisc-linux.org> Cc: Chris Metcalf <cmetcalf@mellanox.com> Cc: David Howells <dhowells@redhat.com> Cc: Geert Uytterhoeven <geert@linux-m68k.org> Cc: Hans-Christian Noren Egtvedt <egtvedt@samfundet.no> Cc: Helge Deller <deller@gmx.de> Cc: James Hogan <james.hogan@imgtec.com> Cc: Jeff Kirsher <jeffrey.t.kirsher@intel.com> Cc: Jonas Bonn <jonas@southpole.se> Cc: Keguang Zhang <keguang.zhang@gmail.com> Cc: Ley Foon Tan <lftan@altera.com> Cc: Mark Salter <msalter@redhat.com> Cc: Max Filippov <jcmvbkbc@gmail.com> Cc: Michael Ellerman <mpe@ellerman.id.au> Cc: Michal Simek <monstr@monstr.eu> Cc: Ralf Baechle <ralf@linux-mips.org> Cc: Rich Felker <dalias@libc.org> Cc: Richard Kuo <rkuo@codeaurora.org> Cc: Russell King <linux@armlinux.org.uk> Cc: Steven Miao <realmz6@gmail.com> Cc: Tobias Klauser <tklauser@distanz.ch> Cc: Vineet Gupta <vgupta@synopsys.com> Cc: Yoshinori Sato <ysato@users.sourceforge.jp> Signed-off-by: Andrew Morton <akpm@linux-foundation.org> Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
-rw-r--r--include/linux/gfp.h2
-rw-r--r--mm/page_alloc.c14
2 files changed, 16 insertions, 0 deletions
diff --git a/include/linux/gfp.h b/include/linux/gfp.h
index f8041f9de31e..4175dca4ac39 100644
--- a/include/linux/gfp.h
+++ b/include/linux/gfp.h
@@ -506,6 +506,8 @@ extern void free_hot_cold_page(struct page *page, bool cold);
506extern void free_hot_cold_page_list(struct list_head *list, bool cold); 506extern void free_hot_cold_page_list(struct list_head *list, bool cold);
507 507
508struct page_frag_cache; 508struct page_frag_cache;
509extern void __page_frag_drain(struct page *page, unsigned int order,
510 unsigned int count);
509extern void *__alloc_page_frag(struct page_frag_cache *nc, 511extern void *__alloc_page_frag(struct page_frag_cache *nc,
510 unsigned int fragsz, gfp_t gfp_mask); 512 unsigned int fragsz, gfp_t gfp_mask);
511extern void __free_page_frag(void *addr); 513extern void __free_page_frag(void *addr);
diff --git a/mm/page_alloc.c b/mm/page_alloc.c
index f64e7bcb43b7..2c6d5f64feca 100644
--- a/mm/page_alloc.c
+++ b/mm/page_alloc.c
@@ -3925,6 +3925,20 @@ static struct page *__page_frag_refill(struct page_frag_cache *nc,
3925 return page; 3925 return page;
3926} 3926}
3927 3927
3928void __page_frag_drain(struct page *page, unsigned int order,
3929 unsigned int count)
3930{
3931 VM_BUG_ON_PAGE(page_ref_count(page) == 0, page);
3932
3933 if (page_ref_sub_and_test(page, count)) {
3934 if (order == 0)
3935 free_hot_cold_page(page, false);
3936 else
3937 __free_pages_ok(page, order);
3938 }
3939}
3940EXPORT_SYMBOL(__page_frag_drain);
3941
3928void *__alloc_page_frag(struct page_frag_cache *nc, 3942void *__alloc_page_frag(struct page_frag_cache *nc,
3929 unsigned int fragsz, gfp_t gfp_mask) 3943 unsigned int fragsz, gfp_t gfp_mask)
3930{ 3944{