1 /*
2 * Copyright (C) 2012 Samsung Electronics Co., Ltd.
3 *
4 * Permission is hereby granted, free of charge, to any person obtaining a
5 * copy of this software and associated documentation files (the "Software"),
6 * to deal in the Software without restriction, including without limitation
7 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8 * and/or sell copies of the Software, and to permit persons to whom the
9 * Software is furnished to do so, subject to the following conditions:
10 *
11 * The above copyright notice and this permission notice (including the next
12 * paragraph) shall be included in all copies or substantial portions of the
13 * Software.
14 *
15 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21 * SOFTWARE.
22 *
23 * Authors:
24 * Inki Dae <inki.dae@samsung.com>
25 */
26
27 #ifdef HAVE_CONFIG_H
28 #include "config.h"
29 #endif
30
31 #include <stdlib.h>
32 #include <stdio.h>
33 #include <string.h>
34 #include <errno.h>
35
36 #include <sys/mman.h>
37 #include <linux/stddef.h>
38
39 #include <xf86drm.h>
40
41 #include "libdrm.h"
42 #include "exynos_drm.h"
43 #include "exynos_drmif.h"
44
45 /*
46 * Create exynos drm device object.
47 *
48 * @fd: file descriptor to exynos drm driver opened.
49 *
50 * if true, return the device object else NULL.
51 */
exynos_device_create(int fd)52 drm_public struct exynos_device * exynos_device_create(int fd)
53 {
54 struct exynos_device *dev;
55
56 dev = calloc(sizeof(*dev), 1);
57 if (!dev) {
58 fprintf(stderr, "failed to create device[%s].\n",
59 strerror(errno));
60 return NULL;
61 }
62
63 dev->fd = fd;
64
65 return dev;
66 }
67
68 /*
69 * Destroy exynos drm device object
70 *
71 * @dev: exynos drm device object.
72 */
exynos_device_destroy(struct exynos_device * dev)73 drm_public void exynos_device_destroy(struct exynos_device *dev)
74 {
75 free(dev);
76 }
77
78 /*
79 * Create a exynos buffer object to exynos drm device.
80 *
81 * @dev: exynos drm device object.
82 * @size: user-desired size.
83 * flags: user-desired memory type.
84 * user can set one or more types among several types to memory
85 * allocation and cache attribute types. and as default,
86 * EXYNOS_BO_NONCONTIG and EXYNOS-BO_NONCACHABLE types would
87 * be used.
88 *
89 * if true, return a exynos buffer object else NULL.
90 */
exynos_bo_create(struct exynos_device * dev,size_t size,uint32_t flags)91 drm_public struct exynos_bo * exynos_bo_create(struct exynos_device *dev,
92 size_t size, uint32_t flags)
93 {
94 struct exynos_bo *bo;
95 struct drm_exynos_gem_create req = {
96 .size = size,
97 .flags = flags,
98 };
99
100 if (size == 0) {
101 fprintf(stderr, "invalid size.\n");
102 goto fail;
103 }
104
105 bo = calloc(sizeof(*bo), 1);
106 if (!bo) {
107 fprintf(stderr, "failed to create bo[%s].\n",
108 strerror(errno));
109 goto err_free_bo;
110 }
111
112 bo->dev = dev;
113
114 if (drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_GEM_CREATE, &req)){
115 fprintf(stderr, "failed to create gem object[%s].\n",
116 strerror(errno));
117 goto err_free_bo;
118 }
119
120 bo->handle = req.handle;
121 bo->size = size;
122 bo->flags = flags;
123
124 return bo;
125
126 err_free_bo:
127 free(bo);
128 fail:
129 return NULL;
130 }
131
132 /*
133 * Get information to gem region allocated.
134 *
135 * @dev: exynos drm device object.
136 * @handle: gem handle to request gem info.
137 * @size: size to gem object and returned by kernel side.
138 * @flags: gem flags to gem object and returned by kernel side.
139 *
140 * with this function call, you can get flags and size to gem handle
141 * through bo object.
142 *
143 * if true, return 0 else negative.
144 */
exynos_bo_get_info(struct exynos_device * dev,uint32_t handle,size_t * size,uint32_t * flags)145 drm_public int exynos_bo_get_info(struct exynos_device *dev, uint32_t handle,
146 size_t *size, uint32_t *flags)
147 {
148 int ret;
149 struct drm_exynos_gem_info req = {
150 .handle = handle,
151 };
152
153 ret = drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_GEM_GET, &req);
154 if (ret < 0) {
155 fprintf(stderr, "failed to get gem object information[%s].\n",
156 strerror(errno));
157 return ret;
158 }
159
160 *size = req.size;
161 *flags = req.flags;
162
163 return 0;
164 }
165
166 /*
167 * Destroy a exynos buffer object.
168 *
169 * @bo: a exynos buffer object to be destroyed.
170 */
exynos_bo_destroy(struct exynos_bo * bo)171 drm_public void exynos_bo_destroy(struct exynos_bo *bo)
172 {
173 if (!bo)
174 return;
175
176 if (bo->vaddr)
177 munmap(bo->vaddr, bo->size);
178
179 if (bo->handle) {
180 struct drm_gem_close req = {
181 .handle = bo->handle,
182 };
183
184 drmIoctl(bo->dev->fd, DRM_IOCTL_GEM_CLOSE, &req);
185 }
186
187 free(bo);
188 }
189
190
191 /*
192 * Get a exynos buffer object from a gem global object name.
193 *
194 * @dev: a exynos device object.
195 * @name: a gem global object name exported by another process.
196 *
197 * this interface is used to get a exynos buffer object from a gem
198 * global object name sent by another process for buffer sharing.
199 *
200 * if true, return a exynos buffer object else NULL.
201 *
202 */
203 drm_public struct exynos_bo *
exynos_bo_from_name(struct exynos_device * dev,uint32_t name)204 exynos_bo_from_name(struct exynos_device *dev, uint32_t name)
205 {
206 struct exynos_bo *bo;
207 struct drm_gem_open req = {
208 .name = name,
209 };
210
211 bo = calloc(sizeof(*bo), 1);
212 if (!bo) {
213 fprintf(stderr, "failed to allocate bo[%s].\n",
214 strerror(errno));
215 return NULL;
216 }
217
218 if (drmIoctl(dev->fd, DRM_IOCTL_GEM_OPEN, &req)) {
219 fprintf(stderr, "failed to open gem object[%s].\n",
220 strerror(errno));
221 goto err_free_bo;
222 }
223
224 bo->dev = dev;
225 bo->name = name;
226 bo->handle = req.handle;
227
228 return bo;
229
230 err_free_bo:
231 free(bo);
232 return NULL;
233 }
234
235 /*
236 * Get a gem global object name from a gem object handle.
237 *
238 * @bo: a exynos buffer object including gem handle.
239 * @name: a gem global object name to be got by kernel driver.
240 *
241 * this interface is used to get a gem global object name from a gem object
242 * handle to a buffer that wants to share it with another process.
243 *
244 * if true, return 0 else negative.
245 */
exynos_bo_get_name(struct exynos_bo * bo,uint32_t * name)246 drm_public int exynos_bo_get_name(struct exynos_bo *bo, uint32_t *name)
247 {
248 if (!bo->name) {
249 struct drm_gem_flink req = {
250 .handle = bo->handle,
251 };
252 int ret;
253
254 ret = drmIoctl(bo->dev->fd, DRM_IOCTL_GEM_FLINK, &req);
255 if (ret) {
256 fprintf(stderr, "failed to get gem global name[%s].\n",
257 strerror(errno));
258 return ret;
259 }
260
261 bo->name = req.name;
262 }
263
264 *name = bo->name;
265
266 return 0;
267 }
268
exynos_bo_handle(struct exynos_bo * bo)269 drm_public uint32_t exynos_bo_handle(struct exynos_bo *bo)
270 {
271 return bo->handle;
272 }
273
274 /*
275 * Mmap a buffer to user space.
276 *
277 * @bo: a exynos buffer object including a gem object handle to be mmapped
278 * to user space.
279 *
280 * if true, user pointer mmaped else NULL.
281 */
exynos_bo_map(struct exynos_bo * bo)282 drm_public void *exynos_bo_map(struct exynos_bo *bo)
283 {
284 if (!bo->vaddr) {
285 struct exynos_device *dev = bo->dev;
286 struct drm_exynos_gem_mmap req = {
287 .handle = bo->handle,
288 .size = bo->size,
289 };
290 int ret;
291
292 ret = drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_GEM_MMAP, &req);
293 if (ret) {
294 fprintf(stderr, "failed to mmap[%s].\n",
295 strerror(errno));
296 return NULL;
297 }
298
299 bo->vaddr = (void *)(uintptr_t)req.mapped;
300 }
301
302 return bo->vaddr;
303 }
304
305 /*
306 * Export gem object to dmabuf as file descriptor.
307 *
308 * @dev: exynos device object
309 * @handle: gem handle to export as file descriptor of dmabuf
310 * @fd: file descriptor returned from kernel
311 *
312 * @return: 0 on success, -1 on error, and errno will be set
313 */
314 drm_public int
exynos_prime_handle_to_fd(struct exynos_device * dev,uint32_t handle,int * fd)315 exynos_prime_handle_to_fd(struct exynos_device *dev, uint32_t handle, int *fd)
316 {
317 return drmPrimeHandleToFD(dev->fd, handle, 0, fd);
318 }
319
320 /*
321 * Import file descriptor into gem handle.
322 *
323 * @dev: exynos device object
324 * @fd: file descriptor of dmabuf to import
325 * @handle: gem handle returned from kernel
326 *
327 * @return: 0 on success, -1 on error, and errno will be set
328 */
329 drm_public int
exynos_prime_fd_to_handle(struct exynos_device * dev,int fd,uint32_t * handle)330 exynos_prime_fd_to_handle(struct exynos_device *dev, int fd, uint32_t *handle)
331 {
332 return drmPrimeFDToHandle(dev->fd, fd, handle);
333 }
334
335
336
337 /*
338 * Request Wireless Display connection or disconnection.
339 *
340 * @dev: a exynos device object.
341 * @connect: indicate whether connectoin or disconnection request.
342 * @ext: indicate whether edid data includes extentions data or not.
343 * @edid: a pointer to edid data from Wireless Display device.
344 *
345 * this interface is used to request Virtual Display driver connection or
346 * disconnection. for this, user should get a edid data from the Wireless
347 * Display device and then send that data to kernel driver with connection
348 * request
349 *
350 * if true, return 0 else negative.
351 */
352 drm_public int
exynos_vidi_connection(struct exynos_device * dev,uint32_t connect,uint32_t ext,void * edid)353 exynos_vidi_connection(struct exynos_device *dev, uint32_t connect,
354 uint32_t ext, void *edid)
355 {
356 struct drm_exynos_vidi_connection req = {
357 .connection = connect,
358 .extensions = ext,
359 .edid = (uint64_t)(uintptr_t)edid,
360 };
361 int ret;
362
363 ret = drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_VIDI_CONNECTION, &req);
364 if (ret) {
365 fprintf(stderr, "failed to request vidi connection[%s].\n",
366 strerror(errno));
367 return ret;
368 }
369
370 return 0;
371 }
372