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