xref: /linux/drivers/accel/rocket/rocket_gem.c (revision 58809f614e0e3f4e12b489bddf680bfeb31c0a20)
1658ebeacSTomeu Vizoso // SPDX-License-Identifier: GPL-2.0-only
2658ebeacSTomeu Vizoso /* Copyright 2024-2025 Tomeu Vizoso <tomeu@tomeuvizoso.net> */
3658ebeacSTomeu Vizoso 
4658ebeacSTomeu Vizoso #include <drm/drm_device.h>
5658ebeacSTomeu Vizoso #include <drm/drm_utils.h>
6658ebeacSTomeu Vizoso #include <drm/rocket_accel.h>
7658ebeacSTomeu Vizoso #include <linux/dma-mapping.h>
8658ebeacSTomeu Vizoso #include <linux/iommu.h>
9658ebeacSTomeu Vizoso 
10658ebeacSTomeu Vizoso #include "rocket_drv.h"
11658ebeacSTomeu Vizoso #include "rocket_gem.h"
12658ebeacSTomeu Vizoso 
13658ebeacSTomeu Vizoso static void rocket_gem_bo_free(struct drm_gem_object *obj)
14658ebeacSTomeu Vizoso {
15658ebeacSTomeu Vizoso 	struct rocket_gem_object *bo = to_rocket_bo(obj);
16658ebeacSTomeu Vizoso 	struct rocket_file_priv *rocket_priv = bo->driver_priv;
17658ebeacSTomeu Vizoso 	size_t unmapped;
18658ebeacSTomeu Vizoso 
19658ebeacSTomeu Vizoso 	drm_WARN_ON(obj->dev, refcount_read(&bo->base.pages_use_count) > 1);
20658ebeacSTomeu Vizoso 
21658ebeacSTomeu Vizoso 	unmapped = iommu_unmap(bo->domain->domain, bo->mm.start, bo->size);
22658ebeacSTomeu Vizoso 	drm_WARN_ON(obj->dev, unmapped != bo->size);
23658ebeacSTomeu Vizoso 
24658ebeacSTomeu Vizoso 	mutex_lock(&rocket_priv->mm_lock);
25658ebeacSTomeu Vizoso 	drm_mm_remove_node(&bo->mm);
26658ebeacSTomeu Vizoso 	mutex_unlock(&rocket_priv->mm_lock);
27658ebeacSTomeu Vizoso 
28658ebeacSTomeu Vizoso 	rocket_iommu_domain_put(bo->domain);
29658ebeacSTomeu Vizoso 	bo->domain = NULL;
30658ebeacSTomeu Vizoso 
31658ebeacSTomeu Vizoso 	drm_gem_shmem_free(&bo->base);
32658ebeacSTomeu Vizoso }
33658ebeacSTomeu Vizoso 
34658ebeacSTomeu Vizoso static const struct drm_gem_object_funcs rocket_gem_funcs = {
35658ebeacSTomeu Vizoso 	.free = rocket_gem_bo_free,
36658ebeacSTomeu Vizoso 	.print_info = drm_gem_shmem_object_print_info,
37658ebeacSTomeu Vizoso 	.pin = drm_gem_shmem_object_pin,
38658ebeacSTomeu Vizoso 	.unpin = drm_gem_shmem_object_unpin,
39658ebeacSTomeu Vizoso 	.get_sg_table = drm_gem_shmem_object_get_sg_table,
40658ebeacSTomeu Vizoso 	.vmap = drm_gem_shmem_object_vmap,
41658ebeacSTomeu Vizoso 	.vunmap = drm_gem_shmem_object_vunmap,
42658ebeacSTomeu Vizoso 	.mmap = drm_gem_shmem_object_mmap,
43658ebeacSTomeu Vizoso 	.vm_ops = &drm_gem_shmem_vm_ops,
44658ebeacSTomeu Vizoso };
45658ebeacSTomeu Vizoso 
46658ebeacSTomeu Vizoso struct drm_gem_object *rocket_gem_create_object(struct drm_device *dev, size_t size)
47658ebeacSTomeu Vizoso {
48658ebeacSTomeu Vizoso 	struct rocket_gem_object *obj;
49658ebeacSTomeu Vizoso 
50658ebeacSTomeu Vizoso 	obj = kzalloc(sizeof(*obj), GFP_KERNEL);
51658ebeacSTomeu Vizoso 	if (!obj)
52658ebeacSTomeu Vizoso 		return ERR_PTR(-ENOMEM);
53658ebeacSTomeu Vizoso 
54658ebeacSTomeu Vizoso 	obj->base.base.funcs = &rocket_gem_funcs;
55658ebeacSTomeu Vizoso 
56658ebeacSTomeu Vizoso 	return &obj->base.base;
57658ebeacSTomeu Vizoso }
58658ebeacSTomeu Vizoso 
59658ebeacSTomeu Vizoso int rocket_ioctl_create_bo(struct drm_device *dev, void *data, struct drm_file *file)
60658ebeacSTomeu Vizoso {
61658ebeacSTomeu Vizoso 	struct rocket_file_priv *rocket_priv = file->driver_priv;
62658ebeacSTomeu Vizoso 	struct drm_rocket_create_bo *args = data;
63658ebeacSTomeu Vizoso 	struct drm_gem_shmem_object *shmem_obj;
64658ebeacSTomeu Vizoso 	struct rocket_gem_object *rkt_obj;
65658ebeacSTomeu Vizoso 	struct drm_gem_object *gem_obj;
66658ebeacSTomeu Vizoso 	struct sg_table *sgt;
67658ebeacSTomeu Vizoso 	int ret;
68658ebeacSTomeu Vizoso 
69658ebeacSTomeu Vizoso 	shmem_obj = drm_gem_shmem_create(dev, args->size);
70658ebeacSTomeu Vizoso 	if (IS_ERR(shmem_obj))
71658ebeacSTomeu Vizoso 		return PTR_ERR(shmem_obj);
72658ebeacSTomeu Vizoso 
73658ebeacSTomeu Vizoso 	gem_obj = &shmem_obj->base;
74658ebeacSTomeu Vizoso 	rkt_obj = to_rocket_bo(gem_obj);
75658ebeacSTomeu Vizoso 
76658ebeacSTomeu Vizoso 	rkt_obj->driver_priv = rocket_priv;
77658ebeacSTomeu Vizoso 	rkt_obj->domain = rocket_iommu_domain_get(rocket_priv);
78658ebeacSTomeu Vizoso 	rkt_obj->size = args->size;
79658ebeacSTomeu Vizoso 	rkt_obj->offset = 0;
80658ebeacSTomeu Vizoso 
81658ebeacSTomeu Vizoso 	ret = drm_gem_handle_create(file, gem_obj, &args->handle);
82658ebeacSTomeu Vizoso 	drm_gem_object_put(gem_obj);
83658ebeacSTomeu Vizoso 	if (ret)
84658ebeacSTomeu Vizoso 		goto err;
85658ebeacSTomeu Vizoso 
86658ebeacSTomeu Vizoso 	sgt = drm_gem_shmem_get_pages_sgt(shmem_obj);
87658ebeacSTomeu Vizoso 	if (IS_ERR(sgt)) {
88658ebeacSTomeu Vizoso 		ret = PTR_ERR(sgt);
89658ebeacSTomeu Vizoso 		goto err;
90658ebeacSTomeu Vizoso 	}
91658ebeacSTomeu Vizoso 
92658ebeacSTomeu Vizoso 	mutex_lock(&rocket_priv->mm_lock);
93658ebeacSTomeu Vizoso 	ret = drm_mm_insert_node_generic(&rocket_priv->mm, &rkt_obj->mm,
94658ebeacSTomeu Vizoso 					 rkt_obj->size, PAGE_SIZE,
95658ebeacSTomeu Vizoso 					 0, 0);
96658ebeacSTomeu Vizoso 	mutex_unlock(&rocket_priv->mm_lock);
97658ebeacSTomeu Vizoso 
98658ebeacSTomeu Vizoso 	ret = iommu_map_sgtable(rocket_priv->domain->domain,
99658ebeacSTomeu Vizoso 				rkt_obj->mm.start,
100658ebeacSTomeu Vizoso 				shmem_obj->sgt,
101658ebeacSTomeu Vizoso 				IOMMU_READ | IOMMU_WRITE);
102658ebeacSTomeu Vizoso 	if (ret < 0 || ret < args->size) {
103658ebeacSTomeu Vizoso 		drm_err(dev, "failed to map buffer: size=%d request_size=%u\n",
104658ebeacSTomeu Vizoso 			ret, args->size);
105658ebeacSTomeu Vizoso 		ret = -ENOMEM;
106658ebeacSTomeu Vizoso 		goto err_remove_node;
107658ebeacSTomeu Vizoso 	}
108658ebeacSTomeu Vizoso 
109658ebeacSTomeu Vizoso 	/* iommu_map_sgtable might have aligned the size */
110658ebeacSTomeu Vizoso 	rkt_obj->size = ret;
111658ebeacSTomeu Vizoso 	args->offset = drm_vma_node_offset_addr(&gem_obj->vma_node);
112658ebeacSTomeu Vizoso 	args->dma_address = rkt_obj->mm.start;
113658ebeacSTomeu Vizoso 
114658ebeacSTomeu Vizoso 	return 0;
115658ebeacSTomeu Vizoso 
116658ebeacSTomeu Vizoso err_remove_node:
117658ebeacSTomeu Vizoso 	mutex_lock(&rocket_priv->mm_lock);
118658ebeacSTomeu Vizoso 	drm_mm_remove_node(&rkt_obj->mm);
119658ebeacSTomeu Vizoso 	mutex_unlock(&rocket_priv->mm_lock);
120658ebeacSTomeu Vizoso 
121658ebeacSTomeu Vizoso err:
122658ebeacSTomeu Vizoso 	drm_gem_shmem_object_free(gem_obj);
123658ebeacSTomeu Vizoso 
124658ebeacSTomeu Vizoso 	return ret;
125658ebeacSTomeu Vizoso }
126*525ad89dSTomeu Vizoso 
127*525ad89dSTomeu Vizoso int rocket_ioctl_prep_bo(struct drm_device *dev, void *data, struct drm_file *file)
128*525ad89dSTomeu Vizoso {
129*525ad89dSTomeu Vizoso 	struct drm_rocket_prep_bo *args = data;
130*525ad89dSTomeu Vizoso 	unsigned long timeout = drm_timeout_abs_to_jiffies(args->timeout_ns);
131*525ad89dSTomeu Vizoso 	struct drm_gem_object *gem_obj;
132*525ad89dSTomeu Vizoso 	struct drm_gem_shmem_object *shmem_obj;
133*525ad89dSTomeu Vizoso 	long ret = 0;
134*525ad89dSTomeu Vizoso 
135*525ad89dSTomeu Vizoso 	if (args->reserved != 0) {
136*525ad89dSTomeu Vizoso 		drm_dbg(dev, "Reserved field in drm_rocket_prep_bo struct should be 0.\n");
137*525ad89dSTomeu Vizoso 		return -EINVAL;
138*525ad89dSTomeu Vizoso 	}
139*525ad89dSTomeu Vizoso 
140*525ad89dSTomeu Vizoso 	gem_obj = drm_gem_object_lookup(file, args->handle);
141*525ad89dSTomeu Vizoso 	if (!gem_obj)
142*525ad89dSTomeu Vizoso 		return -ENOENT;
143*525ad89dSTomeu Vizoso 
144*525ad89dSTomeu Vizoso 	ret = dma_resv_wait_timeout(gem_obj->resv, DMA_RESV_USAGE_WRITE, true, timeout);
145*525ad89dSTomeu Vizoso 	if (!ret)
146*525ad89dSTomeu Vizoso 		ret = timeout ? -ETIMEDOUT : -EBUSY;
147*525ad89dSTomeu Vizoso 
148*525ad89dSTomeu Vizoso 	shmem_obj = &to_rocket_bo(gem_obj)->base;
149*525ad89dSTomeu Vizoso 
150*525ad89dSTomeu Vizoso 	dma_sync_sgtable_for_cpu(dev->dev, shmem_obj->sgt, DMA_BIDIRECTIONAL);
151*525ad89dSTomeu Vizoso 
152*525ad89dSTomeu Vizoso 	drm_gem_object_put(gem_obj);
153*525ad89dSTomeu Vizoso 
154*525ad89dSTomeu Vizoso 	return ret;
155*525ad89dSTomeu Vizoso }
156*525ad89dSTomeu Vizoso 
157*525ad89dSTomeu Vizoso int rocket_ioctl_fini_bo(struct drm_device *dev, void *data, struct drm_file *file)
158*525ad89dSTomeu Vizoso {
159*525ad89dSTomeu Vizoso 	struct drm_rocket_fini_bo *args = data;
160*525ad89dSTomeu Vizoso 	struct drm_gem_shmem_object *shmem_obj;
161*525ad89dSTomeu Vizoso 	struct rocket_gem_object *rkt_obj;
162*525ad89dSTomeu Vizoso 	struct drm_gem_object *gem_obj;
163*525ad89dSTomeu Vizoso 
164*525ad89dSTomeu Vizoso 	if (args->reserved != 0) {
165*525ad89dSTomeu Vizoso 		drm_dbg(dev, "Reserved field in drm_rocket_fini_bo struct should be 0.\n");
166*525ad89dSTomeu Vizoso 		return -EINVAL;
167*525ad89dSTomeu Vizoso 	}
168*525ad89dSTomeu Vizoso 
169*525ad89dSTomeu Vizoso 	gem_obj = drm_gem_object_lookup(file, args->handle);
170*525ad89dSTomeu Vizoso 	if (!gem_obj)
171*525ad89dSTomeu Vizoso 		return -ENOENT;
172*525ad89dSTomeu Vizoso 
173*525ad89dSTomeu Vizoso 	rkt_obj = to_rocket_bo(gem_obj);
174*525ad89dSTomeu Vizoso 	shmem_obj = &rkt_obj->base;
175*525ad89dSTomeu Vizoso 
176*525ad89dSTomeu Vizoso 	dma_sync_sgtable_for_device(dev->dev, shmem_obj->sgt, DMA_BIDIRECTIONAL);
177*525ad89dSTomeu Vizoso 
178*525ad89dSTomeu Vizoso 	drm_gem_object_put(gem_obj);
179*525ad89dSTomeu Vizoso 
180*525ad89dSTomeu Vizoso 	return 0;
181*525ad89dSTomeu Vizoso }
182