xref: /openbmc/linux/drivers/vfio/fsl-mc/vfio_fsl_mc.c (revision 1e04ec14204dec28131855d8dd160c3d55d12797)
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