1 /*
2 * Mesa 3-D graphics library
3 *
4 * Copyright (c) 2021 Huawei Device Co., Ltd. *
5 *
6 * Based on platform_android, which has
7 *
8 * Copyright (C) 2010-2011 Chia-I Wu <olvaffe@gmail.com>
9 * Copyright (C) 2010-2011 LunarG Inc.
10 *
11 * Based on platform_x11, which has
12 *
13 * Copyright © 2011 Intel Corporation
14 *
15 * Permission is hereby granted, free of charge, to any person obtaining a
16 * copy of this software and associated documentation files (the "Software"),
17 * to deal in the Software without restriction, including without limitation
18 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
19 * and/or sell copies of the Software, and to permit persons to whom the
20 * Software is furnished to do so, subject to the following conditions:
21 *
22 * The above copyright notice and this permission notice shall be included
23 * in all copies or substantial portions of the Software.
24 *
25 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
26 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
27 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
28 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
29 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
30 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
31 * DEALINGS IN THE SOFTWARE.
32 */
33
34 #include <errno.h>
35 #include <dirent.h>
36 #include <dlfcn.h>
37 #include <fcntl.h>
38 #include <xf86drm.h>
39 #include <stdbool.h>
40 #include <stdio.h>
41 #include <sys/types.h>
42 #include <drm-uapi/drm_fourcc.h>
43
44 #include "util/compiler.h"
45 #include "util/os_file.h"
46
47 #include "loader.h"
48 #include "egl_dri2.h"
49 #include "platform_ohos.h"
50 #include "libsync.h"
51 #ifdef HAVE_DRM_GRALLOC
52 #include <gralloc_drm_handle.h>
53 #include "gralloc_drm.h"
54 #endif /* HAVE_DRM_GRALLOC */
55 #include "ohos_log.h"
56
57 #define ALIGN(val, align) (((val) + (align)-1) & ~((align)-1))
58
59 static int
get_format_bpp(int native)60 get_format_bpp(int native)
61 {
62 int bpp;
63
64 switch (native) {
65 case PIXEL_FMT_RGBA_8888:
66 case PIXEL_FMT_RGBX_8888:
67 case PIXEL_FMT_BGRA_8888:
68 bpp = 4;
69 break;
70 case PIXEL_FMT_RGB_565:
71 bpp = 2;
72 break;
73 default:
74 bpp = 0;
75 break;
76 }
77
78 return bpp;
79 }
80
81 /* returns # of fds, and by reference the actual fds */
82 static unsigned
get_native_buffer_fds(struct ANativeWindowBuffer *buf, int fds[3])83 get_native_buffer_fds(struct ANativeWindowBuffer *buf, int fds[3])
84 {
85 BufferHandle *handle = GetBufferHandleFromNative(buf);
86 if (handle == NULL) {
87 return 0;
88 }
89
90 fds[0] = handle->fd;
91 return 1;
92 }
93
94 /* createImageFromFds requires fourcc format */
get_fourcc(int native)95 static int get_fourcc(int native)
96 {
97 switch (native) {
98 case PIXEL_FMT_RGB_565:
99 return DRM_FORMAT_RGB565;
100 case PIXEL_FMT_BGRA_8888:
101 return DRM_FORMAT_ARGB8888;
102 case PIXEL_FMT_RGBA_8888:
103 return DRM_FORMAT_ABGR8888;
104 case PIXEL_FMT_RGBX_8888:
105 return DRM_FORMAT_XBGR8888;
106 default:
107 _eglLog(_EGL_WARNING, "unsupported native buffer format 0x%{public}x", native);
108 }
109 return -1;
110 }
111
112 static int
native_window_buffer_get_buffer_info(struct dri2_egl_display *dri2_dpy, struct ANativeWindowBuffer *buf, struct buffer_info *out_buf_info)113 native_window_buffer_get_buffer_info(struct dri2_egl_display *dri2_dpy,
114 struct ANativeWindowBuffer *buf,
115 struct buffer_info *out_buf_info)
116 {
117 int num_planes = 0;
118 int drm_fourcc = 0;
119 int pitch = 0;
120 int fds[3];
121 /*
122 * Non-YUV formats could *also* have multiple planes, such as ancillary
123 * color compression state buffer, but the rest of the code isn't ready
124 * yet to deal with modifiers:
125 */
126 num_planes = get_native_buffer_fds(buf, fds);
127 if (num_planes == 0) {
128 return -EINVAL;
129 }
130
131 assert(num_planes == 1);
132 BufferHandle *bufferHandle;
133 bufferHandle = GetBufferHandleFromNative(buf);
134 drm_fourcc = get_fourcc(bufferHandle->format);
135 if (drm_fourcc == -1) {
136 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
137 return -EINVAL;
138 }
139 pitch = bufferHandle->stride;
140 if (pitch == 0) {
141 _eglError(EGL_BAD_PARAMETER, "eglCreateEGLImageKHR");
142 return -EINVAL;
143 }
144
145 *out_buf_info = (struct buffer_info) {
146 .width = bufferHandle->width,
147 .height = bufferHandle->height,
148 .drm_fourcc = drm_fourcc,
149 .num_planes = num_planes,
150 .fds = {fds[0], -1, -1, -1},
151 .modifier = DRM_FORMAT_MOD_INVALID,
152 .offsets = {0, 0, 0, 0},
153 .pitches = {pitch, 0, 0, 0},
154 .yuv_color_space = EGL_ITU_REC601_EXT,
155 .sample_range = EGL_YUV_NARROW_RANGE_EXT,
156 .horizontal_siting = EGL_YUV_CHROMA_SITING_0_EXT,
157 .vertical_siting = EGL_YUV_CHROMA_SITING_0_EXT,
158 };
159
160 return 0;
161 }
162
163 static __DRIimage *
ohos_create_image_from_buffer_info(struct dri2_egl_display *dri2_dpy, struct buffer_info *buf_info, void *priv)164 ohos_create_image_from_buffer_info(struct dri2_egl_display *dri2_dpy,
165 struct buffer_info *buf_info,
166 void *priv)
167 {
168 unsigned error;
169
170 if (dri2_dpy->image->base.version >= 15 &&
171 dri2_dpy->image->createImageFromDmaBufs2 != NULL) {
172 return dri2_dpy->image->createImageFromDmaBufs2(
173 dri2_dpy->dri_screen, buf_info->width, buf_info->height,
174 buf_info->drm_fourcc, buf_info->modifier, buf_info->fds,
175 buf_info->num_planes, buf_info->pitches, buf_info->offsets,
176 buf_info->yuv_color_space, buf_info->sample_range,
177 buf_info->horizontal_siting, buf_info->vertical_siting, &error,
178 priv);
179 }
180
181 return dri2_dpy->image->createImageFromDmaBufs(
182 dri2_dpy->dri_screen, buf_info->width, buf_info->height,
183 buf_info->drm_fourcc, buf_info->fds, buf_info->num_planes,
184 buf_info->pitches, buf_info->offsets, buf_info->yuv_color_space,
185 buf_info->sample_range, buf_info->horizontal_siting,
186 buf_info->vertical_siting, &error, priv);
187 }
188
189 static __DRIimage *
ohos_create_image_from_native_buffer(_EGLDisplay *disp, struct ANativeWindowBuffer *buf, void *priv)190 ohos_create_image_from_native_buffer(_EGLDisplay *disp,
191 struct ANativeWindowBuffer *buf,
192 void *priv)
193 {
194 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
195 struct buffer_info buf_info;
196 __DRIimage *img = NULL;
197
198 /* If dri driver is gallium virgl, real modifier info queried back from
199 * CrOS info (and potentially mapper metadata if integrated later) cannot
200 * get resolved and the buffer import will fail. Thus the fallback behavior
201 * is preserved down to native_window_buffer_get_buffer_info() so that the
202 * buffer can be imported without modifier info as a last resort.
203 */
204
205 if (!native_window_buffer_get_buffer_info(dri2_dpy, buf, &buf_info)) {
206 img = ohos_create_image_from_buffer_info(dri2_dpy, &buf_info, priv);
207 }
208
209 return img;
210 }
211
212 static EGLBoolean
ohos_window_dequeue_buffer(struct dri2_egl_surface *dri2_surf)213 ohos_window_dequeue_buffer(struct dri2_egl_surface *dri2_surf)
214 {
215 int fence_fd;
216
217 if (ANativeWindow_dequeueBuffer(dri2_surf->window, &dri2_surf->buffer,
218 &fence_fd)) {
219 return EGL_FALSE;
220 }
221
222 /* If access to the buffer is controlled by a sync fence, then block on the
223 * fence.
224 *
225 * It may be more performant to postpone blocking until there is an
226 * immediate need to write to the buffer. But doing so would require adding
227 * hooks to the DRI2 loader.
228 *
229 * From the ANativeWindow_dequeueBuffer documentation:
230 *
231 * The libsync fence file descriptor returned in the int pointed to by
232 * the fenceFd argument will refer to the fence that must signal
233 * before the dequeued buffer may be written to. A value of -1
234 * indicates that the caller may access the buffer immediately without
235 * waiting on a fence. If a valid file descriptor is returned (i.e.
236 * any value except -1) then the caller is responsible for closing the
237 * file descriptor.
238 */
239 if (fence_fd >= 0) {
240 /* From the SYNC_IOC_WAIT documentation in <linux/sync.h>:
241 *
242 * Waits indefinitely if timeout < 0.
243 */
244 int timeout = -1;
245 sync_wait(fence_fd, timeout);
246 close(fence_fd);
247 }
248 /* Record all the buffers created by ANativeWindow and update back buffer
249 * for updating buffer's age in swap_buffers.
250 */
251 EGLBoolean updated = EGL_FALSE;
252 for (int i = 0; i < dri2_surf->color_buffers_count; i++) {
253 if (!dri2_surf->color_buffers[i].buffer) {
254 dri2_surf->color_buffers[i].buffer = dri2_surf->buffer;
255 }
256 if (dri2_surf->color_buffers[i].buffer == dri2_surf->buffer) {
257 dri2_surf->back = &dri2_surf->color_buffers[i];
258 updated = EGL_TRUE;
259 break;
260 }
261 }
262
263 if (!updated) {
264 /* In case of all the buffers were recreated by ANativeWindow, reset
265 * the color_buffers
266 */
267 for (int i = 0; i < dri2_surf->color_buffers_count; i++) {
268 dri2_surf->color_buffers[i].buffer = NULL;
269 dri2_surf->color_buffers[i].age = 0;
270 }
271 dri2_surf->color_buffers[0].buffer = dri2_surf->buffer;
272 dri2_surf->back = &dri2_surf->color_buffers[0];
273 }
274
275 return EGL_TRUE;
276 }
277
278 static EGLBoolean
ohos_window_enqueue_buffer(_EGLDisplay *disp, struct dri2_egl_surface *dri2_surf)279 ohos_window_enqueue_buffer(_EGLDisplay *disp, struct dri2_egl_surface *dri2_surf)
280 {
281 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
282
283 /* To avoid blocking other EGL calls, release the display mutex before
284 * we enter ohos_window_enqueue_buffer() and re-acquire the mutex upon
285 * return.
286 */
287 mtx_unlock(&disp->Mutex);
288
289 /* Queue the buffer with stored out fence fd. The ANativeWindow or buffer
290 * consumer may choose to wait for the fence to signal before accessing
291 * it. If fence fd value is -1, buffer can be accessed by consumer
292 * immediately. Consumer or application shouldn't rely on timestamp
293 * associated with fence if the fence fd is -1.
294 *
295 * Ownership of fd is transferred to consumer after queueBuffer and the
296 * consumer is responsible for closing it. Caller must not use the fd
297 * after passing it to queueBuffer.
298 */
299 int fence_fd = dri2_surf->out_fence_fd;
300 dri2_surf->out_fence_fd = -1;
301 ANativeWindow_queueBuffer(dri2_surf->window, dri2_surf->buffer, fence_fd);
302
303 dri2_surf->buffer = NULL;
304 dri2_surf->back = NULL;
305
306 mtx_lock(&disp->Mutex);
307
308 if (dri2_surf->dri_image_back) {
309 dri2_dpy->image->destroyImage(dri2_surf->dri_image_back);
310 dri2_surf->dri_image_back = NULL;
311 }
312
313 return EGL_TRUE;
314 }
315
316 static void
ohos_window_cancel_buffer(struct dri2_egl_surface *dri2_surf)317 ohos_window_cancel_buffer(struct dri2_egl_surface *dri2_surf)
318 {
319 int ret;
320 int fence_fd = dri2_surf->out_fence_fd;
321
322 dri2_surf->out_fence_fd = -1;
323 ret = ANativeWindow_cancelBuffer(dri2_surf->window, dri2_surf->buffer,
324 fence_fd);
325 dri2_surf->buffer = NULL;
326 if (ret < 0) {
327 _eglLog(_EGL_WARNING, "ANativeWindow_cancelBuffer failed");
328 dri2_surf->base.Lost = EGL_TRUE;
329 }
330 }
331
332 static _EGLSurface *
ohos_create_surface(_EGLDisplay *disp, EGLint type, _EGLConfig *conf, void *native_window, const EGLint *attrib_list)333 ohos_create_surface(_EGLDisplay *disp, EGLint type, _EGLConfig *conf,
334 void *native_window, const EGLint *attrib_list)
335 {
336 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
337 struct dri2_egl_config *dri2_conf = dri2_egl_config(conf);
338 struct dri2_egl_surface *dri2_surf;
339 struct ANativeWindow *window = native_window;
340 const __DRIconfig *config;
341
342 dri2_surf = calloc(1, sizeof *dri2_surf);
343 if (!dri2_surf) {
344 _eglError(EGL_BAD_ALLOC, "ohos_create_surface");
345 return NULL;
346 }
347
348 if (!dri2_init_surface(&dri2_surf->base, disp, type, conf, attrib_list,
349 true, native_window)) {
350 goto cleanup_surface;
351 }
352
353 if (type == EGL_WINDOW_BIT) {
354 int format;
355 int buffer_count;
356
357 format = ANativeWindow_getFormat(window);
358 if (format < 0) {
359 _eglError(EGL_BAD_NATIVE_WINDOW, "ohos_create_surface");
360 goto cleanup_surface;
361 }
362
363 /* Required buffer caching slots. */
364 buffer_count = 3; // default use 3 buffer
365
366 dri2_surf->color_buffers = calloc(buffer_count,
367 sizeof(*dri2_surf->color_buffers));
368 if (!dri2_surf->color_buffers) {
369 _eglError(EGL_BAD_ALLOC, "ohos_create_surface");
370 goto cleanup_surface;
371 }
372 dri2_surf->color_buffers_count = buffer_count;
373
374 if (format != dri2_conf->base.NativeVisualID) {
375 _eglLog(_EGL_WARNING, "Native format mismatch: 0x%{public}x != 0x%{public}x",
376 format, dri2_conf->base.NativeVisualID);
377 }
378
379 NativeWindowHandleOpt(window, GET_BUFFER_GEOMETRY, &dri2_surf->base.Height, &dri2_surf->base.Width);
380 }
381
382 config = dri2_get_dri_config(dri2_conf, type,
383 dri2_surf->base.GLColorspace);
384 if (!config) {
385 _eglError(EGL_BAD_MATCH, "Unsupported surfacetype/colorspace configuration");
386 goto cleanup_surface;
387 }
388
389 if (!dri2_create_drawable(dri2_dpy, config, dri2_surf, dri2_surf)) {
390 goto cleanup_surface;
391 }
392
393 if (window) {
394 ANativeWindow_acquire(window);
395 dri2_surf->window = window;
396 }
397
398 return &dri2_surf->base;
399
400 cleanup_surface:
401 if (dri2_surf->color_buffers_count) {
402 free(dri2_surf->color_buffers);
403 }
404 free(dri2_surf);
405
406 return NULL;
407 }
408
409 static _EGLSurface *
ohos_create_window_surface(_EGLDisplay *disp, _EGLConfig *conf, void *native_window, const EGLint *attrib_list)410 ohos_create_window_surface(_EGLDisplay *disp, _EGLConfig *conf,
411 void *native_window, const EGLint *attrib_list)
412 {
413 return ohos_create_surface(disp, EGL_WINDOW_BIT, conf,
414 native_window, attrib_list);
415 }
416
417 static _EGLSurface *
ohos_create_pbuffer_surface(_EGLDisplay *disp, _EGLConfig *conf, const EGLint *attrib_list)418 ohos_create_pbuffer_surface(_EGLDisplay *disp, _EGLConfig *conf,
419 const EGLint *attrib_list)
420 {
421 return ohos_create_surface(disp, EGL_PBUFFER_BIT, conf,
422 NULL, attrib_list);
423 }
424
425 static EGLBoolean
ohos_destroy_surface(_EGLDisplay *disp, _EGLSurface *surf)426 ohos_destroy_surface(_EGLDisplay *disp, _EGLSurface *surf)
427 {
428 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
429 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surf);
430
431 dri2_egl_surface_free_local_buffers(dri2_surf);
432
433 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
434 if (dri2_surf->buffer) {
435 ohos_window_cancel_buffer(dri2_surf);
436 }
437
438 ANativeWindow_release(dri2_surf->window);
439 }
440
441 if (dri2_surf->dri_image_back) {
442 _eglLog(_EGL_DEBUG, "%{public}s : %{public}d : destroy dri_image_back", __func__, __LINE__);
443 dri2_dpy->image->destroyImage(dri2_surf->dri_image_back);
444 dri2_surf->dri_image_back = NULL;
445 }
446
447 if (dri2_surf->dri_image_front) {
448 _eglLog(_EGL_DEBUG, "%{public}s : %{public}d : destroy dri_image_front", __func__, __LINE__);
449 dri2_dpy->image->destroyImage(dri2_surf->dri_image_front);
450 dri2_surf->dri_image_front = NULL;
451 }
452
453 dri2_dpy->core->destroyDrawable(dri2_surf->dri_drawable);
454
455 dri2_fini_surface(surf);
456 free(dri2_surf->color_buffers);
457 free(dri2_surf);
458
459 return EGL_TRUE;
460 }
461
462 static int
update_buffers(struct dri2_egl_surface *dri2_surf)463 update_buffers(struct dri2_egl_surface *dri2_surf)
464 {
465 if (dri2_surf->base.Lost) {
466 return -1;
467 }
468
469 if (dri2_surf->base.Type != EGL_WINDOW_BIT) {
470 return 0;
471 }
472
473 /* try to dequeue the next back buffer */
474 if (!dri2_surf->buffer && !ohos_window_dequeue_buffer(dri2_surf)) {
475 _eglLog(_EGL_WARNING, "Could not dequeue buffer from native window");
476 dri2_surf->base.Lost = EGL_TRUE;
477 return -1;
478 }
479 BufferHandle *handle = GetBufferHandleFromNative(dri2_surf->buffer);
480 /* free outdated buffers and update the surface size */
481 if (dri2_surf->base.Width != handle->width ||
482 dri2_surf->base.Height != handle->height) {
483 dri2_egl_surface_free_local_buffers(dri2_surf);
484 dri2_surf->base.Width = handle->width;
485 dri2_surf->base.Height = handle->height;
486 }
487
488 return 0;
489 }
490
491 static int
get_front_bo(struct dri2_egl_surface *dri2_surf, unsigned int format)492 get_front_bo(struct dri2_egl_surface *dri2_surf, unsigned int format)
493 {
494 struct dri2_egl_display *dri2_dpy =
495 dri2_egl_display(dri2_surf->base.Resource.Display);
496
497 if (dri2_surf->dri_image_front) {
498 return 0;
499 }
500
501 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
502 /* According current EGL spec, front buffer rendering
503 * for window surface is not supported now.
504 * and mesa doesn't have the implementation of this case.
505 * Add warning message, but not treat it as error.
506 */
507 _eglLog(_EGL_DEBUG, "DRI driver requested unsupported front buffer for window surface");
508 } else if (dri2_surf->base.Type == EGL_PBUFFER_BIT) {
509 dri2_surf->dri_image_front =
510 dri2_dpy->image->createImage(dri2_dpy->dri_screen,
511 dri2_surf->base.Width,
512 dri2_surf->base.Height,
513 format,
514 0,
515 NULL);
516 if (!dri2_surf->dri_image_front) {
517 _eglLog(_EGL_WARNING, "dri2_image_front allocation failed");
518 return -1;
519 }
520 }
521
522 return 0;
523 }
524
525 static int
get_back_bo(struct dri2_egl_surface *dri2_surf)526 get_back_bo(struct dri2_egl_surface *dri2_surf)
527 {
528 _EGLDisplay *disp = dri2_surf->base.Resource.Display;
529
530 if (dri2_surf->dri_image_back) {
531 return 0;
532 }
533
534 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
535 if (!dri2_surf->buffer) {
536 _eglLog(_EGL_WARNING, "Could not get native buffer");
537 return -1;
538 }
539
540 dri2_surf->dri_image_back =
541 ohos_create_image_from_native_buffer(disp, dri2_surf->buffer, NULL);
542 if (!dri2_surf->dri_image_back) {
543 _eglLog(_EGL_WARNING, "failed to create DRI image from FD");
544 return -1;
545 }
546 } else if (dri2_surf->base.Type == EGL_PBUFFER_BIT) {
547 /* The EGL 1.5 spec states that pbuffers are single-buffered. Specifically,
548 * the spec states that they have a back buffer but no front buffer, in
549 * contrast to pixmaps, which have a front buffer but no back buffer.
550 *
551 * Single-buffered surfaces with no front buffer confuse Mesa; so we deviate
552 * from the spec, following the precedent of Mesa's EGL X11 platform. The
553 * X11 platform correctly assigns pbuffers to single-buffered configs, but
554 * assigns the pbuffer a front buffer instead of a back buffer.
555 *
556 * Pbuffers in the X11 platform mostly work today, so let's just copy its
557 * behavior instead of trying to fix (and hence potentially breaking) the
558 * world.
559 */
560 _eglLog(_EGL_DEBUG, "DRI driver requested unsupported back buffer for pbuffer surface");
561 }
562
563 return 0;
564 }
565
566 /* Some drivers will pass multiple bits in buffer_mask.
567 * For such case, will go through all the bits, and
568 * will not return error when unsupported buffer is requested, only
569 * return error when the allocation for supported buffer failed.
570 */
571 static int
ohos_image_get_buffers(__DRIdrawable *driDrawable, unsigned int format, uint32_t *stamp, void *loaderPrivate, uint32_t buffer_mask, struct __DRIimageList *images)572 ohos_image_get_buffers(__DRIdrawable *driDrawable,
573 unsigned int format,
574 uint32_t *stamp,
575 void *loaderPrivate,
576 uint32_t buffer_mask,
577 struct __DRIimageList *images)
578 {
579 struct dri2_egl_surface *dri2_surf = loaderPrivate;
580
581 images->image_mask = 0;
582 images->front = NULL;
583 images->back = NULL;
584
585 if (update_buffers(dri2_surf) < 0) {
586 return 0;
587 }
588
589 if (_eglSurfaceInSharedBufferMode(&dri2_surf->base)) {
590 if (get_back_bo(dri2_surf) < 0) {
591 return 0;
592 }
593
594 /* We have dri_image_back because this is a window surface and
595 * get_back_bo() succeeded.
596 */
597 assert(dri2_surf->dri_image_back);
598 images->back = dri2_surf->dri_image_back;
599 images->image_mask |= __DRI_IMAGE_BUFFER_SHARED;
600
601 /* There exists no accompanying back nor front buffer. */
602 return 1;
603 }
604
605 if (buffer_mask & __DRI_IMAGE_BUFFER_FRONT) {
606 if (get_front_bo(dri2_surf, format) < 0) {
607 return 0;
608 }
609
610 if (dri2_surf->dri_image_front) {
611 images->front = dri2_surf->dri_image_front;
612 images->image_mask |= __DRI_IMAGE_BUFFER_FRONT;
613 }
614 }
615
616 if (buffer_mask & __DRI_IMAGE_BUFFER_BACK) {
617 if (get_back_bo(dri2_surf) < 0) {
618 return 0;
619 }
620
621 if (dri2_surf->dri_image_back) {
622 images->back = dri2_surf->dri_image_back;
623 images->image_mask |= __DRI_IMAGE_BUFFER_BACK;
624 }
625 }
626
627 return 1;
628 }
629
630 static EGLint
ohos_query_buffer_age(_EGLDisplay *disp, _EGLSurface *surface)631 ohos_query_buffer_age(_EGLDisplay *disp, _EGLSurface *surface)
632 {
633 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surface);
634
635 if (update_buffers(dri2_surf) < 0) {
636 _eglError(EGL_BAD_ALLOC, "ohos_query_buffer_age");
637 return -1;
638 }
639
640 return dri2_surf->back ? dri2_surf->back->age : 0;
641 }
642
643 static EGLBoolean
ohos_swap_buffers(_EGLDisplay *disp, _EGLSurface *draw)644 ohos_swap_buffers(_EGLDisplay *disp, _EGLSurface *draw)
645 {
646 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
647 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(draw);
648 const bool has_mutable_rb = _eglSurfaceHasMutableRenderBuffer(draw);
649
650 /* From the EGL_KHR_mutable_render_buffer spec (v12):
651 *
652 * If surface is a single-buffered window, pixmap, or pbuffer surface
653 * for which there is no pending change to the EGL_RENDER_BUFFER
654 * attribute, eglSwapBuffers has no effect.
655 */
656 if (has_mutable_rb &&
657 draw->RequestedRenderBuffer == EGL_SINGLE_BUFFER &&
658 draw->ActiveRenderBuffer == EGL_SINGLE_BUFFER) {
659 _eglLog(_EGL_DEBUG, "%{public}s: remain in shared buffer mode", __func__);
660 return EGL_TRUE;
661 }
662
663 for (int i = 0; i < dri2_surf->color_buffers_count; i++) {
664 if (dri2_surf->color_buffers[i].age > 0) {
665 dri2_surf->color_buffers[i].age++;
666 }
667 }
668
669 /* "XXX: we don't use get_back_bo() since it causes regressions in
670 * several dEQP tests.
671 */
672 if (dri2_surf->back) {
673 dri2_surf->back->age = 1;
674 }
675
676 dri2_flush_drawable_for_swapbuffers(disp, draw);
677
678 /* dri2_surf->buffer can be null even when no error has occured. For
679 * example, if the user has called no GL rendering commands since the
680 * previous eglSwapBuffers, then the driver may have not triggered
681 * a callback to ANativeWindow_dequeueBuffer, in which case
682 * dri2_surf->buffer remains null.
683 */
684 if (dri2_surf->buffer) {
685 ohos_window_enqueue_buffer(disp, dri2_surf);
686 }
687
688 dri2_dpy->flush->invalidate(dri2_surf->dri_drawable);
689
690 return EGL_TRUE;
691 }
692
693 static EGLBoolean
ohos_query_surface(_EGLDisplay *disp, _EGLSurface *surf, EGLint attribute, EGLint *value)694 ohos_query_surface(_EGLDisplay *disp, _EGLSurface *surf,
695 EGLint attribute, EGLint *value)
696 {
697 struct dri2_egl_surface *dri2_surf = dri2_egl_surface(surf);
698 EGLint dummy;
699 switch (attribute) {
700 case EGL_WIDTH:
701 if (dri2_surf->base.Type == EGL_WINDOW_BIT && dri2_surf->window) {
702 NativeWindowHandleOpt(dri2_surf->window,
703 GET_BUFFER_GEOMETRY, &dummy, value);
704 return EGL_TRUE;
705 }
706 break;
707 case EGL_HEIGHT:
708 if (dri2_surf->base.Type == EGL_WINDOW_BIT && dri2_surf->window) {
709 NativeWindowHandleOpt(dri2_surf->window,
710 GET_BUFFER_GEOMETRY, value, &dummy);
711 return EGL_TRUE;
712 }
713 break;
714 default:
715 break;
716 }
717 return _eglQuerySurface(disp, surf, attribute, value);
718 }
719
720 static _EGLImage *
dri2_create_image_ohos_native_buffer(_EGLDisplay *disp, _EGLContext *ctx, struct ANativeWindowBuffer *buf)721 dri2_create_image_ohos_native_buffer(_EGLDisplay *disp,
722 _EGLContext *ctx,
723 struct ANativeWindowBuffer *buf)
724 {
725 if (ctx != NULL) {
726 /* From the EGL_OHOS_image_native_buffer spec:
727 *
728 * * If <target> is EGL_NATIVE_BUFFER_OHOS and <ctx> is not
729 * EGL_NO_CONTEXT, the error EGL_BAD_CONTEXT is generated.
730 */
731 _eglError(EGL_BAD_CONTEXT, "eglCreateEGLImageKHR: for "
732 "EGL_NATIVE_BUFFER_OHOS, the context must be "
733 "EGL_NO_CONTEXT");
734 return NULL;
735 }
736 __DRIimage *dri_image =
737 ohos_create_image_from_native_buffer(disp, buf, buf);
738
739 if (dri_image) {
740 return dri2_create_image_from_dri(disp, dri_image);
741 }
742
743 return NULL;
744 }
745
746 static _EGLImage *
ohos_create_image_khr(_EGLDisplay *disp, _EGLContext *ctx, EGLenum target, EGLClientBuffer buffer, const EGLint *attr_list)747 ohos_create_image_khr(_EGLDisplay *disp, _EGLContext *ctx, EGLenum target,
748 EGLClientBuffer buffer, const EGLint *attr_list)
749 {
750 switch (target) {
751 case EGL_NATIVE_BUFFER_OHOS:
752 return dri2_create_image_ohos_native_buffer(disp, ctx,
753 (struct ANativeWindowBuffer *)buffer);
754 default:
755 return dri2_create_image_khr(disp, ctx, target, buffer, attr_list);
756 }
757 }
758
759 static void
ohos_flush_front_buffer(__DRIdrawable *driDrawable, void *loaderPrivate)760 ohos_flush_front_buffer(__DRIdrawable *driDrawable, void *loaderPrivate)
761 {
762 }
763
764 #ifdef HAVE_DRM_GRALLOC
765 static int
ohos_get_buffers_parse_attachments(struct dri2_egl_surface *dri2_surf, unsigned int *attachments, int count)766 ohos_get_buffers_parse_attachments(struct dri2_egl_surface *dri2_surf,
767 unsigned int *attachments, int count)
768 {
769 int num_buffers = 0;
770
771 /* fill dri2_surf->buffers */
772 for (int i = 0; i < count * 2; i += 2) {
773 __DRIbuffer *buf, *local;
774
775 assert(num_buffers < ARRAY_SIZE(dri2_surf->buffers));
776 buf = &dri2_surf->buffers[num_buffers];
777
778 switch (attachments[i]) {
779 case __DRI_BUFFER_BACK_LEFT:
780 if (dri2_surf->base.Type == EGL_WINDOW_BIT) {
781 buf->attachment = attachments[i];
782 buf->name = get_native_buffer_name(dri2_surf->buffer);
783 buf->cpp = get_format_bpp(dri2_surf->buffer->format);
784 buf->pitch = dri2_surf->buffer->stride * buf->cpp;
785 buf->flags = 0;
786
787 buf->name ? num_buffers++ : 0;
788
789 break;
790 }
791 FALLTHROUGH; /* for pbuffers */
792 case __DRI_BUFFER_DEPTH:
793 case __DRI_BUFFER_STENCIL:
794 case __DRI_BUFFER_ACCUM:
795 case __DRI_BUFFER_DEPTH_STENCIL:
796 case __DRI_BUFFER_HIZ:
797 local = dri2_egl_surface_alloc_local_buffer(dri2_surf,
798 attachments[i], attachments[i + 1]);
799
800 if (local) {
801 *buf = *local;
802 num_buffers++;
803 }
804 break;
805 case __DRI_BUFFER_FRONT_LEFT:
806 case __DRI_BUFFER_FRONT_RIGHT:
807 case __DRI_BUFFER_FAKE_FRONT_LEFT:
808 case __DRI_BUFFER_FAKE_FRONT_RIGHT:
809 case __DRI_BUFFER_BACK_RIGHT:
810 default:
811 /* no front or right buffers */
812 break;
813 }
814 }
815
816 return num_buffers;
817 }
818
819 static __DRIbuffer *
ohos_get_buffers_with_format(__DRIdrawable *driDrawable, int *width, int *height, unsigned int *attachments, int count, int *out_count, void *loaderPrivate)820 ohos_get_buffers_with_format(__DRIdrawable *driDrawable,
821 int *width, int *height,
822 unsigned int *attachments, int count,
823 int *out_count, void *loaderPrivate)
824 {
825 struct dri2_egl_surface *dri2_surf = loaderPrivate;
826
827 if (update_buffers(dri2_surf) < 0) {
828 return NULL;
829 }
830
831 *out_count = ohos_get_buffers_parse_attachments(dri2_surf, attachments, count);
832
833 if (width) {
834 *width = dri2_surf->base.Width;
835 }
836 if (height) {
837 *height = dri2_surf->base.Height;
838 }
839
840 return dri2_surf->buffers;
841 }
842 #endif /* HAVE_DRM_GRALLOC */
843
844 static unsigned
ohos_get_capability(void *loaderPrivate, enum dri_loader_cap cap)845 ohos_get_capability(void *loaderPrivate, enum dri_loader_cap cap)
846 {
847 /* Note: loaderPrivate is _EGLDisplay* */
848 switch (cap) {
849 case DRI_LOADER_CAP_RGBA_ORDERING:
850 return 1;
851 default:
852 return 0;
853 }
854 }
855
856 static void
ohos_destroy_loader_image_state(void *loaderPrivate)857 ohos_destroy_loader_image_state(void *loaderPrivate)
858 {
859 }
860
861 static EGLBoolean
ohos_add_configs_for_visuals(_EGLDisplay *disp)862 ohos_add_configs_for_visuals(_EGLDisplay *disp)
863 {
864 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
865 static const struct {
866 int format;
867 int rgba_shifts[4];
868 unsigned int rgba_sizes[4];
869 } visuals[] = {
870 {PIXEL_FMT_RGBA_8888, {0, 8, 16, 24}, {8, 8, 8, 8}},
871 {PIXEL_FMT_RGBX_8888, {0, 8, 16, -1}, {8, 8, 8, 0}},
872 {PIXEL_FMT_RGB_565, {11, 5, 0, -1}, {5, 6, 5, 0}},
873 /* This must be after PIXEL_FMT_RGBA_8888, we only keep BGRA
874 * visual if it turns out RGBA visual is not available.
875 */
876 {PIXEL_FMT_BGRA_8888, {16, 8, 0, 24}, {8, 8, 8, 8}},
877 };
878
879 unsigned int format_count[ARRAY_SIZE(visuals)] = {0};
880 int config_count = 0;
881
882 bool has_rgba = false;
883 for (int i = 0; i < ARRAY_SIZE(visuals); i++) {
884 if (visuals[i].format == PIXEL_FMT_BGRA_8888 && has_rgba) {
885 continue;
886 }
887 for (int j = 0; dri2_dpy->driver_configs[j]; j++) {
888 const EGLint surface_type = EGL_WINDOW_BIT | EGL_PBUFFER_BIT;
889
890 const EGLint config_attrs[] = {
891 EGL_NATIVE_VISUAL_ID, visuals[i].format,
892 EGL_NATIVE_VISUAL_TYPE, visuals[i].format,
893 EGL_NONE};
894
895 struct dri2_egl_config *dri2_conf =
896 dri2_add_config(disp, dri2_dpy->driver_configs[j],
897 config_count + 1, surface_type, config_attrs,
898 visuals[i].rgba_shifts, visuals[i].rgba_sizes);
899 if (dri2_conf) {
900 (dri2_conf->base.ConfigID == config_count + 1) ? config_count++ : 0;
901 format_count[i]++;
902 }
903 }
904 if (visuals[i].format == PIXEL_FMT_RGBA_8888 && format_count[i]) {
905 has_rgba = true;
906 }
907 }
908
909 for (int i = 0; i < ARRAY_SIZE(format_count); i++) {
910 if (!format_count[i]) {
911 _eglLog(_EGL_DEBUG, "No DRI config supports native format 0x%{public}x",
912 visuals[i].format);
913 }
914 }
915
916 return (config_count != 0);
917 }
918
919 static const struct dri2_egl_display_vtbl ohos_display_vtbl = {
920 .authenticate = NULL,
921 .create_window_surface = ohos_create_window_surface,
922 .create_pbuffer_surface = ohos_create_pbuffer_surface,
923 .destroy_surface = ohos_destroy_surface,
924 .create_image = ohos_create_image_khr,
925 .swap_buffers = ohos_swap_buffers,
926 .swap_interval = NULL,
927 .query_buffer_age = ohos_query_buffer_age,
928 .query_surface = ohos_query_surface,
929 .get_dri_drawable = dri2_surface_get_dri_drawable,
930 };
931
932 static const __DRIimageLoaderExtension ohos_image_loader_extension = {
933 .base = {__DRI_IMAGE_LOADER, 4},
934
935 .getBuffers = ohos_image_get_buffers,
936 .flushFrontBuffer = ohos_flush_front_buffer,
937 .getCapability = ohos_get_capability,
938 .flushSwapBuffers = NULL,
939 .destroyLoaderImageState = ohos_destroy_loader_image_state,
940 };
941
942 static void
ohos_display_shared_buffer(__DRIdrawable *driDrawable, int fence_fd, void *loaderPrivate)943 ohos_display_shared_buffer(__DRIdrawable *driDrawable, int fence_fd,
944 void *loaderPrivate)
945 {
946 struct dri2_egl_surface *dri2_surf = loaderPrivate;
947 struct ANativeWindowBuffer *old_buffer UNUSED = dri2_surf->buffer;
948
949 if (!_eglSurfaceInSharedBufferMode(&dri2_surf->base)) {
950 _eglLog(_EGL_WARNING, "%{public}s: internal error: buffer is not shared",
951 __func__);
952 return;
953 }
954
955 if (fence_fd >= 0) {
956 /* The driver's fence is more recent than the surface's out fence, if it
957 * exists at all. So use the driver's fence.
958 */
959 if (dri2_surf->out_fence_fd >= 0) {
960 close(dri2_surf->out_fence_fd);
961 dri2_surf->out_fence_fd = -1;
962 }
963 } else if (dri2_surf->out_fence_fd >= 0) {
964 fence_fd = dri2_surf->out_fence_fd;
965 dri2_surf->out_fence_fd = -1;
966 }
967
968 if (ANativeWindow_queueBuffer(dri2_surf->window, dri2_surf->buffer,
969 fence_fd)) {
970 _eglLog(_EGL_WARNING, "%{public}s: ANativeWindow_queueBuffer failed", __func__);
971 close(fence_fd);
972 return;
973 }
974
975 fence_fd = -1;
976
977 if (ANativeWindow_dequeueBuffer(dri2_surf->window, &dri2_surf->buffer,
978 &fence_fd)) {
979 /* Tear down the surface because it no longer has a back buffer. */
980 struct dri2_egl_display *dri2_dpy =
981 dri2_egl_display(dri2_surf->base.Resource.Display);
982
983 _eglLog(_EGL_WARNING, "%{public}s: ANativeWindow_dequeueBuffer failed", __func__);
984
985 dri2_surf->base.Lost = true;
986 dri2_surf->buffer = NULL;
987 dri2_surf->back = NULL;
988
989 if (dri2_surf->dri_image_back) {
990 dri2_dpy->image->destroyImage(dri2_surf->dri_image_back);
991 dri2_surf->dri_image_back = NULL;
992 }
993
994 dri2_dpy->flush->invalidate(dri2_surf->dri_drawable);
995 return;
996 }
997
998 if (fence_fd < 0) {
999 return;
1000 }
1001
1002 /* Access to the buffer is controlled by a sync fence. Block on it.
1003 *
1004 * Ideally, we would submit the fence to the driver, and the driver would
1005 * postpone command execution until it signalled. But DRI lacks API for
1006 * that (as of 2018-04-11).
1007 *
1008 * SYNC_IOC_WAIT waits forever if timeout < 0
1009 */
1010 sync_wait(fence_fd, -1);
1011 close(fence_fd);
1012 }
1013
1014 static const __DRImutableRenderBufferLoaderExtension ohos_mutable_render_buffer_extension = {
1015 .base = {__DRI_MUTABLE_RENDER_BUFFER_LOADER, 1},
1016 .displaySharedBuffer = ohos_display_shared_buffer,
1017 };
1018
1019 static const __DRIextension *ohos_image_loader_extensions[] = {
1020 &ohos_image_loader_extension.base,
1021 &image_lookup_extension.base,
1022 &use_invalidate.base,
1023 &ohos_mutable_render_buffer_extension.base,
1024 NULL,
1025 };
1026
1027 static EGLBoolean
ohos_load_driver(_EGLDisplay *disp, bool swrast)1028 ohos_load_driver(_EGLDisplay *disp, bool swrast)
1029 {
1030 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
1031
1032 dri2_dpy->driver_name = loader_get_driver_for_fd(dri2_dpy->fd);
1033 if (dri2_dpy->driver_name == NULL) {
1034 return false;
1035 }
1036
1037 #ifdef HAVE_DRM_GRALLOC
1038 /* Handle control nodes using __DRI_DRI2_LOADER extension and GEM names
1039 * for backwards compatibility with drm_gralloc. (Do not use on new
1040 * systems.) */
1041 dri2_dpy->loader_extensions = ohos_dri2_loader_extensions;
1042 if (!dri2_load_driver(disp)) {
1043 goto error;
1044 }
1045 #else
1046 if (swrast) {
1047 /* Use kms swrast only with vgem / virtio_gpu.
1048 * virtio-gpu fallbacks to software rendering when 3D features
1049 * are unavailable since 6c5ab.
1050 */
1051 if (strcmp(dri2_dpy->driver_name, "vgem") == 0 ||
1052 strcmp(dri2_dpy->driver_name, "virtio_gpu") == 0) {
1053 free(dri2_dpy->driver_name);
1054 dri2_dpy->driver_name = strdup("kms_swrast");
1055 } else {
1056 goto error;
1057 }
1058 }
1059
1060 dri2_dpy->loader_extensions = ohos_image_loader_extensions;
1061 if (!dri2_load_driver_dri3(disp)) {
1062 goto error;
1063 }
1064 #endif
1065
1066 return true;
1067
1068 error:
1069 free(dri2_dpy->driver_name);
1070 dri2_dpy->driver_name = NULL;
1071 return false;
1072 }
1073
1074 static void
ohos_unload_driver(_EGLDisplay *disp)1075 ohos_unload_driver(_EGLDisplay *disp)
1076 {
1077 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
1078
1079 dlclose(dri2_dpy->driver);
1080 dri2_dpy->driver = NULL;
1081 free(dri2_dpy->driver_name);
1082 dri2_dpy->driver_name = NULL;
1083 }
1084
1085 static int
ohos_filter_device(_EGLDisplay *disp, int fd, const char *vendor)1086 ohos_filter_device(_EGLDisplay *disp, int fd, const char *vendor)
1087 {
1088 drmVersionPtr ver = drmGetVersion(fd);
1089 if (!ver) {
1090 return -1;
1091 }
1092
1093 if (strcmp(vendor, ver->name) != 0) {
1094 drmFreeVersion(ver);
1095 return -1;
1096 }
1097
1098 drmFreeVersion(ver);
1099 return 0;
1100 }
1101
1102 static EGLBoolean
ohos_probe_device(_EGLDisplay *disp, bool swrast)1103 ohos_probe_device(_EGLDisplay *disp, bool swrast)
1104 {
1105 /* Check that the device is supported, by attempting to:
1106 * - load the dri module
1107 * - and, create a screen
1108 */
1109 if (!ohos_load_driver(disp, swrast)) {
1110 return EGL_FALSE;
1111 }
1112
1113 if (!dri2_create_screen(disp)) {
1114 _eglLog(_EGL_WARNING, "DRI2: failed to create screen");
1115 ohos_unload_driver(disp);
1116 return EGL_FALSE;
1117 }
1118 return EGL_TRUE;
1119 }
1120
1121 #ifdef HAVE_DRM_GRALLOC
1122 static EGLBoolean
ohos_open_device(_EGLDisplay *disp, bool swrast)1123 ohos_open_device(_EGLDisplay *disp, bool swrast)
1124 {
1125 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
1126 int fd = -1, err = -EINVAL;
1127
1128 if (swrast) {
1129 return EGL_FALSE;
1130 }
1131
1132 if (dri2_dpy->gralloc->perform) {
1133 err = dri2_dpy->gralloc->perform(dri2_dpy->gralloc,
1134 GRALLOC_MODULE_PERFORM_GET_DRM_FD,
1135 &fd);
1136 }
1137 if (err || fd < 0) {
1138 _eglLog(_EGL_WARNING, "fail to get drm fd");
1139 return EGL_FALSE;
1140 }
1141
1142 dri2_dpy->fd = os_dupfd_cloexec(fd);
1143 if (dri2_dpy->fd < 0) {
1144 return EGL_FALSE;
1145 }
1146
1147 if (drmGetNodeTypeFromFd(dri2_dpy->fd) == DRM_NODE_RENDER) {
1148 return EGL_FALSE;
1149 }
1150
1151 return ohos_probe_device(disp, swrast);
1152 }
1153 #else
1154 static EGLBoolean
ohos_open_device(_EGLDisplay *disp, bool swrast)1155 ohos_open_device(_EGLDisplay *disp, bool swrast)
1156 {
1157 #define MAX_DRM_DEVICES 64
1158 struct dri2_egl_display *dri2_dpy = dri2_egl_display(disp);
1159 drmDevicePtr device, devices[MAX_DRM_DEVICES] = {NULL};
1160 int num_devices;
1161
1162 char *vendor_name = NULL;
1163
1164 #ifdef EGL_FORCE_RENDERNODE
1165 const unsigned node_type = DRM_NODE_RENDER;
1166 #else
1167 const unsigned node_type = swrast ? DRM_NODE_PRIMARY : DRM_NODE_RENDER;
1168 #endif
1169
1170 num_devices = drmGetDevices2(0, devices, ARRAY_SIZE(devices));
1171 _eglLog(_EGL_WARNING, "ohos_open_device %{public}d", num_devices);
1172 if (num_devices < 0) {
1173 return EGL_FALSE;
1174 }
1175
1176 for (int i = 0; i < num_devices; i++) {
1177 device = devices[i];
1178
1179 if (!(device->available_nodes & (1 << node_type))) {
1180 continue;
1181 }
1182
1183 dri2_dpy->fd = loader_open_device(device->nodes[node_type]);
1184 if (dri2_dpy->fd < 0) {
1185 DISPLAY_LOGI();
1186 _eglLog(_EGL_WARNING, "%{public}s() Failed to open DRM device %{public}s",
1187 __func__, device->nodes[node_type]);
1188 continue;
1189 }
1190
1191 /* If a vendor is explicitly provided, we use only that.
1192 * Otherwise we fall-back the first device that is supported.
1193 */
1194 DISPLAY_LOGI("vendor_name %{public}s", vendor_name);
1195 if (vendor_name) {
1196 if (ohos_filter_device(disp, dri2_dpy->fd, vendor_name)) {
1197 /* Device does not match - try next device */
1198 close(dri2_dpy->fd);
1199 dri2_dpy->fd = -1;
1200 DISPLAY_LOGI();
1201 continue;
1202 }
1203 /* If the requested device matches - use it. Regardless if
1204 * init fails, do not fall-back to any other device.
1205 */
1206 if (!ohos_probe_device(disp, false)) {
1207 close(dri2_dpy->fd);
1208 dri2_dpy->fd = -1;
1209 }
1210 DISPLAY_LOGI();
1211 break;
1212 }
1213 if (ohos_probe_device(disp, swrast)) {
1214 break;
1215 }
1216 DISPLAY_LOGI();
1217 /* No explicit request - attempt the next device */
1218 close(dri2_dpy->fd);
1219 dri2_dpy->fd = -1;
1220 }
1221 drmFreeDevices(devices, num_devices);
1222 DISPLAY_LOGI();
1223 if (dri2_dpy->fd < 0) {
1224 DISPLAY_LOGI();
1225 _eglLog(_EGL_WARNING, "Failed to open %{public}s DRM device",
1226 vendor_name ? "desired" : "any");
1227 return EGL_FALSE;
1228 }
1229
1230 return EGL_TRUE;
1231 #undef MAX_DRM_DEVICES
1232 }
1233
1234 #endif
1235
1236 EGLBoolean
dri2_initialize_ohos(_EGLDisplay *disp)1237 dri2_initialize_ohos(_EGLDisplay *disp)
1238 {
1239 _EGLDevice *dev;
1240 bool device_opened = false;
1241 struct dri2_egl_display *dri2_dpy;
1242 const char *err;
1243
1244 dri2_dpy = calloc(1, sizeof(*dri2_dpy));
1245 if (!dri2_dpy)
1246 return _eglError(EGL_BAD_ALLOC, "eglInitialize");
1247
1248 dri2_dpy->fd = -1;
1249
1250 disp->DriverData = (void *)dri2_dpy;
1251 device_opened = ohos_open_device(disp, disp->Options.ForceSoftware);
1252
1253 if (!device_opened) {
1254 err = "DRI2: failed to open device";
1255 goto cleanup;
1256 }
1257
1258 dev = _eglAddDevice(dri2_dpy->fd, false);
1259 if (!dev) {
1260 err = "DRI2: failed to find EGLDevice";
1261 goto cleanup;
1262 }
1263
1264 disp->Device = dev;
1265
1266 if (!dri2_setup_extensions(disp)) {
1267 err = "DRI2: failed to setup extensions";
1268 goto cleanup;
1269 }
1270
1271 dri2_setup_screen(disp);
1272
1273 dri2_setup_swap_interval(disp, 1);
1274
1275 disp->Extensions.KHR_image = EGL_TRUE;
1276
1277 /* Create configs *after* enabling extensions because presence of DRI
1278 * driver extensions can affect the capabilities of EGLConfigs.
1279 */
1280 if (!ohos_add_configs_for_visuals(disp)) {
1281 err = "DRI2: failed to add configs";
1282 goto cleanup;
1283 }
1284
1285 /* Fill vtbl last to prevent accidentally calling virtual function during
1286 * initialization.
1287 */
1288 dri2_dpy->vtbl = &ohos_display_vtbl;
1289
1290 return EGL_TRUE;
1291
1292 cleanup:
1293 dri2_display_destroy(disp);
1294 return _eglError(EGL_NOT_INITIALIZED, err);
1295 }
1296