2016-05-07 23:07:46 +08:00
|
|
|
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
|
|
|
|
|
|
/*
|
2016-05-09 19:51:29 +08:00
|
|
|
* Copyright (C) 2011 Intel Corporation.
|
2016-05-07 23:07:46 +08:00
|
|
|
* Copyright (C) 2016 Red Hat
|
2018-11-29 13:34:40 +02:00
|
|
|
* Copyright (c) 2018,2019 DisplayLink (UK) Ltd.
|
2016-05-07 23:07:46 +08:00
|
|
|
*
|
|
|
|
* 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 19:51:29 +08: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 23:07:46 +08:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
2018-07-10 10:36:24 +02:00
|
|
|
#include <drm_fourcc.h>
|
2016-05-09 19:51:29 +08:00
|
|
|
#include <errno.h>
|
|
|
|
#include <gbm.h>
|
2016-05-09 21:22:01 +08:00
|
|
|
#include <gio/gio.h>
|
|
|
|
#include <glib-object.h>
|
2016-05-09 19:51:29 +08:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
2016-05-07 23:07:46 +08:00
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
#include "backends/meta-backend-private.h"
|
2017-03-28 12:35:19 +08:00
|
|
|
#include "backends/meta-crtc.h"
|
2016-08-18 11:28:59 +08:00
|
|
|
#include "backends/meta-egl-ext.h"
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "backends/meta-egl.h"
|
|
|
|
#include "backends/meta-gles3.h"
|
2016-12-13 10:37:11 +08:00
|
|
|
#include "backends/meta-logical-monitor.h"
|
2017-03-24 17:35:51 +08:00
|
|
|
#include "backends/meta-output.h"
|
2016-06-08 17:05:31 +08:00
|
|
|
#include "backends/meta-renderer-view.h"
|
2020-08-28 21:58:18 -03:00
|
|
|
#include "backends/native/meta-cogl-utils.h"
|
2017-08-03 15:42:50 +01:00
|
|
|
#include "backends/native/meta-crtc-kms.h"
|
2019-05-20 22:08:35 +02:00
|
|
|
#include "backends/native/meta-drm-buffer-dumb.h"
|
|
|
|
#include "backends/native/meta-drm-buffer-gbm.h"
|
2019-05-24 17:07:14 +03:00
|
|
|
#include "backends/native/meta-drm-buffer-import.h"
|
2019-05-20 21:20:17 +02:00
|
|
|
#include "backends/native/meta-drm-buffer.h"
|
2017-07-10 18:19:32 +08:00
|
|
|
#include "backends/native/meta-gpu-kms.h"
|
2020-11-14 10:41:23 +01:00
|
|
|
#include "backends/native/meta-kms-device.h"
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
#include "backends/native/meta-kms-update.h"
|
2019-03-11 11:13:01 +01:00
|
|
|
#include "backends/native/meta-kms-utils.h"
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
#include "backends/native/meta-kms.h"
|
|
|
|
#include "backends/native/meta-output-kms.h"
|
2017-07-25 10:21:04 +08:00
|
|
|
#include "backends/native/meta-renderer-native-gles3.h"
|
2018-07-10 10:36:24 +02:00
|
|
|
#include "backends/native/meta-renderer-native.h"
|
2016-05-07 23:09:59 +08:00
|
|
|
#include "cogl/cogl.h"
|
2016-06-08 17:05:31 +08:00
|
|
|
#include "core/boxes-private.h"
|
2016-05-07 23:09:59 +08:00
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifndef EGL_DRM_MASTER_FD_EXT
|
|
|
|
#define EGL_DRM_MASTER_FD_EXT 0x333C
|
|
|
|
#endif
|
|
|
|
|
2018-11-21 15:25:07 +02:00
|
|
|
/* added in libdrm 2.4.95 */
|
|
|
|
#ifndef DRM_FORMAT_INVALID
|
|
|
|
#define DRM_FORMAT_INVALID 0
|
|
|
|
#endif
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
typedef enum _MetaSharedFramebufferCopyMode
|
|
|
|
{
|
2019-05-24 17:07:14 +03:00
|
|
|
/* Zero-copy: primary GPU exports, secondary GPU imports as KMS FB */
|
|
|
|
META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO,
|
2018-11-29 13:34:40 +02:00
|
|
|
/* the secondary GPU will make the copy */
|
|
|
|
META_SHARED_FRAMEBUFFER_COPY_MODE_SECONDARY_GPU,
|
|
|
|
/*
|
|
|
|
* The copy is made in the primary GPU rendering context, either
|
|
|
|
* as a CPU copy through Cogl read-pixels or as primary GPU copy
|
|
|
|
* using glBlitFramebuffer.
|
|
|
|
*/
|
|
|
|
META_SHARED_FRAMEBUFFER_COPY_MODE_PRIMARY
|
2017-07-25 10:21:04 +08:00
|
|
|
} MetaSharedFramebufferCopyMode;
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
typedef struct _MetaRendererNativeGpuData
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
struct gbm_device *device;
|
|
|
|
} gbm;
|
|
|
|
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
struct {
|
|
|
|
EGLDeviceEXT device;
|
|
|
|
} egl;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
MetaRendererNativeMode mode;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
EGLDisplay egl_display;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fields used for blitting iGPU framebuffer content onto dGPU framebuffers.
|
|
|
|
*/
|
|
|
|
struct {
|
|
|
|
MetaSharedFramebufferCopyMode copy_mode;
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
gboolean is_hardware_rendering;
|
2018-11-29 13:34:40 +02:00
|
|
|
gboolean has_EGL_EXT_image_dma_buf_import_modifiers;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
/* For GPU blit mode */
|
|
|
|
EGLContext egl_context;
|
|
|
|
EGLConfig egl_config;
|
|
|
|
} secondary;
|
2017-07-10 18:19:32 +08:00
|
|
|
} MetaRendererNativeGpuData;
|
|
|
|
|
2019-05-24 17:07:14 +03:00
|
|
|
typedef enum _MetaSharedFramebufferImportStatus
|
|
|
|
{
|
|
|
|
/* Not tried importing yet. */
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_NONE,
|
|
|
|
/* Tried before and failed. */
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_FAILED,
|
|
|
|
/* Tried before and succeeded. */
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_OK
|
|
|
|
} MetaSharedFramebufferImportStatus;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
typedef struct _MetaOnscreenNativeSecondaryGpuState
|
2016-05-27 15:07:41 +08:00
|
|
|
{
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaGpuKms *gpu_kms;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
|
|
|
EGLSurface egl_surface;
|
|
|
|
|
|
|
|
struct {
|
|
|
|
struct gbm_surface *surface;
|
2019-05-20 21:20:17 +02:00
|
|
|
MetaDrmBuffer *current_fb;
|
|
|
|
MetaDrmBuffer *next_fb;
|
2017-07-25 10:21:04 +08:00
|
|
|
} gbm;
|
|
|
|
|
|
|
|
struct {
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBufferDumb *current_dumb_fb;
|
|
|
|
MetaDrmBufferDumb *dumb_fbs[2];
|
2017-07-25 10:21:04 +08:00
|
|
|
} cpu;
|
|
|
|
|
2019-07-05 14:09:31 +03:00
|
|
|
gboolean noted_primary_gpu_copy_ok;
|
|
|
|
gboolean noted_primary_gpu_copy_failed;
|
2019-05-24 17:07:14 +03:00
|
|
|
MetaSharedFramebufferImportStatus import_status;
|
2017-07-25 10:21:04 +08:00
|
|
|
} MetaOnscreenNativeSecondaryGpuState;
|
|
|
|
|
|
|
|
typedef struct _MetaOnscreenNative
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native;
|
|
|
|
MetaGpuKms *render_gpu;
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOutput *output;
|
|
|
|
MetaCrtc *crtc;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
2017-07-10 18:19:32 +08:00
|
|
|
|
2016-08-18 11:11:30 +08:00
|
|
|
struct {
|
|
|
|
struct gbm_surface *surface;
|
2019-05-20 21:20:17 +02:00
|
|
|
MetaDrmBuffer *current_fb;
|
|
|
|
MetaDrmBuffer *next_fb;
|
2016-08-18 11:11:30 +08:00
|
|
|
} gbm;
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
struct {
|
|
|
|
EGLStreamKHR stream;
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBufferDumb *dumb_fb;
|
2016-08-18 11:28:59 +08:00
|
|
|
} egl;
|
|
|
|
#endif
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
MetaRendererView *view;
|
2016-05-27 15:07:41 +08:00
|
|
|
} MetaOnscreenNative;
|
|
|
|
|
2016-05-07 23:07:46 +08:00
|
|
|
struct _MetaRendererNative
|
|
|
|
{
|
|
|
|
MetaRenderer parent;
|
2016-05-07 23:09:59 +08:00
|
|
|
|
2018-12-10 14:19:25 +02:00
|
|
|
MetaGpuKms *primary_gpu_kms;
|
2019-01-11 15:35:42 +01:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaGles3 *gles3;
|
2016-08-18 10:37:16 +08:00
|
|
|
|
2018-04-11 14:39:15 +02:00
|
|
|
gboolean use_modifiers;
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
GHashTable *gpu_datas;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2016-05-09 21:22:01 +08:00
|
|
|
CoglClosure *swap_notify_idle;
|
2016-05-10 10:16:45 +08:00
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
GList *pending_mode_set_views;
|
|
|
|
gboolean pending_mode_set;
|
|
|
|
guint mode_set_failed_feedback_source_id;
|
2019-03-25 10:24:46 +01:00
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
GList *kept_alive_onscreens;
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
GList *power_save_page_flip_onscreens;
|
2019-03-25 10:24:46 +01:00
|
|
|
guint power_save_page_flip_source_id;
|
2016-05-07 23:07:46 +08:00
|
|
|
};
|
|
|
|
|
2016-05-09 21:22:01 +08: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 23:07:46 +08:00
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
static const CoglWinsysEGLVtable _cogl_winsys_egl_vtable;
|
|
|
|
static const CoglWinsysVtable *parent_vtable;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static MetaEgl *
|
|
|
|
meta_renderer_native_get_egl (MetaRendererNative *renderer_native);
|
|
|
|
|
|
|
|
static void
|
2020-01-17 21:08:02 +01:00
|
|
|
free_current_secondary_bo (CoglOnscreen *onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2019-06-19 20:57:14 +02:00
|
|
|
static void
|
|
|
|
meta_renderer_native_queue_modes_reset (MetaRendererNative *renderer_native);
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static void
|
|
|
|
meta_renderer_native_gpu_data_free (MetaRendererNativeGpuData *renderer_gpu_data)
|
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
|
|
|
|
|
|
|
if (renderer_gpu_data->egl_display != EGL_NO_DISPLAY)
|
|
|
|
meta_egl_terminate (egl, renderer_gpu_data->egl_display, NULL);
|
|
|
|
|
2017-11-06 20:39:56 +01:00
|
|
|
g_clear_pointer (&renderer_gpu_data->gbm.device, gbm_device_destroy);
|
2017-07-10 18:19:32 +08:00
|
|
|
g_free (renderer_gpu_data);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaRendererNativeGpuData *
|
|
|
|
meta_renderer_native_get_gpu_data (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms)
|
|
|
|
{
|
|
|
|
return g_hash_table_lookup (renderer_native->gpu_datas, gpu_kms);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaRendererNative *
|
|
|
|
meta_renderer_native_from_gpu (MetaGpuKms *gpu_kms)
|
|
|
|
{
|
2019-01-11 15:35:42 +01:00
|
|
|
MetaBackend *backend = meta_gpu_get_backend (META_GPU (gpu_kms));
|
2017-07-10 18:19:32 +08:00
|
|
|
|
|
|
|
return META_RENDERER_NATIVE (meta_backend_get_renderer (backend));
|
|
|
|
}
|
|
|
|
|
|
|
|
struct gbm_device *
|
|
|
|
meta_gbm_device_from_gpu (MetaGpuKms *gpu_kms)
|
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNative *renderer_native = meta_renderer_native_from_gpu (gpu_kms);
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
gpu_kms);
|
|
|
|
|
|
|
|
return renderer_gpu_data->gbm.device;
|
|
|
|
}
|
|
|
|
|
2019-09-12 11:50:34 +02:00
|
|
|
MetaGpuKms *
|
|
|
|
meta_renderer_native_get_primary_gpu (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
return renderer_native->primary_gpu_kms;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static MetaRendererNativeGpuData *
|
|
|
|
meta_create_renderer_native_gpu_data (MetaGpuKms *gpu_kms)
|
|
|
|
{
|
|
|
|
return g_new0 (MetaRendererNativeGpuData, 1);
|
|
|
|
}
|
|
|
|
|
2017-07-24 15:54:29 +08:00
|
|
|
static MetaEgl *
|
|
|
|
meta_renderer_native_get_egl (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
|
|
|
|
return meta_backend_get_egl (meta_renderer_get_backend (renderer));
|
2017-07-24 15:54:29 +08:00
|
|
|
}
|
|
|
|
|
2020-09-14 20:36:17 +02:00
|
|
|
gboolean
|
|
|
|
meta_renderer_native_use_modifiers (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
return renderer_native->use_modifiers;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static MetaEgl *
|
|
|
|
meta_onscreen_native_get_egl (MetaOnscreenNative *onscreen_native)
|
|
|
|
{
|
|
|
|
return meta_renderer_native_get_egl (onscreen_native->renderer_native);
|
|
|
|
}
|
|
|
|
|
2018-02-14 13:27:28 -05:00
|
|
|
static GArray *
|
2020-02-26 19:47:44 +01:00
|
|
|
get_supported_kms_modifiers (MetaCrtcKms *crtc_kms,
|
|
|
|
uint32_t format)
|
2018-02-14 13:27:28 -05:00
|
|
|
{
|
|
|
|
GArray *modifiers;
|
2020-01-17 21:08:02 +01:00
|
|
|
GArray *crtc_mods;
|
2018-02-14 13:27:28 -05:00
|
|
|
unsigned int i;
|
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
crtc_mods = meta_crtc_kms_get_modifiers (crtc_kms, format);
|
2020-01-17 21:08:02 +01:00
|
|
|
if (!crtc_mods)
|
2018-02-14 13:27:28 -05:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
modifiers = g_array_new (FALSE, FALSE, sizeof (uint64_t));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* For each modifier from base_crtc, check if it's available on all other
|
|
|
|
* CRTCs.
|
|
|
|
*/
|
2020-01-17 21:08:02 +01:00
|
|
|
for (i = 0; i < crtc_mods->len; i++)
|
2018-02-14 13:27:28 -05:00
|
|
|
{
|
2020-01-17 21:08:02 +01:00
|
|
|
uint64_t modifier = g_array_index (crtc_mods, uint64_t, i);
|
2018-02-14 13:27:28 -05:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
g_array_append_val (modifiers, modifier);
|
2018-02-14 13:27:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
if (modifiers->len == 0)
|
|
|
|
{
|
|
|
|
g_array_free (modifiers, TRUE);
|
2020-01-17 21:08:02 +01:00
|
|
|
return NULL;
|
2018-02-14 13:27:28 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
return modifiers;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GArray *
|
|
|
|
get_supported_egl_modifiers (CoglOnscreen *onscreen,
|
2020-02-26 19:47:44 +01:00
|
|
|
MetaCrtcKms *crtc_kms,
|
2018-02-14 13:27:28 -05:00
|
|
|
uint32_t format)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2018-02-14 13:27:28 -05:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaGpu *gpu;
|
2018-02-14 13:27:28 -05:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
EGLint num_modifiers;
|
|
|
|
GArray *modifiers;
|
|
|
|
GError *error = NULL;
|
|
|
|
gboolean ret;
|
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
gpu = meta_crtc_get_gpu (META_CRTC (crtc_kms));
|
2018-02-14 13:27:28 -05:00
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
META_GPU_KMS (gpu));
|
|
|
|
|
|
|
|
if (!meta_egl_has_extensions (egl, renderer_gpu_data->egl_display, NULL,
|
|
|
|
"EGL_EXT_image_dma_buf_import_modifiers",
|
|
|
|
NULL))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret = meta_egl_query_dma_buf_modifiers (egl, renderer_gpu_data->egl_display,
|
|
|
|
format, 0, NULL, NULL,
|
|
|
|
&num_modifiers, NULL);
|
|
|
|
if (!ret || num_modifiers == 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
modifiers = g_array_sized_new (FALSE, FALSE, sizeof (uint64_t),
|
|
|
|
num_modifiers);
|
|
|
|
ret = meta_egl_query_dma_buf_modifiers (egl, renderer_gpu_data->egl_display,
|
|
|
|
format, num_modifiers,
|
|
|
|
(EGLuint64KHR *) modifiers->data, NULL,
|
|
|
|
&num_modifiers, &error);
|
|
|
|
|
|
|
|
if (!ret)
|
|
|
|
{
|
|
|
|
g_warning ("Failed to query DMABUF modifiers: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
g_array_free (modifiers, TRUE);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return modifiers;
|
|
|
|
}
|
|
|
|
|
|
|
|
static GArray *
|
|
|
|
get_supported_modifiers (CoglOnscreen *onscreen,
|
|
|
|
uint32_t format)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2018-02-14 13:27:28 -05:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-02-26 19:47:44 +01:00
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (onscreen_native->crtc);
|
2018-11-21 15:25:07 +02:00
|
|
|
MetaGpu *gpu;
|
2020-01-17 21:08:02 +01:00
|
|
|
g_autoptr (GArray) modifiers = NULL;
|
2018-11-21 15:25:07 +02:00
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
gpu = meta_crtc_get_gpu (META_CRTC (crtc_kms));
|
2020-01-17 21:08:02 +01:00
|
|
|
if (gpu == META_GPU (onscreen_native->render_gpu))
|
2020-02-26 19:47:44 +01:00
|
|
|
modifiers = get_supported_kms_modifiers (crtc_kms, format);
|
2020-01-17 21:08:02 +01:00
|
|
|
else
|
2020-02-26 19:47:44 +01:00
|
|
|
modifiers = get_supported_egl_modifiers (onscreen, crtc_kms, format);
|
2018-11-21 15:25:07 +02:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
return g_steal_pointer (&modifiers);
|
2018-11-21 15:25:07 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static GArray *
|
2020-01-17 21:08:02 +01:00
|
|
|
get_supported_kms_formats (CoglOnscreen *onscreen)
|
2018-11-21 15:25:07 +02:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2018-11-21 15:25:07 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-02-26 19:47:44 +01:00
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (onscreen_native->crtc);
|
2018-11-21 15:25:07 +02:00
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
return meta_crtc_kms_copy_drm_format_list (crtc_kms);
|
2018-11-21 15:25:07 +02:00
|
|
|
}
|
|
|
|
|
2020-10-12 16:57:45 +08:00
|
|
|
static uint32_t
|
|
|
|
get_gbm_format_from_egl (MetaEgl *egl,
|
|
|
|
EGLDisplay egl_display,
|
|
|
|
EGLConfig egl_config)
|
|
|
|
{
|
|
|
|
uint32_t gbm_format;
|
|
|
|
EGLint native_visual_id;
|
|
|
|
|
|
|
|
if (meta_egl_get_config_attrib (egl,
|
|
|
|
egl_display,
|
|
|
|
egl_config,
|
|
|
|
EGL_NATIVE_VISUAL_ID,
|
|
|
|
&native_visual_id,
|
|
|
|
NULL))
|
|
|
|
gbm_format = (uint32_t) native_visual_id;
|
|
|
|
else
|
|
|
|
g_assert_not_reached ();
|
|
|
|
|
|
|
|
return gbm_format;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static gboolean
|
|
|
|
init_secondary_gpu_state_gpu_copy_mode (MetaRendererNative *renderer_native,
|
|
|
|
CoglOnscreen *onscreen,
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
|
|
|
int width, height;
|
|
|
|
EGLNativeWindowType egl_native_window;
|
|
|
|
struct gbm_surface *gbm_surface;
|
|
|
|
EGLSurface egl_surface;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaGpuKms *gpu_kms;
|
2020-10-12 16:57:45 +08:00
|
|
|
uint32_t format;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
width = cogl_framebuffer_get_width (framebuffer);
|
|
|
|
height = cogl_framebuffer_get_height (framebuffer);
|
2020-10-12 16:57:45 +08:00
|
|
|
format = get_gbm_format_from_egl (egl,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
renderer_gpu_data->secondary.egl_config);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
gbm_surface = gbm_surface_create (renderer_gpu_data->gbm.device,
|
|
|
|
width, height,
|
2020-10-12 16:57:45 +08:00
|
|
|
format,
|
2017-07-25 10:21:04 +08:00
|
|
|
GBM_BO_USE_SCANOUT | GBM_BO_USE_RENDERING);
|
|
|
|
if (!gbm_surface)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"Failed to create gbm_surface: %s", strerror (errno));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_native_window = (EGLNativeWindowType) gbm_surface;
|
|
|
|
egl_surface =
|
|
|
|
meta_egl_create_window_surface (egl,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
renderer_gpu_data->secondary.egl_config,
|
|
|
|
egl_native_window,
|
|
|
|
NULL,
|
|
|
|
error);
|
|
|
|
if (egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
gbm_surface_destroy (gbm_surface);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
secondary_gpu_state = g_new0 (MetaOnscreenNativeSecondaryGpuState, 1);
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
gpu_kms = META_GPU_KMS (meta_crtc_get_gpu (onscreen_native->crtc));
|
2017-07-25 10:21:04 +08:00
|
|
|
secondary_gpu_state->gpu_kms = gpu_kms;
|
|
|
|
secondary_gpu_state->renderer_gpu_data = renderer_gpu_data;
|
|
|
|
secondary_gpu_state->gbm.surface = gbm_surface;
|
|
|
|
secondary_gpu_state->egl_surface = egl_surface;
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
onscreen_native->secondary_gpu_state = secondary_gpu_state;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2019-09-17 13:23:22 +03:00
|
|
|
static void
|
|
|
|
secondary_gpu_release_dumb (MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state)
|
|
|
|
{
|
|
|
|
unsigned i;
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (secondary_gpu_state->cpu.dumb_fbs); i++)
|
2020-09-29 09:40:53 +02:00
|
|
|
g_clear_object (&secondary_gpu_state->cpu.dumb_fbs[i]);
|
2019-09-17 13:23:22 +03:00
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
|
|
|
secondary_gpu_state_free (MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaEgl *egl = meta_backend_get_egl (backend);
|
|
|
|
|
|
|
|
if (secondary_gpu_state->egl_surface != EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
|
|
|
renderer_gpu_data = secondary_gpu_state->renderer_gpu_data;
|
|
|
|
meta_egl_destroy_surface (egl,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
secondary_gpu_state->egl_surface,
|
|
|
|
NULL);
|
|
|
|
}
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
g_clear_object (&secondary_gpu_state->gbm.current_fb);
|
2019-09-16 16:47:49 +03:00
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
2017-07-25 10:21:04 +08:00
|
|
|
g_clear_pointer (&secondary_gpu_state->gbm.surface, gbm_surface_destroy);
|
|
|
|
|
2019-09-17 13:23:22 +03:00
|
|
|
secondary_gpu_release_dumb (secondary_gpu_state);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
g_free (secondary_gpu_state);
|
|
|
|
}
|
|
|
|
|
2018-11-21 15:25:07 +02:00
|
|
|
static uint32_t
|
2020-01-17 21:08:02 +01:00
|
|
|
pick_secondary_gpu_framebuffer_format_for_cpu (CoglOnscreen *onscreen)
|
2018-11-21 15:25:07 +02:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* cogl_framebuffer_read_pixels_into_bitmap () supported formats in
|
|
|
|
* preference order. Ideally these should depend on the render buffer
|
|
|
|
* format copy_shared_framebuffer_cpu () will be reading from but
|
|
|
|
* alpha channel ignored.
|
|
|
|
*/
|
|
|
|
static const uint32_t preferred_formats[] =
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* DRM_FORMAT_XBGR8888 a.k.a GL_RGBA, GL_UNSIGNED_BYTE on
|
|
|
|
* little-endian is possibly the most optimized glReadPixels
|
|
|
|
* output format. glReadPixels cannot avoid manufacturing an alpha
|
|
|
|
* channel if the render buffer does not have one and converting
|
|
|
|
* to ABGR8888 may be more optimized than ARGB8888.
|
|
|
|
*/
|
|
|
|
DRM_FORMAT_XBGR8888,
|
|
|
|
/* The rest are other fairly commonly used formats in OpenGL. */
|
|
|
|
DRM_FORMAT_XRGB8888,
|
|
|
|
};
|
2019-06-20 10:22:32 +02:00
|
|
|
g_autoptr (GArray) formats = NULL;
|
2018-11-21 15:25:07 +02:00
|
|
|
size_t k;
|
|
|
|
unsigned int i;
|
|
|
|
uint32_t drm_format;
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
formats = get_supported_kms_formats (onscreen);
|
2018-11-21 15:25:07 +02:00
|
|
|
|
|
|
|
/* Check if any of our preferred formats are supported. */
|
|
|
|
for (k = 0; k < G_N_ELEMENTS (preferred_formats); k++)
|
|
|
|
{
|
2020-08-28 21:58:18 -03:00
|
|
|
g_assert (meta_cogl_pixel_format_from_drm_format (preferred_formats[k],
|
|
|
|
NULL,
|
|
|
|
NULL));
|
2018-11-21 15:25:07 +02:00
|
|
|
|
|
|
|
for (i = 0; i < formats->len; i++)
|
|
|
|
{
|
|
|
|
drm_format = g_array_index (formats, uint32_t, i);
|
|
|
|
|
|
|
|
if (drm_format == preferred_formats[k])
|
|
|
|
return drm_format;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Otherwise just pick an arbitrary format we recognize. The formats
|
|
|
|
* list is not in any specific order and we don't know any better
|
|
|
|
* either.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < formats->len; i++)
|
|
|
|
{
|
|
|
|
drm_format = g_array_index (formats, uint32_t, i);
|
|
|
|
|
2020-08-28 21:58:18 -03:00
|
|
|
if (meta_cogl_pixel_format_from_drm_format (drm_format, NULL, NULL))
|
2018-11-21 15:25:07 +02:00
|
|
|
return drm_format;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DRM_FORMAT_INVALID;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static gboolean
|
|
|
|
init_secondary_gpu_state_cpu_copy_mode (MetaRendererNative *renderer_native,
|
|
|
|
CoglOnscreen *onscreen,
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaGpuKms *gpu_kms;
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaKmsDevice *kms_device;
|
2017-07-25 10:21:04 +08:00
|
|
|
int width, height;
|
|
|
|
unsigned int i;
|
2018-11-21 15:25:07 +02:00
|
|
|
uint32_t drm_format;
|
2018-11-22 16:26:01 +02:00
|
|
|
MetaDrmFormatBuf tmp;
|
2018-11-21 15:25:07 +02:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
drm_format = pick_secondary_gpu_framebuffer_format_for_cpu (onscreen);
|
2018-11-21 15:25:07 +02:00
|
|
|
if (drm_format == DRM_FORMAT_INVALID)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"Could not find a suitable pixel format in CPU copy mode");
|
|
|
|
return FALSE;
|
|
|
|
}
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
width = cogl_framebuffer_get_width (framebuffer);
|
|
|
|
height = cogl_framebuffer_get_height (framebuffer);
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
gpu_kms = META_GPU_KMS (meta_crtc_get_gpu (onscreen_native->crtc));
|
2020-09-29 09:40:53 +02:00
|
|
|
kms_device = meta_gpu_kms_get_kms_device (gpu_kms);
|
2018-11-22 16:26:01 +02:00
|
|
|
g_debug ("Secondary GPU %s using DRM format '%s' (0x%x) for a %dx%d output.",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms),
|
|
|
|
meta_drm_format_to_string (&tmp, drm_format),
|
|
|
|
drm_format,
|
|
|
|
width, height);
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
secondary_gpu_state = g_new0 (MetaOnscreenNativeSecondaryGpuState, 1);
|
|
|
|
secondary_gpu_state->renderer_gpu_data = renderer_gpu_data;
|
|
|
|
secondary_gpu_state->gpu_kms = gpu_kms;
|
|
|
|
secondary_gpu_state->egl_surface = EGL_NO_SURFACE;
|
|
|
|
|
|
|
|
for (i = 0; i < G_N_ELEMENTS (secondary_gpu_state->cpu.dumb_fbs); i++)
|
|
|
|
{
|
2020-09-29 09:40:53 +02:00
|
|
|
secondary_gpu_state->cpu.dumb_fbs[i] =
|
|
|
|
meta_drm_buffer_dumb_new (kms_device,
|
|
|
|
width, height,
|
|
|
|
drm_format,
|
|
|
|
error);
|
|
|
|
if (!secondary_gpu_state->cpu.dumb_fbs[i])
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
|
|
|
secondary_gpu_state_free (secondary_gpu_state);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-24 17:07:14 +03:00
|
|
|
/*
|
|
|
|
* This function initializes everything needed for
|
|
|
|
* META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO as well.
|
|
|
|
*/
|
|
|
|
secondary_gpu_state->import_status =
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_NONE;
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
onscreen_native->secondary_gpu_state = secondary_gpu_state;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
init_secondary_gpu_state (MetaRendererNative *renderer_native,
|
|
|
|
CoglOnscreen *onscreen,
|
|
|
|
GError **error)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaGpu *gpu = meta_crtc_get_gpu (onscreen_native->crtc);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
2020-01-17 21:08:02 +01:00
|
|
|
META_GPU_KMS (gpu));
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
switch (renderer_gpu_data->secondary.copy_mode)
|
|
|
|
{
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_SECONDARY_GPU:
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!init_secondary_gpu_state_gpu_copy_mode (renderer_native,
|
|
|
|
onscreen,
|
|
|
|
renderer_gpu_data,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
|
|
|
break;
|
2019-05-24 17:07:14 +03:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO:
|
|
|
|
/*
|
|
|
|
* Initialize also the primary copy mode, so that if zero-copy
|
|
|
|
* path fails, which is quite likely, we can simply continue
|
|
|
|
* with the primary copy path on the very first frame.
|
|
|
|
*/
|
|
|
|
G_GNUC_FALLTHROUGH;
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_PRIMARY:
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!init_secondary_gpu_state_cpu_copy_mode (renderer_native,
|
|
|
|
onscreen,
|
|
|
|
renderer_gpu_data,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
static void
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_disconnect (CoglRenderer *cogl_renderer)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
g_slice_free (CoglRendererEGL, cogl_renderer_egl);
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
2020-01-17 21:08:02 +01:00
|
|
|
free_current_secondary_bo (CoglOnscreen *onscreen)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
|
|
|
|
|
|
|
secondary_gpu_state = onscreen_native->secondary_gpu_state;
|
|
|
|
if (!secondary_gpu_state)
|
|
|
|
return;
|
|
|
|
|
2019-09-16 16:39:02 +03:00
|
|
|
g_clear_object (&secondary_gpu_state->gbm.current_fb);
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
static void
|
|
|
|
free_current_bo (CoglOnscreen *onscreen)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2018-01-30 17:47:17 +08:00
|
|
|
g_clear_object (&onscreen_native->gbm.current_fb);
|
2020-01-17 21:08:02 +01:00
|
|
|
free_current_secondary_bo (onscreen);
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2020-10-21 11:36:40 +02:00
|
|
|
meta_onscreen_native_notify_frame_complete (CoglOnscreen *onscreen)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
CoglFrameInfo *info;
|
2016-12-05 19:18:24 +01:00
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
info = cogl_onscreen_pop_head_frame_info (onscreen);
|
|
|
|
|
|
|
|
g_assert (!cogl_onscreen_peek_head_frame_info (onscreen));
|
2016-11-23 22:46:26 +08:00
|
|
|
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
_cogl_onscreen_notify_frame_sync (onscreen, info);
|
|
|
|
_cogl_onscreen_notify_complete (onscreen, info);
|
|
|
|
cogl_object_unref (info);
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean
|
2016-05-27 15:13:09 +08:00
|
|
|
meta_renderer_native_connect (CoglRenderer *cogl_renderer,
|
2016-08-18 10:21:12 +08:00
|
|
|
GError **error)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaGpuKms *gpu_kms = cogl_renderer->custom_winsys_user_data;
|
|
|
|
MetaRendererNative *renderer_native = meta_renderer_native_from_gpu (gpu_kms);
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
cogl_renderer->winsys = g_slice_new0 (CoglRendererEGL);
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_renderer_egl = cogl_renderer->winsys;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
gpu_kms);
|
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_renderer_egl->platform_vtable = &_cogl_winsys_egl_vtable;
|
2017-07-25 10:21:04 +08:00
|
|
|
cogl_renderer_egl->platform = renderer_gpu_data;
|
|
|
|
cogl_renderer_egl->edpy = renderer_gpu_data->egl_display;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
if (!_cogl_winsys_egl_renderer_connect_common (cogl_renderer, error))
|
|
|
|
goto fail;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
fail:
|
2016-05-27 15:13:09 +08:00
|
|
|
meta_renderer_native_disconnect (cogl_renderer);
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-09-26 15:54:27 +08:00
|
|
|
static int
|
2020-10-12 23:31:38 +02:00
|
|
|
meta_renderer_native_add_egl_config_attributes (CoglDisplay *cogl_display,
|
|
|
|
const CoglFramebufferConfig *config,
|
|
|
|
EGLint *attributes)
|
2016-09-26 15:54:27 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_display->renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
2016-09-26 15:54:27 +08:00
|
|
|
int i = 0;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
attributes[i++] = EGL_SURFACE_TYPE;
|
|
|
|
attributes[i++] = EGL_WINDOW_BIT;
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
attributes[i++] = EGL_SURFACE_TYPE;
|
|
|
|
attributes[i++] = EGL_STREAM_BIT_KHR;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
2016-09-26 15:54:27 +08:00
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2018-02-23 22:14:07 +08:00
|
|
|
static gboolean
|
|
|
|
choose_egl_config_from_gbm_format (MetaEgl *egl,
|
|
|
|
EGLDisplay egl_display,
|
|
|
|
const EGLint *attributes,
|
|
|
|
uint32_t gbm_format,
|
|
|
|
EGLConfig *out_config,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
EGLConfig *egl_configs;
|
|
|
|
EGLint n_configs;
|
|
|
|
EGLint i;
|
|
|
|
|
|
|
|
egl_configs = meta_egl_choose_all_configs (egl, egl_display,
|
|
|
|
attributes,
|
|
|
|
&n_configs,
|
|
|
|
error);
|
|
|
|
if (!egl_configs)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
for (i = 0; i < n_configs; i++)
|
|
|
|
{
|
|
|
|
EGLint visual_id;
|
|
|
|
|
|
|
|
if (!meta_egl_get_config_attrib (egl, egl_display,
|
|
|
|
egl_configs[i],
|
|
|
|
EGL_NATIVE_VISUAL_ID,
|
|
|
|
&visual_id,
|
|
|
|
error))
|
|
|
|
{
|
|
|
|
g_free (egl_configs);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((uint32_t) visual_id == gbm_format)
|
|
|
|
{
|
|
|
|
*out_config = egl_configs[i];
|
|
|
|
g_free (egl_configs);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_free (egl_configs);
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"No EGL config matching supported GBM format found");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_renderer_native_choose_egl_config (CoglDisplay *cogl_display,
|
|
|
|
EGLint *attributes,
|
|
|
|
EGLConfig *out_config,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaEgl *egl = meta_backend_get_egl (backend);
|
2018-06-07 16:29:44 -07:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
2018-02-23 22:14:07 +08:00
|
|
|
EGLDisplay egl_display = cogl_renderer_egl->edpy;
|
|
|
|
|
2018-06-07 16:29:44 -07:00
|
|
|
switch (renderer_gpu_data->mode)
|
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
return choose_egl_config_from_gbm_format (egl,
|
|
|
|
egl_display,
|
|
|
|
attributes,
|
|
|
|
GBM_FORMAT_XRGB8888,
|
|
|
|
out_config,
|
|
|
|
error);
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
return meta_egl_choose_first_config (egl,
|
|
|
|
egl_display,
|
|
|
|
attributes,
|
|
|
|
out_config,
|
|
|
|
error);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
2018-02-23 22:14:07 +08:00
|
|
|
}
|
|
|
|
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_setup_egl_display (CoglDisplay *cogl_display,
|
2016-08-18 10:21:12 +08:00
|
|
|
GError **error)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_display->renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_display_egl->platform = renderer_native;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
/* Force a full modeset / drmModeSetCrtc on
|
|
|
|
* the first swap buffers call.
|
|
|
|
*/
|
2016-06-08 17:05:31 +08:00
|
|
|
meta_renderer_native_queue_modes_reset (renderer_native);
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_destroy_egl_display (CoglDisplay *cogl_display)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-08-31 11:25:45 +08:00
|
|
|
static EGLSurface
|
|
|
|
create_dummy_pbuffer_surface (EGLDisplay egl_display,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaBackend *backend = meta_get_backend ();
|
|
|
|
MetaEgl *egl = meta_backend_get_egl (backend);
|
|
|
|
EGLConfig pbuffer_config;
|
|
|
|
static const EGLint pbuffer_config_attribs[] = {
|
|
|
|
EGL_SURFACE_TYPE, EGL_PBUFFER_BIT,
|
|
|
|
EGL_RED_SIZE, 1,
|
|
|
|
EGL_GREEN_SIZE, 1,
|
|
|
|
EGL_BLUE_SIZE, 1,
|
|
|
|
EGL_ALPHA_SIZE, 0,
|
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES2_BIT,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
static const EGLint pbuffer_attribs[] = {
|
|
|
|
EGL_WIDTH, 16,
|
|
|
|
EGL_HEIGHT, 16,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
2018-02-23 22:13:11 +08:00
|
|
|
if (!meta_egl_choose_first_config (egl, egl_display, pbuffer_config_attribs,
|
|
|
|
&pbuffer_config, error))
|
2016-08-31 11:25:45 +08:00
|
|
|
return EGL_NO_SURFACE;
|
|
|
|
|
|
|
|
return meta_egl_create_pbuffer_surface (egl, egl_display,
|
|
|
|
pbuffer_config, pbuffer_attribs,
|
|
|
|
error);
|
|
|
|
}
|
|
|
|
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_egl_context_created (CoglDisplay *cogl_display,
|
2016-08-18 10:21:12 +08:00
|
|
|
GError **error)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
2016-06-01 17:53:07 +08:00
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
if ((cogl_renderer_egl->private_features &
|
2016-05-09 19:51:29 +08:00
|
|
|
COGL_EGL_WINSYS_FEATURE_SURFACELESS_CONTEXT) == 0)
|
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_display_egl->dummy_surface =
|
|
|
|
create_dummy_pbuffer_surface (cogl_renderer_egl->edpy, error);
|
|
|
|
if (cogl_display_egl->dummy_surface == EGL_NO_SURFACE)
|
2016-08-31 11:25:45 +08:00
|
|
|
return FALSE;
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2016-06-01 17:53:07 +08:00
|
|
|
if (!_cogl_winsys_egl_make_current (cogl_display,
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_display_egl->dummy_surface,
|
|
|
|
cogl_display_egl->dummy_surface,
|
|
|
|
cogl_display_egl->egl_context))
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2019-06-17 18:18:12 +02:00
|
|
|
g_set_error (error, COGL_WINSYS_ERROR,
|
2016-05-09 19:51:29 +08:00
|
|
|
COGL_WINSYS_ERROR_CREATE_CONTEXT,
|
|
|
|
"Failed to make context current");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_egl_cleanup_context (CoglDisplay *cogl_display)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
2016-06-01 17:53:07 +08:00
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
2017-07-24 16:19:55 +08:00
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
if (cogl_display_egl->dummy_surface != EGL_NO_SURFACE)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-24 16:19:55 +08:00
|
|
|
meta_egl_destroy_surface (egl,
|
|
|
|
cogl_renderer_egl->edpy,
|
|
|
|
cogl_display_egl->dummy_surface,
|
|
|
|
NULL);
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_display_egl->dummy_surface = EGL_NO_SURFACE;
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
2020-01-17 21:08:02 +01:00
|
|
|
swap_secondary_drm_fb (CoglOnscreen *onscreen)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
|
|
|
|
|
|
|
secondary_gpu_state = onscreen_native->secondary_gpu_state;
|
|
|
|
if (!secondary_gpu_state)
|
|
|
|
return;
|
|
|
|
|
2018-01-30 17:47:17 +08:00
|
|
|
g_set_object (&secondary_gpu_state->gbm.current_fb,
|
|
|
|
secondary_gpu_state->gbm.next_fb);
|
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2016-05-12 14:55:06 +08:00
|
|
|
static void
|
2016-06-08 17:05:31 +08:00
|
|
|
meta_onscreen_native_swap_drm_fb (CoglOnscreen *onscreen)
|
2016-05-12 14:55:06 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2016-05-12 14:55:06 +08:00
|
|
|
|
2020-10-10 01:22:54 +02:00
|
|
|
if (!onscreen_native->gbm.next_fb)
|
|
|
|
return;
|
|
|
|
|
2016-05-12 14:55:06 +08:00
|
|
|
free_current_bo (onscreen);
|
|
|
|
|
2018-01-30 17:47:17 +08:00
|
|
|
g_set_object (&onscreen_native->gbm.current_fb, onscreen_native->gbm.next_fb);
|
|
|
|
g_clear_object (&onscreen_native->gbm.next_fb);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
swap_secondary_drm_fb (onscreen);
|
2016-06-08 17:05:31 +08:00
|
|
|
}
|
2016-05-12 14:55:06 +08:00
|
|
|
|
2020-02-27 00:08:58 +01:00
|
|
|
static void
|
|
|
|
maybe_update_frame_info (MetaCrtc *crtc,
|
|
|
|
CoglFrameInfo *frame_info,
|
|
|
|
int64_t time_ns)
|
|
|
|
{
|
|
|
|
const MetaCrtcConfig *crtc_config;
|
|
|
|
const MetaCrtcModeInfo *crtc_mode_info;
|
|
|
|
float refresh_rate;
|
|
|
|
|
|
|
|
g_return_if_fail (crtc);
|
|
|
|
|
|
|
|
crtc_config = meta_crtc_get_config (crtc);
|
|
|
|
if (!crtc_config)
|
|
|
|
return;
|
|
|
|
|
|
|
|
crtc_mode_info = meta_crtc_mode_get_info (crtc_config->mode);
|
|
|
|
refresh_rate = crtc_mode_info->refresh_rate;
|
|
|
|
if (refresh_rate >= frame_info->refresh_rate)
|
|
|
|
{
|
|
|
|
frame_info->presentation_time = time_ns;
|
|
|
|
frame_info->refresh_rate = refresh_rate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
static void
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
notify_view_crtc_presented (MetaRendererView *view,
|
|
|
|
MetaKmsCrtc *kms_crtc,
|
|
|
|
int64_t time_ns)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
|
|
|
ClutterStageView *stage_view = CLUTTER_STAGE_VIEW (view);
|
|
|
|
CoglFramebuffer *framebuffer =
|
2016-08-01 02:44:57 +02:00
|
|
|
clutter_stage_view_get_onscreen (stage_view);
|
2016-06-08 17:05:31 +08:00
|
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
2018-06-27 17:19:27 +08:00
|
|
|
CoglFrameInfo *frame_info;
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaCrtc *crtc;
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2018-06-27 17:19:27 +08:00
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
frame_info = cogl_onscreen_peek_head_frame_info (onscreen);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
crtc = META_CRTC (meta_crtc_kms_from_kms_crtc (kms_crtc));
|
2020-02-27 00:08:58 +01:00
|
|
|
maybe_update_frame_info (crtc, frame_info, time_ns);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2020-10-21 11:36:40 +02:00
|
|
|
meta_onscreen_native_notify_frame_complete (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
renderer_gpu_data =
|
|
|
|
meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
onscreen_native->render_gpu);
|
|
|
|
switch (renderer_gpu_data->mode)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
meta_onscreen_native_swap_drm_fb (onscreen);
|
|
|
|
break;
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
clutter: Paint views with individual frame clocks
Replace the default master clock with multiple frame clocks, each
driving its own stage view. As each stage view represents one CRTC, this
means we draw each CRTC with its own designated frame clock,
disconnected from all the others.
For example this means we when using the native backend will never need
to wait for one monitor to vsync before painting another, so e.g. having
a 144 Hz monitor next to a 60 Hz monitor, things including both Wayland
and X11 applications and shell UI will be able to render at the
corresponding monitor refresh rate.
This also changes a warning about missed frames when sending
_NETWM_FRAME_TIMINGS messages to a debug log entry, as it's expected
that we'll start missing frames e.g. when a X11 window (via Xwayland) is
exclusively within a stage view that was not painted, while another one
was, still increasing the global frame clock.
Addititonally, this also requires the X11 window actor to schedule
timeouts for _NET_WM_FRAME_DRAWN/_NET_WM_FRAME_TIMINGS event emitting,
if the actor wasn't on any stage views, as now we'll only get the frame
callbacks on actors when they actually were painted, while in the past,
we'd invoke that vfunc when anything was painted.
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/903
Closes: https://gitlab.gnome.org/GNOME/mutter/-/issues/3
https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1285
2020-05-30 00:27:56 +02:00
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
break;
|
2016-08-18 11:28:59 +08:00
|
|
|
#endif
|
2016-06-08 17:05:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
static int64_t
|
|
|
|
timeval_to_nanoseconds (const struct timeval *tv)
|
|
|
|
{
|
|
|
|
int64_t usec = ((int64_t) tv->tv_sec) * G_USEC_PER_SEC + tv->tv_usec;
|
|
|
|
int64_t nsec = usec * 1000;
|
|
|
|
|
|
|
|
return nsec;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
page_flip_feedback_flipped (MetaKmsCrtc *kms_crtc,
|
|
|
|
unsigned int sequence,
|
|
|
|
unsigned int tv_sec,
|
|
|
|
unsigned int tv_usec,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = user_data;
|
|
|
|
struct timeval page_flip_time;
|
|
|
|
|
|
|
|
page_flip_time = (struct timeval) {
|
|
|
|
.tv_sec = tv_sec,
|
|
|
|
.tv_usec = tv_usec,
|
|
|
|
};
|
|
|
|
|
|
|
|
notify_view_crtc_presented (view, kms_crtc,
|
|
|
|
timeval_to_nanoseconds (&page_flip_time));
|
|
|
|
}
|
|
|
|
|
2020-10-10 00:52:11 +02:00
|
|
|
static void
|
|
|
|
page_flip_feedback_ready (MetaKmsCrtc *kms_crtc,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = user_data;
|
|
|
|
CoglFramebuffer *framebuffer =
|
|
|
|
clutter_stage_view_get_onscreen (CLUTTER_STAGE_VIEW (view));
|
|
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
|
|
|
CoglFrameInfo *frame_info;
|
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
frame_info = cogl_onscreen_peek_head_frame_info (onscreen);
|
2020-10-10 00:52:11 +02:00
|
|
|
frame_info->flags |= COGL_FRAME_INFO_FLAG_SYMBOLIC;
|
|
|
|
|
|
|
|
meta_onscreen_native_notify_frame_complete (onscreen);
|
|
|
|
}
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
static void
|
|
|
|
page_flip_feedback_mode_set_fallback (MetaKmsCrtc *kms_crtc,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = user_data;
|
|
|
|
MetaCrtc *crtc;
|
|
|
|
MetaGpuKms *gpu_kms;
|
|
|
|
int64_t now_ns;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* We ended up not page flipping, thus we don't have a presentation time to
|
|
|
|
* use. Lets use the next best thing: the current time.
|
|
|
|
*/
|
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
crtc = META_CRTC (meta_crtc_kms_from_kms_crtc (kms_crtc));
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
gpu_kms = META_GPU_KMS (meta_crtc_get_gpu (crtc));
|
|
|
|
now_ns = meta_gpu_kms_get_current_time_ns (gpu_kms);
|
|
|
|
|
|
|
|
notify_view_crtc_presented (view, kms_crtc, now_ns);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
page_flip_feedback_discarded (MetaKmsCrtc *kms_crtc,
|
|
|
|
gpointer user_data,
|
|
|
|
const GError *error)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = user_data;
|
|
|
|
MetaCrtc *crtc;
|
|
|
|
MetaGpuKms *gpu_kms;
|
|
|
|
int64_t now_ns;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Page flipping failed, but we want to fail gracefully, so to avoid freezing
|
|
|
|
* the frame clack, pretend we flipped.
|
|
|
|
*/
|
|
|
|
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
if (error &&
|
|
|
|
!g_error_matches (error,
|
|
|
|
G_IO_ERROR,
|
|
|
|
G_IO_ERROR_PERMISSION_DENIED))
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
g_warning ("Page flip discarded: %s", error->message);
|
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
crtc = META_CRTC (meta_crtc_kms_from_kms_crtc (kms_crtc));
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
gpu_kms = META_GPU_KMS (meta_crtc_get_gpu (crtc));
|
|
|
|
now_ns = meta_gpu_kms_get_current_time_ns (gpu_kms);
|
|
|
|
|
|
|
|
notify_view_crtc_presented (view, kms_crtc, now_ns);
|
|
|
|
}
|
|
|
|
|
2020-10-02 16:06:35 +02:00
|
|
|
static const MetaKmsPageFlipListenerVtable page_flip_listener_vtable = {
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
.flipped = page_flip_feedback_flipped,
|
2020-10-10 00:52:11 +02:00
|
|
|
.ready = page_flip_feedback_ready,
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
.mode_set_fallback = page_flip_feedback_mode_set_fallback,
|
|
|
|
.discarded = page_flip_feedback_discarded,
|
|
|
|
};
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
static int
|
|
|
|
custom_egl_stream_page_flip (gpointer custom_page_flip_data,
|
|
|
|
gpointer user_data)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = custom_page_flip_data;
|
|
|
|
MetaRendererView *view = user_data;
|
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2017-07-25 10:21:04 +08:00
|
|
|
EGLDisplay *egl_display;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLAttrib *acquire_attribs;
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
g_autoptr (GError) error = NULL;
|
2017-11-14 16:08:52 +08:00
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
acquire_attribs = (EGLAttrib[]) {
|
|
|
|
EGL_DRM_FLIP_EVENT_DATA_NV,
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
(EGLAttrib) view,
|
2016-08-18 11:28:59 +08:00
|
|
|
EGL_NONE
|
|
|
|
};
|
2017-02-01 18:15:53 -08:00
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
renderer_gpu_data =
|
|
|
|
meta_renderer_native_get_gpu_data (onscreen_native->renderer_native,
|
|
|
|
onscreen_native->render_gpu);
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
egl_display = renderer_gpu_data->egl_display;
|
2016-08-18 11:28:59 +08:00
|
|
|
if (!meta_egl_stream_consumer_acquire_attrib (egl,
|
2017-07-25 10:21:04 +08:00
|
|
|
egl_display,
|
2016-08-18 11:28:59 +08:00
|
|
|
onscreen_native->egl.stream,
|
|
|
|
acquire_attribs,
|
|
|
|
&error))
|
|
|
|
{
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
if (g_error_matches (error, META_EGL_ERROR, EGL_RESOURCE_BUSY_EXT))
|
|
|
|
return -EBUSY;
|
|
|
|
else
|
|
|
|
return -EINVAL;
|
2016-08-18 11:28:59 +08:00
|
|
|
}
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
return 0;
|
2016-08-18 11:28:59 +08:00
|
|
|
}
|
|
|
|
#endif /* HAVE_EGL_DEVICE */
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
static void
|
|
|
|
dummy_power_save_page_flip (CoglOnscreen *onscreen)
|
2019-02-12 19:10:59 +01:00
|
|
|
{
|
2020-10-21 19:40:08 +02:00
|
|
|
CoglFrameInfo *frame_info;
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
meta_onscreen_native_swap_drm_fb (onscreen);
|
2020-10-21 19:40:08 +02:00
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
frame_info = cogl_onscreen_peek_tail_frame_info (onscreen);
|
2020-10-21 19:40:08 +02:00
|
|
|
frame_info->flags |= COGL_FRAME_INFO_FLAG_SYMBOLIC;
|
2020-10-21 11:36:40 +02:00
|
|
|
meta_onscreen_native_notify_frame_complete (onscreen);
|
2019-02-12 19:10:59 +01:00
|
|
|
}
|
|
|
|
|
2019-03-25 10:24:46 +01:00
|
|
|
static gboolean
|
|
|
|
dummy_power_save_page_flip_cb (gpointer user_data)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = user_data;
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
g_list_foreach (renderer_native->power_save_page_flip_onscreens,
|
|
|
|
(GFunc) dummy_power_save_page_flip, NULL);
|
|
|
|
g_list_free_full (renderer_native->power_save_page_flip_onscreens,
|
2020-10-18 11:22:31 +02:00
|
|
|
g_object_unref);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
renderer_native->power_save_page_flip_onscreens = NULL;
|
2019-03-25 10:24:46 +01:00
|
|
|
renderer_native->power_save_page_flip_source_id = 0;
|
|
|
|
|
|
|
|
return G_SOURCE_REMOVE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
queue_dummy_power_save_page_flip (CoglOnscreen *onscreen)
|
2019-03-25 10:24:46 +01:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
2019-03-25 10:24:46 +01:00
|
|
|
const unsigned int timeout_ms = 100;
|
|
|
|
|
|
|
|
if (!renderer_native->power_save_page_flip_source_id)
|
|
|
|
{
|
|
|
|
renderer_native->power_save_page_flip_source_id =
|
|
|
|
g_timeout_add (timeout_ms,
|
|
|
|
dummy_power_save_page_flip_cb,
|
|
|
|
renderer_native);
|
|
|
|
}
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
renderer_native->power_save_page_flip_onscreens =
|
|
|
|
g_list_prepend (renderer_native->power_save_page_flip_onscreens,
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_ref (onscreen));
|
2019-02-12 19:10:59 +01:00
|
|
|
}
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
static void
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
meta_onscreen_native_flip_crtc (CoglOnscreen *onscreen,
|
|
|
|
MetaRendererView *view,
|
|
|
|
MetaCrtc *crtc,
|
|
|
|
MetaKmsPageFlipListenerFlag flags)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
|
|
|
MetaGpuKms *render_gpu = onscreen_native->render_gpu;
|
2020-02-26 19:47:44 +01:00
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (crtc);
|
2020-10-02 16:06:35 +02:00
|
|
|
MetaKmsCrtc *kms_crtc = meta_crtc_kms_get_kms_crtc (crtc_kms);
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaGpuKms *gpu_kms;
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaKmsDevice *kms_device;
|
|
|
|
MetaKms *kms;
|
|
|
|
MetaKmsUpdate *kms_update;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state = NULL;
|
2020-09-21 19:32:46 +02:00
|
|
|
MetaDrmBuffer *buffer;
|
2016-06-08 17:05:31 +08:00
|
|
|
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaOnscreenNativeFlipCrtcs,
|
|
|
|
"Onscreen (flip CRTCs)");
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
gpu_kms = META_GPU_KMS (meta_crtc_get_gpu (crtc));
|
2020-07-16 23:38:10 +02:00
|
|
|
kms_device = meta_gpu_kms_get_kms_device (gpu_kms);
|
|
|
|
kms = meta_kms_device_get_kms (kms_device);
|
|
|
|
kms_update = meta_kms_ensure_pending_update (kms, kms_device);
|
2019-03-25 10:24:46 +01:00
|
|
|
|
|
|
|
g_assert (meta_gpu_kms_is_crtc_active (gpu_kms, crtc));
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
2017-07-25 10:21:04 +08:00
|
|
|
render_gpu);
|
2017-07-10 18:19:32 +08:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
2017-07-25 10:21:04 +08:00
|
|
|
if (gpu_kms == render_gpu)
|
|
|
|
{
|
2020-09-21 19:32:46 +02:00
|
|
|
buffer = onscreen_native->gbm.next_fb;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-01-17 21:08:02 +01:00
|
|
|
secondary_gpu_state = onscreen_native->secondary_gpu_state;
|
2020-09-21 19:32:46 +02:00
|
|
|
buffer = secondary_gpu_state->gbm.next_fb;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2020-09-21 19:32:46 +02:00
|
|
|
meta_crtc_kms_assign_primary_plane (crtc_kms, buffer, kms_update);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
2020-10-01 18:28:13 +02:00
|
|
|
meta_kms_update_set_custom_page_flip (kms_update,
|
|
|
|
custom_egl_stream_page_flip,
|
|
|
|
onscreen_native);
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
2020-10-01 18:28:13 +02:00
|
|
|
|
2020-10-02 16:06:35 +02:00
|
|
|
meta_kms_update_add_page_flip_listener (kms_update,
|
|
|
|
kms_crtc,
|
|
|
|
&page_flip_listener_vtable,
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
flags,
|
|
|
|
g_object_ref (view),
|
|
|
|
g_object_unref);
|
2019-02-12 19:10:59 +01:00
|
|
|
}
|
2017-03-28 12:17:38 +08:00
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
static void
|
2020-01-17 21:08:02 +01:00
|
|
|
meta_onscreen_native_set_crtc_mode (CoglOnscreen *onscreen,
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-02-26 19:47:44 +01:00
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (onscreen_native->crtc);
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaKmsCrtc *kms_crtc = meta_crtc_kms_get_kms_crtc (crtc_kms);
|
|
|
|
MetaKmsDevice *kms_device = meta_kms_crtc_get_device (kms_crtc);
|
|
|
|
MetaKms *kms = meta_kms_device_get_kms (kms_device);
|
|
|
|
MetaKmsUpdate *kms_update;
|
2020-01-17 21:08:02 +01:00
|
|
|
|
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaOnscreenNativeSetCrtcModes,
|
|
|
|
"Onscreen (set CRTC modes)");
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2020-07-16 23:38:10 +02:00
|
|
|
kms_update = meta_kms_ensure_pending_update (kms, kms_device);
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
{
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBuffer *buffer;
|
2017-03-28 12:17:38 +08:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
buffer = META_DRM_BUFFER (onscreen_native->egl.dumb_fb);
|
2020-09-21 19:32:46 +02:00
|
|
|
meta_crtc_kms_assign_primary_plane (crtc_kms, buffer, kms_update);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
2016-06-08 17:05:31 +08:00
|
|
|
}
|
|
|
|
|
2020-02-26 19:47:44 +01:00
|
|
|
meta_crtc_kms_set_mode (crtc_kms, kms_update);
|
2020-02-26 16:47:03 +01:00
|
|
|
meta_output_kms_set_underscan (META_OUTPUT_KMS (onscreen_native->output),
|
|
|
|
kms_update);
|
2017-03-28 12:17:38 +08:00
|
|
|
}
|
|
|
|
|
2019-05-24 17:07:14 +03:00
|
|
|
static gboolean
|
|
|
|
import_shared_framebuffer (CoglOnscreen *onscreen,
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-05-24 17:07:14 +03:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaGpuKms *gpu_kms;
|
|
|
|
MetaKmsDevice *kms_device;
|
|
|
|
struct gbm_device *gbm_device;
|
2019-05-24 17:07:14 +03:00
|
|
|
MetaDrmBufferGbm *buffer_gbm;
|
|
|
|
MetaDrmBufferImport *buffer_import;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
|
|
|
|
buffer_gbm = META_DRM_BUFFER_GBM (onscreen_native->gbm.next_fb);
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
gpu_kms = secondary_gpu_state->gpu_kms;
|
|
|
|
kms_device = meta_gpu_kms_get_kms_device (gpu_kms);
|
|
|
|
gbm_device = meta_gbm_device_from_gpu (gpu_kms);
|
|
|
|
buffer_import = meta_drm_buffer_import_new (kms_device,
|
|
|
|
gbm_device,
|
2019-05-24 17:07:14 +03:00
|
|
|
buffer_gbm,
|
|
|
|
&error);
|
|
|
|
if (!buffer_import)
|
|
|
|
{
|
|
|
|
g_debug ("Zero-copy disabled for %s, meta_drm_buffer_import_new failed: %s",
|
|
|
|
meta_gpu_kms_get_file_path (secondary_gpu_state->gpu_kms),
|
|
|
|
error->message);
|
|
|
|
|
|
|
|
g_warn_if_fail (secondary_gpu_state->import_status ==
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_NONE);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Fall back. If META_SHARED_FRAMEBUFFER_IMPORT_STATUS_NONE is
|
|
|
|
* in effect, we have COPY_MODE_PRIMARY prepared already, so we
|
|
|
|
* simply retry with that path. Import status cannot be FAILED,
|
|
|
|
* because we should not retry if failed once.
|
|
|
|
*
|
|
|
|
* If import status is OK, that is unexpected and we do not
|
|
|
|
* have the fallback path prepared which means this output cannot
|
|
|
|
* work anymore.
|
|
|
|
*/
|
|
|
|
secondary_gpu_state->renderer_gpu_data->secondary.copy_mode =
|
|
|
|
META_SHARED_FRAMEBUFFER_COPY_MODE_PRIMARY;
|
|
|
|
|
|
|
|
secondary_gpu_state->import_status =
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_FAILED;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* next_fb may already contain a fallback buffer, so clear it only
|
|
|
|
* when we are sure to succeed.
|
|
|
|
*/
|
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
|
|
|
secondary_gpu_state->gbm.next_fb = META_DRM_BUFFER (buffer_import);
|
|
|
|
|
|
|
|
if (secondary_gpu_state->import_status ==
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_NONE)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Clean up the cpu-copy part of
|
|
|
|
* init_secondary_gpu_state_cpu_copy_mode ()
|
|
|
|
*/
|
|
|
|
secondary_gpu_release_dumb (secondary_gpu_state);
|
|
|
|
|
|
|
|
g_debug ("Using zero-copy for %s succeeded once.",
|
|
|
|
meta_gpu_kms_get_file_path (secondary_gpu_state->gpu_kms));
|
|
|
|
}
|
|
|
|
|
|
|
|
secondary_gpu_state->import_status =
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_OK;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
|
|
|
copy_shared_framebuffer_gpu (CoglOnscreen *onscreen,
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state,
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data,
|
|
|
|
gboolean *egl_context_changed)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
|
|
|
GError *error = NULL;
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaKmsDevice *kms_device;
|
2019-05-20 22:08:35 +02:00
|
|
|
MetaDrmBufferGbm *buffer_gbm;
|
|
|
|
struct gbm_bo *bo;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2019-06-11 11:16:39 +03:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (CopySharedFramebufferSecondaryGpu,
|
|
|
|
"FB Copy (secondary GPU)");
|
|
|
|
|
2019-12-02 14:00:56 +02:00
|
|
|
g_warn_if_fail (secondary_gpu_state->gbm.next_fb == NULL);
|
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!meta_egl_make_current (egl,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
secondary_gpu_state->egl_surface,
|
|
|
|
secondary_gpu_state->egl_surface,
|
|
|
|
renderer_gpu_data->secondary.egl_context,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to make current: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*egl_context_changed = TRUE;
|
|
|
|
|
2019-05-20 22:08:35 +02:00
|
|
|
buffer_gbm = META_DRM_BUFFER_GBM (onscreen_native->gbm.next_fb);
|
|
|
|
bo = meta_drm_buffer_gbm_get_bo (buffer_gbm);
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!meta_renderer_native_gles3_blit_shared_bo (egl,
|
|
|
|
renderer_native->gles3,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
renderer_gpu_data->secondary.egl_context,
|
|
|
|
secondary_gpu_state->egl_surface,
|
2019-05-20 22:08:35 +02:00
|
|
|
bo,
|
2017-07-25 10:21:04 +08:00
|
|
|
&error))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to blit shared framebuffer: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!meta_egl_swap_buffers (egl,
|
|
|
|
renderer_gpu_data->egl_display,
|
|
|
|
secondary_gpu_state->egl_surface,
|
|
|
|
&error))
|
|
|
|
{
|
|
|
|
g_warning ("Failed to swap buffers: %s", error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
kms_device = meta_gpu_kms_get_kms_device (secondary_gpu_state->gpu_kms);
|
2019-08-20 21:28:19 +02:00
|
|
|
buffer_gbm =
|
2020-09-29 09:40:53 +02:00
|
|
|
meta_drm_buffer_gbm_new_lock_front (kms_device,
|
2019-05-20 22:08:35 +02:00
|
|
|
secondary_gpu_state->gbm.surface,
|
|
|
|
renderer_native->use_modifiers,
|
|
|
|
&error);
|
2019-09-16 15:24:52 +03:00
|
|
|
if (!buffer_gbm)
|
2018-01-30 17:47:17 +08:00
|
|
|
{
|
2019-08-20 21:28:19 +02:00
|
|
|
g_warning ("meta_drm_buffer_gbm_new_lock_front failed: %s",
|
2018-01-30 17:47:17 +08:00
|
|
|
error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
return;
|
|
|
|
}
|
2019-09-16 15:24:52 +03:00
|
|
|
|
|
|
|
secondary_gpu_state->gbm.next_fb = META_DRM_BUFFER (buffer_gbm);
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
static MetaDrmBufferDumb *
|
2018-11-29 13:54:31 +02:00
|
|
|
secondary_gpu_get_next_dumb_buffer (MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state)
|
|
|
|
{
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBufferDumb *current_dumb_fb;
|
2018-11-29 13:54:31 +02:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
current_dumb_fb = secondary_gpu_state->cpu.current_dumb_fb;
|
|
|
|
if (current_dumb_fb == secondary_gpu_state->cpu.dumb_fbs[0])
|
|
|
|
return secondary_gpu_state->cpu.dumb_fbs[1];
|
2018-11-29 13:54:31 +02:00
|
|
|
else
|
2020-09-29 09:40:53 +02:00
|
|
|
return secondary_gpu_state->cpu.dumb_fbs[0];
|
2018-11-29 13:54:31 +02:00
|
|
|
}
|
|
|
|
|
2019-12-09 10:04:56 -03:00
|
|
|
static CoglContext *
|
|
|
|
cogl_context_from_renderer_native (MetaRendererNative *renderer_native)
|
2018-11-29 13:34:40 +02:00
|
|
|
{
|
2019-12-09 10:04:56 -03:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
ClutterBackend *clutter_backend = meta_backend_get_clutter_backend (backend);
|
|
|
|
|
|
|
|
return clutter_backend_get_cogl_context (clutter_backend);
|
|
|
|
}
|
|
|
|
|
|
|
|
static CoglFramebuffer *
|
|
|
|
create_dma_buf_framebuffer (MetaRendererNative *renderer_native,
|
|
|
|
int dmabuf_fd,
|
|
|
|
uint32_t width,
|
|
|
|
uint32_t height,
|
|
|
|
uint32_t stride,
|
|
|
|
uint32_t offset,
|
|
|
|
uint64_t modifier,
|
|
|
|
uint32_t drm_format,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglContext *cogl_context =
|
|
|
|
cogl_context_from_renderer_native (renderer_native);
|
2018-11-29 13:34:40 +02:00
|
|
|
CoglDisplay *cogl_display = cogl_context->display;
|
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
EGLDisplay egl_display = cogl_renderer_egl->edpy;
|
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
|
|
|
EGLImageKHR egl_image;
|
|
|
|
uint32_t strides[1];
|
|
|
|
uint32_t offsets[1];
|
|
|
|
uint64_t modifiers[1];
|
|
|
|
CoglPixelFormat cogl_format;
|
2019-07-16 18:29:29 +02:00
|
|
|
CoglEglImageFlags flags;
|
2018-11-29 13:34:40 +02:00
|
|
|
CoglTexture2D *cogl_tex;
|
|
|
|
CoglOffscreen *cogl_fbo;
|
|
|
|
int ret;
|
|
|
|
|
2020-08-28 21:58:18 -03:00
|
|
|
ret = meta_cogl_pixel_format_from_drm_format (drm_format,
|
|
|
|
&cogl_format,
|
|
|
|
NULL);
|
2018-11-29 13:34:40 +02:00
|
|
|
g_assert (ret);
|
|
|
|
|
2019-12-09 10:04:56 -03:00
|
|
|
strides[0] = stride;
|
|
|
|
offsets[0] = offset;
|
|
|
|
modifiers[0] = modifier;
|
2018-11-29 13:34:40 +02:00
|
|
|
egl_image = meta_egl_create_dmabuf_image (egl,
|
|
|
|
egl_display,
|
2019-12-09 10:04:56 -03:00
|
|
|
width,
|
|
|
|
height,
|
|
|
|
drm_format,
|
2018-11-29 13:34:40 +02:00
|
|
|
1 /* n_planes */,
|
|
|
|
&dmabuf_fd,
|
|
|
|
strides,
|
|
|
|
offsets,
|
|
|
|
modifiers,
|
2019-12-09 10:04:56 -03:00
|
|
|
error);
|
2018-11-29 13:34:40 +02:00
|
|
|
if (egl_image == EGL_NO_IMAGE_KHR)
|
2019-12-09 10:04:56 -03:00
|
|
|
return NULL;
|
2018-11-29 13:34:40 +02:00
|
|
|
|
2019-07-16 18:30:12 +02:00
|
|
|
flags = COGL_EGL_IMAGE_FLAG_NO_GET_DATA;
|
2018-11-29 13:34:40 +02:00
|
|
|
cogl_tex = cogl_egl_texture_2d_new_from_image (cogl_context,
|
2019-12-09 10:04:56 -03:00
|
|
|
width,
|
|
|
|
height,
|
2018-11-29 13:34:40 +02:00
|
|
|
cogl_format,
|
|
|
|
egl_image,
|
2019-07-16 18:29:29 +02:00
|
|
|
flags,
|
2019-12-09 10:04:56 -03:00
|
|
|
error);
|
2018-11-29 13:34:40 +02:00
|
|
|
|
|
|
|
meta_egl_destroy_image (egl, egl_display, egl_image, NULL);
|
|
|
|
|
|
|
|
if (!cogl_tex)
|
2019-12-09 10:04:56 -03:00
|
|
|
return NULL;
|
2018-11-29 13:34:40 +02:00
|
|
|
|
|
|
|
cogl_fbo = cogl_offscreen_new_with_texture (COGL_TEXTURE (cogl_tex));
|
|
|
|
cogl_object_unref (cogl_tex);
|
|
|
|
|
2019-12-09 10:04:56 -03:00
|
|
|
if (!cogl_framebuffer_allocate (COGL_FRAMEBUFFER (cogl_fbo), error))
|
2018-11-29 13:34:40 +02:00
|
|
|
{
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (cogl_fbo);
|
2019-12-09 10:04:56 -03:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return COGL_FRAMEBUFFER (cogl_fbo);
|
|
|
|
}
|
2018-11-29 13:34:40 +02:00
|
|
|
|
2019-12-09 10:04:56 -03:00
|
|
|
static gboolean
|
|
|
|
copy_shared_framebuffer_primary_gpu (CoglOnscreen *onscreen,
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-12-09 10:04:56 -03:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
|
|
|
MetaRendererNativeGpuData *primary_gpu_data;
|
|
|
|
MetaDrmBufferDumb *buffer_dumb;
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBuffer *buffer;
|
|
|
|
int width, height, stride;
|
|
|
|
uint32_t drm_format;
|
2019-12-09 10:04:56 -03:00
|
|
|
CoglFramebuffer *dmabuf_fb;
|
|
|
|
int dmabuf_fd;
|
|
|
|
g_autoptr (GError) error = NULL;
|
|
|
|
CoglPixelFormat cogl_format;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
COGL_TRACE_BEGIN_SCOPED (CopySharedFramebufferPrimaryGpu,
|
|
|
|
"FB Copy (primary GPU)");
|
|
|
|
|
|
|
|
primary_gpu_data =
|
|
|
|
meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
renderer_native->primary_gpu_kms);
|
|
|
|
if (!primary_gpu_data->secondary.has_EGL_EXT_image_dma_buf_import_modifiers)
|
|
|
|
return FALSE;
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
buffer_dumb = secondary_gpu_get_next_dumb_buffer (secondary_gpu_state);
|
|
|
|
buffer = META_DRM_BUFFER (buffer_dumb);
|
2019-12-09 10:04:56 -03:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
width = meta_drm_buffer_get_width (buffer);
|
|
|
|
height = meta_drm_buffer_get_height (buffer);
|
|
|
|
stride = meta_drm_buffer_get_stride (buffer);
|
|
|
|
drm_format = meta_drm_buffer_get_format (buffer);
|
2019-12-09 10:04:56 -03:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
g_assert (cogl_framebuffer_get_width (framebuffer) == width);
|
|
|
|
g_assert (cogl_framebuffer_get_height (framebuffer) == height);
|
|
|
|
|
|
|
|
ret = meta_cogl_pixel_format_from_drm_format (drm_format,
|
2020-08-28 21:58:18 -03:00
|
|
|
&cogl_format,
|
|
|
|
NULL);
|
2019-12-09 10:04:56 -03:00
|
|
|
g_assert (ret);
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
dmabuf_fd = meta_drm_buffer_dumb_ensure_dmabuf_fd (buffer_dumb, &error);
|
|
|
|
if (!dmabuf_fd)
|
|
|
|
{
|
|
|
|
g_debug ("Failed to create DMA buffer: %s", error->message);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2019-12-09 10:04:56 -03:00
|
|
|
|
|
|
|
dmabuf_fb = create_dma_buf_framebuffer (renderer_native,
|
|
|
|
dmabuf_fd,
|
2020-09-29 09:40:53 +02:00
|
|
|
width,
|
|
|
|
height,
|
|
|
|
stride,
|
2019-12-09 10:04:56 -03:00
|
|
|
0, DRM_FORMAT_MOD_LINEAR,
|
2020-09-29 09:40:53 +02:00
|
|
|
drm_format,
|
2019-12-09 10:04:56 -03:00
|
|
|
&error);
|
|
|
|
|
|
|
|
if (error)
|
|
|
|
{
|
|
|
|
g_debug ("%s: Failed to blit DMA buffer image: %s",
|
|
|
|
G_STRFUNC, error->message);
|
2018-11-29 13:34:40 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2019-12-09 10:04:56 -03:00
|
|
|
if (!cogl_blit_framebuffer (framebuffer, COGL_FRAMEBUFFER (dmabuf_fb),
|
2018-11-29 13:34:40 +02:00
|
|
|
0, 0, 0, 0,
|
2020-09-29 09:40:53 +02:00
|
|
|
width, height,
|
2019-12-09 10:04:56 -03:00
|
|
|
&error))
|
2018-11-29 13:34:40 +02:00
|
|
|
{
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (dmabuf_fb);
|
2018-11-29 13:34:40 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (dmabuf_fb);
|
2018-11-29 13:34:40 +02:00
|
|
|
|
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
2020-09-29 09:40:53 +02:00
|
|
|
secondary_gpu_state->gbm.next_fb = buffer;
|
|
|
|
secondary_gpu_state->cpu.current_dumb_fb = buffer_dumb;
|
2018-11-29 13:34:40 +02:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
|
|
|
copy_shared_framebuffer_cpu (CoglOnscreen *onscreen,
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state,
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaDrmBufferDumb *buffer_dumb;
|
|
|
|
MetaDrmBuffer *buffer;
|
|
|
|
int width, height, stride;
|
|
|
|
uint32_t drm_format;
|
|
|
|
void *buffer_data;
|
renderer/native: use cogl for CPU copy path
Use cogl_framebuffer_read_pixels_into_bitmap () instead of
glReadPixels () for the CPU copy path in multi-GPU support.
The cogl function employs several tricks to make the read-pixels as fast
as possible and does the y-flip as necessary. This should make the copy
more performant over all kinds of hardware.
This is expected to be used on virtual outputs (e.g. DisplayLink USB
docks and monitors) foremost, where the dumb buffer memory is just
regular system memory. If the dumb buffer memory is somehow slow, like
residing in discrete VRAM or having an unexpected caching mode, it may
be possible for the cogl function perform worse because it might do the
y-flip in-place in the dumb buffer. Hopefully that does not happen in
any practical scenario.
Calling meta_renderer_native_gles3_read_pixels () here was conceptually
wrong to begin with because it was done with the Cogl GL context of the
primary GPU, not on the GL ES 3 context of a secondary GPU. However,
due eglBindAPI being a no-op in Mesa and the glReadPixels () arguments
being compatible, it worked.
This patch adds a pixel format conversion table between DRM and Cogl
formats. It contains more formats than absolutely necessary and the
texture components field which is currently unused for completeness. See
Mutter issue #323. Making the table more complete documents better how
the pixel formats actually map so that posterity should be less likely
to be confused. This table could be shared with
shm_buffer_get_cogl_pixel_format () as well, but not with
meta_wayland_dma_buf_buffer_attach ().
On HP ProBook 4520s laptop (Mesa DRI Intel(R) Ironlake Mobile, Mesa
18.0.5), without this patch copy_shared_framebuffer_cpu () for a
DisplayLink output takes 5 seconds with a 1080p frame. Obviously that
makes Mutter and gnome-shell completely unusable. With this patch, that
function takes 13-18 ms which makes it usable if not fluent.
On Intel i7-4790 (Mesa DRI Intel(R) Haswell Desktop) machine, this patch
makes no significant difference (the copy takes 4-5 ms).
2018-09-27 16:42:19 +03:00
|
|
|
CoglBitmap *dumb_bitmap;
|
|
|
|
CoglPixelFormat cogl_format;
|
|
|
|
gboolean ret;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2019-06-11 11:16:39 +03:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (CopySharedFramebufferCpu,
|
|
|
|
"FB Copy (CPU)");
|
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
buffer_dumb = secondary_gpu_get_next_dumb_buffer (secondary_gpu_state);
|
|
|
|
buffer = META_DRM_BUFFER (buffer_dumb);
|
|
|
|
|
|
|
|
width = meta_drm_buffer_get_width (buffer);
|
|
|
|
height = meta_drm_buffer_get_height (buffer);
|
|
|
|
stride = meta_drm_buffer_get_stride (buffer);
|
|
|
|
drm_format = meta_drm_buffer_get_format (buffer);
|
|
|
|
buffer_data = meta_drm_buffer_dumb_get_data (buffer_dumb);
|
2018-09-05 11:48:08 +03:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
g_assert (cogl_framebuffer_get_width (framebuffer) == width);
|
|
|
|
g_assert (cogl_framebuffer_get_height (framebuffer) == height);
|
renderer/native: use cogl for CPU copy path
Use cogl_framebuffer_read_pixels_into_bitmap () instead of
glReadPixels () for the CPU copy path in multi-GPU support.
The cogl function employs several tricks to make the read-pixels as fast
as possible and does the y-flip as necessary. This should make the copy
more performant over all kinds of hardware.
This is expected to be used on virtual outputs (e.g. DisplayLink USB
docks and monitors) foremost, where the dumb buffer memory is just
regular system memory. If the dumb buffer memory is somehow slow, like
residing in discrete VRAM or having an unexpected caching mode, it may
be possible for the cogl function perform worse because it might do the
y-flip in-place in the dumb buffer. Hopefully that does not happen in
any practical scenario.
Calling meta_renderer_native_gles3_read_pixels () here was conceptually
wrong to begin with because it was done with the Cogl GL context of the
primary GPU, not on the GL ES 3 context of a secondary GPU. However,
due eglBindAPI being a no-op in Mesa and the glReadPixels () arguments
being compatible, it worked.
This patch adds a pixel format conversion table between DRM and Cogl
formats. It contains more formats than absolutely necessary and the
texture components field which is currently unused for completeness. See
Mutter issue #323. Making the table more complete documents better how
the pixel formats actually map so that posterity should be less likely
to be confused. This table could be shared with
shm_buffer_get_cogl_pixel_format () as well, but not with
meta_wayland_dma_buf_buffer_attach ().
On HP ProBook 4520s laptop (Mesa DRI Intel(R) Ironlake Mobile, Mesa
18.0.5), without this patch copy_shared_framebuffer_cpu () for a
DisplayLink output takes 5 seconds with a 1080p frame. Obviously that
makes Mutter and gnome-shell completely unusable. With this patch, that
function takes 13-18 ms which makes it usable if not fluent.
On Intel i7-4790 (Mesa DRI Intel(R) Haswell Desktop) machine, this patch
makes no significant difference (the copy takes 4-5 ms).
2018-09-27 16:42:19 +03:00
|
|
|
|
2020-09-29 09:40:53 +02:00
|
|
|
ret = meta_cogl_pixel_format_from_drm_format (drm_format,
|
2020-08-28 21:58:18 -03:00
|
|
|
&cogl_format,
|
|
|
|
NULL);
|
renderer/native: use cogl for CPU copy path
Use cogl_framebuffer_read_pixels_into_bitmap () instead of
glReadPixels () for the CPU copy path in multi-GPU support.
The cogl function employs several tricks to make the read-pixels as fast
as possible and does the y-flip as necessary. This should make the copy
more performant over all kinds of hardware.
This is expected to be used on virtual outputs (e.g. DisplayLink USB
docks and monitors) foremost, where the dumb buffer memory is just
regular system memory. If the dumb buffer memory is somehow slow, like
residing in discrete VRAM or having an unexpected caching mode, it may
be possible for the cogl function perform worse because it might do the
y-flip in-place in the dumb buffer. Hopefully that does not happen in
any practical scenario.
Calling meta_renderer_native_gles3_read_pixels () here was conceptually
wrong to begin with because it was done with the Cogl GL context of the
primary GPU, not on the GL ES 3 context of a secondary GPU. However,
due eglBindAPI being a no-op in Mesa and the glReadPixels () arguments
being compatible, it worked.
This patch adds a pixel format conversion table between DRM and Cogl
formats. It contains more formats than absolutely necessary and the
texture components field which is currently unused for completeness. See
Mutter issue #323. Making the table more complete documents better how
the pixel formats actually map so that posterity should be less likely
to be confused. This table could be shared with
shm_buffer_get_cogl_pixel_format () as well, but not with
meta_wayland_dma_buf_buffer_attach ().
On HP ProBook 4520s laptop (Mesa DRI Intel(R) Ironlake Mobile, Mesa
18.0.5), without this patch copy_shared_framebuffer_cpu () for a
DisplayLink output takes 5 seconds with a 1080p frame. Obviously that
makes Mutter and gnome-shell completely unusable. With this patch, that
function takes 13-18 ms which makes it usable if not fluent.
On Intel i7-4790 (Mesa DRI Intel(R) Haswell Desktop) machine, this patch
makes no significant difference (the copy takes 4-5 ms).
2018-09-27 16:42:19 +03:00
|
|
|
g_assert (ret);
|
|
|
|
|
|
|
|
dumb_bitmap = cogl_bitmap_new_for_data (cogl_context,
|
2020-09-29 09:40:53 +02:00
|
|
|
width,
|
|
|
|
height,
|
renderer/native: use cogl for CPU copy path
Use cogl_framebuffer_read_pixels_into_bitmap () instead of
glReadPixels () for the CPU copy path in multi-GPU support.
The cogl function employs several tricks to make the read-pixels as fast
as possible and does the y-flip as necessary. This should make the copy
more performant over all kinds of hardware.
This is expected to be used on virtual outputs (e.g. DisplayLink USB
docks and monitors) foremost, where the dumb buffer memory is just
regular system memory. If the dumb buffer memory is somehow slow, like
residing in discrete VRAM or having an unexpected caching mode, it may
be possible for the cogl function perform worse because it might do the
y-flip in-place in the dumb buffer. Hopefully that does not happen in
any practical scenario.
Calling meta_renderer_native_gles3_read_pixels () here was conceptually
wrong to begin with because it was done with the Cogl GL context of the
primary GPU, not on the GL ES 3 context of a secondary GPU. However,
due eglBindAPI being a no-op in Mesa and the glReadPixels () arguments
being compatible, it worked.
This patch adds a pixel format conversion table between DRM and Cogl
formats. It contains more formats than absolutely necessary and the
texture components field which is currently unused for completeness. See
Mutter issue #323. Making the table more complete documents better how
the pixel formats actually map so that posterity should be less likely
to be confused. This table could be shared with
shm_buffer_get_cogl_pixel_format () as well, but not with
meta_wayland_dma_buf_buffer_attach ().
On HP ProBook 4520s laptop (Mesa DRI Intel(R) Ironlake Mobile, Mesa
18.0.5), without this patch copy_shared_framebuffer_cpu () for a
DisplayLink output takes 5 seconds with a 1080p frame. Obviously that
makes Mutter and gnome-shell completely unusable. With this patch, that
function takes 13-18 ms which makes it usable if not fluent.
On Intel i7-4790 (Mesa DRI Intel(R) Haswell Desktop) machine, this patch
makes no significant difference (the copy takes 4-5 ms).
2018-09-27 16:42:19 +03:00
|
|
|
cogl_format,
|
2020-09-29 09:40:53 +02:00
|
|
|
stride,
|
|
|
|
buffer_data);
|
renderer/native: use cogl for CPU copy path
Use cogl_framebuffer_read_pixels_into_bitmap () instead of
glReadPixels () for the CPU copy path in multi-GPU support.
The cogl function employs several tricks to make the read-pixels as fast
as possible and does the y-flip as necessary. This should make the copy
more performant over all kinds of hardware.
This is expected to be used on virtual outputs (e.g. DisplayLink USB
docks and monitors) foremost, where the dumb buffer memory is just
regular system memory. If the dumb buffer memory is somehow slow, like
residing in discrete VRAM or having an unexpected caching mode, it may
be possible for the cogl function perform worse because it might do the
y-flip in-place in the dumb buffer. Hopefully that does not happen in
any practical scenario.
Calling meta_renderer_native_gles3_read_pixels () here was conceptually
wrong to begin with because it was done with the Cogl GL context of the
primary GPU, not on the GL ES 3 context of a secondary GPU. However,
due eglBindAPI being a no-op in Mesa and the glReadPixels () arguments
being compatible, it worked.
This patch adds a pixel format conversion table between DRM and Cogl
formats. It contains more formats than absolutely necessary and the
texture components field which is currently unused for completeness. See
Mutter issue #323. Making the table more complete documents better how
the pixel formats actually map so that posterity should be less likely
to be confused. This table could be shared with
shm_buffer_get_cogl_pixel_format () as well, but not with
meta_wayland_dma_buf_buffer_attach ().
On HP ProBook 4520s laptop (Mesa DRI Intel(R) Ironlake Mobile, Mesa
18.0.5), without this patch copy_shared_framebuffer_cpu () for a
DisplayLink output takes 5 seconds with a 1080p frame. Obviously that
makes Mutter and gnome-shell completely unusable. With this patch, that
function takes 13-18 ms which makes it usable if not fluent.
On Intel i7-4790 (Mesa DRI Intel(R) Haswell Desktop) machine, this patch
makes no significant difference (the copy takes 4-5 ms).
2018-09-27 16:42:19 +03:00
|
|
|
|
|
|
|
if (!cogl_framebuffer_read_pixels_into_bitmap (framebuffer,
|
|
|
|
0 /* x */,
|
|
|
|
0 /* y */,
|
|
|
|
COGL_READ_PIXELS_COLOR_BUFFER,
|
|
|
|
dumb_bitmap))
|
|
|
|
g_warning ("Failed to CPU-copy to a secondary GPU output");
|
|
|
|
|
|
|
|
cogl_object_unref (dumb_bitmap);
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2018-01-30 17:47:17 +08:00
|
|
|
g_clear_object (&secondary_gpu_state->gbm.next_fb);
|
2020-09-29 09:40:53 +02:00
|
|
|
secondary_gpu_state->gbm.next_fb = buffer;
|
|
|
|
secondary_gpu_state->cpu.current_dumb_fb = buffer_dumb;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
update_secondary_gpu_state_pre_swap_buffers (CoglOnscreen *onscreen)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
|
|
|
|
2019-08-27 13:11:07 +03:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaRendererNativeGpuStatePreSwapBuffers,
|
|
|
|
"Onscreen (secondary gpu pre-swap-buffers)");
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
secondary_gpu_state = onscreen_native->secondary_gpu_state;
|
|
|
|
if (secondary_gpu_state)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
renderer_gpu_data = secondary_gpu_state->renderer_gpu_data;
|
|
|
|
switch (renderer_gpu_data->secondary.copy_mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_SECONDARY_GPU:
|
2017-07-25 10:21:04 +08:00
|
|
|
/* Done after eglSwapBuffers. */
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
2019-05-24 17:07:14 +03:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO:
|
|
|
|
/* Done after eglSwapBuffers. */
|
|
|
|
if (secondary_gpu_state->import_status ==
|
|
|
|
META_SHARED_FRAMEBUFFER_IMPORT_STATUS_OK)
|
|
|
|
break;
|
|
|
|
/* prepare fallback */
|
|
|
|
G_GNUC_FALLTHROUGH;
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_PRIMARY:
|
|
|
|
if (!copy_shared_framebuffer_primary_gpu (onscreen,
|
|
|
|
secondary_gpu_state))
|
|
|
|
{
|
2019-07-05 14:09:31 +03:00
|
|
|
if (!secondary_gpu_state->noted_primary_gpu_copy_failed)
|
|
|
|
{
|
|
|
|
g_debug ("Using primary GPU to copy for %s failed once.",
|
|
|
|
meta_gpu_kms_get_file_path (secondary_gpu_state->gpu_kms));
|
|
|
|
secondary_gpu_state->noted_primary_gpu_copy_failed = TRUE;
|
|
|
|
}
|
|
|
|
|
2018-11-29 13:34:40 +02:00
|
|
|
copy_shared_framebuffer_cpu (onscreen,
|
|
|
|
secondary_gpu_state,
|
|
|
|
renderer_gpu_data);
|
|
|
|
}
|
2019-07-05 14:09:31 +03:00
|
|
|
else if (!secondary_gpu_state->noted_primary_gpu_copy_ok)
|
|
|
|
{
|
|
|
|
g_debug ("Using primary GPU to copy for %s succeeded once.",
|
|
|
|
meta_gpu_kms_get_file_path (secondary_gpu_state->gpu_kms));
|
|
|
|
secondary_gpu_state->noted_primary_gpu_copy_ok = TRUE;
|
|
|
|
}
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
}
|
2016-06-08 17:05:31 +08:00
|
|
|
}
|
2016-05-12 14:55:06 +08:00
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static void
|
|
|
|
update_secondary_gpu_state_post_swap_buffers (CoglOnscreen *onscreen,
|
|
|
|
gboolean *egl_context_changed)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
|
|
|
MetaOnscreenNativeSecondaryGpuState *secondary_gpu_state;
|
|
|
|
|
2019-08-27 13:11:07 +03:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaRendererNativeGpuStatePostSwapBuffers,
|
|
|
|
"Onscreen (secondary gpu post-swap-buffers)");
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
secondary_gpu_state = onscreen_native->secondary_gpu_state;
|
|
|
|
if (secondary_gpu_state)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
renderer_gpu_data =
|
|
|
|
meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
secondary_gpu_state->gpu_kms);
|
2019-05-24 17:07:14 +03:00
|
|
|
retry:
|
2017-07-25 10:21:04 +08:00
|
|
|
switch (renderer_gpu_data->secondary.copy_mode)
|
|
|
|
{
|
2019-05-24 17:07:14 +03:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO:
|
|
|
|
if (!import_shared_framebuffer (onscreen,
|
|
|
|
secondary_gpu_state))
|
|
|
|
goto retry;
|
|
|
|
break;
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_SECONDARY_GPU:
|
2017-07-25 10:21:04 +08:00
|
|
|
copy_shared_framebuffer_gpu (onscreen,
|
|
|
|
secondary_gpu_state,
|
|
|
|
renderer_gpu_data,
|
|
|
|
egl_context_changed);
|
|
|
|
break;
|
2018-11-29 13:34:40 +02:00
|
|
|
case META_SHARED_FRAMEBUFFER_COPY_MODE_PRIMARY:
|
2017-07-25 10:21:04 +08:00
|
|
|
/* Done before eglSwapBuffers. */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-09-29 22:33:02 +08:00
|
|
|
}
|
|
|
|
|
2019-09-12 10:47:46 +02:00
|
|
|
static void
|
2020-07-16 23:38:10 +02:00
|
|
|
ensure_crtc_modes (CoglOnscreen *onscreen)
|
2019-09-12 10:47:46 +02:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-09-12 10:47:46 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2019-09-12 10:47:46 +02:00
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
2020-10-09 23:17:06 +02:00
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
|
|
|
MetaPowerSave power_save_mode;
|
|
|
|
GList *link;
|
2019-09-12 10:47:46 +02:00
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
power_save_mode = meta_monitor_manager_get_power_save_mode (monitor_manager);
|
|
|
|
link = g_list_find (renderer_native->pending_mode_set_views,
|
|
|
|
onscreen_native->view);
|
|
|
|
if (link && power_save_mode == META_POWER_SAVE_ON)
|
2019-09-12 10:47:46 +02:00
|
|
|
{
|
2020-07-16 23:38:10 +02:00
|
|
|
meta_onscreen_native_set_crtc_mode (onscreen, renderer_gpu_data);
|
2020-10-09 23:17:06 +02:00
|
|
|
renderer_native->pending_mode_set_views =
|
|
|
|
g_list_delete_link (renderer_native->pending_mode_set_views, link);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaKmsDevice *
|
|
|
|
kms_device_from_view (MetaRendererView *view)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer =
|
|
|
|
clutter_stage_view_get_onscreen (CLUTTER_STAGE_VIEW (view));
|
|
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-10-09 23:17:06 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (onscreen_native->crtc);
|
|
|
|
MetaKmsCrtc *kms_crtc = meta_crtc_kms_get_kms_crtc (crtc_kms);
|
|
|
|
|
|
|
|
return meta_kms_crtc_get_device (kms_crtc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaGpu *
|
|
|
|
gpu_from_view (MetaRendererView *view)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer =
|
|
|
|
clutter_stage_view_get_onscreen (CLUTTER_STAGE_VIEW (view));
|
|
|
|
CoglOnscreen *onscreen = COGL_ONSCREEN (framebuffer);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-10-09 23:17:06 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
|
|
|
|
return meta_crtc_get_gpu (onscreen_native->crtc);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
configure_disabled_crtcs (MetaGpu *gpu,
|
|
|
|
MetaKmsUpdate *kms_update)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
for (l = meta_gpu_get_crtcs (gpu); l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaCrtc *crtc = l->data;
|
|
|
|
MetaKmsCrtc *kms_crtc;
|
|
|
|
|
|
|
|
if (meta_crtc_get_config (crtc))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
kms_crtc = meta_crtc_kms_get_kms_crtc (META_CRTC_KMS (crtc));
|
|
|
|
if (!meta_kms_crtc_is_active (kms_crtc))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
meta_kms_update_mode_set (kms_update, kms_crtc, NULL, NULL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
static void
|
|
|
|
clear_kept_alive_onscreens (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
g_list_free_full (renderer_native->kept_alive_onscreens,
|
|
|
|
g_object_unref);
|
|
|
|
renderer_native->kept_alive_onscreens = NULL;
|
|
|
|
}
|
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
static void
|
|
|
|
meta_renderer_native_post_mode_set_updates (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaKms *kms = meta_backend_native_get_kms (META_BACKEND_NATIVE (backend));
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
for (l = meta_renderer_get_views (renderer); l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaRendererView *view = l->data;
|
|
|
|
MetaKmsDevice *kms_device;
|
|
|
|
MetaKmsUpdate *kms_update;
|
|
|
|
MetaKmsUpdateFlag flags;
|
|
|
|
g_autoptr (MetaKmsFeedback) kms_feedback = NULL;
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
const GError *feedback_error;
|
2020-10-09 23:17:06 +02:00
|
|
|
|
|
|
|
kms_device = kms_device_from_view (view);
|
|
|
|
|
|
|
|
kms_update = meta_kms_get_pending_update (kms, kms_device);
|
|
|
|
if (!kms_update)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
configure_disabled_crtcs (gpu_from_view (view), kms_update);
|
|
|
|
|
|
|
|
flags = META_KMS_UPDATE_FLAG_NONE;
|
|
|
|
kms_feedback = meta_kms_post_pending_update_sync (kms, kms_device, flags);
|
|
|
|
|
|
|
|
switch (meta_kms_feedback_get_result (kms_feedback))
|
|
|
|
{
|
|
|
|
case META_KMS_FEEDBACK_PASSED:
|
|
|
|
break;
|
|
|
|
case META_KMS_FEEDBACK_FAILED:
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
feedback_error = meta_kms_feedback_get_error (kms_feedback);
|
|
|
|
if (!g_error_matches (feedback_error,
|
|
|
|
G_IO_ERROR,
|
|
|
|
G_IO_ERROR_PERMISSION_DENIED))
|
|
|
|
g_warning ("Failed to post KMS update: %s", feedback_error->message);
|
2020-10-09 23:17:06 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
clear_kept_alive_onscreens (renderer_native);
|
2019-09-12 10:47:46 +02:00
|
|
|
}
|
|
|
|
|
2020-10-02 10:38:57 +02:00
|
|
|
static void
|
2020-12-21 14:34:43 +08:00
|
|
|
unset_disabled_crtcs (MetaBackend *backend,
|
|
|
|
MetaKms *kms)
|
|
|
|
{
|
|
|
|
GList *l;
|
|
|
|
|
2020-10-02 18:37:18 +02:00
|
|
|
meta_topic (META_DEBUG_KMS, "Disabling all disabled CRTCs");
|
|
|
|
|
2020-12-21 14:34:43 +08:00
|
|
|
for (l = meta_backend_get_gpus (backend); l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaGpu *gpu = l->data;
|
2020-10-02 10:38:57 +02:00
|
|
|
MetaKmsDevice *kms_device =
|
|
|
|
meta_gpu_kms_get_kms_device (META_GPU_KMS (gpu));
|
2020-12-21 14:34:43 +08:00
|
|
|
GList *k;
|
2020-10-02 10:38:57 +02:00
|
|
|
gboolean did_mode_set = FALSE;
|
2020-10-02 16:48:34 +02:00
|
|
|
MetaKmsUpdateFlag flags;
|
2020-10-02 10:38:57 +02:00
|
|
|
g_autoptr (MetaKmsFeedback) kms_feedback = NULL;
|
2020-12-21 14:34:43 +08:00
|
|
|
|
|
|
|
for (k = meta_gpu_get_crtcs (gpu); k; k = k->next)
|
|
|
|
{
|
|
|
|
MetaCrtc *crtc = k->data;
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaKmsUpdate *kms_update;
|
2020-12-21 14:34:43 +08:00
|
|
|
|
|
|
|
if (meta_crtc_get_config (crtc))
|
|
|
|
continue;
|
|
|
|
|
2020-07-16 23:38:10 +02:00
|
|
|
kms_update = meta_kms_ensure_pending_update (kms, kms_device);
|
2020-12-21 14:34:43 +08:00
|
|
|
meta_crtc_kms_set_mode (META_CRTC_KMS (crtc), kms_update);
|
2020-07-16 23:38:10 +02:00
|
|
|
|
|
|
|
did_mode_set = TRUE;
|
2020-12-21 14:34:43 +08:00
|
|
|
}
|
|
|
|
|
2020-10-02 10:38:57 +02:00
|
|
|
if (!did_mode_set)
|
|
|
|
continue;
|
2020-12-21 14:34:43 +08:00
|
|
|
|
2020-10-02 16:48:34 +02:00
|
|
|
flags = META_KMS_UPDATE_FLAG_NONE;
|
|
|
|
kms_feedback = meta_kms_post_pending_update_sync (kms,
|
|
|
|
kms_device,
|
|
|
|
flags);
|
2020-10-02 10:38:57 +02:00
|
|
|
if (meta_kms_feedback_get_result (kms_feedback) !=
|
|
|
|
META_KMS_FEEDBACK_PASSED)
|
|
|
|
{
|
|
|
|
const GError *error = meta_kms_feedback_get_error (kms_feedback);
|
2020-12-21 14:34:43 +08:00
|
|
|
|
2020-10-02 10:38:57 +02:00
|
|
|
if (!g_error_matches (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_PERMISSION_DENIED))
|
|
|
|
g_warning ("Failed to post KMS update: %s", error->message);
|
|
|
|
}
|
2020-12-21 14:34:43 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-29 22:33:02 +08:00
|
|
|
static void
|
2020-05-22 21:52:29 +02:00
|
|
|
meta_onscreen_native_swap_buffers_with_damage (CoglOnscreen *onscreen,
|
|
|
|
const int *rectangles,
|
|
|
|
int n_rectangles,
|
2020-10-09 22:00:29 +02:00
|
|
|
CoglFrameInfo *frame_info,
|
|
|
|
gpointer user_data)
|
2016-09-29 22:33:02 +08:00
|
|
|
{
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2017-07-25 10:21:04 +08:00
|
|
|
CoglDisplay *cogl_display = cogl_context_get_display (cogl_context);
|
2016-09-29 22:33:02 +08:00
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaGpuKms *render_gpu = onscreen_native->render_gpu;
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaKmsDevice *render_kms_device = meta_gpu_kms_get_kms_device (render_gpu);
|
2020-10-09 22:29:24 +02:00
|
|
|
ClutterFrame *frame = user_data;
|
2017-07-25 10:21:04 +08:00
|
|
|
gboolean egl_context_changed = FALSE;
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaPowerSave power_save_mode;
|
2018-01-30 17:47:17 +08:00
|
|
|
g_autoptr (GError) error = NULL;
|
2019-05-20 22:08:35 +02:00
|
|
|
MetaDrmBufferGbm *buffer_gbm;
|
2020-10-02 10:38:57 +02:00
|
|
|
MetaKmsCrtc *kms_crtc;
|
|
|
|
MetaKmsDevice *kms_device;
|
2020-10-02 16:48:34 +02:00
|
|
|
MetaKmsUpdateFlag flags;
|
2019-11-09 00:14:36 +01:00
|
|
|
g_autoptr (MetaKmsFeedback) kms_feedback = NULL;
|
2020-10-09 22:29:24 +02:00
|
|
|
const GError *feedback_error;
|
2016-09-29 22:33:02 +08:00
|
|
|
|
2019-08-27 13:11:07 +03:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaRendererNativeSwapBuffers,
|
|
|
|
"Onscreen (swap-buffers)");
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
update_secondary_gpu_state_pre_swap_buffers (onscreen);
|
2016-09-29 22:33:02 +08:00
|
|
|
|
|
|
|
parent_vtable->onscreen_swap_buffers_with_damage (onscreen,
|
|
|
|
rectangles,
|
2020-05-22 21:52:29 +02:00
|
|
|
n_rectangles,
|
2020-10-09 22:00:29 +02:00
|
|
|
frame_info,
|
|
|
|
user_data);
|
2016-09-29 22:33:02 +08:00
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
2017-07-25 10:21:04 +08:00
|
|
|
render_gpu);
|
2017-07-10 18:19:32 +08:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
2018-01-30 17:47:17 +08:00
|
|
|
g_warn_if_fail (onscreen_native->gbm.next_fb == NULL);
|
|
|
|
g_clear_object (&onscreen_native->gbm.next_fb);
|
2019-05-20 22:08:35 +02:00
|
|
|
|
2019-08-20 21:28:19 +02:00
|
|
|
buffer_gbm =
|
2020-09-29 09:40:53 +02:00
|
|
|
meta_drm_buffer_gbm_new_lock_front (render_kms_device,
|
2019-05-20 22:08:35 +02:00
|
|
|
onscreen_native->gbm.surface,
|
|
|
|
renderer_native->use_modifiers,
|
|
|
|
&error);
|
2019-09-16 15:24:52 +03:00
|
|
|
if (!buffer_gbm)
|
2018-01-30 17:47:17 +08:00
|
|
|
{
|
2019-08-20 21:28:19 +02:00
|
|
|
g_warning ("meta_drm_buffer_gbm_new_lock_front failed: %s",
|
2018-01-30 17:47:17 +08:00
|
|
|
error->message);
|
|
|
|
return;
|
|
|
|
}
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2019-09-16 15:24:52 +03:00
|
|
|
onscreen_native->gbm.next_fb = META_DRM_BUFFER (buffer_gbm);
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
2016-09-29 22:33:02 +08:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
update_secondary_gpu_state_post_swap_buffers (onscreen, &egl_context_changed);
|
|
|
|
|
2020-07-16 23:38:10 +02:00
|
|
|
power_save_mode = meta_monitor_manager_get_power_save_mode (monitor_manager);
|
|
|
|
if (power_save_mode == META_POWER_SAVE_ON)
|
|
|
|
{
|
|
|
|
ensure_crtc_modes (onscreen);
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
meta_onscreen_native_flip_crtc (onscreen,
|
|
|
|
onscreen_native->view,
|
|
|
|
onscreen_native->crtc,
|
|
|
|
META_KMS_PAGE_FLIP_LISTENER_FLAG_NONE);
|
2020-07-16 23:38:10 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
queue_dummy_power_save_page_flip (onscreen);
|
2020-10-09 22:29:24 +02:00
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
2020-07-16 23:38:10 +02:00
|
|
|
return;
|
|
|
|
}
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If we changed EGL context, cogl will have the wrong idea about what is
|
|
|
|
* current, making it fail to set it when it needs to. Avoid that by making
|
|
|
|
* EGL_NO_CONTEXT current now, making cogl eventually set the correct
|
|
|
|
* context.
|
|
|
|
*/
|
|
|
|
if (egl_context_changed)
|
2018-04-19 17:28:07 +02:00
|
|
|
_cogl_winsys_egl_ensure_current (cogl_display);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
COGL_TRACE_BEGIN_SCOPED (MetaRendererNativePostKmsUpdate,
|
|
|
|
"Onscreen (post pending update)");
|
2020-10-02 10:38:57 +02:00
|
|
|
kms_crtc = meta_crtc_kms_get_kms_crtc (META_CRTC_KMS (onscreen_native->crtc));
|
|
|
|
kms_device = meta_kms_crtc_get_device (kms_crtc);
|
2020-10-02 16:48:34 +02:00
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
switch (renderer_gpu_data->mode)
|
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
if (renderer_native->pending_mode_set_views)
|
|
|
|
{
|
|
|
|
meta_topic (META_DEBUG_KMS,
|
|
|
|
"Postponing primary plane composite update for CRTC %u (%s)",
|
|
|
|
meta_kms_crtc_get_id (kms_crtc),
|
|
|
|
meta_kms_device_get_path (kms_device));
|
|
|
|
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else if (renderer_native->pending_mode_set)
|
|
|
|
{
|
|
|
|
meta_topic (META_DEBUG_KMS, "Posting global mode set updates on %s",
|
|
|
|
meta_kms_device_get_path (kms_device));
|
|
|
|
|
|
|
|
renderer_native->pending_mode_set = FALSE;
|
|
|
|
meta_renderer_native_post_mode_set_updates (renderer_native);
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
if (renderer_native->pending_mode_set)
|
|
|
|
{
|
|
|
|
renderer_native->pending_mode_set = FALSE;
|
|
|
|
meta_renderer_native_post_mode_set_updates (renderer_native);
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2020-10-02 18:37:18 +02:00
|
|
|
meta_topic (META_DEBUG_KMS,
|
|
|
|
"Posting primary plane composite update for CRTC %u (%s)",
|
|
|
|
meta_kms_crtc_get_id (kms_crtc),
|
|
|
|
meta_kms_device_get_path (kms_device));
|
|
|
|
|
2020-10-02 16:48:34 +02:00
|
|
|
flags = META_KMS_UPDATE_FLAG_NONE;
|
|
|
|
kms_feedback = meta_kms_post_pending_update_sync (kms, kms_device, flags);
|
2020-10-02 10:38:57 +02:00
|
|
|
|
2020-10-09 22:29:24 +02:00
|
|
|
switch (meta_kms_feedback_get_result (kms_feedback))
|
|
|
|
{
|
|
|
|
case META_KMS_FEEDBACK_PASSED:
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
break;
|
|
|
|
case META_KMS_FEEDBACK_FAILED:
|
|
|
|
clutter_frame_set_result (frame,
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
2020-10-09 22:29:24 +02:00
|
|
|
|
|
|
|
feedback_error = meta_kms_feedback_get_error (kms_feedback);
|
|
|
|
if (!g_error_matches (feedback_error,
|
|
|
|
G_IO_ERROR,
|
|
|
|
G_IO_ERROR_PERMISSION_DENIED))
|
|
|
|
g_warning ("Failed to post KMS update: %s", feedback_error->message);
|
|
|
|
break;
|
2020-10-02 10:38:57 +02:00
|
|
|
}
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2019-12-09 10:05:37 -03:00
|
|
|
static CoglDmaBufHandle *
|
|
|
|
meta_renderer_native_create_dma_buf (CoglRenderer *cogl_renderer,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
|
|
|
|
switch (renderer_gpu_data->mode)
|
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
{
|
|
|
|
CoglFramebuffer *dmabuf_fb;
|
2020-06-02 18:34:57 +02:00
|
|
|
CoglDmaBufHandle *dmabuf_handle;
|
2019-12-09 10:05:37 -03:00
|
|
|
struct gbm_bo *new_bo;
|
2020-04-29 16:26:52 +02:00
|
|
|
int stride;
|
|
|
|
int offset;
|
|
|
|
int bpp;
|
2019-12-09 10:05:37 -03:00
|
|
|
int dmabuf_fd = -1;
|
|
|
|
|
|
|
|
new_bo = gbm_bo_create (renderer_gpu_data->gbm.device,
|
|
|
|
width, height, DRM_FORMAT_XRGB8888,
|
|
|
|
GBM_BO_USE_RENDERING | GBM_BO_USE_LINEAR);
|
|
|
|
|
|
|
|
if (!new_bo)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"Failed to allocate buffer");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
dmabuf_fd = gbm_bo_get_fd (new_bo);
|
|
|
|
|
|
|
|
if (dmabuf_fd == -1)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_EXISTS,
|
|
|
|
"Failed to export buffer's DMA fd: %s",
|
|
|
|
g_strerror (errno));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-04-29 16:26:52 +02:00
|
|
|
stride = gbm_bo_get_stride (new_bo);
|
|
|
|
offset = gbm_bo_get_offset (new_bo, 0);
|
|
|
|
bpp = 4;
|
2019-12-09 10:05:37 -03:00
|
|
|
dmabuf_fb = create_dma_buf_framebuffer (renderer_native,
|
|
|
|
dmabuf_fd,
|
|
|
|
width, height,
|
2020-04-29 16:26:52 +02:00
|
|
|
stride,
|
|
|
|
offset,
|
2019-12-09 10:05:37 -03:00
|
|
|
DRM_FORMAT_MOD_LINEAR,
|
|
|
|
DRM_FORMAT_XRGB8888,
|
|
|
|
error);
|
|
|
|
|
|
|
|
if (!dmabuf_fb)
|
|
|
|
return NULL;
|
|
|
|
|
2020-06-02 18:34:57 +02:00
|
|
|
dmabuf_handle =
|
|
|
|
cogl_dma_buf_handle_new (dmabuf_fb, dmabuf_fd,
|
|
|
|
width, height, stride, offset, bpp,
|
|
|
|
new_bo,
|
|
|
|
(GDestroyNotify) gbm_bo_destroy);
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (dmabuf_fb);
|
2020-06-02 18:34:57 +02:00
|
|
|
return dmabuf_handle;
|
2019-12-09 10:05:37 -03:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_UNKNOWN,
|
|
|
|
"Current mode does not support exporting DMA buffers");
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2019-09-12 21:41:17 +02:00
|
|
|
gboolean
|
|
|
|
meta_onscreen_native_is_buffer_scanout_compatible (CoglOnscreen *onscreen,
|
|
|
|
uint32_t drm_format,
|
|
|
|
uint64_t drm_modifier,
|
|
|
|
uint32_t stride)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-09-12 21:41:17 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-02-26 10:37:53 +01:00
|
|
|
const MetaCrtcConfig *crtc_config;
|
2019-09-12 21:41:17 +02:00
|
|
|
MetaDrmBuffer *fb;
|
|
|
|
struct gbm_bo *gbm_bo;
|
|
|
|
|
2020-02-26 10:37:53 +01:00
|
|
|
crtc_config = meta_crtc_get_config (onscreen_native->crtc);
|
|
|
|
if (crtc_config->transform != META_MONITOR_TRANSFORM_NORMAL)
|
2019-09-12 21:41:17 +02:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (onscreen_native->secondary_gpu_state)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!onscreen_native->gbm.surface)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
fb = onscreen_native->gbm.current_fb ? onscreen_native->gbm.current_fb
|
|
|
|
: onscreen_native->gbm.next_fb;
|
|
|
|
if (!fb)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!META_IS_DRM_BUFFER_GBM (fb))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
gbm_bo = meta_drm_buffer_gbm_get_bo (META_DRM_BUFFER_GBM (fb));
|
|
|
|
|
|
|
|
if (gbm_bo_get_format (gbm_bo) != drm_format)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (gbm_bo_get_modifier (gbm_bo) != drm_modifier)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (gbm_bo_get_stride (gbm_bo) != stride)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2020-09-10 17:41:06 +02:00
|
|
|
static gboolean
|
|
|
|
meta_onscreen_native_direct_scanout (CoglOnscreen *onscreen,
|
|
|
|
CoglScanout *scanout,
|
|
|
|
CoglFrameInfo *frame_info,
|
2020-10-09 22:00:29 +02:00
|
|
|
gpointer user_data,
|
2020-09-10 17:41:06 +02:00
|
|
|
GError **error)
|
2019-09-12 10:47:46 +02:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-09-12 10:47:46 +02:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaGpuKms *render_gpu = onscreen_native->render_gpu;
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2019-09-12 10:47:46 +02:00
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
2020-07-16 23:38:10 +02:00
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
|
|
|
MetaPowerSave power_save_mode;
|
2020-10-09 22:29:24 +02:00
|
|
|
ClutterFrame *frame = user_data;
|
2020-10-02 10:38:57 +02:00
|
|
|
MetaKmsCrtc *kms_crtc;
|
|
|
|
MetaKmsDevice *kms_device;
|
2020-10-02 16:48:34 +02:00
|
|
|
MetaKmsUpdateFlag flags;
|
2020-09-10 17:41:06 +02:00
|
|
|
g_autoptr (MetaKmsFeedback) kms_feedback = NULL;
|
2020-10-09 22:29:24 +02:00
|
|
|
const GError *feedback_error;
|
2019-09-12 10:47:46 +02:00
|
|
|
|
2020-07-16 23:38:10 +02:00
|
|
|
power_save_mode = meta_monitor_manager_get_power_save_mode (monitor_manager);
|
|
|
|
if (power_save_mode != META_POWER_SAVE_ON)
|
|
|
|
{
|
2020-10-09 23:17:06 +02:00
|
|
|
g_set_error_literal (error,
|
|
|
|
COGL_SCANOUT_ERROR,
|
|
|
|
COGL_SCANOUT_ERROR_INHIBITED,
|
|
|
|
NULL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (renderer_native->pending_mode_set_views)
|
|
|
|
{
|
|
|
|
g_set_error_literal (error,
|
|
|
|
COGL_SCANOUT_ERROR,
|
|
|
|
COGL_SCANOUT_ERROR_INHIBITED,
|
|
|
|
NULL);
|
2020-07-16 23:38:10 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
2019-09-12 10:47:46 +02:00
|
|
|
|
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
render_gpu);
|
|
|
|
|
2020-09-10 17:41:06 +02:00
|
|
|
g_warn_if_fail (renderer_gpu_data->mode == META_RENDERER_NATIVE_MODE_GBM);
|
2019-09-12 10:47:46 +02:00
|
|
|
g_warn_if_fail (!onscreen_native->gbm.next_fb);
|
2020-09-10 17:41:06 +02:00
|
|
|
|
2019-09-12 10:47:46 +02:00
|
|
|
g_set_object (&onscreen_native->gbm.next_fb, META_DRM_BUFFER (scanout));
|
|
|
|
|
2020-07-16 23:38:10 +02:00
|
|
|
ensure_crtc_modes (onscreen);
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
meta_onscreen_native_flip_crtc (onscreen,
|
|
|
|
onscreen_native->view,
|
|
|
|
onscreen_native->crtc,
|
|
|
|
META_KMS_PAGE_FLIP_LISTENER_FLAG_NO_DISCARD);
|
2019-09-12 10:47:46 +02:00
|
|
|
|
2020-10-02 10:38:57 +02:00
|
|
|
kms_crtc = meta_crtc_kms_get_kms_crtc (META_CRTC_KMS (onscreen_native->crtc));
|
|
|
|
kms_device = meta_kms_crtc_get_device (kms_crtc);
|
2020-10-02 16:48:34 +02:00
|
|
|
|
2020-10-02 18:37:18 +02:00
|
|
|
meta_topic (META_DEBUG_KMS,
|
|
|
|
"Posting direct scanout update for CRTC %u (%s)",
|
|
|
|
meta_kms_crtc_get_id (kms_crtc),
|
|
|
|
meta_kms_device_get_path (kms_device));
|
|
|
|
|
2020-10-02 16:48:34 +02:00
|
|
|
flags = META_KMS_UPDATE_FLAG_PRESERVE_ON_ERROR;
|
|
|
|
kms_feedback = meta_kms_post_pending_update_sync (kms, kms_device, flags);
|
2020-10-09 22:29:24 +02:00
|
|
|
switch (meta_kms_feedback_get_result (kms_feedback))
|
2020-09-10 17:41:06 +02:00
|
|
|
{
|
2020-10-09 22:29:24 +02:00
|
|
|
case META_KMS_FEEDBACK_PASSED:
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
break;
|
|
|
|
case META_KMS_FEEDBACK_FAILED:
|
|
|
|
feedback_error = meta_kms_feedback_get_error (kms_feedback);
|
2020-09-10 17:41:06 +02:00
|
|
|
|
|
|
|
if (g_error_matches (feedback_error,
|
|
|
|
G_IO_ERROR, G_IO_ERROR_PERMISSION_DENIED))
|
2020-10-09 22:29:24 +02:00
|
|
|
break;
|
2020-09-10 17:41:06 +02:00
|
|
|
|
|
|
|
g_clear_object (&onscreen_native->gbm.next_fb);
|
|
|
|
g_propagate_error (error, g_error_copy (feedback_error));
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
2019-09-12 10:47:46 +02:00
|
|
|
}
|
|
|
|
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean
|
2016-06-01 17:53:07 +08:00
|
|
|
meta_renderer_native_init_egl_context (CoglContext *cogl_context,
|
2016-08-18 10:21:12 +08:00
|
|
|
GError **error)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2016-10-20 15:59:09 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
2016-10-20 15:59:09 +08:00
|
|
|
#endif
|
|
|
|
|
2016-06-01 17:53:07 +08:00
|
|
|
COGL_FLAGS_SET (cogl_context->winsys_features,
|
2016-05-09 19:51:29 +08:00
|
|
|
COGL_WINSYS_FEATURE_SWAP_BUFFERS_EVENT,
|
|
|
|
TRUE);
|
2016-06-01 17:53:07 +08:00
|
|
|
COGL_FLAGS_SET (cogl_context->winsys_features,
|
2016-05-09 19:51:29 +08:00
|
|
|
COGL_WINSYS_FEATURE_SYNC_AND_COMPLETE_EVENT,
|
|
|
|
TRUE);
|
2016-06-08 17:05:31 +08:00
|
|
|
COGL_FLAGS_SET (cogl_context->winsys_features,
|
|
|
|
COGL_WINSYS_FEATURE_MULTIPLE_ONSCREEN,
|
|
|
|
TRUE);
|
|
|
|
|
2016-10-20 15:59:09 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
2017-07-25 10:21:04 +08:00
|
|
|
if (renderer_gpu_data->mode == META_RENDERER_NATIVE_MODE_EGL_DEVICE)
|
2016-10-20 15:59:09 +08:00
|
|
|
COGL_FLAGS_SET (cogl_context->features,
|
|
|
|
COGL_FEATURE_ID_TEXTURE_EGL_IMAGE_EXTERNAL, TRUE);
|
|
|
|
#endif
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
2017-07-25 10:21:04 +08:00
|
|
|
should_surface_be_sharable (CoglOnscreen *onscreen)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
if (META_GPU_KMS (meta_crtc_get_gpu (onscreen_native->crtc)) ==
|
|
|
|
onscreen_native->render_gpu)
|
2017-07-25 10:21:04 +08:00
|
|
|
return FALSE;
|
2020-01-17 21:08:02 +01:00
|
|
|
else
|
|
|
|
return TRUE;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_renderer_native_create_surface_gbm (CoglOnscreen *onscreen,
|
2016-08-18 11:11:30 +08:00
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
struct gbm_surface **gbm_surface,
|
|
|
|
EGLSurface *egl_surface,
|
|
|
|
GError **error)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaRendererNative *renderer_native = onscreen_native->renderer_native;
|
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2016-06-08 17:05:31 +08:00
|
|
|
CoglDisplay *cogl_display = cogl_context->display;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
2018-04-02 17:14:53 +01:00
|
|
|
struct gbm_surface *new_gbm_surface = NULL;
|
2016-06-08 17:05:31 +08:00
|
|
|
EGLNativeWindowType egl_native_window;
|
|
|
|
EGLSurface new_egl_surface;
|
2020-10-12 16:57:45 +08:00
|
|
|
uint32_t format;
|
2017-08-03 15:42:50 +01:00
|
|
|
GArray *modifiers;
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data =
|
|
|
|
meta_renderer_native_get_gpu_data (renderer_native,
|
2017-07-25 10:21:04 +08:00
|
|
|
onscreen_native->render_gpu);
|
2017-08-03 15:42:50 +01:00
|
|
|
|
2020-10-12 16:57:45 +08:00
|
|
|
format = get_gbm_format_from_egl (egl,
|
|
|
|
cogl_renderer_egl->edpy,
|
|
|
|
cogl_display_egl->egl_config);
|
|
|
|
|
2018-04-11 14:39:15 +02:00
|
|
|
if (renderer_native->use_modifiers)
|
|
|
|
modifiers = get_supported_modifiers (onscreen, format);
|
|
|
|
else
|
|
|
|
modifiers = NULL;
|
2017-08-03 15:42:50 +01:00
|
|
|
|
|
|
|
if (modifiers)
|
|
|
|
{
|
|
|
|
new_gbm_surface =
|
|
|
|
gbm_surface_create_with_modifiers (renderer_gpu_data->gbm.device,
|
|
|
|
width, height, format,
|
|
|
|
(uint64_t *) modifiers->data,
|
|
|
|
modifiers->len);
|
|
|
|
g_array_free (modifiers, TRUE);
|
|
|
|
}
|
2018-04-02 17:14:53 +01:00
|
|
|
|
|
|
|
if (!new_gbm_surface)
|
2017-08-03 15:42:50 +01:00
|
|
|
{
|
|
|
|
uint32_t flags = GBM_BO_USE_SCANOUT | GBM_BO_USE_RENDERING;
|
|
|
|
|
|
|
|
if (should_surface_be_sharable (onscreen))
|
|
|
|
flags |= GBM_BO_USE_LINEAR;
|
|
|
|
|
|
|
|
new_gbm_surface = gbm_surface_create (renderer_gpu_data->gbm.device,
|
|
|
|
width, height,
|
|
|
|
format,
|
|
|
|
flags);
|
|
|
|
}
|
2016-06-08 17:05:31 +08:00
|
|
|
|
|
|
|
if (!new_gbm_surface)
|
|
|
|
{
|
|
|
|
g_set_error (error, COGL_WINSYS_ERROR,
|
|
|
|
COGL_WINSYS_ERROR_CREATE_ONSCREEN,
|
|
|
|
"Failed to allocate surface");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
egl_native_window = (EGLNativeWindowType) new_gbm_surface;
|
2017-07-24 16:19:55 +08:00
|
|
|
new_egl_surface =
|
|
|
|
meta_egl_create_window_surface (egl,
|
|
|
|
cogl_renderer_egl->edpy,
|
|
|
|
cogl_display_egl->egl_config,
|
|
|
|
egl_native_window,
|
|
|
|
NULL,
|
|
|
|
error);
|
2016-06-08 17:05:31 +08:00
|
|
|
if (new_egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
gbm_surface_destroy (new_gbm_surface);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*gbm_surface = new_gbm_surface;
|
|
|
|
*egl_surface = new_egl_surface;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
static gboolean
|
2020-01-17 21:08:02 +01:00
|
|
|
meta_renderer_native_create_surface_egl_device (CoglOnscreen *onscreen,
|
|
|
|
int width,
|
|
|
|
int height,
|
|
|
|
EGLStreamKHR *out_egl_stream,
|
|
|
|
EGLSurface *out_egl_surface,
|
|
|
|
GError **error)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2017-07-25 10:21:04 +08:00
|
|
|
CoglDisplay *cogl_display = cogl_context->display;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
2016-08-18 11:28:59 +08:00
|
|
|
CoglRenderer *cogl_renderer = cogl_display->renderer;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data = cogl_renderer_egl->platform;
|
|
|
|
MetaEgl *egl =
|
|
|
|
meta_renderer_native_get_egl (renderer_gpu_data->renderer_native);
|
|
|
|
EGLDisplay egl_display = renderer_gpu_data->egl_display;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLConfig egl_config;
|
|
|
|
EGLStreamKHR egl_stream;
|
|
|
|
EGLSurface egl_surface;
|
|
|
|
EGLint num_layers;
|
|
|
|
EGLOutputLayerEXT output_layer;
|
2016-12-13 21:44:16 +08:00
|
|
|
EGLAttrib output_attribs[3];
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLint stream_attribs[] = {
|
2019-01-11 16:03:37 +01:00
|
|
|
EGL_STREAM_FIFO_LENGTH_KHR, 0,
|
2016-08-18 11:28:59 +08:00
|
|
|
EGL_CONSUMER_AUTO_ACQUIRE_EXT, EGL_FALSE,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
EGLint stream_producer_attribs[] = {
|
|
|
|
EGL_WIDTH, width,
|
|
|
|
EGL_HEIGHT, height,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
egl_stream = meta_egl_create_stream (egl, egl_display, stream_attribs, error);
|
|
|
|
if (egl_stream == EGL_NO_STREAM_KHR)
|
|
|
|
return FALSE;
|
|
|
|
|
2016-12-13 21:44:16 +08:00
|
|
|
output_attribs[0] = EGL_DRM_CRTC_EXT;
|
2020-02-25 11:34:43 +01:00
|
|
|
output_attribs[1] = meta_crtc_get_id (onscreen_native->crtc);
|
2016-12-13 21:44:16 +08:00
|
|
|
output_attribs[2] = EGL_NONE;
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
if (!meta_egl_get_output_layers (egl, egl_display,
|
|
|
|
output_attribs,
|
|
|
|
&output_layer, 1, &num_layers,
|
|
|
|
error))
|
|
|
|
{
|
|
|
|
meta_egl_destroy_stream (egl, egl_display, egl_stream, NULL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (num_layers < 1)
|
|
|
|
{
|
|
|
|
meta_egl_destroy_stream (egl, egl_display, egl_stream, NULL);
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Unable to find output layers.");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!meta_egl_stream_consumer_output (egl, egl_display,
|
|
|
|
egl_stream, output_layer,
|
|
|
|
error))
|
|
|
|
{
|
|
|
|
meta_egl_destroy_stream (egl, egl_display, egl_stream, NULL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
egl_config = cogl_display_egl->egl_config;
|
2016-08-18 11:28:59 +08:00
|
|
|
egl_surface = meta_egl_create_stream_producer_surface (egl,
|
|
|
|
egl_display,
|
|
|
|
egl_config,
|
|
|
|
egl_stream,
|
|
|
|
stream_producer_attribs,
|
|
|
|
error);
|
|
|
|
if (egl_surface == EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
meta_egl_destroy_stream (egl, egl_display, egl_stream, NULL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*out_egl_stream = egl_stream;
|
|
|
|
*out_egl_surface = egl_surface;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
2017-10-09 13:39:50 +02:00
|
|
|
#endif /* HAVE_EGL_DEVICE */
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean
|
2016-05-27 15:13:09 +08:00
|
|
|
meta_renderer_native_init_onscreen (CoglOnscreen *onscreen,
|
2016-08-18 10:21:12 +08:00
|
|
|
GError **error)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2016-06-01 17:53:07 +08:00
|
|
|
CoglDisplay *cogl_display = cogl_context->display;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
|
|
|
CoglOnscreenEGL *onscreen_egl;
|
2016-05-27 15:07:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2019-06-17 23:42:01 +02:00
|
|
|
g_return_val_if_fail (cogl_display_egl->egl_context, FALSE);
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
onscreen_egl = g_slice_new0 (CoglOnscreenEGL);
|
|
|
|
cogl_onscreen_set_winsys (onscreen, onscreen_egl);
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2016-05-27 15:07:41 +08:00
|
|
|
onscreen_native = g_slice_new0 (MetaOnscreenNative);
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_egl->platform = onscreen_native;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2016-09-29 22:46:34 +08:00
|
|
|
/*
|
|
|
|
* Don't actually initialize anything here, since we may not have the
|
|
|
|
* information available yet, and there is no way to pass it at this stage.
|
|
|
|
* To properly allocate a MetaOnscreenNative, the caller must call
|
|
|
|
* meta_onscreen_native_allocate() after cogl_framebuffer_allocate().
|
|
|
|
*
|
|
|
|
* TODO: Turn CoglFramebuffer/CoglOnscreen into GObjects, so it's possible
|
|
|
|
* to add backend specific properties.
|
|
|
|
*/
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
meta_onscreen_native_allocate (CoglOnscreen *onscreen,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-09-29 22:46:34 +08:00
|
|
|
struct gbm_surface *gbm_surface;
|
|
|
|
EGLSurface egl_surface;
|
|
|
|
int width;
|
|
|
|
int height;
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
2020-09-29 09:40:53 +02:00
|
|
|
MetaKmsDevice *render_kms_device;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLStreamKHR egl_stream;
|
|
|
|
#endif
|
2016-09-29 22:46:34 +08:00
|
|
|
|
2016-05-09 19:51:29 +08: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-06-08 17:05:31 +08:00
|
|
|
width = cogl_framebuffer_get_width (framebuffer);
|
|
|
|
height = cogl_framebuffer_get_height (framebuffer);
|
|
|
|
if (width == 0 || height == 0)
|
2016-05-09 19:51:29 +08:00
|
|
|
return TRUE;
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data =
|
2017-07-25 10:21:04 +08:00
|
|
|
meta_renderer_native_get_gpu_data (onscreen_native->renderer_native,
|
|
|
|
onscreen_native->render_gpu);
|
2017-07-10 18:19:32 +08:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!meta_renderer_native_create_surface_gbm (onscreen,
|
2016-08-18 11:28:59 +08:00
|
|
|
width, height,
|
|
|
|
&gbm_surface,
|
|
|
|
&egl_surface,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
2016-08-18 11:11:30 +08:00
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
onscreen_native->gbm.surface = gbm_surface;
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_egl->egl_surface = egl_surface;
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
2020-09-29 09:40:53 +02:00
|
|
|
render_kms_device =
|
|
|
|
meta_gpu_kms_get_kms_device (onscreen_native->render_gpu);
|
|
|
|
onscreen_native->egl.dumb_fb =
|
|
|
|
meta_drm_buffer_dumb_new (render_kms_device,
|
|
|
|
width, height,
|
|
|
|
DRM_FORMAT_XRGB8888,
|
|
|
|
error);
|
|
|
|
if (!onscreen_native->egl.dumb_fb)
|
2016-08-18 11:28:59 +08:00
|
|
|
return FALSE;
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!meta_renderer_native_create_surface_egl_device (onscreen,
|
2016-08-18 11:28:59 +08:00
|
|
|
width, height,
|
|
|
|
&egl_stream,
|
|
|
|
&egl_surface,
|
|
|
|
error))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
onscreen_native->egl.stream = egl_stream;
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_egl->egl_surface = egl_surface;
|
2016-08-18 11:28:59 +08:00
|
|
|
break;
|
|
|
|
#endif /* HAVE_EGL_DEVICE */
|
|
|
|
}
|
2016-08-18 11:11:30 +08:00
|
|
|
|
|
|
|
return TRUE;
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2019-06-17 19:16:12 +02:00
|
|
|
static void
|
|
|
|
destroy_egl_surface (CoglOnscreen *onscreen)
|
|
|
|
{
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2019-06-17 19:16:12 +02:00
|
|
|
|
|
|
|
if (onscreen_egl->egl_surface != EGL_NO_SURFACE)
|
|
|
|
{
|
|
|
|
MetaOnscreenNative *onscreen_native = onscreen_egl->platform;
|
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2019-06-17 19:16:12 +02:00
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
|
|
|
|
|
|
|
meta_egl_destroy_surface (egl,
|
|
|
|
cogl_renderer_egl->edpy,
|
|
|
|
onscreen_egl->egl_surface,
|
|
|
|
NULL);
|
|
|
|
onscreen_egl->egl_surface = EGL_NO_SURFACE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
static void
|
2016-05-27 15:13:09 +08:00
|
|
|
meta_renderer_native_release_onscreen (CoglOnscreen *onscreen)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
|
|
|
CoglFramebuffer *framebuffer = COGL_FRAMEBUFFER (onscreen);
|
2020-10-12 23:31:38 +02:00
|
|
|
CoglContext *cogl_context = cogl_framebuffer_get_context (framebuffer);
|
2019-06-17 18:18:42 +02:00
|
|
|
CoglDisplay *cogl_display = cogl_context_get_display (cogl_context);
|
|
|
|
CoglDisplayEGL *cogl_display_egl = cogl_display->winsys;
|
2020-10-17 00:11:27 +02:00
|
|
|
CoglOnscreenEGL *onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2016-05-27 15:07:41 +08:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
2019-01-11 15:35:42 +01:00
|
|
|
MetaRendererNative *renderer_native;
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
/* If we never successfully allocated then there's nothing to do */
|
2017-07-14 15:20:41 +08:00
|
|
|
if (onscreen_egl == NULL)
|
2016-05-09 19:51:29 +08:00
|
|
|
return;
|
|
|
|
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_native = onscreen_egl->platform;
|
2019-01-11 15:35:42 +01:00
|
|
|
renderer_native = onscreen_native->renderer_native;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2019-06-17 18:18:42 +02:00
|
|
|
if (onscreen_egl->egl_surface != EGL_NO_SURFACE &&
|
|
|
|
(cogl_display_egl->current_draw_surface == onscreen_egl->egl_surface ||
|
|
|
|
cogl_display_egl->current_read_surface == onscreen_egl->egl_surface))
|
|
|
|
{
|
|
|
|
if (!_cogl_winsys_egl_make_current (cogl_display,
|
|
|
|
cogl_display_egl->dummy_surface,
|
|
|
|
cogl_display_egl->dummy_surface,
|
|
|
|
cogl_display_egl->egl_context))
|
|
|
|
g_warning ("Failed to clear current context");
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data =
|
2019-01-11 15:35:42 +01:00
|
|
|
meta_renderer_native_get_gpu_data (renderer_native,
|
2017-07-25 10:21:04 +08:00
|
|
|
onscreen_native->render_gpu);
|
2017-07-10 18:19:32 +08:00
|
|
|
switch (renderer_gpu_data->mode)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2016-08-18 11:28:59 +08:00
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
/* flip state takes a reference on the onscreen so there should
|
|
|
|
* never be outstanding flips when we reach here. */
|
2018-01-30 17:47:17 +08:00
|
|
|
g_return_if_fail (onscreen_native->gbm.next_fb == NULL);
|
2016-08-18 11:28:59 +08:00
|
|
|
|
|
|
|
free_current_bo (onscreen);
|
|
|
|
|
2019-06-17 19:16:12 +02:00
|
|
|
destroy_egl_surface (onscreen);
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
if (onscreen_native->gbm.surface)
|
|
|
|
{
|
|
|
|
gbm_surface_destroy (onscreen_native->gbm.surface);
|
|
|
|
onscreen_native->gbm.surface = NULL;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
2020-09-29 09:40:53 +02:00
|
|
|
g_clear_object (&onscreen_native->egl.dumb_fb);
|
2019-06-17 19:16:12 +02:00
|
|
|
|
|
|
|
destroy_egl_surface (onscreen);
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
if (onscreen_native->egl.stream != EGL_NO_STREAM_KHR)
|
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaEgl *egl = meta_onscreen_native_get_egl (onscreen_native);
|
2019-06-27 09:24:19 +02:00
|
|
|
CoglRenderer *cogl_renderer = cogl_context->display->renderer;
|
|
|
|
CoglRendererEGL *cogl_renderer_egl = cogl_renderer->winsys;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
|
|
|
meta_egl_destroy_stream (egl,
|
2017-07-14 15:20:41 +08:00
|
|
|
cogl_renderer_egl->edpy,
|
2016-08-18 11:28:59 +08:00
|
|
|
onscreen_native->egl.stream,
|
|
|
|
NULL);
|
|
|
|
onscreen_native->egl.stream = EGL_NO_STREAM_KHR;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
#endif /* HAVE_EGL_DEVICE */
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
g_clear_pointer (&onscreen_native->secondary_gpu_state,
|
|
|
|
secondary_gpu_state_free);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2016-05-27 15:07:41 +08:00
|
|
|
g_slice_free (MetaOnscreenNative, onscreen_native);
|
2020-10-17 00:11:27 +02:00
|
|
|
g_slice_free (CoglOnscreenEGL, cogl_onscreen_get_winsys (onscreen));
|
|
|
|
cogl_onscreen_set_winsys (onscreen, NULL);
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static const CoglWinsysEGLVtable
|
|
|
|
_cogl_winsys_egl_vtable = {
|
2016-09-26 15:54:27 +08:00
|
|
|
.add_config_attributes = meta_renderer_native_add_egl_config_attributes,
|
2018-02-23 22:14:07 +08:00
|
|
|
.choose_config = meta_renderer_native_choose_egl_config,
|
2016-05-27 15:13:09 +08: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 19:51:29 +08:00
|
|
|
};
|
|
|
|
|
2019-06-19 20:57:14 +02:00
|
|
|
static void
|
2016-05-09 19:51:29 +08:00
|
|
|
meta_renderer_native_queue_modes_reset (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2016-06-08 17:05:31 +08:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
g_list_free (renderer_native->pending_mode_set_views);
|
|
|
|
renderer_native->pending_mode_set_views =
|
|
|
|
g_list_copy (meta_renderer_get_views (renderer));
|
|
|
|
renderer_native->pending_mode_set = TRUE;
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
g_clear_handle_id (&renderer_native->mode_set_failed_feedback_source_id,
|
|
|
|
g_source_remove);
|
2017-09-28 11:59:27 -04:00
|
|
|
|
2020-10-02 18:37:18 +02:00
|
|
|
meta_topic (META_DEBUG_KMS, "Queue mode set");
|
2016-05-09 19:51:29 +08:00
|
|
|
}
|
|
|
|
|
2016-08-01 03:47:11 +02:00
|
|
|
static CoglOnscreen *
|
2017-07-25 10:21:04 +08:00
|
|
|
meta_renderer_native_create_onscreen (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *render_gpu,
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaOutput *output,
|
|
|
|
MetaCrtc *crtc,
|
2017-07-10 18:19:32 +08:00
|
|
|
CoglContext *context,
|
2020-01-17 21:08:02 +01:00
|
|
|
int width,
|
|
|
|
int height,
|
2017-07-24 17:35:55 +08:00
|
|
|
GError **error)
|
2016-08-01 03:47:11 +02:00
|
|
|
{
|
|
|
|
CoglOnscreen *onscreen;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglOnscreenEGL *onscreen_egl;
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
2016-08-01 03:47:11 +02:00
|
|
|
|
|
|
|
onscreen = cogl_onscreen_new (context, width, height);
|
2018-07-10 17:46:02 +08:00
|
|
|
|
2017-07-24 17:35:55 +08:00
|
|
|
if (!cogl_framebuffer_allocate (COGL_FRAMEBUFFER (onscreen), error))
|
2016-08-01 03:47:11 +02:00
|
|
|
{
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (onscreen);
|
2016-08-01 03:47:11 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_native = onscreen_egl->platform;
|
2017-07-25 10:21:04 +08:00
|
|
|
onscreen_native->renderer_native = renderer_native;
|
|
|
|
onscreen_native->render_gpu = render_gpu;
|
2020-01-17 21:08:02 +01:00
|
|
|
onscreen_native->output = output;
|
|
|
|
onscreen_native->crtc = crtc;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
if (META_GPU_KMS (meta_crtc_get_gpu (crtc)) != render_gpu)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
2020-01-17 21:08:02 +01:00
|
|
|
if (!init_secondary_gpu_state (renderer_native, onscreen, error))
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
2017-07-10 18:19:32 +08:00
|
|
|
|
2016-08-01 03:47:11 +02:00
|
|
|
return onscreen;
|
|
|
|
}
|
|
|
|
|
2016-08-01 03:48:30 +02:00
|
|
|
static CoglOffscreen *
|
|
|
|
meta_renderer_native_create_offscreen (MetaRendererNative *renderer,
|
|
|
|
CoglContext *context,
|
|
|
|
gint view_width,
|
2017-07-24 17:35:55 +08:00
|
|
|
gint view_height,
|
|
|
|
GError **error)
|
2016-08-01 03:48:30 +02:00
|
|
|
{
|
|
|
|
CoglOffscreen *fb;
|
|
|
|
CoglTexture2D *tex;
|
|
|
|
|
|
|
|
tex = cogl_texture_2d_new_with_size (context, view_width, view_height);
|
|
|
|
cogl_primitive_texture_set_auto_mipmap (COGL_PRIMITIVE_TEXTURE (tex), FALSE);
|
|
|
|
|
2017-07-24 17:35:55 +08:00
|
|
|
if (!cogl_texture_allocate (COGL_TEXTURE (tex), error))
|
2016-08-01 03:48:30 +02:00
|
|
|
{
|
|
|
|
cogl_object_unref (tex);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
fb = cogl_offscreen_new_with_texture (COGL_TEXTURE (tex));
|
|
|
|
cogl_object_unref (tex);
|
2017-07-24 17:35:55 +08:00
|
|
|
if (!cogl_framebuffer_allocate (COGL_FRAMEBUFFER (fb), error))
|
2016-08-01 03:48:30 +02:00
|
|
|
{
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (fb);
|
2016-08-01 03:48:30 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return fb;
|
|
|
|
}
|
|
|
|
|
2018-06-27 17:19:27 +08:00
|
|
|
static int64_t
|
|
|
|
meta_renderer_native_get_clock_time (CoglContext *context)
|
|
|
|
{
|
|
|
|
CoglRenderer *cogl_renderer = cogl_context_get_renderer (context);
|
|
|
|
MetaGpuKms *gpu_kms = cogl_renderer->custom_winsys_user_data;
|
|
|
|
|
|
|
|
return meta_gpu_kms_get_current_time_ns (gpu_kms);
|
|
|
|
}
|
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
static const CoglWinsysVtable *
|
2016-09-23 11:49:10 +08:00
|
|
|
get_native_cogl_winsys_vtable (CoglRenderer *cogl_renderer)
|
2016-05-09 19:51:29 +08:00
|
|
|
{
|
2016-08-18 10:21:12 +08:00
|
|
|
static gboolean vtable_inited = FALSE;
|
2016-05-09 19:51:29 +08:00
|
|
|
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 15:13:09 +08:00
|
|
|
vtable.renderer_connect = meta_renderer_native_connect;
|
|
|
|
vtable.renderer_disconnect = meta_renderer_native_disconnect;
|
2019-12-09 10:05:37 -03:00
|
|
|
vtable.renderer_create_dma_buf = meta_renderer_native_create_dma_buf;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2016-05-27 15:13:09 +08:00
|
|
|
vtable.onscreen_init = meta_renderer_native_init_onscreen;
|
|
|
|
vtable.onscreen_deinit = meta_renderer_native_release_onscreen;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
|
|
|
/* The KMS winsys doesn't support swap region */
|
|
|
|
vtable.onscreen_swap_region = NULL;
|
|
|
|
vtable.onscreen_swap_buffers_with_damage =
|
2016-05-27 15:13:09 +08:00
|
|
|
meta_onscreen_native_swap_buffers_with_damage;
|
2019-09-12 10:47:46 +02:00
|
|
|
vtable.onscreen_direct_scanout = meta_onscreen_native_direct_scanout;
|
2016-05-09 19:51:29 +08:00
|
|
|
|
2018-06-27 17:19:27 +08:00
|
|
|
vtable.context_get_clock_time = meta_renderer_native_get_clock_time;
|
|
|
|
|
2016-05-09 19:51:29 +08:00
|
|
|
vtable_inited = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return &vtable;
|
|
|
|
}
|
|
|
|
|
2016-05-09 09:59:54 +08:00
|
|
|
static CoglRenderer *
|
2017-07-25 10:21:04 +08:00
|
|
|
create_cogl_renderer_for_gpu (MetaGpuKms *gpu_kms)
|
2016-05-09 09:59:54 +08:00
|
|
|
{
|
|
|
|
CoglRenderer *cogl_renderer;
|
|
|
|
|
|
|
|
cogl_renderer = cogl_renderer_new ();
|
|
|
|
cogl_renderer_set_custom_winsys (cogl_renderer,
|
2017-07-24 16:03:02 +08:00
|
|
|
get_native_cogl_winsys_vtable,
|
2017-07-25 10:21:04 +08:00
|
|
|
gpu_kms);
|
2016-05-09 09:59:54 +08:00
|
|
|
|
|
|
|
return cogl_renderer;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static CoglRenderer *
|
|
|
|
meta_renderer_native_create_cogl_renderer (MetaRenderer *renderer)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (renderer);
|
|
|
|
|
2018-12-10 14:19:25 +02:00
|
|
|
return create_cogl_renderer_for_gpu (renderer_native->primary_gpu_kms);
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
static void
|
|
|
|
meta_onscreen_native_set_view (CoglOnscreen *onscreen,
|
|
|
|
MetaRendererView *view)
|
|
|
|
{
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglOnscreenEGL *onscreen_egl;
|
2016-06-08 17:05:31 +08:00
|
|
|
MetaOnscreenNative *onscreen_native;
|
|
|
|
|
2020-10-17 00:11:27 +02:00
|
|
|
onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_native = onscreen_egl->platform;
|
2016-06-08 17:05:31 +08:00
|
|
|
onscreen_native->view = view;
|
|
|
|
}
|
|
|
|
|
2017-03-08 16:05:00 +08:00
|
|
|
static MetaMonitorTransform
|
|
|
|
calculate_view_transform (MetaMonitorManager *monitor_manager,
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaLogicalMonitor *logical_monitor,
|
|
|
|
MetaOutput *output,
|
|
|
|
MetaCrtc *crtc)
|
2017-03-08 16:05:00 +08:00
|
|
|
{
|
2018-07-23 21:36:57 +02:00
|
|
|
MetaMonitorTransform crtc_transform;
|
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
crtc = meta_output_get_assigned_crtc (output);
|
2018-07-23 21:36:57 +02:00
|
|
|
crtc_transform =
|
2020-01-17 21:08:02 +01:00
|
|
|
meta_output_logical_to_crtc_transform (output, logical_monitor->transform);
|
2017-03-08 16:05:00 +08:00
|
|
|
|
|
|
|
if (meta_monitor_manager_is_transform_handled (monitor_manager,
|
2017-11-03 11:25:30 +01:00
|
|
|
crtc,
|
2018-07-23 21:36:57 +02:00
|
|
|
crtc_transform))
|
2017-03-08 16:05:00 +08:00
|
|
|
return META_MONITOR_TRANSFORM_NORMAL;
|
|
|
|
else
|
2018-07-23 21:36:57 +02:00
|
|
|
return crtc_transform;
|
2017-03-08 16:05:00 +08:00
|
|
|
}
|
|
|
|
|
2018-10-09 15:55:23 +02:00
|
|
|
static gboolean
|
|
|
|
should_force_shadow_fb (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *primary_gpu)
|
|
|
|
{
|
2019-10-01 11:54:32 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
2020-05-05 17:06:35 +02:00
|
|
|
CoglContext *cogl_context =
|
|
|
|
cogl_context_from_renderer_native (renderer_native);
|
2018-10-09 15:55:23 +02:00
|
|
|
int kms_fd;
|
|
|
|
uint64_t prefer_shadow = 0;
|
|
|
|
|
2019-10-01 11:54:32 +02:00
|
|
|
if (meta_renderer_is_hardware_accelerated (renderer))
|
|
|
|
return FALSE;
|
2018-10-09 15:55:23 +02:00
|
|
|
|
2020-05-05 17:06:35 +02:00
|
|
|
if (!cogl_has_feature (cogl_context, COGL_FEATURE_ID_BLIT_FRAMEBUFFER))
|
|
|
|
return FALSE;
|
|
|
|
|
2018-10-09 15:55:23 +02:00
|
|
|
kms_fd = meta_gpu_kms_get_fd (primary_gpu);
|
|
|
|
if (drmGetCap (kms_fd, DRM_CAP_DUMB_PREFER_SHADOW, &prefer_shadow) == 0)
|
|
|
|
{
|
|
|
|
if (prefer_shadow)
|
|
|
|
{
|
|
|
|
static gboolean logged_once = FALSE;
|
|
|
|
|
|
|
|
if (!logged_once)
|
|
|
|
{
|
|
|
|
g_message ("Forcing shadow framebuffer");
|
|
|
|
logged_once = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
static MetaRendererView *
|
2016-11-25 14:31:38 +08:00
|
|
|
meta_renderer_native_create_view (MetaRenderer *renderer,
|
2020-01-17 21:08:02 +01:00
|
|
|
MetaLogicalMonitor *logical_monitor,
|
|
|
|
MetaOutput *output,
|
|
|
|
MetaCrtc *crtc)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
2017-07-06 16:00:56 +08:00
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (renderer);
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
2017-03-08 16:05:00 +08:00
|
|
|
MetaMonitorManager *monitor_manager =
|
2019-01-11 15:35:42 +01:00
|
|
|
meta_backend_get_monitor_manager (backend);
|
2017-07-10 18:19:32 +08:00
|
|
|
CoglContext *cogl_context =
|
2018-10-09 15:55:23 +02:00
|
|
|
cogl_context_from_renderer_native (renderer_native);
|
2016-08-01 03:48:30 +02:00
|
|
|
CoglDisplay *cogl_display = cogl_context_get_display (cogl_context);
|
2017-07-25 10:21:04 +08:00
|
|
|
CoglDisplayEGL *cogl_display_egl;
|
2017-07-14 15:20:41 +08:00
|
|
|
CoglOnscreenEGL *onscreen_egl;
|
2020-02-26 10:37:53 +01:00
|
|
|
const MetaCrtcConfig *crtc_config;
|
2020-02-27 00:08:58 +01:00
|
|
|
const MetaCrtcModeInfo *crtc_mode_info;
|
2017-03-08 16:05:00 +08:00
|
|
|
MetaMonitorTransform view_transform;
|
2016-08-01 03:47:11 +02:00
|
|
|
CoglOnscreen *onscreen = NULL;
|
2016-08-01 03:48:30 +02:00
|
|
|
CoglOffscreen *offscreen = NULL;
|
2020-04-30 17:53:30 +02:00
|
|
|
gboolean use_shadowfb;
|
2017-05-25 16:12:51 +08:00
|
|
|
float scale;
|
2020-01-17 21:08:02 +01:00
|
|
|
int onscreen_width;
|
|
|
|
int onscreen_height;
|
2020-04-14 10:44:16 +02:00
|
|
|
MetaRectangle view_layout;
|
2016-06-08 17:05:31 +08:00
|
|
|
MetaRendererView *view;
|
2016-09-29 22:46:34 +08:00
|
|
|
GError *error = NULL;
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2020-02-26 10:37:53 +01:00
|
|
|
crtc_config = meta_crtc_get_config (crtc);
|
2020-02-27 00:08:58 +01:00
|
|
|
crtc_mode_info = meta_crtc_mode_get_info (crtc_config->mode);
|
|
|
|
onscreen_width = crtc_mode_info->width;
|
|
|
|
onscreen_height = crtc_mode_info->height;
|
2017-02-24 18:10:52 +08:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
onscreen = meta_renderer_native_create_onscreen (renderer_native,
|
2018-12-10 14:19:25 +02:00
|
|
|
renderer_native->primary_gpu_kms,
|
2020-01-17 21:08:02 +01:00
|
|
|
output,
|
|
|
|
crtc,
|
2016-08-01 03:47:11 +02:00
|
|
|
cogl_context,
|
2020-01-17 21:08:02 +01:00
|
|
|
onscreen_width,
|
|
|
|
onscreen_height,
|
2017-07-24 17:35:55 +08:00
|
|
|
&error);
|
2016-08-01 03:47:11 +02:00
|
|
|
if (!onscreen)
|
2017-07-24 17:35:55 +08:00
|
|
|
g_error ("Failed to allocate onscreen framebuffer: %s", error->message);
|
2016-06-08 17:05:31 +08:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
view_transform = calculate_view_transform (monitor_manager,
|
|
|
|
logical_monitor,
|
|
|
|
output,
|
|
|
|
crtc);
|
2020-01-17 19:35:19 +01:00
|
|
|
if (view_transform != META_MONITOR_TRANSFORM_NORMAL)
|
2016-08-01 03:48:30 +02:00
|
|
|
{
|
2020-01-17 21:08:02 +01:00
|
|
|
int offscreen_width;
|
|
|
|
int offscreen_height;
|
|
|
|
|
|
|
|
if (meta_monitor_transform_is_rotated (view_transform))
|
|
|
|
{
|
|
|
|
offscreen_width = onscreen_height;
|
|
|
|
offscreen_height = onscreen_width;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
offscreen_width = onscreen_width;
|
|
|
|
offscreen_height = onscreen_height;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
offscreen = meta_renderer_native_create_offscreen (renderer_native,
|
2016-08-01 03:48:30 +02:00
|
|
|
cogl_context,
|
2020-01-17 21:08:02 +01:00
|
|
|
offscreen_width,
|
|
|
|
offscreen_height,
|
2017-07-24 17:35:55 +08:00
|
|
|
&error);
|
2016-08-01 03:48:30 +02:00
|
|
|
if (!offscreen)
|
2017-07-24 17:35:55 +08:00
|
|
|
g_error ("Failed to allocate back buffer texture: %s", error->message);
|
2019-10-22 17:05:46 +02:00
|
|
|
}
|
|
|
|
|
2020-04-30 17:53:30 +02:00
|
|
|
use_shadowfb = should_force_shadow_fb (renderer_native,
|
|
|
|
renderer_native->primary_gpu_kms);
|
2016-08-01 03:48:30 +02:00
|
|
|
|
2020-01-17 21:08:02 +01:00
|
|
|
if (meta_is_stage_views_scaled ())
|
|
|
|
scale = meta_logical_monitor_get_scale (logical_monitor);
|
|
|
|
else
|
|
|
|
scale = 1.0;
|
|
|
|
|
2020-02-26 10:37:53 +01:00
|
|
|
meta_rectangle_from_graphene_rect (&crtc_config->layout,
|
2020-04-14 10:44:16 +02:00
|
|
|
META_ROUNDING_STRATEGY_ROUND,
|
|
|
|
&view_layout);
|
2016-06-08 17:05:31 +08:00
|
|
|
view = g_object_new (META_TYPE_RENDERER_VIEW,
|
2020-04-30 10:42:30 +02:00
|
|
|
"name", meta_output_get_name (output),
|
2020-03-30 17:34:43 +02:00
|
|
|
"stage", meta_backend_get_stage (backend),
|
2020-04-14 10:44:16 +02:00
|
|
|
"layout", &view_layout,
|
2020-04-01 11:13:22 +02:00
|
|
|
"crtc", crtc,
|
2017-05-25 16:12:51 +08:00
|
|
|
"scale", scale,
|
2016-08-01 03:47:11 +02:00
|
|
|
"framebuffer", onscreen,
|
2016-08-01 03:48:30 +02:00
|
|
|
"offscreen", offscreen,
|
2020-04-30 17:53:30 +02:00
|
|
|
"use-shadowfb", use_shadowfb,
|
2017-03-08 16:05:00 +08:00
|
|
|
"transform", view_transform,
|
2020-04-17 09:18:37 +02:00
|
|
|
"refresh-rate", crtc_mode_info->refresh_rate,
|
2016-06-08 17:05:31 +08:00
|
|
|
NULL);
|
2020-10-13 11:35:47 +02:00
|
|
|
g_clear_object (&offscreen);
|
2016-06-08 17:05:31 +08:00
|
|
|
|
|
|
|
meta_onscreen_native_set_view (onscreen, view);
|
|
|
|
|
2016-09-29 22:46:34 +08:00
|
|
|
if (!meta_onscreen_native_allocate (onscreen, &error))
|
|
|
|
{
|
|
|
|
g_warning ("Could not create onscreen: %s", error->message);
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (onscreen);
|
2016-09-29 22:46:34 +08:00
|
|
|
g_object_unref (view);
|
|
|
|
g_error_free (error);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref (onscreen);
|
2016-09-29 22:46:34 +08:00
|
|
|
|
|
|
|
/* Ensure we don't point to stale surfaces when creating the offscreen */
|
2020-10-17 00:11:27 +02:00
|
|
|
onscreen_egl = cogl_onscreen_get_winsys (onscreen);
|
2017-07-25 10:21:04 +08:00
|
|
|
cogl_display_egl = cogl_display->winsys;
|
2016-09-29 22:46:34 +08:00
|
|
|
_cogl_winsys_egl_make_current (cogl_display,
|
2017-07-14 15:20:41 +08:00
|
|
|
onscreen_egl->egl_surface,
|
|
|
|
onscreen_egl->egl_surface,
|
|
|
|
cogl_display_egl->egl_context);
|
2016-09-29 22:46:34 +08:00
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
return view;
|
|
|
|
}
|
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
static void
|
|
|
|
keep_current_onscreens_alive (MetaRenderer *renderer)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (renderer);
|
|
|
|
GList *views;
|
|
|
|
GList *l;
|
|
|
|
|
|
|
|
views = meta_renderer_get_views (renderer);
|
|
|
|
for (l = views; l; l = l->next)
|
|
|
|
{
|
|
|
|
ClutterStageView *stage_view = l->data;
|
|
|
|
CoglFramebuffer *onscreen = clutter_stage_view_get_onscreen (stage_view);
|
|
|
|
|
|
|
|
renderer_native->kept_alive_onscreens =
|
|
|
|
g_list_prepend (renderer_native->kept_alive_onscreens,
|
|
|
|
g_object_ref (onscreen));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-19 20:57:14 +02:00
|
|
|
static void
|
|
|
|
meta_renderer_native_rebuild_views (MetaRenderer *renderer)
|
|
|
|
{
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
2019-06-19 21:14:05 +02:00
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
2019-06-19 20:57:14 +02:00
|
|
|
MetaRendererClass *parent_renderer_class =
|
|
|
|
META_RENDERER_CLASS (meta_renderer_native_parent_class);
|
|
|
|
|
2019-06-19 21:14:05 +02:00
|
|
|
meta_kms_discard_pending_page_flips (kms);
|
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
keep_current_onscreens_alive (renderer);
|
|
|
|
|
2019-06-19 20:57:14 +02:00
|
|
|
parent_renderer_class->rebuild_views (renderer);
|
|
|
|
|
|
|
|
meta_renderer_native_queue_modes_reset (META_RENDERER_NATIVE (renderer));
|
|
|
|
}
|
|
|
|
|
2020-10-10 11:47:58 +02:00
|
|
|
void
|
|
|
|
meta_renderer_native_prepare_frame (MetaRendererNative *renderer_native,
|
|
|
|
MetaRendererView *view,
|
|
|
|
ClutterFrame *frame)
|
|
|
|
{
|
|
|
|
MetaCrtc *crtc = meta_renderer_view_get_crtc (view);
|
|
|
|
MetaCrtcKms *crtc_kms = META_CRTC_KMS (crtc);
|
|
|
|
MetaKmsCrtc *kms_crtc = meta_crtc_kms_get_kms_crtc (META_CRTC_KMS (crtc));
|
|
|
|
MetaKmsDevice *kms_device = meta_kms_crtc_get_device (kms_crtc);;
|
|
|
|
|
|
|
|
meta_crtc_kms_maybe_set_gamma (crtc_kms, kms_device);
|
|
|
|
}
|
|
|
|
|
2020-10-10 01:02:28 +02:00
|
|
|
static void
|
|
|
|
add_onscreen_frame_info (MetaCrtc *crtc)
|
|
|
|
{
|
|
|
|
MetaGpu *gpu = meta_crtc_get_gpu (crtc);
|
|
|
|
MetaBackend *backend = meta_gpu_get_backend (gpu);
|
|
|
|
ClutterBackend *clutter_backend = meta_backend_get_clutter_backend (backend);
|
|
|
|
MetaStageNative *stage_native =
|
|
|
|
meta_clutter_backend_native_get_stage_native (clutter_backend);
|
|
|
|
MetaRenderer *renderer = meta_backend_get_renderer (backend);
|
|
|
|
MetaRendererView *view = meta_renderer_get_view_for_crtc (renderer, crtc);
|
|
|
|
|
|
|
|
clutter_stage_cogl_add_onscreen_frame_info (CLUTTER_STAGE_COGL (stage_native),
|
|
|
|
CLUTTER_STAGE_VIEW (view));
|
|
|
|
}
|
|
|
|
|
2016-06-08 17:05:31 +08:00
|
|
|
void
|
2020-10-10 01:02:28 +02:00
|
|
|
meta_renderer_native_finish_frame (MetaRendererNative *renderer_native,
|
|
|
|
MetaRendererView *view,
|
|
|
|
ClutterFrame *frame)
|
2016-06-08 17:05:31 +08:00
|
|
|
{
|
2020-10-10 01:02:28 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
|
|
|
|
|
|
|
if (!clutter_frame_has_result (frame))
|
|
|
|
{
|
|
|
|
MetaCrtc *crtc = meta_renderer_view_get_crtc (view);
|
|
|
|
MetaKmsCrtc *kms_crtc = meta_crtc_kms_get_kms_crtc (META_CRTC_KMS (crtc));
|
|
|
|
MetaKmsDevice *kms_device = meta_kms_crtc_get_device (kms_crtc);;
|
|
|
|
MetaKmsUpdateFlag flags;
|
|
|
|
MetaKmsUpdate *kms_update;
|
|
|
|
g_autoptr (MetaKmsFeedback) kms_feedback = NULL;
|
|
|
|
const GError *error;
|
|
|
|
|
|
|
|
kms_update = meta_kms_get_pending_update (kms, kms_device);
|
|
|
|
if (!kms_update)
|
|
|
|
{
|
|
|
|
clutter_frame_set_result (frame, CLUTTER_FRAME_RESULT_IDLE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
meta_kms_update_add_page_flip_listener (kms_update,
|
|
|
|
kms_crtc,
|
|
|
|
&page_flip_listener_vtable,
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
META_KMS_PAGE_FLIP_LISTENER_FLAG_NONE,
|
|
|
|
g_object_ref (view),
|
|
|
|
g_object_unref);
|
2020-10-10 01:02:28 +02:00
|
|
|
|
|
|
|
flags = META_KMS_UPDATE_FLAG_NONE;
|
|
|
|
kms_feedback = meta_kms_post_pending_update_sync (kms,
|
|
|
|
kms_device,
|
|
|
|
flags);
|
|
|
|
switch (meta_kms_feedback_get_result (kms_feedback))
|
|
|
|
{
|
|
|
|
case META_KMS_FEEDBACK_PASSED:
|
|
|
|
add_onscreen_frame_info (crtc);
|
|
|
|
clutter_frame_set_result (frame,
|
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
|
|
|
break;
|
|
|
|
case META_KMS_FEEDBACK_FAILED:
|
2020-10-10 01:25:06 +02:00
|
|
|
add_onscreen_frame_info (crtc);
|
2020-10-10 01:02:28 +02:00
|
|
|
clutter_frame_set_result (frame,
|
kms/page-flip: Pass ownership of listener user data along with closure
In order to reliably manage the reference count of the user data passed
to page flip listeners - being the stage view - make the ownership of
this data travel through the different objects that take responsibility
of the next step.
Initially this is the MetaKmsPageFlipListener that belongs to a
MetaKmsUpdate.
When a page flip is successfully queued, the ownership is transferred to
a MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. In the
simple impl device, the MetaKmsPageFlipData is passed to
drmModePageFlip(), then returned back via the DRM event. In the future
atomic impl device, the MetaKmsPageFlipData is stored in a table, then
retrieved when DRM event are handled.
When the DRM events are handled, the page flip listener's interface
callbacks are invoked, and after that, the user data is freed using the
passed GDestroyNotify function, in the main context, the same as where
the interface callbacks were called.
When a page flip fails, the ownership is also transferred to a
MetaKmsPageFlipClosure that is part of a MetaKmsPageFlipData. This page
flip data will be passed to the main context via a callback, where it
will discard the page flip, and free the user data using the provided
GDestroyNotify.
Note that this adds back a page flip listener type flag for telling the
KMS implementation whether to actively discard a page flip via the
interface, or just free the user data. Avoiding discarding via the
interface is needed for the direct scanout case, where we immediately
need to know the result in order to fall back to the composite pipeline
if the direct scanout failed. We do in fact also need active discard via
the interface paths, e.g. in the simple impl device when we're
asynchronously retrying a page flip, so replace the ad-hoc discard paths
in meta-renderer-native.c and replace them by not asking for no-discard
page flip error handling.
Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/1488>
2020-12-16 08:41:14 +01:00
|
|
|
CLUTTER_FRAME_RESULT_PENDING_PRESENTED);
|
2020-10-10 01:02:28 +02:00
|
|
|
|
|
|
|
error = meta_kms_feedback_get_error (kms_feedback);
|
|
|
|
if (!g_error_matches (error,
|
|
|
|
G_IO_ERROR,
|
|
|
|
G_IO_ERROR_PERMISSION_DENIED))
|
|
|
|
g_warning ("Failed to post KMS update: %s", error->message);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2016-06-08 17:05:31 +08:00
|
|
|
}
|
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
static gboolean
|
2018-06-07 16:29:44 -07:00
|
|
|
create_secondary_egl_config (MetaEgl *egl,
|
|
|
|
MetaRendererNativeMode mode,
|
|
|
|
EGLDisplay egl_display,
|
|
|
|
EGLConfig *egl_config,
|
|
|
|
GError **error)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
|
|
|
EGLint attributes[] = {
|
|
|
|
EGL_RED_SIZE, 1,
|
|
|
|
EGL_GREEN_SIZE, 1,
|
|
|
|
EGL_BLUE_SIZE, 1,
|
|
|
|
EGL_ALPHA_SIZE, EGL_DONT_CARE,
|
|
|
|
EGL_BUFFER_SIZE, EGL_DONT_CARE,
|
|
|
|
EGL_RENDERABLE_TYPE, EGL_OPENGL_ES3_BIT,
|
|
|
|
EGL_SURFACE_TYPE, EGL_WINDOW_BIT,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
2018-06-07 16:29:44 -07:00
|
|
|
switch (mode)
|
|
|
|
{
|
|
|
|
case META_RENDERER_NATIVE_MODE_GBM:
|
|
|
|
return choose_egl_config_from_gbm_format (egl,
|
|
|
|
egl_display,
|
|
|
|
attributes,
|
|
|
|
GBM_FORMAT_XRGB8888,
|
|
|
|
egl_config,
|
|
|
|
error);
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
case META_RENDERER_NATIVE_MODE_EGL_DEVICE:
|
|
|
|
return meta_egl_choose_first_config (egl,
|
|
|
|
egl_display,
|
|
|
|
attributes,
|
|
|
|
egl_config,
|
|
|
|
error);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static EGLContext
|
|
|
|
create_secondary_egl_context (MetaEgl *egl,
|
|
|
|
EGLDisplay egl_display,
|
|
|
|
EGLConfig egl_config,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
EGLint attributes[] = {
|
|
|
|
EGL_CONTEXT_CLIENT_VERSION, 3,
|
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
return meta_egl_create_context (egl,
|
|
|
|
egl_display,
|
|
|
|
egl_config,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
attributes,
|
|
|
|
error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_renderer_native_ensure_gles3 (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
|
|
|
|
|
|
|
if (renderer_native->gles3)
|
|
|
|
return;
|
|
|
|
|
|
|
|
renderer_native->gles3 = meta_gles3_new (egl);
|
|
|
|
}
|
|
|
|
|
|
|
|
static gboolean
|
|
|
|
init_secondary_gpu_data_gpu (MetaRendererNativeGpuData *renderer_gpu_data,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = renderer_gpu_data->renderer_native;
|
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
|
|
|
EGLDisplay egl_display = renderer_gpu_data->egl_display;
|
|
|
|
EGLConfig egl_config;
|
|
|
|
EGLContext egl_context;
|
2019-01-20 11:46:46 +01:00
|
|
|
const char **missing_gl_extensions;
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
const char *renderer_str;
|
2017-07-25 10:21:04 +08:00
|
|
|
|
2018-06-07 16:29:44 -07:00
|
|
|
if (!create_secondary_egl_config (egl, renderer_gpu_data->mode, egl_display,
|
|
|
|
&egl_config, error))
|
2017-07-25 10:21:04 +08:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
egl_context = create_secondary_egl_context (egl, egl_display, egl_config, error);
|
|
|
|
if (egl_context == EGL_NO_CONTEXT)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
meta_renderer_native_ensure_gles3 (renderer_native);
|
|
|
|
|
|
|
|
if (!meta_egl_make_current (egl,
|
|
|
|
egl_display,
|
|
|
|
EGL_NO_SURFACE,
|
|
|
|
EGL_NO_SURFACE,
|
|
|
|
egl_context,
|
|
|
|
error))
|
|
|
|
{
|
|
|
|
meta_egl_destroy_context (egl, egl_display, egl_context, NULL);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
renderer_str = (const char *) glGetString (GL_RENDERER);
|
|
|
|
if (g_str_has_prefix (renderer_str, "llvmpipe") ||
|
|
|
|
g_str_has_prefix (renderer_str, "softpipe") ||
|
|
|
|
g_str_has_prefix (renderer_str, "swrast"))
|
2018-10-23 10:09:22 +02:00
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"Do not want to use software renderer (%s), falling back to CPU copy path",
|
|
|
|
renderer_str);
|
2019-01-28 13:02:07 +02:00
|
|
|
goto out_fail_with_context;
|
2018-10-23 10:09:22 +02:00
|
|
|
}
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
|
2017-07-25 10:21:04 +08:00
|
|
|
if (!meta_gles3_has_extensions (renderer_native->gles3,
|
|
|
|
&missing_gl_extensions,
|
|
|
|
"GL_OES_EGL_image_external",
|
|
|
|
NULL))
|
|
|
|
{
|
|
|
|
char *missing_gl_extensions_str;
|
|
|
|
|
2019-01-20 11:46:46 +01:00
|
|
|
missing_gl_extensions_str = g_strjoinv (", ",
|
|
|
|
(char **) missing_gl_extensions);
|
2017-07-25 10:21:04 +08:00
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"Missing OpenGL ES extensions: %s",
|
|
|
|
missing_gl_extensions_str);
|
|
|
|
g_free (missing_gl_extensions_str);
|
|
|
|
g_free (missing_gl_extensions);
|
2019-01-28 13:02:07 +02:00
|
|
|
|
|
|
|
goto out_fail_with_context;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
2018-10-23 10:09:22 +02:00
|
|
|
renderer_gpu_data->secondary.is_hardware_rendering = TRUE;
|
2017-07-25 10:21:04 +08:00
|
|
|
renderer_gpu_data->secondary.egl_context = egl_context;
|
|
|
|
renderer_gpu_data->secondary.egl_config = egl_config;
|
2018-11-29 13:34:40 +02:00
|
|
|
renderer_gpu_data->secondary.copy_mode = META_SHARED_FRAMEBUFFER_COPY_MODE_SECONDARY_GPU;
|
|
|
|
|
|
|
|
renderer_gpu_data->secondary.has_EGL_EXT_image_dma_buf_import_modifiers =
|
|
|
|
meta_egl_has_extensions (egl, egl_display, NULL,
|
|
|
|
"EGL_EXT_image_dma_buf_import_modifiers",
|
|
|
|
NULL);
|
2017-07-25 10:21:04 +08:00
|
|
|
|
|
|
|
return TRUE;
|
2019-01-28 13:02:07 +02:00
|
|
|
|
|
|
|
out_fail_with_context:
|
|
|
|
meta_egl_make_current (egl,
|
|
|
|
egl_display,
|
|
|
|
EGL_NO_SURFACE,
|
|
|
|
EGL_NO_SURFACE,
|
|
|
|
EGL_NO_CONTEXT,
|
|
|
|
NULL);
|
|
|
|
meta_egl_destroy_context (egl, egl_display, egl_context, NULL);
|
|
|
|
|
|
|
|
return FALSE;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
init_secondary_gpu_data_cpu (MetaRendererNativeGpuData *renderer_gpu_data)
|
|
|
|
{
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
renderer_gpu_data->secondary.is_hardware_rendering = FALSE;
|
2019-05-24 17:07:14 +03:00
|
|
|
|
|
|
|
/* First try ZERO, it automatically falls back to PRIMARY as needed */
|
|
|
|
renderer_gpu_data->secondary.copy_mode =
|
|
|
|
META_SHARED_FRAMEBUFFER_COPY_MODE_ZERO;
|
2017-07-25 10:21:04 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
init_secondary_gpu_data (MetaRendererNativeGpuData *renderer_gpu_data)
|
|
|
|
{
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (init_secondary_gpu_data_gpu (renderer_gpu_data, &error))
|
|
|
|
return;
|
|
|
|
|
|
|
|
g_warning ("Failed to initialize accelerated iGPU/dGPU framebuffer sharing: %s",
|
|
|
|
error->message);
|
|
|
|
g_error_free (error);
|
|
|
|
|
|
|
|
init_secondary_gpu_data_cpu (renderer_gpu_data);
|
|
|
|
}
|
|
|
|
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
static gboolean
|
|
|
|
gpu_kms_is_hardware_rendering (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms)
|
|
|
|
{
|
|
|
|
MetaRendererNativeGpuData *data;
|
|
|
|
|
|
|
|
data = meta_renderer_native_get_gpu_data (renderer_native, gpu_kms);
|
|
|
|
return data->secondary.is_hardware_rendering;
|
|
|
|
}
|
|
|
|
|
2019-04-03 13:11:45 +03:00
|
|
|
static EGLDisplay
|
|
|
|
init_gbm_egl_display (MetaRendererNative *renderer_native,
|
|
|
|
struct gbm_device *gbm_device,
|
|
|
|
GError **error)
|
2016-05-09 21:22:01 +08:00
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
2016-08-18 10:37:16 +08:00
|
|
|
EGLDisplay egl_display;
|
2016-05-09 21:22:01 +08:00
|
|
|
|
2016-09-01 11:04:22 +08:00
|
|
|
if (!meta_egl_has_extensions (egl, EGL_NO_DISPLAY, NULL,
|
|
|
|
"EGL_MESA_platform_gbm",
|
2017-11-03 03:59:00 +00:00
|
|
|
NULL) &&
|
|
|
|
!meta_egl_has_extensions (egl, EGL_NO_DISPLAY, NULL,
|
|
|
|
"EGL_KHR_platform_gbm",
|
2016-09-01 11:04:22 +08:00
|
|
|
NULL))
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Missing extension for GBM renderer: EGL_KHR_platform_gbm");
|
2019-04-03 13:11:45 +03:00
|
|
|
return EGL_NO_DISPLAY;
|
2016-09-01 11:04:22 +08:00
|
|
|
}
|
|
|
|
|
2019-04-03 13:11:45 +03:00
|
|
|
egl_display = meta_egl_get_platform_display (egl,
|
|
|
|
EGL_PLATFORM_GBM_KHR,
|
|
|
|
gbm_device, NULL, error);
|
|
|
|
if (egl_display == EGL_NO_DISPLAY)
|
|
|
|
return EGL_NO_DISPLAY;
|
|
|
|
|
|
|
|
if (!meta_egl_initialize (egl, egl_display, error))
|
|
|
|
return EGL_NO_DISPLAY;
|
|
|
|
|
|
|
|
return egl_display;
|
|
|
|
}
|
|
|
|
|
|
|
|
static MetaRendererNativeGpuData *
|
|
|
|
create_renderer_gpu_data_gbm (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
struct gbm_device *gbm_device;
|
|
|
|
int kms_fd;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
g_autoptr (GError) local_error = NULL;
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
kms_fd = meta_gpu_kms_get_fd (gpu_kms);
|
2017-07-06 16:00:56 +08:00
|
|
|
|
|
|
|
gbm_device = gbm_create_device (kms_fd);
|
2016-08-18 10:37:16 +08:00
|
|
|
if (!gbm_device)
|
2016-05-09 21:22:01 +08:00
|
|
|
{
|
2016-05-10 10:16:45 +08:00
|
|
|
g_set_error (error, G_IO_ERROR,
|
2016-05-09 21:22:01 +08:00
|
|
|
G_IO_ERROR_FAILED,
|
2016-10-28 15:35:31 +08:00
|
|
|
"Failed to create gbm device: %s", g_strerror (errno));
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
2016-05-09 21:22:01 +08:00
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = meta_create_renderer_native_gpu_data (gpu_kms);
|
|
|
|
renderer_gpu_data->renderer_native = renderer_native;
|
|
|
|
renderer_gpu_data->gbm.device = gbm_device;
|
|
|
|
renderer_gpu_data->mode = META_RENDERER_NATIVE_MODE_GBM;
|
|
|
|
|
2019-04-03 13:11:45 +03:00
|
|
|
renderer_gpu_data->egl_display = init_gbm_egl_display (renderer_native,
|
|
|
|
gbm_device,
|
|
|
|
&local_error);
|
|
|
|
if (renderer_gpu_data->egl_display == EGL_NO_DISPLAY)
|
|
|
|
{
|
|
|
|
g_debug ("GBM EGL init for %s failed: %s",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms),
|
|
|
|
local_error->message);
|
2016-08-18 10:37:16 +08:00
|
|
|
|
2019-04-03 13:11:45 +03:00
|
|
|
init_secondary_gpu_data_cpu (renderer_gpu_data);
|
|
|
|
return renderer_gpu_data;
|
|
|
|
}
|
|
|
|
|
|
|
|
init_secondary_gpu_data (renderer_gpu_data);
|
2017-07-10 18:19:32 +08:00
|
|
|
return renderer_gpu_data;
|
2016-05-09 21:22:01 +08:00
|
|
|
}
|
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
static const char *
|
|
|
|
get_drm_device_file (MetaEgl *egl,
|
|
|
|
EGLDeviceEXT device,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
if (!meta_egl_egl_device_has_extensions (egl, device,
|
|
|
|
NULL,
|
|
|
|
"EGL_EXT_device_drm",
|
|
|
|
NULL))
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Missing required EGLDevice extension EGL_EXT_device_drm");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return meta_egl_query_device_string (egl, device,
|
|
|
|
EGL_DRM_DEVICE_FILE_EXT,
|
|
|
|
error);
|
|
|
|
}
|
|
|
|
|
|
|
|
static EGLDeviceEXT
|
2017-07-10 18:19:32 +08:00
|
|
|
find_egl_device (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
GError **error)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2017-07-24 15:54:29 +08:00
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
2019-01-20 11:46:46 +01:00
|
|
|
const char **missing_extensions;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLint num_devices;
|
|
|
|
EGLDeviceEXT *devices;
|
2017-07-06 16:00:56 +08:00
|
|
|
const char *kms_file_path;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLDeviceEXT device;
|
|
|
|
EGLint i;
|
|
|
|
|
|
|
|
if (!meta_egl_has_extensions (egl,
|
|
|
|
EGL_NO_DISPLAY,
|
|
|
|
&missing_extensions,
|
|
|
|
"EGL_EXT_device_base",
|
|
|
|
NULL))
|
|
|
|
{
|
|
|
|
char *missing_extensions_str;
|
|
|
|
|
2019-01-20 11:46:46 +01:00
|
|
|
missing_extensions_str = g_strjoinv (", ", (char **) missing_extensions);
|
2016-08-18 11:28:59 +08:00
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Missing EGL extensions required for EGLDevice renderer: %s",
|
|
|
|
missing_extensions_str);
|
|
|
|
g_free (missing_extensions_str);
|
|
|
|
g_free (missing_extensions);
|
|
|
|
return EGL_NO_DEVICE_EXT;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!meta_egl_query_devices (egl, 0, NULL, &num_devices, error))
|
|
|
|
return EGL_NO_DEVICE_EXT;
|
|
|
|
|
|
|
|
devices = g_new0 (EGLDeviceEXT, num_devices);
|
|
|
|
if (!meta_egl_query_devices (egl, num_devices, devices, &num_devices,
|
|
|
|
error))
|
|
|
|
{
|
|
|
|
g_free (devices);
|
|
|
|
return EGL_NO_DEVICE_EXT;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
kms_file_path = meta_gpu_kms_get_file_path (gpu_kms);
|
2017-07-06 16:00:56 +08:00
|
|
|
|
2016-08-18 11:28:59 +08:00
|
|
|
device = EGL_NO_DEVICE_EXT;
|
|
|
|
for (i = 0; i < num_devices; i++)
|
|
|
|
{
|
|
|
|
const char *egl_device_drm_path;
|
|
|
|
|
|
|
|
g_clear_error (error);
|
|
|
|
|
|
|
|
egl_device_drm_path = get_drm_device_file (egl, devices[i], error);
|
|
|
|
if (!egl_device_drm_path)
|
|
|
|
continue;
|
|
|
|
|
2017-07-06 16:00:56 +08:00
|
|
|
if (g_str_equal (egl_device_drm_path, kms_file_path))
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
|
|
|
device = devices[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
g_free (devices);
|
|
|
|
|
|
|
|
if (device == EGL_NO_DEVICE_EXT)
|
|
|
|
{
|
|
|
|
if (!*error)
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Failed to find matching EGLDeviceEXT");
|
|
|
|
return EGL_NO_DEVICE_EXT;
|
|
|
|
}
|
|
|
|
|
|
|
|
return device;
|
|
|
|
}
|
|
|
|
|
|
|
|
static EGLDisplay
|
2017-07-10 18:19:32 +08:00
|
|
|
get_egl_device_display (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
EGLDeviceEXT egl_device,
|
|
|
|
GError **error)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2017-07-24 15:54:29 +08:00
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
2017-07-10 18:19:32 +08:00
|
|
|
int kms_fd = meta_gpu_kms_get_fd (gpu_kms);
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLint platform_attribs[] = {
|
2017-07-06 16:00:56 +08:00
|
|
|
EGL_DRM_MASTER_FD_EXT, kms_fd,
|
2016-08-18 11:28:59 +08:00
|
|
|
EGL_NONE
|
|
|
|
};
|
|
|
|
|
|
|
|
return meta_egl_get_platform_display (egl, EGL_PLATFORM_DEVICE_EXT,
|
|
|
|
(void *) egl_device,
|
|
|
|
platform_attribs,
|
|
|
|
error);
|
|
|
|
}
|
|
|
|
|
2018-12-10 15:46:56 +02:00
|
|
|
static int
|
|
|
|
count_drm_devices (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
|
|
|
|
return g_list_length (meta_backend_get_gpus (backend));
|
2018-12-10 15:46:56 +02:00
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static MetaRendererNativeGpuData *
|
|
|
|
create_renderer_gpu_data_egl_device (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
GError **error)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2017-07-24 15:54:29 +08:00
|
|
|
MetaEgl *egl = meta_renderer_native_get_egl (renderer_native);
|
2019-01-20 11:46:46 +01:00
|
|
|
const char **missing_extensions;
|
2016-08-18 11:28:59 +08:00
|
|
|
EGLDeviceEXT egl_device;
|
|
|
|
EGLDisplay egl_display;
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2018-12-10 15:46:56 +02:00
|
|
|
if (count_drm_devices (renderer_native) != 1)
|
2017-07-25 10:21:04 +08:00
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"EGLDevice currently only works with single GPU systems");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
egl_device = find_egl_device (renderer_native, gpu_kms, error);
|
2016-08-18 11:28:59 +08:00
|
|
|
if (egl_device == EGL_NO_DEVICE_EXT)
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
egl_display = get_egl_device_display (renderer_native, gpu_kms,
|
|
|
|
egl_device, error);
|
2016-08-18 11:28:59 +08:00
|
|
|
if (egl_display == EGL_NO_DISPLAY)
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
|
|
|
if (!meta_egl_initialize (egl, egl_display, error))
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
|
|
|
if (!meta_egl_has_extensions (egl,
|
|
|
|
egl_display,
|
|
|
|
&missing_extensions,
|
|
|
|
"EGL_NV_output_drm_flip_event",
|
|
|
|
"EGL_EXT_output_base",
|
|
|
|
"EGL_EXT_output_drm",
|
|
|
|
"EGL_KHR_stream",
|
|
|
|
"EGL_KHR_stream_producer_eglsurface",
|
|
|
|
"EGL_EXT_stream_consumer_egloutput",
|
|
|
|
"EGL_EXT_stream_acquire_mode",
|
|
|
|
NULL))
|
|
|
|
{
|
|
|
|
char *missing_extensions_str;
|
|
|
|
|
2019-01-20 11:46:46 +01:00
|
|
|
missing_extensions_str = g_strjoinv (", ", (char **) missing_extensions);
|
2016-08-18 11:28:59 +08:00
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Missing EGL extensions required for EGLDevice renderer: %s",
|
|
|
|
missing_extensions_str);
|
2019-06-12 17:58:54 +01:00
|
|
|
meta_egl_terminate (egl, egl_display, NULL);
|
2016-08-18 11:28:59 +08:00
|
|
|
g_free (missing_extensions_str);
|
|
|
|
g_free (missing_extensions);
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
2016-08-18 11:28:59 +08:00
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = meta_create_renderer_native_gpu_data (gpu_kms);
|
|
|
|
renderer_gpu_data->renderer_native = renderer_native;
|
|
|
|
renderer_gpu_data->egl.device = egl_device;
|
|
|
|
renderer_gpu_data->mode = META_RENDERER_NATIVE_MODE_EGL_DEVICE;
|
2017-07-25 10:21:04 +08:00
|
|
|
renderer_gpu_data->egl_display = egl_display;
|
2016-08-18 11:28:59 +08:00
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
return renderer_gpu_data;
|
2016-08-18 11:28:59 +08:00
|
|
|
}
|
|
|
|
#endif /* HAVE_EGL_DEVICE */
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static MetaRendererNativeGpuData *
|
|
|
|
meta_renderer_native_create_renderer_gpu_data (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
GError **error)
|
2016-08-18 11:14:03 +08:00
|
|
|
{
|
2017-07-10 18:19:32 +08:00
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
2016-08-18 11:28:59 +08:00
|
|
|
GError *gbm_error = NULL;
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
GError *egl_device_error = NULL;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
2017-06-27 18:31:48 -07:00
|
|
|
/* Try to initialize the EGLDevice backend first. Whenever we use a
|
|
|
|
* non-NVIDIA GPU, the EGLDevice enumeration function won't find a match, and
|
|
|
|
* we'll fall back to GBM (which will always succeed as it has a software
|
|
|
|
* rendering fallback)
|
|
|
|
*/
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = create_renderer_gpu_data_egl_device (renderer_native,
|
|
|
|
gpu_kms,
|
|
|
|
&egl_device_error);
|
|
|
|
if (renderer_gpu_data)
|
|
|
|
return renderer_gpu_data;
|
2017-06-27 18:31:48 -07:00
|
|
|
#endif
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_gpu_data = create_renderer_gpu_data_gbm (renderer_native,
|
|
|
|
gpu_kms,
|
|
|
|
&gbm_error);
|
|
|
|
if (renderer_gpu_data)
|
2016-08-18 11:28:59 +08:00
|
|
|
{
|
2017-07-07 17:59:20 +08:00
|
|
|
#ifdef HAVE_EGL_DEVICE
|
2017-06-27 18:31:48 -07:00
|
|
|
g_error_free (egl_device_error);
|
2017-07-07 17:59:20 +08:00
|
|
|
#endif
|
2017-07-10 18:19:32 +08:00
|
|
|
return renderer_gpu_data;
|
2016-08-18 11:28:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
g_set_error (error, G_IO_ERROR,
|
|
|
|
G_IO_ERROR_FAILED,
|
|
|
|
"Failed to initialize renderer: "
|
|
|
|
"%s"
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
", %s"
|
|
|
|
#endif
|
|
|
|
, gbm_error->message
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
, egl_device_error->message
|
|
|
|
#endif
|
|
|
|
);
|
|
|
|
|
|
|
|
g_error_free (gbm_error);
|
|
|
|
#ifdef HAVE_EGL_DEVICE
|
|
|
|
g_error_free (egl_device_error);
|
|
|
|
#endif
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-09-17 13:03:18 +02:00
|
|
|
static gboolean
|
|
|
|
create_renderer_gpu_data (MetaRendererNative *renderer_native,
|
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
|
|
|
renderer_gpu_data =
|
|
|
|
meta_renderer_native_create_renderer_gpu_data (renderer_native,
|
|
|
|
gpu_kms,
|
|
|
|
error);
|
|
|
|
if (!renderer_gpu_data)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
g_hash_table_insert (renderer_native->gpu_datas,
|
|
|
|
gpu_kms,
|
|
|
|
renderer_gpu_data);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2019-01-11 15:35:42 +01:00
|
|
|
on_gpu_added (MetaBackendNative *backend_native,
|
2018-09-17 13:03:18 +02:00
|
|
|
MetaGpuKms *gpu_kms,
|
|
|
|
MetaRendererNative *renderer_native)
|
|
|
|
{
|
2019-01-11 15:35:42 +01:00
|
|
|
MetaBackend *backend = META_BACKEND (backend_native);
|
2018-09-17 13:03:18 +02:00
|
|
|
ClutterBackend *clutter_backend = meta_backend_get_clutter_backend (backend);
|
|
|
|
CoglContext *cogl_context = clutter_backend_get_cogl_context (clutter_backend);
|
|
|
|
CoglDisplay *cogl_display = cogl_context_get_display (cogl_context);
|
|
|
|
GError *error = NULL;
|
|
|
|
|
|
|
|
if (!create_renderer_gpu_data (renderer_native, gpu_kms, &error))
|
|
|
|
{
|
|
|
|
g_warning ("on_gpu_added: could not create gpu_data for gpu %s: %s",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms), error->message);
|
|
|
|
g_clear_error (&error);
|
|
|
|
}
|
|
|
|
|
|
|
|
_cogl_winsys_egl_ensure_current (cogl_display);
|
|
|
|
}
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
static void
|
|
|
|
on_power_save_mode_changed (MetaMonitorManager *monitor_manager,
|
|
|
|
MetaRendererNative *renderer_native)
|
|
|
|
{
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
|
|
|
MetaPowerSave power_save_mode;
|
|
|
|
|
|
|
|
power_save_mode = meta_monitor_manager_get_power_save_mode (monitor_manager);
|
|
|
|
if (power_save_mode == META_POWER_SAVE_ON)
|
|
|
|
meta_renderer_native_queue_modes_reset (renderer_native);
|
|
|
|
else
|
|
|
|
meta_kms_discard_pending_page_flips (kms);
|
|
|
|
}
|
|
|
|
|
2020-12-21 14:59:32 +08:00
|
|
|
void
|
|
|
|
meta_renderer_native_reset_modes (MetaRendererNative *renderer_native)
|
|
|
|
{
|
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
|
|
|
MetaBackendNative *backend_native = META_BACKEND_NATIVE (backend);
|
|
|
|
MetaKms *kms = meta_backend_native_get_kms (backend_native);
|
|
|
|
|
2020-10-02 10:38:57 +02:00
|
|
|
unset_disabled_crtcs (backend, kms);
|
2020-12-21 14:59:32 +08:00
|
|
|
}
|
|
|
|
|
2018-12-10 14:19:25 +02:00
|
|
|
static MetaGpuKms *
|
2019-01-11 15:35:42 +01:00
|
|
|
choose_primary_gpu_unchecked (MetaBackend *backend,
|
2019-04-03 13:34:09 +03:00
|
|
|
MetaRendererNative *renderer_native)
|
2018-12-10 14:19:25 +02:00
|
|
|
{
|
2019-01-11 15:35:42 +01:00
|
|
|
GList *gpus = meta_backend_get_gpus (backend);
|
2018-12-10 14:19:25 +02:00
|
|
|
GList *l;
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
int allow_sw;
|
2018-12-10 14:19:25 +02:00
|
|
|
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
/*
|
|
|
|
* Check first hardware rendering devices, and if none found,
|
|
|
|
* then software rendering devices.
|
|
|
|
*/
|
|
|
|
for (allow_sw = 0; allow_sw < 2; allow_sw++)
|
|
|
|
{
|
2020-11-14 10:41:23 +01:00
|
|
|
/* First check if one was explicitly configured. */
|
|
|
|
for (l = gpus; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaGpuKms *gpu_kms = META_GPU_KMS (l->data);
|
|
|
|
MetaKmsDevice *kms_device = meta_gpu_kms_get_kms_device (gpu_kms);
|
|
|
|
|
|
|
|
if (meta_kms_device_get_flags (kms_device) &
|
|
|
|
META_KMS_DEVICE_FLAG_PREFERRED_PRIMARY)
|
|
|
|
{
|
|
|
|
g_message ("GPU %s selected primary given udev rule",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms));
|
|
|
|
return gpu_kms;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
/* Prefer a platform device */
|
|
|
|
for (l = gpus; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaGpuKms *gpu_kms = META_GPU_KMS (l->data);
|
|
|
|
|
|
|
|
if (meta_gpu_kms_is_platform_device (gpu_kms) &&
|
|
|
|
(allow_sw == 1 ||
|
|
|
|
gpu_kms_is_hardware_rendering (renderer_native, gpu_kms)))
|
2020-11-14 10:41:23 +01:00
|
|
|
{
|
|
|
|
g_message ("Integrated GPU %s selected as primary",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms));
|
|
|
|
return gpu_kms;
|
|
|
|
}
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Otherwise a device we booted with */
|
|
|
|
for (l = gpus; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaGpuKms *gpu_kms = META_GPU_KMS (l->data);
|
|
|
|
|
|
|
|
if (meta_gpu_kms_is_boot_vga (gpu_kms) &&
|
|
|
|
(allow_sw == 1 ||
|
|
|
|
gpu_kms_is_hardware_rendering (renderer_native, gpu_kms)))
|
2020-11-14 10:41:23 +01:00
|
|
|
{
|
|
|
|
g_message ("Boot VGA GPU %s selected as primary",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms));
|
|
|
|
return gpu_kms;
|
|
|
|
}
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Fall back to any device */
|
|
|
|
for (l = gpus; l; l = l->next)
|
|
|
|
{
|
|
|
|
MetaGpuKms *gpu_kms = META_GPU_KMS (l->data);
|
|
|
|
|
|
|
|
if (allow_sw == 1 ||
|
|
|
|
gpu_kms_is_hardware_rendering (renderer_native, gpu_kms))
|
2020-11-14 10:41:23 +01:00
|
|
|
{
|
|
|
|
g_message ("GPU %s selected as primary",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms));
|
|
|
|
return gpu_kms;
|
|
|
|
}
|
renderer/native: Prefer hardware rendering for primary GPU
Mutter prefers platform devices over anything else as the primary GPU.
This will not work too well, when a platform device does not actually
have a rendering GPU but is a display-only device. An example of this
are DisplayLink devices with the proprietary driver stack, which exposes
a DRM KMS platform device but without any rendering driver.
Mutter cannot rely on EGL init failing on such devices either, because
nowadays Mesa supports software renderers on GBM, so the initialization
may well succeed.
The hardware rendering capability is recognized by matching the GL
renderer string to the known Mesa software renderers. At this time,
there is no better alternative to detecting this.
The secondary GPU data is abused for the GL renderer, as the Cogl
context may not have been created yet. Also, the Cogl context would
only be created on the primary GPU, but at this point the primary GPU
has not been chosen yet. Hence, GPU copy path GL context is used as a
proxy and predictor of what the Cogl context might be if it was created.
Mind, that even the GL flavour are not the same between Cogl and
secondary contexts, so this is stretch but it should be just enough.
The logic to choose the primary GPU is changed to always prefer hardware
rendering devices while also maintaining the old order of preferring
platform over boot_vga devices.
Co-authored by: Emilio Pozuelo Monfort <emilio.pozuelo@collabora.co.uk>
https://gitlab.gnome.org/GNOME/mutter/merge_requests/271
2018-12-10 16:49:58 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
g_assert_not_reached ();
|
|
|
|
return NULL;
|
2018-12-10 14:19:25 +02:00
|
|
|
}
|
|
|
|
|
2019-04-03 13:34:09 +03:00
|
|
|
static MetaGpuKms *
|
2019-01-11 15:35:42 +01:00
|
|
|
choose_primary_gpu (MetaBackend *backend,
|
2019-04-03 13:34:09 +03:00
|
|
|
MetaRendererNative *renderer_native,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaGpuKms *gpu_kms;
|
|
|
|
MetaRendererNativeGpuData *renderer_gpu_data;
|
|
|
|
|
2019-01-11 15:35:42 +01:00
|
|
|
gpu_kms = choose_primary_gpu_unchecked (backend, renderer_native);
|
2019-04-03 13:34:09 +03:00
|
|
|
renderer_gpu_data = meta_renderer_native_get_gpu_data (renderer_native,
|
|
|
|
gpu_kms);
|
|
|
|
if (renderer_gpu_data->egl_display == EGL_NO_DISPLAY)
|
|
|
|
{
|
|
|
|
g_set_error (error, G_IO_ERROR, G_IO_ERROR_FAILED,
|
|
|
|
"The GPU %s chosen as primary is not supported by EGL.",
|
|
|
|
meta_gpu_kms_get_file_path (gpu_kms));
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return gpu_kms;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static gboolean
|
|
|
|
meta_renderer_native_initable_init (GInitable *initable,
|
|
|
|
GCancellable *cancellable,
|
|
|
|
GError **error)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (initable);
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
2017-07-10 18:19:32 +08:00
|
|
|
GList *gpus;
|
2017-07-25 10:21:04 +08:00
|
|
|
GList *l;
|
2017-07-10 18:19:32 +08:00
|
|
|
|
2019-01-11 15:35:42 +01:00
|
|
|
gpus = meta_backend_get_gpus (backend);
|
2017-07-25 10:21:04 +08:00
|
|
|
for (l = gpus; l; l = l->next)
|
2017-07-10 18:19:32 +08:00
|
|
|
{
|
2017-07-25 10:21:04 +08:00
|
|
|
MetaGpuKms *gpu_kms = META_GPU_KMS (l->data);
|
2017-07-10 18:19:32 +08:00
|
|
|
|
2018-09-17 13:03:18 +02:00
|
|
|
if (!create_renderer_gpu_data (renderer_native, gpu_kms, error))
|
2017-07-25 10:21:04 +08:00
|
|
|
return FALSE;
|
|
|
|
}
|
2017-07-10 18:19:32 +08:00
|
|
|
|
2019-01-11 15:35:42 +01:00
|
|
|
renderer_native->primary_gpu_kms = choose_primary_gpu (backend,
|
2019-04-03 13:34:09 +03:00
|
|
|
renderer_native,
|
|
|
|
error);
|
|
|
|
if (!renderer_native->primary_gpu_kms)
|
|
|
|
return FALSE;
|
2018-10-12 14:19:24 +02:00
|
|
|
|
2020-09-14 20:36:17 +02:00
|
|
|
if (meta_gpu_kms_requires_modifiers (renderer_native->primary_gpu_kms))
|
|
|
|
renderer_native->use_modifiers = TRUE;
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
return TRUE;
|
2016-08-18 11:14:03 +08:00
|
|
|
}
|
|
|
|
|
2016-05-09 21:22:01 +08:00
|
|
|
static void
|
|
|
|
initable_iface_init (GInitableIface *initable_iface)
|
|
|
|
{
|
|
|
|
initable_iface->init = meta_renderer_native_initable_init;
|
|
|
|
}
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
static void
|
|
|
|
meta_renderer_native_finalize (GObject *object)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (object);
|
|
|
|
|
2020-10-23 10:21:54 +02:00
|
|
|
clear_kept_alive_onscreens (renderer_native);
|
|
|
|
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
if (renderer_native->power_save_page_flip_onscreens)
|
2019-03-25 10:24:46 +01:00
|
|
|
{
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
g_list_free_full (renderer_native->power_save_page_flip_onscreens,
|
2020-10-13 11:35:47 +02:00
|
|
|
g_object_unref);
|
2019-11-22 00:25:30 +01:00
|
|
|
g_clear_handle_id (&renderer_native->power_save_page_flip_source_id,
|
|
|
|
g_source_remove);
|
2019-03-25 10:24:46 +01:00
|
|
|
}
|
|
|
|
|
2020-10-09 23:17:06 +02:00
|
|
|
g_list_free (renderer_native->pending_mode_set_views);
|
|
|
|
g_clear_handle_id (&renderer_native->mode_set_failed_feedback_source_id,
|
|
|
|
g_source_remove);
|
|
|
|
|
2017-07-10 18:19:32 +08:00
|
|
|
g_hash_table_destroy (renderer_native->gpu_datas);
|
2017-07-25 10:21:04 +08:00
|
|
|
g_clear_object (&renderer_native->gles3);
|
2017-07-10 18:19:32 +08:00
|
|
|
|
|
|
|
G_OBJECT_CLASS (meta_renderer_native_parent_class)->finalize (object);
|
|
|
|
}
|
|
|
|
|
2018-04-11 14:39:15 +02:00
|
|
|
static void
|
|
|
|
meta_renderer_native_constructed (GObject *object)
|
|
|
|
{
|
|
|
|
MetaRendererNative *renderer_native = META_RENDERER_NATIVE (object);
|
2019-10-01 11:51:53 +02:00
|
|
|
MetaRenderer *renderer = META_RENDERER (renderer_native);
|
|
|
|
MetaBackend *backend = meta_renderer_get_backend (renderer);
|
2018-04-11 14:39:15 +02:00
|
|
|
MetaSettings *settings = meta_backend_get_settings (backend);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
MetaMonitorManager *monitor_manager =
|
|
|
|
meta_backend_get_monitor_manager (backend);
|
2018-04-11 14:39:15 +02:00
|
|
|
|
|
|
|
if (meta_settings_is_experimental_feature_enabled (
|
|
|
|
settings, META_EXPERIMENTAL_FEATURE_KMS_MODIFIERS))
|
|
|
|
renderer_native->use_modifiers = TRUE;
|
|
|
|
|
2019-01-11 15:35:42 +01:00
|
|
|
g_signal_connect (backend, "gpu-added",
|
2018-09-17 13:03:18 +02:00
|
|
|
G_CALLBACK (on_gpu_added), renderer_native);
|
backend/native: Add and use transactional KMS API
This commit introduces, and makes use of, a transactional API used for
setting up KMS state, later to be applied, potentially atomically. From
an API point of view, so is always the case, but in the current
implementation, it still uses legacy drmMode* API to apply the state
non-atomically.
The API consists of various buliding blocks:
* MetaKmsUpdate - a set of configuration changes, the higher level
handle for handing over configuration to the impl backend. It's used to
set mode, assign framebuffers to planes, queue page flips and set
connector properties.
* MetaKmsPlaneAssignment - the assignment of a framebuffer to a plane.
Currently used to map a framebuffer to the primary plane of a CRTC. In
the legacy KMS implementation, the plane assignment is used to derive
the framebuffer used for mode setting and page flipping.
This also means various high level changes:
State, excluding configuring the cursor plane and creating/destroying
DRM framebuffer handles, are applied in the end of a clutter frame, in
one go. From an API point of view, this is done atomically, but as
mentioned, only the non-atomic implementation exists so far.
From MetaRendererNative's point of view, a page flip now initially
always succeeds; the handling of EBUSY errors are done asynchronously in
the MetaKmsImpl backend (still by retrying at refresh rate, but
postponing flip callbacks instead of manipulating the frame clock).
Handling of falling back to mode setting instead of page flipping is
notified after the fact by a more precise page flip feedback API.
EGLStream based page flipping relies on the impl backend not being
atomic, as the page flipping is done in the EGLStream backend (e.g.
nvidia driver). It uses a 'custom' page flip queueing method, keeping
the EGLStream logic inside meta-renderer-native.c.
Page flip handling is moved to meta-kms-impl-device.c from
meta-gpu-kms.c. It goes via an extra idle callback before reaching
meta-renderer-native.c to make sure callbacks are invoked outside of the
impl context.
While dummy power save page flipping is kept in meta-renderer-native.c, the
EBUSY handling is moved to meta-kms-impl-simple.c. Instead of freezing the
frame clock, actual page flip callbacks are postponed until all EBUSY retries
have either succeeded or failed due to some other error than EBUSY. This
effectively inhibits new frames to be drawn, meaning we won't stall waiting on
the file descriptor for pending page flips.
https://gitlab.gnome.org/GNOME/mutter/issues/548
https://gitlab.gnome.org/GNOME/mutter/merge_requests/525
2019-04-04 22:36:41 +02:00
|
|
|
g_signal_connect (monitor_manager, "power-save-mode-changed",
|
|
|
|
G_CALLBACK (on_power_save_mode_changed), renderer_native);
|
2018-09-17 13:03:18 +02:00
|
|
|
|
2018-04-11 14:39:15 +02:00
|
|
|
G_OBJECT_CLASS (meta_renderer_native_parent_class)->constructed (object);
|
|
|
|
}
|
|
|
|
|
2016-05-07 23:07:46 +08:00
|
|
|
static void
|
|
|
|
meta_renderer_native_init (MetaRendererNative *renderer_native)
|
|
|
|
{
|
2017-07-10 18:19:32 +08:00
|
|
|
renderer_native->gpu_datas =
|
|
|
|
g_hash_table_new_full (NULL, NULL,
|
|
|
|
NULL,
|
|
|
|
(GDestroyNotify) meta_renderer_native_gpu_data_free);
|
2016-05-07 23:07:46 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
meta_renderer_native_class_init (MetaRendererNativeClass *klass)
|
|
|
|
{
|
2016-05-07 23:09:59 +08:00
|
|
|
GObjectClass *object_class = G_OBJECT_CLASS (klass);
|
2016-05-09 09:59:54 +08:00
|
|
|
MetaRendererClass *renderer_class = META_RENDERER_CLASS (klass);
|
2016-05-07 23:09:59 +08:00
|
|
|
|
2016-05-09 21:22:01 +08:00
|
|
|
object_class->finalize = meta_renderer_native_finalize;
|
2018-04-11 14:39:15 +02:00
|
|
|
object_class->constructed = meta_renderer_native_constructed;
|
2016-05-07 23:09:59 +08:00
|
|
|
|
2016-05-09 09:59:54 +08:00
|
|
|
renderer_class->create_cogl_renderer = meta_renderer_native_create_cogl_renderer;
|
2016-06-08 17:05:31 +08:00
|
|
|
renderer_class->create_view = meta_renderer_native_create_view;
|
2019-06-19 20:57:14 +02:00
|
|
|
renderer_class->rebuild_views = meta_renderer_native_rebuild_views;
|
2016-05-07 23:07:46 +08:00
|
|
|
}
|
2016-05-09 21:22:01 +08:00
|
|
|
|
|
|
|
MetaRendererNative *
|
2019-01-11 15:35:42 +01:00
|
|
|
meta_renderer_native_new (MetaBackendNative *backend_native,
|
|
|
|
GError **error)
|
2016-05-09 21:22:01 +08:00
|
|
|
{
|
2017-10-05 10:47:25 -04:00
|
|
|
return g_initable_new (META_TYPE_RENDERER_NATIVE,
|
|
|
|
NULL,
|
|
|
|
error,
|
2019-01-11 15:35:42 +01:00
|
|
|
"backend", backend_native,
|
2017-10-05 10:47:25 -04:00
|
|
|
NULL);
|
2016-05-09 21:22:01 +08:00
|
|
|
}
|