1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * IBM/3270 Driver - fullscreen driver.
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Author(s):
6*4882a593Smuzhiyun * Original 3270 Code for 2.4 written by Richard Hitt (UTS Global)
7*4882a593Smuzhiyun * Rewritten for 2.5/2.6 by Martin Schwidefsky <schwidefsky@de.ibm.com>
8*4882a593Smuzhiyun * Copyright IBM Corp. 2003, 2009
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <linux/memblock.h>
12*4882a593Smuzhiyun #include <linux/console.h>
13*4882a593Smuzhiyun #include <linux/init.h>
14*4882a593Smuzhiyun #include <linux/interrupt.h>
15*4882a593Smuzhiyun #include <linux/compat.h>
16*4882a593Smuzhiyun #include <linux/sched/signal.h>
17*4882a593Smuzhiyun #include <linux/module.h>
18*4882a593Smuzhiyun #include <linux/list.h>
19*4882a593Smuzhiyun #include <linux/slab.h>
20*4882a593Smuzhiyun #include <linux/types.h>
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun #include <asm/ccwdev.h>
23*4882a593Smuzhiyun #include <asm/cio.h>
24*4882a593Smuzhiyun #include <asm/ebcdic.h>
25*4882a593Smuzhiyun #include <asm/idals.h>
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #include "raw3270.h"
28*4882a593Smuzhiyun #include "ctrlchar.h"
29*4882a593Smuzhiyun
30*4882a593Smuzhiyun static struct raw3270_fn fs3270_fn;
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun struct fs3270 {
33*4882a593Smuzhiyun struct raw3270_view view;
34*4882a593Smuzhiyun struct pid *fs_pid; /* Pid of controlling program. */
35*4882a593Smuzhiyun int read_command; /* ccw command to use for reads. */
36*4882a593Smuzhiyun int write_command; /* ccw command to use for writes. */
37*4882a593Smuzhiyun int attention; /* Got attention. */
38*4882a593Smuzhiyun int active; /* Fullscreen view is active. */
39*4882a593Smuzhiyun struct raw3270_request *init; /* single init request. */
40*4882a593Smuzhiyun wait_queue_head_t wait; /* Init & attention wait queue. */
41*4882a593Smuzhiyun struct idal_buffer *rdbuf; /* full-screen-deactivate buffer */
42*4882a593Smuzhiyun size_t rdbuf_size; /* size of data returned by RDBUF */
43*4882a593Smuzhiyun };
44*4882a593Smuzhiyun
45*4882a593Smuzhiyun static DEFINE_MUTEX(fs3270_mutex);
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun static void
fs3270_wake_up(struct raw3270_request * rq,void * data)48*4882a593Smuzhiyun fs3270_wake_up(struct raw3270_request *rq, void *data)
49*4882a593Smuzhiyun {
50*4882a593Smuzhiyun wake_up((wait_queue_head_t *) data);
51*4882a593Smuzhiyun }
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun static inline int
fs3270_working(struct fs3270 * fp)54*4882a593Smuzhiyun fs3270_working(struct fs3270 *fp)
55*4882a593Smuzhiyun {
56*4882a593Smuzhiyun /*
57*4882a593Smuzhiyun * The fullscreen view is in working order if the view
58*4882a593Smuzhiyun * has been activated AND the initial request is finished.
59*4882a593Smuzhiyun */
60*4882a593Smuzhiyun return fp->active && raw3270_request_final(fp->init);
61*4882a593Smuzhiyun }
62*4882a593Smuzhiyun
63*4882a593Smuzhiyun static int
fs3270_do_io(struct raw3270_view * view,struct raw3270_request * rq)64*4882a593Smuzhiyun fs3270_do_io(struct raw3270_view *view, struct raw3270_request *rq)
65*4882a593Smuzhiyun {
66*4882a593Smuzhiyun struct fs3270 *fp;
67*4882a593Smuzhiyun int rc;
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun fp = (struct fs3270 *) view;
70*4882a593Smuzhiyun rq->callback = fs3270_wake_up;
71*4882a593Smuzhiyun rq->callback_data = &fp->wait;
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun do {
74*4882a593Smuzhiyun if (!fs3270_working(fp)) {
75*4882a593Smuzhiyun /* Fullscreen view isn't ready yet. */
76*4882a593Smuzhiyun rc = wait_event_interruptible(fp->wait,
77*4882a593Smuzhiyun fs3270_working(fp));
78*4882a593Smuzhiyun if (rc != 0)
79*4882a593Smuzhiyun break;
80*4882a593Smuzhiyun }
81*4882a593Smuzhiyun rc = raw3270_start(view, rq);
82*4882a593Smuzhiyun if (rc == 0) {
83*4882a593Smuzhiyun /* Started successfully. Now wait for completion. */
84*4882a593Smuzhiyun wait_event(fp->wait, raw3270_request_final(rq));
85*4882a593Smuzhiyun }
86*4882a593Smuzhiyun } while (rc == -EACCES);
87*4882a593Smuzhiyun return rc;
88*4882a593Smuzhiyun }
89*4882a593Smuzhiyun
90*4882a593Smuzhiyun /*
91*4882a593Smuzhiyun * Switch to the fullscreen view.
92*4882a593Smuzhiyun */
93*4882a593Smuzhiyun static void
fs3270_reset_callback(struct raw3270_request * rq,void * data)94*4882a593Smuzhiyun fs3270_reset_callback(struct raw3270_request *rq, void *data)
95*4882a593Smuzhiyun {
96*4882a593Smuzhiyun struct fs3270 *fp;
97*4882a593Smuzhiyun
98*4882a593Smuzhiyun fp = (struct fs3270 *) rq->view;
99*4882a593Smuzhiyun raw3270_request_reset(rq);
100*4882a593Smuzhiyun wake_up(&fp->wait);
101*4882a593Smuzhiyun }
102*4882a593Smuzhiyun
103*4882a593Smuzhiyun static void
fs3270_restore_callback(struct raw3270_request * rq,void * data)104*4882a593Smuzhiyun fs3270_restore_callback(struct raw3270_request *rq, void *data)
105*4882a593Smuzhiyun {
106*4882a593Smuzhiyun struct fs3270 *fp;
107*4882a593Smuzhiyun
108*4882a593Smuzhiyun fp = (struct fs3270 *) rq->view;
109*4882a593Smuzhiyun if (rq->rc != 0 || rq->rescnt != 0) {
110*4882a593Smuzhiyun if (fp->fs_pid)
111*4882a593Smuzhiyun kill_pid(fp->fs_pid, SIGHUP, 1);
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun fp->rdbuf_size = 0;
114*4882a593Smuzhiyun raw3270_request_reset(rq);
115*4882a593Smuzhiyun wake_up(&fp->wait);
116*4882a593Smuzhiyun }
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun static int
fs3270_activate(struct raw3270_view * view)119*4882a593Smuzhiyun fs3270_activate(struct raw3270_view *view)
120*4882a593Smuzhiyun {
121*4882a593Smuzhiyun struct fs3270 *fp;
122*4882a593Smuzhiyun char *cp;
123*4882a593Smuzhiyun int rc;
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun fp = (struct fs3270 *) view;
126*4882a593Smuzhiyun
127*4882a593Smuzhiyun /* If an old init command is still running just return. */
128*4882a593Smuzhiyun if (!raw3270_request_final(fp->init))
129*4882a593Smuzhiyun return 0;
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun if (fp->rdbuf_size == 0) {
132*4882a593Smuzhiyun /* No saved buffer. Just clear the screen. */
133*4882a593Smuzhiyun raw3270_request_set_cmd(fp->init, TC_EWRITEA);
134*4882a593Smuzhiyun fp->init->callback = fs3270_reset_callback;
135*4882a593Smuzhiyun } else {
136*4882a593Smuzhiyun /* Restore fullscreen buffer saved by fs3270_deactivate. */
137*4882a593Smuzhiyun raw3270_request_set_cmd(fp->init, TC_EWRITEA);
138*4882a593Smuzhiyun raw3270_request_set_idal(fp->init, fp->rdbuf);
139*4882a593Smuzhiyun fp->init->ccw.count = fp->rdbuf_size;
140*4882a593Smuzhiyun cp = fp->rdbuf->data[0];
141*4882a593Smuzhiyun cp[0] = TW_KR;
142*4882a593Smuzhiyun cp[1] = TO_SBA;
143*4882a593Smuzhiyun cp[2] = cp[6];
144*4882a593Smuzhiyun cp[3] = cp[7];
145*4882a593Smuzhiyun cp[4] = TO_IC;
146*4882a593Smuzhiyun cp[5] = TO_SBA;
147*4882a593Smuzhiyun cp[6] = 0x40;
148*4882a593Smuzhiyun cp[7] = 0x40;
149*4882a593Smuzhiyun fp->init->rescnt = 0;
150*4882a593Smuzhiyun fp->init->callback = fs3270_restore_callback;
151*4882a593Smuzhiyun }
152*4882a593Smuzhiyun rc = fp->init->rc = raw3270_start_locked(view, fp->init);
153*4882a593Smuzhiyun if (rc)
154*4882a593Smuzhiyun fp->init->callback(fp->init, NULL);
155*4882a593Smuzhiyun else
156*4882a593Smuzhiyun fp->active = 1;
157*4882a593Smuzhiyun return rc;
158*4882a593Smuzhiyun }
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun /*
161*4882a593Smuzhiyun * Shutdown fullscreen view.
162*4882a593Smuzhiyun */
163*4882a593Smuzhiyun static void
fs3270_save_callback(struct raw3270_request * rq,void * data)164*4882a593Smuzhiyun fs3270_save_callback(struct raw3270_request *rq, void *data)
165*4882a593Smuzhiyun {
166*4882a593Smuzhiyun struct fs3270 *fp;
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun fp = (struct fs3270 *) rq->view;
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun /* Correct idal buffer element 0 address. */
171*4882a593Smuzhiyun fp->rdbuf->data[0] -= 5;
172*4882a593Smuzhiyun fp->rdbuf->size += 5;
173*4882a593Smuzhiyun
174*4882a593Smuzhiyun /*
175*4882a593Smuzhiyun * If the rdbuf command failed or the idal buffer is
176*4882a593Smuzhiyun * to small for the amount of data returned by the
177*4882a593Smuzhiyun * rdbuf command, then we have no choice but to send
178*4882a593Smuzhiyun * a SIGHUP to the application.
179*4882a593Smuzhiyun */
180*4882a593Smuzhiyun if (rq->rc != 0 || rq->rescnt == 0) {
181*4882a593Smuzhiyun if (fp->fs_pid)
182*4882a593Smuzhiyun kill_pid(fp->fs_pid, SIGHUP, 1);
183*4882a593Smuzhiyun fp->rdbuf_size = 0;
184*4882a593Smuzhiyun } else
185*4882a593Smuzhiyun fp->rdbuf_size = fp->rdbuf->size - rq->rescnt;
186*4882a593Smuzhiyun raw3270_request_reset(rq);
187*4882a593Smuzhiyun wake_up(&fp->wait);
188*4882a593Smuzhiyun }
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun static void
fs3270_deactivate(struct raw3270_view * view)191*4882a593Smuzhiyun fs3270_deactivate(struct raw3270_view *view)
192*4882a593Smuzhiyun {
193*4882a593Smuzhiyun struct fs3270 *fp;
194*4882a593Smuzhiyun
195*4882a593Smuzhiyun fp = (struct fs3270 *) view;
196*4882a593Smuzhiyun fp->active = 0;
197*4882a593Smuzhiyun
198*4882a593Smuzhiyun /* If an old init command is still running just return. */
199*4882a593Smuzhiyun if (!raw3270_request_final(fp->init))
200*4882a593Smuzhiyun return;
201*4882a593Smuzhiyun
202*4882a593Smuzhiyun /* Prepare read-buffer request. */
203*4882a593Smuzhiyun raw3270_request_set_cmd(fp->init, TC_RDBUF);
204*4882a593Smuzhiyun /*
205*4882a593Smuzhiyun * Hackish: skip first 5 bytes of the idal buffer to make
206*4882a593Smuzhiyun * room for the TW_KR/TO_SBA/<address>/<address>/TO_IC sequence
207*4882a593Smuzhiyun * in the activation command.
208*4882a593Smuzhiyun */
209*4882a593Smuzhiyun fp->rdbuf->data[0] += 5;
210*4882a593Smuzhiyun fp->rdbuf->size -= 5;
211*4882a593Smuzhiyun raw3270_request_set_idal(fp->init, fp->rdbuf);
212*4882a593Smuzhiyun fp->init->rescnt = 0;
213*4882a593Smuzhiyun fp->init->callback = fs3270_save_callback;
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun /* Start I/O to read in the 3270 buffer. */
216*4882a593Smuzhiyun fp->init->rc = raw3270_start_locked(view, fp->init);
217*4882a593Smuzhiyun if (fp->init->rc)
218*4882a593Smuzhiyun fp->init->callback(fp->init, NULL);
219*4882a593Smuzhiyun }
220*4882a593Smuzhiyun
221*4882a593Smuzhiyun static void
fs3270_irq(struct fs3270 * fp,struct raw3270_request * rq,struct irb * irb)222*4882a593Smuzhiyun fs3270_irq(struct fs3270 *fp, struct raw3270_request *rq, struct irb *irb)
223*4882a593Smuzhiyun {
224*4882a593Smuzhiyun /* Handle ATTN. Set indication and wake waiters for attention. */
225*4882a593Smuzhiyun if (irb->scsw.cmd.dstat & DEV_STAT_ATTENTION) {
226*4882a593Smuzhiyun fp->attention = 1;
227*4882a593Smuzhiyun wake_up(&fp->wait);
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun
230*4882a593Smuzhiyun if (rq) {
231*4882a593Smuzhiyun if (irb->scsw.cmd.dstat & DEV_STAT_UNIT_CHECK)
232*4882a593Smuzhiyun rq->rc = -EIO;
233*4882a593Smuzhiyun else
234*4882a593Smuzhiyun /* Normal end. Copy residual count. */
235*4882a593Smuzhiyun rq->rescnt = irb->scsw.cmd.count;
236*4882a593Smuzhiyun }
237*4882a593Smuzhiyun }
238*4882a593Smuzhiyun
239*4882a593Smuzhiyun /*
240*4882a593Smuzhiyun * Process reads from fullscreen 3270.
241*4882a593Smuzhiyun */
242*4882a593Smuzhiyun static ssize_t
fs3270_read(struct file * filp,char __user * data,size_t count,loff_t * off)243*4882a593Smuzhiyun fs3270_read(struct file *filp, char __user *data, size_t count, loff_t *off)
244*4882a593Smuzhiyun {
245*4882a593Smuzhiyun struct fs3270 *fp;
246*4882a593Smuzhiyun struct raw3270_request *rq;
247*4882a593Smuzhiyun struct idal_buffer *ib;
248*4882a593Smuzhiyun ssize_t rc;
249*4882a593Smuzhiyun
250*4882a593Smuzhiyun if (count == 0 || count > 65535)
251*4882a593Smuzhiyun return -EINVAL;
252*4882a593Smuzhiyun fp = filp->private_data;
253*4882a593Smuzhiyun if (!fp)
254*4882a593Smuzhiyun return -ENODEV;
255*4882a593Smuzhiyun ib = idal_buffer_alloc(count, 0);
256*4882a593Smuzhiyun if (IS_ERR(ib))
257*4882a593Smuzhiyun return -ENOMEM;
258*4882a593Smuzhiyun rq = raw3270_request_alloc(0);
259*4882a593Smuzhiyun if (!IS_ERR(rq)) {
260*4882a593Smuzhiyun if (fp->read_command == 0 && fp->write_command != 0)
261*4882a593Smuzhiyun fp->read_command = 6;
262*4882a593Smuzhiyun raw3270_request_set_cmd(rq, fp->read_command ? : 2);
263*4882a593Smuzhiyun raw3270_request_set_idal(rq, ib);
264*4882a593Smuzhiyun rc = wait_event_interruptible(fp->wait, fp->attention);
265*4882a593Smuzhiyun fp->attention = 0;
266*4882a593Smuzhiyun if (rc == 0) {
267*4882a593Smuzhiyun rc = fs3270_do_io(&fp->view, rq);
268*4882a593Smuzhiyun if (rc == 0) {
269*4882a593Smuzhiyun count -= rq->rescnt;
270*4882a593Smuzhiyun if (idal_buffer_to_user(ib, data, count) != 0)
271*4882a593Smuzhiyun rc = -EFAULT;
272*4882a593Smuzhiyun else
273*4882a593Smuzhiyun rc = count;
274*4882a593Smuzhiyun
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun }
277*4882a593Smuzhiyun raw3270_request_free(rq);
278*4882a593Smuzhiyun } else
279*4882a593Smuzhiyun rc = PTR_ERR(rq);
280*4882a593Smuzhiyun idal_buffer_free(ib);
281*4882a593Smuzhiyun return rc;
282*4882a593Smuzhiyun }
283*4882a593Smuzhiyun
284*4882a593Smuzhiyun /*
285*4882a593Smuzhiyun * Process writes to fullscreen 3270.
286*4882a593Smuzhiyun */
287*4882a593Smuzhiyun static ssize_t
fs3270_write(struct file * filp,const char __user * data,size_t count,loff_t * off)288*4882a593Smuzhiyun fs3270_write(struct file *filp, const char __user *data, size_t count, loff_t *off)
289*4882a593Smuzhiyun {
290*4882a593Smuzhiyun struct fs3270 *fp;
291*4882a593Smuzhiyun struct raw3270_request *rq;
292*4882a593Smuzhiyun struct idal_buffer *ib;
293*4882a593Smuzhiyun int write_command;
294*4882a593Smuzhiyun ssize_t rc;
295*4882a593Smuzhiyun
296*4882a593Smuzhiyun fp = filp->private_data;
297*4882a593Smuzhiyun if (!fp)
298*4882a593Smuzhiyun return -ENODEV;
299*4882a593Smuzhiyun ib = idal_buffer_alloc(count, 0);
300*4882a593Smuzhiyun if (IS_ERR(ib))
301*4882a593Smuzhiyun return -ENOMEM;
302*4882a593Smuzhiyun rq = raw3270_request_alloc(0);
303*4882a593Smuzhiyun if (!IS_ERR(rq)) {
304*4882a593Smuzhiyun if (idal_buffer_from_user(ib, data, count) == 0) {
305*4882a593Smuzhiyun write_command = fp->write_command ? : 1;
306*4882a593Smuzhiyun if (write_command == 5)
307*4882a593Smuzhiyun write_command = 13;
308*4882a593Smuzhiyun raw3270_request_set_cmd(rq, write_command);
309*4882a593Smuzhiyun raw3270_request_set_idal(rq, ib);
310*4882a593Smuzhiyun rc = fs3270_do_io(&fp->view, rq);
311*4882a593Smuzhiyun if (rc == 0)
312*4882a593Smuzhiyun rc = count - rq->rescnt;
313*4882a593Smuzhiyun } else
314*4882a593Smuzhiyun rc = -EFAULT;
315*4882a593Smuzhiyun raw3270_request_free(rq);
316*4882a593Smuzhiyun } else
317*4882a593Smuzhiyun rc = PTR_ERR(rq);
318*4882a593Smuzhiyun idal_buffer_free(ib);
319*4882a593Smuzhiyun return rc;
320*4882a593Smuzhiyun }
321*4882a593Smuzhiyun
322*4882a593Smuzhiyun /*
323*4882a593Smuzhiyun * process ioctl commands for the tube driver
324*4882a593Smuzhiyun */
325*4882a593Smuzhiyun static long
fs3270_ioctl(struct file * filp,unsigned int cmd,unsigned long arg)326*4882a593Smuzhiyun fs3270_ioctl(struct file *filp, unsigned int cmd, unsigned long arg)
327*4882a593Smuzhiyun {
328*4882a593Smuzhiyun char __user *argp;
329*4882a593Smuzhiyun struct fs3270 *fp;
330*4882a593Smuzhiyun struct raw3270_iocb iocb;
331*4882a593Smuzhiyun int rc;
332*4882a593Smuzhiyun
333*4882a593Smuzhiyun fp = filp->private_data;
334*4882a593Smuzhiyun if (!fp)
335*4882a593Smuzhiyun return -ENODEV;
336*4882a593Smuzhiyun if (is_compat_task())
337*4882a593Smuzhiyun argp = compat_ptr(arg);
338*4882a593Smuzhiyun else
339*4882a593Smuzhiyun argp = (char __user *)arg;
340*4882a593Smuzhiyun rc = 0;
341*4882a593Smuzhiyun mutex_lock(&fs3270_mutex);
342*4882a593Smuzhiyun switch (cmd) {
343*4882a593Smuzhiyun case TUBICMD:
344*4882a593Smuzhiyun fp->read_command = arg;
345*4882a593Smuzhiyun break;
346*4882a593Smuzhiyun case TUBOCMD:
347*4882a593Smuzhiyun fp->write_command = arg;
348*4882a593Smuzhiyun break;
349*4882a593Smuzhiyun case TUBGETI:
350*4882a593Smuzhiyun rc = put_user(fp->read_command, argp);
351*4882a593Smuzhiyun break;
352*4882a593Smuzhiyun case TUBGETO:
353*4882a593Smuzhiyun rc = put_user(fp->write_command, argp);
354*4882a593Smuzhiyun break;
355*4882a593Smuzhiyun case TUBGETMOD:
356*4882a593Smuzhiyun iocb.model = fp->view.model;
357*4882a593Smuzhiyun iocb.line_cnt = fp->view.rows;
358*4882a593Smuzhiyun iocb.col_cnt = fp->view.cols;
359*4882a593Smuzhiyun iocb.pf_cnt = 24;
360*4882a593Smuzhiyun iocb.re_cnt = 20;
361*4882a593Smuzhiyun iocb.map = 0;
362*4882a593Smuzhiyun if (copy_to_user(argp, &iocb, sizeof(struct raw3270_iocb)))
363*4882a593Smuzhiyun rc = -EFAULT;
364*4882a593Smuzhiyun break;
365*4882a593Smuzhiyun }
366*4882a593Smuzhiyun mutex_unlock(&fs3270_mutex);
367*4882a593Smuzhiyun return rc;
368*4882a593Smuzhiyun }
369*4882a593Smuzhiyun
370*4882a593Smuzhiyun /*
371*4882a593Smuzhiyun * Allocate fs3270 structure.
372*4882a593Smuzhiyun */
373*4882a593Smuzhiyun static struct fs3270 *
fs3270_alloc_view(void)374*4882a593Smuzhiyun fs3270_alloc_view(void)
375*4882a593Smuzhiyun {
376*4882a593Smuzhiyun struct fs3270 *fp;
377*4882a593Smuzhiyun
378*4882a593Smuzhiyun fp = kzalloc(sizeof(struct fs3270),GFP_KERNEL);
379*4882a593Smuzhiyun if (!fp)
380*4882a593Smuzhiyun return ERR_PTR(-ENOMEM);
381*4882a593Smuzhiyun fp->init = raw3270_request_alloc(0);
382*4882a593Smuzhiyun if (IS_ERR(fp->init)) {
383*4882a593Smuzhiyun kfree(fp);
384*4882a593Smuzhiyun return ERR_PTR(-ENOMEM);
385*4882a593Smuzhiyun }
386*4882a593Smuzhiyun return fp;
387*4882a593Smuzhiyun }
388*4882a593Smuzhiyun
389*4882a593Smuzhiyun /*
390*4882a593Smuzhiyun * Free fs3270 structure.
391*4882a593Smuzhiyun */
392*4882a593Smuzhiyun static void
fs3270_free_view(struct raw3270_view * view)393*4882a593Smuzhiyun fs3270_free_view(struct raw3270_view *view)
394*4882a593Smuzhiyun {
395*4882a593Smuzhiyun struct fs3270 *fp;
396*4882a593Smuzhiyun
397*4882a593Smuzhiyun fp = (struct fs3270 *) view;
398*4882a593Smuzhiyun if (fp->rdbuf)
399*4882a593Smuzhiyun idal_buffer_free(fp->rdbuf);
400*4882a593Smuzhiyun raw3270_request_free(((struct fs3270 *) view)->init);
401*4882a593Smuzhiyun kfree(view);
402*4882a593Smuzhiyun }
403*4882a593Smuzhiyun
404*4882a593Smuzhiyun /*
405*4882a593Smuzhiyun * Unlink fs3270 data structure from filp.
406*4882a593Smuzhiyun */
407*4882a593Smuzhiyun static void
fs3270_release(struct raw3270_view * view)408*4882a593Smuzhiyun fs3270_release(struct raw3270_view *view)
409*4882a593Smuzhiyun {
410*4882a593Smuzhiyun struct fs3270 *fp;
411*4882a593Smuzhiyun
412*4882a593Smuzhiyun fp = (struct fs3270 *) view;
413*4882a593Smuzhiyun if (fp->fs_pid)
414*4882a593Smuzhiyun kill_pid(fp->fs_pid, SIGHUP, 1);
415*4882a593Smuzhiyun }
416*4882a593Smuzhiyun
417*4882a593Smuzhiyun /* View to a 3270 device. Can be console, tty or fullscreen. */
418*4882a593Smuzhiyun static struct raw3270_fn fs3270_fn = {
419*4882a593Smuzhiyun .activate = fs3270_activate,
420*4882a593Smuzhiyun .deactivate = fs3270_deactivate,
421*4882a593Smuzhiyun .intv = (void *) fs3270_irq,
422*4882a593Smuzhiyun .release = fs3270_release,
423*4882a593Smuzhiyun .free = fs3270_free_view
424*4882a593Smuzhiyun };
425*4882a593Smuzhiyun
426*4882a593Smuzhiyun /*
427*4882a593Smuzhiyun * This routine is called whenever a 3270 fullscreen device is opened.
428*4882a593Smuzhiyun */
429*4882a593Smuzhiyun static int
fs3270_open(struct inode * inode,struct file * filp)430*4882a593Smuzhiyun fs3270_open(struct inode *inode, struct file *filp)
431*4882a593Smuzhiyun {
432*4882a593Smuzhiyun struct fs3270 *fp;
433*4882a593Smuzhiyun struct idal_buffer *ib;
434*4882a593Smuzhiyun int minor, rc = 0;
435*4882a593Smuzhiyun
436*4882a593Smuzhiyun if (imajor(file_inode(filp)) != IBM_FS3270_MAJOR)
437*4882a593Smuzhiyun return -ENODEV;
438*4882a593Smuzhiyun minor = iminor(file_inode(filp));
439*4882a593Smuzhiyun /* Check for minor 0 multiplexer. */
440*4882a593Smuzhiyun if (minor == 0) {
441*4882a593Smuzhiyun struct tty_struct *tty = get_current_tty();
442*4882a593Smuzhiyun if (!tty || tty->driver->major != IBM_TTY3270_MAJOR) {
443*4882a593Smuzhiyun tty_kref_put(tty);
444*4882a593Smuzhiyun return -ENODEV;
445*4882a593Smuzhiyun }
446*4882a593Smuzhiyun minor = tty->index;
447*4882a593Smuzhiyun tty_kref_put(tty);
448*4882a593Smuzhiyun }
449*4882a593Smuzhiyun mutex_lock(&fs3270_mutex);
450*4882a593Smuzhiyun /* Check if some other program is already using fullscreen mode. */
451*4882a593Smuzhiyun fp = (struct fs3270 *) raw3270_find_view(&fs3270_fn, minor);
452*4882a593Smuzhiyun if (!IS_ERR(fp)) {
453*4882a593Smuzhiyun raw3270_put_view(&fp->view);
454*4882a593Smuzhiyun rc = -EBUSY;
455*4882a593Smuzhiyun goto out;
456*4882a593Smuzhiyun }
457*4882a593Smuzhiyun /* Allocate fullscreen view structure. */
458*4882a593Smuzhiyun fp = fs3270_alloc_view();
459*4882a593Smuzhiyun if (IS_ERR(fp)) {
460*4882a593Smuzhiyun rc = PTR_ERR(fp);
461*4882a593Smuzhiyun goto out;
462*4882a593Smuzhiyun }
463*4882a593Smuzhiyun
464*4882a593Smuzhiyun init_waitqueue_head(&fp->wait);
465*4882a593Smuzhiyun fp->fs_pid = get_pid(task_pid(current));
466*4882a593Smuzhiyun rc = raw3270_add_view(&fp->view, &fs3270_fn, minor,
467*4882a593Smuzhiyun RAW3270_VIEW_LOCK_BH);
468*4882a593Smuzhiyun if (rc) {
469*4882a593Smuzhiyun fs3270_free_view(&fp->view);
470*4882a593Smuzhiyun goto out;
471*4882a593Smuzhiyun }
472*4882a593Smuzhiyun
473*4882a593Smuzhiyun /* Allocate idal-buffer. */
474*4882a593Smuzhiyun ib = idal_buffer_alloc(2*fp->view.rows*fp->view.cols + 5, 0);
475*4882a593Smuzhiyun if (IS_ERR(ib)) {
476*4882a593Smuzhiyun raw3270_put_view(&fp->view);
477*4882a593Smuzhiyun raw3270_del_view(&fp->view);
478*4882a593Smuzhiyun rc = PTR_ERR(ib);
479*4882a593Smuzhiyun goto out;
480*4882a593Smuzhiyun }
481*4882a593Smuzhiyun fp->rdbuf = ib;
482*4882a593Smuzhiyun
483*4882a593Smuzhiyun rc = raw3270_activate_view(&fp->view);
484*4882a593Smuzhiyun if (rc) {
485*4882a593Smuzhiyun raw3270_put_view(&fp->view);
486*4882a593Smuzhiyun raw3270_del_view(&fp->view);
487*4882a593Smuzhiyun goto out;
488*4882a593Smuzhiyun }
489*4882a593Smuzhiyun stream_open(inode, filp);
490*4882a593Smuzhiyun filp->private_data = fp;
491*4882a593Smuzhiyun out:
492*4882a593Smuzhiyun mutex_unlock(&fs3270_mutex);
493*4882a593Smuzhiyun return rc;
494*4882a593Smuzhiyun }
495*4882a593Smuzhiyun
496*4882a593Smuzhiyun /*
497*4882a593Smuzhiyun * This routine is called when the 3270 tty is closed. We wait
498*4882a593Smuzhiyun * for the remaining request to be completed. Then we clean up.
499*4882a593Smuzhiyun */
500*4882a593Smuzhiyun static int
fs3270_close(struct inode * inode,struct file * filp)501*4882a593Smuzhiyun fs3270_close(struct inode *inode, struct file *filp)
502*4882a593Smuzhiyun {
503*4882a593Smuzhiyun struct fs3270 *fp;
504*4882a593Smuzhiyun
505*4882a593Smuzhiyun fp = filp->private_data;
506*4882a593Smuzhiyun filp->private_data = NULL;
507*4882a593Smuzhiyun if (fp) {
508*4882a593Smuzhiyun put_pid(fp->fs_pid);
509*4882a593Smuzhiyun fp->fs_pid = NULL;
510*4882a593Smuzhiyun raw3270_reset(&fp->view);
511*4882a593Smuzhiyun raw3270_put_view(&fp->view);
512*4882a593Smuzhiyun raw3270_del_view(&fp->view);
513*4882a593Smuzhiyun }
514*4882a593Smuzhiyun return 0;
515*4882a593Smuzhiyun }
516*4882a593Smuzhiyun
517*4882a593Smuzhiyun static const struct file_operations fs3270_fops = {
518*4882a593Smuzhiyun .owner = THIS_MODULE, /* owner */
519*4882a593Smuzhiyun .read = fs3270_read, /* read */
520*4882a593Smuzhiyun .write = fs3270_write, /* write */
521*4882a593Smuzhiyun .unlocked_ioctl = fs3270_ioctl, /* ioctl */
522*4882a593Smuzhiyun .compat_ioctl = fs3270_ioctl, /* ioctl */
523*4882a593Smuzhiyun .open = fs3270_open, /* open */
524*4882a593Smuzhiyun .release = fs3270_close, /* release */
525*4882a593Smuzhiyun .llseek = no_llseek,
526*4882a593Smuzhiyun };
527*4882a593Smuzhiyun
fs3270_create_cb(int minor)528*4882a593Smuzhiyun static void fs3270_create_cb(int minor)
529*4882a593Smuzhiyun {
530*4882a593Smuzhiyun __register_chrdev(IBM_FS3270_MAJOR, minor, 1, "tub", &fs3270_fops);
531*4882a593Smuzhiyun device_create(class3270, NULL, MKDEV(IBM_FS3270_MAJOR, minor),
532*4882a593Smuzhiyun NULL, "3270/tub%d", minor);
533*4882a593Smuzhiyun }
534*4882a593Smuzhiyun
fs3270_destroy_cb(int minor)535*4882a593Smuzhiyun static void fs3270_destroy_cb(int minor)
536*4882a593Smuzhiyun {
537*4882a593Smuzhiyun device_destroy(class3270, MKDEV(IBM_FS3270_MAJOR, minor));
538*4882a593Smuzhiyun __unregister_chrdev(IBM_FS3270_MAJOR, minor, 1, "tub");
539*4882a593Smuzhiyun }
540*4882a593Smuzhiyun
541*4882a593Smuzhiyun static struct raw3270_notifier fs3270_notifier =
542*4882a593Smuzhiyun {
543*4882a593Smuzhiyun .create = fs3270_create_cb,
544*4882a593Smuzhiyun .destroy = fs3270_destroy_cb,
545*4882a593Smuzhiyun };
546*4882a593Smuzhiyun
547*4882a593Smuzhiyun /*
548*4882a593Smuzhiyun * 3270 fullscreen driver initialization.
549*4882a593Smuzhiyun */
550*4882a593Smuzhiyun static int __init
fs3270_init(void)551*4882a593Smuzhiyun fs3270_init(void)
552*4882a593Smuzhiyun {
553*4882a593Smuzhiyun int rc;
554*4882a593Smuzhiyun
555*4882a593Smuzhiyun rc = __register_chrdev(IBM_FS3270_MAJOR, 0, 1, "fs3270", &fs3270_fops);
556*4882a593Smuzhiyun if (rc)
557*4882a593Smuzhiyun return rc;
558*4882a593Smuzhiyun device_create(class3270, NULL, MKDEV(IBM_FS3270_MAJOR, 0),
559*4882a593Smuzhiyun NULL, "3270/tub");
560*4882a593Smuzhiyun raw3270_register_notifier(&fs3270_notifier);
561*4882a593Smuzhiyun return 0;
562*4882a593Smuzhiyun }
563*4882a593Smuzhiyun
564*4882a593Smuzhiyun static void __exit
fs3270_exit(void)565*4882a593Smuzhiyun fs3270_exit(void)
566*4882a593Smuzhiyun {
567*4882a593Smuzhiyun raw3270_unregister_notifier(&fs3270_notifier);
568*4882a593Smuzhiyun device_destroy(class3270, MKDEV(IBM_FS3270_MAJOR, 0));
569*4882a593Smuzhiyun __unregister_chrdev(IBM_FS3270_MAJOR, 0, 1, "fs3270");
570*4882a593Smuzhiyun }
571*4882a593Smuzhiyun
572*4882a593Smuzhiyun MODULE_LICENSE("GPL");
573*4882a593Smuzhiyun MODULE_ALIAS_CHARDEV_MAJOR(IBM_FS3270_MAJOR);
574*4882a593Smuzhiyun
575*4882a593Smuzhiyun module_init(fs3270_init);
576*4882a593Smuzhiyun module_exit(fs3270_exit);
577