mirror of
https://github.com/torvalds/linux.git
synced 2024-12-22 02:52:56 +00:00
57319d80e1
We have chosen to perform the allocation of bounds tables in kernel (See the patch "on-demand kernel allocation of bounds tables") and to mark these VMAs with VM_MPX. However, there is currently no suitable interface to actually do this. Existing interfaces, like do_mmap_pgoff(), have no way to set a modified ->vm_ops or ->vm_flags and don't hold mmap_sem long enough to let a caller do it. This patch wraps mmap_region() and hold mmap_sem long enough to make the modifications to the VMA which we need. Also note the 32/64-bit #ifdef in the header. We actually need to do this at runtime eventually. But, for now, we don't support running 32-bit binaries on 64-bit kernels. Support for this will come in later patches. Signed-off-by: Qiaowei Ren <qiaowei.ren@intel.com> Signed-off-by: Dave Hansen <dave.hansen@linux.intel.com> Cc: linux-mm@kvack.org Cc: linux-mips@linux-mips.org Cc: Dave Hansen <dave@sr71.net> Link: http://lkml.kernel.org/r/20141114151827.CE440F67@viggo.jf.intel.com Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
87 lines
1.9 KiB
C
87 lines
1.9 KiB
C
/*
|
|
* mpx.c - Memory Protection eXtensions
|
|
*
|
|
* Copyright (c) 2014, Intel Corporation.
|
|
* Qiaowei Ren <qiaowei.ren@intel.com>
|
|
* Dave Hansen <dave.hansen@intel.com>
|
|
*/
|
|
#include <linux/kernel.h>
|
|
#include <linux/syscalls.h>
|
|
#include <linux/sched/sysctl.h>
|
|
|
|
#include <asm/mman.h>
|
|
#include <asm/mpx.h>
|
|
|
|
static const char *mpx_mapping_name(struct vm_area_struct *vma)
|
|
{
|
|
return "[mpx]";
|
|
}
|
|
|
|
static struct vm_operations_struct mpx_vma_ops = {
|
|
.name = mpx_mapping_name,
|
|
};
|
|
|
|
/*
|
|
* This is really a simplified "vm_mmap". it only handles MPX
|
|
* bounds tables (the bounds directory is user-allocated).
|
|
*
|
|
* Later on, we use the vma->vm_ops to uniquely identify these
|
|
* VMAs.
|
|
*/
|
|
static unsigned long mpx_mmap(unsigned long len)
|
|
{
|
|
unsigned long ret;
|
|
unsigned long addr, pgoff;
|
|
struct mm_struct *mm = current->mm;
|
|
vm_flags_t vm_flags;
|
|
struct vm_area_struct *vma;
|
|
|
|
/* Only bounds table and bounds directory can be allocated here */
|
|
if (len != MPX_BD_SIZE_BYTES && len != MPX_BT_SIZE_BYTES)
|
|
return -EINVAL;
|
|
|
|
down_write(&mm->mmap_sem);
|
|
|
|
/* Too many mappings? */
|
|
if (mm->map_count > sysctl_max_map_count) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
|
|
/* Obtain the address to map to. we verify (or select) it and ensure
|
|
* that it represents a valid section of the address space.
|
|
*/
|
|
addr = get_unmapped_area(NULL, 0, len, 0, MAP_ANONYMOUS | MAP_PRIVATE);
|
|
if (addr & ~PAGE_MASK) {
|
|
ret = addr;
|
|
goto out;
|
|
}
|
|
|
|
vm_flags = VM_READ | VM_WRITE | VM_MPX |
|
|
mm->def_flags | VM_MAYREAD | VM_MAYWRITE | VM_MAYEXEC;
|
|
|
|
/* Set pgoff according to addr for anon_vma */
|
|
pgoff = addr >> PAGE_SHIFT;
|
|
|
|
ret = mmap_region(NULL, addr, len, vm_flags, pgoff);
|
|
if (IS_ERR_VALUE(ret))
|
|
goto out;
|
|
|
|
vma = find_vma(mm, ret);
|
|
if (!vma) {
|
|
ret = -ENOMEM;
|
|
goto out;
|
|
}
|
|
vma->vm_ops = &mpx_vma_ops;
|
|
|
|
if (vm_flags & VM_LOCKED) {
|
|
up_write(&mm->mmap_sem);
|
|
mm_populate(ret, len);
|
|
return ret;
|
|
}
|
|
|
|
out:
|
|
up_write(&mm->mmap_sem);
|
|
return ret;
|
|
}
|