2016-05-07 15:07:46 +00:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
2016-05-09 11:51:29 +00:00
|
|
|
* Copyright (C) 2011 Intel Corporation.
|
2016-05-07 15:07:46 +00:00
|
|
|
* Copyright (C) 2016 Red Hat
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person
|
|
|
|
* obtaining a copy of this software and associated documentation
|
|
|
|
* files (the "Software"), to deal in the Software without
|
|
|
|
* restriction, including without limitation the rights to use, copy,
|
|
|
|
* modify, merge, publish, distribute, sublicense, and/or sell copies
|
|
|
|
* of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be
|
|
|
|
* included in all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
|
|
|
|
* BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
|
|
|
|
* ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
|
|
* SOFTWARE.
|
|
|
|
*
|
2016-05-09 11:51:29 +00:00
|
|
|
* Authors:
|
|
|
|
* Rob Bradford <rob@linux.intel.com> (from cogl-winsys-egl-kms.c)
|
|
|
|
* Kristian Høgsberg (from eglkms.c)
|
|
|
|
* Benjamin Franzke (from eglkms.c)
|
|
|
|
* Robert Bragg <robert@linux.intel.com> (from cogl-winsys-egl-kms.c)
|
|
|
|
* Neil Roberts <neil@linux.intel.com> (from cogl-winsys-egl-kms.c)
|
|
|
|
* Jonas Ådahl <jadahl@redhat.com>
|
|
|
|
*
|
2016-05-07 15:07:46 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2016-05-09 11:51:29 +00:00
|
|
|
#include <errno.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <gbm.h>
|
2016-05-09 13:22:01 +00:00
|
|
|
#include <gio/gio.h>
|
|
|
|
#include <glib-object.h>
|
2016-05-09 11:51:29 +00:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <xf86drm.h>
|
2016-05-07 15:07:46 +00:00
|
|
|
|
2016-05-09 11:51:29 +00:00
|
|
|
#include "backends/meta-backend-private.h"
|
2016-05-11 08:26:34 +00:00
|
|
|
#include "backends/native/meta-monitor-manager-kms.h"
|
2016-05-07 15:07:46 +00:00
|
|
|
#include "backends/native/meta-renderer-native.h"
|
2016-05-07 15:09:59 +00:00
|
|
|
#include "cogl/cogl.h"
|
|
|
|
|
|
|
|
enum
|
|
|
|
{
|
|
|
|
PROP_0,
|
|
|
|
|
|
|
|
PROP_KMS_FD,
|
|
|
|
|
|
|
|
PROP_LAST
|
|
|
|
};
|
2016-05-07 15:07:46 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
typedef struct _MetaOnscreenNative
|
|
|
|
{
|
|
|
|
struct gbm_surface *surface;
|
|
|
|
uint32_t current_fb_id;
|
|
|
|
uint32_t next_fb_id;
|
|
|
|
struct gbm_bo *current_bo;
|
|
|
|
struct gbm_bo *next_bo;
|
|
|
|
gboolean pending_swap_notify;
|
|
|
|
|
|
|
|
EGLSurface *pending_egl_surface;
|
|
|
|
struct gbm_surface *pending_surface;
|
|
|
|
} MetaOnscreenNative;
|
|
|
|
|
2016-05-07 15:07:46 +00:00
|
|
|
struct _MetaRendererNative
|
|
|
|
{
|
|
|
|
MetaRenderer parent;
|
2016-05-07 15:09:59 +00:00
|
|
|
|
|
|
|
int kms_fd;
|
2016-05-09 13:22:01 +00:00
|
|
|
struct gbm_device *gbm;
|
|
|
|
CoglClosure *swap_notify_idle;
|
2016-05-10 02:16:45 +00:00
|
|
|
|
|
|
|
int width, height;
|
2016-05-10 07:10:23 +00:00
|
|
|
gboolean pending_set_crtc;
|
2016-05-10 02:16:45 +00:00
|
|
|
struct gbm_surface *dummy_gbm_surface;
|
|
|
|
|
|
|
|
CoglOnscreen *onscreen;
|
2016-05-07 15:07:46 +00:00
|
|
|
};
|
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
static void
|
|
|
|
initable_iface_init (GInitableIface *initable_iface);
|
|
|
|
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (MetaRendererNative,
|
|
|
|
meta_renderer_native,
|
|
|
|
META_TYPE_RENDERER,
|
|
|
|
G_IMPLEMENT_INTERFACE (G_TYPE_INITABLE,
|
|
|
|
initable_iface_init))
|
2016-05-07 15:07:46 +00:00
|
|
|
|
2016-05-09 11:51:29 +00:00
|
|
|
static const CoglWinsysEGLVtable _cogl_winsys_egl_vtable;
|
|
|
|
static const CoglWinsysVtable *parent_vtable;
|
|
|
|
|
|
|
|
static void
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_disconnect (CoglRenderer *renderer)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
|
|
|
|
|
|
|
if (egl_renderer->edpy != EGL_NO_DISPLAY)
|
|
|
|
eglTerminate (egl_renderer->edpy);
|
|
|
|
|
|
|
|
g_slice_free (CoglRendererEGL, egl_renderer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
flush_pending_swap_notify_cb (void *data,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = data;
|
|
|
|
|
|
|
|
if (framebuffer->type == COGL_FRAMEBUFFER_TYPE_ONSCREEN)
|
|
|
|
{
|
|
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->pending_swap_notify)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
CoglFrameInfo *info =
|
|
|
|
g_queue_pop_head (&onscreen->pending_frame_infos);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
_cogl_onscreen_notify_frame_sync (onscreen, info);
|
|
|
|
_cogl_onscreen_notify_complete (onscreen, info);
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->pending_swap_notify = FALSE;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
cogl_object_unref (info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
flush_pending_swap_notify_idle (void *user_data)
|
|
|
|
{
|
|
|
|
CoglContext *context = user_data;
|
|
|
|
CoglRendererEGL *egl_renderer = context->display->renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* This needs to be disconnected before invoking the callbacks in
|
|
|
|
* case the callbacks cause it to be queued again */
|
2016-05-09 13:22:01 +00:00
|
|
|
_cogl_closure_disconnect (renderer_native->swap_notify_idle);
|
|
|
|
renderer_native->swap_notify_idle = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
g_list_foreach (context->framebuffers,
|
|
|
|
flush_pending_swap_notify_cb,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
free_current_bo (CoglOnscreen *onscreen)
|
|
|
|
{
|
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglContext *context = COGL_FRAMEBUFFER (onscreen)->context;
|
|
|
|
CoglRenderer *renderer = context->display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->current_fb_id)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-09 13:22:01 +00:00
|
|
|
drmModeRmFB (renderer_native->kms_fd,
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->current_fb_id);
|
|
|
|
onscreen_native->current_fb_id = 0;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->current_bo)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
gbm_surface_release_buffer (onscreen_native->surface,
|
|
|
|
onscreen_native->current_bo);
|
|
|
|
onscreen_native->current_bo = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
queue_swap_notify_for_onscreen (CoglOnscreen *onscreen)
|
|
|
|
{
|
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglContext *context = COGL_FRAMEBUFFER (onscreen)->context;
|
|
|
|
CoglRenderer *renderer = context->display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* We only want to notify that the swap is complete when the
|
|
|
|
* application calls cogl_context_dispatch so instead of
|
|
|
|
* immediately notifying we queue an idle callback */
|
2016-05-09 13:22:01 +00:00
|
|
|
if (!renderer_native->swap_notify_idle)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-09 13:22:01 +00:00
|
|
|
renderer_native->swap_notify_idle =
|
2016-05-09 11:51:29 +00:00
|
|
|
_cogl_poll_renderer_add_idle (renderer,
|
|
|
|
flush_pending_swap_notify_idle,
|
|
|
|
context,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->pending_swap_notify = TRUE;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static CoglBool
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_connect (CoglRenderer *cogl_renderer,
|
|
|
|
CoglError **error)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (renderer);
|
|
|
|
CoglRendererEGL *egl_renderer;
|
|
|
|
|
|
|
|
cogl_renderer->winsys = g_slice_new0 (CoglRendererEGL);
|
|
|
|
egl_renderer = cogl_renderer->winsys;
|
|
|
|
|
|
|
|
egl_renderer->platform_vtable = &_cogl_winsys_egl_vtable;
|
2016-05-09 13:22:01 +00:00
|
|
|
egl_renderer->platform = renderer_native;
|
2016-05-09 11:51:29 +00:00
|
|
|
egl_renderer->edpy = EGL_NO_DISPLAY;
|
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
if (renderer_native->gbm == NULL)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_INIT,
|
|
|
|
"Couldn't create gbm device");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
egl_renderer->edpy =
|
|
|
|
eglGetDisplay ((EGLNativeDisplayType) renderer_native->gbm);
|
2016-05-09 11:51:29 +00:00
|
|
|
if (egl_renderer->edpy == EGL_NO_DISPLAY)
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_INIT,
|
|
|
|
"Couldn't get eglDisplay");
|
|
|
|
goto fail;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_cogl_winsys_egl_renderer_connect_common (cogl_renderer, error))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
fail:
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_disconnect (cogl_renderer);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
setup_crtc_modes (CoglDisplay *display, int fb_id)
|
|
|
|
{
|
2016-05-11 08:26:34 +00:00
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
|
|
|
MetaMonitorManagerKms *monitor_manager_kms =
|
|
|
|
META_MONITOR_MANAGER_KMS (monitor_manager);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-11 08:26:34 +00:00
|
|
|
meta_monitor_manager_kms_apply_crtc_modes (monitor_manager_kms, fb_id);
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static CoglBool
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_setup_egl_display (CoglDisplay *display,
|
|
|
|
CoglError **error)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglDisplayEGL *egl_display = display->winsys;
|
|
|
|
CoglRendererEGL *egl_renderer = display->renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
egl_display->platform = renderer_native;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* Force a full modeset / drmModeSetCrtc on
|
|
|
|
* the first swap buffers call.
|
|
|
|
*/
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->pending_set_crtc = TRUE;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_destroy_egl_display (CoglDisplay *display)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglBool
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_egl_context_created (CoglDisplay *display,
|
|
|
|
CoglError **error)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglDisplayEGL *egl_display = display->winsys;
|
|
|
|
CoglRenderer *renderer = display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
if ((egl_renderer->private_features &
|
|
|
|
COGL_EGL_WINSYS_FEATURE_SURFACELESS_CONTEXT) == 0)
|
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->dummy_gbm_surface =
|
2016-05-09 13:22:01 +00:00
|
|
|
gbm_surface_create (renderer_native->gbm,
|
2016-05-09 11:51:29 +00:00
|
|
|
16, 16,
|
|
|
|
GBM_FORMAT_XRGB8888,
|
|
|
|
GBM_BO_USE_RENDERING);
|
2016-05-10 02:16:45 +00:00
|
|
|
if (!renderer_native->dummy_gbm_surface)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_CONTEXT,
|
|
|
|
"Failed to create dummy GBM surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_display->dummy_surface =
|
|
|
|
eglCreateWindowSurface (egl_renderer->edpy,
|
|
|
|
egl_display->egl_config,
|
|
|
|
(EGLNativeWindowType)
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->dummy_gbm_surface,
|
2016-05-09 11:51:29 +00:00
|
|
|
NULL);
|
|
|
|
if (egl_display->dummy_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_CONTEXT,
|
|
|
|
"Failed to create dummy EGL surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!_cogl_winsys_egl_make_current (display,
|
|
|
|
egl_display->dummy_surface,
|
|
|
|
egl_display->dummy_surface,
|
|
|
|
egl_display->egl_context))
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_CONTEXT,
|
|
|
|
"Failed to make context current");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_egl_cleanup_context (CoglDisplay *display)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglDisplayEGL *egl_display = display->winsys;
|
|
|
|
CoglRenderer *renderer = display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
|
|
|
|
|
|
|
if (egl_display->dummy_surface != EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
eglDestroySurface (egl_renderer->edpy, egl_display->dummy_surface);
|
|
|
|
egl_display->dummy_surface = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-12 06:55:06 +00:00
|
|
|
static void
|
|
|
|
flip_callback (void *user_data)
|
|
|
|
{
|
|
|
|
CoglOnscreen *onscreen = user_data;
|
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-12 06:55:06 +00:00
|
|
|
|
|
|
|
queue_swap_notify_for_onscreen (onscreen);
|
|
|
|
|
|
|
|
free_current_bo (onscreen);
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->current_fb_id = onscreen_native->next_fb_id;
|
|
|
|
onscreen_native->next_fb_id = 0;
|
2016-05-12 06:55:06 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->current_bo = onscreen_native->next_bo;
|
|
|
|
onscreen_native->next_bo = NULL;
|
2016-05-12 06:55:06 +00:00
|
|
|
|
|
|
|
cogl_object_unref (onscreen);
|
|
|
|
}
|
|
|
|
|
2016-05-09 11:51:29 +00:00
|
|
|
static void
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_onscreen_native_swap_buffers_with_damage (CoglOnscreen *onscreen,
|
|
|
|
const int *rectangles,
|
|
|
|
int n_rectangles)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-12 06:55:06 +00:00
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
|
|
|
MetaMonitorManagerKms *monitor_manager_kms =
|
|
|
|
META_MONITOR_MANAGER_KMS (monitor_manager);
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglContext *context = COGL_FRAMEBUFFER (onscreen)->context;
|
|
|
|
CoglRenderer *renderer = context->display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
uint32_t handle, stride;
|
2016-05-12 06:55:06 +00:00
|
|
|
gboolean fb_in_use;
|
|
|
|
uint32_t next_fb_id;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* If we already have a pending swap then block until it completes */
|
2016-05-27 07:07:41 +00:00
|
|
|
while (onscreen_native->next_fb_id != 0)
|
2016-05-12 06:55:06 +00:00
|
|
|
meta_monitor_manager_kms_wait_for_flip (monitor_manager_kms);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->pending_egl_surface)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
CoglFramebuffer *fb = COGL_FRAMEBUFFER (renderer_native->onscreen);
|
|
|
|
|
2016-05-09 11:51:29 +00:00
|
|
|
eglDestroySurface (egl_renderer->edpy, egl_onscreen->egl_surface);
|
2016-05-27 07:07:41 +00:00
|
|
|
egl_onscreen->egl_surface = onscreen_native->pending_egl_surface;
|
|
|
|
onscreen_native->pending_egl_surface = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
_cogl_framebuffer_winsys_update_size (fb,
|
|
|
|
renderer_native->width,
|
|
|
|
renderer_native->height);
|
2016-05-09 11:51:29 +00:00
|
|
|
context->current_draw_buffer_changes |= COGL_FRAMEBUFFER_STATE_BIND;
|
|
|
|
}
|
|
|
|
parent_vtable->onscreen_swap_buffers_with_damage (onscreen,
|
|
|
|
rectangles,
|
|
|
|
n_rectangles);
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->pending_surface)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
free_current_bo (onscreen);
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->surface)
|
|
|
|
gbm_surface_destroy (onscreen_native->surface);
|
|
|
|
onscreen_native->surface = onscreen_native->pending_surface;
|
|
|
|
onscreen_native->pending_surface = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
/* Now we need to set the CRTC to whatever is the front buffer */
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->next_bo =
|
|
|
|
gbm_surface_lock_front_buffer (onscreen_native->surface);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
stride = gbm_bo_get_stride (onscreen_native->next_bo);
|
|
|
|
handle = gbm_bo_get_handle (onscreen_native->next_bo).u32;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
if (drmModeAddFB (renderer_native->kms_fd,
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->width,
|
|
|
|
renderer_native->height,
|
2016-05-09 11:51:29 +00:00
|
|
|
24, /* depth */
|
|
|
|
32, /* bpp */
|
|
|
|
stride,
|
|
|
|
handle,
|
2016-05-27 07:07:41 +00:00
|
|
|
&onscreen_native->next_fb_id))
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
g_warning ("Failed to create new back buffer handle: %m");
|
2016-05-27 07:07:41 +00:00
|
|
|
gbm_surface_release_buffer (onscreen_native->surface,
|
|
|
|
onscreen_native->next_bo);
|
|
|
|
onscreen_native->next_bo = NULL;
|
|
|
|
onscreen_native->next_fb_id = 0;
|
2016-05-09 11:51:29 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If this is the first framebuffer to be presented then we now setup the
|
|
|
|
* crtc modes, else we flip from the previous buffer */
|
2016-05-10 02:16:45 +00:00
|
|
|
if (renderer_native->pending_set_crtc)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
setup_crtc_modes (context->display, onscreen_native->next_fb_id);
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->pending_set_crtc = FALSE;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
next_fb_id = onscreen_native->next_fb_id;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-12 06:55:06 +00:00
|
|
|
/* Reference will either be released in flip_callback, or if the fb
|
|
|
|
* wasn't used, indicated by the return value below.
|
|
|
|
*/
|
|
|
|
cogl_object_ref (onscreen);
|
|
|
|
fb_in_use = meta_monitor_manager_kms_flip_all_crtcs (monitor_manager_kms,
|
|
|
|
next_fb_id,
|
|
|
|
flip_callback, onscreen);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-12 06:55:06 +00:00
|
|
|
if (!fb_in_use)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
drmModeRmFB (renderer_native->kms_fd, onscreen_native->next_fb_id);
|
|
|
|
gbm_surface_release_buffer (onscreen_native->surface,
|
|
|
|
onscreen_native->next_bo);
|
|
|
|
onscreen_native->next_bo = NULL;
|
|
|
|
onscreen_native->next_fb_id = 0;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
queue_swap_notify_for_onscreen (onscreen);
|
|
|
|
|
2016-05-12 06:55:06 +00:00
|
|
|
g_object_unref (onscreen);
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglBool
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_init_egl_context (CoglContext *context,
|
|
|
|
CoglError **error)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
COGL_FLAGS_SET (context->features,
|
|
|
|
COGL_FEATURE_ID_SWAP_BUFFERS_EVENT, TRUE);
|
|
|
|
/* TODO: remove this deprecated feature */
|
|
|
|
COGL_FLAGS_SET (context->winsys_features,
|
|
|
|
COGL_WINSYS_FEATURE_SWAP_BUFFERS_EVENT,
|
|
|
|
TRUE);
|
|
|
|
COGL_FLAGS_SET (context->winsys_features,
|
|
|
|
COGL_WINSYS_FEATURE_SYNC_AND_COMPLETE_EVENT,
|
|
|
|
TRUE);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglBool
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_init_onscreen (CoglOnscreen *onscreen,
|
|
|
|
CoglError **error)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
|
|
|
CoglContext *context = framebuffer->context;
|
|
|
|
CoglDisplay *display = context->display;
|
|
|
|
CoglDisplayEGL *egl_display = display->winsys;
|
|
|
|
CoglRenderer *renderer = display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
2016-05-09 13:22:01 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_renderer->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglOnscreenEGL *egl_onscreen;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
_COGL_RETURN_VAL_IF_FAIL (egl_display->egl_context, FALSE);
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
if (renderer_native->onscreen)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Cannot have multiple onscreens in the KMS platform");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->onscreen = onscreen;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
onscreen->winsys = g_slice_new0 (CoglOnscreenEGL);
|
|
|
|
egl_onscreen = onscreen->winsys;
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native = g_slice_new0 (MetaOnscreenNative);
|
|
|
|
egl_onscreen->platform = onscreen_native;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* If a kms_fd is set then the display width and height
|
|
|
|
* won't be available until meta_renderer_native_set_layout
|
|
|
|
* is called. In that case, defer creating the surface
|
|
|
|
* until then.
|
|
|
|
*/
|
2016-05-10 02:16:45 +00:00
|
|
|
if (renderer_native->width == 0 ||
|
|
|
|
renderer_native->height == 0)
|
2016-05-09 11:51:29 +00:00
|
|
|
return TRUE;
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->surface =
|
2016-05-09 13:22:01 +00:00
|
|
|
gbm_surface_create (renderer_native->gbm,
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->width,
|
|
|
|
renderer_native->height,
|
2016-05-09 11:51:29 +00:00
|
|
|
GBM_FORMAT_XRGB8888,
|
|
|
|
GBM_BO_USE_SCANOUT |
|
|
|
|
GBM_BO_USE_RENDERING);
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (!onscreen_native->surface)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Failed to allocate surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_onscreen->egl_surface =
|
|
|
|
eglCreateWindowSurface (egl_renderer->edpy,
|
|
|
|
egl_display->egl_config,
|
2016-05-27 07:07:41 +00:00
|
|
|
(EGLNativeWindowType) onscreen_native->surface,
|
2016-05-09 11:51:29 +00:00
|
|
|
NULL);
|
|
|
|
if (egl_onscreen->egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Failed to allocate surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
_cogl_framebuffer_winsys_update_size (framebuffer,
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->width,
|
|
|
|
renderer_native->height);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_renderer_native_release_onscreen (CoglOnscreen *onscreen)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
|
|
|
CoglContext *context = framebuffer->context;
|
|
|
|
CoglDisplay *display = context->display;
|
|
|
|
CoglDisplayEGL *egl_display = display->winsys;
|
2016-05-10 02:16:45 +00:00
|
|
|
MetaRendererNative *renderer_native = egl_display->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
CoglRenderer *renderer = context->display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
|
|
|
CoglOnscreenEGL *egl_onscreen = onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* If we never successfully allocated then there's nothing to do */
|
|
|
|
if (egl_onscreen == NULL)
|
|
|
|
return;
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->onscreen = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* flip state takes a reference on the onscreen so there should
|
|
|
|
* never be outstanding flips when we reach here. */
|
2016-05-27 07:07:41 +00:00
|
|
|
g_return_if_fail (onscreen_native->next_fb_id == 0);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
free_current_bo (onscreen);
|
|
|
|
|
|
|
|
if (egl_onscreen->egl_surface != EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
eglDestroySurface (egl_renderer->edpy, egl_onscreen->egl_surface);
|
|
|
|
egl_onscreen->egl_surface = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->surface)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
gbm_surface_destroy (onscreen_native->surface);
|
|
|
|
onscreen_native->surface = NULL;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
g_slice_free (MetaOnscreenNative, onscreen_native);
|
2016-05-09 11:51:29 +00:00
|
|
|
g_slice_free (CoglOnscreenEGL, onscreen->winsys);
|
|
|
|
onscreen->winsys = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const CoglWinsysEGLVtable
|
|
|
|
_cogl_winsys_egl_vtable = {
|
2016-05-27 07:13:09 +00:00
|
|
|
.display_setup = meta_renderer_native_setup_egl_display,
|
|
|
|
.display_destroy = meta_renderer_native_destroy_egl_display,
|
|
|
|
.context_created = meta_renderer_native_egl_context_created,
|
|
|
|
.cleanup_context = meta_renderer_native_egl_cleanup_context,
|
|
|
|
.context_init = meta_renderer_native_init_egl_context
|
2016-05-09 11:51:29 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
struct gbm_device *
|
|
|
|
meta_renderer_native_get_gbm (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2016-05-09 13:22:01 +00:00
|
|
|
return renderer_native->gbm;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
meta_renderer_native_get_kms_fd (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
return renderer_native->kms_fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_renderer_native_queue_modes_reset (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->pending_set_crtc = TRUE;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
gboolean
|
|
|
|
meta_renderer_native_set_layout (MetaRendererNative *renderer_native,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
ClutterBackend *clutter_backend = clutter_get_default_backend ();
|
|
|
|
CoglContext *cogl_context = clutter_backend_get_cogl_context (clutter_backend);
|
|
|
|
CoglDisplay *cogl_display = cogl_context_get_display (cogl_context);
|
|
|
|
CoglDisplayEGL *egl_display = cogl_display->winsys;
|
|
|
|
CoglRenderer *renderer = cogl_display->renderer;
|
|
|
|
CoglRendererEGL *egl_renderer = renderer->winsys;
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
if ((width != renderer_native->width ||
|
|
|
|
height != renderer_native->height) &&
|
|
|
|
renderer_native->onscreen)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
CoglOnscreenEGL *egl_onscreen = renderer_native->onscreen->winsys;
|
2016-05-27 07:07:41 +00:00
|
|
|
MetaOnscreenNative *onscreen_native = egl_onscreen->platform;
|
2016-05-09 11:51:29 +00:00
|
|
|
struct gbm_surface *new_surface;
|
|
|
|
EGLSurface new_egl_surface;
|
|
|
|
|
|
|
|
/* Need to drop the GBM surface and create a new one */
|
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
new_surface = gbm_surface_create (renderer_native->gbm,
|
2016-05-09 11:51:29 +00:00
|
|
|
width, height,
|
|
|
|
GBM_FORMAT_XRGB8888,
|
|
|
|
GBM_BO_USE_SCANOUT |
|
|
|
|
GBM_BO_USE_RENDERING);
|
|
|
|
|
|
|
|
if (!new_surface)
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Failed to allocate new surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_egl_surface =
|
|
|
|
eglCreateWindowSurface (egl_renderer->edpy,
|
|
|
|
egl_display->egl_config,
|
|
|
|
(EGLNativeWindowType) new_surface,
|
|
|
|
NULL);
|
|
|
|
if (new_egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
_cogl_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Failed to allocate new surface");
|
|
|
|
gbm_surface_destroy (new_surface);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->pending_egl_surface)
|
|
|
|
eglDestroySurface (egl_renderer->edpy,
|
|
|
|
onscreen_native->pending_egl_surface);
|
|
|
|
if (onscreen_native->pending_surface)
|
|
|
|
gbm_surface_destroy (onscreen_native->pending_surface);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* If there's already a surface, wait until the next swap to switch
|
|
|
|
* it out, otherwise, if we're just starting up we can use the new
|
|
|
|
* surface right away.
|
|
|
|
*/
|
2016-05-27 07:07:41 +00:00
|
|
|
if (onscreen_native->surface != NULL)
|
2016-05-09 11:51:29 +00:00
|
|
|
{
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->pending_surface = new_surface;
|
|
|
|
onscreen_native->pending_egl_surface = new_egl_surface;
|
2016-05-09 11:51:29 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (renderer_native->onscreen);
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:07:41 +00:00
|
|
|
onscreen_native->surface = new_surface;
|
2016-05-09 11:51:29 +00:00
|
|
|
egl_onscreen->egl_surface = new_egl_surface;
|
|
|
|
|
|
|
|
_cogl_framebuffer_winsys_update_size (framebuffer, width, height);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->width = width;
|
|
|
|
renderer_native->height = height;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
renderer_native->pending_set_crtc = TRUE;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const CoglWinsysVtable *
|
|
|
|
get_native_cogl_winsys_vtable (void)
|
|
|
|
{
|
|
|
|
static CoglBool vtable_inited = FALSE;
|
|
|
|
static CoglWinsysVtable vtable;
|
|
|
|
|
|
|
|
if (!vtable_inited)
|
|
|
|
{
|
|
|
|
/* The this winsys is a subclass of the EGL winsys so we
|
|
|
|
start by copying its vtable */
|
|
|
|
|
|
|
|
parent_vtable = _cogl_winsys_egl_get_vtable ();
|
|
|
|
vtable = *parent_vtable;
|
|
|
|
|
|
|
|
vtable.id = COGL_WINSYS_ID_CUSTOM;
|
|
|
|
vtable.name = "EGL_KMS";
|
|
|
|
|
2016-05-27 07:13:09 +00:00
|
|
|
vtable.renderer_connect = meta_renderer_native_connect;
|
|
|
|
vtable.renderer_disconnect = meta_renderer_native_disconnect;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
2016-05-27 07:13:09 +00:00
|
|
|
vtable.onscreen_init = meta_renderer_native_init_onscreen;
|
|
|
|
vtable.onscreen_deinit = meta_renderer_native_release_onscreen;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
/* The KMS winsys doesn't support swap region */
|
|
|
|
vtable.onscreen_swap_region = NULL;
|
|
|
|
vtable.onscreen_swap_buffers_with_damage =
|
2016-05-27 07:13:09 +00:00
|
|
|
meta_onscreen_native_swap_buffers_with_damage;
|
2016-05-09 11:51:29 +00:00
|
|
|
|
|
|
|
vtable_inited = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return &vtable;
|
|
|
|
}
|
|
|
|
|
2016-05-09 01:59:54 +00:00
|
|
|
static CoglRenderer *
|
|
|
|
meta_renderer_native_create_cogl_renderer (MetaRenderer *renderer)
|
|
|
|
{
|
|
|
|
CoglRenderer *cogl_renderer;
|
|
|
|
|
|
|
|
cogl_renderer = cogl_renderer_new ();
|
|
|
|
cogl_renderer_set_custom_winsys (cogl_renderer,
|
2016-05-09 11:51:29 +00:00
|
|
|
get_native_cogl_winsys_vtable);
|
2016-05-09 01:59:54 +00:00
|
|
|
|
|
|
|
return cogl_renderer;
|
|
|
|
}
|
|
|
|
|
2016-05-07 15:09:59 +00:00
|
|
|
static void
|
|
|
|
meta_renderer_native_get_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_KMS_FD:
|
|
|
|
g_value_set_int (value, renderer_native->kms_fd);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_renderer_native_set_property (GObject *object,
|
|
|
|
guint prop_id,
|
|
|
|
const GValue *value,
|
|
|
|
GParamSpec *pspec)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (object);
|
|
|
|
|
|
|
|
switch (prop_id)
|
|
|
|
{
|
|
|
|
case PROP_KMS_FD:
|
|
|
|
renderer_native->kms_fd = g_value_get_int (value);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-09 13:22:01 +00:00
|
|
|
static void
|
|
|
|
meta_renderer_native_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (object);
|
|
|
|
|
2016-05-10 02:16:45 +00:00
|
|
|
g_clear_pointer (&renderer_native->dummy_gbm_surface, gbm_surface_destroy);
|
2016-05-09 13:22:01 +00:00
|
|
|
g_clear_pointer (&renderer_native->gbm, gbm_device_destroy);
|
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_renderer_native_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_renderer_native_initable_init (GInitable *initable,
|
|
|
|
GCancellable *cancellable,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (initable);
|
2016-05-10 02:16:45 +00:00
|
|
|
drmModeRes *resources;
|
2016-05-09 13:22:01 +00:00
|
|
|
|
|
|
|
renderer_native->gbm = gbm_create_device (renderer_native->kms_fd);
|
|
|
|
if (!renderer_native->gbm)
|
|
|
|
{
|
2016-05-10 02:16:45 +00:00
|
|
|
g_set_error (error, G_IO_ERROR,
|
2016-05-09 13:22:01 +00:00
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Failed to create gbm device");
|
2016-05-10 02:16:45 +00:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
resources = drmModeGetResources (renderer_native->kms_fd);
|
|
|
|
if (!resources)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"drmModeGetResources failed");
|
|
|
|
goto err_resources;
|
2016-05-09 13:22:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
2016-05-10 02:16:45 +00:00
|
|
|
|
|
|
|
err_resources:
|
|
|
|
g_clear_pointer (&renderer_native->gbm, gbm_device_destroy);
|
|
|
|
|
|
|
|
err:
|
|
|
|
return FALSE;
|
2016-05-09 13:22:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
initable_iface_init (GInitableIface *initable_iface)
|
|
|
|
{
|
|
|
|
initable_iface->init = meta_renderer_native_initable_init;
|
|
|
|
}
|
|
|
|
|
2016-05-07 15:07:46 +00:00
|
|
|
static void
|
|
|
|
meta_renderer_native_init (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_renderer_native_class_init (MetaRendererNativeClass *klass)
|
|
|
|
{
|
2016-05-07 15:09:59 +00:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2016-05-09 01:59:54 +00:00
|
|
|
MetaRendererClass *renderer_class = META_RENDERER_CLASS (klass);
|
2016-05-07 15:09:59 +00:00
|
|
|
|
|
|
|
object_class->get_property = meta_renderer_native_get_property;
|
|
|
|
object_class->set_property = meta_renderer_native_set_property;
|
2016-05-09 13:22:01 +00:00
|
|
|
object_class->finalize = meta_renderer_native_finalize;
|
2016-05-07 15:09:59 +00:00
|
|
|
|
2016-05-09 01:59:54 +00:00
|
|
|
renderer_class->create_cogl_renderer = meta_renderer_native_create_cogl_renderer;
|
|
|
|
|
2016-05-07 15:09:59 +00:00
|
|
|
g_object_class_install_property (object_class,
|
|
|
|
PROP_KMS_FD,
|
|
|
|
g_param_spec_int ("kms-fd",
|
|
|
|
"KMS fd",
|
|
|
|
"The KMS file descriptor",
|
|
|
|
0, G_MAXINT, 0,
|
|
|
|
G_PARAM_READWRITE |
|
|
|
|
G_PARAM_CONSTRUCT_ONLY));
|
2016-05-07 15:07:46 +00:00
|
|
|
}
|
2016-05-09 13:22:01 +00:00
|
|
|
|
|
|
|
MetaRendererNative *
|
|
|
|
meta_renderer_native_new (int kms_fd,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native;
|
|
|
|
|
|
|
|
renderer_native = g_object_new (META_TYPE_RENDERER_NATIVE,
|
|
|
|
"kms-fd", kms_fd,
|
|
|
|
NULL);
|
|
|
|
if (!g_initable_init (G_INITABLE (renderer_native), NULL, error))
|
|
|
|
{
|
|
|
|
g_object_unref (renderer_native);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return renderer_native;
|
|
|
|
}
|