cregit-Linux how code gets into the kernel

Release 4.8 mm/vmalloc.c

Directory: mm
/*
 *  linux/mm/vmalloc.c
 *
 *  Copyright (C) 1993  Linus Torvalds
 *  Support of BIGMEM added by Gerhard Wichert, Siemens AG, July 1999
 *  SMP-safe vmalloc/vfree/ioremap, Tigran Aivazian <tigran@veritas.com>, May 2000
 *  Major rework to support vmap/vunmap, Christoph Hellwig, SGI, August 2002
 *  Numa awareness, Christoph Lameter, SGI, June 2005
 */

#include <linux/vmalloc.h>
#include <linux/mm.h>
#include <linux/module.h>
#include <linux/highmem.h>
#include <linux/sched.h>
#include <linux/slab.h>
#include <linux/spinlock.h>
#include <linux/interrupt.h>
#include <linux/proc_fs.h>
#include <linux/seq_file.h>
#include <linux/debugobjects.h>
#include <linux/kallsyms.h>
#include <linux/list.h>
#include <linux/notifier.h>
#include <linux/rbtree.h>
#include <linux/radix-tree.h>
#include <linux/rcupdate.h>
#include <linux/pfn.h>
#include <linux/kmemleak.h>
#include <linux/atomic.h>
#include <linux/compiler.h>
#include <linux/llist.h>
#include <linux/bitops.h>

#include <asm/uaccess.h>
#include <asm/tlbflush.h>
#include <asm/shmparam.h>

#include "internal.h"


struct vfree_deferred {
	
struct llist_head list;
	
struct work_struct wq;
};
static DEFINE_PER_CPU(struct vfree_deferred, vfree_deferred);

static void __vunmap(const void *, int);


static void free_work(struct work_struct *w) { struct vfree_deferred *p = container_of(w, struct vfree_deferred, wq); struct llist_node *llnode = llist_del_all(&p->list); while (llnode) { void *p = llnode; llnode = llist_next(llnode); __vunmap(p, 1); } }

Contributors

PersonTokensPropCommitsCommitProp
al viroal viro65100.00%1100.00%
Total65100.00%1100.00%

/*** Page table manipulation functions ***/
static void vunmap_pte_range(pmd_t *pmd, unsigned long addr, unsigned long end) { pte_t *pte; pte = pte_offset_kernel(pmd, addr); do { pte_t ptent = ptep_get_and_clear(&init_mm, addr, pte); WARN_ON(!pte_none(ptent) && !pte_present(ptent)); } while (pte++, addr += PAGE_SIZE, addr != end); }

Contributors

PersonTokensPropCommitsCommitProp
pre-gitpre-git4356.58%660.00%
hugh dickinshugh dickins2431.58%110.00%
david s. millerdavid s. miller45.26%110.00%
christoph hellwigchristoph hellwig45.26%110.00%
ingo molnaringo molnar11.32%110.00%
Total76100.00%10100.00%


static void vunmap_pmd_range(pud_t *pud, unsigned long addr, unsigned long end) { pmd_t *pmd; unsigned long next; pmd = pmd_offset(pud, addr); do { next = pmd_addr_end(addr, end); if (pmd_clear_huge(pmd)) continue; if (pmd_none_or_clear_bad(pmd)) continue; vunmap_pte_range(pmd, addr, next); } while (pmd++, addr = next, addr != end); }

Contributors

PersonTokensPropCommitsCommitProp
hugh dickinshugh dickins3641.86%222.22%
pre-gitpre-git3540.70%444.44%
toshi kanitoshi kani89.30%111.11%
christoph hellwigchristoph hellwig44.65%111.11%
andi kleenandi kleen33.49%111.11%
Total86100.00%9100.00%


static void vunmap_pud_range(pgd_t *pgd, unsigned long addr, unsigned long end) { pud_t *pud; unsigned long next; pud = pud_offset(pgd, addr); do { next = pud_addr_end(addr, end); if (pud_clear_huge(pud)) continue; if (pud_none_or_clear_bad(pud)) continue; vunmap_pmd_range(pud, addr, next); } while (pud++, addr = next, addr != end); }

Contributors

PersonTokensPropCommitsCommitProp
andi kleenandi kleen4552.33%125.00%
hugh dickinshugh dickins3338.37%250.00%
toshi kanitoshi kani89.30%125.00%
Total86100.00%4100.00%


static void vunmap_page_range(unsigned long addr, unsigned long end) { pgd_t *pgd; unsigned long next; BUG_ON(addr >= end); pgd = pgd_offset_k(addr); do { next = pgd_addr_end(addr, end); if (pgd_none_or_clear_bad(pgd)) continue; vunmap_pud_range(pgd, addr, next); } while (pgd++, addr = next, addr != end); }

Contributors

PersonTokensPropCommitsCommitProp
hugh dickinshugh dickins4860.76%222.22%
pre-gitpre-git2126.58%444.44%
benjamin herrenschmidtbenjamin herrenschmidt67.59%111.11%
nick pigginnick piggin33.80%111.11%
andrew mortonandrew morton11.27%111.11%
Total79100.00%9100.00%


static int vmap_pte_range(pmd_t *pmd, unsigned long addr, unsigned long end, pgprot_t prot, struct page **pages, int *nr) { pte_t *pte; /* * nr is a running index into the array which helps higher level * callers keep track of where we're up to. */ pte = pte_alloc_kernel(pmd, addr); if (!pte) return -ENOMEM; do { struct page *page = pages[*nr]; if (WARN_ON(!pte_none(*pte))) return -EBUSY; if (WARN_ON(!page)) return -ENOMEM; set_pte_at(&init_mm, addr, pte, mk_pte(page, prot)); (*nr)++; } while (pte++, addr += PAGE_SIZE, addr != end); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
pre-gitpre-git4835.29%440.00%
hugh dickinshugh dickins4533.09%110.00%
nick pigginnick piggin2014.71%110.00%
andrew mortonandrew morton1712.50%110.00%
christoph hellwigchristoph hellwig42.94%110.00%
oleg nesterovoleg nesterov10.74%110.00%
ingo molnaringo molnar10.74%110.00%
Total136100.00%10100.00%


static int vmap_pmd_range(pud_t *pud, unsigned long addr, unsigned long end, pgprot_t prot, struct page **pages, int *nr) { pmd_t *pmd; unsigned long next; pmd = pmd_alloc(&init_mm, pud, addr); if (!pmd) return -ENOMEM; do { next = pmd_addr_end(addr, end); if (vmap_pte_range(pmd, addr, next, prot, pages, nr)) return -ENOMEM; } while (pmd++, addr = next, addr != end); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
andi kleenandi kleen6861.82%125.00%
hugh dickinshugh dickins3430.91%125.00%
nick pigginnick piggin65.45%125.00%
andrew mortonandrew morton21.82%125.00%
Total110100.00%4100.00%


static int vmap_pud_range(pgd_t *pgd, unsigned long addr, unsigned long end, pgprot_t prot, struct page **pages, int *nr) { pud_t *pud; unsigned long next; pud = pud_alloc(&init_mm, pgd, addr); if (!pud) return -ENOMEM; do { next = pud_addr_end(addr, end); if (vmap_pmd_range(pud, addr, next, prot, pages, nr)) return -ENOMEM; } while (pud++, addr = next, addr != end); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
hugh dickinshugh dickins6760.91%114.29%
andi kleenandi kleen1614.55%114.29%
christoph hellwigchristoph hellwig87.27%114.29%
pre-gitpre-git76.36%114.29%
nick pigginnick piggin65.45%114.29%
andrew mortonandrew morton65.45%228.57%
Total110100.00%7100.00%

/* * Set up page tables in kva (addr, end). The ptes shall have prot "prot", and * will have pfns corresponding to the "pages" array. * * Ie. pte at addr+N*PAGE_SIZE shall point to pfn corresponding to pages[N] */
static int vmap_page_range_noflush(unsigned long start, unsigned long end, pgprot_t prot, struct page **pages) { pgd_t *pgd; unsigned long next; unsigned long addr = start; int err = 0; int nr = 0; BUG_ON(addr >= end); pgd = pgd_offset_k(addr); do { next = pgd_addr_end(addr, end); err = vmap_pud_range(pgd, addr, next, prot, pages, &nr); if (err) return err; } while (pgd++, addr = next, addr != end); return nr; }

Contributors

PersonTokensPropCommitsCommitProp
hugh dickinshugh dickins4034.78%18.33%
christoph hellwigchristoph hellwig2219.13%18.33%
nick pigginnick piggin1916.52%18.33%
pre-gitpre-git86.96%325.00%
adam lackorzynskiadam lackorzynski76.09%18.33%
andi kleenandi kleen65.22%18.33%
andrew mortonandrew morton65.22%18.33%
marcus alanenmarcus alanen32.61%18.33%
figo zhangfigo zhang32.61%18.33%
tejun heotejun heo10.87%18.33%
Total115100.00%12100.00%


static int vmap_page_range(unsigned long start, unsigned long end, pgprot_t prot, struct page **pages) { int ret; ret = vmap_page_range_noflush(start, end, prot, pages); flush_cache_vmap(start, end); return ret; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo49100.00%1100.00%
Total49100.00%1100.00%


int is_vmalloc_or_module_addr(const void *x) { /* * ARM, x86-64 and sparc64 put modules in a special place, * and fall back on vmalloc() if that fails. Others * just put it in the vmalloc space. */ #if defined(CONFIG_MODULES) && defined(MODULES_VADDR) unsigned long addr = (unsigned long)x; if (addr >= MODULES_VADDR && addr < MODULES_END) return 1; #endif return is_vmalloc_addr(x); }

Contributors

PersonTokensPropCommitsCommitProp
linus torvaldslinus torvalds5298.11%150.00%
russell kingrussell king11.89%150.00%
Total53100.00%2100.00%

/* * Walk a vmap address to the struct page it maps. */
struct page *vmalloc_to_page(const void *vmalloc_addr) { unsigned long addr = (unsigned long) vmalloc_addr; struct page *page = NULL; pgd_t *pgd = pgd_offset_k(addr); /* * XXX we might need to change this if we add VIRTUAL_BUG_ON for * architectures that do not vmalloc module space */ VIRTUAL_BUG_ON(!is_vmalloc_or_module_addr(vmalloc_addr)); if (!pgd_none(*pgd)) { pud_t *pud = pud_offset(pgd, addr); if (!pud_none(*pud)) { pmd_t *pmd = pmd_offset(pud, addr); if (!pmd_none(*pmd)) { pte_t *ptep, pte; ptep = pte_offset_map(pmd, addr); pte = *ptep; if (pte_present(pte)) page = pte_page(pte); pte_unmap(ptep); } } } return page; }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter10975.17%228.57%
nick pigginnick piggin149.66%114.29%
vassili karpovvassili karpov128.28%114.29%
jiri slabyjiri slaby85.52%114.29%
ingo molnaringo molnar10.69%114.29%
linus torvaldslinus torvalds10.69%114.29%
Total145100.00%7100.00%

EXPORT_SYMBOL(vmalloc_to_page); /* * Map a vmalloc()-space virtual address to the physical page frame number. */
unsigned long vmalloc_to_pfn(const void *vmalloc_addr) { return page_to_pfn(vmalloc_to_page(vmalloc_addr)); }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter1575.00%266.67%
vassili karpovvassili karpov525.00%133.33%
Total20100.00%3100.00%

EXPORT_SYMBOL(vmalloc_to_pfn); /*** Global kva allocator ***/ #define VM_VM_AREA 0x04 static DEFINE_SPINLOCK(vmap_area_lock); /* Export for kexec only */ LIST_HEAD(vmap_area_list); static LLIST_HEAD(vmap_purge_list); static struct rb_root vmap_area_root = RB_ROOT; /* The vmap cache globals are protected by vmap_area_lock */ static struct rb_node *free_vmap_cache; static unsigned long cached_hole_size; static unsigned long cached_vstart; static unsigned long cached_align; static unsigned long vmap_area_pcpu_hole;
static struct vmap_area *__find_vmap_area(unsigned long addr) { struct rb_node *n = vmap_area_root.rb_node; while (n) { struct vmap_area *va; va = rb_entry(n, struct vmap_area, rb_node); if (addr < va->va_start) n = n->rb_left; else if (addr >= va->va_end) n = n->rb_right; else return va; } return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin5467.50%120.00%
james bottomleyjames bottomley2025.00%120.00%
hirofumi ogawahirofumi ogawa33.75%120.00%
daisuke hatayamadaisuke hatayama22.50%120.00%
christoph hellwigchristoph hellwig11.25%120.00%
Total80100.00%5100.00%


static void __insert_vmap_area(struct vmap_area *va) { struct rb_node **p = &vmap_area_root.rb_node; struct rb_node *parent = NULL; struct rb_node *tmp; while (*p) { struct vmap_area *tmp_va; parent = *p; tmp_va = rb_entry(parent, struct vmap_area, rb_node); if (va->va_start < tmp_va->va_end) p = &(*p)->rb_left; else if (va->va_end > tmp_va->va_start) p = &(*p)->rb_right; else BUG(); } rb_link_node(&va->rb_node, parent, p); rb_insert_color(&va->rb_node, &vmap_area_root); /* address-sort this list */ tmp = rb_prev(&va->rb_node); if (tmp) { struct vmap_area *prev; prev = rb_entry(tmp, struct vmap_area, rb_node); list_add_rcu(&va->list, &prev->list); } else list_add_rcu(&va->list, &vmap_area_list); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin16988.95%120.00%
pre-gitpre-git157.89%120.00%
namhyung kimnamhyung kim42.11%120.00%
joonsoo kimjoonsoo kim10.53%120.00%
christoph lameterchristoph lameter10.53%120.00%
Total190100.00%5100.00%

static void purge_vmap_area_lazy(void); static BLOCKING_NOTIFIER_HEAD(vmap_notify_list); /* * Allocate a region of KVA of the specified size and alignment, within the * vstart and vend. */
static struct vmap_area *alloc_vmap_area(unsigned long size, unsigned long align, unsigned long vstart, unsigned long vend, int node, gfp_t gfp_mask) { struct vmap_area *va; struct rb_node *n; unsigned long addr; int purged = 0; struct vmap_area *first; BUG_ON(!size); BUG_ON(offset_in_page(size)); BUG_ON(!is_power_of_2(align)); might_sleep_if(gfpflags_allow_blocking(gfp_mask)); va = kmalloc_node(sizeof(struct vmap_area), gfp_mask & GFP_RECLAIM_MASK, node); if (unlikely(!va)) return ERR_PTR(-ENOMEM); /* * Only scan the relevant parts containing pointers to other objects * to avoid false negatives. */ kmemleak_scan_area(&va->rb_node, SIZE_MAX, gfp_mask & GFP_RECLAIM_MASK); retry: spin_lock(&vmap_area_lock); /* * Invalidate cache if we have more permissive parameters. * cached_hole_size notes the largest hole noticed _below_ * the vmap_area cached in free_vmap_cache: if size fits * into that hole, we want to scan from vstart to reuse * the hole instead of allocating above free_vmap_cache. * Note that __free_vmap_area may update free_vmap_cache * without updating cached_hole_size or cached_align. */ if (!free_vmap_cache || size < cached_hole_size || vstart < cached_vstart || align < cached_align) { nocache: cached_hole_size = 0; free_vmap_cache = NULL; } /* record if we encounter less permissive parameters */ cached_vstart = vstart; cached_align = align; /* find starting point for our search */ if (free_vmap_cache) { first = rb_entry(free_vmap_cache, struct vmap_area, rb_node); addr = ALIGN(first->va_end, align); if (addr < vstart) goto nocache; if (addr + size < addr) goto overflow; } else { addr = ALIGN(vstart, align); if (addr + size < addr) goto overflow; n = vmap_area_root.rb_node; first = NULL; while (n) { struct vmap_area *tmp; tmp = rb_entry(n, struct vmap_area, rb_node); if (tmp->va_end >= addr) { first = tmp; if (tmp->va_start <= addr) break; n = n->rb_left; } else n = n->rb_right; } if (!first) goto found; } /* from the starting point, walk areas until a suitable hole is found */ while (addr + size > first->va_start && addr + size <= vend) { if (addr + cached_hole_size < first->va_start) cached_hole_size = first->va_start - addr; addr = ALIGN(first->va_end, align); if (addr + size < addr) goto overflow; if (list_is_last(&first->list, &vmap_area_list)) goto found; first = list_next_entry(first, list); } found: if (addr + size > vend) goto overflow; va->va_start = addr; va->va_end = addr + size; va->flags = 0; __insert_vmap_area(va); free_vmap_cache = &va->rb_node; spin_unlock(&vmap_area_lock); BUG_ON(!IS_ALIGNED(va->va_start, align)); BUG_ON(va->va_start < vstart); BUG_ON(va->va_end > vend); return va; overflow: spin_unlock(&vmap_area_lock); if (!purged) { purge_vmap_area_lazy(); purged = 1; goto retry; } if (gfpflags_allow_blocking(gfp_mask)) { unsigned long freed = 0; blocking_notifier_call_chain(&vmap_notify_list, 0, &freed); if (freed > 0) { purged = 0; goto retry; } } if (printk_ratelimit()) pr_warn("vmap allocation for size %lu failed: use vmalloc=<size> to increase size\n", size); kfree(va); return ERR_PTR(-EBUSY); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin48582.48%320.00%
chris wilsonchris wilson498.33%16.67%
catalin marinascatalin marinas152.55%16.67%
hong zhi guohong zhi guo122.04%16.67%
glauber costaglauber costa101.70%213.33%
wang xiaoqiangwang xiaoqiang50.85%16.67%
ralph wuerthnerralph wuerthner50.85%16.67%
alexander kuleshovalexander kuleshov30.51%16.67%
geliang tanggeliang tang10.17%16.67%
johannes weinerjohannes weiner10.17%16.67%
pintu kumarpintu kumar10.17%16.67%
joe perchesjoe perches10.17%16.67%
Total588100.00%15100.00%


int register_vmap_purge_notifier(struct notifier_block *nb) { return blocking_notifier_chain_register(&vmap_notify_list, nb); }

Contributors

PersonTokensPropCommitsCommitProp
chris wilsonchris wilson19100.00%1100.00%
Total19100.00%1100.00%

EXPORT_SYMBOL_GPL(register_vmap_purge_notifier);
int unregister_vmap_purge_notifier(struct notifier_block *nb) { return blocking_notifier_chain_unregister(&vmap_notify_list, nb); }

Contributors

PersonTokensPropCommitsCommitProp
chris wilsonchris wilson19100.00%1100.00%
Total19100.00%1100.00%

EXPORT_SYMBOL_GPL(unregister_vmap_purge_notifier);
static void __free_vmap_area(struct vmap_area *va) { BUG_ON(RB_EMPTY_NODE(&va->rb_node)); if (free_vmap_cache) { if (va->va_end < cached_vstart) { free_vmap_cache = NULL; } else { struct vmap_area *cache; cache = rb_entry(free_vmap_cache, struct vmap_area, rb_node); if (va->va_start <= cache->va_start) { free_vmap_cache = rb_prev(&va->rb_node); /* * We don't try to update cached_hole_size or * cached_align, but it won't go very wrong. */ } } } rb_erase(&va->rb_node, &vmap_area_root); RB_CLEAR_NODE(&va->rb_node); list_del_rcu(&va->list); /* * Track the highest possible candidate for pcpu area * allocation. Areas outside of vmalloc area can be returned * here too, consider only end addresses which fall inside * vmalloc area proper. */ if (va->va_end > VMALLOC_START && va->va_end <= VMALLOC_END) vmap_area_pcpu_hole = max(vmap_area_pcpu_hole, va->va_end); kfree_rcu(va, rcu_head); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin11780.69%250.00%
tejun heotejun heo2617.93%125.00%
lai jiangshanlai jiangshan21.38%125.00%
Total145100.00%4100.00%

/* * Free a region of KVA allocated by alloc_vmap_area */
static void free_vmap_area(struct vmap_area *va) { spin_lock(&vmap_area_lock); __free_vmap_area(va); spin_unlock(&vmap_area_lock); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin28100.00%1100.00%
Total28100.00%1100.00%

/* * Clear the pagetable entries of a given vmap_area */
static void unmap_vmap_area(struct vmap_area *va) { vunmap_page_range(va->va_start, va->va_end); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin22100.00%1100.00%
Total22100.00%1100.00%


static void vmap_debug_free_range(unsigned long start, unsigned long end) { /* * Unmap page tables and force a TLB flush immediately if pagealloc * debugging is enabled. This catches use after free bugs similarly to * those in linear kernel virtual address space after a page has been * freed. * * All the lazy freeing logic is still retained, in order to minimise * intrusiveness of this debugging feature. * * This is going to be *slow* (linear kernel virtual address debugging * doesn't do a broadcast TLB flush so it is a lot faster). */ if (debug_pagealloc_enabled()) { vunmap_page_range(start, end); flush_tlb_kernel_range(start, end); } }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin2877.78%150.00%
joonsoo kimjoonsoo kim822.22%150.00%
Total36100.00%2100.00%

/* * lazy_max_pages is the maximum amount of virtual address space we gather up * before attempting to purge with a TLB flush. * * There is a tradeoff here: a larger number will cover more kernel page tables * and take slightly longer to purge, but it will linearly reduce the number of * global TLB flushes that must be performed. It would seem natural to scale * this number up linearly with the number of CPUs (because vmapping activity * could also scale linearly with the number of CPUs), however it is likely * that in practice, workloads might be constrained in other ways that mean * vmap activity will not scale linearly with CPUs. Also, I want to be * conservative and not introduce a big latency on huge systems, so go with * a less aggressive log scale. It will still be an improvement over the old * code, and it will be simple to change the scale factor if we find that it * becomes a problem on bigger systems. */
static unsigned long lazy_max_pages(void) { unsigned int log; log = fls(num_online_cpus()); return log * (32UL * 1024 * 1024 / PAGE_SIZE); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin34100.00%1100.00%
Total34100.00%1100.00%

static atomic_t vmap_lazy_nr = ATOMIC_INIT(0); /* for per-CPU blocks */ static void purge_fragmented_blocks_allcpus(void); /* * called before a call to iounmap() if the caller wants vm_area_struct's * immediately freed. */
void set_iounmap_nonlazy(void) { atomic_set(&vmap_lazy_nr, lazy_max_pages()+1); }

Contributors

PersonTokensPropCommitsCommitProp
cliff wickmancliff wickman18100.00%1100.00%
Total18100.00%1100.00%

/* * Purges all lazily-freed vmap areas. * * If sync is 0 then don't purge if there is already a purge in progress. * If force_flush is 1, then flush kernel TLBs between *start and *end even * if we found no lazy vmap areas to unmap (callers can use this to optimise * their own TLB flushing). * Returns with *start = min(*start, lowest purged address) * *end = max(*end, highest purged address) */
static void __purge_vmap_area_lazy(unsigned long *start, unsigned long *end, int sync, int force_flush) { static DEFINE_SPINLOCK(purge_lock); struct llist_node *valist; struct vmap_area *va; struct vmap_area *n_va; int nr = 0; /* * If sync is 0 but force_flush is 1, we'll go sync anyway but callers * should not expect such behaviour. This just simplifies locking for * the case that isn't actually used at the moment anyway. */ if (!sync && !force_flush) { if (!spin_trylock(&purge_lock)) return; } else spin_lock(&purge_lock); if (sync) purge_fragmented_blocks_allcpus(); valist = llist_del_all(&vmap_purge_list); llist_for_each_entry(va, valist, purge_list) { if (va->va_start < *start) *start = va->va_start; if (va->va_end > *end) *end = va->va_end; nr += (va->va_end - va->va_start) >> PAGE_SHIFT; } if (nr) atomic_sub(nr, &vmap_lazy_nr); if (nr || force_flush) flush_tlb_kernel_range(*start, *end); if (nr) { spin_lock(&vmap_area_lock); llist_for_each_entry_safe(va, n_va, valist, purge_list) __free_vmap_area(va); spin_unlock(&vmap_area_lock); } spin_unlock(&purge_lock); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin18687.32%240.00%
chris wilsonchris wilson167.51%120.00%
vegard nossumvegard nossum73.29%120.00%
andrew mortonandrew morton41.88%120.00%
Total213100.00%5100.00%

/* * Kick off a purge of the outstanding lazy areas. Don't bother if somebody * is already purging. */
static void try_purge_vmap_area_lazy(void) { unsigned long start = ULONG_MAX, end = 0; __purge_vmap_area_lazy(&start, &end, 0, 0); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin31100.00%1100.00%
Total31100.00%1100.00%

/* * Kick off a purge of the outstanding lazy areas. */
static void purge_vmap_area_lazy(void) { unsigned long start = ULONG_MAX, end = 0; __purge_vmap_area_lazy(&start, &end, 1, 0); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin31100.00%2100.00%
Total31100.00%2100.00%

/* * Free a vmap area, caller ensuring that the area has been unmapped * and flush_cache_vunmap had been called for the correct range * previously. */
static void free_vmap_area_noflush(struct vmap_area *va) { int nr_lazy; nr_lazy = atomic_add_return((va->va_end - va->va_start) >> PAGE_SHIFT, &vmap_lazy_nr); /* After this point, we may free va at any time */ llist_add(&va->purge_list, &vmap_purge_list); if (unlikely(nr_lazy > lazy_max_pages())) try_purge_vmap_area_lazy(); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin3966.10%250.00%
chris wilsonchris wilson1932.20%125.00%
jeremy fitzhardingejeremy fitzhardinge11.69%125.00%
Total59100.00%4100.00%

/* * Free and unmap a vmap area, caller ensuring flush_cache_vunmap had been * called for the correct range previously. */
static void free_unmap_vmap_area_noflush(struct vmap_area *va) { unmap_vmap_area(va); free_vmap_area_noflush(va); }

Contributors

PersonTokensPropCommitsCommitProp
jeremy fitzhardingejeremy fitzhardinge21100.00%1100.00%
Total21100.00%1100.00%

/* * Free and unmap a vmap area */
static void free_unmap_vmap_area(struct vmap_area *va) { flush_cache_vunmap(va->va_start, va->va_end); free_unmap_vmap_area_noflush(va); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin27100.00%1100.00%
Total27100.00%1100.00%


static struct vmap_area *find_vmap_area(unsigned long addr) { struct vmap_area *va; spin_lock(&vmap_area_lock); va = __find_vmap_area(addr); spin_unlock(&vmap_area_lock); return va; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin39100.00%1100.00%
Total39100.00%1100.00%


static void free_unmap_vmap_area_addr(unsigned long addr) { struct vmap_area *va; va = find_vmap_area(addr); BUG_ON(!va); free_unmap_vmap_area(va); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin33100.00%1100.00%
Total33100.00%1100.00%

/*** Per cpu kva allocator ***/ /* * vmap space is limited especially on 32 bit architectures. Ensure there is * room for at least 16 percpu vmap blocks per CPU. */ /* * If we had a constant VMALLOC_START and VMALLOC_END, we'd like to be able * to #define VMALLOC_SPACE (VMALLOC_END-VMALLOC_START). Guess * instead (we just need a rough idea) */ #if BITS_PER_LONG == 32 #define VMALLOC_SPACE (128UL*1024*1024) #else #define VMALLOC_SPACE (128UL*1024*1024*1024) #endif #define VMALLOC_PAGES (VMALLOC_SPACE / PAGE_SIZE) #define VMAP_MAX_ALLOC BITS_PER_LONG /* 256K with 4K pages */ #define VMAP_BBMAP_BITS_MAX 1024 /* 4MB with 4K pages */ #define VMAP_BBMAP_BITS_MIN (VMAP_MAX_ALLOC*2) #define VMAP_MIN(x, y) ((x) < (y) ? (x) : (y)) /* can't use min() */ #define VMAP_MAX(x, y) ((x) > (y) ? (x) : (y)) /* can't use max() */ #define VMAP_BBMAP_BITS \ VMAP_MIN(VMAP_BBMAP_BITS_MAX, \ VMAP_MAX(VMAP_BBMAP_BITS_MIN, \ VMALLOC_PAGES / roundup_pow_of_two(NR_CPUS) / 16)) #define VMAP_BLOCK_SIZE (VMAP_BBMAP_BITS * PAGE_SIZE) static bool vmap_initialized __read_mostly = false; struct vmap_block_queue { spinlock_t lock; struct list_head free; }; struct vmap_block { spinlock_t lock; struct vmap_area *va; unsigned long free, dirty; unsigned long dirty_min, dirty_max; /*< dirty range */ struct list_head free_list; struct rcu_head rcu_head; struct list_head purge; }; /* Queue of free and dirty vmap blocks, for allocation and flushing purposes */ static DEFINE_PER_CPU(struct vmap_block_queue, vmap_block_queue); /* * Radix tree of vmap blocks, indexed by address, to quickly find a vmap block * in the free path. Could get rid of this if we change the API to return a * "cookie" from alloc, to be passed to free. But no big deal yet. */ static DEFINE_SPINLOCK(vmap_block_tree_lock); static RADIX_TREE(vmap_block_tree, GFP_ATOMIC); /* * We should probably have a fallback mechanism to allocate virtual memory * out of partially filled vmap blocks. However vmap block sizing should be * fairly reasonable according to the vmalloc size, so it shouldn't be a * big problem. */
static unsigned long addr_to_vb_idx(unsigned long addr) { addr -= VMALLOC_START & ~(VMAP_BLOCK_SIZE-1); addr /= VMAP_BLOCK_SIZE; return addr; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin29100.00%1100.00%
Total29100.00%1100.00%


static void *vmap_block_vaddr(unsigned long va_start, unsigned long pages_off) { unsigned long addr; addr = va_start + (pages_off << PAGE_SHIFT); BUG_ON(addr_to_vb_idx(addr) != addr_to_vb_idx(va_start)); return (void *)addr; }

Contributors

PersonTokensPropCommitsCommitProp
roman peniaevroman peniaev49100.00%1100.00%
Total49100.00%1100.00%

/** * new_vmap_block - allocates new vmap_block and occupies 2^order pages in this * block. Of course pages number can't exceed VMAP_BBMAP_BITS * @order: how many 2^order pages should be occupied in newly allocated block * @gfp_mask: flags for the page level allocator * * Returns: virtual address in a newly allocated block or ERR_PTR(-errno) */
static void *new_vmap_block(unsigned int order, gfp_t gfp_mask) { struct vmap_block_queue *vbq; struct vmap_block *vb; struct vmap_area *va; unsigned long vb_idx; int node, err; void *vaddr; node = numa_node_id(); vb = kmalloc_node(sizeof(struct vmap_block), gfp_mask & GFP_RECLAIM_MASK, node); if (unlikely(!vb)) return ERR_PTR(-ENOMEM); va = alloc_vmap_area(VMAP_BLOCK_SIZE, VMAP_BLOCK_SIZE, VMALLOC_START, VMALLOC_END, node, gfp_mask); if (IS_ERR(va)) { kfree(vb); return ERR_CAST(va); } err = radix_tree_preload(gfp_mask); if (unlikely(err)) { kfree(vb); free_vmap_area(va); return ERR_PTR(err); } vaddr = vmap_block_vaddr(va->va_start, 0); spin_lock_init(&vb->lock); vb->va = va; /* At least something should be left free */ BUG_ON(VMAP_BBMAP_BITS <= (1UL << order)); vb->free = VMAP_BBMAP_BITS - (1UL << order); vb->dirty = 0; vb->dirty_min = VMAP_BBMAP_BITS; vb->dirty_max = 0; INIT_LIST_HEAD(&vb->free_list); vb_idx = addr_to_vb_idx(va->va_start); spin_lock(&vmap_block_tree_lock); err = radix_tree_insert(&vmap_block_tree, vb_idx, vb); spin_unlock(&vmap_block_tree_lock); BUG_ON(err); radix_tree_preload_end(); vbq = &get_cpu_var(vmap_block_queue); spin_lock(&vbq->lock); list_add_tail_rcu(&vb->free_list, &vbq->free); spin_unlock(&vbq->lock); put_cpu_var(vmap_block_queue); return vaddr; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin25983.82%116.67%
roman peniaevroman peniaev4815.53%350.00%
julia lawalljulia lawall10.32%116.67%
tejun heotejun heo10.32%116.67%
Total309100.00%6100.00%


static void free_vmap_block(struct vmap_block *vb) { struct vmap_block *tmp; unsigned long vb_idx; vb_idx = addr_to_vb_idx(vb->va->va_start); spin_lock(&vmap_block_tree_lock); tmp = radix_tree_delete(&vmap_block_tree, vb_idx); spin_unlock(&vmap_block_tree_lock); BUG_ON(tmp != vb); free_vmap_area_noflush(vb->va); kfree_rcu(vb, rcu_head); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin7195.95%133.33%
lai jiangshanlai jiangshan22.70%133.33%
jeremy fitzhardingejeremy fitzhardinge11.35%133.33%
Total74100.00%3100.00%


static void purge_fragmented_blocks(int cpu) { LIST_HEAD(purge); struct vmap_block *vb; struct vmap_block *n_vb; struct vmap_block_queue *vbq = &per_cpu(vmap_block_queue, cpu); rcu_read_lock(); list_for_each_entry_rcu(vb, &vbq->free, free_list) { if (!(vb->free + vb->dirty == VMAP_BBMAP_BITS && vb->dirty != VMAP_BBMAP_BITS)) continue; spin_lock(&vb->lock); if (vb->free + vb->dirty == VMAP_BBMAP_BITS && vb->dirty != VMAP_BBMAP_BITS) { vb->free = 0; /* prevent further allocs after releasing lock */ vb->dirty = VMAP_BBMAP_BITS; /* prevent purging it again */ vb->dirty_min = 0; vb->dirty_max = VMAP_BBMAP_BITS; spin_lock(&vbq->lock); list_del_rcu(&vb->free_list); spin_unlock(&vbq->lock); spin_unlock(&vb->lock); list_add_tail(&vb->purge, &purge); } else spin_unlock(&vb->lock); } rcu_read_unlock(); list_for_each_entry_safe(vb, n_vb, &purge, purge) { list_del(&vb->purge); free_vmap_block(vb); } }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin19896.12%150.00%
roman peniaevroman peniaev83.88%150.00%
Total206100.00%2100.00%


static void purge_fragmented_blocks_allcpus(void) { int cpu; for_each_possible_cpu(cpu) purge_fragmented_blocks(cpu); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin20100.00%1100.00%
Total20100.00%1100.00%


static void *vb_alloc(unsigned long size, gfp_t gfp_mask) { struct vmap_block_queue *vbq; struct vmap_block *vb; void *vaddr = NULL; unsigned int order; BUG_ON(offset_in_page(size)); BUG_ON(size > PAGE_SIZE*VMAP_MAX_ALLOC); if (WARN_ON(size == 0)) { /* * Allocating 0 bytes isn't what caller wants since * get_order(0) returns funny result. Just warn and terminate * early. */ return NULL; } order = get_order(size); rcu_read_lock(); vbq = &get_cpu_var(vmap_block_queue); list_for_each_entry_rcu(vb, &vbq->free, free_list) { unsigned long pages_off; spin_lock(&vb->lock); if (vb->free < (1UL << order)) { spin_unlock(&vb->lock); continue; } pages_off = VMAP_BBMAP_BITS - vb->free; vaddr = vmap_block_vaddr(vb->va->va_start, pages_off); vb->free -= 1UL << order; if (vb->free == 0) { spin_lock(&vbq->lock); list_del_rcu(&vb->free_list); spin_unlock(&vbq->lock); } spin_unlock(&vb->lock); break; } put_cpu_var(vmap_block_queue); rcu_read_unlock(); /* Allocate new block if nothing was found */ if (!vaddr) vaddr = new_vmap_block(order, gfp_mask); return vaddr; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin17677.53%337.50%
roman peniaevroman peniaev3113.66%112.50%
jan karajan kara156.61%112.50%
alexander kuleshovalexander kuleshov31.32%112.50%
tejun heotejun heo10.44%112.50%
zhang yanfeizhang yanfei10.44%112.50%
Total227100.00%8100.00%


static void vb_free(const void *addr, unsigned long size) { unsigned long offset; unsigned long vb_idx; unsigned int order; struct vmap_block *vb; BUG_ON(offset_in_page(size)); BUG_ON(size > PAGE_SIZE*VMAP_MAX_ALLOC); flush_cache_vunmap((unsigned long)addr, (unsigned long)addr + size); order = get_order(size); offset = (unsigned long)addr & (VMAP_BLOCK_SIZE - 1); offset >>= PAGE_SHIFT; vb_idx = addr_to_vb_idx((unsigned long)addr); rcu_read_lock(); vb = radix_tree_lookup(&vmap_block_tree, vb_idx); rcu_read_unlock(); BUG_ON(!vb); vunmap_page_range((unsigned long)addr, (unsigned long)addr + size); spin_lock(&vb->lock); /* Expand dirty range */ vb->dirty_min = min(vb->dirty_min, offset); vb->dirty_max = max(vb->dirty_max, offset + (1UL << order)); vb->dirty += 1UL << order; if (vb->dirty == VMAP_BBMAP_BITS) { BUG_ON(vb->free); spin_unlock(&vb->lock); free_vmap_block(vb); } else spin_unlock(&vb->lock); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin18179.04%350.00%
roman peniaevroman peniaev2812.23%116.67%
jeremy fitzhardingejeremy fitzhardinge177.42%116.67%
alexander kuleshovalexander kuleshov31.31%116.67%
Total229100.00%6100.00%

/** * vm_unmap_aliases - unmap outstanding lazy aliases in the vmap layer * * The vmap/vmalloc layer lazily flushes kernel virtual mappings primarily * to amortize TLB flushing overheads. What this means is that any page you * have now, may, in a former life, have been mapped into kernel virtual * address by the vmap layer and so there might be some CPUs with TLB entries * still referencing that page (additional to the regular 1:1 kernel mapping). * * vm_unmap_aliases flushes all such lazy mappings. After it returns, we can * be sure that none of the pages we have control over will have any aliases * from the vmap layer. */
void vm_unmap_aliases(void) { unsigned long start = ULONG_MAX, end = 0; int cpu; int flush = 0; if (unlikely(!vmap_initialized)) return; for_each_possible_cpu(cpu) { struct vmap_block_queue *vbq = &per_cpu(vmap_block_queue, cpu); struct vmap_block *vb; rcu_read_lock(); list_for_each_entry_rcu(vb, &vbq->free, free_list) { spin_lock(&vb->lock); if (vb->dirty) { unsigned long va_start = vb->va->va_start; unsigned long s, e; s = va_start + (vb->dirty_min << PAGE_SHIFT); e = va_start + (vb->dirty_max << PAGE_SHIFT); start = min(s, start); end = max(e, end); flush = 1; } spin_unlock(&vb->lock); } rcu_read_unlock(); } __purge_vmap_area_lazy(&start, &end, 1, flush); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin13075.14%125.00%
roman peniaevroman peniaev3319.08%125.00%
jeremy fitzhardingejeremy fitzhardinge95.20%125.00%
joonsoo kimjoonsoo kim10.58%125.00%
Total173100.00%4100.00%

EXPORT_SYMBOL_GPL(vm_unmap_aliases); /** * vm_unmap_ram - unmap linear kernel address space set up by vm_map_ram * @mem: the pointer returned by vm_map_ram * @count: the count passed to that vm_map_ram call (cannot unmap partial) */
void vm_unmap_ram(const void *mem, unsigned int count) { unsigned long size = (unsigned long)count << PAGE_SHIFT; unsigned long addr = (unsigned long)mem; BUG_ON(!addr); BUG_ON(addr < VMALLOC_START); BUG_ON(addr > VMALLOC_END); BUG_ON(!PAGE_ALIGNED(addr)); debug_check_no_locks_freed(mem, size); vmap_debug_free_range(addr, addr+size); if (likely(count <= VMAP_MAX_ALLOC)) vb_free(mem, size); else free_unmap_vmap_area_addr(addr); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin9592.23%240.00%
guillermo julián morenoguillermo julián moreno43.88%120.00%
wang xiaoqiangwang xiaoqiang32.91%120.00%
shawn linshawn lin10.97%120.00%
Total103100.00%5100.00%

EXPORT_SYMBOL(vm_unmap_ram); /** * vm_map_ram - map pages linearly into kernel virtual address (vmalloc space) * @pages: an array of pointers to the pages to be mapped * @count: number of pages * @node: prefer to allocate data structures on this node * @prot: memory protection to use. PAGE_KERNEL for regular RAM * * If you use this function for less than VMAP_MAX_ALLOC pages, it could be * faster than vmap so it's good. But if you mix long-life and short-life * objects with vm_map_ram(), it could consume lots of address space through * fragmentation (especially on a 32bit machine). You could see failures in * the end. Please use this function for short-lived objects. * * Returns: a pointer to the address that has been mapped, or %NULL on failure */
void *vm_map_ram(struct page **pages, unsigned int count, int node, pgprot_t prot) { unsigned long size = (unsigned long)count << PAGE_SHIFT; unsigned long addr; void *mem; if (likely(count <= VMAP_MAX_ALLOC)) { mem = vb_alloc(size, GFP_KERNEL); if (IS_ERR(mem)) return NULL; addr = (unsigned long)mem; } else { struct vmap_area *va; va = alloc_vmap_area(size, PAGE_SIZE, VMALLOC_START, VMALLOC_END, node, GFP_KERNEL); if (IS_ERR(va)) return NULL; addr = va->va_start; mem = (void *)addr; } if (vmap_page_range(addr, addr + size, prot, pages) < 0) { vm_unmap_ram(mem, count); return NULL; } return mem; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin15696.89%133.33%
guillermo julián morenoguillermo julián moreno42.48%133.33%
mika kukkonenmika kukkonen10.62%133.33%
Total161100.00%3100.00%

EXPORT_SYMBOL(vm_map_ram); static struct vm_struct *vmlist __initdata; /** * vm_area_add_early - add vmap area early during boot * @vm: vm_struct to add * * This function is used to add fixed kernel vm area to vmlist before * vmalloc_init() is called. @vm->addr, @vm->size, and @vm->flags * should contain proper values and the other fields should be zero. * * DO NOT USE THIS FUNCTION UNLESS YOU KNOW WHAT YOU'RE DOING. */
void __init vm_area_add_early(struct vm_struct *vm) { struct vm_struct *tmp, **p; BUG_ON(vmap_initialized); for (p = &vmlist; (tmp = *p) != NULL; p = &tmp->next) { if (tmp->addr >= vm->addr) { BUG_ON(tmp->addr < vm->addr + vm->size); break; } else BUG_ON(tmp->addr + tmp->size > vm->addr); } vm->next = *p; *p = vm; }

Contributors

PersonTokensPropCommitsCommitProp
nicolas pitrenicolas pitre106100.00%1100.00%
Total106100.00%1100.00%

/** * vm_area_register_early - register vmap area early during boot * @vm: vm_struct to register * @align: requested alignment * * This function is used to register kernel vm area before * vmalloc_init() is called. @vm->size and @vm->flags should contain * proper values on entry and other fields should be zero. On return, * vm->addr contains the allocated address. * * DO NOT USE THIS FUNCTION UNLESS YOU KNOW WHAT YOU'RE DOING. */
void __init vm_area_register_early(struct vm_struct *vm, size_t align) { static size_t vm_init_off __initdata; unsigned long addr; addr = ALIGN(VMALLOC_START + vm_init_off, align); vm_init_off = PFN_ALIGN(addr + vm->size) - VMALLOC_START; vm->addr = (void *)addr; vm_area_add_early(vm); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo5995.16%266.67%
nicolas pitrenicolas pitre34.84%133.33%
Total62100.00%3100.00%


void __init vmalloc_init(void) { struct vmap_area *va; struct vm_struct *tmp; int i; for_each_possible_cpu(i) { struct vmap_block_queue *vbq; struct vfree_deferred *p; vbq = &per_cpu(vmap_block_queue, i); spin_lock_init(&vbq->lock); INIT_LIST_HEAD(&vbq->free); p = &per_cpu(vfree_deferred, i); init_llist_head(&p->list); INIT_WORK(&p->wq, free_work); } /* Import existing vmlist entries. */ for (tmp = vmlist; tmp; tmp = tmp->next) { va = kzalloc(sizeof(struct vmap_area), GFP_NOWAIT); va->flags = VM_VM_AREA; va->va_start = (unsigned long)tmp->addr; va->va_end = va->va_start + tmp->size; va->vm = tmp; __insert_vmap_area(va); } vmap_area_pcpu_hole = VMALLOC_END; vmap_initialized = true; }

Contributors

PersonTokensPropCommitsCommitProp
ivan kokshayskyivan kokshaysky7242.35%112.50%
nick pigginnick piggin4526.47%112.50%
al viroal viro3319.41%112.50%
kyongho chokyongho cho63.53%112.50%
jeremy fitzhardingejeremy fitzhardinge42.35%112.50%
tejun heotejun heo42.35%112.50%
pekka j enbergpekka j enberg31.76%112.50%
pre-gitpre-git31.76%112.50%
Total170100.00%8100.00%

/** * map_kernel_range_noflush - map kernel VM area with the specified pages * @addr: start of the VM area to map * @size: size of the VM area to map * @prot: page protection flags to use * @pages: pages to map * * Map PFN_UP(@size) pages at @addr. The VM area @addr and @size * specify should have been allocated using get_vm_area() and its * friends. * * NOTE: * This function does NOT do any cache flushing. The caller is * responsible for calling flush_cache_vmap() on to-be-mapped areas * before calling this function. * * RETURNS: * The number of pages mapped on success, -errno on failure. */
int map_kernel_range_noflush(unsigned long addr, unsigned long size, pgprot_t prot, struct page **pages) { return vmap_page_range_noflush(addr, addr + size, prot, pages); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo36100.00%1100.00%
Total36100.00%1100.00%

/** * unmap_kernel_range_noflush - unmap kernel VM area * @addr: start of the VM area to unmap * @size: size of the VM area to unmap * * Unmap PFN_UP(@size) pages at @addr. The VM area @addr and @size * specify should have been allocated using get_vm_area() and its * friends. * * NOTE: * This function does NOT do any cache flushing. The caller is * responsible for calling flush_cache_vunmap() on to-be-mapped areas * before calling this function and flush_tlb_kernel_range() after. */
void unmap_kernel_range_noflush(unsigned long addr, unsigned long size) { vunmap_page_range(addr, addr + size); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo22100.00%1100.00%
Total22100.00%1100.00%

EXPORT_SYMBOL_GPL(unmap_kernel_range_noflush); /** * unmap_kernel_range - unmap kernel VM area and flush cache and TLB * @addr: start of the VM area to unmap * @size: size of the VM area to unmap * * Similar to unmap_kernel_range_noflush() but flushes vcache before * the unmapping and tlb after. */
void unmap_kernel_range(unsigned long addr, unsigned long size) { unsigned long end = addr + size; flush_cache_vunmap(addr, end); vunmap_page_range(addr, end); flush_tlb_kernel_range(addr, end); }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin3071.43%125.00%
tejun heotejun heo716.67%125.00%
russell kingrussell king49.52%125.00%
pre-gitpre-git12.38%125.00%
Total42100.00%4100.00%

EXPORT_SYMBOL_GPL(unmap_kernel_range);
int map_vm_area(struct vm_struct *area, pgprot_t prot, struct page **pages) { unsigned long addr = (unsigned long)area->addr; unsigned long end = addr + get_vm_area_size(area); int err; err = vmap_page_range(addr, end, prot, pages); return err > 0 ? 0 : err; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin5277.61%125.00%
james bottomleyjames bottomley913.43%125.00%
wang chaowang chao34.48%125.00%
wanpeng liwanpeng li34.48%125.00%
Total67100.00%4100.00%

EXPORT_SYMBOL_GPL(map_vm_area);
static void setup_vmalloc_vm(struct vm_struct *vm, struct vmap_area *va, unsigned long flags, const void *caller) { spin_lock(&vmap_area_lock); vm->flags = flags; vm->addr = (void *)va->va_start; vm->size = va->va_end - va->va_start; vm->caller = caller; va->vm = vm; va->flags |= VM_VM_AREA; spin_unlock(&vmap_area_lock); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo6981.18%120.00%
joonsoo kimjoonsoo kim1214.12%120.00%
mitsuo hayasakamitsuo hayasaka22.35%120.00%
marek szyprowskimarek szyprowski11.18%120.00%
minchan kimminchan kim11.18%120.00%
Total85100.00%5100.00%


static void clear_vm_uninitialized_flag(struct vm_struct *vm) { /* * Before removing VM_UNINITIALIZED, * we should make sure that vm has proper values. * Pair with smp_rmb() in show_numa_info(). */ smp_wmb(); vm->flags &= ~VM_UNINITIALIZED; }

Contributors

PersonTokensPropCommitsCommitProp
mitsuo hayasakamitsuo hayasaka1568.18%125.00%
zhang yanfeizhang yanfei313.64%125.00%
joonsoo kimjoonsoo kim313.64%125.00%
tejun heotejun heo14.55%125.00%
Total22100.00%4100.00%


static struct vm_struct *__get_vm_area_node(unsigned long size, unsigned long align, unsigned long flags, unsigned long start, unsigned long end, int node, gfp_t gfp_mask, const void *caller) { struct vmap_area *va; struct vm_struct *area; BUG_ON(in_interrupt()); if (flags & VM_IOREMAP) align = 1ul << clamp_t(int, fls_long(size), PAGE_SHIFT, IOREMAP_MAX_ORDER); size = PAGE_ALIGN(size); if (unlikely(!size)) return NULL; area = kzalloc_node(sizeof(*area), gfp_mask & GFP_RECLAIM_MASK, node); if (unlikely(!area)) return NULL; if (!(flags & VM_NO_GUARD)) size += PAGE_SIZE; va = alloc_vmap_area(size, align, start, end, node, gfp_mask); if (IS_ERR(va)) { kfree(area); return NULL; } setup_vmalloc_vm(area, va, flags, caller); return area; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin12265.59%15.88%
mitsuo hayasakamitsuo hayasaka115.91%15.88%
andrey ryabininandrey ryabinin94.84%15.88%
zhang yanfeizhang yanfei94.84%15.88%
james bottomleyjames bottomley73.76%211.76%
pre-gitpre-git63.23%317.65%
toshi kanitoshi kani42.15%15.88%
robert braggrobert bragg42.15%15.88%
christoph hellwigchristoph hellwig42.15%15.88%
david s. millerdavid s. miller42.15%15.88%
linus torvaldslinus torvalds21.08%15.88%
andi kleenandi kleen21.08%15.88%
tejun heotejun heo10.54%15.88%
marek szyprowskimarek szyprowski10.54%15.88%
Total186100.00%17100.00%


struct vm_struct *__get_vm_area(unsigned long size, unsigned long flags, unsigned long start, unsigned long end) { return __get_vm_area_node(size, 1, flags, start, end, NUMA_NO_NODE, GFP_KERNEL, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter4189.13%240.00%
giridhar pemmasanigiridhar pemmasani24.35%120.00%
david s. millerdavid s. miller24.35%120.00%
david rientjesdavid rientjes12.17%120.00%
Total46100.00%5100.00%

EXPORT_SYMBOL_GPL(__get_vm_area);
struct vm_struct *__get_vm_area_caller(unsigned long size, unsigned long flags, unsigned long start, unsigned long end, const void *caller) { return __get_vm_area_node(size, 1, flags, start, end, NUMA_NO_NODE, GFP_KERNEL, caller); }

Contributors

PersonTokensPropCommitsCommitProp
benjamin herrenschmidtbenjamin herrenschmidt4491.67%125.00%
david s. millerdavid s. miller24.17%125.00%
david rientjesdavid rientjes12.08%125.00%
marek szyprowskimarek szyprowski12.08%125.00%
Total48100.00%4100.00%

/** * get_vm_area - reserve a contiguous kernel virtual area * @size: size of the area * @flags: %VM_IOREMAP for I/O mappings or VM_ALLOC * * Search an area of @size in the kernel virtual mapping area, * and reserved it for out purposes. Returns the area descriptor * on success or %NULL on failure. */
struct vm_struct *get_vm_area(unsigned long size, unsigned long flags) { return __get_vm_area_node(size, 1, flags, VMALLOC_START, VMALLOC_END, NUMA_NO_NODE, GFP_KERNEL, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
russell kingrussell king2360.53%125.00%
christoph lameterchristoph lameter1231.58%125.00%
david s. millerdavid s. miller25.26%125.00%
david rientjesdavid rientjes12.63%125.00%
Total38100.00%4100.00%


struct vm_struct *get_vm_area_caller(unsigned long size, unsigned long flags, const void *caller) { return __get_vm_area_node(size, 1, flags, VMALLOC_START, VMALLOC_END, NUMA_NO_NODE, GFP_KERNEL, caller); }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter3382.50%120.00%
russell kingrussell king37.50%120.00%
david s. millerdavid s. miller25.00%120.00%
david rientjesdavid rientjes12.50%120.00%
marek szyprowskimarek szyprowski12.50%120.00%
Total40100.00%5100.00%

/** * find_vm_area - find a continuous kernel virtual area * @addr: base address * * Search for the kernel VM area starting at @addr, and return it. * It is up to the caller to do all required locking to keep the returned * pointer valid. */
struct vm_struct *find_vm_area(const void *addr) { struct vmap_area *va; va = find_vmap_area((unsigned long)addr); if (va && va->flags & VM_VM_AREA) return va->vm; return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin4495.65%250.00%
minchan kimminchan kim12.17%125.00%
christoph lameterchristoph lameter12.17%125.00%
Total46100.00%4100.00%

/** * remove_vm_area - find and remove a continuous kernel virtual area * @addr: base address * * Search for the kernel VM area starting at @addr, and remove it. * This function returns the found VM area, but using it is NOT safe * on SMP machines, except for its size or flags. */
struct vm_struct *remove_vm_area(const void *addr) { struct vmap_area *va; va = find_vmap_area((unsigned long)addr); if (va && va->flags & VM_VM_AREA) { struct vm_struct *vm = va->vm; spin_lock(&vmap_area_lock); va->vm = NULL; va->flags &= ~VM_VM_AREA; spin_unlock(&vmap_area_lock); vmap_debug_free_range(va->va_start, va->va_end); kasan_free_shadow(vm); free_unmap_vmap_area(va); return vm; } return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin3837.62%111.11%
joonsoo kimjoonsoo kim2524.75%111.11%
kamezawa hiroyukikamezawa hiroyuki1615.84%111.11%
pre-gitpre-git98.91%111.11%
andi kleenandi kleen65.94%222.22%
andrey ryabininandrey ryabinin54.95%111.11%
christoph lameterchristoph lameter10.99%111.11%
minchan kimminchan kim10.99%111.11%
Total101100.00%9100.00%


static void __vunmap(const void *addr, int deallocate_pages) { struct vm_struct *area; if (!addr) return; if (WARN(!PAGE_ALIGNED(addr), "Trying to vfree() bad address (%p)\n", addr)) return; area = remove_vm_area(addr); if (unlikely(!area)) { WARN(1, KERN_ERR "Trying to vfree() nonexistent vm area (%p)\n", addr); return; } debug_check_no_locks_freed(addr, get_vm_area_size(area)); debug_check_no_obj_freed(addr, get_vm_area_size(area)); if (deallocate_pages) { int i; for (i = 0; i < area->nr_pages; i++) { struct page *page = area->pages[i]; BUG_ON(!page); __free_pages(page, 0); } kvfree(area->pages); } kfree(area); return; }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton3220.92%15.56%
andi kleenandi kleen3220.92%15.56%
christoph hellwigchristoph hellwig3120.26%15.56%
christoph lameterchristoph lameter127.84%211.11%
pre-gitpre-git106.54%422.22%
thomas gleixnerthomas gleixner74.58%15.56%
ingo molnaringo molnar74.58%15.56%
daisuke hatayamadaisuke hatayama63.92%15.56%
jerome marchandjerome marchand63.92%15.56%
arjan van de venarjan van de ven53.27%15.56%
vladimir davydovvladimir davydov31.96%211.11%
benjamin herrenschmidtbenjamin herrenschmidt10.65%15.56%
david rientjesdavid rientjes10.65%15.56%
Total153100.00%18100.00%

/** * vfree - release memory allocated by vmalloc() * @addr: memory base address * * Free the virtually continuous memory area starting at @addr, as * obtained from vmalloc(), vmalloc_32() or __vmalloc(). If @addr is * NULL, no operation is performed. * * Must not be called in NMI context (strictly speaking, only if we don't * have CONFIG_ARCH_HAVE_NMI_SAFE_CMPXCHG, but making the calling * conventions for vfree() arch-depenedent would be a really bad idea) * * NOTE: assumes that the object at *addr has a size >= sizeof(llist_node) */
void vfree(const void *addr) { BUG_ON(in_nmi()); kmemleak_free(addr); if (!addr) return; if (unlikely(in_interrupt())) { struct vfree_deferred *p = this_cpu_ptr(&vfree_deferred); if (llist_add((struct llist_node *)addr, &p->list)) schedule_work(&p->wq); } else __vunmap(addr, 1); }

Contributors

PersonTokensPropCommitsCommitProp
al viroal viro4960.49%112.50%
christoph hellwigchristoph hellwig1012.35%112.50%
tom rinitom rini67.41%112.50%
catalin marinascatalin marinas56.17%112.50%
andrew mortonandrew morton56.17%112.50%
christoph lameterchristoph lameter33.70%225.00%
oleg nesterovoleg nesterov33.70%112.50%
Total81100.00%8100.00%

EXPORT_SYMBOL(vfree); /** * vunmap - release virtual mapping obtained by vmap() * @addr: memory base address * * Free the virtually contiguous memory area starting at @addr, * which was created from the page array passed to vmap(). * * Must not be called in interrupt context. */
void vunmap(const void *addr) { BUG_ON(in_interrupt()); might_sleep(); if (addr) __vunmap(addr, 0); }

Contributors

PersonTokensPropCommitsCommitProp
tom rinitom rini826.67%116.67%
christoph hellwigchristoph hellwig826.67%116.67%
andrew mortonandrew morton620.00%116.67%
al viroal viro413.33%116.67%
peter zijlstrapeter zijlstra310.00%116.67%
christoph lameterchristoph lameter13.33%116.67%
Total30100.00%6100.00%

EXPORT_SYMBOL(vunmap); /** * vmap - map an array of pages into virtually contiguous space * @pages: array of page pointers * @count: number of pages to map * @flags: vm_area->flags * @prot: page protection for the mapping * * Maps @count pages from @pages into contiguous kernel virtual * space. */
void *vmap(struct page **pages, unsigned int count, unsigned long flags, pgprot_t prot) { struct vm_struct *area; unsigned long size; /* In bytes */ might_sleep(); if (count > totalram_pages) return NULL; size = (unsigned long)count << PAGE_SHIFT; area = get_vm_area_caller(size, flags, __builtin_return_address(0)); if (!area) return NULL; if (map_vm_area(area, prot, pages)) { vunmap(area->addr); return NULL; } return area->addr; }

Contributors

PersonTokensPropCommitsCommitProp
christoph hellwigchristoph hellwig5148.57%112.50%
andrew mortonandrew morton1918.10%225.00%
guillermo julián morenoguillermo julián moreno1514.29%112.50%
tom rinitom rini1110.48%112.50%
christoph lameterchristoph lameter54.76%112.50%
peter zijlstrapeter zijlstra32.86%112.50%
jan beulichjan beulich10.95%112.50%
Total105100.00%8100.00%

EXPORT_SYMBOL(vmap); static void *__vmalloc_node(unsigned long size, unsigned long align, gfp_t gfp_mask, pgprot_t prot, int node, const void *caller);
static void *__vmalloc_area_node(struct vm_struct *area, gfp_t gfp_mask, pgprot_t prot, int node) { const int order = 0; struct page **pages; unsigned int nr_pages, array_size, i; const gfp_t nested_gfp = (gfp_mask & GFP_RECLAIM_MASK) | __GFP_ZERO; const gfp_t alloc_mask = gfp_mask | __GFP_NOWARN; nr_pages = get_vm_area_size(area) >> PAGE_SHIFT; array_size = (nr_pages * sizeof(struct page *)); area->nr_pages = nr_pages; /* Please note that the recursion is strictly bounded. */ if (array_size > PAGE_SIZE) { pages = __vmalloc_node(array_size, 1, nested_gfp|__GFP_HIGHMEM, PAGE_KERNEL, node, area->caller); } else { pages = kmalloc_node(array_size, nested_gfp, node); } area->pages = pages; if (!area->pages) { remove_vm_area(area->addr); kfree(area); return NULL; } for (i = 0; i < area->nr_pages; i++) { struct page *page; if (node == NUMA_NO_NODE) page = alloc_pages(alloc_mask, order); else page = alloc_pages_node(node, alloc_mask, order); if (unlikely(!page)) { /* Successfully allocated i pages, free them in __vunmap() */ area->nr_pages = i; goto fail; } area->pages[i] = page; if (gfpflags_allow_blocking(gfp_mask)) cond_resched(); } if (map_vm_area(area, prot, pages)) goto fail; return area->addr; fail: warn_alloc_failed(gfp_mask, order, "vmalloc: allocation failure, allocated %ld of %ld bytes\n", (area->nr_pages*PAGE_SIZE), area->size); vfree(area->addr); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
andrew mortonandrew morton11437.62%26.90%
christoph lameterchristoph lameter4213.86%413.79%
christoph hellwigchristoph hellwig309.90%13.45%
pre-gitpre-git289.24%413.79%
dave hansendave hansen278.91%13.45%
jan beulichjan beulich144.62%13.45%
david rientjesdavid rientjes113.63%13.45%
oleg nesterovoleg nesterov72.31%26.90%
eric dumazeteric dumazet72.31%13.45%
wanpeng liwanpeng li51.65%26.90%
vladimir davydovvladimir davydov41.32%26.90%
mel gormanmel gorman30.99%13.45%
marcus alanenmarcus alanen20.66%13.45%
david s. millerdavid s. miller20.66%13.45%
jan kiszkajan kiszka20.66%13.45%
jianguo wujianguo wu20.66%13.45%
joe perchesjoe perches10.33%13.45%
al viroal viro10.33%13.45%
adrian bunkadrian bunk10.33%13.45%
Total303100.00%29100.00%

/** * __vmalloc_node_range - allocate virtually contiguous memory * @size: allocation size * @align: desired alignment * @start: vm area range start * @end: vm area range end * @gfp_mask: flags for the page level allocator * @prot: protection mask for the allocated pages * @vm_flags: additional vm area flags (e.g. %VM_NO_GUARD) * @node: node to use for allocation or NUMA_NO_NODE * @caller: caller's return address * * Allocate enough pages to cover @size from the page level * allocator with @gfp_mask flags. Map them into contiguous * kernel virtual space, using a pagetable protection of @prot. */
void *__vmalloc_node_range(unsigned long size, unsigned long align, unsigned long start, unsigned long end, gfp_t gfp_mask, pgprot_t prot, unsigned long vm_flags, int node, const void *caller) { struct vm_struct *area; void *addr; unsigned long real_size = size; size = PAGE_ALIGN(size); if (!size || (size >> PAGE_SHIFT) > totalram_pages) goto fail; area = __get_vm_area_node(size, align, VM_ALLOC | VM_UNINITIALIZED | vm_flags, start, end, node, gfp_mask, caller); if (!area) goto fail; addr = __vmalloc_area_node(area, gfp_mask, prot, node); if (!addr) return NULL; /* * In this function, newly allocated vm_struct has VM_UNINITIALIZED * flag. It means that vm_struct is not fully initialized. * Now, it is fully initialized, so remove this flag here. */ clear_vm_uninitialized_flag(area); /* * A ref_count = 2 is needed because vm_struct allocated in * __get_vm_area_node() contains a reference to the virtual address of * the vmalloc'ed block. */ kmemleak_alloc(addr, real_size, 2, gfp_mask); return addr; fail: warn_alloc_failed(gfp_mask, 0, "vmalloc: allocation failure: %lu bytes\n", real_size); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
oleg nesterovoleg nesterov5633.14%15.88%
catalin marinascatalin marinas2715.98%211.76%
joe perchesjoe perches2213.02%15.88%
christoph lameterchristoph lameter2011.83%211.76%
david rientjesdavid rientjes116.51%15.88%
andrey ryabininandrey ryabinin63.55%15.88%
david s. millerdavid s. miller63.55%15.88%
mitsuo hayasakamitsuo hayasaka52.96%15.88%
mel gormanmel gorman52.96%15.88%
wanpeng liwanpeng li31.78%15.88%
zhang yanfeizhang yanfei31.78%15.88%
giridhar pemmasanigiridhar pemmasani21.18%15.88%
al viroal viro10.59%15.88%
marek szyprowskimarek szyprowski10.59%15.88%
jan beulichjan beulich10.59%15.88%
Total169100.00%17100.00%

/** * __vmalloc_node - allocate virtually contiguous memory * @size: allocation size * @align: desired alignment * @gfp_mask: flags for the page level allocator * @prot: protection mask for the allocated pages * @node: node to use for allocation or NUMA_NO_NODE * @caller: caller's return address * * Allocate enough pages to cover @size from the page level * allocator with @gfp_mask flags. Map them into contiguous * kernel virtual space, using a pagetable protection of @prot. */
static void *__vmalloc_node(unsigned long size, unsigned long align, gfp_t gfp_mask, pgprot_t prot, int node, const void *caller) { return __vmalloc_node_range(size, align, VMALLOC_START, VMALLOC_END, gfp_mask, prot, 0, node, caller); }

Contributors

PersonTokensPropCommitsCommitProp
david rientjesdavid rientjes4894.12%133.33%
andrey ryabininandrey ryabinin23.92%133.33%
marek szyprowskimarek szyprowski11.96%133.33%
Total51100.00%3100.00%


void *__vmalloc(unsigned long size, gfp_t gfp_mask, pgprot_t prot) { return __vmalloc_node(size, 1, gfp_mask, prot, NUMA_NO_NODE, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter2982.86%240.00%
oleg nesterovoleg nesterov38.57%120.00%
david s. millerdavid s. miller25.71%120.00%
david rientjesdavid rientjes12.86%120.00%
Total35100.00%5100.00%

EXPORT_SYMBOL(__vmalloc);
static inline void *__vmalloc_node_flags(unsigned long size, int node, gfp_t flags) { return __vmalloc_node(size, 1, flags, PAGE_KERNEL, node, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
dave youngdave young37100.00%1100.00%
Total37100.00%1100.00%

/** * vmalloc - allocate virtually contiguous memory * @size: allocation size * Allocate enough pages to cover @size from the page level * allocator and map them into contiguous kernel virtual space. * * For tight control over page level allocator and protection flags * use __vmalloc() instead. */
void *vmalloc(unsigned long size) { return __vmalloc_node_flags(size, NUMA_NO_NODE, GFP_KERNEL | __GFP_HIGHMEM); }

Contributors

PersonTokensPropCommitsCommitProp
christoph hellwigchristoph hellwig1672.73%125.00%
dave youngdave young418.18%125.00%
david rientjesdavid rientjes14.55%125.00%
david s. millerdavid s. miller14.55%125.00%
Total22100.00%4100.00%

EXPORT_SYMBOL(vmalloc); /** * vzalloc - allocate virtually contiguous memory with zero fill * @size: allocation size * Allocate enough pages to cover @size from the page level * allocator and map them into contiguous kernel virtual space. * The memory allocated is set to zero. * * For tight control over page level allocator and protection flags * use __vmalloc() instead. */
void *vzalloc(unsigned long size) { return __vmalloc_node_flags(size, NUMA_NO_NODE, GFP_KERNEL | __GFP_HIGHMEM | __GFP_ZERO); }

Contributors

PersonTokensPropCommitsCommitProp
dave youngdave young1875.00%125.00%
christoph hellwigchristoph hellwig312.50%125.00%
christoph lameterchristoph lameter28.33%125.00%
david rientjesdavid rientjes14.17%125.00%
Total24100.00%4100.00%

EXPORT_SYMBOL(vzalloc); /** * vmalloc_user - allocate zeroed virtually contiguous memory for userspace * @size: allocation size * * The resulting memory area is zeroed so it can be mapped to userspace * without leaking data. */
void *vmalloc_user(unsigned long size) { struct vm_struct *area; void *ret; ret = __vmalloc_node(size, SHMLBA, GFP_KERNEL | __GFP_HIGHMEM | __GFP_ZERO, PAGE_KERNEL, NUMA_NO_NODE, __builtin_return_address(0)); if (ret) { area = find_vm_area(ret); area->flags |= VM_USERMAP; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin4975.38%233.33%
glauber costaglauber costa710.77%116.67%
eric dumazeteric dumazet69.23%116.67%
david s. millerdavid s. miller23.08%116.67%
david rientjesdavid rientjes11.54%116.67%
Total65100.00%6100.00%

EXPORT_SYMBOL(vmalloc_user); /** * vmalloc_node - allocate memory on a specific node * @size: allocation size * @node: numa node * * Allocate enough pages to cover @size from the page level * allocator and map them into contiguous kernel virtual space. * * For tight control over page level allocator and protection flags * use __vmalloc() instead. */
void *vmalloc_node(unsigned long size, int node) { return __vmalloc_node(size, 1, GFP_KERNEL | __GFP_HIGHMEM, PAGE_KERNEL, node, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter3294.12%266.67%
david s. millerdavid s. miller25.88%133.33%
Total34100.00%3100.00%

EXPORT_SYMBOL(vmalloc_node); /** * vzalloc_node - allocate memory on a specific node with zero fill * @size: allocation size * @node: numa node * * Allocate enough pages to cover @size from the page level * allocator and map them into contiguous kernel virtual space. * The memory allocated is set to zero. * * For tight control over page level allocator and protection flags * use __vmalloc_node() instead. */
void *vzalloc_node(unsigned long size, int node) { return __vmalloc_node_flags(size, node, GFP_KERNEL | __GFP_HIGHMEM | __GFP_ZERO); }

Contributors

PersonTokensPropCommitsCommitProp
dave youngdave young27100.00%1100.00%
Total27100.00%1100.00%

EXPORT_SYMBOL(vzalloc_node); #ifndef PAGE_KERNEL_EXEC # define PAGE_KERNEL_EXEC PAGE_KERNEL #endif /** * vmalloc_exec - allocate virtually contiguous, executable memory * @size: allocation size * * Kernel-internal function to allocate enough pages to cover @size * the page level allocator and map them into contiguous and * executable kernel virtual space. * * For tight control over page level allocator and protection flags * use __vmalloc() instead. */
void *vmalloc_exec(unsigned long size) { return __vmalloc_node(size, 1, GFP_KERNEL | __GFP_HIGHMEM, PAGE_KERNEL_EXEC, NUMA_NO_NODE, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
ingo molnaringo molnar2167.74%125.00%
glauber costaglauber costa722.58%125.00%
david s. millerdavid s. miller26.45%125.00%
david rientjesdavid rientjes13.23%125.00%
Total31100.00%4100.00%

#if defined(CONFIG_64BIT) && defined(CONFIG_ZONE_DMA32) #define GFP_VMALLOC32 GFP_DMA32 | GFP_KERNEL #elif defined(CONFIG_64BIT) && defined(CONFIG_ZONE_DMA) #define GFP_VMALLOC32 GFP_DMA | GFP_KERNEL #else #define GFP_VMALLOC32 GFP_KERNEL #endif /** * vmalloc_32 - allocate virtually contiguous memory (32bit addressable) * @size: allocation size * * Allocate enough 32bit PA addressable pages to cover @size from the * page level allocator and map them into contiguous kernel virtual space. */
void *vmalloc_32(unsigned long size) { return __vmalloc_node(size, 1, GFP_VMALLOC32, PAGE_KERNEL, NUMA_NO_NODE, __builtin_return_address(0)); }

Contributors

PersonTokensPropCommitsCommitProp
christoph hellwigchristoph hellwig1551.72%114.29%
glauber costaglauber costa724.14%114.29%
pre-gitpre-git310.34%228.57%
david s. millerdavid s. miller26.90%114.29%
david rientjesdavid rientjes13.45%114.29%
andi kleenandi kleen13.45%114.29%
Total29100.00%7100.00%

EXPORT_SYMBOL(vmalloc_32); /** * vmalloc_32_user - allocate zeroed virtually contiguous 32bit memory * @size: allocation size * * The resulting memory area is 32bit addressable and zeroed so it can be * mapped to userspace without leaking data. */
void *vmalloc_32_user(unsigned long size) { struct vm_struct *area; void *ret; ret = __vmalloc_node(size, 1, GFP_VMALLOC32 | __GFP_ZERO, PAGE_KERNEL, NUMA_NO_NODE, __builtin_return_address(0)); if (ret) { area = find_vm_area(ret); area->flags |= VM_USERMAP; } return ret; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin4673.02%228.57%
glauber costaglauber costa711.11%114.29%
eric dumazeteric dumazet69.52%114.29%
david s. millerdavid s. miller23.17%114.29%
andi kleenandi kleen11.59%114.29%
david rientjesdavid rientjes11.59%114.29%
Total63100.00%7100.00%

EXPORT_SYMBOL(vmalloc_32_user); /* * small helper routine , copy contents to buf from addr. * If the page is not present, fill zero. */
static int aligned_vread(char *buf, char *addr, unsigned long count) { struct page *p; int copied = 0; while (count) { unsigned long offset, length; offset = offset_in_page(addr); length = PAGE_SIZE - offset; if (length > count) length = count; p = vmalloc_to_page(addr); /* * To do safe access to this _mapped_ area, we need * lock. But adding lock here means that we need to add * overhead of vmalloc()/vfree() calles for this _debug_ * interface, rarely used. Instead of that, we'll use * kmap() and get small overhead in this access function. */ if (p) { /* * we can expect USER0 is not used (see vread/vwrite's * function description) */ void *map = kmap_atomic(p); memcpy(buf, map + offset, length); kunmap_atomic(map); } else memset(buf, 0, length); addr += length; buf += length; copied += length; count -= length; } return copied; }

Contributors

PersonTokensPropCommitsCommitProp
kamezawa hiroyukikamezawa hiroyuki12997.73%150.00%
alexander kuleshovalexander kuleshov32.27%150.00%
Total132100.00%2100.00%


static int aligned_vwrite(char *buf, char *addr, unsigned long count) { struct page *p; int copied = 0; while (count) { unsigned long offset, length; offset = offset_in_page(addr); length = PAGE_SIZE - offset; if (length > count) length = count; p = vmalloc_to_page(addr); /* * To do safe access to this _mapped_ area, we need * lock. But adding lock here means that we need to add * overhead of vmalloc()/vfree() calles for this _debug_ * interface, rarely used. Instead of that, we'll use * kmap() and get small overhead in this access function. */ if (p) { /* * we can expect USER0 is not used (see vread/vwrite's * function description) */ void *map = kmap_atomic(p); memcpy(map + offset, buf, length); kunmap_atomic(map); } addr += length; buf += length; copied += length; count -= length; } return copied; }

Contributors

PersonTokensPropCommitsCommitProp
kamezawa hiroyukikamezawa hiroyuki11997.54%150.00%
alexander kuleshovalexander kuleshov32.46%150.00%
Total122100.00%2100.00%

/** * vread() - read vmalloc area in a safe way. * @buf: buffer for reading data * @addr: vm address. * @count: number of bytes to be read. * * Returns # of bytes which addr and buf should be increased. * (same number to @count). Returns 0 if [addr...addr+count) doesn't * includes any intersect with alive vmalloc area. * * This function checks that addr is a valid vmalloc'ed area, and * copy data from that area to a given buffer. If the given memory range * of [addr...addr+count) includes some valid address, data is copied to * proper area of @buf. If there are memory holes, they'll be zero-filled. * IOREMAP area is treated as memory hole and no copy is done. * * If [addr...addr+count) doesn't includes any intersects with alive * vm_struct area, returns 0. @buf should be kernel's buffer. * * Note: In usual ops, vread() is never necessary because the caller * should know vmalloc() area is valid and can use memcpy(). * This is for routines which have to access vmalloc area without * any informaion, as /dev/kmem. * */
long vread(char *buf, char *addr, unsigned long count) { struct vmap_area *va; struct vm_struct *vm; char *vaddr, *buf_start = buf; unsigned long buflen = count; unsigned long n; /* Don't allow overflow */ if ((unsigned long) addr + count < count) count = -(unsigned long) addr; spin_lock(&vmap_area_lock); list_for_each_entry(va, &vmap_area_list, list) { if (!count) break; if (!(va->flags & VM_VM_AREA)) continue; vm = va->vm; vaddr = (char *) vm->addr; if (addr >= vaddr + get_vm_area_size(vm)) continue; while (addr < vaddr) { if (count == 0) goto finished; *buf = '\0'; buf++; addr++; count--; } n = vaddr + get_vm_area_size(vm) - addr; if (n > count) n = count; if (!(vm->flags & VM_IOREMAP)) aligned_vread(buf, addr, n); else /* IOREMAP area is treated as memory hole */ memset(buf, 0, n); buf += n; addr += n; count -= n; } finished: spin_unlock(&vmap_area_lock); if (buf == buf_start) return 0; /* zero-fill memory holes */ if (buf != buf_start + buflen) memset(buf, 0, buflen - (buf - buf_start)); return buflen; }

Contributors

PersonTokensPropCommitsCommitProp
pre-gitpre-git14252.79%666.67%
kamezawa hiroyukikamezawa hiroyuki7728.62%111.11%
joonsoo kimjoonsoo kim4416.36%111.11%
wanpeng liwanpeng li62.23%111.11%
Total269100.00%9100.00%

/** * vwrite() - write vmalloc area in a safe way. * @buf: buffer for source data * @addr: vm address. * @count: number of bytes to be read. * * Returns # of bytes which addr and buf should be incresed. * (same number to @count). * If [addr...addr+count) doesn't includes any intersect with valid * vmalloc area, returns 0. * * This function checks that addr is a valid vmalloc'ed area, and * copy data from a buffer to the given addr. If specified range of * [addr...addr+count) includes some valid address, data is copied from * proper area of @buf. If there are memory holes, no copy to hole. * IOREMAP area is treated as memory hole and no copy is done. * * If [addr...addr+count) doesn't includes any intersects with alive * vm_struct area, returns 0. @buf should be kernel's buffer. * * Note: In usual ops, vwrite() is never necessary because the caller * should know vmalloc() area is valid and can use memcpy(). * This is for routines which have to access vmalloc area without * any informaion, as /dev/kmem. */
long vwrite(char *buf, char *addr, unsigned long count) { struct vmap_area *va; struct vm_struct *vm; char *vaddr; unsigned long n, buflen; int copied = 0; /* Don't allow overflow */ if ((unsigned long) addr + count < count) count = -(unsigned long) addr; buflen = count; spin_lock(&vmap_area_lock); list_for_each_entry(va, &vmap_area_list, list) { if (!count) break; if (!(va->flags & VM_VM_AREA)) continue; vm = va->vm; vaddr = (char *) vm->addr; if (addr >= vaddr + get_vm_area_size(vm)) continue; while (addr < vaddr) { if (count == 0) goto finished; buf++; addr++; count--; } n = vaddr + get_vm_area_size(vm) - addr; if (n > count) n = count; if (!(vm->flags & VM_IOREMAP)) { aligned_vwrite(buf, addr, n); copied++; } buf += n; addr += n; count -= n; } finished: spin_unlock(&vmap_area_lock); if (!copied) return 0; return buflen; }

Contributors

PersonTokensPropCommitsCommitProp
dave jonesdave jones12955.36%125.00%
kamezawa hiroyukikamezawa hiroyuki5423.18%125.00%
joonsoo kimjoonsoo kim4418.88%125.00%
wanpeng liwanpeng li62.58%125.00%
Total233100.00%4100.00%

/** * remap_vmalloc_range_partial - map vmalloc pages to userspace * @vma: vma to cover * @uaddr: target user address to start at * @kaddr: virtual address of vmalloc kernel memory * @size: size of map area * * Returns: 0 for success, -Exxx on failure * * This function checks that @kaddr is a valid vmalloc'ed area, * and that it is big enough to cover the range starting at * @uaddr in @vma. Will return failure if that criteria isn't * met. * * Similar to remap_pfn_range() (see mm/memory.c) */
int remap_vmalloc_range_partial(struct vm_area_struct *vma, unsigned long uaddr, void *kaddr, unsigned long size) { struct vm_struct *area; size = PAGE_ALIGN(size); if (!PAGE_ALIGNED(uaddr) || !PAGE_ALIGNED(kaddr)) return -EINVAL; area = find_vm_area(kaddr); if (!area) return -EINVAL; if (!(area->flags & VM_USERMAP)) return -EINVAL; if (kaddr + size > area->addr + area->size) return -EINVAL; do { struct page *page = vmalloc_to_page(kaddr); int ret; ret = vm_insert_page(vma, uaddr, page); if (ret) return ret; uaddr += PAGE_SIZE; kaddr += PAGE_SIZE; size -= PAGE_SIZE; } while (size > 0); vma->vm_flags |= VM_DONTEXPAND | VM_DONTDUMP; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin12676.36%250.00%
daisuke hatayamadaisuke hatayama3621.82%125.00%
konstantin khlebnikovkonstantin khlebnikov31.82%125.00%
Total165100.00%4100.00%

EXPORT_SYMBOL(remap_vmalloc_range_partial); /** * remap_vmalloc_range - map vmalloc pages to userspace * @vma: vma to cover (map full range of vma) * @addr: vmalloc memory * @pgoff: number of pages into addr before first page to map * * Returns: 0 for success, -Exxx on failure * * This function checks that addr is a valid vmalloc'ed area, and * that it is big enough to cover the vma. Will return failure if * that criteria isn't met. * * Similar to remap_pfn_range() (see mm/memory.c) */
int remap_vmalloc_range(struct vm_area_struct *vma, void *addr, unsigned long pgoff) { return remap_vmalloc_range_partial(vma, vma->vm_start, addr + (pgoff << PAGE_SHIFT), vma->vm_end - vma->vm_start); }

Contributors

PersonTokensPropCommitsCommitProp
daisuke hatayamadaisuke hatayama44100.00%1100.00%
Total44100.00%1100.00%

EXPORT_SYMBOL(remap_vmalloc_range); /* * Implement a stub for vmalloc_sync_all() if the architecture chose not to * have one. */
void __weak vmalloc_sync_all(void) { }

Contributors

PersonTokensPropCommitsCommitProp
christoph hellwigchristoph hellwig457.14%150.00%
gideon israel dsouzagideon israel dsouza342.86%150.00%
Total7100.00%2100.00%


static int f(pte_t *pte, pgtable_t table, unsigned long addr, void *data) { pte_t ***p = data; if (p) { *(*p) = pte; (*p)++; } return 0; }

Contributors

PersonTokensPropCommitsCommitProp
david vrabeldavid vrabel2853.85%133.33%
jeremy fitzhardingejeremy fitzhardinge2242.31%133.33%
martin schwidefskymartin schwidefsky23.85%133.33%
Total52100.00%3100.00%

/** * alloc_vm_area - allocate a range of kernel address space * @size: size of the area * @ptes: returns the PTEs for the address space * * Returns: NULL on failure, vm_struct on success * * This function reserves a range of kernel address space, and * allocates pagetables to map that range. No actual mappings * are created. * * If @ptes is non-NULL, pointers to the PTEs (in init_mm) * allocated for the VM area are returned. */
struct vm_struct *alloc_vm_area(size_t size, pte_t **ptes) { struct vm_struct *area; area = get_vm_area_caller(size, VM_IOREMAP, __builtin_return_address(0)); if (area == NULL) return NULL; /* * This ensures that page tables are constructed for this region * of kernel virtual address space and mapped into init_mm. */ if (apply_to_page_range(&init_mm, (unsigned long)area->addr, size, f, ptes ? &ptes : NULL)) { free_vm_area(area); return NULL; } return area; }

Contributors

PersonTokensPropCommitsCommitProp
jeremy fitzhardingejeremy fitzhardinge6880.95%133.33%
david vrabeldavid vrabel1011.90%133.33%
christoph lameterchristoph lameter67.14%133.33%
Total84100.00%3100.00%

EXPORT_SYMBOL_GPL(alloc_vm_area);
void free_vm_area(struct vm_struct *area) { struct vm_struct *ret; ret = remove_vm_area(area->addr); BUG_ON(ret != area); kfree(area); }

Contributors

PersonTokensPropCommitsCommitProp
jeremy fitzhardingejeremy fitzhardinge36100.00%1100.00%
Total36100.00%1100.00%

EXPORT_SYMBOL_GPL(free_vm_area); #ifdef CONFIG_SMP
static struct vmap_area *node_to_va(struct rb_node *n) { return n ? rb_entry(n, struct vmap_area, rb_node) : NULL; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo28100.00%1100.00%
Total28100.00%1100.00%

/** * pvm_find_next_prev - find the next and prev vmap_area surrounding @end * @end: target address * @pnext: out arg for the next vmap_area * @pprev: out arg for the previous vmap_area * * Returns: %true if either or both of next and prev are found, * %false if no vmap_area exists * * Find vmap_areas end addresses of which enclose @end. ie. if not * NULL, *pnext->va_end > @end and *pprev->va_end <= @end. */
static bool pvm_find_next_prev(unsigned long end, struct vmap_area **pnext, struct vmap_area **pprev) { struct rb_node *n = vmap_area_root.rb_node; struct vmap_area *va = NULL; while (n) { va = rb_entry(n, struct vmap_area, rb_node); if (end < va->va_end) n = n->rb_left; else if (end > va->va_end) n = n->rb_right; else break; } if (!va) return false; if (va->va_end > end) { *pnext = va; *pprev = node_to_va(rb_prev(&(*pnext)->rb_node)); } else { *pprev = va; *pnext = node_to_va(rb_next(&(*pprev)->rb_node)); } return true; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo155100.00%1100.00%
Total155100.00%1100.00%

/** * pvm_determine_end - find the highest aligned address between two vmap_areas * @pnext: in/out arg for the next vmap_area * @pprev: in/out arg for the previous vmap_area * @align: alignment * * Returns: determined end address * * Find the highest aligned address between *@pnext and *@pprev below * VMALLOC_END. *@pnext and *@pprev are adjusted so that the aligned * down address is between the end addresses of the two vmap_areas. * * Please note that the address returned by this function may fall * inside *@pnext vmap_area. The caller is responsible for checking * that. */
static unsigned long pvm_determine_end(struct vmap_area **pnext, struct vmap_area **pprev, unsigned long align) { const unsigned long vmalloc_end = VMALLOC_END & ~(align - 1); unsigned long addr; if (*pnext) addr = min((*pnext)->va_start & ~(align - 1), vmalloc_end); else addr = vmalloc_end; while (*pprev && (*pprev)->va_end > addr) { *pnext = *pprev; *pprev = node_to_va(rb_prev(&(*pnext)->rb_node)); } return addr; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo114100.00%1100.00%
Total114100.00%1100.00%

/** * pcpu_get_vm_areas - allocate vmalloc areas for percpu allocator * @offsets: array containing offset of each area * @sizes: array containing size of each area * @nr_vms: the number of areas to allocate * @align: alignment, all entries in @offsets and @sizes must be aligned to this * * Returns: kmalloc'd vm_struct pointer array pointing to allocated * vm_structs on success, %NULL on failure * * Percpu allocator wants to use congruent vm areas so that it can * maintain the offsets among percpu areas. This function allocates * congruent vmalloc areas for it with GFP_KERNEL. These areas tend to * be scattered pretty far, distance between two areas easily going up * to gigabytes. To avoid interacting with regular vmallocs, these * areas are allocated from top. * * Despite its complicated look, this allocator is rather simple. It * does everything top-down and scans areas from the end looking for * matching slot. While scanning, if any of the areas overlaps with * existing vmap_area, the base address is pulled down to fit the * area. Scanning is repeated till all the areas fit and then all * necessary data structres are inserted and the result is returned. */
struct vm_struct **pcpu_get_vm_areas(const unsigned long *offsets, const size_t *sizes, int nr_vms, size_t align) { const unsigned long vmalloc_start = ALIGN(VMALLOC_START, align); const unsigned long vmalloc_end = VMALLOC_END & ~(align - 1); struct vmap_area **vas, *prev, *next; struct vm_struct **vms; int area, area2, last_area, term_area; unsigned long base, start, end, last_end; bool purged = false; /* verify parameters and allocate data structures */ BUG_ON(offset_in_page(align) || !is_power_of_2(align)); for (last_area = 0, area = 0; area < nr_vms; area++) { start = offsets[area]; end = start + sizes[area]; /* is everything aligned properly? */ BUG_ON(!IS_ALIGNED(offsets[area], align)); BUG_ON(!IS_ALIGNED(sizes[area], align)); /* detect the area with the highest address */ if (start > offsets[last_area]) last_area = area; for (area2 = 0; area2 < nr_vms; area2++) { unsigned long start2 = offsets[area2]; unsigned long end2 = start2 + sizes[area2]; if (area2 == area) continue; BUG_ON(start2 >= start && start2 < end); BUG_ON(end2 <= end && end2 > start); } } last_end = offsets[last_area] + sizes[last_area]; if (vmalloc_end - vmalloc_start < last_end) { WARN_ON(true); return NULL; } vms = kcalloc(nr_vms, sizeof(vms[0]), GFP_KERNEL); vas = kcalloc(nr_vms, sizeof(vas[0]), GFP_KERNEL); if (!vas || !vms) goto err_free2; for (area = 0; area < nr_vms; area++) { vas[area] = kzalloc(sizeof(struct vmap_area), GFP_KERNEL); vms[area] = kzalloc(sizeof(struct vm_struct), GFP_KERNEL); if (!vas[area] || !vms[area]) goto err_free; } retry: spin_lock(&vmap_area_lock); /* start scanning - we scan from the top, begin with the last area */ area = term_area = last_area; start = offsets[area]; end = start + sizes[area]; if (!pvm_find_next_prev(vmap_area_pcpu_hole, &next, &prev)) { base = vmalloc_end - last_end; goto found; } base = pvm_determine_end(&next, &prev, align) - end; while (true) { BUG_ON(next && next->va_end <= base + end); BUG_ON(prev && prev->va_end > base + end); /* * base might have underflowed, add last_end before * comparing. */ if (base + last_end < vmalloc_start + last_end) { spin_unlock(&vmap_area_lock); if (!purged) { purge_vmap_area_lazy(); purged = true; goto retry; } goto err_free; } /* * If next overlaps, move base downwards so that it's * right below next and then recheck. */ if (next && next->va_start < base + end) { base = pvm_determine_end(&next, &prev, align) - end; term_area = area; continue; } /* * If prev overlaps, shift down next and prev and move * base so that it's right below new next and then * recheck. */ if (prev && prev->va_end > base + start) { next = prev; prev = node_to_va(rb_prev(&next->rb_node)); base = pvm_determine_end(&next, &prev, align) - end; term_area = area; continue; } /* * This area fits, move on to the previous one. If * the previous one is the terminal one, we're done. */ area = (area + nr_vms - 1) % nr_vms; if (area == term_area) break; start = offsets[area]; end = start + sizes[area]; pvm_find_next_prev(base + end, &next, &prev); } found: /* we've found a fitting base, insert all va's */ for (area = 0; area < nr_vms; area++) { struct vmap_area *va = vas[area]; va->va_start = base + offsets[area]; va->va_end = va->va_start + sizes[area]; __insert_vmap_area(va); } vmap_area_pcpu_hole = base + offsets[last_area]; spin_unlock(&vmap_area_lock); /* insert all vm's */ for (area = 0; area < nr_vms; area++) setup_vmalloc_vm(vms[area], vas[area], VM_ALLOC, pcpu_get_vm_areas); kfree(vas); return vms; err_free: for (area = 0; area < nr_vms; area++) { kfree(vas[area]); kfree(vms[area]); } err_free2: kfree(vas); kfree(vms); return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo80997.94%116.67%
thomas meyerthomas meyer60.73%116.67%
david rientjesdavid rientjes40.48%116.67%
kautuk consulkautuk consul30.36%116.67%
alexander kuleshovalexander kuleshov30.36%116.67%
zhang yanfeizhang yanfei10.12%116.67%
Total826100.00%6100.00%

/** * pcpu_free_vm_areas - free vmalloc areas for percpu allocator * @vms: vm_struct pointer array returned by pcpu_get_vm_areas() * @nr_vms: the number of allocated areas * * Free vm_structs and the array allocated by pcpu_get_vm_areas(). */
void pcpu_free_vm_areas(struct vm_struct **vms, int nr_vms) { int i; for (i = 0; i < nr_vms; i++) free_vm_area(vms[i]); kfree(vms); }

Contributors

PersonTokensPropCommitsCommitProp
tejun heotejun heo43100.00%1100.00%
Total43100.00%1100.00%

#endif /* CONFIG_SMP */ #ifdef CONFIG_PROC_FS
static void *s_start(struct seq_file *m, loff_t *pos) __acquires(&vmap_area_lock

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter1482.35%133.33%
namhyung kimnamhyung kim211.76%133.33%
joonsoo kimjoonsoo kim15.88%133.33%
Total17100.00%3100.00%

) { loff_t n = *pos; struct vmap_area *va; spin_lock(&vmap_area_lock); va = list_first_entry(&vmap_area_list, typeof(*va), list); while (n > 0 && &va->list != &vmap_area_list) { n--; va = list_next_entry(va, list); } if (!n && &va->list != &vmap_area_list) return va; return NULL; }
static void *s_next(struct seq_file *m, void *p, loff_t *pos) { struct vmap_area *va = p, *next; ++*pos; next = list_next_entry(va, list); if (&next->list != &vmap_area_list) return next; return NULL; }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter3254.24%133.33%
joonsoo kimjoonsoo kim2644.07%133.33%
geliang tanggeliang tang11.69%133.33%
Total59100.00%3100.00%


static void s_stop(struct seq_file *m, void *p) __releases(&vmap_area_lock

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter1381.25%133.33%
namhyung kimnamhyung kim212.50%133.33%
joonsoo kimjoonsoo kim16.25%133.33%
Total16100.00%3100.00%

) { spin_unlock(&vmap_area_lock); }
static void show_numa_info(struct seq_file *m, struct vm_struct *v) { if (IS_ENABLED(CONFIG_NUMA)) { unsigned int nr, *counters = m->private; if (!counters) return; if (v->flags & VM_UNINITIALIZED) return; /* Pair with smp_wmb() in clear_vm_uninitialized_flag() */ smp_rmb(); memset(counters, 0, nr_node_ids * sizeof(unsigned int)); for (nr = 0; nr < v->nr_pages; nr++) counters[page_to_nid(v->pages[nr])]++; for_each_node_state(nr, N_HIGH_MEMORY) if (counters[nr]) seq_printf(m, " N%u=%u", nr, counters[nr]); } }

Contributors

PersonTokensPropCommitsCommitProp
eric dumazeteric dumazet10986.51%125.00%
wanpeng liwanpeng li97.14%125.00%
dmitriy vyukovdmitriy vyukov43.17%125.00%
kirill a. shutemovkirill a. shutemov43.17%125.00%
Total126100.00%4100.00%


static int s_show(struct seq_file *m, void *p) { struct vmap_area *va = p; struct vm_struct *v; /* * s_show can encounter race with remove_vm_area, !VM_VM_AREA on * behalf of vmap area is being tear down or vm_map_ram allocation. */ if (!(va->flags & VM_VM_AREA)) return 0; v = va->vm; seq_printf(m, "0x%pK-0x%pK %7ld", v->addr, v->addr + v->size, v->size); if (v->caller) seq_printf(m, " %pS", v->caller); if (v->nr_pages) seq_printf(m, " pages=%d", v->nr_pages); if (v->phys_addr) seq_printf(m, " phys=%llx", (unsigned long long)v->phys_addr); if (v->flags & VM_IOREMAP) seq_puts(m, " ioremap"); if (v->flags & VM_ALLOC) seq_puts(m, " vmalloc"); if (v->flags & VM_MAP) seq_puts(m, " vmap"); if (v->flags & VM_USERMAP) seq_puts(m, " user"); if (is_vmalloc_addr(v->pages)) seq_puts(m, " vpages"); show_numa_info(m, v); seq_putc(m, '\n'); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
christoph lameterchristoph lameter16775.91%220.00%
joonsoo kimjoonsoo kim2712.27%110.00%
eric dumazeteric dumazet73.18%110.00%
kenji kaneshigekenji kaneshige62.73%110.00%
fabian frederickfabian frederick52.27%110.00%
david rientjesdavid rientjes41.82%110.00%
joe perchesjoe perches20.91%110.00%
kees cookkees cook10.45%110.00%
wanpeng liwanpeng li10.45%110.00%
Total220100.00%10100.00%

static const struct seq_operations vmalloc_op = { .start = s_start, .next = s_next, .stop = s_stop, .show = s_show, };
static int vmalloc_open(struct inode *inode, struct file *file) { if (IS_ENABLED(CONFIG_NUMA)) return seq_open_private(file, &vmalloc_op, nr_node_ids * sizeof(unsigned int)); else return seq_open(file, &vmalloc_op); }

Contributors

PersonTokensPropCommitsCommitProp
alexey dobriyanalexey dobriyan3672.00%125.00%
rob jonesrob jones918.00%125.00%
kirill a. shutemovkirill a. shutemov48.00%125.00%
vasiliy kulikovvasiliy kulikov12.00%125.00%
Total50100.00%4100.00%

static const struct file_operations proc_vmalloc_operations = { .open = vmalloc_open, .read = seq_read, .llseek = seq_lseek, .release = seq_release_private, };
static int __init proc_vmalloc_init(void) { proc_create("vmallocinfo", S_IRUSR, NULL, &proc_vmalloc_operations); return 0; }

Contributors

PersonTokensPropCommitsCommitProp
alexey dobriyanalexey dobriyan24100.00%1100.00%
Total24100.00%1100.00%

module_init(proc_vmalloc_init); #endif

Overall Contributors

PersonTokensPropCommitsCommitProp
nick pigginnick piggin354634.36%94.21%
tejun heotejun heo144714.02%83.74%
christoph lameterchristoph lameter6906.69%83.74%
kamezawa hiroyukikamezawa hiroyuki3963.84%20.93%
pre-gitpre-git3913.79%2310.75%
hugh dickinshugh dickins3273.17%20.93%
joonsoo kimjoonsoo kim2422.34%73.27%
christoph hellwigchristoph hellwig2192.12%20.93%
andi kleenandi kleen2162.09%41.87%
andrew mortonandrew morton2152.08%73.27%
roman peniaevroman peniaev2051.99%31.40%
jeremy fitzhardingejeremy fitzhardinge1981.92%31.40%
al viroal viro1891.83%20.93%
chris wilsonchris wilson1471.42%20.93%
eric dumazeteric dumazet1351.31%31.40%
dave jonesdave jones1311.27%10.47%
nicolas pitrenicolas pitre1101.07%10.47%
alexey dobriyanalexey dobriyan1000.97%20.93%
dave youngdave young990.96%10.47%
daisuke hatayamadaisuke hatayama950.92%20.93%
david rientjesdavid rientjes920.89%52.34%
ivan kokshayskyivan kokshaysky720.70%10.47%
oleg nesterovoleg nesterov710.69%31.40%
linus torvaldslinus torvalds550.53%20.93%
benjamin herrenschmidtbenjamin herrenschmidt530.51%41.87%
catalin marinascatalin marinas500.48%20.93%
david s. millerdavid s. miller440.43%20.93%
ingo molnaringo molnar400.39%41.87%
james bottomleyjames bottomley390.38%20.93%
david vrabeldavid vrabel390.38%10.47%
glauber costaglauber costa380.37%31.40%
mitsuo hayasakamitsuo hayasaka330.32%10.47%
wanpeng liwanpeng li330.32%52.34%
arnaldo carvalho de meloarnaldo carvalho de melo320.31%10.47%
russell kingrussell king310.30%20.93%
dave hansendave hansen270.26%10.47%
joe perchesjoe perches260.25%41.87%
tom rinitom rini250.24%10.47%
andrey ryabininandrey ryabinin230.22%31.40%
toshi kanitoshi kani230.22%20.93%
guillermo julián morenoguillermo julián moreno230.22%10.47%
vassili karpovvassili karpov210.20%10.47%
cliff wickmancliff wickman190.18%10.47%
alexander kuleshovalexander kuleshov180.17%10.47%
zhang yanfeizhang yanfei170.16%41.87%
jan beulichjan beulich160.16%20.93%
jan karajan kara150.15%10.47%
hong zhi guohong zhi guo120.12%10.47%
mel gormanmel gorman110.11%31.40%
namhyung kimnamhyung kim100.10%20.93%
thomas gleixnerthomas gleixner100.10%10.47%
rob jonesrob jones90.09%10.47%
marek szyprowskimarek szyprowski80.08%20.93%
minchan kimminchan kim80.08%20.93%
kirill a. shutemovkirill a. shutemov80.08%10.47%
jiri slabyjiri slaby80.08%10.47%
wang xiaoqiangwang xiaoqiang80.08%10.47%
vegard nossumvegard nossum70.07%10.47%
vladimir davydovvladimir davydov70.07%20.93%
adam lackorzynskiadam lackorzynski70.07%10.47%
kyongho chokyongho cho60.06%10.47%
jerome marchandjerome marchand60.06%10.47%
thomas meyerthomas meyer60.06%10.47%
peter zijlstrapeter zijlstra60.06%10.47%
gideon israel dsouzagideon israel dsouza60.06%10.47%
kenji kaneshigekenji kaneshige60.06%10.47%
rusty russellrusty russell50.05%10.47%
arjan van de venarjan van de ven50.05%10.47%
marcus alanenmarcus alanen50.05%10.47%
huang yinghuang ying50.05%10.47%
rolf eike beerrolf eike beer50.05%10.47%
fabian frederickfabian frederick50.05%10.47%
ralph wuerthnerralph wuerthner50.05%10.47%
dmitriy vyukovdmitriy vyukov40.04%10.47%
robert braggrobert bragg40.04%10.47%
geliang tanggeliang tang40.04%10.47%
giridhar pemmasanigiridhar pemmasani40.04%10.47%
lai jiangshanlai jiangshan40.04%20.93%
wang chaowang chao30.03%10.47%
konstantin khlebnikovkonstantin khlebnikov30.03%10.47%
pekka j enbergpekka j enberg30.03%10.47%
figo zhangfigo zhang30.03%10.47%
michael opdenackermichael opdenacker30.03%10.47%
hirofumi ogawahirofumi ogawa30.03%10.47%
kautuk consulkautuk consul30.03%10.47%
clemens ladischclemens ladisch20.02%10.47%
jan kiszkajan kiszka20.02%10.47%
martin schwidefskymartin schwidefsky20.02%10.47%
americo wangamerico wang20.02%10.47%
jianguo wujianguo wu20.02%10.47%
vasiliy kulikovvasiliy kulikov10.01%10.47%
adrian bunkadrian bunk10.01%10.47%
gioh kimgioh kim10.01%10.47%
julia lawalljulia lawall10.01%10.47%
mika kukkonenmika kukkonen10.01%10.47%
pintu kumarpintu kumar10.01%10.47%
shawn linshawn lin10.01%10.47%
johannes weinerjohannes weiner10.01%10.47%
simon arlottsimon arlott10.01%10.47%
arun sharmaarun sharma10.01%10.47%
richard hendersonrichard henderson10.01%10.47%
kees cookkees cook10.01%10.47%
Total10320100.00%214100.00%
Directory: mm
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.