cregit-Linux how code gets into the kernel

Release 4.11 arch/arm/mm/mmap.c

Directory: arch/arm/mm
/*
 *  linux/arch/arm/mm/mmap.c
 */
#include <linux/fs.h>
#include <linux/mm.h>
#include <linux/mman.h>
#include <linux/shm.h>
#include <linux/sched/signal.h>
#include <linux/sched/mm.h>
#include <linux/io.h>
#include <linux/personality.h>
#include <linux/random.h>
#include <asm/cachetype.h>


#define COLOUR_ALIGN(addr,pgoff)		\
	((((addr)+SHMLBA-1)&~(SHMLBA-1)) +      \
         (((pgoff)<<PAGE_SHIFT) & (SHMLBA-1)))

/* gap between mmap and stack */

#define MIN_GAP (128*1024*1024UL)

#define MAX_GAP ((TASK_SIZE)/6*5)


static int mmap_is_legacy(void) { if (current->personality & ADDR_COMPAT_LAYOUT) return 1; if (rlimit(RLIMIT_STACK) == RLIM_INFINITY) return 1; return sysctl_legacy_va_layout; }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring34100.00%1100.00%
Total34100.00%1100.00%


static unsigned long mmap_base(unsigned long rnd) { unsigned long gap = rlimit(RLIMIT_STACK); if (gap < MIN_GAP) gap = MIN_GAP; else if (gap > MAX_GAP) gap = MAX_GAP; return PAGE_ALIGN(TASK_SIZE - gap - rnd); }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring51100.00%1100.00%
Total51100.00%1100.00%

/* * We need to ensure that shared mappings are correctly aligned to * avoid aliasing issues with VIPT caches. We need to ensure that * a specific page of an object is always mapped at a multiple of * SHMLBA bytes. * * We unconditionally provide this function for all cases, however * in the VIVT case, we optimise out the alignment rules. */
unsigned long arch_get_unmapped_area(struct file *filp, unsigned long addr, unsigned long len, unsigned long pgoff, unsigned long flags) { struct mm_struct *mm = current->mm; struct vm_area_struct *vma; int do_align = 0; int aliasing = cache_is_vipt_aliasing(); struct vm_unmapped_area_info info; /* * We only need to do colour alignment if either the I or D * caches alias. */ if (aliasing) do_align = filp || (flags & MAP_SHARED); /* * We enforce the MAP_FIXED case. */ if (flags & MAP_FIXED) { if (aliasing && flags & MAP_SHARED && (addr - (pgoff << PAGE_SHIFT)) & (SHMLBA - 1)) return -EINVAL; return addr; } if (len > TASK_SIZE) return -ENOMEM; if (addr) { if (do_align) addr = COLOUR_ALIGN(addr, pgoff); else addr = PAGE_ALIGN(addr); vma = find_vma(mm, addr); if (TASK_SIZE - len >= addr && (!vma || addr + len <= vma->vm_start)) return addr; } info.flags = 0; info.length = len; info.low_limit = mm->mmap_base; info.high_limit = TASK_SIZE; info.align_mask = do_align ? (PAGE_MASK & (SHMLBA - 1)) : 0; info.align_offset = pgoff << PAGE_SHIFT; return vm_unmapped_area(&info); }

Contributors

PersonTokensPropCommitsCommitProp
Russell King17171.25%225.00%
Michel Lespinasse4418.33%112.50%
Rob Herring104.17%225.00%
Al Viro83.33%112.50%
Wolfgang Wander62.50%112.50%
Benjamin Herrenschmidt10.42%112.50%
Total240100.00%8100.00%


unsigned long arch_get_unmapped_area_topdown(struct file *filp, const unsigned long addr0, const unsigned long len, const unsigned long pgoff, const unsigned long flags) { struct vm_area_struct *vma; struct mm_struct *mm = current->mm; unsigned long addr = addr0; int do_align = 0; int aliasing = cache_is_vipt_aliasing(); struct vm_unmapped_area_info info; /* * We only need to do colour alignment if either the I or D * caches alias. */ if (aliasing) do_align = filp || (flags & MAP_SHARED); /* requested length too big for entire address space */ if (len > TASK_SIZE) return -ENOMEM; if (flags & MAP_FIXED) { if (aliasing && flags & MAP_SHARED && (addr - (pgoff << PAGE_SHIFT)) & (SHMLBA - 1)) return -EINVAL; return addr; } /* requesting a specific address */ if (addr) { if (do_align) addr = COLOUR_ALIGN(addr, pgoff); else addr = PAGE_ALIGN(addr); vma = find_vma(mm, addr); if (TASK_SIZE - len >= addr && (!vma || addr + len <= vma->vm_start)) return addr; } info.flags = VM_UNMAPPED_AREA_TOPDOWN; info.length = len; info.low_limit = FIRST_USER_ADDRESS; info.high_limit = mm->mmap_base; info.align_mask = do_align ? (PAGE_MASK & (SHMLBA - 1)) : 0; info.align_offset = pgoff << PAGE_SHIFT; addr = vm_unmapped_area(&info); /* * A failed mmap() very likely causes application failure, * so fall back to the bottom-up function here. This scenario * can happen with large stack limits and large mmap() * allocations. */ if (addr & ~PAGE_MASK) { VM_BUG_ON(addr != -ENOMEM); info.flags = 0; info.low_limit = mm->mmap_base; info.high_limit = TASK_SIZE; addr = vm_unmapped_area(&info); } return addr; }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring23277.08%133.33%
Michel Lespinasse6822.59%133.33%
Russell King10.33%133.33%
Total301100.00%3100.00%


unsigned long arch_mmap_rnd(void) { unsigned long rnd; rnd = get_random_long() & ((1UL << mmap_rnd_bits) - 1); return rnd << PAGE_SHIFT; }

Contributors

PersonTokensPropCommitsCommitProp
Kees Cook2475.00%250.00%
Daniel Cashman825.00%250.00%
Total32100.00%4100.00%


void arch_pick_mmap_layout(struct mm_struct *mm) { unsigned long random_factor = 0UL; if (current->flags & PF_RANDOMIZE) random_factor = arch_mmap_rnd(); if (mmap_is_legacy()) { mm->mmap_base = TASK_UNMAPPED_BASE + random_factor; mm->get_unmapped_area = arch_get_unmapped_area; } else { mm->mmap_base = mmap_base(random_factor); mm->get_unmapped_area = arch_get_unmapped_area_topdown; } }

Contributors

PersonTokensPropCommitsCommitProp
Rob Herring6798.53%150.00%
Kees Cook11.47%150.00%
Total68100.00%2100.00%

/* * You really shouldn't be using read() or write() on /dev/mem. This * might go away in the future. */
int valid_phys_addr_range(phys_addr_t addr, size_t size) { if (addr < PHYS_OFFSET) return 0; if (addr + size > __pa(high_memory - 1) + 1) return 0; return 1; }

Contributors

PersonTokensPropCommitsCommitProp
Lennert Buytenhek2663.41%120.00%
Alexandre Rusev921.95%120.00%
Greg Ungerer37.32%120.00%
Russell King24.88%120.00%
Cyril Chemparathy12.44%120.00%
Total41100.00%5100.00%

/* * Do not allow /dev/mem mappings beyond the supported physical range. */
int valid_mmap_phys_addr_range(unsigned long pfn, size_t size) { return (pfn + (size >> PAGE_SHIFT)) <= (1 + (PHYS_MASK >> PAGE_SHIFT)); }

Contributors

PersonTokensPropCommitsCommitProp
Lennert Buytenhek2369.70%150.00%
Sergey Dyasly1030.30%150.00%
Total33100.00%2100.00%

#ifdef CONFIG_STRICT_DEVMEM #include <linux/ioport.h> /* * devmem_is_allowed() checks to see if /dev/mem access to a certain * address is valid. The argument is a physical page number. * We mimic x86 here by disallowing access to system RAM as well as * device-exclusive MMIO regions. This effectively disable read()/write() * on /dev/mem. */
int devmem_is_allowed(unsigned long pfn) { if (iomem_is_exclusive(pfn << PAGE_SHIFT)) return 0; if (!page_is_ram(pfn)) return 1; return 0; }

Contributors

PersonTokensPropCommitsCommitProp
Nico Pitre35100.00%1100.00%
Total35100.00%1100.00%

#endif

Overall Contributors

PersonTokensPropCommitsCommitProp
Rob Herring40445.04%27.69%
Russell King20322.63%519.23%
Michel Lespinasse11212.49%13.85%
Nico Pitre505.57%311.54%
Lennert Buytenhek505.57%13.85%
Kees Cook252.79%27.69%
Sergey Dyasly111.23%13.85%
Alexandre Rusev91.00%13.85%
Daniel Cashman80.89%27.69%
Al Viro80.89%13.85%
Wolfgang Wander60.67%13.85%
Ingo Molnar40.45%27.69%
Greg Ungerer30.33%13.85%
Alexey Dobriyan20.22%13.85%
Cyril Chemparathy10.11%13.85%
Benjamin Herrenschmidt10.11%13.85%
Total897100.00%26100.00%
Directory: arch/arm/mm
Information contained on this website is for historical information purposes only and does not indicate or represent copyright ownership.
Created with cregit.