mirror of
https://gitlab.gnome.org/GNOME/libmks.git
synced 2024-12-22 13:45:21 +00:00
2e842f7e7f
This just starts on the DMA-BUF code and abstracts MksPaintable so it can encapsulate both MksDmabufPaintable and MksCairoFramebuffer. Currently, the dmabuf just tailes everything (fully) but we can fix that with snapshot work stil. Either way, want to get the abstraction landed first before we dive deeper into that.
175 lines
5.2 KiB
C
175 lines
5.2 KiB
C
/*
|
|
* Copyright 2021 Red Hat, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* SPDX-License-Identifier: LGPL-2.1-or-later
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <gdk/gdk.h>
|
|
|
|
#ifdef GDK_WINDOWING_WAYLAND
|
|
# include <gdk/wayland/gdkwayland.h>
|
|
#endif
|
|
|
|
#ifdef GDK_WINDOWING_X11
|
|
# include <gdk/x11/gdkx.h>
|
|
#endif
|
|
|
|
#include "mks-gl-context-private.h"
|
|
|
|
GLuint
|
|
mks_gl_context_import_dmabuf (GdkGLContext *context,
|
|
uint32_t format,
|
|
unsigned int width,
|
|
unsigned int height,
|
|
uint32_t n_planes,
|
|
const int *fds,
|
|
const uint32_t *strides,
|
|
const uint32_t *offsets,
|
|
const uint64_t *modifiers)
|
|
{
|
|
GdkDisplay *display;
|
|
EGLDisplay egl_display;
|
|
EGLint attribs[2 * (3 + 4 * 5) + 1];
|
|
EGLImage image;
|
|
guint texture_id;
|
|
int i;
|
|
|
|
g_return_val_if_fail (GDK_IS_GL_CONTEXT (context), 0);
|
|
g_return_val_if_fail (0 < n_planes && n_planes <= 4, 0);
|
|
|
|
display = gdk_gl_context_get_display (context);
|
|
|
|
#ifdef GDK_WINDOWING_WAYLAND
|
|
if (GDK_IS_WAYLAND_DISPLAY (display))
|
|
egl_display = gdk_wayland_display_get_egl_display (display);
|
|
else
|
|
#endif
|
|
#ifdef GDK_WINDOWING_X11
|
|
if (GDK_IS_X11_DISPLAY (display))
|
|
egl_display = gdk_x11_display_get_egl_display (display);
|
|
else
|
|
#endif
|
|
egl_display = NULL;
|
|
|
|
if (egl_display == NULL)
|
|
{
|
|
g_warning ("Can't import dmabufs when not using EGL");
|
|
return 0;
|
|
}
|
|
|
|
i = 0;
|
|
attribs[i++] = EGL_WIDTH;
|
|
attribs[i++] = width;
|
|
attribs[i++] = EGL_HEIGHT;
|
|
attribs[i++] = height;
|
|
attribs[i++] = EGL_LINUX_DRM_FOURCC_EXT;
|
|
attribs[i++] = format;
|
|
|
|
if (n_planes > 0)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE0_FD_EXT;
|
|
attribs[i++] = fds[0];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE0_OFFSET_EXT;
|
|
attribs[i++] = offsets[0];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE0_PITCH_EXT;
|
|
attribs[i++] = strides[0];
|
|
if (modifiers)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE0_MODIFIER_LO_EXT;
|
|
attribs[i++] = modifiers[0] & 0xFFFFFFFF;
|
|
attribs[i++] = EGL_DMA_BUF_PLANE0_MODIFIER_HI_EXT;
|
|
attribs[i++] = modifiers[0] >> 32;
|
|
}
|
|
}
|
|
|
|
if (n_planes > 1)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE1_FD_EXT;
|
|
attribs[i++] = fds[1];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE1_OFFSET_EXT;
|
|
attribs[i++] = offsets[1];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE1_PITCH_EXT;
|
|
attribs[i++] = strides[1];
|
|
if (modifiers)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE1_MODIFIER_LO_EXT;
|
|
attribs[i++] = modifiers[1] & 0xFFFFFFFF;
|
|
attribs[i++] = EGL_DMA_BUF_PLANE1_MODIFIER_HI_EXT;
|
|
attribs[i++] = modifiers[1] >> 32;
|
|
}
|
|
}
|
|
|
|
if (n_planes > 2)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE2_FD_EXT;
|
|
attribs[i++] = fds[2];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE2_OFFSET_EXT;
|
|
attribs[i++] = offsets[2];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE2_PITCH_EXT;
|
|
attribs[i++] = strides[2];
|
|
if (modifiers)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE2_MODIFIER_LO_EXT;
|
|
attribs[i++] = modifiers[2] & 0xFFFFFFFF;
|
|
attribs[i++] = EGL_DMA_BUF_PLANE2_MODIFIER_HI_EXT;
|
|
attribs[i++] = modifiers[2] >> 32;
|
|
}
|
|
}
|
|
if (n_planes > 3)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE3_FD_EXT;
|
|
attribs[i++] = fds[3];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE3_OFFSET_EXT;
|
|
attribs[i++] = offsets[3];
|
|
attribs[i++] = EGL_DMA_BUF_PLANE3_PITCH_EXT;
|
|
attribs[i++] = strides[3];
|
|
if (modifiers)
|
|
{
|
|
attribs[i++] = EGL_DMA_BUF_PLANE3_MODIFIER_LO_EXT;
|
|
attribs[i++] = modifiers[3] & 0xFFFFFFFF;
|
|
attribs[i++] = EGL_DMA_BUF_PLANE3_MODIFIER_HI_EXT;
|
|
attribs[i++] = modifiers[3] >> 32;
|
|
}
|
|
}
|
|
|
|
attribs[i++] = EGL_NONE;
|
|
|
|
image = eglCreateImageKHR (egl_display,
|
|
EGL_NO_CONTEXT,
|
|
EGL_LINUX_DMA_BUF_EXT,
|
|
(EGLClientBuffer)NULL,
|
|
attribs);
|
|
if (image == EGL_NO_IMAGE)
|
|
{
|
|
g_warning ("Failed to create EGL image: %d\n", eglGetError ());
|
|
return 0;
|
|
}
|
|
|
|
gdk_gl_context_make_current (context);
|
|
|
|
glGenTextures (1, &texture_id);
|
|
glBindTexture (GL_TEXTURE_2D, texture_id);
|
|
glEGLImageTargetTexture2DOES (GL_TEXTURE_2D, image);
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
|
glTexParameteri (GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
|
|
|
eglDestroyImageKHR (egl_display, image);
|
|
|
|
return texture_id;
|
|
}
|