Commit f6c876d6 authored by Eric Farman's avatar Eric Farman Committed by Alex Williamson

vfio/ccw: Fix FSM state if mdev probe fails

The FSM is in STANDBY state when arriving in vfio_ccw_mdev_probe(),
and this routine converts it to IDLE as part of its processing.
The error exit sets it to IDLE (again) but clears the private->mdev
pointer.

The FSM should of course be managing the state itself, but the
correct thing for vfio_ccw_mdev_probe() to do would be to put
the state back the way it found it.

The corresponding check of private->mdev in vfio_ccw_sch_io_todo()
can be removed, since the distinction is unnecessary at this point.

Fixes: 3bf1311f ("vfio/ccw: Convert to use vfio_register_emulated_iommu_dev()")
Signed-off-by: default avatarEric Farman <farman@linux.ibm.com>
Reviewed-by: default avatarJason Gunthorpe <jgg@nvidia.com>
Reviewed-by: default avatarMatthew Rosato <mjrosato@linux.ibm.com>
Link: https://lore.kernel.org/r/20220707135737.720765-3-farman@linux.ibm.comSigned-off-by: default avatarAlex Williamson <alex.williamson@redhat.com>
parent 3566ee1d
...@@ -106,9 +106,10 @@ static void vfio_ccw_sch_io_todo(struct work_struct *work) ...@@ -106,9 +106,10 @@ static void vfio_ccw_sch_io_todo(struct work_struct *work)
/* /*
* Reset to IDLE only if processing of a channel program * Reset to IDLE only if processing of a channel program
* has finished. Do not overwrite a possible processing * has finished. Do not overwrite a possible processing
* state if the final interrupt was for HSCH or CSCH. * state if the interrupt was unsolicited, or if the final
* interrupt was for HSCH or CSCH.
*/ */
if (private->mdev && cp_is_finished) if (cp_is_finished)
private->state = VFIO_CCW_STATE_IDLE; private->state = VFIO_CCW_STATE_IDLE;
if (private->io_trigger) if (private->io_trigger)
......
...@@ -146,7 +146,7 @@ static int vfio_ccw_mdev_probe(struct mdev_device *mdev) ...@@ -146,7 +146,7 @@ static int vfio_ccw_mdev_probe(struct mdev_device *mdev)
vfio_uninit_group_dev(&private->vdev); vfio_uninit_group_dev(&private->vdev);
atomic_inc(&private->avail); atomic_inc(&private->avail);
private->mdev = NULL; private->mdev = NULL;
private->state = VFIO_CCW_STATE_IDLE; private->state = VFIO_CCW_STATE_STANDBY;
return ret; return ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment