egl: android: add dma-buf fd support
[mesa.git] / src / egl / drivers / dri2 / platform_android.c
1 /*
2 * Mesa 3-D graphics library
3 *
4 * Copyright (C) 2010-2011 Chia-I Wu <olvaffe@gmail.com>
5 * Copyright (C) 2010-2011 LunarG Inc.
6 *
7 * Based on platform_x11, which has
8 *
9 * Copyright © 2011 Intel Corporation
10 *
11 * Permission is hereby granted, free of charge, to any person obtaining a
12 * copy of this software and associated documentation files (the "Software"),
13 * to deal in the Software without restriction, including without limitation
14 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
15 * and/or sell copies of the Software, and to permit persons to whom the
16 * Software is furnished to do so, subject to the following conditions:
17 *
18 * The above copyright notice and this permission notice shall be included
19 * in all copies or substantial portions of the Software.
20 *
21 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
22 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
23 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
24 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
25 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
26 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
27 * DEALINGS IN THE SOFTWARE.
28 */
29
30 #include <errno.h>
31 #include <dlfcn.h>
32 #include <xf86drm.h>
33
34 #if ANDROID_VERSION >= 0x402
35 #include <sync/sync.h>
36 #endif
37
38 #include "loader.h"
39 #include "egl_dri2.h"
40 #include "egl_dri2_fallbacks.h"
41 #include "gralloc_drm.h"
42
43 static int
44 get_format_bpp(int native)
45 {
46 int bpp;
47
48 switch (native) {
49 case HAL_PIXEL_FORMAT_RGBA_8888:
50 case HAL_PIXEL_FORMAT_RGBX_8888:
51 case HAL_PIXEL_FORMAT_BGRA_8888:
52 bpp = 4;
53 break;
54 case HAL_PIXEL_FORMAT_RGB_888:
55 bpp = 3;
56 break;
57 case HAL_PIXEL_FORMAT_RGB_565:
58 bpp = 2;
59 break;
60 default:
61 bpp = 0;
62 break;
63 }
64
65 return bpp;
66 }
67
68 /* createImageFromFds requires fourcc format */
69 static int get_fourcc(int format)
70 {
71 switch(format) {
72 case __DRI_IMAGE_FORMAT_RGB565: return __DRI_IMAGE_FOURCC_RGB565;
73 case __DRI_IMAGE_FORMAT_ARGB8888: return __DRI_IMAGE_FOURCC_ARGB8888;
74 case __DRI_IMAGE_FORMAT_XRGB8888: return __DRI_IMAGE_FOURCC_XRGB8888;
75 case __DRI_IMAGE_FORMAT_ABGR8888: return __DRI_IMAGE_FOURCC_ABGR8888;
76 case __DRI_IMAGE_FORMAT_XBGR8888: return __DRI_IMAGE_FOURCC_XBGR8888;
77 }
78 return -1;
79 }
80
81 static int get_format(int format)
82 {
83 switch (format) {
84 case HAL_PIXEL_FORMAT_BGRA_8888: return __DRI_IMAGE_FORMAT_ARGB8888;
85 case HAL_PIXEL_FORMAT_RGB_565: return __DRI_IMAGE_FORMAT_RGB565;
86 case HAL_PIXEL_FORMAT_RGBA_8888: return __DRI_IMAGE_FORMAT_ABGR8888;
87 case HAL_PIXEL_FORMAT_RGBX_8888: return __DRI_IMAGE_FORMAT_XBGR8888;
88 case HAL_PIXEL_FORMAT_RGB_888:
89 /* unsupported */
90 default:
91 _eglLog(_EGL_WARNING, "unsupported native buffer format 0x%x", format);
92 }
93 return -1;
94 }
95 static int
96 get_native_buffer_fd(struct ANativeWindowBuffer *buf)
97 {
98 native_handle_t *handle = (native_handle_t *)buf->handle;
99 /*
100 * Various gralloc implementations exist, but the dma-buf fd tends
101 * to be first. Access it directly to avoid a dependency on specific
102 * gralloc versions.
103 */
104 return (handle && handle->numFds) ? handle->data[0] : -1;
105 }
106
107 static int
108 get_native_buffer_name(struct ANativeWindowBuffer *buf)
109 {
110 return gralloc_drm_get_gem_handle(buf->handle);
111 }
112
113 static EGLBoolean
114 droid_window_dequeue_buffer(struct dri2_egl_surface *dri2_surf)
115 {
116 #if ANDROID_VERSION >= 0x0402
117 int fence_fd;
118
119 if (dri2_surf->window->dequeueBuffer(dri2_surf->window, &dri2_surf->buffer,
120 &fence_fd))
121 return EGL_FALSE;
122
123 /* If access to the buffer is controlled by a sync fence, then block on the
124 * fence.
125 *
126 * It may be more performant to postpone blocking until there is an
127 * immediate need to write to the buffer. But doing so would require adding
128 * hooks to the DRI2 loader.
129 *
130 * From the ANativeWindow::dequeueBuffer documentation:
131 *
132 * The libsync fence file descriptor returned in the int pointed to by
133 * the fenceFd argument will refer to the fence that must signal
134 * before the dequeued buffer may be written to. A value of -1
135 * indicates that the caller may access the buffer immediately without
136 * waiting on a fence. If a valid file descriptor is returned (i.e.
137 * any value except -1) then the caller is responsible for closing the
138 * file descriptor.
139 */
140 if (fence_fd >= 0) {
141 /* From the SYNC_IOC_WAIT documentation in <linux/sync.h>:
142 *
143 * Waits indefinitely if timeout < 0.
144 */
145 int timeout = -1;
146 sync_wait(fence_fd, timeout);
147 close(fence_fd);
148 }
149
150 dri2_surf->buffer->common.incRef(&dri2_surf->buffer->common);
151 #else
152 if (dri2_surf->window->dequeueBuffer(dri2_surf->window, &dri2_surf->buffer))
153 return EGL_FALSE;
154
155 dri2_surf->buffer->common.incRef(&dri2_surf->buffer->common);
156 dri2_surf->window->lockBuffer(dri2_surf->window, dri2_surf->buffer);
157 #endif
158
159 return EGL_TRUE;
160 }
161
162 static EGLBoolean
163 droid_window_enqueue_buffer(struct dri2_egl_surface *dri2_surf)
164 {
165 #if ANDROID_VERSION >= 0x0402
166 /* Queue the buffer without a sync fence. This informs the ANativeWindow
167 * that it may access the buffer immediately.
168 *
169 * From ANativeWindow::dequeueBuffer:
170 *
171 * The fenceFd argument specifies a libsync fence file descriptor for
172 * a fence that must signal before the buffer can be accessed. If
173 * the buffer can be accessed immediately then a value of -1 should
174 * be used. The caller must not use the file descriptor after it
175 * is passed to queueBuffer, and the ANativeWindow implementation
176 * is responsible for closing it.
177 */
178 int fence_fd = -1;
179 dri2_surf->window->queueBuffer(dri2_surf->window, dri2_surf->buffer,
180 fence_fd);
181 #else
182 dri2_surf->window->queueBuffer(dri2_surf->window, dri2_surf->buffer);
183 #endif
184
185 dri2_surf->buffer->common.decRef(&dri2_surf->buffer->common);
186 dri2_surf->buffer = NULL;
187
188 return EGL_TRUE;
189 }
190
191 static void
192 droid_window_cancel_buffer(struct dri2_egl_surface *dri2_surf)
193 {
194 /* no cancel buffer? */
195 droid_window_enqueue_buffer(dri2_surf);
196 }
197
198 static __DRIbuffer *
199 droid_alloc_local_buffer(struct dri2_egl_surface *dri2_surf,
200 unsigned int att, unsigned int format)
201 {
202 struct dri2_egl_display *dri2_dpy =
203 dri2_egl_display(dri2_surf->base.Resource.Display);
204
205 if (att >= ARRAY_SIZE(dri2_surf->local_buffers))
206 return NULL;
207
208 if (!dri2_surf->local_buffers[att]) {
209 dri2_surf->local_buffers[att] =
210 dri2_dpy->dri2->allocateBuffer(dri2_dpy->dri_screen, att, format,
211 dri2_surf->base.Width, dri2_surf->base.Height);
212 }
213
214 return dri2_surf->local_buffers[att];
215 }
216
217 static void
218 droid_free_local_buffers(struct dri2_egl_surface *dri2_surf)
219 {
220 struct dri2_egl_display *dri2_dpy =
221 dri2_egl_display(dri2_surf->base.Resource.Display);
222 int i;
223
224 for (i = 0; i < ARRAY_SIZE(dri2_surf->local_buffers); i++) {
225 if (dri2_surf->local_buffers[i]) {
226 dri2_dpy->dri2->releaseBuffer(dri2_dpy->dri_screen,
227 dri2_surf->local_buffers[i]);
228 dri2_surf->local_buffers[i] = NULL;
229 }
230 }
231 }
232
233 static _EGLSurface *
234 droid_create_surface(_EGLDriver *drv, _EGLDisplay *disp, EGLint type,
235 _EGLConfig *conf, void *native_window,
236 const EGLint *attrib_list)
237 {
238 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
239 struct dri2_egl_config *dri2_conf = dri2_egl_config(conf);
240 struct dri2_egl_surface *dri2_surf;
241 struct ANativeWindow *window = native_window;
242 const __DRIconfig *config;
243
244 dri2_surf = calloc(1, sizeof *dri2_surf);
245 if (!dri2_surf) {
246 _eglError(EGL_BAD_ALLOC, "droid_create_surface");
247 return NULL;
248 }
249
250 if (!_eglInitSurface(&dri2_surf->base, disp, type, conf, attrib_list))
251 goto cleanup_surface;
252
253 if (type == EGL_WINDOW_BIT) {
254 int format;
255
256 if (!window || window->common.magic != ANDROID_NATIVE_WINDOW_MAGIC) {
257 _eglError(EGL_BAD_NATIVE_WINDOW, "droid_create_surface");
258 goto cleanup_surface;
259 }
260 if (window->query(window, NATIVE_WINDOW_FORMAT, &format)) {
261 _eglError(EGL_BAD_NATIVE_WINDOW, "droid_create_surface");
262 goto cleanup_surface;
263 }
264
265 if (format != dri2_conf->base.NativeVisualID) {
266 _eglLog(_EGL_WARNING, "Native format mismatch: 0x%x != 0x%x",
267 format, dri2_conf->base.NativeVisualID);
268 }
269
270 window->query(window, NATIVE_WINDOW_WIDTH, &dri2_surf->base.Width);
271 window->query(window, NATIVE_WINDOW_HEIGHT, &dri2_surf->base.Height);
272 }
273
274 config = dri2_get_dri_config(dri2_conf, EGL_WINDOW_BIT,
275 dri2_surf->base.GLColorspace);
276
277 dri2_surf->dri_drawable =
278 (*dri2_dpy->dri2->createNewDrawable)(dri2_dpy->dri_screen, config,
279 dri2_surf);
280 if (dri2_surf->dri_drawable == NULL) {
281 _eglError(EGL_BAD_ALLOC, "dri2->createNewDrawable");
282 goto cleanup_surface;
283 }
284
285 if (window) {
286 window->common.incRef(&window->common);
287 dri2_surf->window = window;
288 }
289
290 return &dri2_surf->base;
291
292 cleanup_surface:
293 free(dri2_surf);
294
295 return NULL;
296 }
297
298 static _EGLSurface *
299 droid_create_window_surface(_EGLDriver *drv, _EGLDisplay *disp,
300 _EGLConfig *conf, void *native_window,
301 const EGLint *attrib_list)
302 {
303 return droid_create_surface(drv, disp, EGL_WINDOW_BIT, conf,
304 native_window, attrib_list);
305 }
306
307 static _EGLSurface *
308 droid_create_pbuffer_surface(_EGLDriver *drv, _EGLDisplay *disp,
309 _EGLConfig *conf, const EGLint *attrib_list)
310 {
311 return droid_create_surface(drv, disp, EGL_PBUFFER_BIT, conf,
312 NULL, attrib_list);
313 }
314
315 static EGLBoolean
316 droid_destroy_surface(_EGLDriver *drv, _EGLDisplay *disp, _EGLSurface *surf)
317 {
318 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
319 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surf);
320
321 if (!_eglPutSurface(surf))
322 return EGL_TRUE;
323
324 droid_free_local_buffers(dri2_surf);
325
326 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
327 if (dri2_surf->buffer)
328 droid_window_cancel_buffer(dri2_surf);
329
330 dri2_surf->window->common.decRef(&dri2_surf->window->common);
331 }
332
333 (*dri2_dpy->core->destroyDrawable)(dri2_surf->dri_drawable);
334
335 free(dri2_surf);
336
337 return EGL_TRUE;
338 }
339
340 static int
341 update_buffers(struct dri2_egl_surface *dri2_surf)
342 {
343 if (dri2_surf->base.Type != EGL_WINDOW_BIT)
344 return 0;
345
346 /* try to dequeue the next back buffer */
347 if (!dri2_surf->buffer && !droid_window_dequeue_buffer(dri2_surf))
348 return -1;
349
350 /* free outdated buffers and update the surface size */
351 if (dri2_surf->base.Width != dri2_surf->buffer->width ||
352 dri2_surf->base.Height != dri2_surf->buffer->height) {
353 droid_free_local_buffers(dri2_surf);
354 dri2_surf->base.Width = dri2_surf->buffer->width;
355 dri2_surf->base.Height = dri2_surf->buffer->height;
356 }
357
358 return 0;
359 }
360
361 static int
362 get_back_bo(struct dri2_egl_surface *dri2_surf)
363 {
364 struct dri2_egl_display *dri2_dpy =
365 dri2_egl_display(dri2_surf->base.Resource.Display);
366 int format, stride;
367 int offset = 0, fd;
368
369 if (!dri2_surf->buffer)
370 return -1;
371
372 fd = get_native_buffer_fd(dri2_surf->buffer);
373 if (fd < 0)
374 return -1;
375
376 format = get_format(dri2_surf->buffer->format);
377
378 stride = dri2_surf->buffer->stride *
379 get_format_bpp(dri2_surf->buffer->format);
380
381 dri2_surf->dri_image =
382 dri2_dpy->image->createImageFromFds(dri2_dpy->dri_screen,
383 dri2_surf->base.Width,
384 dri2_surf->base.Height,
385 get_fourcc(format),
386 &fd,
387 1,
388 &stride,
389 &offset,
390 dri2_surf);
391 if (!dri2_surf->dri_image)
392 return -1;
393
394 return 0;
395 }
396
397 static int
398 droid_image_get_buffers(__DRIdrawable *driDrawable,
399 unsigned int format,
400 uint32_t *stamp,
401 void *loaderPrivate,
402 uint32_t buffer_mask,
403 struct __DRIimageList *images)
404 {
405 struct dri2_egl_surface *dri2_surf = loaderPrivate;
406
407 if (update_buffers(dri2_surf) < 0)
408 return 0;
409
410 if (get_back_bo(dri2_surf) < 0) {
411 _eglError(EGL_BAD_PARAMETER, "get_back_bo");
412 return 0;
413 }
414
415 images->image_mask = __DRI_IMAGE_BUFFER_BACK;
416 images->back = dri2_surf->dri_image;
417
418 return 1;
419 }
420
421 static EGLBoolean
422 droid_swap_buffers(_EGLDriver *drv, _EGLDisplay *disp, _EGLSurface *draw)
423 {
424 struct dri2_egl_driver *dri2_drv = dri2_egl_driver(drv);
425 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
426 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(draw);
427 _EGLContext *ctx;
428
429 if (dri2_surf->base.Type != EGL_WINDOW_BIT)
430 return EGL_TRUE;
431
432 if (dri2_drv->glFlush) {
433 ctx = _eglGetCurrentContext();
434 if (ctx && ctx->DrawSurface == &dri2_surf->base)
435 dri2_drv->glFlush();
436 }
437
438 dri2_flush_drawable_for_swapbuffers(disp, draw);
439
440 if (dri2_surf->buffer)
441 droid_window_enqueue_buffer(dri2_surf);
442
443 (*dri2_dpy->flush->invalidate)(dri2_surf->dri_drawable);
444
445 return EGL_TRUE;
446 }
447
448 static _EGLImage *
449 dri2_create_image_android_native_buffer(_EGLDriver *drv, _EGLDisplay *disp,
450 _EGLContext *ctx,
451 struct ANativeWindowBuffer *buf)
452 {
453 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
454 struct dri2_egl_image *dri2_img;
455 int name, fd;
456
457 if (ctx != NULL) {
458 /* From the EGL_ANDROID_image_native_buffer spec:
459 *
460 * * If <target> is EGL_NATIVE_BUFFER_ANDROID and <ctx> is not
461 * EGL_NO_CONTEXT, the error EGL_BAD_CONTEXT is generated.
462 */
463 _eglError(EGL_BAD_CONTEXT, "eglCreateEGLImageKHR: for "
464 "EGL_NATIVE_BUFFER_ANDROID, the context must be "
465 "EGL_NO_CONTEXT");
466 return NULL;
467 }
468
469 if (!buf || buf->common.magic != ANDROID_NATIVE_BUFFER_MAGIC ||
470 buf->common.version != sizeof(*buf)) {
471 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
472 return NULL;
473 }
474
475 fd = get_native_buffer_fd(buf);
476 if (fd >= 0) {
477 const EGLint attr_list[14] = {
478 EGL_WIDTH, buf->width,
479 EGL_HEIGHT, buf->height,
480 EGL_LINUX_DRM_FOURCC_EXT, get_fourcc(get_format(buf->format)),
481 EGL_DMA_BUF_PLANE0_FD_EXT, fd,
482 EGL_DMA_BUF_PLANE0_PITCH_EXT, buf->stride * get_format_bpp(buf->format),
483 EGL_DMA_BUF_PLANE0_OFFSET_EXT, 0,
484 EGL_NONE, 0
485 };
486
487 return dri2_create_image_khr(drv, disp, ctx, EGL_LINUX_DMA_BUF_EXT,
488 NULL, attr_list);
489 }
490
491 name = get_native_buffer_name(buf);
492 if (!name) {
493 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
494 return NULL;
495 }
496
497 dri2_img = calloc(1, sizeof(*dri2_img));
498 if (!dri2_img) {
499 _eglError(EGL_BAD_ALLOC, "droid_create_image_mesa_drm");
500 return NULL;
501 }
502
503 if (!_eglInitImage(&dri2_img->base, disp)) {
504 free(dri2_img);
505 return NULL;
506 }
507
508 dri2_img->dri_image =
509 dri2_dpy->image->createImageFromName(dri2_dpy->dri_screen,
510 buf->width,
511 buf->height,
512 get_format(buf->format),
513 name,
514 buf->stride,
515 dri2_img);
516 if (!dri2_img->dri_image) {
517 free(dri2_img);
518 _eglError(EGL_BAD_ALLOC, "droid_create_image_mesa_drm");
519 return NULL;
520 }
521
522 return &dri2_img->base;
523 }
524
525 static _EGLImage *
526 droid_create_image_khr(_EGLDriver *drv, _EGLDisplay *disp,
527 _EGLContext *ctx, EGLenum target,
528 EGLClientBuffer buffer, const EGLint *attr_list)
529 {
530 switch (target) {
531 case EGL_NATIVE_BUFFER_ANDROID:
532 return dri2_create_image_android_native_buffer(drv, disp, ctx,
533 (struct ANativeWindowBuffer *) buffer);
534 default:
535 return dri2_create_image_khr(drv, disp, ctx, target, buffer, attr_list);
536 }
537 }
538
539 static void
540 droid_flush_front_buffer(__DRIdrawable * driDrawable, void *loaderPrivate)
541 {
542 }
543
544 static int
545 droid_get_buffers_parse_attachments(struct dri2_egl_surface *dri2_surf,
546 unsigned int *attachments, int count)
547 {
548 int num_buffers = 0, i;
549
550 /* fill dri2_surf->buffers */
551 for (i = 0; i < count * 2; i += 2) {
552 __DRIbuffer *buf, *local;
553
554 assert(num_buffers < ARRAY_SIZE(dri2_surf->buffers));
555 buf = &dri2_surf->buffers[num_buffers];
556
557 switch (attachments[i]) {
558 case __DRI_BUFFER_BACK_LEFT:
559 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
560 buf->attachment = attachments[i];
561 buf->name = get_native_buffer_name(dri2_surf->buffer);
562 buf->cpp = get_format_bpp(dri2_surf->buffer->format);
563 buf->pitch = dri2_surf->buffer->stride * buf->cpp;
564 buf->flags = 0;
565
566 if (buf->name)
567 num_buffers++;
568
569 break;
570 }
571 /* fall through for pbuffers */
572 case __DRI_BUFFER_DEPTH:
573 case __DRI_BUFFER_STENCIL:
574 case __DRI_BUFFER_ACCUM:
575 case __DRI_BUFFER_DEPTH_STENCIL:
576 case __DRI_BUFFER_HIZ:
577 local = droid_alloc_local_buffer(dri2_surf,
578 attachments[i], attachments[i + 1]);
579
580 if (local) {
581 *buf = *local;
582 num_buffers++;
583 }
584 break;
585 case __DRI_BUFFER_FRONT_LEFT:
586 case __DRI_BUFFER_FRONT_RIGHT:
587 case __DRI_BUFFER_FAKE_FRONT_LEFT:
588 case __DRI_BUFFER_FAKE_FRONT_RIGHT:
589 case __DRI_BUFFER_BACK_RIGHT:
590 default:
591 /* no front or right buffers */
592 break;
593 }
594 }
595
596 return num_buffers;
597 }
598
599 static __DRIbuffer *
600 droid_get_buffers_with_format(__DRIdrawable * driDrawable,
601 int *width, int *height,
602 unsigned int *attachments, int count,
603 int *out_count, void *loaderPrivate)
604 {
605 struct dri2_egl_surface *dri2_surf = loaderPrivate;
606 struct dri2_egl_display *dri2_dpy =
607 dri2_egl_display(dri2_surf->base.Resource.Display);
608 int i;
609
610 if (update_buffers(dri2_surf) < 0)
611 return NULL;
612
613 dri2_surf->buffer_count =
614 droid_get_buffers_parse_attachments(dri2_surf, attachments, count);
615
616 if (width)
617 *width = dri2_surf->base.Width;
618 if (height)
619 *height = dri2_surf->base.Height;
620
621 *out_count = dri2_surf->buffer_count;
622
623 return dri2_surf->buffers;
624 }
625
626 static EGLBoolean
627 droid_add_configs_for_visuals(_EGLDriver *drv, _EGLDisplay *dpy)
628 {
629 struct dri2_egl_display *dri2_dpy = dri2_egl_display(dpy);
630 const struct {
631 int format;
632 unsigned int rgba_masks[4];
633 } visuals[] = {
634 { HAL_PIXEL_FORMAT_RGBA_8888, { 0xff, 0xff00, 0xff0000, 0xff000000 } },
635 { HAL_PIXEL_FORMAT_RGBX_8888, { 0xff, 0xff00, 0xff0000, 0x0 } },
636 { HAL_PIXEL_FORMAT_RGB_888, { 0xff, 0xff00, 0xff0000, 0x0 } },
637 { HAL_PIXEL_FORMAT_RGB_565, { 0xf800, 0x7e0, 0x1f, 0x0 } },
638 { HAL_PIXEL_FORMAT_BGRA_8888, { 0xff0000, 0xff00, 0xff, 0xff000000 } },
639 { 0, { 0, 0, 0, 0 } }
640 };
641 EGLint config_attrs[] = {
642 EGL_NATIVE_VISUAL_ID, 0,
643 EGL_NATIVE_VISUAL_TYPE, 0,
644 EGL_FRAMEBUFFER_TARGET_ANDROID, EGL_TRUE,
645 EGL_RECORDABLE_ANDROID, EGL_TRUE,
646 EGL_NONE
647 };
648 int count, i, j;
649
650 count = 0;
651 for (i = 0; visuals[i].format; i++) {
652 int format_count = 0;
653
654 config_attrs[1] = visuals[i].format;
655 config_attrs[3] = visuals[i].format;
656
657 for (j = 0; dri2_dpy->driver_configs[j]; j++) {
658 const EGLint surface_type = EGL_WINDOW_BIT | EGL_PBUFFER_BIT;
659 struct dri2_egl_config *dri2_conf;
660 unsigned int double_buffered = 0;
661
662 dri2_dpy->core->getConfigAttrib(dri2_dpy->driver_configs[j],
663 __DRI_ATTRIB_DOUBLE_BUFFER, &double_buffered);
664
665 /* support only double buffered configs */
666 if (!double_buffered)
667 continue;
668
669 dri2_conf = dri2_add_config(dpy, dri2_dpy->driver_configs[j],
670 count + 1, surface_type, config_attrs, visuals[i].rgba_masks);
671 if (dri2_conf) {
672 count++;
673 format_count++;
674 }
675 }
676
677 if (!format_count) {
678 _eglLog(_EGL_DEBUG, "No DRI config supports native format 0x%x",
679 visuals[i].format);
680 }
681 }
682
683 /* post-process configs */
684 for (i = 0; i < dpy->Configs->Size; i++) {
685 struct dri2_egl_config *dri2_conf = dri2_egl_config(dpy->Configs->Elements[i]);
686
687 /* there is no front buffer so no OpenGL */
688 dri2_conf->base.RenderableType &= ~EGL_OPENGL_BIT;
689 dri2_conf->base.Conformant &= ~EGL_OPENGL_BIT;
690 }
691
692 return (count != 0);
693 }
694
695 static int
696 droid_open_device(void)
697 {
698 const hw_module_t *mod;
699 int fd = -1, err;
700
701 err = hw_get_module(GRALLOC_HARDWARE_MODULE_ID, &mod);
702 if (!err) {
703 const gralloc_module_t *gr = (gralloc_module_t *) mod;
704
705 err = -EINVAL;
706 if (gr->perform)
707 err = gr->perform(gr, GRALLOC_MODULE_PERFORM_GET_DRM_FD, &fd);
708 }
709 if (err || fd < 0) {
710 _eglLog(_EGL_WARNING, "fail to get drm fd");
711 fd = -1;
712 }
713
714 return (fd >= 0) ? dup(fd) : -1;
715 }
716
717 /* support versions < JellyBean */
718 #ifndef ALOGW
719 #define ALOGW LOGW
720 #endif
721 #ifndef ALOGD
722 #define ALOGD LOGD
723 #endif
724 #ifndef ALOGI
725 #define ALOGI LOGI
726 #endif
727
728 static void
729 droid_log(EGLint level, const char *msg)
730 {
731 switch (level) {
732 case _EGL_DEBUG:
733 ALOGD("%s", msg);
734 break;
735 case _EGL_INFO:
736 ALOGI("%s", msg);
737 break;
738 case _EGL_WARNING:
739 ALOGW("%s", msg);
740 break;
741 case _EGL_FATAL:
742 LOG_FATAL("%s", msg);
743 break;
744 default:
745 break;
746 }
747 }
748
749 static struct dri2_egl_display_vtbl droid_display_vtbl = {
750 .authenticate = NULL,
751 .create_window_surface = droid_create_window_surface,
752 .create_pixmap_surface = dri2_fallback_create_pixmap_surface,
753 .create_pbuffer_surface = droid_create_pbuffer_surface,
754 .destroy_surface = droid_destroy_surface,
755 .create_image = droid_create_image_khr,
756 .swap_interval = dri2_fallback_swap_interval,
757 .swap_buffers = droid_swap_buffers,
758 .swap_buffers_with_damage = dri2_fallback_swap_buffers_with_damage,
759 .swap_buffers_region = dri2_fallback_swap_buffers_region,
760 .post_sub_buffer = dri2_fallback_post_sub_buffer,
761 .copy_buffers = dri2_fallback_copy_buffers,
762 .query_buffer_age = dri2_fallback_query_buffer_age,
763 .create_wayland_buffer_from_image = dri2_fallback_create_wayland_buffer_from_image,
764 .get_sync_values = dri2_fallback_get_sync_values,
765 .get_dri_drawable = dri2_surface_get_dri_drawable,
766 };
767
768 static const __DRIimageLoaderExtension droid_image_loader_extension = {
769 .base = { __DRI_IMAGE_LOADER, 1 },
770
771 .getBuffers = droid_image_get_buffers,
772 .flushFrontBuffer = droid_flush_front_buffer,
773 };
774
775 EGLBoolean
776 dri2_initialize_android(_EGLDriver *drv, _EGLDisplay *dpy)
777 {
778 struct dri2_egl_display *dri2_dpy;
779 const char *err;
780
781 _eglSetLogProc(droid_log);
782
783 loader_set_logger(_eglLog);
784
785 dri2_dpy = calloc(1, sizeof(*dri2_dpy));
786 if (!dri2_dpy)
787 return _eglError(EGL_BAD_ALLOC, "eglInitialize");
788
789 dpy->DriverData = (void *) dri2_dpy;
790
791 dri2_dpy->fd = droid_open_device();
792 if (dri2_dpy->fd < 0) {
793 err = "DRI2: failed to open device";
794 goto cleanup_display;
795 }
796
797 dri2_dpy->driver_name = loader_get_driver_for_fd(dri2_dpy->fd, 0);
798 if (dri2_dpy->driver_name == NULL) {
799 err = "DRI2: failed to get driver name";
800 goto cleanup_device;
801 }
802
803 if (!dri2_load_driver(dpy)) {
804 err = "DRI2: failed to load driver";
805 goto cleanup_driver_name;
806 }
807
808 dri2_dpy->is_render_node = drmGetNodeTypeFromFd(dri2_dpy->fd) == DRM_NODE_RENDER;
809
810 dri2_dpy->extensions[0] = &droid_image_loader_extension.base;
811 dri2_dpy->extensions[1] = &use_invalidate.base;
812 dri2_dpy->extensions[2] = &image_lookup_extension.base;
813
814 /* render nodes cannot use Gem names, and thus do not support
815 * the __DRI_DRI2_LOADER extension */
816 if (!dri2_dpy->is_render_node) {
817 dri2_dpy->dri2_loader_extension.base.name = __DRI_DRI2_LOADER;
818 dri2_dpy->dri2_loader_extension.base.version = 3;
819 dri2_dpy->dri2_loader_extension.getBuffers = NULL;
820 dri2_dpy->dri2_loader_extension.flushFrontBuffer = droid_flush_front_buffer;
821 dri2_dpy->dri2_loader_extension.getBuffersWithFormat =
822 droid_get_buffers_with_format;
823 dri2_dpy->extensions[3] = &dri2_dpy->dri2_loader_extension.base;
824 dri2_dpy->extensions[4] = NULL;
825 } else
826 dri2_dpy->extensions[3] = NULL;
827
828
829 if (!dri2_create_screen(dpy)) {
830 err = "DRI2: failed to create screen";
831 goto cleanup_driver;
832 }
833
834 if (!droid_add_configs_for_visuals(drv, dpy)) {
835 err = "DRI2: failed to add configs";
836 goto cleanup_screen;
837 }
838
839 dpy->Extensions.ANDROID_framebuffer_target = EGL_TRUE;
840 dpy->Extensions.ANDROID_image_native_buffer = EGL_TRUE;
841 dpy->Extensions.ANDROID_recordable = EGL_TRUE;
842 dpy->Extensions.KHR_image_base = EGL_TRUE;
843
844 /* Fill vtbl last to prevent accidentally calling virtual function during
845 * initialization.
846 */
847 dri2_dpy->vtbl = &droid_display_vtbl;
848
849 return EGL_TRUE;
850
851 cleanup_screen:
852 dri2_dpy->core->destroyScreen(dri2_dpy->dri_screen);
853 cleanup_driver:
854 dlclose(dri2_dpy->driver);
855 cleanup_driver_name:
856 free(dri2_dpy->driver_name);
857 cleanup_device:
858 close(dri2_dpy->fd);
859 cleanup_display:
860 free(dri2_dpy);
861
862 return _eglError(EGL_NOT_INITIALIZED, err);
863 }