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 | } | ||