|
|
@@ -373,11 +373,20 @@ static void vfio_pci_disable(struct vfio_pci_device *vdev)
|
|
|
pci_write_config_word(pdev, PCI_COMMAND, PCI_COMMAND_INTX_DISABLE);
|
|
|
|
|
|
/*
|
|
|
- * Try to reset the device. The success of this is dependent on
|
|
|
- * being able to lock the device, which is not always possible.
|
|
|
+ * Try to get the locks ourselves to prevent a deadlock. The
|
|
|
+ * success of this is dependent on being able to lock the device,
|
|
|
+ * which is not always possible.
|
|
|
+ * We can not use the "try" reset interface here, which will
|
|
|
+ * overwrite the previously restored configuration information.
|
|
|
*/
|
|
|
- if (vdev->reset_works && !pci_try_reset_function(pdev))
|
|
|
- vdev->needs_reset = false;
|
|
|
+ if (vdev->reset_works && pci_cfg_access_trylock(pdev)) {
|
|
|
+ if (device_trylock(&pdev->dev)) {
|
|
|
+ if (!__pci_reset_function_locked(pdev))
|
|
|
+ vdev->needs_reset = false;
|
|
|
+ device_unlock(&pdev->dev);
|
|
|
+ }
|
|
|
+ pci_cfg_access_unlock(pdev);
|
|
|
+ }
|
|
|
|
|
|
pci_restore_state(pdev);
|
|
|
out:
|