> From: Yishai Hadas <yishaih@xxxxxxxxxx> > Sent: Tuesday, February 8, 2022 1:22 AM > > From: Jason Gunthorpe <jgg@xxxxxxxxxx> > > The RUNNING_P2P state is designed to support multiple devices in the same > VM that are doing P2P transactions between themselves. When in > RUNNING_P2P > the device must be able to accept incoming P2P transactions but should not > generate outgoing transactions. outgoing 'P2P' transactions. > > As an optional extension to the mandatory states it is defined as > inbetween STOP and RUNNING: > STOP -> RUNNING_P2P -> RUNNING -> RUNNING_P2P -> STOP > > For drivers that are unable to support RUNNING_P2P the core code silently > merges RUNNING_P2P and RUNNING together. Drivers that support this will It would be clearer if following message could be also reflected here: + * The optional states cannot be used with SET_STATE if the device does not + * support them. The user can discover if these states are supported by using + * VFIO_DEVICE_FEATURE_MIGRATION. Otherwise the original context reads like RUNNING_P2P can be used as end state even if the underlying driver doesn't support it then makes me wonder what is the point of the new capability bit. > be > required to implement 4 FSM arcs beyond the basic FSM. 2 of the basic FSM > arcs become combination transitions. > > Compared to the v1 clarification, NDMA is redefined into FSM states and is > described in terms of the desired P2P quiescent behavior, noting that > halting all DMA is an acceptable implementation. > > Signed-off-by: Jason Gunthorpe <jgg@xxxxxxxxxx> > Signed-off-by: Yishai Hadas <yishaih@xxxxxxxxxx> > --- > drivers/vfio/vfio.c | 79 ++++++++++++++++++++++++++++++--------- > include/linux/vfio.h | 1 + > include/uapi/linux/vfio.h | 34 ++++++++++++++++- > 3 files changed, 95 insertions(+), 19 deletions(-) > > @@ -1631,17 +1657,36 @@ int vfio_mig_get_next_state(struct vfio_device [...] > *next_fsm = vfio_from_fsm_table[cur_fsm][new_fsm]; > + while ((state_flags_table[*next_fsm] & device->migration_flags) != > + state_flags_table[*next_fsm]) > + *next_fsm = vfio_from_fsm_table[*next_fsm][new_fsm]; > + A comment highlighting the silent merging of unsupported states would be informative here. and I have a puzzle on following messages: > * > + * And 1 optional state to support VFIO_MIGRATION_P2P: > + * RUNNING_P2P - RUNNING, except the device cannot do peer to peer > DMA > * and > + * RUNNING_P2P -> RUNNING > * While in RUNNING the device is fully operational, the device may > generate > * interrupts, DMA, respond to MMIO, all vfio device regions are functional, > * and the device may advance its internal state. > * and below > + * The optional peer to peer (P2P) quiescent state is intended to be a > quiescent > + * state for the device for the purposes of managing multiple devices within > a > + * user context where peer-to-peer DMA between devices may be active. > The > + * RUNNING_P2P states must prevent the device from initiating > + * any new P2P DMA transactions. If the device can identify P2P transactions > + * then it can stop only P2P DMA, otherwise it must stop all DMA. The > migration > + * driver must complete any such outstanding operations prior to > completing the > + * FSM arc into a P2P state. For the purpose of specification the states > + * behave as though the device was fully running if not supported. Defining RUNNING_P2P in above way implies that RUNNING_P2P inherits all behaviors in RUNNING except blocking outbound P2P: * generate interrupts and DMAs * respond to MMIO * all vfio regions are functional * device may advance its internal state * drain and block outstanding P2P requests I think this is not the intended behavior when NDMA was being discussed in previous threads, as above definition suggests the user could continue to submit new requests after outstanding P2P requests are completed given all vfio regions are functional when the device is in RUNNING_P2P. Though just a naming thing, possibly what we really require is a STOPPING_P2P state which indicates the device is moving to the STOP (or STOPPED) state. In this state the device is functional but vfio regions are not so the user still needs to restrict device access: * generate interrupts and DMAs * respond to MMIO * all vfio regions are NOT functional (no user access) * device may advance its internal state * drain and block outstanding P2P requests In virtualization this means Qemu must stop vCPU first before entering STOPPING_P2P for a device. Back to your earlier suggestion on reusing RUNNING_P2P to cover vPRI usage via a new capability bit [1]: "A cap like "running_p2p returns an event fd, doesn't finish until the VCPU does stuff, and stops pri as well as p2p" might be all that is required here (and not an actual new state)" vPRI requires a RUNNING semantics. A new capability bit can change the behaviors listed above for STOPPING_P2P to below: * both P2P and vPRI requests should be drained and blocked; * all vfio regions are functional (with a RUNNING behavior) so vCPUs can continue running to help drain vPRI requests; * an eventfd is returned for the user to poll-wait the completion of state transition; and in this regard possibly it makes more sense to call this state as STOPPING to encapsulate any optional preparation work before the device can be transitioned to STOP (with default as defined for STOPPING_P2P above and actual behavior changeable by future capability bits)? One additional requirement in driver side is to dynamically mediate the fast path and queue any new request which may trigger vPRI or P2P before moving out of RUNNING_P2P. If moving to STOP_COPY, then queued requests will also be included as device state to be replayed in the resuming path. Does above sound a reasonable understanding of this FSM mechanism? > + * > + * The optional states cannot be used with SET_STATE if the device does not > + * support them. The user can disocver if these states are supported by 'disocver' -> 'discover' Thanks Kevin