#ifndef LINUX_QUICKLIST_H #define LINUX_QUICKLIST_H /* * Fast allocations and disposal of pages. Pages must be in the condition * as needed after allocation when they are freed. Per cpu lists of pages * are kept that only contain node local pages. * * (C) 2007, SGI. Christoph Lameter <clameter@sgi.com> */ #include <linux/kernel.h> #include <linux/gfp.h> #include <linux/percpu.h> #ifdef CONFIG_QUICKLIST struct quicklist { void *page; int nr_pages; }; DECLARE_PER_CPU(struct quicklist, quicklist)[CONFIG_NR_QUICK]; /* * The two key functions quicklist_alloc and quicklist_free are inline so * that they may be custom compiled for the platform. * Specifying a NULL ctor can remove constructor support. Specifying * a constant quicklist allows the determination of the exact address * in the per cpu area. * * The fast patch in quicklist_alloc touched only a per cpu cacheline and * the first cacheline of the page itself. There is minmal overhead involved. */ static inline void *quicklist_alloc(int nr, gfp_t flags, void (*ctor)(void *)) { struct quicklist *q; void **p = NULL; q =&get_cpu_var(quicklist)[nr]; p = q->page; if (likely(p)) { q->page = p[0]; p[0] = NULL; q->nr_pages--; } put_cpu_var(quicklist); if (likely(p)) return p; p = (void *)__get_free_page(flags | __GFP_ZERO); if (ctor && p) ctor(p); return p; } static inline void __quicklist_free(int nr, void (*dtor)(void *), void *p, struct page *page) { struct quicklist *q; int nid = page_to_nid(page); if (unlikely(nid != numa_node_id())) { if (dtor) dtor(p); __free_page(page); return; } q = &get_cpu_var(quicklist)[nr]; *(void **)p = q->page; q->page = p; q->nr_pages++; put_cpu_var(quicklist); } static inline void quicklist_free(int nr, void (*dtor)(void *), void *pp) { __quicklist_free(nr, dtor, pp, virt_to_page(pp)); } static inline void quicklist_free_page(int nr, void (*dtor)(void *), struct page *page) { __quicklist_free(nr, dtor, page_address(page), page); } void quicklist_trim(int nr, void (*dtor)(void *), unsigned long min_pages, unsigned long max_free); unsigned long quicklist_total_size(void); #endif #endif /* LINUX_QUICKLIST_H */