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 183cd90214SHalil Pasic #define CREATE_TRACE_POINTS 193cd90214SHalil Pasic #include "vfio_ccw_trace.h" 203cd90214SHalil Pasic 21bbe37e4cSDong Jia Shi static int fsm_io_helper(struct vfio_ccw_private *private) 22bbe37e4cSDong Jia Shi { 23bbe37e4cSDong Jia Shi struct subchannel *sch; 24bbe37e4cSDong Jia Shi union orb *orb; 25bbe37e4cSDong Jia Shi int ccode; 26bbe37e4cSDong Jia Shi __u8 lpm; 27bbe37e4cSDong Jia Shi unsigned long flags; 283368e547SCornelia Huck int ret; 29bbe37e4cSDong Jia Shi 30bbe37e4cSDong Jia Shi sch = private->sch; 31bbe37e4cSDong Jia Shi 32bbe37e4cSDong Jia Shi spin_lock_irqsave(sch->lock, flags); 33bbe37e4cSDong Jia Shi 34bbe37e4cSDong Jia Shi orb = cp_get_orb(&private->cp, (u32)(addr_t)sch, sch->lpm); 3571189f26SCornelia Huck if (!orb) { 3671189f26SCornelia Huck ret = -EIO; 3771189f26SCornelia Huck goto out; 3871189f26SCornelia Huck } 39bbe37e4cSDong Jia Shi 40*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(5, "stIO"); 41*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(5, dev_name(&sch->dev)); 42*60e05d1cSCornelia Huck 43bbe37e4cSDong Jia Shi /* Issue "Start Subchannel" */ 44bbe37e4cSDong Jia Shi ccode = ssch(sch->schid, orb); 45bbe37e4cSDong Jia Shi 46*60e05d1cSCornelia Huck VFIO_CCW_HEX_EVENT(5, &ccode, sizeof(ccode)); 47*60e05d1cSCornelia Huck 48bbe37e4cSDong Jia Shi switch (ccode) { 49bbe37e4cSDong Jia Shi case 0: 50bbe37e4cSDong Jia Shi /* 51bbe37e4cSDong Jia Shi * Initialize device status information 52bbe37e4cSDong Jia Shi */ 53bbe37e4cSDong Jia Shi sch->schib.scsw.cmd.actl |= SCSW_ACTL_START_PEND; 543368e547SCornelia Huck ret = 0; 55690f6a15SCornelia Huck private->state = VFIO_CCW_STATE_CP_PENDING; 563368e547SCornelia Huck break; 57bbe37e4cSDong Jia Shi case 1: /* Status pending */ 58bbe37e4cSDong Jia Shi case 2: /* Busy */ 593368e547SCornelia Huck ret = -EBUSY; 603368e547SCornelia Huck break; 61bbe37e4cSDong Jia Shi case 3: /* Device/path not operational */ 62bbe37e4cSDong Jia Shi { 63bbe37e4cSDong Jia Shi lpm = orb->cmd.lpm; 64bbe37e4cSDong Jia Shi if (lpm != 0) 65bbe37e4cSDong Jia Shi sch->lpm &= ~lpm; 66bbe37e4cSDong Jia Shi else 67bbe37e4cSDong Jia Shi sch->lpm = 0; 68bbe37e4cSDong Jia Shi 69bbe37e4cSDong Jia Shi if (cio_update_schib(sch)) 703368e547SCornelia Huck ret = -ENODEV; 713368e547SCornelia Huck else 723368e547SCornelia Huck ret = sch->lpm ? -EACCES : -ENODEV; 733368e547SCornelia Huck break; 74bbe37e4cSDong Jia Shi } 75bbe37e4cSDong Jia Shi default: 763368e547SCornelia Huck ret = ccode; 77bbe37e4cSDong Jia Shi } 7871189f26SCornelia Huck out: 793368e547SCornelia Huck spin_unlock_irqrestore(sch->lock, flags); 803368e547SCornelia Huck return ret; 81bbe37e4cSDong Jia Shi } 82bbe37e4cSDong Jia Shi 83d5afd5d1SCornelia Huck static int fsm_do_halt(struct vfio_ccw_private *private) 84d5afd5d1SCornelia Huck { 85d5afd5d1SCornelia Huck struct subchannel *sch; 86d5afd5d1SCornelia Huck unsigned long flags; 87d5afd5d1SCornelia Huck int ccode; 88d5afd5d1SCornelia Huck int ret; 89d5afd5d1SCornelia Huck 90d5afd5d1SCornelia Huck sch = private->sch; 91d5afd5d1SCornelia Huck 92d5afd5d1SCornelia Huck spin_lock_irqsave(sch->lock, flags); 93d5afd5d1SCornelia Huck 94*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, "haltIO"); 95*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev)); 96*60e05d1cSCornelia Huck 97d5afd5d1SCornelia Huck /* Issue "Halt Subchannel" */ 98d5afd5d1SCornelia Huck ccode = hsch(sch->schid); 99d5afd5d1SCornelia Huck 100*60e05d1cSCornelia Huck VFIO_CCW_HEX_EVENT(2, &ccode, sizeof(ccode)); 101*60e05d1cSCornelia Huck 102d5afd5d1SCornelia Huck switch (ccode) { 103d5afd5d1SCornelia Huck case 0: 104d5afd5d1SCornelia Huck /* 105d5afd5d1SCornelia Huck * Initialize device status information 106d5afd5d1SCornelia Huck */ 107d5afd5d1SCornelia Huck sch->schib.scsw.cmd.actl |= SCSW_ACTL_HALT_PEND; 108d5afd5d1SCornelia Huck ret = 0; 109d5afd5d1SCornelia Huck break; 110d5afd5d1SCornelia Huck case 1: /* Status pending */ 111d5afd5d1SCornelia Huck case 2: /* Busy */ 112d5afd5d1SCornelia Huck ret = -EBUSY; 113d5afd5d1SCornelia Huck break; 114d5afd5d1SCornelia Huck case 3: /* Device not operational */ 115d5afd5d1SCornelia Huck ret = -ENODEV; 116d5afd5d1SCornelia Huck break; 117d5afd5d1SCornelia Huck default: 118d5afd5d1SCornelia Huck ret = ccode; 119d5afd5d1SCornelia Huck } 120d5afd5d1SCornelia Huck spin_unlock_irqrestore(sch->lock, flags); 121d5afd5d1SCornelia Huck return ret; 122d5afd5d1SCornelia Huck } 123d5afd5d1SCornelia Huck 124d5afd5d1SCornelia Huck static int fsm_do_clear(struct vfio_ccw_private *private) 125d5afd5d1SCornelia Huck { 126d5afd5d1SCornelia Huck struct subchannel *sch; 127d5afd5d1SCornelia Huck unsigned long flags; 128d5afd5d1SCornelia Huck int ccode; 129d5afd5d1SCornelia Huck int ret; 130d5afd5d1SCornelia Huck 131d5afd5d1SCornelia Huck sch = private->sch; 132d5afd5d1SCornelia Huck 133d5afd5d1SCornelia Huck spin_lock_irqsave(sch->lock, flags); 134d5afd5d1SCornelia Huck 135*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, "clearIO"); 136*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev)); 137*60e05d1cSCornelia Huck 138d5afd5d1SCornelia Huck /* Issue "Clear Subchannel" */ 139d5afd5d1SCornelia Huck ccode = csch(sch->schid); 140d5afd5d1SCornelia Huck 141*60e05d1cSCornelia Huck VFIO_CCW_HEX_EVENT(2, &ccode, sizeof(ccode)); 142*60e05d1cSCornelia Huck 143d5afd5d1SCornelia Huck switch (ccode) { 144d5afd5d1SCornelia Huck case 0: 145d5afd5d1SCornelia Huck /* 146d5afd5d1SCornelia Huck * Initialize device status information 147d5afd5d1SCornelia Huck */ 148d5afd5d1SCornelia Huck sch->schib.scsw.cmd.actl = SCSW_ACTL_CLEAR_PEND; 149d5afd5d1SCornelia Huck /* TODO: check what else we might need to clear */ 150d5afd5d1SCornelia Huck ret = 0; 151d5afd5d1SCornelia Huck break; 152d5afd5d1SCornelia Huck case 3: /* Device not operational */ 153d5afd5d1SCornelia Huck ret = -ENODEV; 154d5afd5d1SCornelia Huck break; 155d5afd5d1SCornelia Huck default: 156d5afd5d1SCornelia Huck ret = ccode; 157d5afd5d1SCornelia Huck } 158d5afd5d1SCornelia Huck spin_unlock_irqrestore(sch->lock, flags); 159d5afd5d1SCornelia Huck return ret; 160d5afd5d1SCornelia Huck } 161d5afd5d1SCornelia Huck 162bbe37e4cSDong Jia Shi static void fsm_notoper(struct vfio_ccw_private *private, 163bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 164bbe37e4cSDong Jia Shi { 165bbe37e4cSDong Jia Shi struct subchannel *sch = private->sch; 166bbe37e4cSDong Jia Shi 167*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, "notoper"); 168*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(2, dev_name(&sch->dev)); 169*60e05d1cSCornelia Huck 170bbe37e4cSDong Jia Shi /* 171bbe37e4cSDong Jia Shi * TODO: 172bbe37e4cSDong Jia Shi * Probably we should send the machine check to the guest. 173bbe37e4cSDong Jia Shi */ 174bbe37e4cSDong Jia Shi css_sched_sch_todo(sch, SCH_TODO_UNREG); 175bbe37e4cSDong Jia Shi private->state = VFIO_CCW_STATE_NOT_OPER; 176bbe37e4cSDong Jia Shi } 177bbe37e4cSDong Jia Shi 178bbe37e4cSDong Jia Shi /* 179bbe37e4cSDong Jia Shi * No operation action. 180bbe37e4cSDong Jia Shi */ 181bbe37e4cSDong Jia Shi static void fsm_nop(struct vfio_ccw_private *private, 182bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 183bbe37e4cSDong Jia Shi { 184bbe37e4cSDong Jia Shi } 185bbe37e4cSDong Jia Shi 186bbe37e4cSDong Jia Shi static void fsm_io_error(struct vfio_ccw_private *private, 187bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 188bbe37e4cSDong Jia Shi { 189bbe37e4cSDong Jia Shi pr_err("vfio-ccw: FSM: I/O request from state:%d\n", private->state); 190c98e16b2SEric Farman private->io_region->ret_code = -EIO; 191bbe37e4cSDong Jia Shi } 192bbe37e4cSDong Jia Shi 193bbe37e4cSDong Jia Shi static void fsm_io_busy(struct vfio_ccw_private *private, 194bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 195bbe37e4cSDong Jia Shi { 196c98e16b2SEric Farman private->io_region->ret_code = -EBUSY; 197bbe37e4cSDong Jia Shi } 198bbe37e4cSDong Jia Shi 199690f6a15SCornelia Huck static void fsm_io_retry(struct vfio_ccw_private *private, 200690f6a15SCornelia Huck enum vfio_ccw_event event) 201690f6a15SCornelia Huck { 202690f6a15SCornelia Huck private->io_region->ret_code = -EAGAIN; 203690f6a15SCornelia Huck } 204690f6a15SCornelia Huck 205d5afd5d1SCornelia Huck static void fsm_async_error(struct vfio_ccw_private *private, 206d5afd5d1SCornelia Huck enum vfio_ccw_event event) 207d5afd5d1SCornelia Huck { 208d5afd5d1SCornelia Huck struct ccw_cmd_region *cmd_region = private->cmd_region; 209d5afd5d1SCornelia Huck 210d5afd5d1SCornelia Huck pr_err("vfio-ccw: FSM: %s request from state:%d\n", 211d5afd5d1SCornelia Huck cmd_region->command == VFIO_CCW_ASYNC_CMD_HSCH ? "halt" : 212d5afd5d1SCornelia Huck cmd_region->command == VFIO_CCW_ASYNC_CMD_CSCH ? "clear" : 213d5afd5d1SCornelia Huck "<unknown>", private->state); 214d5afd5d1SCornelia Huck cmd_region->ret_code = -EIO; 215d5afd5d1SCornelia Huck } 216d5afd5d1SCornelia Huck 217d5afd5d1SCornelia Huck static void fsm_async_retry(struct vfio_ccw_private *private, 218d5afd5d1SCornelia Huck enum vfio_ccw_event event) 219d5afd5d1SCornelia Huck { 220d5afd5d1SCornelia Huck private->cmd_region->ret_code = -EAGAIN; 221d5afd5d1SCornelia Huck } 222d5afd5d1SCornelia Huck 223bbe37e4cSDong Jia Shi static void fsm_disabled_irq(struct vfio_ccw_private *private, 224bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 225bbe37e4cSDong Jia Shi { 226bbe37e4cSDong Jia Shi struct subchannel *sch = private->sch; 227bbe37e4cSDong Jia Shi 228bbe37e4cSDong Jia Shi /* 229bbe37e4cSDong Jia Shi * An interrupt in a disabled state means a previous disable was not 230bbe37e4cSDong Jia Shi * successful - should not happen, but we try to disable again. 231bbe37e4cSDong Jia Shi */ 232bbe37e4cSDong Jia Shi cio_disable_subchannel(sch); 233bbe37e4cSDong Jia Shi } 2343cd90214SHalil Pasic inline struct subchannel_id get_schid(struct vfio_ccw_private *p) 2353cd90214SHalil Pasic { 2363cd90214SHalil Pasic return p->sch->schid; 2373cd90214SHalil Pasic } 238bbe37e4cSDong Jia Shi 239bbe37e4cSDong Jia Shi /* 240bbe37e4cSDong Jia Shi * Deal with the ccw command request from the userspace. 241bbe37e4cSDong Jia Shi */ 242bbe37e4cSDong Jia Shi static void fsm_io_request(struct vfio_ccw_private *private, 243bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 244bbe37e4cSDong Jia Shi { 245bbe37e4cSDong Jia Shi union orb *orb; 246bbe37e4cSDong Jia Shi union scsw *scsw = &private->scsw; 247c98e16b2SEric Farman struct ccw_io_region *io_region = private->io_region; 248bbe37e4cSDong Jia Shi struct mdev_device *mdev = private->mdev; 2493cd90214SHalil Pasic char *errstr = "request"; 250*60e05d1cSCornelia Huck struct subchannel_id schid = get_schid(private); 251bbe37e4cSDong Jia Shi 252690f6a15SCornelia Huck private->state = VFIO_CCW_STATE_CP_PROCESSING; 253bbe37e4cSDong Jia Shi memcpy(scsw, io_region->scsw_area, sizeof(*scsw)); 254bbe37e4cSDong Jia Shi 255bbe37e4cSDong Jia Shi if (scsw->cmd.fctl & SCSW_FCTL_START_FUNC) { 256bbe37e4cSDong Jia Shi orb = (union orb *)io_region->orb_area; 257bbe37e4cSDong Jia Shi 2589851bc77SCornelia Huck /* Don't try to build a cp if transport mode is specified. */ 2599851bc77SCornelia Huck if (orb->tm.b) { 2609851bc77SCornelia Huck io_region->ret_code = -EOPNOTSUPP; 261*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 262*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): transport mode\n", 263*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 264*60e05d1cSCornelia Huck schid.ssid, schid.sch_no); 2653cd90214SHalil Pasic errstr = "transport mode"; 2669851bc77SCornelia Huck goto err_out; 2679851bc77SCornelia Huck } 268bbe37e4cSDong Jia Shi io_region->ret_code = cp_init(&private->cp, mdev_dev(mdev), 269bbe37e4cSDong Jia Shi orb); 2703cd90214SHalil Pasic if (io_region->ret_code) { 271*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 272*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): cp_init=%d\n", 273*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 274*60e05d1cSCornelia Huck schid.ssid, schid.sch_no, 275*60e05d1cSCornelia Huck io_region->ret_code); 2763cd90214SHalil Pasic errstr = "cp init"; 277bbe37e4cSDong Jia Shi goto err_out; 2783cd90214SHalil Pasic } 279bbe37e4cSDong Jia Shi 280bbe37e4cSDong Jia Shi io_region->ret_code = cp_prefetch(&private->cp); 281bbe37e4cSDong Jia Shi if (io_region->ret_code) { 282*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 283*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): cp_prefetch=%d\n", 284*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 285*60e05d1cSCornelia Huck schid.ssid, schid.sch_no, 286*60e05d1cSCornelia Huck io_region->ret_code); 2873cd90214SHalil Pasic errstr = "cp prefetch"; 288bbe37e4cSDong Jia Shi cp_free(&private->cp); 289bbe37e4cSDong Jia Shi goto err_out; 290bbe37e4cSDong Jia Shi } 291bbe37e4cSDong Jia Shi 292bbe37e4cSDong Jia Shi /* Start channel program and wait for I/O interrupt. */ 293bbe37e4cSDong Jia Shi io_region->ret_code = fsm_io_helper(private); 294bbe37e4cSDong Jia Shi if (io_region->ret_code) { 295*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 296*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): fsm_io_helper=%d\n", 297*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 298*60e05d1cSCornelia Huck schid.ssid, schid.sch_no, 299*60e05d1cSCornelia Huck io_region->ret_code); 3003cd90214SHalil Pasic errstr = "cp fsm_io_helper"; 301bbe37e4cSDong Jia Shi cp_free(&private->cp); 302bbe37e4cSDong Jia Shi goto err_out; 303bbe37e4cSDong Jia Shi } 304bbe37e4cSDong Jia Shi return; 305bbe37e4cSDong Jia Shi } else if (scsw->cmd.fctl & SCSW_FCTL_HALT_FUNC) { 306*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 307*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): halt on io_region\n", 308*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 309*60e05d1cSCornelia Huck schid.ssid, schid.sch_no); 310d5afd5d1SCornelia Huck /* halt is handled via the async cmd region */ 311bbe37e4cSDong Jia Shi io_region->ret_code = -EOPNOTSUPP; 312bbe37e4cSDong Jia Shi goto err_out; 313bbe37e4cSDong Jia Shi } else if (scsw->cmd.fctl & SCSW_FCTL_CLEAR_FUNC) { 314*60e05d1cSCornelia Huck VFIO_CCW_MSG_EVENT(2, 315*60e05d1cSCornelia Huck "%pUl (%x.%x.%04x): clear on io_region\n", 316*60e05d1cSCornelia Huck mdev_uuid(mdev), schid.cssid, 317*60e05d1cSCornelia Huck schid.ssid, schid.sch_no); 318d5afd5d1SCornelia Huck /* clear is handled via the async cmd region */ 319bbe37e4cSDong Jia Shi io_region->ret_code = -EOPNOTSUPP; 320bbe37e4cSDong Jia Shi goto err_out; 321bbe37e4cSDong Jia Shi } 322bbe37e4cSDong Jia Shi 323bbe37e4cSDong Jia Shi err_out: 324*60e05d1cSCornelia Huck trace_vfio_ccw_io_fctl(scsw->cmd.fctl, schid, 3253cd90214SHalil Pasic io_region->ret_code, errstr); 326bbe37e4cSDong Jia Shi } 327bbe37e4cSDong Jia Shi 328bbe37e4cSDong Jia Shi /* 329d5afd5d1SCornelia Huck * Deal with an async request from userspace. 330d5afd5d1SCornelia Huck */ 331d5afd5d1SCornelia Huck static void fsm_async_request(struct vfio_ccw_private *private, 332d5afd5d1SCornelia Huck enum vfio_ccw_event event) 333d5afd5d1SCornelia Huck { 334d5afd5d1SCornelia Huck struct ccw_cmd_region *cmd_region = private->cmd_region; 335d5afd5d1SCornelia Huck 336d5afd5d1SCornelia Huck switch (cmd_region->command) { 337d5afd5d1SCornelia Huck case VFIO_CCW_ASYNC_CMD_HSCH: 338d5afd5d1SCornelia Huck cmd_region->ret_code = fsm_do_halt(private); 339d5afd5d1SCornelia Huck break; 340d5afd5d1SCornelia Huck case VFIO_CCW_ASYNC_CMD_CSCH: 341d5afd5d1SCornelia Huck cmd_region->ret_code = fsm_do_clear(private); 342d5afd5d1SCornelia Huck break; 343d5afd5d1SCornelia Huck default: 344d5afd5d1SCornelia Huck /* should not happen? */ 345d5afd5d1SCornelia Huck cmd_region->ret_code = -EINVAL; 346d5afd5d1SCornelia Huck } 347d5afd5d1SCornelia Huck } 348d5afd5d1SCornelia Huck 349d5afd5d1SCornelia Huck /* 350bbe37e4cSDong Jia Shi * Got an interrupt for a normal io (state busy). 351bbe37e4cSDong Jia Shi */ 352bbe37e4cSDong Jia Shi static void fsm_irq(struct vfio_ccw_private *private, 353bbe37e4cSDong Jia Shi enum vfio_ccw_event event) 354bbe37e4cSDong Jia Shi { 355c9c31b07SDong Jia Shi struct irb *irb = this_cpu_ptr(&cio_irb); 356bbe37e4cSDong Jia Shi 357*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(6, "IRQ"); 358*60e05d1cSCornelia Huck VFIO_CCW_TRACE_EVENT(6, dev_name(&private->sch->dev)); 359*60e05d1cSCornelia Huck 360bbe37e4cSDong Jia Shi memcpy(&private->irb, irb, sizeof(*irb)); 361bbe37e4cSDong Jia Shi 362bbe37e4cSDong Jia Shi queue_work(vfio_ccw_work_q, &private->io_work); 363bbe37e4cSDong Jia Shi 364bbe37e4cSDong Jia Shi if (private->completion) 365bbe37e4cSDong Jia Shi complete(private->completion); 366bbe37e4cSDong Jia Shi } 367bbe37e4cSDong Jia Shi 368bbe37e4cSDong Jia Shi /* 369bbe37e4cSDong Jia Shi * Device statemachine 370bbe37e4cSDong Jia Shi */ 371bbe37e4cSDong Jia Shi fsm_func_t *vfio_ccw_jumptable[NR_VFIO_CCW_STATES][NR_VFIO_CCW_EVENTS] = { 372bbe37e4cSDong Jia Shi [VFIO_CCW_STATE_NOT_OPER] = { 373bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_NOT_OPER] = fsm_nop, 374bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_IO_REQ] = fsm_io_error, 375d5afd5d1SCornelia Huck [VFIO_CCW_EVENT_ASYNC_REQ] = fsm_async_error, 376bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_INTERRUPT] = fsm_disabled_irq, 377bbe37e4cSDong Jia Shi }, 378bbe37e4cSDong Jia Shi [VFIO_CCW_STATE_STANDBY] = { 379bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_NOT_OPER] = fsm_notoper, 380bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_IO_REQ] = fsm_io_error, 381d5afd5d1SCornelia Huck [VFIO_CCW_EVENT_ASYNC_REQ] = fsm_async_error, 382bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_INTERRUPT] = fsm_irq, 383bbe37e4cSDong Jia Shi }, 384bbe37e4cSDong Jia Shi [VFIO_CCW_STATE_IDLE] = { 385bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_NOT_OPER] = fsm_notoper, 386bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_IO_REQ] = fsm_io_request, 387d5afd5d1SCornelia Huck [VFIO_CCW_EVENT_ASYNC_REQ] = fsm_async_request, 388bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_INTERRUPT] = fsm_irq, 389bbe37e4cSDong Jia Shi }, 390690f6a15SCornelia Huck [VFIO_CCW_STATE_CP_PROCESSING] = { 391690f6a15SCornelia Huck [VFIO_CCW_EVENT_NOT_OPER] = fsm_notoper, 392690f6a15SCornelia Huck [VFIO_CCW_EVENT_IO_REQ] = fsm_io_retry, 393d5afd5d1SCornelia Huck [VFIO_CCW_EVENT_ASYNC_REQ] = fsm_async_retry, 394690f6a15SCornelia Huck [VFIO_CCW_EVENT_INTERRUPT] = fsm_irq, 395690f6a15SCornelia Huck }, 396690f6a15SCornelia Huck [VFIO_CCW_STATE_CP_PENDING] = { 397bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_NOT_OPER] = fsm_notoper, 398bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_IO_REQ] = fsm_io_busy, 399d5afd5d1SCornelia Huck [VFIO_CCW_EVENT_ASYNC_REQ] = fsm_async_request, 400bbe37e4cSDong Jia Shi [VFIO_CCW_EVENT_INTERRUPT] = fsm_irq, 401bbe37e4cSDong Jia Shi }, 402bbe37e4cSDong Jia Shi }; 403