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 #include <unistd.h>
36
37 #include <sys/mman.h>
38 #include <linux/stddef.h>
39
40 #include <xf86drm.h>
41
42 #include "libdrm_macros.h"
43 #include "exynos_drm.h"
44 #include "exynos_drmif.h"
45
46 #define U642VOID(x) ((void *)(unsigned long)(x))
47
48 /*
49 * Create exynos drm device object.
50 *
51 * @fd: file descriptor to exynos drm driver opened.
52 *
53 * if true, return the device object else NULL.
54 */
exynos_device_create(int fd)55 struct exynos_device * exynos_device_create(int fd)
56 {
57 struct exynos_device *dev;
58
59 dev = calloc(sizeof(*dev), 1);
60 if (!dev) {
61 fprintf(stderr, "failed to create device[%s].\n",
62 strerror(errno));
63 return NULL;
64 }
65
66 dev->fd = fd;
67
68 return dev;
69 }
70
71 /*
72 * Destroy exynos drm device object
73 *
74 * @dev: exynos drm device object.
75 */
exynos_device_destroy(struct exynos_device * dev)76 void exynos_device_destroy(struct exynos_device *dev)
77 {
78 free(dev);
79 }
80
81 /*
82 * Create a exynos buffer object to exynos drm device.
83 *
84 * @dev: exynos drm device object.
85 * @size: user-desired size.
86 * flags: user-desired memory type.
87 * user can set one or more types among several types to memory
88 * allocation and cache attribute types. and as default,
89 * EXYNOS_BO_NONCONTIG and EXYNOS-BO_NONCACHABLE types would
90 * be used.
91 *
92 * if true, return a exynos buffer object else NULL.
93 */
exynos_bo_create(struct exynos_device * dev,size_t size,uint32_t flags)94 struct exynos_bo * exynos_bo_create(struct exynos_device *dev,
95 size_t size, uint32_t flags)
96 {
97 struct exynos_bo *bo;
98 struct drm_exynos_gem_create req = {
99 .size = size,
100 .flags = flags,
101 };
102
103 if (size == 0) {
104 fprintf(stderr, "invalid size.\n");
105 goto fail;
106 }
107
108 bo = calloc(sizeof(*bo), 1);
109 if (!bo) {
110 fprintf(stderr, "failed to create bo[%s].\n",
111 strerror(errno));
112 goto err_free_bo;
113 }
114
115 bo->dev = dev;
116
117 if (drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_GEM_CREATE, &req)){
118 fprintf(stderr, "failed to create gem object[%s].\n",
119 strerror(errno));
120 goto err_free_bo;
121 }
122
123 bo->handle = req.handle;
124 bo->size = size;
125 bo->flags = flags;
126
127 return bo;
128
129 err_free_bo:
130 free(bo);
131 fail:
132 return NULL;
133 }
134
135 /*
136 * Get information to gem region allocated.
137 *
138 * @dev: exynos drm device object.
139 * @handle: gem handle to request gem info.
140 * @size: size to gem object and returned by kernel side.
141 * @flags: gem flags to gem object and returned by kernel side.
142 *
143 * with this function call, you can get flags and size to gem handle
144 * through bo object.
145 *
146 * if true, return 0 else negative.
147 */
exynos_bo_get_info(struct exynos_device * dev,uint32_t handle,size_t * size,uint32_t * flags)148 int exynos_bo_get_info(struct exynos_device *dev, uint32_t handle,
149 size_t *size, uint32_t *flags)
150 {
151 int ret;
152 struct drm_exynos_gem_info req = {
153 .handle = handle,
154 };
155
156 ret = drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_GEM_GET, &req);
157 if (ret < 0) {
158 fprintf(stderr, "failed to get gem object information[%s].\n",
159 strerror(errno));
160 return ret;
161 }
162
163 *size = req.size;
164 *flags = req.flags;
165
166 return 0;
167 }
168
169 /*
170 * Destroy a exynos buffer object.
171 *
172 * @bo: a exynos buffer object to be destroyed.
173 */
exynos_bo_destroy(struct exynos_bo * bo)174 void exynos_bo_destroy(struct exynos_bo *bo)
175 {
176 if (!bo)
177 return;
178
179 if (bo->vaddr)
180 munmap(bo->vaddr, bo->size);
181
182 if (bo->handle) {
183 struct drm_gem_close req = {
184 .handle = bo->handle,
185 };
186
187 drmIoctl(bo->dev->fd, DRM_IOCTL_GEM_CLOSE, &req);
188 }
189
190 free(bo);
191 }
192
193
194 /*
195 * Get a exynos buffer object from a gem global object name.
196 *
197 * @dev: a exynos device object.
198 * @name: a gem global object name exported by another process.
199 *
200 * this interface is used to get a exynos buffer object from a gem
201 * global object name sent by another process for buffer sharing.
202 *
203 * if true, return a exynos buffer object else NULL.
204 *
205 */
206 struct exynos_bo *
exynos_bo_from_name(struct exynos_device * dev,uint32_t name)207 exynos_bo_from_name(struct exynos_device *dev, uint32_t name)
208 {
209 struct exynos_bo *bo;
210 struct drm_gem_open req = {
211 .name = name,
212 };
213
214 bo = calloc(sizeof(*bo), 1);
215 if (!bo) {
216 fprintf(stderr, "failed to allocate bo[%s].\n",
217 strerror(errno));
218 return NULL;
219 }
220
221 if (drmIoctl(dev->fd, DRM_IOCTL_GEM_OPEN, &req)) {
222 fprintf(stderr, "failed to open gem object[%s].\n",
223 strerror(errno));
224 goto err_free_bo;
225 }
226
227 bo->dev = dev;
228 bo->name = name;
229 bo->handle = req.handle;
230
231 return bo;
232
233 err_free_bo:
234 free(bo);
235 return NULL;
236 }
237
238 /*
239 * Get a gem global object name from a gem object handle.
240 *
241 * @bo: a exynos buffer object including gem handle.
242 * @name: a gem global object name to be got by kernel driver.
243 *
244 * this interface is used to get a gem global object name from a gem object
245 * handle to a buffer that wants to share it with another process.
246 *
247 * if true, return 0 else negative.
248 */
exynos_bo_get_name(struct exynos_bo * bo,uint32_t * name)249 int exynos_bo_get_name(struct exynos_bo *bo, uint32_t *name)
250 {
251 if (!bo->name) {
252 struct drm_gem_flink req = {
253 .handle = bo->handle,
254 };
255 int ret;
256
257 ret = drmIoctl(bo->dev->fd, DRM_IOCTL_GEM_FLINK, &req);
258 if (ret) {
259 fprintf(stderr, "failed to get gem global name[%s].\n",
260 strerror(errno));
261 return ret;
262 }
263
264 bo->name = req.name;
265 }
266
267 *name = bo->name;
268
269 return 0;
270 }
271
exynos_bo_handle(struct exynos_bo * bo)272 uint32_t exynos_bo_handle(struct exynos_bo *bo)
273 {
274 return bo->handle;
275 }
276
277 /*
278 * Mmap a buffer to user space.
279 *
280 * @bo: a exynos buffer object including a gem object handle to be mmapped
281 * to user space.
282 *
283 * if true, user pointer mmaped else NULL.
284 */
exynos_bo_map(struct exynos_bo * bo)285 void *exynos_bo_map(struct exynos_bo *bo)
286 {
287 if (!bo->vaddr) {
288 struct exynos_device *dev = bo->dev;
289 struct drm_mode_map_dumb arg;
290 void *map = NULL;
291 int ret;
292
293 memset(&arg, 0, sizeof(arg));
294 arg.handle = bo->handle;
295
296 ret = drmIoctl(dev->fd, DRM_IOCTL_MODE_MAP_DUMB, &arg);
297 if (ret) {
298 fprintf(stderr, "failed to map dumb buffer[%s].\n",
299 strerror(errno));
300 return NULL;
301 }
302
303 map = drm_mmap(0, bo->size, PROT_READ | PROT_WRITE, MAP_SHARED,
304 dev->fd, arg.offset);
305
306 if (map != MAP_FAILED)
307 bo->vaddr = map;
308 }
309
310 return bo->vaddr;
311 }
312
313 /*
314 * Export gem object to dmabuf as file descriptor.
315 *
316 * @dev: exynos device object
317 * @handle: gem handle to export as file descriptor of dmabuf
318 * @fd: file descriptor returned from kernel
319 *
320 * @return: 0 on success, -1 on error, and errno will be set
321 */
322 int
exynos_prime_handle_to_fd(struct exynos_device * dev,uint32_t handle,int * fd)323 exynos_prime_handle_to_fd(struct exynos_device *dev, uint32_t handle, int *fd)
324 {
325 return drmPrimeHandleToFD(dev->fd, handle, 0, fd);
326 }
327
328 /*
329 * Import file descriptor into gem handle.
330 *
331 * @dev: exynos device object
332 * @fd: file descriptor of dmabuf to import
333 * @handle: gem handle returned from kernel
334 *
335 * @return: 0 on success, -1 on error, and errno will be set
336 */
337 int
exynos_prime_fd_to_handle(struct exynos_device * dev,int fd,uint32_t * handle)338 exynos_prime_fd_to_handle(struct exynos_device *dev, int fd, uint32_t *handle)
339 {
340 return drmPrimeFDToHandle(dev->fd, fd, handle);
341 }
342
343
344
345 /*
346 * Request Wireless Display connection or disconnection.
347 *
348 * @dev: a exynos device object.
349 * @connect: indicate whether connectoin or disconnection request.
350 * @ext: indicate whether edid data includes extentions data or not.
351 * @edid: a pointer to edid data from Wireless Display device.
352 *
353 * this interface is used to request Virtual Display driver connection or
354 * disconnection. for this, user should get a edid data from the Wireless
355 * Display device and then send that data to kernel driver with connection
356 * request
357 *
358 * if true, return 0 else negative.
359 */
360 int
exynos_vidi_connection(struct exynos_device * dev,uint32_t connect,uint32_t ext,void * edid)361 exynos_vidi_connection(struct exynos_device *dev, uint32_t connect,
362 uint32_t ext, void *edid)
363 {
364 struct drm_exynos_vidi_connection req = {
365 .connection = connect,
366 .extensions = ext,
367 .edid = (uint64_t)(uintptr_t)edid,
368 };
369 int ret;
370
371 ret = drmIoctl(dev->fd, DRM_IOCTL_EXYNOS_VIDI_CONNECTION, &req);
372 if (ret) {
373 fprintf(stderr, "failed to request vidi connection[%s].\n",
374 strerror(errno));
375 return ret;
376 }
377
378 return 0;
379 }
380
381 static void
exynos_handle_vendor(int fd,struct drm_event * e,void * ctx)382 exynos_handle_vendor(int fd, struct drm_event *e, void *ctx)
383 {
384 struct drm_exynos_g2d_event *g2d;
385 struct exynos_event_context *ectx = ctx;
386
387 switch (e->type) {
388 case DRM_EXYNOS_G2D_EVENT:
389 if (ectx->version < 1 || ectx->g2d_event_handler == NULL)
390 break;
391 g2d = (struct drm_exynos_g2d_event *)e;
392 ectx->g2d_event_handler(fd, g2d->cmdlist_no, g2d->tv_sec,
393 g2d->tv_usec, U642VOID(g2d->user_data));
394 break;
395
396 default:
397 break;
398 }
399 }
400
401 int
exynos_handle_event(struct exynos_device * dev,struct exynos_event_context * ctx)402 exynos_handle_event(struct exynos_device *dev, struct exynos_event_context *ctx)
403 {
404 char buffer[1024];
405 int len, i;
406 struct drm_event *e;
407 struct drm_event_vblank *vblank;
408 drmEventContextPtr evctx = &ctx->base;
409
410 /* The DRM read semantics guarantees that we always get only
411 * complete events. */
412 len = read(dev->fd, buffer, sizeof buffer);
413 if (len == 0)
414 return 0;
415 if (len < (int)sizeof *e)
416 return -1;
417
418 i = 0;
419 while (i < len) {
420 e = (struct drm_event *) &buffer[i];
421 switch (e->type) {
422 case DRM_EVENT_VBLANK:
423 if (evctx->version < 1 ||
424 evctx->vblank_handler == NULL)
425 break;
426 vblank = (struct drm_event_vblank *) e;
427 evctx->vblank_handler(dev->fd,
428 vblank->sequence,
429 vblank->tv_sec,
430 vblank->tv_usec,
431 U642VOID (vblank->user_data));
432 break;
433 case DRM_EVENT_FLIP_COMPLETE:
434 if (evctx->version < 2 ||
435 evctx->page_flip_handler == NULL)
436 break;
437 vblank = (struct drm_event_vblank *) e;
438 evctx->page_flip_handler(dev->fd,
439 vblank->sequence,
440 vblank->tv_sec,
441 vblank->tv_usec,
442 U642VOID (vblank->user_data));
443 break;
444 default:
445 exynos_handle_vendor(dev->fd, e, evctx);
446 break;
447 }
448 i += e->length;
449 }
450
451 return 0;
452 }
453