-obj-$(CONFIG_VFIO) += vfio.o
+vfio_core-y := vfio.o virqfd.o
+
+obj-$(CONFIG_VFIO) += vfio_core.o
 obj-$(CONFIG_VFIO_IOMMU_TYPE1) += vfio_iommu_type1.o
 obj-$(CONFIG_VFIO_IOMMU_SPAPR_TCE) += vfio_iommu_spapr_tce.o
 obj-$(CONFIG_VFIO_SPAPR_EEH) += vfio_spapr_eeh.o
 
 
-vfio-pci-y := vfio_pci.o vfio_pci_intrs.o vfio_pci_rdwr.o vfio_pci_config.o \
-             ../virqfd.o
+vfio-pci-y := vfio_pci.o vfio_pci_intrs.o vfio_pci_rdwr.o vfio_pci_config.o
 
 obj-$(CONFIG_VFIO_PCI) += vfio-pci.o
 
 static void __exit vfio_pci_cleanup(void)
 {
        pci_unregister_driver(&vfio_pci_driver);
-       vfio_virqfd_exit();
        vfio_pci_uninit_perm_bits();
 }
 
        if (ret)
                return ret;
 
-       /* Start the virqfd cleanup handler */
-       ret = vfio_virqfd_init();
-       if (ret)
-               goto out_virqfd;
-
        /* Register and scan for devices */
        ret = pci_register_driver(&vfio_pci_driver);
        if (ret)
        return 0;
 
 out_driver:
-       vfio_virqfd_exit();
-out_virqfd:
        vfio_pci_uninit_perm_bits();
        return ret;
 }
 
        if (ret)
                goto err_cdev_add;
 
+       /* Start the virqfd cleanup handler used by some VFIO bus drivers */
+       ret = vfio_virqfd_init();
+       if (ret)
+               goto err_virqfd;
+
        pr_info(DRIVER_DESC " version: " DRIVER_VERSION "\n");
 
        /*
 
        return 0;
 
+err_virqfd:
+       cdev_del(&vfio.group_cdev);
 err_cdev_add:
        unregister_chrdev_region(vfio.group_devt, MINORMASK);
 err_alloc_chrdev:
 {
        WARN_ON(!list_empty(&vfio.group_list));
 
+       vfio_virqfd_exit();
        idr_destroy(&vfio.group_idr);
        cdev_del(&vfio.group_cdev);
        unregister_chrdev_region(vfio.group_devt, MINORMASK);