egl/android: remove config post-processing
[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 #define ALIGN(val, align) (((val) + (align) - 1) & ~((align) - 1))
44
45 static int
46 get_format_bpp(int native)
47 {
48 int bpp;
49
50 switch (native) {
51 case HAL_PIXEL_FORMAT_RGBA_8888:
52 case HAL_PIXEL_FORMAT_RGBX_8888:
53 case HAL_PIXEL_FORMAT_BGRA_8888:
54 bpp = 4;
55 break;
56 case HAL_PIXEL_FORMAT_RGB_888:
57 bpp = 3;
58 break;
59 case HAL_PIXEL_FORMAT_RGB_565:
60 bpp = 2;
61 break;
62 case HAL_PIXEL_FORMAT_YV12:
63 bpp = 1;
64 break;
65 default:
66 bpp = 0;
67 break;
68 }
69
70 return bpp;
71 }
72
73 /* createImageFromFds requires fourcc format */
74 static int get_fourcc(int native)
75 {
76 switch (native) {
77 case HAL_PIXEL_FORMAT_RGB_565: return __DRI_IMAGE_FOURCC_RGB565;
78 case HAL_PIXEL_FORMAT_BGRA_8888: return __DRI_IMAGE_FOURCC_ARGB8888;
79 case HAL_PIXEL_FORMAT_RGBA_8888: return __DRI_IMAGE_FOURCC_ABGR8888;
80 case HAL_PIXEL_FORMAT_RGBX_8888: return __DRI_IMAGE_FOURCC_XBGR8888;
81 case HAL_PIXEL_FORMAT_YV12: return __DRI_IMAGE_FOURCC_YVU420;
82 default:
83 _eglLog(_EGL_WARNING, "unsupported native buffer format 0x%x", native);
84 }
85 return -1;
86 }
87
88 static int get_format(int format)
89 {
90 switch (format) {
91 case HAL_PIXEL_FORMAT_BGRA_8888: return __DRI_IMAGE_FORMAT_ARGB8888;
92 case HAL_PIXEL_FORMAT_RGB_565: return __DRI_IMAGE_FORMAT_RGB565;
93 case HAL_PIXEL_FORMAT_RGBA_8888: return __DRI_IMAGE_FORMAT_ABGR8888;
94 case HAL_PIXEL_FORMAT_RGBX_8888: return __DRI_IMAGE_FORMAT_XBGR8888;
95 case HAL_PIXEL_FORMAT_RGB_888:
96 /* unsupported */
97 default:
98 _eglLog(_EGL_WARNING, "unsupported native buffer format 0x%x", format);
99 }
100 return -1;
101 }
102
103 static int
104 get_native_buffer_fd(struct ANativeWindowBuffer *buf)
105 {
106 native_handle_t *handle = (native_handle_t *)buf->handle;
107 /*
108 * Various gralloc implementations exist, but the dma-buf fd tends
109 * to be first. Access it directly to avoid a dependency on specific
110 * gralloc versions.
111 */
112 return (handle && handle->numFds) ? handle->data[0] : -1;
113 }
114
115 static int
116 get_native_buffer_name(struct ANativeWindowBuffer *buf)
117 {
118 return gralloc_drm_get_gem_handle(buf->handle);
119 }
120
121 static EGLBoolean
122 droid_window_dequeue_buffer(struct dri2_egl_surface *dri2_surf)
123 {
124 #if ANDROID_VERSION >= 0x0402
125 int fence_fd;
126
127 if (dri2_surf->window->dequeueBuffer(dri2_surf->window, &dri2_surf->buffer,
128 &fence_fd))
129 return EGL_FALSE;
130
131 /* If access to the buffer is controlled by a sync fence, then block on the
132 * fence.
133 *
134 * It may be more performant to postpone blocking until there is an
135 * immediate need to write to the buffer. But doing so would require adding
136 * hooks to the DRI2 loader.
137 *
138 * From the ANativeWindow::dequeueBuffer documentation:
139 *
140 * The libsync fence file descriptor returned in the int pointed to by
141 * the fenceFd argument will refer to the fence that must signal
142 * before the dequeued buffer may be written to. A value of -1
143 * indicates that the caller may access the buffer immediately without
144 * waiting on a fence. If a valid file descriptor is returned (i.e.
145 * any value except -1) then the caller is responsible for closing the
146 * file descriptor.
147 */
148 if (fence_fd >= 0) {
149 /* From the SYNC_IOC_WAIT documentation in <linux/sync.h>:
150 *
151 * Waits indefinitely if timeout < 0.
152 */
153 int timeout = -1;
154 sync_wait(fence_fd, timeout);
155 close(fence_fd);
156 }
157
158 dri2_surf->buffer->common.incRef(&dri2_surf->buffer->common);
159 #else
160 if (dri2_surf->window->dequeueBuffer(dri2_surf->window, &dri2_surf->buffer))
161 return EGL_FALSE;
162
163 dri2_surf->buffer->common.incRef(&dri2_surf->buffer->common);
164 dri2_surf->window->lockBuffer(dri2_surf->window, dri2_surf->buffer);
165 #endif
166
167 return EGL_TRUE;
168 }
169
170 static EGLBoolean
171 droid_window_enqueue_buffer(_EGLDisplay *disp, struct dri2_egl_surface *dri2_surf)
172 {
173 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
174
175 /* To avoid blocking other EGL calls, release the display mutex before
176 * we enter droid_window_enqueue_buffer() and re-acquire the mutex upon
177 * return.
178 */
179 mtx_unlock(&disp->Mutex);
180
181 #if ANDROID_VERSION >= 0x0402
182 /* Queue the buffer without a sync fence. This informs the ANativeWindow
183 * that it may access the buffer immediately.
184 *
185 * From ANativeWindow::dequeueBuffer:
186 *
187 * The fenceFd argument specifies a libsync fence file descriptor for
188 * a fence that must signal before the buffer can be accessed. If
189 * the buffer can be accessed immediately then a value of -1 should
190 * be used. The caller must not use the file descriptor after it
191 * is passed to queueBuffer, and the ANativeWindow implementation
192 * is responsible for closing it.
193 */
194 int fence_fd = -1;
195 dri2_surf->window->queueBuffer(dri2_surf->window, dri2_surf->buffer,
196 fence_fd);
197 #else
198 dri2_surf->window->queueBuffer(dri2_surf->window, dri2_surf->buffer);
199 #endif
200
201 dri2_surf->buffer->common.decRef(&dri2_surf->buffer->common);
202 dri2_surf->buffer = NULL;
203
204 mtx_lock(&disp->Mutex);
205
206 if (dri2_surf->dri_image) {
207 dri2_dpy->image->destroyImage(dri2_surf->dri_image);
208 dri2_surf->dri_image = NULL;
209 }
210
211 return EGL_TRUE;
212 }
213
214 static void
215 droid_window_cancel_buffer(_EGLDisplay *disp, struct dri2_egl_surface *dri2_surf)
216 {
217 /* no cancel buffer? */
218 droid_window_enqueue_buffer(disp, dri2_surf);
219 }
220
221 static __DRIbuffer *
222 droid_alloc_local_buffer(struct dri2_egl_surface *dri2_surf,
223 unsigned int att, unsigned int format)
224 {
225 struct dri2_egl_display *dri2_dpy =
226 dri2_egl_display(dri2_surf->base.Resource.Display);
227
228 if (att >= ARRAY_SIZE(dri2_surf->local_buffers))
229 return NULL;
230
231 if (!dri2_surf->local_buffers[att]) {
232 dri2_surf->local_buffers[att] =
233 dri2_dpy->dri2->allocateBuffer(dri2_dpy->dri_screen, att, format,
234 dri2_surf->base.Width, dri2_surf->base.Height);
235 }
236
237 return dri2_surf->local_buffers[att];
238 }
239
240 static void
241 droid_free_local_buffers(struct dri2_egl_surface *dri2_surf)
242 {
243 struct dri2_egl_display *dri2_dpy =
244 dri2_egl_display(dri2_surf->base.Resource.Display);
245 int i;
246
247 for (i = 0; i < ARRAY_SIZE(dri2_surf->local_buffers); i++) {
248 if (dri2_surf->local_buffers[i]) {
249 dri2_dpy->dri2->releaseBuffer(dri2_dpy->dri_screen,
250 dri2_surf->local_buffers[i]);
251 dri2_surf->local_buffers[i] = NULL;
252 }
253 }
254 }
255
256 static _EGLSurface *
257 droid_create_surface(_EGLDriver *drv, _EGLDisplay *disp, EGLint type,
258 _EGLConfig *conf, void *native_window,
259 const EGLint *attrib_list)
260 {
261 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
262 struct dri2_egl_config *dri2_conf = dri2_egl_config(conf);
263 struct dri2_egl_surface *dri2_surf;
264 struct ANativeWindow *window = native_window;
265 const __DRIconfig *config;
266
267 dri2_surf = calloc(1, sizeof *dri2_surf);
268 if (!dri2_surf) {
269 _eglError(EGL_BAD_ALLOC, "droid_create_surface");
270 return NULL;
271 }
272
273 if (!_eglInitSurface(&dri2_surf->base, disp, type, conf, attrib_list))
274 goto cleanup_surface;
275
276 if (type == EGL_WINDOW_BIT) {
277 int format;
278
279 if (!window || window->common.magic != ANDROID_NATIVE_WINDOW_MAGIC) {
280 _eglError(EGL_BAD_NATIVE_WINDOW, "droid_create_surface");
281 goto cleanup_surface;
282 }
283 if (window->query(window, NATIVE_WINDOW_FORMAT, &format)) {
284 _eglError(EGL_BAD_NATIVE_WINDOW, "droid_create_surface");
285 goto cleanup_surface;
286 }
287
288 if (format != dri2_conf->base.NativeVisualID) {
289 _eglLog(_EGL_WARNING, "Native format mismatch: 0x%x != 0x%x",
290 format, dri2_conf->base.NativeVisualID);
291 }
292
293 window->query(window, NATIVE_WINDOW_WIDTH, &dri2_surf->base.Width);
294 window->query(window, NATIVE_WINDOW_HEIGHT, &dri2_surf->base.Height);
295 }
296
297 config = dri2_get_dri_config(dri2_conf, EGL_WINDOW_BIT,
298 dri2_surf->base.GLColorspace);
299 if (!config)
300 goto cleanup_surface;
301
302 dri2_surf->dri_drawable =
303 (*dri2_dpy->dri2->createNewDrawable)(dri2_dpy->dri_screen, config,
304 dri2_surf);
305 if (dri2_surf->dri_drawable == NULL) {
306 _eglError(EGL_BAD_ALLOC, "dri2->createNewDrawable");
307 goto cleanup_surface;
308 }
309
310 if (window) {
311 window->common.incRef(&window->common);
312 dri2_surf->window = window;
313 }
314
315 return &dri2_surf->base;
316
317 cleanup_surface:
318 free(dri2_surf);
319
320 return NULL;
321 }
322
323 static _EGLSurface *
324 droid_create_window_surface(_EGLDriver *drv, _EGLDisplay *disp,
325 _EGLConfig *conf, void *native_window,
326 const EGLint *attrib_list)
327 {
328 return droid_create_surface(drv, disp, EGL_WINDOW_BIT, conf,
329 native_window, attrib_list);
330 }
331
332 static _EGLSurface *
333 droid_create_pbuffer_surface(_EGLDriver *drv, _EGLDisplay *disp,
334 _EGLConfig *conf, const EGLint *attrib_list)
335 {
336 return droid_create_surface(drv, disp, EGL_PBUFFER_BIT, conf,
337 NULL, attrib_list);
338 }
339
340 static EGLBoolean
341 droid_destroy_surface(_EGLDriver *drv, _EGLDisplay *disp, _EGLSurface *surf)
342 {
343 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
344 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surf);
345
346 if (!_eglPutSurface(surf))
347 return EGL_TRUE;
348
349 droid_free_local_buffers(dri2_surf);
350
351 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
352 if (dri2_surf->buffer)
353 droid_window_cancel_buffer(disp, dri2_surf);
354
355 dri2_surf->window->common.decRef(&dri2_surf->window->common);
356 }
357
358 (*dri2_dpy->core->destroyDrawable)(dri2_surf->dri_drawable);
359
360 free(dri2_surf);
361
362 return EGL_TRUE;
363 }
364
365 static int
366 update_buffers(struct dri2_egl_surface *dri2_surf)
367 {
368 if (dri2_surf->base.Type != EGL_WINDOW_BIT)
369 return 0;
370
371 /* try to dequeue the next back buffer */
372 if (!dri2_surf->buffer && !droid_window_dequeue_buffer(dri2_surf)) {
373 _eglLog(_EGL_WARNING, "Could not dequeue buffer from native window");
374 return -1;
375 }
376
377 /* free outdated buffers and update the surface size */
378 if (dri2_surf->base.Width != dri2_surf->buffer->width ||
379 dri2_surf->base.Height != dri2_surf->buffer->height) {
380 droid_free_local_buffers(dri2_surf);
381 dri2_surf->base.Width = dri2_surf->buffer->width;
382 dri2_surf->base.Height = dri2_surf->buffer->height;
383 }
384
385 return 0;
386 }
387
388 static int
389 get_back_bo(struct dri2_egl_surface *dri2_surf)
390 {
391 struct dri2_egl_display *dri2_dpy =
392 dri2_egl_display(dri2_surf->base.Resource.Display);
393 int fourcc, pitch;
394 int offset = 0, fd;
395
396 if (dri2_surf->dri_image)
397 return 0;
398
399 if (!dri2_surf->buffer)
400 return -1;
401
402 fd = get_native_buffer_fd(dri2_surf->buffer);
403 if (fd < 0) {
404 _eglLog(_EGL_WARNING, "Could not get native buffer FD");
405 return -1;
406 }
407
408 fourcc = get_fourcc(dri2_surf->buffer->format);
409
410 pitch = dri2_surf->buffer->stride *
411 get_format_bpp(dri2_surf->buffer->format);
412
413 if (fourcc == -1 || pitch == 0) {
414 _eglLog(_EGL_WARNING, "Invalid buffer fourcc(%x) or pitch(%d)",
415 fourcc, pitch);
416 return -1;
417 }
418
419 dri2_surf->dri_image =
420 dri2_dpy->image->createImageFromFds(dri2_dpy->dri_screen,
421 dri2_surf->base.Width,
422 dri2_surf->base.Height,
423 fourcc,
424 &fd,
425 1,
426 &pitch,
427 &offset,
428 dri2_surf);
429 if (!dri2_surf->dri_image)
430 return -1;
431
432 return 0;
433 }
434
435 static int
436 droid_image_get_buffers(__DRIdrawable *driDrawable,
437 unsigned int format,
438 uint32_t *stamp,
439 void *loaderPrivate,
440 uint32_t buffer_mask,
441 struct __DRIimageList *images)
442 {
443 struct dri2_egl_surface *dri2_surf = loaderPrivate;
444
445 images->image_mask = 0;
446
447 if (update_buffers(dri2_surf) < 0)
448 return 0;
449
450 if (buffer_mask & __DRI_IMAGE_BUFFER_FRONT) {
451 /*
452 * We don't support front buffers and GLES doesn't require them for
453 * window surfaces, but some DRI drivers will request them anyway.
454 * We just ignore such request as other platforms backends do.
455 */
456 }
457
458 if (buffer_mask & __DRI_IMAGE_BUFFER_BACK) {
459 if (get_back_bo(dri2_surf) < 0)
460 return 0;
461
462 images->back = dri2_surf->dri_image;
463 images->image_mask |= __DRI_IMAGE_BUFFER_BACK;
464 }
465
466 return 1;
467 }
468
469 static EGLBoolean
470 droid_swap_buffers(_EGLDriver *drv, _EGLDisplay *disp, _EGLSurface *draw)
471 {
472 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
473 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(draw);
474
475 if (dri2_surf->base.Type != EGL_WINDOW_BIT)
476 return EGL_TRUE;
477
478 dri2_flush_drawable_for_swapbuffers(disp, draw);
479
480 if (dri2_surf->buffer)
481 droid_window_enqueue_buffer(disp, dri2_surf);
482
483 (*dri2_dpy->flush->invalidate)(dri2_surf->dri_drawable);
484
485 return EGL_TRUE;
486 }
487
488 static _EGLImage *
489 droid_create_image_from_prime_fd(_EGLDisplay *disp, _EGLContext *ctx,
490 struct ANativeWindowBuffer *buf, int fd)
491 {
492 unsigned int offsets[3] = { 0, 0, 0 };
493 unsigned int pitches[3] = { 0, 0, 0 };
494
495 const int fourcc = get_fourcc(buf->format);
496 if (fourcc == -1) {
497 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
498 return NULL;
499 }
500
501 pitches[0] = buf->stride * get_format_bpp(buf->format);
502 if (pitches[0] == 0) {
503 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
504 return NULL;
505 }
506
507 switch (buf->format) {
508 case HAL_PIXEL_FORMAT_YV12:
509 /* Y plane is assumed to be at offset 0. */
510 /* Cr plane is located after Y plane */
511 offsets[1] = offsets[0] + pitches[0] * buf->height;
512 pitches[1] = ALIGN(pitches[0] / 2, 16);
513 /* Cb plane is located after Cr plane */
514 offsets[2] = offsets[1] + pitches[1] * buf->height / 2;
515 pitches[2] = pitches[1];
516
517 const EGLint attr_list_yv12[] = {
518 EGL_WIDTH, buf->width,
519 EGL_HEIGHT, buf->height,
520 EGL_LINUX_DRM_FOURCC_EXT, fourcc,
521 EGL_DMA_BUF_PLANE0_FD_EXT, fd,
522 EGL_DMA_BUF_PLANE0_PITCH_EXT, pitches[0],
523 EGL_DMA_BUF_PLANE0_OFFSET_EXT, offsets[0],
524 EGL_DMA_BUF_PLANE1_FD_EXT, fd,
525 EGL_DMA_BUF_PLANE1_PITCH_EXT, pitches[1],
526 EGL_DMA_BUF_PLANE1_OFFSET_EXT, offsets[1],
527 EGL_DMA_BUF_PLANE2_FD_EXT, fd,
528 EGL_DMA_BUF_PLANE2_PITCH_EXT, pitches[2],
529 EGL_DMA_BUF_PLANE2_OFFSET_EXT, offsets[2],
530 EGL_NONE, 0
531 };
532
533 return dri2_create_image_dma_buf(disp, ctx, NULL, attr_list_yv12);
534 }
535
536 const EGLint attr_list[] = {
537 EGL_WIDTH, buf->width,
538 EGL_HEIGHT, buf->height,
539 EGL_LINUX_DRM_FOURCC_EXT, fourcc,
540 EGL_DMA_BUF_PLANE0_FD_EXT, fd,
541 EGL_DMA_BUF_PLANE0_PITCH_EXT, pitches[0],
542 EGL_DMA_BUF_PLANE0_OFFSET_EXT, 0,
543 EGL_NONE, 0
544 };
545
546 return dri2_create_image_dma_buf(disp, ctx, NULL, attr_list);
547 }
548
549 static _EGLImage *
550 droid_create_image_from_name(_EGLDisplay *disp, _EGLContext *ctx,
551 struct ANativeWindowBuffer *buf)
552 {
553 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
554 struct dri2_egl_image *dri2_img;
555 int name;
556 int format;
557
558 name = get_native_buffer_name(buf);
559 if (!name) {
560 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
561 return NULL;
562 }
563
564 format = get_format(buf->format);
565 if (format == -1)
566 return NULL;
567
568 dri2_img = calloc(1, sizeof(*dri2_img));
569 if (!dri2_img) {
570 _eglError(EGL_BAD_ALLOC, "droid_create_image_mesa_drm");
571 return NULL;
572 }
573
574 if (!_eglInitImage(&dri2_img->base, disp)) {
575 free(dri2_img);
576 return NULL;
577 }
578
579 dri2_img->dri_image =
580 dri2_dpy->image->createImageFromName(dri2_dpy->dri_screen,
581 buf->width,
582 buf->height,
583 format,
584 name,
585 buf->stride,
586 dri2_img);
587 if (!dri2_img->dri_image) {
588 free(dri2_img);
589 _eglError(EGL_BAD_ALLOC, "droid_create_image_mesa_drm");
590 return NULL;
591 }
592
593 return &dri2_img->base;
594 }
595
596 static EGLBoolean
597 droid_query_surface(_EGLDriver *drv, _EGLDisplay *dpy, _EGLSurface *surf,
598 EGLint attribute, EGLint *value)
599 {
600 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surf);
601 switch (attribute) {
602 case EGL_WIDTH:
603 if (dri2_surf->base.Type == EGL_WINDOW_BIT && dri2_surf->window) {
604 dri2_surf->window->query(dri2_surf->window,
605 NATIVE_WINDOW_DEFAULT_WIDTH, value);
606 return EGL_TRUE;
607 }
608 break;
609 case EGL_HEIGHT:
610 if (dri2_surf->base.Type == EGL_WINDOW_BIT && dri2_surf->window) {
611 dri2_surf->window->query(dri2_surf->window,
612 NATIVE_WINDOW_DEFAULT_HEIGHT, value);
613 return EGL_TRUE;
614 }
615 break;
616 default:
617 break;
618 }
619 return _eglQuerySurface(drv, dpy, surf, attribute, value);
620 }
621
622 static _EGLImage *
623 dri2_create_image_android_native_buffer(_EGLDisplay *disp,
624 _EGLContext *ctx,
625 struct ANativeWindowBuffer *buf)
626 {
627 int fd;
628
629 if (ctx != NULL) {
630 /* From the EGL_ANDROID_image_native_buffer spec:
631 *
632 * * If <target> is EGL_NATIVE_BUFFER_ANDROID and <ctx> is not
633 * EGL_NO_CONTEXT, the error EGL_BAD_CONTEXT is generated.
634 */
635 _eglError(EGL_BAD_CONTEXT, "eglCreateEGLImageKHR: for "
636 "EGL_NATIVE_BUFFER_ANDROID, the context must be "
637 "EGL_NO_CONTEXT");
638 return NULL;
639 }
640
641 if (!buf || buf->common.magic != ANDROID_NATIVE_BUFFER_MAGIC ||
642 buf->common.version != sizeof(*buf)) {
643 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
644 return NULL;
645 }
646
647 fd = get_native_buffer_fd(buf);
648 if (fd >= 0)
649 return droid_create_image_from_prime_fd(disp, ctx, buf, fd);
650
651 return droid_create_image_from_name(disp, ctx, buf);
652 }
653
654 static _EGLImage *
655 droid_create_image_khr(_EGLDriver *drv, _EGLDisplay *disp,
656 _EGLContext *ctx, EGLenum target,
657 EGLClientBuffer buffer, const EGLint *attr_list)
658 {
659 switch (target) {
660 case EGL_NATIVE_BUFFER_ANDROID:
661 return dri2_create_image_android_native_buffer(disp, ctx,
662 (struct ANativeWindowBuffer *) buffer);
663 default:
664 return dri2_create_image_khr(drv, disp, ctx, target, buffer, attr_list);
665 }
666 }
667
668 static void
669 droid_flush_front_buffer(__DRIdrawable * driDrawable, void *loaderPrivate)
670 {
671 }
672
673 static int
674 droid_get_buffers_parse_attachments(struct dri2_egl_surface *dri2_surf,
675 unsigned int *attachments, int count)
676 {
677 int num_buffers = 0, i;
678
679 /* fill dri2_surf->buffers */
680 for (i = 0; i < count * 2; i += 2) {
681 __DRIbuffer *buf, *local;
682
683 assert(num_buffers < ARRAY_SIZE(dri2_surf->buffers));
684 buf = &dri2_surf->buffers[num_buffers];
685
686 switch (attachments[i]) {
687 case __DRI_BUFFER_BACK_LEFT:
688 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
689 buf->attachment = attachments[i];
690 buf->name = get_native_buffer_name(dri2_surf->buffer);
691 buf->cpp = get_format_bpp(dri2_surf->buffer->format);
692 buf->pitch = dri2_surf->buffer->stride * buf->cpp;
693 buf->flags = 0;
694
695 if (buf->name)
696 num_buffers++;
697
698 break;
699 }
700 /* fall through for pbuffers */
701 case __DRI_BUFFER_DEPTH:
702 case __DRI_BUFFER_STENCIL:
703 case __DRI_BUFFER_ACCUM:
704 case __DRI_BUFFER_DEPTH_STENCIL:
705 case __DRI_BUFFER_HIZ:
706 local = droid_alloc_local_buffer(dri2_surf,
707 attachments[i], attachments[i + 1]);
708
709 if (local) {
710 *buf = *local;
711 num_buffers++;
712 }
713 break;
714 case __DRI_BUFFER_FRONT_LEFT:
715 case __DRI_BUFFER_FRONT_RIGHT:
716 case __DRI_BUFFER_FAKE_FRONT_LEFT:
717 case __DRI_BUFFER_FAKE_FRONT_RIGHT:
718 case __DRI_BUFFER_BACK_RIGHT:
719 default:
720 /* no front or right buffers */
721 break;
722 }
723 }
724
725 return num_buffers;
726 }
727
728 static __DRIbuffer *
729 droid_get_buffers_with_format(__DRIdrawable * driDrawable,
730 int *width, int *height,
731 unsigned int *attachments, int count,
732 int *out_count, void *loaderPrivate)
733 {
734 struct dri2_egl_surface *dri2_surf = loaderPrivate;
735
736 if (update_buffers(dri2_surf) < 0)
737 return NULL;
738
739 dri2_surf->buffer_count =
740 droid_get_buffers_parse_attachments(dri2_surf, attachments, count);
741
742 if (width)
743 *width = dri2_surf->base.Width;
744 if (height)
745 *height = dri2_surf->base.Height;
746
747 *out_count = dri2_surf->buffer_count;
748
749 return dri2_surf->buffers;
750 }
751
752 static EGLBoolean
753 droid_add_configs_for_visuals(_EGLDriver *drv, _EGLDisplay *dpy)
754 {
755 struct dri2_egl_display *dri2_dpy = dri2_egl_display(dpy);
756 static const struct {
757 int format;
758 unsigned int rgba_masks[4];
759 } visuals[] = {
760 { HAL_PIXEL_FORMAT_RGBA_8888, { 0xff, 0xff00, 0xff0000, 0xff000000 } },
761 { HAL_PIXEL_FORMAT_RGBX_8888, { 0xff, 0xff00, 0xff0000, 0x0 } },
762 { HAL_PIXEL_FORMAT_RGB_888, { 0xff, 0xff00, 0xff0000, 0x0 } },
763 { HAL_PIXEL_FORMAT_RGB_565, { 0xf800, 0x7e0, 0x1f, 0x0 } },
764 { HAL_PIXEL_FORMAT_BGRA_8888, { 0xff0000, 0xff00, 0xff, 0xff000000 } },
765 { 0, { 0, 0, 0, 0 } }
766 };
767 EGLint config_attrs[] = {
768 EGL_NATIVE_VISUAL_ID, 0,
769 EGL_NATIVE_VISUAL_TYPE, 0,
770 EGL_FRAMEBUFFER_TARGET_ANDROID, EGL_TRUE,
771 EGL_RECORDABLE_ANDROID, EGL_TRUE,
772 EGL_NONE
773 };
774 int count, i, j;
775
776 count = 0;
777 for (i = 0; visuals[i].format; i++) {
778 int format_count = 0;
779
780 config_attrs[1] = visuals[i].format;
781 config_attrs[3] = visuals[i].format;
782
783 for (j = 0; dri2_dpy->driver_configs[j]; j++) {
784 const EGLint surface_type = EGL_WINDOW_BIT | EGL_PBUFFER_BIT;
785 struct dri2_egl_config *dri2_conf;
786 unsigned int double_buffered = 0;
787
788 dri2_dpy->core->getConfigAttrib(dri2_dpy->driver_configs[j],
789 __DRI_ATTRIB_DOUBLE_BUFFER, &double_buffered);
790
791 /* support only double buffered configs */
792 if (!double_buffered)
793 continue;
794
795 dri2_conf = dri2_add_config(dpy, dri2_dpy->driver_configs[j],
796 count + 1, surface_type, config_attrs, visuals[i].rgba_masks);
797 if (dri2_conf) {
798 count++;
799 format_count++;
800 }
801 }
802
803 if (!format_count) {
804 _eglLog(_EGL_DEBUG, "No DRI config supports native format 0x%x",
805 visuals[i].format);
806 }
807 }
808
809 return (count != 0);
810 }
811
812 static int
813 droid_open_device(void)
814 {
815 const hw_module_t *mod;
816 int fd = -1, err;
817
818 err = hw_get_module(GRALLOC_HARDWARE_MODULE_ID, &mod);
819 if (!err) {
820 const gralloc_module_t *gr = (gralloc_module_t *) mod;
821
822 err = -EINVAL;
823 if (gr->perform)
824 err = gr->perform(gr, GRALLOC_MODULE_PERFORM_GET_DRM_FD, &fd);
825 }
826 if (err || fd < 0) {
827 _eglLog(_EGL_WARNING, "fail to get drm fd");
828 fd = -1;
829 }
830
831 return (fd >= 0) ? dup(fd) : -1;
832 }
833
834 /* support versions < JellyBean */
835 #ifndef ALOGW
836 #define ALOGW LOGW
837 #endif
838 #ifndef ALOGD
839 #define ALOGD LOGD
840 #endif
841 #ifndef ALOGI
842 #define ALOGI LOGI
843 #endif
844
845 static void
846 droid_log(EGLint level, const char *msg)
847 {
848 switch (level) {
849 case _EGL_DEBUG:
850 ALOGD("%s", msg);
851 break;
852 case _EGL_INFO:
853 ALOGI("%s", msg);
854 break;
855 case _EGL_WARNING:
856 ALOGW("%s", msg);
857 break;
858 case _EGL_FATAL:
859 LOG_FATAL("%s", msg);
860 break;
861 default:
862 break;
863 }
864 }
865
866 static struct dri2_egl_display_vtbl droid_display_vtbl = {
867 .authenticate = NULL,
868 .create_window_surface = droid_create_window_surface,
869 .create_pixmap_surface = dri2_fallback_create_pixmap_surface,
870 .create_pbuffer_surface = droid_create_pbuffer_surface,
871 .destroy_surface = droid_destroy_surface,
872 .create_image = droid_create_image_khr,
873 .swap_interval = dri2_fallback_swap_interval,
874 .swap_buffers = droid_swap_buffers,
875 .swap_buffers_with_damage = dri2_fallback_swap_buffers_with_damage,
876 .swap_buffers_region = dri2_fallback_swap_buffers_region,
877 .post_sub_buffer = dri2_fallback_post_sub_buffer,
878 .copy_buffers = dri2_fallback_copy_buffers,
879 .query_buffer_age = dri2_fallback_query_buffer_age,
880 .query_surface = droid_query_surface,
881 .create_wayland_buffer_from_image = dri2_fallback_create_wayland_buffer_from_image,
882 .get_sync_values = dri2_fallback_get_sync_values,
883 .get_dri_drawable = dri2_surface_get_dri_drawable,
884 };
885
886 static const __DRIimageLoaderExtension droid_image_loader_extension = {
887 .base = { __DRI_IMAGE_LOADER, 1 },
888
889 .getBuffers = droid_image_get_buffers,
890 .flushFrontBuffer = droid_flush_front_buffer,
891 };
892
893 EGLBoolean
894 dri2_initialize_android(_EGLDriver *drv, _EGLDisplay *dpy)
895 {
896 struct dri2_egl_display *dri2_dpy;
897 const char *err;
898
899 _eglSetLogProc(droid_log);
900
901 loader_set_logger(_eglLog);
902
903 dri2_dpy = calloc(1, sizeof(*dri2_dpy));
904 if (!dri2_dpy)
905 return _eglError(EGL_BAD_ALLOC, "eglInitialize");
906
907 dpy->DriverData = (void *) dri2_dpy;
908
909 dri2_dpy->fd = droid_open_device();
910 if (dri2_dpy->fd < 0) {
911 err = "DRI2: failed to open device";
912 goto cleanup_display;
913 }
914
915 dri2_dpy->driver_name = loader_get_driver_for_fd(dri2_dpy->fd, 0);
916 if (dri2_dpy->driver_name == NULL) {
917 err = "DRI2: failed to get driver name";
918 goto cleanup_device;
919 }
920
921 if (!dri2_load_driver(dpy)) {
922 err = "DRI2: failed to load driver";
923 goto cleanup_driver_name;
924 }
925
926 dri2_dpy->is_render_node = drmGetNodeTypeFromFd(dri2_dpy->fd) == DRM_NODE_RENDER;
927
928 /* render nodes cannot use Gem names, and thus do not support
929 * the __DRI_DRI2_LOADER extension */
930 if (!dri2_dpy->is_render_node) {
931 dri2_dpy->dri2_loader_extension.base.name = __DRI_DRI2_LOADER;
932 dri2_dpy->dri2_loader_extension.base.version = 3;
933 dri2_dpy->dri2_loader_extension.getBuffers = NULL;
934 dri2_dpy->dri2_loader_extension.flushFrontBuffer = droid_flush_front_buffer;
935 dri2_dpy->dri2_loader_extension.getBuffersWithFormat =
936 droid_get_buffers_with_format;
937 dri2_dpy->extensions[0] = &dri2_dpy->dri2_loader_extension.base;
938 } else {
939 dri2_dpy->extensions[0] = &droid_image_loader_extension.base;
940 }
941 dri2_dpy->extensions[1] = &use_invalidate.base;
942 dri2_dpy->extensions[2] = &image_lookup_extension.base;
943 dri2_dpy->extensions[3] = NULL;
944
945
946 if (!dri2_create_screen(dpy)) {
947 err = "DRI2: failed to create screen";
948 goto cleanup_driver;
949 }
950
951 if (!droid_add_configs_for_visuals(drv, dpy)) {
952 err = "DRI2: failed to add configs";
953 goto cleanup_screen;
954 }
955
956 dpy->Extensions.ANDROID_framebuffer_target = EGL_TRUE;
957 dpy->Extensions.ANDROID_image_native_buffer = EGL_TRUE;
958 dpy->Extensions.ANDROID_recordable = EGL_TRUE;
959 dpy->Extensions.KHR_image_base = EGL_TRUE;
960
961 /* Fill vtbl last to prevent accidentally calling virtual function during
962 * initialization.
963 */
964 dri2_dpy->vtbl = &droid_display_vtbl;
965
966 return EGL_TRUE;
967
968 cleanup_screen:
969 dri2_dpy->core->destroyScreen(dri2_dpy->dri_screen);
970 cleanup_driver:
971 dlclose(dri2_dpy->driver);
972 cleanup_driver_name:
973 free(dri2_dpy->driver_name);
974 cleanup_device:
975 close(dri2_dpy->fd);
976 cleanup_display:
977 free(dri2_dpy);
978 dpy->DriverData = NULL;
979
980 return _eglError(EGL_NOT_INITIALIZED, err);
981 }