10#include <private/qeglfsintegration_p.h>
11#include <private/qeglfskmsintegration_p.h>
13#include <QtCore/QLoggingCategory>
15#include <QtGui/private/qguiapplication_p.h>
16#include <QtGui/private/qtguiglobal_p.h>
17#include <QtFbSupport/private/qfbvthandler_p.h>
23QMutex QEglFSKmsGbmScreen::s_nonThreadedFlipMutex;
27 Q_ASSERT(DRM_FORMAT_XRGB8888 == GBM_FORMAT_XRGB8888);
33 Q_ASSERT(DRM_FORMAT_XRGB8888 == GBM_FORMAT_XRGB8888);
37void QEglFSKmsGbmScreen::bufferDestroyedHandler(gbm_bo *bo,
void *data)
39 FrameBuffer *fb =
static_cast<FrameBuffer *>(data);
42 gbm_device *device = gbm_bo_get_device(bo);
43 drmModeRmFB(gbm_device_get_fd(device), fb->fb);
49QEglFSKmsGbmScreen::FrameBuffer *QEglFSKmsGbmScreen::framebufferForBufferObject(gbm_bo *bo)
52 FrameBuffer *fb =
static_cast<FrameBuffer *>(gbm_bo_get_user_data(bo));
57 uint32_t width = gbm_bo_get_width(bo);
58 uint32_t height = gbm_bo_get_height(bo);
59 uint32_t handles[4] = { gbm_bo_get_handle(bo).u32 };
60 uint32_t strides[4] = { gbm_bo_get_stride(bo) };
61 uint32_t offsets[4] = { 0 };
62 uint32_t pixelFormat = gbmFormatToDrmFormat(gbm_bo_get_format(bo));
64 auto fb = std::make_unique<FrameBuffer>();
65 qCDebug(qLcEglfsKmsDebug,
"Adding FB, size %ux%u, DRM format 0x%x, stride %u, handle %u",
66 width, height, pixelFormat, strides[0], handles[0]);
68 int ret = drmModeAddFB2(device()->fd(), width, height, pixelFormat,
69 handles, strides, offsets, &fb->fb, 0);
72 qWarning(
"Failed to create KMS FB!");
77 gbm_bo_set_user_data(bo, fb.release(), bufferDestroyedHandler);
81QEglFSKmsGbmScreen::QEglFSKmsGbmScreen(QEglFSKmsDevice *device,
const QKmsOutput &output,
bool headless)
82 : QEglFSKmsScreen(device, output, headless)
83 , m_gbm_surface(
nullptr)
84 , m_gbm_bo_current(
nullptr)
85 , m_gbm_bo_next(
nullptr)
86 , m_flipPending(
false)
88 , m_cloneSource(
nullptr)
92QEglFSKmsGbmScreen::~QEglFSKmsGbmScreen()
94 const int remainingScreenCount = qGuiApp->screens().count();
95 qCDebug(qLcEglfsKmsDebug,
"Screen dtor. %p Remaining screens: %d",
this, remainingScreenCount);
96 if (!remainingScreenCount && !device()->screenConfig()->separateScreens())
97 static_cast<QEglFSKmsGbmDevice *>(device())->destroyGlobalCursor();
101 QList<CloneDestination> &dests = m_cloneSource->m_cloneDests;
102 auto newEnd = std::remove_if(dests.begin(), dests.end(),
103 [
this](CloneDestination &dest) {
104 return dest.screen ==
this;
106 dests.erase(newEnd, dests.end());
110 for (CloneDestination &dest : m_cloneDests) {
111 dest.screen->m_cloneSource =
nullptr;
113 dest.screen->m_output.mode_set =
false;
117QPlatformCursor *QEglFSKmsGbmScreen::cursor()
const
119 QKmsScreenConfig *config = device()->screenConfig();
120 if (config->headless())
122 if (config->hwCursor()) {
123 if (!config->separateScreens())
124 return static_cast<QEglFSKmsGbmDevice *>(device())->globalCursor();
126 if (m_cursor.isNull()) {
127 QEglFSKmsGbmScreen *that =
const_cast<QEglFSKmsGbmScreen *>(
this);
128 that->m_cursor.reset(
new QEglFSKmsGbmCursor(that));
131 return m_cursor.data();
133 return QEglFSScreen::cursor();
137gbm_surface *QEglFSKmsGbmScreen::createSurface(EGLConfig eglConfig)
139 if (!m_gbm_surface) {
140 qCDebug(qLcEglfsKmsDebug,
"Creating gbm_surface for screen %s", qPrintable(name()));
142 const auto gbmDevice =
static_cast<QEglFSKmsGbmDevice *>(device())->gbmDevice();
145 const bool queryFromEgl = !m_output.drm_format_requested_by_user;
147 EGLint native_format = -1;
148 EGLBoolean success = eglGetConfigAttrib(display(), eglConfig, EGL_NATIVE_VISUAL_ID, &native_format);
149 qCDebug(qLcEglfsKmsDebug) <<
"Got native format" << Qt::hex << native_format << Qt::dec
150 <<
"from eglGetConfigAttrib() with return code" <<
bool(success);
153 m_gbm_surface = gbm_surface_create(gbmDevice,
154 rawGeometry().width(),
155 rawGeometry().height(),
159 m_output.drm_format = gbmFormatToDrmFormat(native_format);
163 const uint32_t gbmFormat = drmFormatToGbmFormat(m_output.drm_format);
168 if (!m_gbm_surface) {
170 qCDebug(qLcEglfsKmsDebug,
"Could not create surface with EGL_NATIVE_VISUAL_ID, falling back to format %x", gbmFormat);
171 m_gbm_surface = gbm_surface_create(gbmDevice,
172 rawGeometry().width(),
173 rawGeometry().height(),
180 if (!m_gbm_surface) {
181 uint64_t modifier = DRM_FORMAT_MOD_LINEAR;
183 m_gbm_surface = gbm_surface_create_with_modifiers(gbmDevice,
184 rawGeometry().width(),
185 rawGeometry().height(),
192 qFatal(
"Could not create GBM surface!");
194 return m_gbm_surface;
197void QEglFSKmsGbmScreen::resetSurface()
199 m_flipPending =
false;
200 m_gbm_bo_current =
nullptr;
201 m_gbm_surface =
nullptr;
210 if (!device()->hasAtomicSupport())
211 needsNewModeSetForNextFb =
true;
214void QEglFSKmsGbmScreen::initCloning(QPlatformScreen *screenThisScreenClones,
215 const QList<QPlatformScreen *> &screensCloningThisScreen)
218 const bool clonesAnother = screenThisScreenClones !=
nullptr;
219 if (clonesAnother && !screensCloningThisScreen.isEmpty()) {
220 qWarning(
"QEglFSKmsGbmScreen %s cannot be clone source and destination at the same time", qPrintable(name()));
224 m_cloneSource =
static_cast<QEglFSKmsGbmScreen *>(screenThisScreenClones);
225 qCDebug(qLcEglfsKmsDebug,
"Screen %s clones %s", qPrintable(name()), qPrintable(m_cloneSource->name()));
227 m_cloneSource =
nullptr;
231 m_cloneDests.clear();
232 for (QPlatformScreen *s : screensCloningThisScreen) {
234 d.screen =
static_cast<QEglFSKmsGbmScreen *>(s);
235 m_cloneDests.append(d);
239void QEglFSKmsGbmScreen::ensureModeSet(uint32_t fb)
241 QKmsOutput &op(output());
242 const int fd = device()->fd();
244 if (!op.mode_set || needsNewModeSetForNextFb) {
246 needsNewModeSetForNextFb =
false;
248 bool doModeSet =
true;
249 drmModeCrtcPtr currentMode = drmModeGetCrtc(fd, op.crtc_id);
250 const bool alreadySet = currentMode && currentMode->buffer_id == fb && !memcmp(¤tMode->mode, &op.modes[op.mode],
sizeof(drmModeModeInfo));
252 drmModeFreeCrtc(currentMode);
257 qCDebug(qLcEglfsKmsDebug,
"Setting mode for screen %s", qPrintable(name()));
259 if (device()->hasAtomicSupport()) {
260#if QT_CONFIG(drm_atomic)
261 drmModeAtomicReq *request = device()->threadLocalAtomicRequest();
263 drmModeAtomicAddProperty(request, op.connector_id, op.crtcIdPropertyId, op.crtc_id);
264 drmModeAtomicAddProperty(request, op.crtc_id, op.modeIdPropertyId, op.mode_blob_id);
265 drmModeAtomicAddProperty(request, op.crtc_id, op.activePropertyId, 1);
269 int ret = drmModeSetCrtc(fd,
277 setPowerState(PowerStateOn);
279 qErrnoWarning(errno,
"Could not set DRM mode for screen %s", qPrintable(name()));
285void QEglFSKmsGbmScreen::nonThreadedPageFlipHandler(
int fd,
286 unsigned int sequence,
288 unsigned int tv_usec,
293 QEglFSKmsGbmScreen *screen =
static_cast<QEglFSKmsGbmScreen *>(user_data);
295 if (QEglFSKmsScreen::isScreenKnown(screen)) {
296 screen->flipFinished();
297 screen->pageFlipped(sequence, tv_sec, tv_usec);
301void QEglFSKmsGbmScreen::waitForFlipWithEventReader(QEglFSKmsGbmScreen *screen)
304 QEglFSKmsGbmDevice *dev =
static_cast<QEglFSKmsGbmDevice *>(device());
305 dev->eventReader()->startWaitFlip(screen, &m_flipMutex, &m_flipCond);
312 succ = m_flipCond.wait(&m_flipMutex);
314 succ = m_flipCond.wait(&m_flipMutex, 300);
317 qCWarning(qLcEglfsKmsDebug) <<
"timeout on waitForFlipWithEventReader, screen to wait for:"
318 << screen <<
", screen waiting (shouldn't be the same screen):"
321 m_flipMutex.unlock();
322 screen->flipFinished();
325void QEglFSKmsGbmScreen::waitForFlip()
327 if (m_headless || m_cloneSource)
334 QEglFSKmsGbmDevice *dev =
static_cast<QEglFSKmsGbmDevice *>(device());
335 if (dev->usesEventReader()) {
336 waitForFlipWithEventReader(
this);
341 for (CloneDestination &d : m_cloneDests) {
342 if (d.screen !=
this)
343 waitForFlipWithEventReader(d.screen);
346 QMutexLocker lock(&s_nonThreadedFlipMutex);
347 while (m_gbm_bo_next) {
348 drmEventContext drmEvent;
349 memset(&drmEvent, 0,
sizeof(drmEvent));
350 drmEvent.version = 2;
351 drmEvent.vblank_handler =
nullptr;
352 drmEvent.page_flip_handler = nonThreadedPageFlipHandler;
353 drmHandleEvent(device()->fd(), &drmEvent);
357#if QT_CONFIG(drm_atomic)
358 device()->threadLocalAtomicReset();
362#if QT_CONFIG(drm_atomic)
363static void addAtomicFlip(drmModeAtomicReq *request,
const QKmsOutput &output, uint32_t fb)
365 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
366 output.eglfs_plane->framebufferPropertyId, fb);
368 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
369 output.eglfs_plane->crtcPropertyId, output.crtc_id);
371 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
372 output.eglfs_plane->srcwidthPropertyId, output.size.width() << 16);
374 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
375 output.eglfs_plane->srcXPropertyId, 0);
377 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
378 output.eglfs_plane->srcYPropertyId, 0);
380 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
381 output.eglfs_plane->srcheightPropertyId, output.size.height() << 16);
383 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
384 output.eglfs_plane->crtcXPropertyId, 0);
386 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
387 output.eglfs_plane->crtcYPropertyId, 0);
389 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
390 output.eglfs_plane->crtcwidthPropertyId, output.modes[output.mode].hdisplay);
392 drmModeAtomicAddProperty(request, output.eglfs_plane->id,
393 output.eglfs_plane->crtcheightPropertyId, output.modes[output.mode].vdisplay);
397void QEglFSKmsGbmScreen::flip()
401 if (m_headless || m_cloneSource)
404 if (!m_gbm_surface) {
405 qWarning(
"Cannot sync before platform init!");
409 m_gbm_bo_next = gbm_surface_lock_front_buffer(m_gbm_surface);
410 if (!m_gbm_bo_next) {
411 qWarning(
"Could not lock GBM surface front buffer for screen %s", qPrintable(name()));
415 auto gbmRelease = qScopeGuard([
this]{
416 m_flipPending =
false;
417 gbm_surface_release_buffer(m_gbm_surface, m_gbm_bo_next);
418 m_gbm_bo_next =
nullptr;
421 FrameBuffer *fb = framebufferForBufferObject(m_gbm_bo_next);
423 qWarning(
"FrameBuffer not available. Cannot flip");
426 ensureModeSet(fb->fb);
428 const QKmsOutput &thisOutput(output());
429 const int fd = device()->fd();
430 m_flipPending =
true;
432 if (device()->hasAtomicSupport()) {
433#if QT_CONFIG(drm_atomic)
434 drmModeAtomicReq *request = device()->threadLocalAtomicRequest();
436 addAtomicFlip(request, thisOutput, fb->fb);
437 static int zpos = qEnvironmentVariableIntValue(
"QT_QPA_EGLFS_KMS_ZPOS");
439 drmModeAtomicAddProperty(request, thisOutput.eglfs_plane->id,
440 thisOutput.eglfs_plane->zposPropertyId, zpos);
442 static uint blendOp = uint(qEnvironmentVariableIntValue(
"QT_QPA_EGLFS_KMS_BLEND_OP"));
444 drmModeAtomicAddProperty(request, thisOutput.eglfs_plane->id,
445 thisOutput.eglfs_plane->blendOpPropertyId, blendOp);
450 int ret = drmModePageFlip(fd,
453 DRM_MODE_PAGE_FLIP_EVENT,
456 qErrnoWarning(
"Could not queue DRM page flip on screen %s", qPrintable(name()));
461 for (CloneDestination &d : m_cloneDests) {
462 if (d.screen !=
this) {
463 d.screen->ensureModeSet(fb->fb);
464 d.cloneFlipPending =
true;
465 const QKmsOutput &destOutput(d.screen->output());
467 if (device()->hasAtomicSupport()) {
468#if QT_CONFIG(drm_atomic)
469 drmModeAtomicReq *request = device()->threadLocalAtomicRequest();
471 addAtomicFlip(request, destOutput, fb->fb);
481 int ret = drmModePageFlip(fd,
484 DRM_MODE_PAGE_FLIP_EVENT,
487 qErrnoWarning(
"Could not queue DRM page flip for screen %s (clones screen %s)",
488 qPrintable(d.screen->name()),
490 d.cloneFlipPending =
false;
496 if (device()->hasAtomicSupport()) {
497#if QT_CONFIG(drm_atomic)
498 if (!device()->threadLocalAtomicCommit(
this)) {
504 gbmRelease.dismiss();
507void QEglFSKmsGbmScreen::flipFinished()
510 m_cloneSource->cloneDestFlipFinished(
this);
514 m_flipPending =
false;
518void QEglFSKmsGbmScreen::cloneDestFlipFinished(QEglFSKmsGbmScreen *cloneDestScreen)
520 for (CloneDestination &d : m_cloneDests) {
521 if (d.screen == cloneDestScreen) {
522 d.cloneFlipPending =
false;
529void QEglFSKmsGbmScreen::updateFlipStatus()
539 for (
const CloneDestination &d : std::as_const(m_cloneDests)) {
540 if (d.cloneFlipPending)
544 if (m_gbm_bo_current) {
545 gbm_surface_release_buffer(m_gbm_surface,
549 m_gbm_bo_current = m_gbm_bo_next;
550 m_gbm_bo_next =
nullptr;
static uint32_t gbmFormatToDrmFormat(uint32_t gbmFormat)
static uint32_t drmFormatToGbmFormat(uint32_t drmFormat)