aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael Holzheu <holzheu@linux.vnet.ibm.com>2013-06-06 03:52:08 -0400
committerMartin Schwidefsky <schwidefsky@de.ibm.com>2013-06-26 15:10:13 -0400
commitd475f942b1dd6a897dac3ad4ed98d6994b275378 (patch)
tree43f599275ef7ac5350ea25f8f1eeb2690d6dc4e4
parente9a8f32a98a6099b009ea7da4f299bb5427db126 (diff)
s390/sclp: Add SCLP character device driver
Add a character misc device "sclp_ctl" that allows to run SCCBs from user space using the SCLP_CTL_SCCB ioctl. Signed-off-by: Michael Holzheu <holzheu@linux.vnet.ibm.com> Signed-off-by: Martin Schwidefsky <schwidefsky@de.ibm.com>
-rw-r--r--Documentation/ioctl/ioctl-number.txt1
-rw-r--r--arch/s390/include/uapi/asm/Kbuild1
-rw-r--r--arch/s390/include/uapi/asm/sclp_ctl.h24
-rw-r--r--drivers/s390/char/Makefile2
-rw-r--r--drivers/s390/char/sclp.c15
-rw-r--r--drivers/s390/char/sclp.h1
-rw-r--r--drivers/s390/char/sclp_cmd.c18
-rw-r--r--drivers/s390/char/sclp_ctl.c145
8 files changed, 192 insertions, 15 deletions
diff --git a/Documentation/ioctl/ioctl-number.txt b/Documentation/ioctl/ioctl-number.txt
index 237acab169dd..2a5f0e14efa3 100644
--- a/Documentation/ioctl/ioctl-number.txt
+++ b/Documentation/ioctl/ioctl-number.txt
@@ -72,6 +72,7 @@ Code Seq#(hex) Include File Comments
720x06 all linux/lp.h 720x06 all linux/lp.h
730x09 all linux/raid/md_u.h 730x09 all linux/raid/md_u.h
740x10 00-0F drivers/char/s390/vmcp.h 740x10 00-0F drivers/char/s390/vmcp.h
750x10 10-1F arch/s390/include/uapi/sclp_ctl.h
750x12 all linux/fs.h 760x12 all linux/fs.h
76 linux/blkpg.h 77 linux/blkpg.h
770x1b all InfiniBand Subsystem <http://infiniband.sourceforge.net/> 780x1b all InfiniBand Subsystem <http://infiniband.sourceforge.net/>
diff --git a/arch/s390/include/uapi/asm/Kbuild b/arch/s390/include/uapi/asm/Kbuild
index 9ccd1905bdad..6a9a9eb645f5 100644
--- a/arch/s390/include/uapi/asm/Kbuild
+++ b/arch/s390/include/uapi/asm/Kbuild
@@ -35,6 +35,7 @@ header-y += siginfo.h
35header-y += signal.h 35header-y += signal.h
36header-y += socket.h 36header-y += socket.h
37header-y += sockios.h 37header-y += sockios.h
38header-y += sclp_ctl.h
38header-y += stat.h 39header-y += stat.h
39header-y += statfs.h 40header-y += statfs.h
40header-y += swab.h 41header-y += swab.h
diff --git a/arch/s390/include/uapi/asm/sclp_ctl.h b/arch/s390/include/uapi/asm/sclp_ctl.h
new file mode 100644
index 000000000000..f2818613ee41
--- /dev/null
+++ b/arch/s390/include/uapi/asm/sclp_ctl.h
@@ -0,0 +1,24 @@
1/*
2 * IOCTL interface for SCLP
3 *
4 * Copyright IBM Corp. 2012
5 *
6 * Author: Michael Holzheu <holzheu@linux.vnet.ibm.com>
7 */
8
9#ifndef _ASM_SCLP_CTL_H
10#define _ASM_SCLP_CTL_H
11
12#include <linux/types.h>
13
14struct sclp_ctl_sccb {
15 __u32 cmdw;
16 __u64 sccb;
17} __attribute__((packed));
18
19#define SCLP_CTL_IOCTL_MAGIC 0x10
20
21#define SCLP_CTL_SCCB \
22 _IOWR(SCLP_CTL_IOCTL_MAGIC, 0x10, struct sclp_ctl_sccb)
23
24#endif
diff --git a/drivers/s390/char/Makefile b/drivers/s390/char/Makefile
index f3c325207445..17821a026c9c 100644
--- a/drivers/s390/char/Makefile
+++ b/drivers/s390/char/Makefile
@@ -3,7 +3,7 @@
3# 3#
4 4
5obj-y += ctrlchar.o keyboard.o defkeymap.o sclp.o sclp_rw.o sclp_quiesce.o \ 5obj-y += ctrlchar.o keyboard.o defkeymap.o sclp.o sclp_rw.o sclp_quiesce.o \
6 sclp_cmd.o sclp_config.o sclp_cpi_sys.o sclp_ocf.o 6 sclp_cmd.o sclp_config.o sclp_cpi_sys.o sclp_ocf.o sclp_ctl.o
7 7
8obj-$(CONFIG_TN3270) += raw3270.o 8obj-$(CONFIG_TN3270) += raw3270.o
9obj-$(CONFIG_TN3270_CONSOLE) += con3270.o 9obj-$(CONFIG_TN3270_CONSOLE) += con3270.o
diff --git a/drivers/s390/char/sclp.c b/drivers/s390/char/sclp.c
index a77febeead1f..97319d692b04 100644
--- a/drivers/s390/char/sclp.c
+++ b/drivers/s390/char/sclp.c
@@ -148,14 +148,19 @@ static int sclp_init(void);
148int 148int
149sclp_service_call(sclp_cmdw_t command, void *sccb) 149sclp_service_call(sclp_cmdw_t command, void *sccb)
150{ 150{
151 int cc; 151 int cc = 4; /* Initialize for program check handling */
152 152
153 asm volatile( 153 asm volatile(
154 " .insn rre,0xb2200000,%1,%2\n" /* servc %1,%2 */ 154 "0: .insn rre,0xb2200000,%1,%2\n" /* servc %1,%2 */
155 " ipm %0\n" 155 "1: ipm %0\n"
156 " srl %0,28" 156 " srl %0,28\n"
157 : "=&d" (cc) : "d" (command), "a" (__pa(sccb)) 157 "2:\n"
158 EX_TABLE(0b, 2b)
159 EX_TABLE(1b, 2b)
160 : "+&d" (cc) : "d" (command), "a" (__pa(sccb))
158 : "cc", "memory"); 161 : "cc", "memory");
162 if (cc == 4)
163 return -EINVAL;
159 if (cc == 3) 164 if (cc == 3)
160 return -EIO; 165 return -EIO;
161 if (cc == 2) 166 if (cc == 2)
diff --git a/drivers/s390/char/sclp.h b/drivers/s390/char/sclp.h
index e11383f5d3d2..40d1406289ed 100644
--- a/drivers/s390/char/sclp.h
+++ b/drivers/s390/char/sclp.h
@@ -171,6 +171,7 @@ int sclp_remove_processed(struct sccb_header *sccb);
171int sclp_deactivate(void); 171int sclp_deactivate(void);
172int sclp_reactivate(void); 172int sclp_reactivate(void);
173int sclp_service_call(sclp_cmdw_t command, void *sccb); 173int sclp_service_call(sclp_cmdw_t command, void *sccb);
174int sclp_sync_request(sclp_cmdw_t command, void *sccb);
174 175
175int sclp_sdias_init(void); 176int sclp_sdias_init(void);
176void sclp_sdias_exit(void); 177void sclp_sdias_exit(void);
diff --git a/drivers/s390/char/sclp_cmd.c b/drivers/s390/char/sclp_cmd.c
index bf07c3a188d4..657b0348579c 100644
--- a/drivers/s390/char/sclp_cmd.c
+++ b/drivers/s390/char/sclp_cmd.c
@@ -195,7 +195,7 @@ static void sclp_sync_callback(struct sclp_req *req, void *data)
195 complete(completion); 195 complete(completion);
196} 196}
197 197
198static int do_sync_request(sclp_cmdw_t cmd, void *sccb) 198int sclp_sync_request(sclp_cmdw_t cmd, void *sccb)
199{ 199{
200 struct completion completion; 200 struct completion completion;
201 struct sclp_req *request; 201 struct sclp_req *request;
@@ -270,7 +270,7 @@ int sclp_get_cpu_info(struct sclp_cpu_info *info)
270 if (!sccb) 270 if (!sccb)
271 return -ENOMEM; 271 return -ENOMEM;
272 sccb->header.length = sizeof(*sccb); 272 sccb->header.length = sizeof(*sccb);
273 rc = do_sync_request(SCLP_CMDW_READ_CPU_INFO, sccb); 273 rc = sclp_sync_request(SCLP_CMDW_READ_CPU_INFO, sccb);
274 if (rc) 274 if (rc)
275 goto out; 275 goto out;
276 if (sccb->header.response_code != 0x0010) { 276 if (sccb->header.response_code != 0x0010) {
@@ -304,7 +304,7 @@ static int do_cpu_configure(sclp_cmdw_t cmd)
304 if (!sccb) 304 if (!sccb)
305 return -ENOMEM; 305 return -ENOMEM;
306 sccb->header.length = sizeof(*sccb); 306 sccb->header.length = sizeof(*sccb);
307 rc = do_sync_request(cmd, sccb); 307 rc = sclp_sync_request(cmd, sccb);
308 if (rc) 308 if (rc)
309 goto out; 309 goto out;
310 switch (sccb->header.response_code) { 310 switch (sccb->header.response_code) {
@@ -374,7 +374,7 @@ static int do_assign_storage(sclp_cmdw_t cmd, u16 rn)
374 return -ENOMEM; 374 return -ENOMEM;
375 sccb->header.length = PAGE_SIZE; 375 sccb->header.length = PAGE_SIZE;
376 sccb->rn = rn; 376 sccb->rn = rn;
377 rc = do_sync_request(cmd, sccb); 377 rc = sclp_sync_request(cmd, sccb);
378 if (rc) 378 if (rc)
379 goto out; 379 goto out;
380 switch (sccb->header.response_code) { 380 switch (sccb->header.response_code) {
@@ -429,7 +429,7 @@ static int sclp_attach_storage(u8 id)
429 if (!sccb) 429 if (!sccb)
430 return -ENOMEM; 430 return -ENOMEM;
431 sccb->header.length = PAGE_SIZE; 431 sccb->header.length = PAGE_SIZE;
432 rc = do_sync_request(0x00080001 | id << 8, sccb); 432 rc = sclp_sync_request(0x00080001 | id << 8, sccb);
433 if (rc) 433 if (rc)
434 goto out; 434 goto out;
435 switch (sccb->header.response_code) { 435 switch (sccb->header.response_code) {
@@ -627,7 +627,7 @@ static int __init sclp_detect_standby_memory(void)
627 for (id = 0; id <= sclp_max_storage_id; id++) { 627 for (id = 0; id <= sclp_max_storage_id; id++) {
628 memset(sccb, 0, PAGE_SIZE); 628 memset(sccb, 0, PAGE_SIZE);
629 sccb->header.length = PAGE_SIZE; 629 sccb->header.length = PAGE_SIZE;
630 rc = do_sync_request(0x00040001 | id << 8, sccb); 630 rc = sclp_sync_request(0x00040001 | id << 8, sccb);
631 if (rc) 631 if (rc)
632 goto out; 632 goto out;
633 switch (sccb->header.response_code) { 633 switch (sccb->header.response_code) {
@@ -714,7 +714,7 @@ static int do_pci_configure(sclp_cmdw_t cmd, u32 fid)
714 sccb->header.length = PAGE_SIZE; 714 sccb->header.length = PAGE_SIZE;
715 sccb->atype = SCLP_RECONFIG_PCI_ATPYE; 715 sccb->atype = SCLP_RECONFIG_PCI_ATPYE;
716 sccb->aid = fid; 716 sccb->aid = fid;
717 rc = do_sync_request(cmd, sccb); 717 rc = sclp_sync_request(cmd, sccb);
718 if (rc) 718 if (rc)
719 goto out; 719 goto out;
720 switch (sccb->header.response_code) { 720 switch (sccb->header.response_code) {
@@ -771,7 +771,7 @@ static int do_chp_configure(sclp_cmdw_t cmd)
771 if (!sccb) 771 if (!sccb)
772 return -ENOMEM; 772 return -ENOMEM;
773 sccb->header.length = sizeof(*sccb); 773 sccb->header.length = sizeof(*sccb);
774 rc = do_sync_request(cmd, sccb); 774 rc = sclp_sync_request(cmd, sccb);
775 if (rc) 775 if (rc)
776 goto out; 776 goto out;
777 switch (sccb->header.response_code) { 777 switch (sccb->header.response_code) {
@@ -846,7 +846,7 @@ int sclp_chp_read_info(struct sclp_chp_info *info)
846 if (!sccb) 846 if (!sccb)
847 return -ENOMEM; 847 return -ENOMEM;
848 sccb->header.length = sizeof(*sccb); 848 sccb->header.length = sizeof(*sccb);
849 rc = do_sync_request(SCLP_CMDW_READ_CHPATH_INFORMATION, sccb); 849 rc = sclp_sync_request(SCLP_CMDW_READ_CHPATH_INFORMATION, sccb);
850 if (rc) 850 if (rc)
851 goto out; 851 goto out;
852 if (sccb->header.response_code != 0x0010) { 852 if (sccb->header.response_code != 0x0010) {
diff --git a/drivers/s390/char/sclp_ctl.c b/drivers/s390/char/sclp_ctl.c
new file mode 100644
index 000000000000..abe8ef13d148
--- /dev/null
+++ b/drivers/s390/char/sclp_ctl.c
@@ -0,0 +1,145 @@
1/*
2 * IOCTL interface for SCLP
3 *
4 * Copyright IBM Corp. 2012
5 *
6 * Author: Michael Holzheu <holzheu@linux.vnet.ibm.com>
7 */
8
9#include <linux/compat.h>
10#include <linux/uaccess.h>
11#include <linux/miscdevice.h>
12#include <linux/gfp.h>
13#include <linux/module.h>
14#include <linux/ioctl.h>
15#include <linux/fs.h>
16#include <linux/compat.h>
17#include <asm/compat.h>
18#include <asm/sclp_ctl.h>
19#include <asm/sclp.h>
20
21#include "sclp.h"
22
23/*
24 * Supported command words
25 */
26static unsigned int sclp_ctl_sccb_wlist[] = {
27 0x00400002,
28 0x00410002,
29};
30
31/*
32 * Check if command word is supported
33 */
34static int sclp_ctl_cmdw_supported(unsigned int cmdw)
35{
36 int i;
37
38 for (i = 0; i < ARRAY_SIZE(sclp_ctl_sccb_wlist); i++) {
39 if (cmdw == sclp_ctl_sccb_wlist[i])
40 return 1;
41 }
42 return 0;
43}
44
45static void __user *u64_to_uptr(u64 value)
46{
47 if (is_compat_task())
48 return compat_ptr(value);
49 else
50 return (void __user *)(unsigned long)value;
51}
52
53/*
54 * Start SCLP request
55 */
56static int sclp_ctl_ioctl_sccb(void __user *user_area)
57{
58 struct sclp_ctl_sccb ctl_sccb;
59 struct sccb_header *sccb;
60 int rc;
61
62 if (copy_from_user(&ctl_sccb, user_area, sizeof(ctl_sccb)))
63 return -EFAULT;
64 if (!sclp_ctl_cmdw_supported(ctl_sccb.cmdw))
65 return -EOPNOTSUPP;
66 sccb = (void *) get_zeroed_page(GFP_KERNEL | GFP_DMA);
67 if (!sccb)
68 return -ENOMEM;
69 if (copy_from_user(sccb, u64_to_uptr(ctl_sccb.sccb), sizeof(*sccb))) {
70 rc = -EFAULT;
71 goto out_free;
72 }
73 if (sccb->length > PAGE_SIZE || sccb->length < 8)
74 return -EINVAL;
75 if (copy_from_user(sccb, u64_to_uptr(ctl_sccb.sccb), sccb->length)) {
76 rc = -EFAULT;
77 goto out_free;
78 }
79 rc = sclp_sync_request(ctl_sccb.cmdw, sccb);
80 if (rc)
81 goto out_free;
82 if (copy_to_user(u64_to_uptr(ctl_sccb.sccb), sccb, sccb->length))
83 rc = -EFAULT;
84out_free:
85 free_page((unsigned long) sccb);
86 return rc;
87}
88
89/*
90 * SCLP SCCB ioctl function
91 */
92static long sclp_ctl_ioctl(struct file *filp, unsigned int cmd,
93 unsigned long arg)
94{
95 void __user *argp;
96
97 if (is_compat_task())
98 argp = compat_ptr(arg);
99 else
100 argp = (void __user *) arg;
101 switch (cmd) {
102 case SCLP_CTL_SCCB:
103 return sclp_ctl_ioctl_sccb(argp);
104 default: /* unknown ioctl number */
105 return -ENOTTY;
106 }
107}
108
109/*
110 * File operations
111 */
112static const struct file_operations sclp_ctl_fops = {
113 .owner = THIS_MODULE,
114 .open = nonseekable_open,
115 .unlocked_ioctl = sclp_ctl_ioctl,
116 .compat_ioctl = sclp_ctl_ioctl,
117 .llseek = no_llseek,
118};
119
120/*
121 * Misc device definition
122 */
123static struct miscdevice sclp_ctl_device = {
124 .minor = MISC_DYNAMIC_MINOR,
125 .name = "sclp",
126 .fops = &sclp_ctl_fops,
127};
128
129/*
130 * Register sclp_ctl misc device
131 */
132static int __init sclp_ctl_init(void)
133{
134 return misc_register(&sclp_ctl_device);
135}
136module_init(sclp_ctl_init);
137
138/*
139 * Deregister sclp_ctl misc device
140 */
141static void __exit sclp_ctl_exit(void)
142{
143 misc_deregister(&sclp_ctl_device);
144}
145module_exit(sclp_ctl_exit);