diff options
| author | Alexey Ishchuk <aishchuk@linux.vnet.ibm.com> | 2014-11-14 08:27:58 -0500 |
|---|---|---|
| committer | Martin Schwidefsky <schwidefsky@de.ibm.com> | 2014-11-19 03:46:43 -0500 |
| commit | 4eafad7febd482092b331ea72c37274d745956be (patch) | |
| tree | 81df1dd45ee04b56180da2214fae5a3bfe66ea2a /arch/s390/pci | |
| parent | 86c558e8d3b774580faf8250092388d52cfde63e (diff) | |
s390/kernel: add system calls for PCI memory access
Add the new __NR_s390_pci_mmio_write and __NR_s390_pci_mmio_read
system calls to allow user space applications to access device PCI I/O
memory pages on s390x platform.
[ Martin Schwidefsky: some code beautification ]
Signed-off-by: Alexey Ishchuk <aishchuk@linux.vnet.ibm.com>
Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
Diffstat (limited to 'arch/s390/pci')
| -rw-r--r-- | arch/s390/pci/Makefile | 2 | ||||
| -rw-r--r-- | arch/s390/pci/pci_mmio.c | 115 |
2 files changed, 116 insertions, 1 deletions
diff --git a/arch/s390/pci/Makefile b/arch/s390/pci/Makefile index a9e1dc4ae442..805d8b29193a 100644 --- a/arch/s390/pci/Makefile +++ b/arch/s390/pci/Makefile | |||
| @@ -3,4 +3,4 @@ | |||
| 3 | # | 3 | # |
| 4 | 4 | ||
| 5 | obj-$(CONFIG_PCI) += pci.o pci_dma.o pci_clp.o pci_sysfs.o \ | 5 | obj-$(CONFIG_PCI) += pci.o pci_dma.o pci_clp.o pci_sysfs.o \ |
| 6 | pci_event.o pci_debug.o pci_insn.o | 6 | pci_event.o pci_debug.o pci_insn.o pci_mmio.o |
diff --git a/arch/s390/pci/pci_mmio.c b/arch/s390/pci/pci_mmio.c new file mode 100644 index 000000000000..62c5ea6d8682 --- /dev/null +++ b/arch/s390/pci/pci_mmio.c | |||
| @@ -0,0 +1,115 @@ | |||
| 1 | /* | ||
| 2 | * Access to PCI I/O memory from user space programs. | ||
| 3 | * | ||
| 4 | * Copyright IBM Corp. 2014 | ||
| 5 | * Author(s): Alexey Ishchuk <aishchuk@linux.vnet.ibm.com> | ||
| 6 | */ | ||
| 7 | #include <linux/kernel.h> | ||
| 8 | #include <linux/syscalls.h> | ||
| 9 | #include <linux/init.h> | ||
| 10 | #include <linux/mm.h> | ||
| 11 | #include <linux/errno.h> | ||
| 12 | #include <linux/pci.h> | ||
| 13 | |||
| 14 | static long get_pfn(unsigned long user_addr, unsigned long access, | ||
| 15 | unsigned long *pfn) | ||
| 16 | { | ||
| 17 | struct vm_area_struct *vma; | ||
| 18 | long ret; | ||
| 19 | |||
| 20 | down_read(¤t->mm->mmap_sem); | ||
| 21 | ret = -EINVAL; | ||
| 22 | vma = find_vma(current->mm, user_addr); | ||
| 23 | if (!vma) | ||
| 24 | goto out; | ||
| 25 | ret = -EACCES; | ||
| 26 | if (!(vma->vm_flags & access)) | ||
| 27 | goto out; | ||
| 28 | ret = follow_pfn(vma, user_addr, pfn); | ||
| 29 | out: | ||
| 30 | up_read(¤t->mm->mmap_sem); | ||
| 31 | return ret; | ||
| 32 | } | ||
| 33 | |||
| 34 | SYSCALL_DEFINE3(s390_pci_mmio_write, unsigned long, mmio_addr, | ||
| 35 | const void __user *, user_buffer, size_t, length) | ||
| 36 | { | ||
| 37 | u8 local_buf[64]; | ||
| 38 | void __iomem *io_addr; | ||
| 39 | void *buf; | ||
| 40 | unsigned long pfn; | ||
| 41 | long ret; | ||
| 42 | |||
| 43 | if (!zpci_is_enabled()) | ||
| 44 | return -ENODEV; | ||
| 45 | |||
| 46 | if (length <= 0 || PAGE_SIZE - (mmio_addr & ~PAGE_MASK) < length) | ||
| 47 | return -EINVAL; | ||
| 48 | if (length > 64) { | ||
| 49 | buf = kmalloc(length, GFP_KERNEL); | ||
| 50 | if (!buf) | ||
| 51 | return -ENOMEM; | ||
| 52 | } else | ||
| 53 | buf = local_buf; | ||
| 54 | |||
| 55 | ret = get_pfn(mmio_addr, VM_WRITE, &pfn); | ||
| 56 | if (ret) | ||
| 57 | goto out; | ||
| 58 | io_addr = (void *)((pfn << PAGE_SHIFT) | (mmio_addr & ~PAGE_MASK)); | ||
| 59 | |||
| 60 | ret = -EFAULT; | ||
| 61 | if ((unsigned long) io_addr < ZPCI_IOMAP_ADDR_BASE) | ||
| 62 | goto out; | ||
| 63 | |||
| 64 | if (copy_from_user(buf, user_buffer, length)) | ||
| 65 | goto out; | ||
| 66 | |||
| 67 | memcpy_toio(io_addr, buf, length); | ||
| 68 | ret = 0; | ||
| 69 | out: | ||
| 70 | if (buf != local_buf) | ||
| 71 | kfree(buf); | ||
| 72 | return ret; | ||
| 73 | } | ||
| 74 | |||
| 75 | SYSCALL_DEFINE3(s390_pci_mmio_read, unsigned long, mmio_addr, | ||
| 76 | void __user *, user_buffer, size_t, length) | ||
| 77 | { | ||
| 78 | u8 local_buf[64]; | ||
| 79 | void __iomem *io_addr; | ||
| 80 | void *buf; | ||
| 81 | unsigned long pfn; | ||
| 82 | long ret; | ||
| 83 | |||
| 84 | if (!zpci_is_enabled()) | ||
| 85 | return -ENODEV; | ||
| 86 | |||
| 87 | if (length <= 0 || PAGE_SIZE - (mmio_addr & ~PAGE_MASK) < length) | ||
| 88 | return -EINVAL; | ||
| 89 | if (length > 64) { | ||
| 90 | buf = kmalloc(length, GFP_KERNEL); | ||
| 91 | if (!buf) | ||
| 92 | return -ENOMEM; | ||
| 93 | } else | ||
| 94 | buf = local_buf; | ||
| 95 | |||
| 96 | ret = get_pfn(mmio_addr, VM_READ, &pfn); | ||
| 97 | if (ret) | ||
| 98 | goto out; | ||
| 99 | io_addr = (void *)((pfn << PAGE_SHIFT) | (mmio_addr & ~PAGE_MASK)); | ||
| 100 | |||
| 101 | ret = -EFAULT; | ||
| 102 | if ((unsigned long) io_addr < ZPCI_IOMAP_ADDR_BASE) | ||
| 103 | goto out; | ||
| 104 | |||
| 105 | memcpy_fromio(buf, io_addr, length); | ||
| 106 | |||
| 107 | if (copy_to_user(user_buffer, buf, length)) | ||
| 108 | goto out; | ||
| 109 | |||
| 110 | ret = 0; | ||
| 111 | out: | ||
| 112 | if (buf != local_buf) | ||
| 113 | kfree(buf); | ||
| 114 | return ret; | ||
| 115 | } | ||
