1fb1ff4c1SBharat Bhushan // SPDX-License-Identifier: (GPL-2.0+ OR BSD-3-Clause) 2fb1ff4c1SBharat Bhushan /* 3fb1ff4c1SBharat Bhushan * Copyright 2013-2016 Freescale Semiconductor Inc. 4fb1ff4c1SBharat Bhushan * Copyright 2016-2017,2019-2020 NXP 5fb1ff4c1SBharat Bhushan */ 6fb1ff4c1SBharat Bhushan 7fb1ff4c1SBharat Bhushan #include <linux/device.h> 8fb1ff4c1SBharat Bhushan #include <linux/iommu.h> 9fb1ff4c1SBharat Bhushan #include <linux/module.h> 10fb1ff4c1SBharat Bhushan #include <linux/mutex.h> 11fb1ff4c1SBharat Bhushan #include <linux/slab.h> 12fb1ff4c1SBharat Bhushan #include <linux/types.h> 13fb1ff4c1SBharat Bhushan #include <linux/vfio.h> 14fb1ff4c1SBharat Bhushan #include <linux/fsl/mc.h> 151bb141edSDiana Craciun #include <linux/delay.h> 1683e49179SDiana Craciun #include <linux/io-64-nonatomic-hi-lo.h> 17fb1ff4c1SBharat Bhushan 18fb1ff4c1SBharat Bhushan #include "vfio_fsl_mc_private.h" 19fb1ff4c1SBharat Bhushan 20704f5082SDiana Craciun static struct fsl_mc_driver vfio_fsl_mc_driver; 21704f5082SDiana Craciun 22f2ba7e8cSDiana Craciun static DEFINE_MUTEX(reflck_lock); 23f2ba7e8cSDiana Craciun 24f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_get(struct vfio_fsl_mc_reflck *reflck) 25f2ba7e8cSDiana Craciun { 26f2ba7e8cSDiana Craciun kref_get(&reflck->kref); 27f2ba7e8cSDiana Craciun } 28f2ba7e8cSDiana Craciun 29f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_release(struct kref *kref) 30f2ba7e8cSDiana Craciun { 31f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck *reflck = container_of(kref, 32f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck, 33f2ba7e8cSDiana Craciun kref); 34f2ba7e8cSDiana Craciun 35f2ba7e8cSDiana Craciun mutex_destroy(&reflck->lock); 36f2ba7e8cSDiana Craciun kfree(reflck); 37f2ba7e8cSDiana Craciun mutex_unlock(&reflck_lock); 38f2ba7e8cSDiana Craciun } 39f2ba7e8cSDiana Craciun 40f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_put(struct vfio_fsl_mc_reflck *reflck) 41f2ba7e8cSDiana Craciun { 42f2ba7e8cSDiana Craciun kref_put_mutex(&reflck->kref, vfio_fsl_mc_reflck_release, &reflck_lock); 43f2ba7e8cSDiana Craciun } 44f2ba7e8cSDiana Craciun 45f2ba7e8cSDiana Craciun static struct vfio_fsl_mc_reflck *vfio_fsl_mc_reflck_alloc(void) 46f2ba7e8cSDiana Craciun { 47f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck *reflck; 48f2ba7e8cSDiana Craciun 49f2ba7e8cSDiana Craciun reflck = kzalloc(sizeof(*reflck), GFP_KERNEL); 50f2ba7e8cSDiana Craciun if (!reflck) 51f2ba7e8cSDiana Craciun return ERR_PTR(-ENOMEM); 52f2ba7e8cSDiana Craciun 53f2ba7e8cSDiana Craciun kref_init(&reflck->kref); 54f2ba7e8cSDiana Craciun mutex_init(&reflck->lock); 55f2ba7e8cSDiana Craciun 56f2ba7e8cSDiana Craciun return reflck; 57f2ba7e8cSDiana Craciun } 58f2ba7e8cSDiana Craciun 59f2ba7e8cSDiana Craciun static int vfio_fsl_mc_reflck_attach(struct vfio_fsl_mc_device *vdev) 60f2ba7e8cSDiana Craciun { 61822e1a90SDiana Craciun int ret = 0; 62f2ba7e8cSDiana Craciun 63f2ba7e8cSDiana Craciun mutex_lock(&reflck_lock); 64f2ba7e8cSDiana Craciun if (is_fsl_mc_bus_dprc(vdev->mc_dev)) { 65f2ba7e8cSDiana Craciun vdev->reflck = vfio_fsl_mc_reflck_alloc(); 66f2ba7e8cSDiana Craciun ret = PTR_ERR_OR_ZERO(vdev->reflck); 67f2ba7e8cSDiana Craciun } else { 68f2ba7e8cSDiana Craciun struct device *mc_cont_dev = vdev->mc_dev->dev.parent; 69f2ba7e8cSDiana Craciun struct vfio_device *device; 70f2ba7e8cSDiana Craciun struct vfio_fsl_mc_device *cont_vdev; 71f2ba7e8cSDiana Craciun 72f2ba7e8cSDiana Craciun device = vfio_device_get_from_dev(mc_cont_dev); 73f2ba7e8cSDiana Craciun if (!device) { 74f2ba7e8cSDiana Craciun ret = -ENODEV; 75f2ba7e8cSDiana Craciun goto unlock; 76f2ba7e8cSDiana Craciun } 77f2ba7e8cSDiana Craciun 78*1e04ec14SJason Gunthorpe cont_vdev = 79*1e04ec14SJason Gunthorpe container_of(device, struct vfio_fsl_mc_device, vdev); 80f2ba7e8cSDiana Craciun if (!cont_vdev || !cont_vdev->reflck) { 81f2ba7e8cSDiana Craciun vfio_device_put(device); 82f2ba7e8cSDiana Craciun ret = -ENODEV; 83f2ba7e8cSDiana Craciun goto unlock; 84f2ba7e8cSDiana Craciun } 85f2ba7e8cSDiana Craciun vfio_fsl_mc_reflck_get(cont_vdev->reflck); 86f2ba7e8cSDiana Craciun vdev->reflck = cont_vdev->reflck; 87f2ba7e8cSDiana Craciun vfio_device_put(device); 88f2ba7e8cSDiana Craciun } 89f2ba7e8cSDiana Craciun 90f2ba7e8cSDiana Craciun unlock: 91f2ba7e8cSDiana Craciun mutex_unlock(&reflck_lock); 92f2ba7e8cSDiana Craciun return ret; 93f2ba7e8cSDiana Craciun } 94f2ba7e8cSDiana Craciun 95df747bcdSDiana Craciun static int vfio_fsl_mc_regions_init(struct vfio_fsl_mc_device *vdev) 96fb1ff4c1SBharat Bhushan { 97df747bcdSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 98df747bcdSDiana Craciun int count = mc_dev->obj_desc.region_count; 99df747bcdSDiana Craciun int i; 100df747bcdSDiana Craciun 101df747bcdSDiana Craciun vdev->regions = kcalloc(count, sizeof(struct vfio_fsl_mc_region), 102df747bcdSDiana Craciun GFP_KERNEL); 103df747bcdSDiana Craciun if (!vdev->regions) 104df747bcdSDiana Craciun return -ENOMEM; 105df747bcdSDiana Craciun 106df747bcdSDiana Craciun for (i = 0; i < count; i++) { 107df747bcdSDiana Craciun struct resource *res = &mc_dev->regions[i]; 10867247289SDiana Craciun int no_mmap = is_fsl_mc_bus_dprc(mc_dev); 109df747bcdSDiana Craciun 110df747bcdSDiana Craciun vdev->regions[i].addr = res->start; 111df747bcdSDiana Craciun vdev->regions[i].size = resource_size(res); 112df747bcdSDiana Craciun vdev->regions[i].type = mc_dev->regions[i].flags & IORESOURCE_BITS; 11367247289SDiana Craciun /* 11467247289SDiana Craciun * Only regions addressed with PAGE granularity may be 11567247289SDiana Craciun * MMAPed securely. 11667247289SDiana Craciun */ 11767247289SDiana Craciun if (!no_mmap && !(vdev->regions[i].addr & ~PAGE_MASK) && 11867247289SDiana Craciun !(vdev->regions[i].size & ~PAGE_MASK)) 11967247289SDiana Craciun vdev->regions[i].flags |= 12067247289SDiana Craciun VFIO_REGION_INFO_FLAG_MMAP; 1211bb141edSDiana Craciun vdev->regions[i].flags |= VFIO_REGION_INFO_FLAG_READ; 1221bb141edSDiana Craciun if (!(mc_dev->regions[i].flags & IORESOURCE_READONLY)) 1231bb141edSDiana Craciun vdev->regions[i].flags |= VFIO_REGION_INFO_FLAG_WRITE; 124df747bcdSDiana Craciun } 125fb1ff4c1SBharat Bhushan 126fb1ff4c1SBharat Bhushan return 0; 127fb1ff4c1SBharat Bhushan } 128fb1ff4c1SBharat Bhushan 129df747bcdSDiana Craciun static void vfio_fsl_mc_regions_cleanup(struct vfio_fsl_mc_device *vdev) 130df747bcdSDiana Craciun { 1311bb141edSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 1321bb141edSDiana Craciun int i; 1331bb141edSDiana Craciun 1341bb141edSDiana Craciun for (i = 0; i < mc_dev->obj_desc.region_count; i++) 1351bb141edSDiana Craciun iounmap(vdev->regions[i].ioaddr); 136df747bcdSDiana Craciun kfree(vdev->regions); 137df747bcdSDiana Craciun } 138df747bcdSDiana Craciun 1396df62c5bSJason Gunthorpe static int vfio_fsl_mc_open(struct vfio_device *core_vdev) 140df747bcdSDiana Craciun { 1416df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 1426df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 143df747bcdSDiana Craciun int ret; 144df747bcdSDiana Craciun 145df747bcdSDiana Craciun if (!try_module_get(THIS_MODULE)) 146df747bcdSDiana Craciun return -ENODEV; 147df747bcdSDiana Craciun 148f2ba7e8cSDiana Craciun mutex_lock(&vdev->reflck->lock); 149df747bcdSDiana Craciun if (!vdev->refcnt) { 150df747bcdSDiana Craciun ret = vfio_fsl_mc_regions_init(vdev); 151df747bcdSDiana Craciun if (ret) 152df747bcdSDiana Craciun goto err_reg_init; 153df747bcdSDiana Craciun } 154df747bcdSDiana Craciun vdev->refcnt++; 155df747bcdSDiana Craciun 156f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 157df747bcdSDiana Craciun 158df747bcdSDiana Craciun return 0; 159df747bcdSDiana Craciun 160df747bcdSDiana Craciun err_reg_init: 161f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 162df747bcdSDiana Craciun module_put(THIS_MODULE); 163df747bcdSDiana Craciun return ret; 164df747bcdSDiana Craciun } 165df747bcdSDiana Craciun 1666df62c5bSJason Gunthorpe static void vfio_fsl_mc_release(struct vfio_device *core_vdev) 167fb1ff4c1SBharat Bhushan { 1686df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 1696df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 170cc0ee20bSDiana Craciun int ret; 171df747bcdSDiana Craciun 172f2ba7e8cSDiana Craciun mutex_lock(&vdev->reflck->lock); 173df747bcdSDiana Craciun 174cc0ee20bSDiana Craciun if (!(--vdev->refcnt)) { 175cc0ee20bSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 176cc0ee20bSDiana Craciun struct device *cont_dev = fsl_mc_cont_dev(&mc_dev->dev); 177cc0ee20bSDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(cont_dev); 178cc0ee20bSDiana Craciun 179df747bcdSDiana Craciun vfio_fsl_mc_regions_cleanup(vdev); 180df747bcdSDiana Craciun 181cc0ee20bSDiana Craciun /* reset the device before cleaning up the interrupts */ 182cc0ee20bSDiana Craciun ret = dprc_reset_container(mc_cont->mc_io, 0, 183cc0ee20bSDiana Craciun mc_cont->mc_handle, 184cc0ee20bSDiana Craciun mc_cont->obj_desc.id, 185cc0ee20bSDiana Craciun DPRC_RESET_OPTION_NON_RECURSIVE); 186cc0ee20bSDiana Craciun 187cc0ee20bSDiana Craciun if (ret) { 188cc0ee20bSDiana Craciun dev_warn(&mc_cont->dev, "VFIO_FLS_MC: reset device has failed (%d)\n", 189cc0ee20bSDiana Craciun ret); 190cc0ee20bSDiana Craciun WARN_ON(1); 191cc0ee20bSDiana Craciun } 192cc0ee20bSDiana Craciun 193cc0ee20bSDiana Craciun vfio_fsl_mc_irqs_cleanup(vdev); 194cc0ee20bSDiana Craciun 195cc0ee20bSDiana Craciun fsl_mc_cleanup_irq_pool(mc_cont); 196cc0ee20bSDiana Craciun } 197cc0ee20bSDiana Craciun 198f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 199df747bcdSDiana Craciun 200fb1ff4c1SBharat Bhushan module_put(THIS_MODULE); 201fb1ff4c1SBharat Bhushan } 202fb1ff4c1SBharat Bhushan 2036df62c5bSJason Gunthorpe static long vfio_fsl_mc_ioctl(struct vfio_device *core_vdev, 2046df62c5bSJason Gunthorpe unsigned int cmd, unsigned long arg) 205fb1ff4c1SBharat Bhushan { 206f97f4c04SDiana Craciun unsigned long minsz; 2076df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 2086df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 209f97f4c04SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 210f97f4c04SDiana Craciun 211fb1ff4c1SBharat Bhushan switch (cmd) { 212fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_INFO: 213fb1ff4c1SBharat Bhushan { 214f97f4c04SDiana Craciun struct vfio_device_info info; 215f97f4c04SDiana Craciun 216f97f4c04SDiana Craciun minsz = offsetofend(struct vfio_device_info, num_irqs); 217f97f4c04SDiana Craciun 218f97f4c04SDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 219f97f4c04SDiana Craciun return -EFAULT; 220f97f4c04SDiana Craciun 221f97f4c04SDiana Craciun if (info.argsz < minsz) 222f97f4c04SDiana Craciun return -EINVAL; 223f97f4c04SDiana Craciun 224f97f4c04SDiana Craciun info.flags = VFIO_DEVICE_FLAGS_FSL_MC; 225ac93ab2bSDiana Craciun 226ac93ab2bSDiana Craciun if (is_fsl_mc_bus_dprc(mc_dev)) 227ac93ab2bSDiana Craciun info.flags |= VFIO_DEVICE_FLAGS_RESET; 228ac93ab2bSDiana Craciun 229f97f4c04SDiana Craciun info.num_regions = mc_dev->obj_desc.region_count; 230f97f4c04SDiana Craciun info.num_irqs = mc_dev->obj_desc.irq_count; 231f97f4c04SDiana Craciun 232f97f4c04SDiana Craciun return copy_to_user((void __user *)arg, &info, minsz) ? 233f97f4c04SDiana Craciun -EFAULT : 0; 234fb1ff4c1SBharat Bhushan } 235fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_REGION_INFO: 236fb1ff4c1SBharat Bhushan { 237df747bcdSDiana Craciun struct vfio_region_info info; 238df747bcdSDiana Craciun 239df747bcdSDiana Craciun minsz = offsetofend(struct vfio_region_info, offset); 240df747bcdSDiana Craciun 241df747bcdSDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 242df747bcdSDiana Craciun return -EFAULT; 243df747bcdSDiana Craciun 244df747bcdSDiana Craciun if (info.argsz < minsz) 245df747bcdSDiana Craciun return -EINVAL; 246df747bcdSDiana Craciun 247df747bcdSDiana Craciun if (info.index >= mc_dev->obj_desc.region_count) 248df747bcdSDiana Craciun return -EINVAL; 249df747bcdSDiana Craciun 250df747bcdSDiana Craciun /* map offset to the physical address */ 251df747bcdSDiana Craciun info.offset = VFIO_FSL_MC_INDEX_TO_OFFSET(info.index); 252df747bcdSDiana Craciun info.size = vdev->regions[info.index].size; 253df747bcdSDiana Craciun info.flags = vdev->regions[info.index].flags; 254df747bcdSDiana Craciun 25509699e56SDan Carpenter if (copy_to_user((void __user *)arg, &info, minsz)) 25609699e56SDan Carpenter return -EFAULT; 25709699e56SDan Carpenter return 0; 258fb1ff4c1SBharat Bhushan } 259fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_IRQ_INFO: 260fb1ff4c1SBharat Bhushan { 2612e0d2956SDiana Craciun struct vfio_irq_info info; 2622e0d2956SDiana Craciun 2632e0d2956SDiana Craciun minsz = offsetofend(struct vfio_irq_info, count); 2642e0d2956SDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 2652e0d2956SDiana Craciun return -EFAULT; 2662e0d2956SDiana Craciun 2672e0d2956SDiana Craciun if (info.argsz < minsz) 2682e0d2956SDiana Craciun return -EINVAL; 2692e0d2956SDiana Craciun 2702e0d2956SDiana Craciun if (info.index >= mc_dev->obj_desc.irq_count) 2712e0d2956SDiana Craciun return -EINVAL; 2722e0d2956SDiana Craciun 2732e0d2956SDiana Craciun info.flags = VFIO_IRQ_INFO_EVENTFD; 2742e0d2956SDiana Craciun info.count = 1; 2752e0d2956SDiana Craciun 27609699e56SDan Carpenter if (copy_to_user((void __user *)arg, &info, minsz)) 27709699e56SDan Carpenter return -EFAULT; 27809699e56SDan Carpenter return 0; 279fb1ff4c1SBharat Bhushan } 280fb1ff4c1SBharat Bhushan case VFIO_DEVICE_SET_IRQS: 281fb1ff4c1SBharat Bhushan { 2822e0d2956SDiana Craciun struct vfio_irq_set hdr; 2832e0d2956SDiana Craciun u8 *data = NULL; 2842e0d2956SDiana Craciun int ret = 0; 2852e0d2956SDiana Craciun size_t data_size = 0; 2862e0d2956SDiana Craciun 2872e0d2956SDiana Craciun minsz = offsetofend(struct vfio_irq_set, count); 2882e0d2956SDiana Craciun 2892e0d2956SDiana Craciun if (copy_from_user(&hdr, (void __user *)arg, minsz)) 2902e0d2956SDiana Craciun return -EFAULT; 2912e0d2956SDiana Craciun 2922e0d2956SDiana Craciun ret = vfio_set_irqs_validate_and_prepare(&hdr, mc_dev->obj_desc.irq_count, 2932e0d2956SDiana Craciun mc_dev->obj_desc.irq_count, &data_size); 2942e0d2956SDiana Craciun if (ret) 2952e0d2956SDiana Craciun return ret; 2962e0d2956SDiana Craciun 2972e0d2956SDiana Craciun if (data_size) { 2982e0d2956SDiana Craciun data = memdup_user((void __user *)(arg + minsz), 2992e0d2956SDiana Craciun data_size); 3002e0d2956SDiana Craciun if (IS_ERR(data)) 3012e0d2956SDiana Craciun return PTR_ERR(data); 3022e0d2956SDiana Craciun } 3032e0d2956SDiana Craciun 3042e0d2956SDiana Craciun mutex_lock(&vdev->igate); 3052e0d2956SDiana Craciun ret = vfio_fsl_mc_set_irqs_ioctl(vdev, hdr.flags, 3062e0d2956SDiana Craciun hdr.index, hdr.start, 3072e0d2956SDiana Craciun hdr.count, data); 3082e0d2956SDiana Craciun mutex_unlock(&vdev->igate); 3092e0d2956SDiana Craciun kfree(data); 3102e0d2956SDiana Craciun 3112e0d2956SDiana Craciun return ret; 312fb1ff4c1SBharat Bhushan } 313fb1ff4c1SBharat Bhushan case VFIO_DEVICE_RESET: 314fb1ff4c1SBharat Bhushan { 315ac93ab2bSDiana Craciun int ret; 316ac93ab2bSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 317ac93ab2bSDiana Craciun 318ac93ab2bSDiana Craciun /* reset is supported only for the DPRC */ 319ac93ab2bSDiana Craciun if (!is_fsl_mc_bus_dprc(mc_dev)) 320fb1ff4c1SBharat Bhushan return -ENOTTY; 321ac93ab2bSDiana Craciun 322ac93ab2bSDiana Craciun ret = dprc_reset_container(mc_dev->mc_io, 0, 323ac93ab2bSDiana Craciun mc_dev->mc_handle, 324ac93ab2bSDiana Craciun mc_dev->obj_desc.id, 325ac93ab2bSDiana Craciun DPRC_RESET_OPTION_NON_RECURSIVE); 326ac93ab2bSDiana Craciun return ret; 327ac93ab2bSDiana Craciun 328fb1ff4c1SBharat Bhushan } 329fb1ff4c1SBharat Bhushan default: 330fb1ff4c1SBharat Bhushan return -ENOTTY; 331fb1ff4c1SBharat Bhushan } 332fb1ff4c1SBharat Bhushan } 333fb1ff4c1SBharat Bhushan 3346df62c5bSJason Gunthorpe static ssize_t vfio_fsl_mc_read(struct vfio_device *core_vdev, char __user *buf, 335fb1ff4c1SBharat Bhushan size_t count, loff_t *ppos) 336fb1ff4c1SBharat Bhushan { 3376df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 3386df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 3391bb141edSDiana Craciun unsigned int index = VFIO_FSL_MC_OFFSET_TO_INDEX(*ppos); 3401bb141edSDiana Craciun loff_t off = *ppos & VFIO_FSL_MC_OFFSET_MASK; 3411bb141edSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 3421bb141edSDiana Craciun struct vfio_fsl_mc_region *region; 3431bb141edSDiana Craciun u64 data[8]; 3441bb141edSDiana Craciun int i; 3451bb141edSDiana Craciun 3461bb141edSDiana Craciun if (index >= mc_dev->obj_desc.region_count) 347fb1ff4c1SBharat Bhushan return -EINVAL; 3481bb141edSDiana Craciun 3491bb141edSDiana Craciun region = &vdev->regions[index]; 3501bb141edSDiana Craciun 3511bb141edSDiana Craciun if (!(region->flags & VFIO_REGION_INFO_FLAG_READ)) 3521bb141edSDiana Craciun return -EINVAL; 3531bb141edSDiana Craciun 3541bb141edSDiana Craciun if (!region->ioaddr) { 3551bb141edSDiana Craciun region->ioaddr = ioremap(region->addr, region->size); 3561bb141edSDiana Craciun if (!region->ioaddr) 3571bb141edSDiana Craciun return -ENOMEM; 3581bb141edSDiana Craciun } 3591bb141edSDiana Craciun 3601bb141edSDiana Craciun if (count != 64 || off != 0) 3611bb141edSDiana Craciun return -EINVAL; 3621bb141edSDiana Craciun 3631bb141edSDiana Craciun for (i = 7; i >= 0; i--) 3641bb141edSDiana Craciun data[i] = readq(region->ioaddr + i * sizeof(uint64_t)); 3651bb141edSDiana Craciun 3661bb141edSDiana Craciun if (copy_to_user(buf, data, 64)) 3671bb141edSDiana Craciun return -EFAULT; 3681bb141edSDiana Craciun 3691bb141edSDiana Craciun return count; 3701bb141edSDiana Craciun } 3711bb141edSDiana Craciun 3721bb141edSDiana Craciun #define MC_CMD_COMPLETION_TIMEOUT_MS 5000 3731bb141edSDiana Craciun #define MC_CMD_COMPLETION_POLLING_MAX_SLEEP_USECS 500 3741bb141edSDiana Craciun 3751bb141edSDiana Craciun static int vfio_fsl_mc_send_command(void __iomem *ioaddr, uint64_t *cmd_data) 3761bb141edSDiana Craciun { 3771bb141edSDiana Craciun int i; 3781bb141edSDiana Craciun enum mc_cmd_status status; 3791bb141edSDiana Craciun unsigned long timeout_usecs = MC_CMD_COMPLETION_TIMEOUT_MS * 1000; 3801bb141edSDiana Craciun 3811bb141edSDiana Craciun /* Write at command parameter into portal */ 3821bb141edSDiana Craciun for (i = 7; i >= 1; i--) 3831bb141edSDiana Craciun writeq_relaxed(cmd_data[i], ioaddr + i * sizeof(uint64_t)); 3841bb141edSDiana Craciun 3851bb141edSDiana Craciun /* Write command header in the end */ 3861bb141edSDiana Craciun writeq(cmd_data[0], ioaddr); 3871bb141edSDiana Craciun 3881bb141edSDiana Craciun /* Wait for response before returning to user-space 3891bb141edSDiana Craciun * This can be optimized in future to even prepare response 3901bb141edSDiana Craciun * before returning to user-space and avoid read ioctl. 3911bb141edSDiana Craciun */ 3921bb141edSDiana Craciun for (;;) { 3931bb141edSDiana Craciun u64 header; 3941bb141edSDiana Craciun struct mc_cmd_header *resp_hdr; 3951bb141edSDiana Craciun 3961bb141edSDiana Craciun header = cpu_to_le64(readq_relaxed(ioaddr)); 3971bb141edSDiana Craciun 3981bb141edSDiana Craciun resp_hdr = (struct mc_cmd_header *)&header; 3991bb141edSDiana Craciun status = (enum mc_cmd_status)resp_hdr->status; 4001bb141edSDiana Craciun if (status != MC_CMD_STATUS_READY) 4011bb141edSDiana Craciun break; 4021bb141edSDiana Craciun 4031bb141edSDiana Craciun udelay(MC_CMD_COMPLETION_POLLING_MAX_SLEEP_USECS); 4041bb141edSDiana Craciun timeout_usecs -= MC_CMD_COMPLETION_POLLING_MAX_SLEEP_USECS; 4051bb141edSDiana Craciun if (timeout_usecs == 0) 4061bb141edSDiana Craciun return -ETIMEDOUT; 4071bb141edSDiana Craciun } 4081bb141edSDiana Craciun 4091bb141edSDiana Craciun return 0; 410fb1ff4c1SBharat Bhushan } 411fb1ff4c1SBharat Bhushan 4126df62c5bSJason Gunthorpe static ssize_t vfio_fsl_mc_write(struct vfio_device *core_vdev, 4136df62c5bSJason Gunthorpe const char __user *buf, size_t count, 4146df62c5bSJason Gunthorpe loff_t *ppos) 415fb1ff4c1SBharat Bhushan { 4166df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 4176df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 4181bb141edSDiana Craciun unsigned int index = VFIO_FSL_MC_OFFSET_TO_INDEX(*ppos); 4191bb141edSDiana Craciun loff_t off = *ppos & VFIO_FSL_MC_OFFSET_MASK; 4201bb141edSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 4211bb141edSDiana Craciun struct vfio_fsl_mc_region *region; 4221bb141edSDiana Craciun u64 data[8]; 4231bb141edSDiana Craciun int ret; 4241bb141edSDiana Craciun 4251bb141edSDiana Craciun if (index >= mc_dev->obj_desc.region_count) 426fb1ff4c1SBharat Bhushan return -EINVAL; 4271bb141edSDiana Craciun 4281bb141edSDiana Craciun region = &vdev->regions[index]; 4291bb141edSDiana Craciun 4301bb141edSDiana Craciun if (!(region->flags & VFIO_REGION_INFO_FLAG_WRITE)) 4311bb141edSDiana Craciun return -EINVAL; 4321bb141edSDiana Craciun 4331bb141edSDiana Craciun if (!region->ioaddr) { 4341bb141edSDiana Craciun region->ioaddr = ioremap(region->addr, region->size); 4351bb141edSDiana Craciun if (!region->ioaddr) 4361bb141edSDiana Craciun return -ENOMEM; 4371bb141edSDiana Craciun } 4381bb141edSDiana Craciun 4391bb141edSDiana Craciun if (count != 64 || off != 0) 4401bb141edSDiana Craciun return -EINVAL; 4411bb141edSDiana Craciun 4421bb141edSDiana Craciun if (copy_from_user(&data, buf, 64)) 4431bb141edSDiana Craciun return -EFAULT; 4441bb141edSDiana Craciun 4451bb141edSDiana Craciun ret = vfio_fsl_mc_send_command(region->ioaddr, data); 4461bb141edSDiana Craciun if (ret) 4471bb141edSDiana Craciun return ret; 4481bb141edSDiana Craciun 4491bb141edSDiana Craciun return count; 4501bb141edSDiana Craciun 451fb1ff4c1SBharat Bhushan } 452fb1ff4c1SBharat Bhushan 45367247289SDiana Craciun static int vfio_fsl_mc_mmap_mmio(struct vfio_fsl_mc_region region, 45467247289SDiana Craciun struct vm_area_struct *vma) 45567247289SDiana Craciun { 45667247289SDiana Craciun u64 size = vma->vm_end - vma->vm_start; 45767247289SDiana Craciun u64 pgoff, base; 45867247289SDiana Craciun u8 region_cacheable; 45967247289SDiana Craciun 46067247289SDiana Craciun pgoff = vma->vm_pgoff & 46167247289SDiana Craciun ((1U << (VFIO_FSL_MC_OFFSET_SHIFT - PAGE_SHIFT)) - 1); 46267247289SDiana Craciun base = pgoff << PAGE_SHIFT; 46367247289SDiana Craciun 46467247289SDiana Craciun if (region.size < PAGE_SIZE || base + size > region.size) 46567247289SDiana Craciun return -EINVAL; 46667247289SDiana Craciun 46767247289SDiana Craciun region_cacheable = (region.type & FSL_MC_REGION_CACHEABLE) && 46867247289SDiana Craciun (region.type & FSL_MC_REGION_SHAREABLE); 46967247289SDiana Craciun if (!region_cacheable) 47067247289SDiana Craciun vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); 47167247289SDiana Craciun 47267247289SDiana Craciun vma->vm_pgoff = (region.addr >> PAGE_SHIFT) + pgoff; 47367247289SDiana Craciun 47467247289SDiana Craciun return remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff, 47567247289SDiana Craciun size, vma->vm_page_prot); 47667247289SDiana Craciun } 47767247289SDiana Craciun 4786df62c5bSJason Gunthorpe static int vfio_fsl_mc_mmap(struct vfio_device *core_vdev, 4796df62c5bSJason Gunthorpe struct vm_area_struct *vma) 480fb1ff4c1SBharat Bhushan { 4816df62c5bSJason Gunthorpe struct vfio_fsl_mc_device *vdev = 4826df62c5bSJason Gunthorpe container_of(core_vdev, struct vfio_fsl_mc_device, vdev); 48367247289SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 48469848cd6SDan Carpenter unsigned int index; 48567247289SDiana Craciun 48667247289SDiana Craciun index = vma->vm_pgoff >> (VFIO_FSL_MC_OFFSET_SHIFT - PAGE_SHIFT); 48767247289SDiana Craciun 48867247289SDiana Craciun if (vma->vm_end < vma->vm_start) 489fb1ff4c1SBharat Bhushan return -EINVAL; 49067247289SDiana Craciun if (vma->vm_start & ~PAGE_MASK) 49167247289SDiana Craciun return -EINVAL; 49267247289SDiana Craciun if (vma->vm_end & ~PAGE_MASK) 49367247289SDiana Craciun return -EINVAL; 49467247289SDiana Craciun if (!(vma->vm_flags & VM_SHARED)) 49567247289SDiana Craciun return -EINVAL; 49667247289SDiana Craciun if (index >= mc_dev->obj_desc.region_count) 49767247289SDiana Craciun return -EINVAL; 49867247289SDiana Craciun 49967247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_MMAP)) 50067247289SDiana Craciun return -EINVAL; 50167247289SDiana Craciun 50267247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_READ) 50367247289SDiana Craciun && (vma->vm_flags & VM_READ)) 50467247289SDiana Craciun return -EINVAL; 50567247289SDiana Craciun 50667247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_WRITE) 50767247289SDiana Craciun && (vma->vm_flags & VM_WRITE)) 50867247289SDiana Craciun return -EINVAL; 50967247289SDiana Craciun 51067247289SDiana Craciun vma->vm_private_data = mc_dev; 51167247289SDiana Craciun 51267247289SDiana Craciun return vfio_fsl_mc_mmap_mmio(vdev->regions[index], vma); 513fb1ff4c1SBharat Bhushan } 514fb1ff4c1SBharat Bhushan 515fb1ff4c1SBharat Bhushan static const struct vfio_device_ops vfio_fsl_mc_ops = { 516fb1ff4c1SBharat Bhushan .name = "vfio-fsl-mc", 517fb1ff4c1SBharat Bhushan .open = vfio_fsl_mc_open, 518fb1ff4c1SBharat Bhushan .release = vfio_fsl_mc_release, 519fb1ff4c1SBharat Bhushan .ioctl = vfio_fsl_mc_ioctl, 520fb1ff4c1SBharat Bhushan .read = vfio_fsl_mc_read, 521fb1ff4c1SBharat Bhushan .write = vfio_fsl_mc_write, 522fb1ff4c1SBharat Bhushan .mmap = vfio_fsl_mc_mmap, 523fb1ff4c1SBharat Bhushan }; 524fb1ff4c1SBharat Bhushan 525704f5082SDiana Craciun static int vfio_fsl_mc_bus_notifier(struct notifier_block *nb, 526704f5082SDiana Craciun unsigned long action, void *data) 527704f5082SDiana Craciun { 528704f5082SDiana Craciun struct vfio_fsl_mc_device *vdev = container_of(nb, 529704f5082SDiana Craciun struct vfio_fsl_mc_device, nb); 530704f5082SDiana Craciun struct device *dev = data; 531704f5082SDiana Craciun struct fsl_mc_device *mc_dev = to_fsl_mc_device(dev); 532704f5082SDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(mc_dev->dev.parent); 533704f5082SDiana Craciun 534704f5082SDiana Craciun if (action == BUS_NOTIFY_ADD_DEVICE && 535704f5082SDiana Craciun vdev->mc_dev == mc_cont) { 536704f5082SDiana Craciun mc_dev->driver_override = kasprintf(GFP_KERNEL, "%s", 537704f5082SDiana Craciun vfio_fsl_mc_ops.name); 538704f5082SDiana Craciun if (!mc_dev->driver_override) 539704f5082SDiana Craciun dev_warn(dev, "VFIO_FSL_MC: Setting driver override for device in dprc %s failed\n", 540704f5082SDiana Craciun dev_name(&mc_cont->dev)); 541704f5082SDiana Craciun else 542704f5082SDiana Craciun dev_info(dev, "VFIO_FSL_MC: Setting driver override for device in dprc %s\n", 543704f5082SDiana Craciun dev_name(&mc_cont->dev)); 544704f5082SDiana Craciun } else if (action == BUS_NOTIFY_BOUND_DRIVER && 545704f5082SDiana Craciun vdev->mc_dev == mc_cont) { 546704f5082SDiana Craciun struct fsl_mc_driver *mc_drv = to_fsl_mc_driver(dev->driver); 547704f5082SDiana Craciun 548704f5082SDiana Craciun if (mc_drv && mc_drv != &vfio_fsl_mc_driver) 549704f5082SDiana Craciun dev_warn(dev, "VFIO_FSL_MC: Object %s bound to driver %s while DPRC bound to vfio-fsl-mc\n", 550704f5082SDiana Craciun dev_name(dev), mc_drv->driver.name); 551704f5082SDiana Craciun } 552704f5082SDiana Craciun 553704f5082SDiana Craciun return 0; 554704f5082SDiana Craciun } 555704f5082SDiana Craciun 556704f5082SDiana Craciun static int vfio_fsl_mc_init_device(struct vfio_fsl_mc_device *vdev) 557704f5082SDiana Craciun { 558704f5082SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 559704f5082SDiana Craciun int ret; 560704f5082SDiana Craciun 561704f5082SDiana Craciun /* Non-dprc devices share mc_io from parent */ 562704f5082SDiana Craciun if (!is_fsl_mc_bus_dprc(mc_dev)) { 563704f5082SDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(mc_dev->dev.parent); 564704f5082SDiana Craciun 565704f5082SDiana Craciun mc_dev->mc_io = mc_cont->mc_io; 566704f5082SDiana Craciun return 0; 567704f5082SDiana Craciun } 568704f5082SDiana Craciun 569704f5082SDiana Craciun vdev->nb.notifier_call = vfio_fsl_mc_bus_notifier; 570704f5082SDiana Craciun ret = bus_register_notifier(&fsl_mc_bus_type, &vdev->nb); 571704f5082SDiana Craciun if (ret) 572704f5082SDiana Craciun return ret; 573704f5082SDiana Craciun 574704f5082SDiana Craciun /* open DPRC, allocate a MC portal */ 575704f5082SDiana Craciun ret = dprc_setup(mc_dev); 576704f5082SDiana Craciun if (ret) { 577704f5082SDiana Craciun dev_err(&mc_dev->dev, "VFIO_FSL_MC: Failed to setup DPRC (%d)\n", ret); 578704f5082SDiana Craciun goto out_nc_unreg; 579704f5082SDiana Craciun } 580704f5082SDiana Craciun return 0; 581704f5082SDiana Craciun 582704f5082SDiana Craciun out_nc_unreg: 583704f5082SDiana Craciun bus_unregister_notifier(&fsl_mc_bus_type, &vdev->nb); 584704f5082SDiana Craciun return ret; 585704f5082SDiana Craciun } 586704f5082SDiana Craciun 5872b1fe162SJason Gunthorpe static int vfio_fsl_mc_scan_container(struct fsl_mc_device *mc_dev) 5882b1fe162SJason Gunthorpe { 5892b1fe162SJason Gunthorpe int ret; 5902b1fe162SJason Gunthorpe 5912b1fe162SJason Gunthorpe /* non dprc devices do not scan for other devices */ 5922b1fe162SJason Gunthorpe if (!is_fsl_mc_bus_dprc(mc_dev)) 5932b1fe162SJason Gunthorpe return 0; 5942b1fe162SJason Gunthorpe ret = dprc_scan_container(mc_dev, false); 5952b1fe162SJason Gunthorpe if (ret) { 5962b1fe162SJason Gunthorpe dev_err(&mc_dev->dev, 5972b1fe162SJason Gunthorpe "VFIO_FSL_MC: Container scanning failed (%d)\n", ret); 5982b1fe162SJason Gunthorpe dprc_remove_devices(mc_dev, NULL, 0); 5992b1fe162SJason Gunthorpe return ret; 6002b1fe162SJason Gunthorpe } 6012b1fe162SJason Gunthorpe return 0; 6022b1fe162SJason Gunthorpe } 6032b1fe162SJason Gunthorpe 6042b1fe162SJason Gunthorpe static void vfio_fsl_uninit_device(struct vfio_fsl_mc_device *vdev) 6052b1fe162SJason Gunthorpe { 6062b1fe162SJason Gunthorpe struct fsl_mc_device *mc_dev = vdev->mc_dev; 6072b1fe162SJason Gunthorpe 6082b1fe162SJason Gunthorpe if (!is_fsl_mc_bus_dprc(mc_dev)) 6092b1fe162SJason Gunthorpe return; 6102b1fe162SJason Gunthorpe 6112b1fe162SJason Gunthorpe dprc_cleanup(mc_dev); 6122b1fe162SJason Gunthorpe bus_unregister_notifier(&fsl_mc_bus_type, &vdev->nb); 6132b1fe162SJason Gunthorpe } 6142b1fe162SJason Gunthorpe 615fb1ff4c1SBharat Bhushan static int vfio_fsl_mc_probe(struct fsl_mc_device *mc_dev) 616fb1ff4c1SBharat Bhushan { 617fb1ff4c1SBharat Bhushan struct iommu_group *group; 618fb1ff4c1SBharat Bhushan struct vfio_fsl_mc_device *vdev; 619fb1ff4c1SBharat Bhushan struct device *dev = &mc_dev->dev; 620fb1ff4c1SBharat Bhushan int ret; 621fb1ff4c1SBharat Bhushan 622fb1ff4c1SBharat Bhushan group = vfio_iommu_group_get(dev); 623fb1ff4c1SBharat Bhushan if (!group) { 624fb1ff4c1SBharat Bhushan dev_err(dev, "VFIO_FSL_MC: No IOMMU group\n"); 625fb1ff4c1SBharat Bhushan return -EINVAL; 626fb1ff4c1SBharat Bhushan } 627fb1ff4c1SBharat Bhushan 6280ca78666SJason Gunthorpe vdev = kzalloc(sizeof(*vdev), GFP_KERNEL); 629fb1ff4c1SBharat Bhushan if (!vdev) { 630fb1ff4c1SBharat Bhushan ret = -ENOMEM; 631fb1ff4c1SBharat Bhushan goto out_group_put; 632fb1ff4c1SBharat Bhushan } 633fb1ff4c1SBharat Bhushan 634*1e04ec14SJason Gunthorpe vfio_init_group_dev(&vdev->vdev, dev, &vfio_fsl_mc_ops); 635fb1ff4c1SBharat Bhushan vdev->mc_dev = mc_dev; 6362b1fe162SJason Gunthorpe mutex_init(&vdev->igate); 637704f5082SDiana Craciun 638f2ba7e8cSDiana Craciun ret = vfio_fsl_mc_reflck_attach(vdev); 639704f5082SDiana Craciun if (ret) 6400ca78666SJason Gunthorpe goto out_kfree; 641704f5082SDiana Craciun 642f2ba7e8cSDiana Craciun ret = vfio_fsl_mc_init_device(vdev); 643f2ba7e8cSDiana Craciun if (ret) 644f2ba7e8cSDiana Craciun goto out_reflck; 645df747bcdSDiana Craciun 6460ca78666SJason Gunthorpe ret = vfio_register_group_dev(&vdev->vdev); 6472b1fe162SJason Gunthorpe if (ret) { 6482b1fe162SJason Gunthorpe dev_err(dev, "VFIO_FSL_MC: Failed to add to vfio group\n"); 6492b1fe162SJason Gunthorpe goto out_device; 6502b1fe162SJason Gunthorpe } 6512e0d2956SDiana Craciun 6522b1fe162SJason Gunthorpe /* 6532b1fe162SJason Gunthorpe * This triggers recursion into vfio_fsl_mc_probe() on another device 6542b1fe162SJason Gunthorpe * and the vfio_fsl_mc_reflck_attach() must succeed, which relies on the 6552b1fe162SJason Gunthorpe * vfio_add_group_dev() above. It has no impact on this vdev, so it is 6562b1fe162SJason Gunthorpe * safe to be after the vfio device is made live. 6572b1fe162SJason Gunthorpe */ 6582b1fe162SJason Gunthorpe ret = vfio_fsl_mc_scan_container(mc_dev); 6592b1fe162SJason Gunthorpe if (ret) 6602b1fe162SJason Gunthorpe goto out_group_dev; 6610ca78666SJason Gunthorpe dev_set_drvdata(dev, vdev); 662fb1ff4c1SBharat Bhushan return 0; 663fb1ff4c1SBharat Bhushan 664704f5082SDiana Craciun out_group_dev: 6650ca78666SJason Gunthorpe vfio_unregister_group_dev(&vdev->vdev); 6662b1fe162SJason Gunthorpe out_device: 6672b1fe162SJason Gunthorpe vfio_fsl_uninit_device(vdev); 6682b1fe162SJason Gunthorpe out_reflck: 6692b1fe162SJason Gunthorpe vfio_fsl_mc_reflck_put(vdev->reflck); 6700ca78666SJason Gunthorpe out_kfree: 6710ca78666SJason Gunthorpe kfree(vdev); 672fb1ff4c1SBharat Bhushan out_group_put: 673fb1ff4c1SBharat Bhushan vfio_iommu_group_put(group, dev); 674fb1ff4c1SBharat Bhushan return ret; 675fb1ff4c1SBharat Bhushan } 676fb1ff4c1SBharat Bhushan 677fb1ff4c1SBharat Bhushan static int vfio_fsl_mc_remove(struct fsl_mc_device *mc_dev) 678fb1ff4c1SBharat Bhushan { 679fb1ff4c1SBharat Bhushan struct device *dev = &mc_dev->dev; 6800ca78666SJason Gunthorpe struct vfio_fsl_mc_device *vdev = dev_get_drvdata(dev); 681fb1ff4c1SBharat Bhushan 6820ca78666SJason Gunthorpe vfio_unregister_group_dev(&vdev->vdev); 6832e0d2956SDiana Craciun mutex_destroy(&vdev->igate); 6842e0d2956SDiana Craciun 685704f5082SDiana Craciun dprc_remove_devices(mc_dev, NULL, 0); 6862b1fe162SJason Gunthorpe vfio_fsl_uninit_device(vdev); 6872b1fe162SJason Gunthorpe vfio_fsl_mc_reflck_put(vdev->reflck); 688704f5082SDiana Craciun 6890ca78666SJason Gunthorpe kfree(vdev); 690fb1ff4c1SBharat Bhushan vfio_iommu_group_put(mc_dev->dev.iommu_group, dev); 691fb1ff4c1SBharat Bhushan 692fb1ff4c1SBharat Bhushan return 0; 693fb1ff4c1SBharat Bhushan } 694fb1ff4c1SBharat Bhushan 695fb1ff4c1SBharat Bhushan static struct fsl_mc_driver vfio_fsl_mc_driver = { 696fb1ff4c1SBharat Bhushan .probe = vfio_fsl_mc_probe, 697fb1ff4c1SBharat Bhushan .remove = vfio_fsl_mc_remove, 698fb1ff4c1SBharat Bhushan .driver = { 699fb1ff4c1SBharat Bhushan .name = "vfio-fsl-mc", 700fb1ff4c1SBharat Bhushan .owner = THIS_MODULE, 701fb1ff4c1SBharat Bhushan }, 702fb1ff4c1SBharat Bhushan }; 703fb1ff4c1SBharat Bhushan 704fb1ff4c1SBharat Bhushan static int __init vfio_fsl_mc_driver_init(void) 705fb1ff4c1SBharat Bhushan { 706fb1ff4c1SBharat Bhushan return fsl_mc_driver_register(&vfio_fsl_mc_driver); 707fb1ff4c1SBharat Bhushan } 708fb1ff4c1SBharat Bhushan 709fb1ff4c1SBharat Bhushan static void __exit vfio_fsl_mc_driver_exit(void) 710fb1ff4c1SBharat Bhushan { 711fb1ff4c1SBharat Bhushan fsl_mc_driver_unregister(&vfio_fsl_mc_driver); 712fb1ff4c1SBharat Bhushan } 713fb1ff4c1SBharat Bhushan 714fb1ff4c1SBharat Bhushan module_init(vfio_fsl_mc_driver_init); 715fb1ff4c1SBharat Bhushan module_exit(vfio_fsl_mc_driver_exit); 716fb1ff4c1SBharat Bhushan 717fb1ff4c1SBharat Bhushan MODULE_LICENSE("Dual BSD/GPL"); 718fb1ff4c1SBharat Bhushan MODULE_DESCRIPTION("VFIO for FSL-MC devices - User Level meta-driver"); 719