xref: /openbmc/linux/drivers/s390/cio/vfio_ccw_fsm.c (revision 6c02ac4c9211edabe17bda437ac97e578756f31b)
1b2441318SGreg Kroah-Hartman // SPDX-License-Identifier: GPL-2.0
2bbe37e4cSDong Jia Shi /*
3bbe37e4cSDong Jia Shi  * Finite state machine for vfio-ccw device handling
4bbe37e4cSDong Jia Shi  *
5bbe37e4cSDong Jia Shi  * Copyright IBM Corp. 2017
6d5afd5d1SCornelia Huck  * Copyright Red Hat, Inc. 2019
7bbe37e4cSDong Jia Shi  *
8bbe37e4cSDong Jia Shi  * Author(s): Dong Jia Shi <bjsdjshi@linux.vnet.ibm.com>
9d5afd5d1SCornelia Huck  *            Cornelia Huck <cohuck@redhat.com>
10bbe37e4cSDong Jia Shi  */
11bbe37e4cSDong Jia Shi 
12bbe37e4cSDong Jia Shi #include <linux/vfio.h>
13bbe37e4cSDong Jia Shi #include <linux/mdev.h>
14bbe37e4cSDong Jia Shi 
15bbe37e4cSDong Jia Shi #include "ioasm.h"
16bbe37e4cSDong Jia Shi #include "vfio_ccw_private.h"
17bbe37e4cSDong Jia Shi 
18bbe37e4cSDong Jia Shi static int fsm_io_helper(struct vfio_ccw_private *private)
19bbe37e4cSDong Jia Shi {
20bbe37e4cSDong Jia Shi 	struct subchannel *sch;
21bbe37e4cSDong Jia Shi 	union orb *orb;
22bbe37e4cSDong Jia Shi 	int ccode;
23bbe37e4cSDong Jia Shi 	__u8 lpm;
24bbe37e4cSDong Jia Shi 	unsigned long flags;
253368e547SCornelia Huck 	int ret;
26bbe37e4cSDong Jia Shi 
27bbe37e4cSDong Jia Shi 	sch = private->sch;
28bbe37e4cSDong Jia Shi 
29bbe37e4cSDong Jia Shi 	spin_lock_irqsave(sch->lock, flags);
30bbe37e4cSDong Jia Shi 
31bbe37e4cSDong Jia Shi 	orb = cp_get_orb(&private->cp, (u32)(addr_t)sch, sch->lpm);
3271189f26SCornelia Huck 	if (!orb) {
3371189f26SCornelia Huck 		ret = -EIO;
3471189f26SCornelia Huck 		goto out;
3571189f26SCornelia Huck 	}
36bbe37e4cSDong Jia Shi 
3760e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(5, "stIO");
3860e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(5, dev_name(&sch->dev));
3960e05d1cSCornelia Huck 
40bbe37e4cSDong Jia Shi 	/* Issue "Start Subchannel" */
41bbe37e4cSDong Jia Shi 	ccode = ssch(sch->schid, orb);
42bbe37e4cSDong Jia Shi 
4360e05d1cSCornelia Huck 	VFIO_CCW_HEX_EVENT(5, &ccode, sizeof(ccode));
4460e05d1cSCornelia Huck 
45bbe37e4cSDong Jia Shi 	switch (ccode) {
46bbe37e4cSDong Jia Shi 	case 0:
47bbe37e4cSDong Jia Shi 		/*
48bbe37e4cSDong Jia Shi 		 * Initialize device status information
49bbe37e4cSDong Jia Shi 		 */
50bbe37e4cSDong Jia Shi 		sch->schib.scsw.cmd.actl |= SCSW_ACTL_START_PEND;
513368e547SCornelia Huck 		ret = 0;
52690f6a15SCornelia Huck 		private->state = VFIO_CCW_STATE_CP_PENDING;
533368e547SCornelia Huck 		break;
54bbe37e4cSDong Jia Shi 	case 1:		/* Status pending */
55bbe37e4cSDong Jia Shi 	case 2:		/* Busy */
563368e547SCornelia Huck 		ret = -EBUSY;
573368e547SCornelia Huck 		break;
58bbe37e4cSDong Jia Shi 	case 3:		/* Device/path not operational */
59bbe37e4cSDong Jia Shi 	{
60bbe37e4cSDong Jia Shi 		lpm = orb->cmd.lpm;
61bbe37e4cSDong Jia Shi 		if (lpm != 0)
62bbe37e4cSDong Jia Shi 			sch->lpm &= ~lpm;
63bbe37e4cSDong Jia Shi 		else
64bbe37e4cSDong Jia Shi 			sch->lpm = 0;
65bbe37e4cSDong Jia Shi 
66bbe37e4cSDong Jia Shi 		if (cio_update_schib(sch))
673368e547SCornelia Huck 			ret = -ENODEV;
683368e547SCornelia Huck 		else
693368e547SCornelia Huck 			ret = sch->lpm ? -EACCES : -ENODEV;
703368e547SCornelia Huck 		break;
71bbe37e4cSDong Jia Shi 	}
72bbe37e4cSDong Jia Shi 	default:
733368e547SCornelia Huck 		ret = ccode;
74bbe37e4cSDong Jia Shi 	}
7571189f26SCornelia Huck out:
763368e547SCornelia Huck 	spin_unlock_irqrestore(sch->lock, flags);
773368e547SCornelia Huck 	return ret;
78bbe37e4cSDong Jia Shi }
79bbe37e4cSDong Jia Shi 
80d5afd5d1SCornelia Huck static int fsm_do_halt(struct vfio_ccw_private *private)
81d5afd5d1SCornelia Huck {
82d5afd5d1SCornelia Huck 	struct subchannel *sch;
83d5afd5d1SCornelia Huck 	unsigned long flags;
84d5afd5d1SCornelia Huck 	int ccode;
85d5afd5d1SCornelia Huck 	int ret;
86d5afd5d1SCornelia Huck 
87d5afd5d1SCornelia Huck 	sch = private->sch;
88d5afd5d1SCornelia Huck 
89d5afd5d1SCornelia Huck 	spin_lock_irqsave(sch->lock, flags);
90d5afd5d1SCornelia Huck 
9160e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, "haltIO");
9260e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev));
9360e05d1cSCornelia Huck 
94d5afd5d1SCornelia Huck 	/* Issue "Halt Subchannel" */
95d5afd5d1SCornelia Huck 	ccode = hsch(sch->schid);
96d5afd5d1SCornelia Huck 
9760e05d1cSCornelia Huck 	VFIO_CCW_HEX_EVENT(2, &ccode, sizeof(ccode));
9860e05d1cSCornelia Huck 
99d5afd5d1SCornelia Huck 	switch (ccode) {
100d5afd5d1SCornelia Huck 	case 0:
101d5afd5d1SCornelia Huck 		/*
102d5afd5d1SCornelia Huck 		 * Initialize device status information
103d5afd5d1SCornelia Huck 		 */
104d5afd5d1SCornelia Huck 		sch->schib.scsw.cmd.actl |= SCSW_ACTL_HALT_PEND;
105d5afd5d1SCornelia Huck 		ret = 0;
106d5afd5d1SCornelia Huck 		break;
107d5afd5d1SCornelia Huck 	case 1:		/* Status pending */
108d5afd5d1SCornelia Huck 	case 2:		/* Busy */
109d5afd5d1SCornelia Huck 		ret = -EBUSY;
110d5afd5d1SCornelia Huck 		break;
111d5afd5d1SCornelia Huck 	case 3:		/* Device not operational */
112d5afd5d1SCornelia Huck 		ret = -ENODEV;
113d5afd5d1SCornelia Huck 		break;
114d5afd5d1SCornelia Huck 	default:
115d5afd5d1SCornelia Huck 		ret = ccode;
116d5afd5d1SCornelia Huck 	}
117d5afd5d1SCornelia Huck 	spin_unlock_irqrestore(sch->lock, flags);
118d5afd5d1SCornelia Huck 	return ret;
119d5afd5d1SCornelia Huck }
120d5afd5d1SCornelia Huck 
121d5afd5d1SCornelia Huck static int fsm_do_clear(struct vfio_ccw_private *private)
122d5afd5d1SCornelia Huck {
123d5afd5d1SCornelia Huck 	struct subchannel *sch;
124d5afd5d1SCornelia Huck 	unsigned long flags;
125d5afd5d1SCornelia Huck 	int ccode;
126d5afd5d1SCornelia Huck 	int ret;
127d5afd5d1SCornelia Huck 
128d5afd5d1SCornelia Huck 	sch = private->sch;
129d5afd5d1SCornelia Huck 
130d5afd5d1SCornelia Huck 	spin_lock_irqsave(sch->lock, flags);
131d5afd5d1SCornelia Huck 
13260e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, "clearIO");
13360e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev));
13460e05d1cSCornelia Huck 
135d5afd5d1SCornelia Huck 	/* Issue "Clear Subchannel" */
136d5afd5d1SCornelia Huck 	ccode = csch(sch->schid);
137d5afd5d1SCornelia Huck 
13860e05d1cSCornelia Huck 	VFIO_CCW_HEX_EVENT(2, &ccode, sizeof(ccode));
13960e05d1cSCornelia Huck 
140d5afd5d1SCornelia Huck 	switch (ccode) {
141d5afd5d1SCornelia Huck 	case 0:
142d5afd5d1SCornelia Huck 		/*
143d5afd5d1SCornelia Huck 		 * Initialize device status information
144d5afd5d1SCornelia Huck 		 */
145d5afd5d1SCornelia Huck 		sch->schib.scsw.cmd.actl = SCSW_ACTL_CLEAR_PEND;
146d5afd5d1SCornelia Huck 		/* TODO: check what else we might need to clear */
147d5afd5d1SCornelia Huck 		ret = 0;
148d5afd5d1SCornelia Huck 		break;
149d5afd5d1SCornelia Huck 	case 3:		/* Device not operational */
150d5afd5d1SCornelia Huck 		ret = -ENODEV;
151d5afd5d1SCornelia Huck 		break;
152d5afd5d1SCornelia Huck 	default:
153d5afd5d1SCornelia Huck 		ret = ccode;
154d5afd5d1SCornelia Huck 	}
155d5afd5d1SCornelia Huck 	spin_unlock_irqrestore(sch->lock, flags);
156d5afd5d1SCornelia Huck 	return ret;
157d5afd5d1SCornelia Huck }
158d5afd5d1SCornelia Huck 
159bbe37e4cSDong Jia Shi static void fsm_notoper(struct vfio_ccw_private *private,
160bbe37e4cSDong Jia Shi 			enum vfio_ccw_event event)
161bbe37e4cSDong Jia Shi {
162bbe37e4cSDong Jia Shi 	struct subchannel *sch = private->sch;
163bbe37e4cSDong Jia Shi 
16460e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, "notoper");
16560e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev));
16660e05d1cSCornelia Huck 
167bbe37e4cSDong Jia Shi 	/*
168bbe37e4cSDong Jia Shi 	 * TODO:
169bbe37e4cSDong Jia Shi 	 * Probably we should send the machine check to the guest.
170bbe37e4cSDong Jia Shi 	 */
171bbe37e4cSDong Jia Shi 	css_sched_sch_todo(sch, SCH_TODO_UNREG);
172bbe37e4cSDong Jia Shi 	private->state = VFIO_CCW_STATE_NOT_OPER;
173bbe37e4cSDong Jia Shi }
174bbe37e4cSDong Jia Shi 
175bbe37e4cSDong Jia Shi /*
176bbe37e4cSDong Jia Shi  * No operation action.
177bbe37e4cSDong Jia Shi  */
178bbe37e4cSDong Jia Shi static void fsm_nop(struct vfio_ccw_private *private,
179bbe37e4cSDong Jia Shi 		    enum vfio_ccw_event event)
180bbe37e4cSDong Jia Shi {
181bbe37e4cSDong Jia Shi }
182bbe37e4cSDong Jia Shi 
183bbe37e4cSDong Jia Shi static void fsm_io_error(struct vfio_ccw_private *private,
184bbe37e4cSDong Jia Shi 			 enum vfio_ccw_event event)
185bbe37e4cSDong Jia Shi {
186bbe37e4cSDong Jia Shi 	pr_err("vfio-ccw: FSM: I/O request from state:%d\n", private->state);
187c98e16b2SEric Farman 	private->io_region->ret_code = -EIO;
188bbe37e4cSDong Jia Shi }
189bbe37e4cSDong Jia Shi 
190bbe37e4cSDong Jia Shi static void fsm_io_busy(struct vfio_ccw_private *private,
191bbe37e4cSDong Jia Shi 			enum vfio_ccw_event event)
192bbe37e4cSDong Jia Shi {
193c98e16b2SEric Farman 	private->io_region->ret_code = -EBUSY;
194bbe37e4cSDong Jia Shi }
195bbe37e4cSDong Jia Shi 
196690f6a15SCornelia Huck static void fsm_io_retry(struct vfio_ccw_private *private,
197690f6a15SCornelia Huck 			 enum vfio_ccw_event event)
198690f6a15SCornelia Huck {
199690f6a15SCornelia Huck 	private->io_region->ret_code = -EAGAIN;
200690f6a15SCornelia Huck }
201690f6a15SCornelia Huck 
202d5afd5d1SCornelia Huck static void fsm_async_error(struct vfio_ccw_private *private,
203d5afd5d1SCornelia Huck 			    enum vfio_ccw_event event)
204d5afd5d1SCornelia Huck {
205d5afd5d1SCornelia Huck 	struct ccw_cmd_region *cmd_region = private->cmd_region;
206d5afd5d1SCornelia Huck 
207d5afd5d1SCornelia Huck 	pr_err("vfio-ccw: FSM: %s request from state:%d\n",
208d5afd5d1SCornelia Huck 	       cmd_region->command == VFIO_CCW_ASYNC_CMD_HSCH ? "halt" :
209d5afd5d1SCornelia Huck 	       cmd_region->command == VFIO_CCW_ASYNC_CMD_CSCH ? "clear" :
210d5afd5d1SCornelia Huck 	       "<unknown>", private->state);
211d5afd5d1SCornelia Huck 	cmd_region->ret_code = -EIO;
212d5afd5d1SCornelia Huck }
213d5afd5d1SCornelia Huck 
214d5afd5d1SCornelia Huck static void fsm_async_retry(struct vfio_ccw_private *private,
215d5afd5d1SCornelia Huck 			    enum vfio_ccw_event event)
216d5afd5d1SCornelia Huck {
217d5afd5d1SCornelia Huck 	private->cmd_region->ret_code = -EAGAIN;
218d5afd5d1SCornelia Huck }
219d5afd5d1SCornelia Huck 
220bbe37e4cSDong Jia Shi static void fsm_disabled_irq(struct vfio_ccw_private *private,
221bbe37e4cSDong Jia Shi 			     enum vfio_ccw_event event)
222bbe37e4cSDong Jia Shi {
223bbe37e4cSDong Jia Shi 	struct subchannel *sch = private->sch;
224bbe37e4cSDong Jia Shi 
225bbe37e4cSDong Jia Shi 	/*
226bbe37e4cSDong Jia Shi 	 * An interrupt in a disabled state means a previous disable was not
227bbe37e4cSDong Jia Shi 	 * successful - should not happen, but we try to disable again.
228bbe37e4cSDong Jia Shi 	 */
229bbe37e4cSDong Jia Shi 	cio_disable_subchannel(sch);
230bbe37e4cSDong Jia Shi }
2313cd90214SHalil Pasic inline struct subchannel_id get_schid(struct vfio_ccw_private *p)
2323cd90214SHalil Pasic {
2333cd90214SHalil Pasic 	return p->sch->schid;
2343cd90214SHalil Pasic }
235bbe37e4cSDong Jia Shi 
236bbe37e4cSDong Jia Shi /*
237bbe37e4cSDong Jia Shi  * Deal with the ccw command request from the userspace.
238bbe37e4cSDong Jia Shi  */
239bbe37e4cSDong Jia Shi static void fsm_io_request(struct vfio_ccw_private *private,
240bbe37e4cSDong Jia Shi 			   enum vfio_ccw_event event)
241bbe37e4cSDong Jia Shi {
242bbe37e4cSDong Jia Shi 	union orb *orb;
243bbe37e4cSDong Jia Shi 	union scsw *scsw = &private->scsw;
244c98e16b2SEric Farman 	struct ccw_io_region *io_region = private->io_region;
245bbe37e4cSDong Jia Shi 	struct mdev_device *mdev = private->mdev;
2463cd90214SHalil Pasic 	char *errstr = "request";
24760e05d1cSCornelia Huck 	struct subchannel_id schid = get_schid(private);
248bbe37e4cSDong Jia Shi 
249690f6a15SCornelia Huck 	private->state = VFIO_CCW_STATE_CP_PROCESSING;
250bbe37e4cSDong Jia Shi 	memcpy(scsw, io_region->scsw_area, sizeof(*scsw));
251bbe37e4cSDong Jia Shi 
252bbe37e4cSDong Jia Shi 	if (scsw->cmd.fctl & SCSW_FCTL_START_FUNC) {
253bbe37e4cSDong Jia Shi 		orb = (union orb *)io_region->orb_area;
254bbe37e4cSDong Jia Shi 
2559851bc77SCornelia Huck 		/* Don't try to build a cp if transport mode is specified. */
2569851bc77SCornelia Huck 		if (orb->tm.b) {
2579851bc77SCornelia Huck 			io_region->ret_code = -EOPNOTSUPP;
25860e05d1cSCornelia Huck 			VFIO_CCW_MSG_EVENT(2,
25960e05d1cSCornelia Huck 					   "%pUl (%x.%x.%04x): transport mode\n",
26060e05d1cSCornelia Huck 					   mdev_uuid(mdev), schid.cssid,
26160e05d1cSCornelia Huck 					   schid.ssid, schid.sch_no);
2623cd90214SHalil Pasic 			errstr = "transport mode";
2639851bc77SCornelia Huck 			goto err_out;
2649851bc77SCornelia Huck 		}
265bbe37e4cSDong Jia Shi 		io_region->ret_code = cp_init(&private->cp, mdev_dev(mdev),
266bbe37e4cSDong Jia Shi 					      orb);
2673cd90214SHalil Pasic 		if (io_region->ret_code) {
26860e05d1cSCornelia Huck 			VFIO_CCW_MSG_EVENT(2,
26960e05d1cSCornelia Huck 					   "%pUl (%x.%x.%04x): cp_init=%d\n",
27060e05d1cSCornelia Huck 					   mdev_uuid(mdev), schid.cssid,
27160e05d1cSCornelia Huck 					   schid.ssid, schid.sch_no,
27260e05d1cSCornelia Huck 					   io_region->ret_code);
2733cd90214SHalil Pasic 			errstr = "cp init";
274bbe37e4cSDong Jia Shi 			goto err_out;
2753cd90214SHalil Pasic 		}
276bbe37e4cSDong Jia Shi 
277bbe37e4cSDong Jia Shi 		io_region->ret_code = cp_prefetch(&private->cp);
278bbe37e4cSDong Jia Shi 		if (io_region->ret_code) {
27960e05d1cSCornelia Huck 			VFIO_CCW_MSG_EVENT(2,
28060e05d1cSCornelia Huck 					   "%pUl (%x.%x.%04x): cp_prefetch=%d\n",
28160e05d1cSCornelia Huck 					   mdev_uuid(mdev), schid.cssid,
28260e05d1cSCornelia Huck 					   schid.ssid, schid.sch_no,
28360e05d1cSCornelia Huck 					   io_region->ret_code);
2843cd90214SHalil Pasic 			errstr = "cp prefetch";
285bbe37e4cSDong Jia Shi 			cp_free(&private->cp);
286bbe37e4cSDong Jia Shi 			goto err_out;
287bbe37e4cSDong Jia Shi 		}
288bbe37e4cSDong Jia Shi 
289bbe37e4cSDong Jia Shi 		/* Start channel program and wait for I/O interrupt. */
290bbe37e4cSDong Jia Shi 		io_region->ret_code = fsm_io_helper(private);
291bbe37e4cSDong Jia Shi 		if (io_region->ret_code) {
29260e05d1cSCornelia Huck 			VFIO_CCW_MSG_EVENT(2,
29360e05d1cSCornelia Huck 					   "%pUl (%x.%x.%04x): fsm_io_helper=%d\n",
29460e05d1cSCornelia Huck 					   mdev_uuid(mdev), schid.cssid,
29560e05d1cSCornelia Huck 					   schid.ssid, schid.sch_no,
29660e05d1cSCornelia Huck 					   io_region->ret_code);
2973cd90214SHalil Pasic 			errstr = "cp fsm_io_helper";
298bbe37e4cSDong Jia Shi 			cp_free(&private->cp);
299bbe37e4cSDong Jia Shi 			goto err_out;
300bbe37e4cSDong Jia Shi 		}
301bbe37e4cSDong Jia Shi 		return;
302bbe37e4cSDong Jia Shi 	} else if (scsw->cmd.fctl & SCSW_FCTL_HALT_FUNC) {
30360e05d1cSCornelia Huck 		VFIO_CCW_MSG_EVENT(2,
30460e05d1cSCornelia Huck 				   "%pUl (%x.%x.%04x): halt on io_region\n",
30560e05d1cSCornelia Huck 				   mdev_uuid(mdev), schid.cssid,
30660e05d1cSCornelia Huck 				   schid.ssid, schid.sch_no);
307d5afd5d1SCornelia Huck 		/* halt is handled via the async cmd region */
308bbe37e4cSDong Jia Shi 		io_region->ret_code = -EOPNOTSUPP;
309bbe37e4cSDong Jia Shi 		goto err_out;
310bbe37e4cSDong Jia Shi 	} else if (scsw->cmd.fctl & SCSW_FCTL_CLEAR_FUNC) {
31160e05d1cSCornelia Huck 		VFIO_CCW_MSG_EVENT(2,
31260e05d1cSCornelia Huck 				   "%pUl (%x.%x.%04x): clear on io_region\n",
31360e05d1cSCornelia Huck 				   mdev_uuid(mdev), schid.cssid,
31460e05d1cSCornelia Huck 				   schid.ssid, schid.sch_no);
315d5afd5d1SCornelia Huck 		/* clear is handled via the async cmd region */
316bbe37e4cSDong Jia Shi 		io_region->ret_code = -EOPNOTSUPP;
317bbe37e4cSDong Jia Shi 		goto err_out;
318bbe37e4cSDong Jia Shi 	}
319bbe37e4cSDong Jia Shi 
320bbe37e4cSDong Jia Shi err_out:
321*6c02ac4cSEric Farman 	private->state = VFIO_CCW_STATE_IDLE;
32285298880SEric Farman 	trace_vfio_ccw_fsm_io_request(scsw->cmd.fctl, schid,
3233cd90214SHalil Pasic 				      io_region->ret_code, errstr);
324bbe37e4cSDong Jia Shi }
325bbe37e4cSDong Jia Shi 
326bbe37e4cSDong Jia Shi /*
327d5afd5d1SCornelia Huck  * Deal with an async request from userspace.
328d5afd5d1SCornelia Huck  */
329d5afd5d1SCornelia Huck static void fsm_async_request(struct vfio_ccw_private *private,
330d5afd5d1SCornelia Huck 			      enum vfio_ccw_event event)
331d5afd5d1SCornelia Huck {
332d5afd5d1SCornelia Huck 	struct ccw_cmd_region *cmd_region = private->cmd_region;
333d5afd5d1SCornelia Huck 
334d5afd5d1SCornelia Huck 	switch (cmd_region->command) {
335d5afd5d1SCornelia Huck 	case VFIO_CCW_ASYNC_CMD_HSCH:
336d5afd5d1SCornelia Huck 		cmd_region->ret_code = fsm_do_halt(private);
337d5afd5d1SCornelia Huck 		break;
338d5afd5d1SCornelia Huck 	case VFIO_CCW_ASYNC_CMD_CSCH:
339d5afd5d1SCornelia Huck 		cmd_region->ret_code = fsm_do_clear(private);
340d5afd5d1SCornelia Huck 		break;
341d5afd5d1SCornelia Huck 	default:
342d5afd5d1SCornelia Huck 		/* should not happen? */
343d5afd5d1SCornelia Huck 		cmd_region->ret_code = -EINVAL;
344d5afd5d1SCornelia Huck 	}
345d5950b02SEric Farman 
346d5950b02SEric Farman 	trace_vfio_ccw_fsm_async_request(get_schid(private),
347d5950b02SEric Farman 					 cmd_region->command,
348d5950b02SEric Farman 					 cmd_region->ret_code);
349d5afd5d1SCornelia Huck }
350d5afd5d1SCornelia Huck 
351d5afd5d1SCornelia Huck /*
352bbe37e4cSDong Jia Shi  * Got an interrupt for a normal io (state busy).
353bbe37e4cSDong Jia Shi  */
354bbe37e4cSDong Jia Shi static void fsm_irq(struct vfio_ccw_private *private,
355bbe37e4cSDong Jia Shi 		    enum vfio_ccw_event event)
356bbe37e4cSDong Jia Shi {
357c9c31b07SDong Jia Shi 	struct irb *irb = this_cpu_ptr(&cio_irb);
358bbe37e4cSDong Jia Shi 
35960e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(6, "IRQ");
36060e05d1cSCornelia Huck 	VFIO_CCW_TRACE_EVENT(6, dev_name(&private->sch->dev));
36160e05d1cSCornelia Huck 
362bbe37e4cSDong Jia Shi 	memcpy(&private->irb, irb, sizeof(*irb));
363bbe37e4cSDong Jia Shi 
364bbe37e4cSDong Jia Shi 	queue_work(vfio_ccw_work_q, &private->io_work);
365bbe37e4cSDong Jia Shi 
366bbe37e4cSDong Jia Shi 	if (private->completion)
367bbe37e4cSDong Jia Shi 		complete(private->completion);
368bbe37e4cSDong Jia Shi }
369bbe37e4cSDong Jia Shi 
370bbe37e4cSDong Jia Shi /*
371bbe37e4cSDong Jia Shi  * Device statemachine
372bbe37e4cSDong Jia Shi  */
373bbe37e4cSDong Jia Shi fsm_func_t *vfio_ccw_jumptable[NR_VFIO_CCW_STATES][NR_VFIO_CCW_EVENTS] = {
374bbe37e4cSDong Jia Shi 	[VFIO_CCW_STATE_NOT_OPER] = {
375bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_NOT_OPER]	= fsm_nop,
376bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_IO_REQ]		= fsm_io_error,
377d5afd5d1SCornelia Huck 		[VFIO_CCW_EVENT_ASYNC_REQ]	= fsm_async_error,
378bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_INTERRUPT]	= fsm_disabled_irq,
379bbe37e4cSDong Jia Shi 	},
380bbe37e4cSDong Jia Shi 	[VFIO_CCW_STATE_STANDBY] = {
381bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_NOT_OPER]	= fsm_notoper,
382bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_IO_REQ]		= fsm_io_error,
383d5afd5d1SCornelia Huck 		[VFIO_CCW_EVENT_ASYNC_REQ]	= fsm_async_error,
384bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_INTERRUPT]	= fsm_irq,
385bbe37e4cSDong Jia Shi 	},
386bbe37e4cSDong Jia Shi 	[VFIO_CCW_STATE_IDLE] = {
387bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_NOT_OPER]	= fsm_notoper,
388bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_IO_REQ]		= fsm_io_request,
389d5afd5d1SCornelia Huck 		[VFIO_CCW_EVENT_ASYNC_REQ]	= fsm_async_request,
390bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_INTERRUPT]	= fsm_irq,
391bbe37e4cSDong Jia Shi 	},
392690f6a15SCornelia Huck 	[VFIO_CCW_STATE_CP_PROCESSING] = {
393690f6a15SCornelia Huck 		[VFIO_CCW_EVENT_NOT_OPER]	= fsm_notoper,
394690f6a15SCornelia Huck 		[VFIO_CCW_EVENT_IO_REQ]		= fsm_io_retry,
395d5afd5d1SCornelia Huck 		[VFIO_CCW_EVENT_ASYNC_REQ]	= fsm_async_retry,
396690f6a15SCornelia Huck 		[VFIO_CCW_EVENT_INTERRUPT]	= fsm_irq,
397690f6a15SCornelia Huck 	},
398690f6a15SCornelia Huck 	[VFIO_CCW_STATE_CP_PENDING] = {
399bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_NOT_OPER]	= fsm_notoper,
400bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_IO_REQ]		= fsm_io_busy,
401d5afd5d1SCornelia Huck 		[VFIO_CCW_EVENT_ASYNC_REQ]	= fsm_async_request,
402bbe37e4cSDong Jia Shi 		[VFIO_CCW_EVENT_INTERRUPT]	= fsm_irq,
403bbe37e4cSDong Jia Shi 	},
404bbe37e4cSDong Jia Shi };
405