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> 15fb1ff4c1SBharat Bhushan 16fb1ff4c1SBharat Bhushan #include "vfio_fsl_mc_private.h" 17fb1ff4c1SBharat Bhushan 18704f5082SDiana Craciun static struct fsl_mc_driver vfio_fsl_mc_driver; 19704f5082SDiana Craciun 20f2ba7e8cSDiana Craciun static DEFINE_MUTEX(reflck_lock); 21f2ba7e8cSDiana Craciun 22f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_get(struct vfio_fsl_mc_reflck *reflck) 23f2ba7e8cSDiana Craciun { 24f2ba7e8cSDiana Craciun kref_get(&reflck->kref); 25f2ba7e8cSDiana Craciun } 26f2ba7e8cSDiana Craciun 27f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_release(struct kref *kref) 28f2ba7e8cSDiana Craciun { 29f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck *reflck = container_of(kref, 30f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck, 31f2ba7e8cSDiana Craciun kref); 32f2ba7e8cSDiana Craciun 33f2ba7e8cSDiana Craciun mutex_destroy(&reflck->lock); 34f2ba7e8cSDiana Craciun kfree(reflck); 35f2ba7e8cSDiana Craciun mutex_unlock(&reflck_lock); 36f2ba7e8cSDiana Craciun } 37f2ba7e8cSDiana Craciun 38f2ba7e8cSDiana Craciun static void vfio_fsl_mc_reflck_put(struct vfio_fsl_mc_reflck *reflck) 39f2ba7e8cSDiana Craciun { 40f2ba7e8cSDiana Craciun kref_put_mutex(&reflck->kref, vfio_fsl_mc_reflck_release, &reflck_lock); 41f2ba7e8cSDiana Craciun } 42f2ba7e8cSDiana Craciun 43f2ba7e8cSDiana Craciun static struct vfio_fsl_mc_reflck *vfio_fsl_mc_reflck_alloc(void) 44f2ba7e8cSDiana Craciun { 45f2ba7e8cSDiana Craciun struct vfio_fsl_mc_reflck *reflck; 46f2ba7e8cSDiana Craciun 47f2ba7e8cSDiana Craciun reflck = kzalloc(sizeof(*reflck), GFP_KERNEL); 48f2ba7e8cSDiana Craciun if (!reflck) 49f2ba7e8cSDiana Craciun return ERR_PTR(-ENOMEM); 50f2ba7e8cSDiana Craciun 51f2ba7e8cSDiana Craciun kref_init(&reflck->kref); 52f2ba7e8cSDiana Craciun mutex_init(&reflck->lock); 53f2ba7e8cSDiana Craciun 54f2ba7e8cSDiana Craciun return reflck; 55f2ba7e8cSDiana Craciun } 56f2ba7e8cSDiana Craciun 57f2ba7e8cSDiana Craciun static int vfio_fsl_mc_reflck_attach(struct vfio_fsl_mc_device *vdev) 58f2ba7e8cSDiana Craciun { 59f2ba7e8cSDiana Craciun int ret; 60f2ba7e8cSDiana Craciun 61f2ba7e8cSDiana Craciun mutex_lock(&reflck_lock); 62f2ba7e8cSDiana Craciun if (is_fsl_mc_bus_dprc(vdev->mc_dev)) { 63f2ba7e8cSDiana Craciun vdev->reflck = vfio_fsl_mc_reflck_alloc(); 64f2ba7e8cSDiana Craciun ret = PTR_ERR_OR_ZERO(vdev->reflck); 65f2ba7e8cSDiana Craciun } else { 66f2ba7e8cSDiana Craciun struct device *mc_cont_dev = vdev->mc_dev->dev.parent; 67f2ba7e8cSDiana Craciun struct vfio_device *device; 68f2ba7e8cSDiana Craciun struct vfio_fsl_mc_device *cont_vdev; 69f2ba7e8cSDiana Craciun 70f2ba7e8cSDiana Craciun device = vfio_device_get_from_dev(mc_cont_dev); 71f2ba7e8cSDiana Craciun if (!device) { 72f2ba7e8cSDiana Craciun ret = -ENODEV; 73f2ba7e8cSDiana Craciun goto unlock; 74f2ba7e8cSDiana Craciun } 75f2ba7e8cSDiana Craciun 76f2ba7e8cSDiana Craciun cont_vdev = vfio_device_data(device); 77f2ba7e8cSDiana Craciun if (!cont_vdev || !cont_vdev->reflck) { 78f2ba7e8cSDiana Craciun vfio_device_put(device); 79f2ba7e8cSDiana Craciun ret = -ENODEV; 80f2ba7e8cSDiana Craciun goto unlock; 81f2ba7e8cSDiana Craciun } 82f2ba7e8cSDiana Craciun vfio_fsl_mc_reflck_get(cont_vdev->reflck); 83f2ba7e8cSDiana Craciun vdev->reflck = cont_vdev->reflck; 84f2ba7e8cSDiana Craciun vfio_device_put(device); 85f2ba7e8cSDiana Craciun } 86f2ba7e8cSDiana Craciun 87f2ba7e8cSDiana Craciun unlock: 88f2ba7e8cSDiana Craciun mutex_unlock(&reflck_lock); 89f2ba7e8cSDiana Craciun return ret; 90f2ba7e8cSDiana Craciun } 91f2ba7e8cSDiana Craciun 92df747bcdSDiana Craciun static int vfio_fsl_mc_regions_init(struct vfio_fsl_mc_device *vdev) 93fb1ff4c1SBharat Bhushan { 94df747bcdSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 95df747bcdSDiana Craciun int count = mc_dev->obj_desc.region_count; 96df747bcdSDiana Craciun int i; 97df747bcdSDiana Craciun 98df747bcdSDiana Craciun vdev->regions = kcalloc(count, sizeof(struct vfio_fsl_mc_region), 99df747bcdSDiana Craciun GFP_KERNEL); 100df747bcdSDiana Craciun if (!vdev->regions) 101df747bcdSDiana Craciun return -ENOMEM; 102df747bcdSDiana Craciun 103df747bcdSDiana Craciun for (i = 0; i < count; i++) { 104df747bcdSDiana Craciun struct resource *res = &mc_dev->regions[i]; 10567247289SDiana Craciun int no_mmap = is_fsl_mc_bus_dprc(mc_dev); 106df747bcdSDiana Craciun 107df747bcdSDiana Craciun vdev->regions[i].addr = res->start; 108df747bcdSDiana Craciun vdev->regions[i].size = resource_size(res); 109df747bcdSDiana Craciun vdev->regions[i].type = mc_dev->regions[i].flags & IORESOURCE_BITS; 11067247289SDiana Craciun /* 11167247289SDiana Craciun * Only regions addressed with PAGE granularity may be 11267247289SDiana Craciun * MMAPed securely. 11367247289SDiana Craciun */ 11467247289SDiana Craciun if (!no_mmap && !(vdev->regions[i].addr & ~PAGE_MASK) && 11567247289SDiana Craciun !(vdev->regions[i].size & ~PAGE_MASK)) 11667247289SDiana Craciun vdev->regions[i].flags |= 11767247289SDiana Craciun VFIO_REGION_INFO_FLAG_MMAP; 11867247289SDiana Craciun 119df747bcdSDiana Craciun } 120fb1ff4c1SBharat Bhushan 121fb1ff4c1SBharat Bhushan return 0; 122fb1ff4c1SBharat Bhushan } 123fb1ff4c1SBharat Bhushan 124df747bcdSDiana Craciun static void vfio_fsl_mc_regions_cleanup(struct vfio_fsl_mc_device *vdev) 125df747bcdSDiana Craciun { 126df747bcdSDiana Craciun kfree(vdev->regions); 127df747bcdSDiana Craciun } 128df747bcdSDiana Craciun 129df747bcdSDiana Craciun static int vfio_fsl_mc_open(void *device_data) 130df747bcdSDiana Craciun { 131df747bcdSDiana Craciun struct vfio_fsl_mc_device *vdev = device_data; 132df747bcdSDiana Craciun int ret; 133df747bcdSDiana Craciun 134df747bcdSDiana Craciun if (!try_module_get(THIS_MODULE)) 135df747bcdSDiana Craciun return -ENODEV; 136df747bcdSDiana Craciun 137f2ba7e8cSDiana Craciun mutex_lock(&vdev->reflck->lock); 138df747bcdSDiana Craciun if (!vdev->refcnt) { 139df747bcdSDiana Craciun ret = vfio_fsl_mc_regions_init(vdev); 140df747bcdSDiana Craciun if (ret) 141df747bcdSDiana Craciun goto err_reg_init; 142df747bcdSDiana Craciun } 143df747bcdSDiana Craciun vdev->refcnt++; 144df747bcdSDiana Craciun 145f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 146df747bcdSDiana Craciun 147df747bcdSDiana Craciun return 0; 148df747bcdSDiana Craciun 149df747bcdSDiana Craciun err_reg_init: 150f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 151df747bcdSDiana Craciun module_put(THIS_MODULE); 152df747bcdSDiana Craciun return ret; 153df747bcdSDiana Craciun } 154df747bcdSDiana Craciun 155fb1ff4c1SBharat Bhushan static void vfio_fsl_mc_release(void *device_data) 156fb1ff4c1SBharat Bhushan { 157df747bcdSDiana Craciun struct vfio_fsl_mc_device *vdev = device_data; 158*cc0ee20bSDiana Craciun int ret; 159df747bcdSDiana Craciun 160f2ba7e8cSDiana Craciun mutex_lock(&vdev->reflck->lock); 161df747bcdSDiana Craciun 162*cc0ee20bSDiana Craciun if (!(--vdev->refcnt)) { 163*cc0ee20bSDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 164*cc0ee20bSDiana Craciun struct device *cont_dev = fsl_mc_cont_dev(&mc_dev->dev); 165*cc0ee20bSDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(cont_dev); 166*cc0ee20bSDiana Craciun 167df747bcdSDiana Craciun vfio_fsl_mc_regions_cleanup(vdev); 168df747bcdSDiana Craciun 169*cc0ee20bSDiana Craciun /* reset the device before cleaning up the interrupts */ 170*cc0ee20bSDiana Craciun ret = dprc_reset_container(mc_cont->mc_io, 0, 171*cc0ee20bSDiana Craciun mc_cont->mc_handle, 172*cc0ee20bSDiana Craciun mc_cont->obj_desc.id, 173*cc0ee20bSDiana Craciun DPRC_RESET_OPTION_NON_RECURSIVE); 174*cc0ee20bSDiana Craciun 175*cc0ee20bSDiana Craciun if (ret) { 176*cc0ee20bSDiana Craciun dev_warn(&mc_cont->dev, "VFIO_FLS_MC: reset device has failed (%d)\n", 177*cc0ee20bSDiana Craciun ret); 178*cc0ee20bSDiana Craciun WARN_ON(1); 179*cc0ee20bSDiana Craciun } 180*cc0ee20bSDiana Craciun 181*cc0ee20bSDiana Craciun vfio_fsl_mc_irqs_cleanup(vdev); 182*cc0ee20bSDiana Craciun 183*cc0ee20bSDiana Craciun fsl_mc_cleanup_irq_pool(mc_cont); 184*cc0ee20bSDiana Craciun } 185*cc0ee20bSDiana Craciun 186f2ba7e8cSDiana Craciun mutex_unlock(&vdev->reflck->lock); 187df747bcdSDiana Craciun 188fb1ff4c1SBharat Bhushan module_put(THIS_MODULE); 189fb1ff4c1SBharat Bhushan } 190fb1ff4c1SBharat Bhushan 191fb1ff4c1SBharat Bhushan static long vfio_fsl_mc_ioctl(void *device_data, unsigned int cmd, 192fb1ff4c1SBharat Bhushan unsigned long arg) 193fb1ff4c1SBharat Bhushan { 194f97f4c04SDiana Craciun unsigned long minsz; 195f97f4c04SDiana Craciun struct vfio_fsl_mc_device *vdev = device_data; 196f97f4c04SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 197f97f4c04SDiana Craciun 198fb1ff4c1SBharat Bhushan switch (cmd) { 199fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_INFO: 200fb1ff4c1SBharat Bhushan { 201f97f4c04SDiana Craciun struct vfio_device_info info; 202f97f4c04SDiana Craciun 203f97f4c04SDiana Craciun minsz = offsetofend(struct vfio_device_info, num_irqs); 204f97f4c04SDiana Craciun 205f97f4c04SDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 206f97f4c04SDiana Craciun return -EFAULT; 207f97f4c04SDiana Craciun 208f97f4c04SDiana Craciun if (info.argsz < minsz) 209f97f4c04SDiana Craciun return -EINVAL; 210f97f4c04SDiana Craciun 211f97f4c04SDiana Craciun info.flags = VFIO_DEVICE_FLAGS_FSL_MC; 212f97f4c04SDiana Craciun info.num_regions = mc_dev->obj_desc.region_count; 213f97f4c04SDiana Craciun info.num_irqs = mc_dev->obj_desc.irq_count; 214f97f4c04SDiana Craciun 215f97f4c04SDiana Craciun return copy_to_user((void __user *)arg, &info, minsz) ? 216f97f4c04SDiana Craciun -EFAULT : 0; 217fb1ff4c1SBharat Bhushan } 218fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_REGION_INFO: 219fb1ff4c1SBharat Bhushan { 220df747bcdSDiana Craciun struct vfio_region_info info; 221df747bcdSDiana Craciun 222df747bcdSDiana Craciun minsz = offsetofend(struct vfio_region_info, offset); 223df747bcdSDiana Craciun 224df747bcdSDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 225df747bcdSDiana Craciun return -EFAULT; 226df747bcdSDiana Craciun 227df747bcdSDiana Craciun if (info.argsz < minsz) 228df747bcdSDiana Craciun return -EINVAL; 229df747bcdSDiana Craciun 230df747bcdSDiana Craciun if (info.index >= mc_dev->obj_desc.region_count) 231df747bcdSDiana Craciun return -EINVAL; 232df747bcdSDiana Craciun 233df747bcdSDiana Craciun /* map offset to the physical address */ 234df747bcdSDiana Craciun info.offset = VFIO_FSL_MC_INDEX_TO_OFFSET(info.index); 235df747bcdSDiana Craciun info.size = vdev->regions[info.index].size; 236df747bcdSDiana Craciun info.flags = vdev->regions[info.index].flags; 237df747bcdSDiana Craciun 238df747bcdSDiana Craciun return copy_to_user((void __user *)arg, &info, minsz); 239fb1ff4c1SBharat Bhushan } 240fb1ff4c1SBharat Bhushan case VFIO_DEVICE_GET_IRQ_INFO: 241fb1ff4c1SBharat Bhushan { 2422e0d2956SDiana Craciun struct vfio_irq_info info; 2432e0d2956SDiana Craciun 2442e0d2956SDiana Craciun minsz = offsetofend(struct vfio_irq_info, count); 2452e0d2956SDiana Craciun if (copy_from_user(&info, (void __user *)arg, minsz)) 2462e0d2956SDiana Craciun return -EFAULT; 2472e0d2956SDiana Craciun 2482e0d2956SDiana Craciun if (info.argsz < minsz) 2492e0d2956SDiana Craciun return -EINVAL; 2502e0d2956SDiana Craciun 2512e0d2956SDiana Craciun if (info.index >= mc_dev->obj_desc.irq_count) 2522e0d2956SDiana Craciun return -EINVAL; 2532e0d2956SDiana Craciun 2542e0d2956SDiana Craciun info.flags = VFIO_IRQ_INFO_EVENTFD; 2552e0d2956SDiana Craciun info.count = 1; 2562e0d2956SDiana Craciun 2572e0d2956SDiana Craciun return copy_to_user((void __user *)arg, &info, minsz); 258fb1ff4c1SBharat Bhushan } 259fb1ff4c1SBharat Bhushan case VFIO_DEVICE_SET_IRQS: 260fb1ff4c1SBharat Bhushan { 2612e0d2956SDiana Craciun struct vfio_irq_set hdr; 2622e0d2956SDiana Craciun u8 *data = NULL; 2632e0d2956SDiana Craciun int ret = 0; 2642e0d2956SDiana Craciun size_t data_size = 0; 2652e0d2956SDiana Craciun 2662e0d2956SDiana Craciun minsz = offsetofend(struct vfio_irq_set, count); 2672e0d2956SDiana Craciun 2682e0d2956SDiana Craciun if (copy_from_user(&hdr, (void __user *)arg, minsz)) 2692e0d2956SDiana Craciun return -EFAULT; 2702e0d2956SDiana Craciun 2712e0d2956SDiana Craciun ret = vfio_set_irqs_validate_and_prepare(&hdr, mc_dev->obj_desc.irq_count, 2722e0d2956SDiana Craciun mc_dev->obj_desc.irq_count, &data_size); 2732e0d2956SDiana Craciun if (ret) 2742e0d2956SDiana Craciun return ret; 2752e0d2956SDiana Craciun 2762e0d2956SDiana Craciun if (data_size) { 2772e0d2956SDiana Craciun data = memdup_user((void __user *)(arg + minsz), 2782e0d2956SDiana Craciun data_size); 2792e0d2956SDiana Craciun if (IS_ERR(data)) 2802e0d2956SDiana Craciun return PTR_ERR(data); 2812e0d2956SDiana Craciun } 2822e0d2956SDiana Craciun 2832e0d2956SDiana Craciun mutex_lock(&vdev->igate); 2842e0d2956SDiana Craciun ret = vfio_fsl_mc_set_irqs_ioctl(vdev, hdr.flags, 2852e0d2956SDiana Craciun hdr.index, hdr.start, 2862e0d2956SDiana Craciun hdr.count, data); 2872e0d2956SDiana Craciun mutex_unlock(&vdev->igate); 2882e0d2956SDiana Craciun kfree(data); 2892e0d2956SDiana Craciun 2902e0d2956SDiana Craciun return ret; 291fb1ff4c1SBharat Bhushan } 292fb1ff4c1SBharat Bhushan case VFIO_DEVICE_RESET: 293fb1ff4c1SBharat Bhushan { 294fb1ff4c1SBharat Bhushan return -ENOTTY; 295fb1ff4c1SBharat Bhushan } 296fb1ff4c1SBharat Bhushan default: 297fb1ff4c1SBharat Bhushan return -ENOTTY; 298fb1ff4c1SBharat Bhushan } 299fb1ff4c1SBharat Bhushan } 300fb1ff4c1SBharat Bhushan 301fb1ff4c1SBharat Bhushan static ssize_t vfio_fsl_mc_read(void *device_data, char __user *buf, 302fb1ff4c1SBharat Bhushan size_t count, loff_t *ppos) 303fb1ff4c1SBharat Bhushan { 304fb1ff4c1SBharat Bhushan return -EINVAL; 305fb1ff4c1SBharat Bhushan } 306fb1ff4c1SBharat Bhushan 307fb1ff4c1SBharat Bhushan static ssize_t vfio_fsl_mc_write(void *device_data, const char __user *buf, 308fb1ff4c1SBharat Bhushan size_t count, loff_t *ppos) 309fb1ff4c1SBharat Bhushan { 310fb1ff4c1SBharat Bhushan return -EINVAL; 311fb1ff4c1SBharat Bhushan } 312fb1ff4c1SBharat Bhushan 31367247289SDiana Craciun static int vfio_fsl_mc_mmap_mmio(struct vfio_fsl_mc_region region, 31467247289SDiana Craciun struct vm_area_struct *vma) 31567247289SDiana Craciun { 31667247289SDiana Craciun u64 size = vma->vm_end - vma->vm_start; 31767247289SDiana Craciun u64 pgoff, base; 31867247289SDiana Craciun u8 region_cacheable; 31967247289SDiana Craciun 32067247289SDiana Craciun pgoff = vma->vm_pgoff & 32167247289SDiana Craciun ((1U << (VFIO_FSL_MC_OFFSET_SHIFT - PAGE_SHIFT)) - 1); 32267247289SDiana Craciun base = pgoff << PAGE_SHIFT; 32367247289SDiana Craciun 32467247289SDiana Craciun if (region.size < PAGE_SIZE || base + size > region.size) 32567247289SDiana Craciun return -EINVAL; 32667247289SDiana Craciun 32767247289SDiana Craciun region_cacheable = (region.type & FSL_MC_REGION_CACHEABLE) && 32867247289SDiana Craciun (region.type & FSL_MC_REGION_SHAREABLE); 32967247289SDiana Craciun if (!region_cacheable) 33067247289SDiana Craciun vma->vm_page_prot = pgprot_noncached(vma->vm_page_prot); 33167247289SDiana Craciun 33267247289SDiana Craciun vma->vm_pgoff = (region.addr >> PAGE_SHIFT) + pgoff; 33367247289SDiana Craciun 33467247289SDiana Craciun return remap_pfn_range(vma, vma->vm_start, vma->vm_pgoff, 33567247289SDiana Craciun size, vma->vm_page_prot); 33667247289SDiana Craciun } 33767247289SDiana Craciun 338fb1ff4c1SBharat Bhushan static int vfio_fsl_mc_mmap(void *device_data, struct vm_area_struct *vma) 339fb1ff4c1SBharat Bhushan { 34067247289SDiana Craciun struct vfio_fsl_mc_device *vdev = device_data; 34167247289SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 34267247289SDiana Craciun int index; 34367247289SDiana Craciun 34467247289SDiana Craciun index = vma->vm_pgoff >> (VFIO_FSL_MC_OFFSET_SHIFT - PAGE_SHIFT); 34567247289SDiana Craciun 34667247289SDiana Craciun if (vma->vm_end < vma->vm_start) 347fb1ff4c1SBharat Bhushan return -EINVAL; 34867247289SDiana Craciun if (vma->vm_start & ~PAGE_MASK) 34967247289SDiana Craciun return -EINVAL; 35067247289SDiana Craciun if (vma->vm_end & ~PAGE_MASK) 35167247289SDiana Craciun return -EINVAL; 35267247289SDiana Craciun if (!(vma->vm_flags & VM_SHARED)) 35367247289SDiana Craciun return -EINVAL; 35467247289SDiana Craciun if (index >= mc_dev->obj_desc.region_count) 35567247289SDiana Craciun return -EINVAL; 35667247289SDiana Craciun 35767247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_MMAP)) 35867247289SDiana Craciun return -EINVAL; 35967247289SDiana Craciun 36067247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_READ) 36167247289SDiana Craciun && (vma->vm_flags & VM_READ)) 36267247289SDiana Craciun return -EINVAL; 36367247289SDiana Craciun 36467247289SDiana Craciun if (!(vdev->regions[index].flags & VFIO_REGION_INFO_FLAG_WRITE) 36567247289SDiana Craciun && (vma->vm_flags & VM_WRITE)) 36667247289SDiana Craciun return -EINVAL; 36767247289SDiana Craciun 36867247289SDiana Craciun vma->vm_private_data = mc_dev; 36967247289SDiana Craciun 37067247289SDiana Craciun return vfio_fsl_mc_mmap_mmio(vdev->regions[index], vma); 371fb1ff4c1SBharat Bhushan } 372fb1ff4c1SBharat Bhushan 373fb1ff4c1SBharat Bhushan static const struct vfio_device_ops vfio_fsl_mc_ops = { 374fb1ff4c1SBharat Bhushan .name = "vfio-fsl-mc", 375fb1ff4c1SBharat Bhushan .open = vfio_fsl_mc_open, 376fb1ff4c1SBharat Bhushan .release = vfio_fsl_mc_release, 377fb1ff4c1SBharat Bhushan .ioctl = vfio_fsl_mc_ioctl, 378fb1ff4c1SBharat Bhushan .read = vfio_fsl_mc_read, 379fb1ff4c1SBharat Bhushan .write = vfio_fsl_mc_write, 380fb1ff4c1SBharat Bhushan .mmap = vfio_fsl_mc_mmap, 381fb1ff4c1SBharat Bhushan }; 382fb1ff4c1SBharat Bhushan 383704f5082SDiana Craciun static int vfio_fsl_mc_bus_notifier(struct notifier_block *nb, 384704f5082SDiana Craciun unsigned long action, void *data) 385704f5082SDiana Craciun { 386704f5082SDiana Craciun struct vfio_fsl_mc_device *vdev = container_of(nb, 387704f5082SDiana Craciun struct vfio_fsl_mc_device, nb); 388704f5082SDiana Craciun struct device *dev = data; 389704f5082SDiana Craciun struct fsl_mc_device *mc_dev = to_fsl_mc_device(dev); 390704f5082SDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(mc_dev->dev.parent); 391704f5082SDiana Craciun 392704f5082SDiana Craciun if (action == BUS_NOTIFY_ADD_DEVICE && 393704f5082SDiana Craciun vdev->mc_dev == mc_cont) { 394704f5082SDiana Craciun mc_dev->driver_override = kasprintf(GFP_KERNEL, "%s", 395704f5082SDiana Craciun vfio_fsl_mc_ops.name); 396704f5082SDiana Craciun if (!mc_dev->driver_override) 397704f5082SDiana Craciun dev_warn(dev, "VFIO_FSL_MC: Setting driver override for device in dprc %s failed\n", 398704f5082SDiana Craciun dev_name(&mc_cont->dev)); 399704f5082SDiana Craciun else 400704f5082SDiana Craciun dev_info(dev, "VFIO_FSL_MC: Setting driver override for device in dprc %s\n", 401704f5082SDiana Craciun dev_name(&mc_cont->dev)); 402704f5082SDiana Craciun } else if (action == BUS_NOTIFY_BOUND_DRIVER && 403704f5082SDiana Craciun vdev->mc_dev == mc_cont) { 404704f5082SDiana Craciun struct fsl_mc_driver *mc_drv = to_fsl_mc_driver(dev->driver); 405704f5082SDiana Craciun 406704f5082SDiana Craciun if (mc_drv && mc_drv != &vfio_fsl_mc_driver) 407704f5082SDiana Craciun dev_warn(dev, "VFIO_FSL_MC: Object %s bound to driver %s while DPRC bound to vfio-fsl-mc\n", 408704f5082SDiana Craciun dev_name(dev), mc_drv->driver.name); 409704f5082SDiana Craciun } 410704f5082SDiana Craciun 411704f5082SDiana Craciun return 0; 412704f5082SDiana Craciun } 413704f5082SDiana Craciun 414704f5082SDiana Craciun static int vfio_fsl_mc_init_device(struct vfio_fsl_mc_device *vdev) 415704f5082SDiana Craciun { 416704f5082SDiana Craciun struct fsl_mc_device *mc_dev = vdev->mc_dev; 417704f5082SDiana Craciun int ret; 418704f5082SDiana Craciun 419704f5082SDiana Craciun /* Non-dprc devices share mc_io from parent */ 420704f5082SDiana Craciun if (!is_fsl_mc_bus_dprc(mc_dev)) { 421704f5082SDiana Craciun struct fsl_mc_device *mc_cont = to_fsl_mc_device(mc_dev->dev.parent); 422704f5082SDiana Craciun 423704f5082SDiana Craciun mc_dev->mc_io = mc_cont->mc_io; 424704f5082SDiana Craciun return 0; 425704f5082SDiana Craciun } 426704f5082SDiana Craciun 427704f5082SDiana Craciun vdev->nb.notifier_call = vfio_fsl_mc_bus_notifier; 428704f5082SDiana Craciun ret = bus_register_notifier(&fsl_mc_bus_type, &vdev->nb); 429704f5082SDiana Craciun if (ret) 430704f5082SDiana Craciun return ret; 431704f5082SDiana Craciun 432704f5082SDiana Craciun /* open DPRC, allocate a MC portal */ 433704f5082SDiana Craciun ret = dprc_setup(mc_dev); 434704f5082SDiana Craciun if (ret) { 435704f5082SDiana Craciun dev_err(&mc_dev->dev, "VFIO_FSL_MC: Failed to setup DPRC (%d)\n", ret); 436704f5082SDiana Craciun goto out_nc_unreg; 437704f5082SDiana Craciun } 438704f5082SDiana Craciun 439704f5082SDiana Craciun ret = dprc_scan_container(mc_dev, false); 440704f5082SDiana Craciun if (ret) { 441704f5082SDiana Craciun dev_err(&mc_dev->dev, "VFIO_FSL_MC: Container scanning failed (%d)\n", ret); 442704f5082SDiana Craciun goto out_dprc_cleanup; 443704f5082SDiana Craciun } 444704f5082SDiana Craciun 445704f5082SDiana Craciun return 0; 446704f5082SDiana Craciun 447704f5082SDiana Craciun out_dprc_cleanup: 448704f5082SDiana Craciun dprc_remove_devices(mc_dev, NULL, 0); 449704f5082SDiana Craciun dprc_cleanup(mc_dev); 450704f5082SDiana Craciun out_nc_unreg: 451704f5082SDiana Craciun bus_unregister_notifier(&fsl_mc_bus_type, &vdev->nb); 452704f5082SDiana Craciun vdev->nb.notifier_call = NULL; 453704f5082SDiana Craciun 454704f5082SDiana Craciun return ret; 455704f5082SDiana Craciun } 456704f5082SDiana Craciun 457fb1ff4c1SBharat Bhushan static int vfio_fsl_mc_probe(struct fsl_mc_device *mc_dev) 458fb1ff4c1SBharat Bhushan { 459fb1ff4c1SBharat Bhushan struct iommu_group *group; 460fb1ff4c1SBharat Bhushan struct vfio_fsl_mc_device *vdev; 461fb1ff4c1SBharat Bhushan struct device *dev = &mc_dev->dev; 462fb1ff4c1SBharat Bhushan int ret; 463fb1ff4c1SBharat Bhushan 464fb1ff4c1SBharat Bhushan group = vfio_iommu_group_get(dev); 465fb1ff4c1SBharat Bhushan if (!group) { 466fb1ff4c1SBharat Bhushan dev_err(dev, "VFIO_FSL_MC: No IOMMU group\n"); 467fb1ff4c1SBharat Bhushan return -EINVAL; 468fb1ff4c1SBharat Bhushan } 469fb1ff4c1SBharat Bhushan 470fb1ff4c1SBharat Bhushan vdev = devm_kzalloc(dev, sizeof(*vdev), GFP_KERNEL); 471fb1ff4c1SBharat Bhushan if (!vdev) { 472fb1ff4c1SBharat Bhushan ret = -ENOMEM; 473fb1ff4c1SBharat Bhushan goto out_group_put; 474fb1ff4c1SBharat Bhushan } 475fb1ff4c1SBharat Bhushan 476fb1ff4c1SBharat Bhushan vdev->mc_dev = mc_dev; 477fb1ff4c1SBharat Bhushan 478fb1ff4c1SBharat Bhushan ret = vfio_add_group_dev(dev, &vfio_fsl_mc_ops, vdev); 479fb1ff4c1SBharat Bhushan if (ret) { 480fb1ff4c1SBharat Bhushan dev_err(dev, "VFIO_FSL_MC: Failed to add to vfio group\n"); 481fb1ff4c1SBharat Bhushan goto out_group_put; 482fb1ff4c1SBharat Bhushan } 483704f5082SDiana Craciun 484f2ba7e8cSDiana Craciun ret = vfio_fsl_mc_reflck_attach(vdev); 485704f5082SDiana Craciun if (ret) 486704f5082SDiana Craciun goto out_group_dev; 487704f5082SDiana Craciun 488f2ba7e8cSDiana Craciun ret = vfio_fsl_mc_init_device(vdev); 489f2ba7e8cSDiana Craciun if (ret) 490f2ba7e8cSDiana Craciun goto out_reflck; 491df747bcdSDiana Craciun 4922e0d2956SDiana Craciun mutex_init(&vdev->igate); 4932e0d2956SDiana Craciun 494fb1ff4c1SBharat Bhushan return 0; 495fb1ff4c1SBharat Bhushan 496f2ba7e8cSDiana Craciun out_reflck: 497f2ba7e8cSDiana Craciun vfio_fsl_mc_reflck_put(vdev->reflck); 498704f5082SDiana Craciun out_group_dev: 499704f5082SDiana Craciun vfio_del_group_dev(dev); 500fb1ff4c1SBharat Bhushan out_group_put: 501fb1ff4c1SBharat Bhushan vfio_iommu_group_put(group, dev); 502fb1ff4c1SBharat Bhushan return ret; 503fb1ff4c1SBharat Bhushan } 504fb1ff4c1SBharat Bhushan 505fb1ff4c1SBharat Bhushan static int vfio_fsl_mc_remove(struct fsl_mc_device *mc_dev) 506fb1ff4c1SBharat Bhushan { 507fb1ff4c1SBharat Bhushan struct vfio_fsl_mc_device *vdev; 508fb1ff4c1SBharat Bhushan struct device *dev = &mc_dev->dev; 509fb1ff4c1SBharat Bhushan 510fb1ff4c1SBharat Bhushan vdev = vfio_del_group_dev(dev); 511fb1ff4c1SBharat Bhushan if (!vdev) 512fb1ff4c1SBharat Bhushan return -EINVAL; 513fb1ff4c1SBharat Bhushan 5142e0d2956SDiana Craciun mutex_destroy(&vdev->igate); 5152e0d2956SDiana Craciun 516f2ba7e8cSDiana Craciun vfio_fsl_mc_reflck_put(vdev->reflck); 517df747bcdSDiana Craciun 518704f5082SDiana Craciun if (is_fsl_mc_bus_dprc(mc_dev)) { 519704f5082SDiana Craciun dprc_remove_devices(mc_dev, NULL, 0); 520704f5082SDiana Craciun dprc_cleanup(mc_dev); 521704f5082SDiana Craciun } 522704f5082SDiana Craciun 523704f5082SDiana Craciun if (vdev->nb.notifier_call) 524704f5082SDiana Craciun bus_unregister_notifier(&fsl_mc_bus_type, &vdev->nb); 525704f5082SDiana Craciun 526fb1ff4c1SBharat Bhushan vfio_iommu_group_put(mc_dev->dev.iommu_group, dev); 527fb1ff4c1SBharat Bhushan 528fb1ff4c1SBharat Bhushan return 0; 529fb1ff4c1SBharat Bhushan } 530fb1ff4c1SBharat Bhushan 531fb1ff4c1SBharat Bhushan static struct fsl_mc_driver vfio_fsl_mc_driver = { 532fb1ff4c1SBharat Bhushan .probe = vfio_fsl_mc_probe, 533fb1ff4c1SBharat Bhushan .remove = vfio_fsl_mc_remove, 534fb1ff4c1SBharat Bhushan .driver = { 535fb1ff4c1SBharat Bhushan .name = "vfio-fsl-mc", 536fb1ff4c1SBharat Bhushan .owner = THIS_MODULE, 537fb1ff4c1SBharat Bhushan }, 538fb1ff4c1SBharat Bhushan }; 539fb1ff4c1SBharat Bhushan 540fb1ff4c1SBharat Bhushan static int __init vfio_fsl_mc_driver_init(void) 541fb1ff4c1SBharat Bhushan { 542fb1ff4c1SBharat Bhushan return fsl_mc_driver_register(&vfio_fsl_mc_driver); 543fb1ff4c1SBharat Bhushan } 544fb1ff4c1SBharat Bhushan 545fb1ff4c1SBharat Bhushan static void __exit vfio_fsl_mc_driver_exit(void) 546fb1ff4c1SBharat Bhushan { 547fb1ff4c1SBharat Bhushan fsl_mc_driver_unregister(&vfio_fsl_mc_driver); 548fb1ff4c1SBharat Bhushan } 549fb1ff4c1SBharat Bhushan 550fb1ff4c1SBharat Bhushan module_init(vfio_fsl_mc_driver_init); 551fb1ff4c1SBharat Bhushan module_exit(vfio_fsl_mc_driver_exit); 552fb1ff4c1SBharat Bhushan 553fb1ff4c1SBharat Bhushan MODULE_LICENSE("Dual BSD/GPL"); 554fb1ff4c1SBharat Bhushan MODULE_DESCRIPTION("VFIO for FSL-MC devices - User Level meta-driver"); 555