Some architectures like PPC, especailly PowerNV platform, need to do additional operations while adding or removing VFIO devices to or from VFIO bus. The patch adds weak functions while to open, release or ioctl for the specific VFIO device. Those functions could be overrided by individual architectures if necessary. Signed-off-by: Gavin Shan <shangw@xxxxxxxxxxxxxxxxxx> --- drivers/vfio/pci/vfio_pci.c | 42 +++++++++++++++++++++++++++++++++++------- include/linux/vfio.h | 7 ++++++- 2 files changed, 41 insertions(+), 8 deletions(-) diff --git a/drivers/vfio/pci/vfio_pci.c b/drivers/vfio/pci/vfio_pci.c index 8189cb6..1a53e77 100644 --- a/drivers/vfio/pci/vfio_pci.c +++ b/drivers/vfio/pci/vfio_pci.c @@ -143,32 +143,51 @@ static void vfio_pci_disable(struct vfio_pci_device *vdev) pci_restore_state(pdev); } +void __weak vfio_pci_arch_release(struct pci_dev *pdev) +{ + return; +} + static void vfio_pci_release(void *device_data) { struct vfio_pci_device *vdev = device_data; - if (atomic_dec_and_test(&vdev->refcnt)) + if (atomic_dec_and_test(&vdev->refcnt)) { + vfio_pci_arch_release(vdev->pdev); + vfio_pci_disable(vdev); + } module_put(THIS_MODULE); } +int __weak vfio_pci_arch_open(struct pci_dev *pdev) +{ + return 0; +} + static int vfio_pci_open(void *device_data) { struct vfio_pci_device *vdev = device_data; + int ret; if (!try_module_get(THIS_MODULE)) return -ENODEV; if (atomic_inc_return(&vdev->refcnt) == 1) { - int ret = vfio_pci_enable(vdev); - if (ret) { - module_put(THIS_MODULE); - return ret; - } + ret = vfio_pci_arch_open(vdev->pdev); + if (ret) + goto fail; + + ret = vfio_pci_enable(vdev); + if (ret) + goto fail; } return 0; +fail: + module_put(THIS_MODULE); + return ret; } static int vfio_pci_get_irq_count(struct vfio_pci_device *vdev, int irq_type) @@ -206,6 +225,12 @@ static int vfio_pci_get_irq_count(struct vfio_pci_device *vdev, int irq_type) return 0; } +long __weak vfio_pci_arch_ioctl(struct pci_dev *pdev, + unsigned int cmd, unsigned long arg) +{ + return -ENOTTY; +} + static long vfio_pci_ioctl(void *device_data, unsigned int cmd, unsigned long arg) { @@ -374,9 +399,12 @@ static long vfio_pci_ioctl(void *device_data, return ret; - } else if (cmd == VFIO_DEVICE_RESET) + } else if (cmd == VFIO_DEVICE_RESET) { return vdev->reset_works ? pci_reset_function(vdev->pdev) : -EINVAL; + } else { + return vfio_pci_arch_ioctl(vdev->pdev, cmd, arg); + } return -ENOTTY; } diff --git a/include/linux/vfio.h b/include/linux/vfio.h index ab9e862..a991c39 100644 --- a/include/linux/vfio.h +++ b/include/linux/vfio.h @@ -11,9 +11,9 @@ #ifndef VFIO_H #define VFIO_H - #include <linux/iommu.h> #include <linux/mm.h> +#include <linux/pci.h> #include <uapi/linux/vfio.h> /** @@ -40,6 +40,11 @@ struct vfio_device_ops { int (*mmap)(void *device_data, struct vm_area_struct *vma); }; +extern int vfio_pci_arch_open(struct pci_dev *pdev); +extern long vfio_pci_arch_ioctl(struct pci_dev *pdev, + unsigned int cmd, + unsigned long arg); +extern void vfio_pci_arch_release(struct pci_dev *pdev); extern int vfio_add_group_dev(struct device *dev, const struct vfio_device_ops *ops, void *device_data); -- 1.7.5.4 -- To unsubscribe from this list: send the line "unsubscribe kvm" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html