/* * This file is provided under a dual BSD/GPLv2 license. When using or * redistributing this file, you may do so under either license. * * GPL LICENSE SUMMARY * * Copyright(c) 2016 Intel Corporation. * * This program is free software; you can redistribute it and/or modify * it under the terms of version 2 of the GNU General Public License as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, but * WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * General Public License for more details. * * Contact Information: * Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com> * Intel Finland Oy - BIC 0357606-4 - Westendinkatu 7, 02160 Espoo * * BSD LICENSE * * Copyright(c) 2016 Intel Corporation. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * Neither the name of Intel Corporation nor the names of its * contributors may be used to endorse or promote products derived * from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. * * Authors: * * Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com> * Suresh Siddha <suresh.b.siddha@intel.com> * Serge Ayoun <serge.ayoun@intel.com> * Shay Katz-zamir <shay.katz-zamir@intel.com> * Sean Christopherson <sean.j.christopherson@intel.com> */ #include "sgx.h" #include <asm/mman.h> #include <linux/delay.h> #include <linux/file.h> #include <linux/highmem.h> #include <linux/ratelimit.h> #include <linux/slab.h> #include <linux/hashtable.h> #include <linux/shmem_fs.h> #include <linux/mm.h> static void sgx_vma_open(struct vm_area_struct *vma) { struct sgx_encl *encl = vma->vm_private_data; if (!encl) return; /* kref cannot underflow because ECREATE ioctl checks that there is only * one single VMA for the enclave before proceeding. */ kref_get(&encl->refcount); } static void sgx_vma_close(struct vm_area_struct *vma) { struct sgx_encl *encl = vma->vm_private_data; if (!encl) return; mutex_lock(&encl->lock); zap_vma_ptes(vma, vma->vm_start, vma->vm_end - vma->vm_start); encl->flags |= SGX_ENCL_DEAD; mutex_unlock(&encl->lock); kref_put(&encl->refcount, sgx_encl_release); } #if (LINUX_VERSION_CODE >= KERNEL_VERSION(4,11,0)) static int sgx_vma_fault(struct vm_fault *vmf) { struct vm_area_struct *vma = vmf->vma; #else static int sgx_vma_fault(struct vm_area_struct *vma, struct vm_fault *vmf) { #endif #if (LINUX_VERSION_CODE >= KERNEL_VERSION(4,10,0)) unsigned long addr = (unsigned long)vmf->address; #else unsigned long addr = (unsigned long) vmf->virtual_address; #endif struct sgx_encl_page *entry; entry = sgx_fault_page(vma, addr, 0); if (!IS_ERR(entry) || PTR_ERR(entry) == -EBUSY) return VM_FAULT_NOPAGE; else return VM_FAULT_SIGBUS; } static inline int sgx_vma_access_word(struct sgx_encl *encl, unsigned long addr, void *buf, int len, int write, struct sgx_encl_page *encl_page, int i) { char data[sizeof(unsigned long)]; int align, cnt, offset; void *vaddr; int ret; offset = ((addr + i) & (PAGE_SIZE - 1)) & ~(sizeof(unsigned long) - 1); align = (addr + i) & (sizeof(unsigned long) - 1); cnt = sizeof(unsigned long) - align; cnt = min(cnt, len - i); if (write) { if (encl_page->flags & SGX_ENCL_PAGE_TCS && (offset < 8 || (offset + (len - i)) > 16)) return -ECANCELED; if (align || (cnt != sizeof(unsigned long))) { vaddr = sgx_get_page(encl_page->epc_page); ret = __edbgrd((void *)((unsigned long)vaddr + offset), (unsigned long *)data); sgx_put_page(vaddr); if (ret) { sgx_dbg(encl, "EDBGRD returned %d\n", ret); return -EFAULT; } } memcpy(data + align, buf + i, cnt); vaddr = sgx_get_page(encl_page->epc_page); ret = __edbgwr((void *)((unsigned long)vaddr + offset), (unsigned long *)data); sgx_put_page(vaddr); if (ret) { sgx_dbg(encl, "EDBGWR returned %d\n", ret); return -EFAULT; } } else { if (encl_page->flags & SGX_ENCL_PAGE_TCS && (offset + (len - i)) > 72) return -ECANCELED; vaddr = sgx_get_page(encl_page->epc_page); ret = __edbgrd((void *)((unsigned long)vaddr + offset), (unsigned long *)data); sgx_put_page(vaddr); if (ret) { sgx_dbg(encl, "EDBGRD returned %d\n", ret); return -EFAULT; } memcpy(buf + i, data + align, cnt); } return cnt; } static int sgx_vma_access(struct vm_area_struct *vma, unsigned long addr, void *buf, int len, int write) { struct sgx_encl *encl = vma->vm_private_data; struct sgx_encl_page *entry = NULL; const char *op_str = write ? "EDBGWR" : "EDBGRD"; int ret = 0; int i; /* If process was forked, VMA is still there but vm_private_data is set * to NULL. */ if (!encl) return -EFAULT; if (!(encl->flags & SGX_ENCL_DEBUG) || !(encl->flags & SGX_ENCL_INITIALIZED) || (encl->flags & SGX_ENCL_DEAD)) return -EFAULT; sgx_dbg(encl, "%s addr=0x%lx, len=%d\n", op_str, addr, len); for (i = 0; i < len; i += ret) { if (!entry || !((addr + i) & (PAGE_SIZE - 1))) { if (entry) entry->flags &= ~SGX_ENCL_PAGE_RESERVED; entry = sgx_fault_page(vma, (addr + i) & PAGE_MASK, SGX_FAULT_RESERVE); if (IS_ERR(entry)) { ret = PTR_ERR(entry); entry = NULL; break; } } /* No locks are needed because used fields are immutable after * intialization. */ ret = sgx_vma_access_word(encl, addr, buf, len, write, entry, i); if (ret < 0) break; } if (entry) entry->flags &= ~SGX_ENCL_PAGE_RESERVED; return (ret < 0 && ret != -ECANCELED) ? ret : i; } const struct vm_operations_struct sgx_vm_ops = { .close = sgx_vma_close, .open = sgx_vma_open, .fault = sgx_vma_fault, .access = sgx_vma_access, };