فهرست منبع

Don't require the GUI thread's assistance to render video

The scene graph is able to render on its own thread and mpv is
notifying us of a new frame from another thread. This gives us
the opportunity to render video without asking if the GUI thread
has any QQuickItem to update, and avoid stuttering the video if
anything unexpected is blocking the GUI thread's event loop.

Possibly helps for #257
Jocelyn Turcotte 9 سال پیش
والد
کامیت
b5d1eb3716
2فایلهای تغییر یافته به همراه36 افزوده شده و 12 حذف شده
  1. 35 10
      src/player/PlayerQuickItem.cpp
  2. 1 2
      src/player/PlayerQuickItem.h

+ 35 - 10
src/player/PlayerQuickItem.cpp

@@ -3,6 +3,7 @@
 #include <stdexcept>
 
 #include <QOpenGLContext>
+#include <QRunnable>
 
 #include <QtGui/QOpenGLFramebufferObject>
 
@@ -125,6 +126,24 @@ static void* get_proc_address(void* ctx, const char* name)
   return res;
 }
 
+namespace {
+
+class RequestRepaintJob : public QRunnable {
+  QQuickWindow *window;
+public:
+  RequestRepaintJob(QQuickWindow *window) : window(window) { }
+  void run() override {
+    // QSGThreadedRenderLoop::update has a special code path that will render
+    // without syncing the render and GUI threads unless asked elsewhere to support
+    // QQuickAnimator animations. This is currently triggered by the fact that
+    // QQuickWindow::update() is called from the render thread.
+    // This allows continuing rendering video while the GUI thread is busy.
+    window->update();
+  }
+};
+
+}
+
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 PlayerRenderer::PlayerRenderer(mpv::qt::Handle mpv, QQuickWindow* window)
 : m_mpv(mpv), m_mpvGL(0), m_window(window), m_size(), m_hAvrtHandle(0)
@@ -140,6 +159,8 @@ bool PlayerRenderer::init()
   DwmEnableMMCSS(TRUE);
 #endif
 
+  mpv_opengl_cb_set_update_callback(m_mpvGL, on_update, (void *)this);
+
   // Signals presence of MPGetNativeDisplay().
   const char *extensions = "GL_MP_MPGetNativeDisplay";
   return mpv_opengl_cb_init_gl(m_mpvGL, extensions, get_proc_address, NULL) >= 0;
@@ -191,6 +212,20 @@ void PlayerRenderer::onPlaybackActive(bool active)
 #endif
 }
 
+///////////////////////////////////////////////////////////////////////////////////////////////////
+void PlayerRenderer::on_update(void *ctx)
+{
+  PlayerRenderer *self = (PlayerRenderer *)ctx;
+  // QQuickWindow::scheduleRenderJob is expected to be called from the GUI thread but
+  // is thread-safe when using the QSGThreadedRenderLoop. We can detect a non-threaded render
+  // loop by checking if QQuickWindow::beforeSynchronizing was called from the GUI thread
+  // (which affects the QObject::thread() of the PlayerRenderer).
+  if (self->thread() == self->m_window->thread())
+    QMetaObject::invokeMethod(self->m_window, "update", Qt::QueuedConnection);
+  else
+    self->m_window->scheduleRenderJob(new RequestRepaintJob(self->m_window), QQuickWindow::NoStage);
+}
+
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 PlayerQuickItem::PlayerQuickItem(QQuickItem* parent)
 : QQuickItem(parent), m_mpvGL(NULL), m_renderer(NULL)
@@ -213,7 +248,6 @@ void PlayerQuickItem::onWindowChanged(QQuickWindow* win)
   {
     connect(win, &QQuickWindow::beforeSynchronizing, this, &PlayerQuickItem::onSynchronize, Qt::DirectConnection);
     connect(win, &QQuickWindow::sceneGraphInvalidated, this, &PlayerQuickItem::onInvalidate, Qt::DirectConnection);
-    connect(this, &PlayerQuickItem::onUpdate, win, &QQuickWindow::update, Qt::QueuedConnection);
   }
 }
 
@@ -269,13 +303,6 @@ void PlayerQuickItem::onInvalidate()
   m_renderer = NULL;
 }
 
-///////////////////////////////////////////////////////////////////////////////////////////////////
-void PlayerQuickItem::on_update(void *ctx)
-{
-  PlayerQuickItem *self = (PlayerQuickItem *)ctx;
-  emit self->onUpdate();
-}
-
 ///////////////////////////////////////////////////////////////////////////////////////////////////
 void PlayerQuickItem::initMpv(PlayerComponent* player)
 {
@@ -285,8 +312,6 @@ void PlayerQuickItem::initMpv(PlayerComponent* player)
   if (!m_mpvGL)
     throw FatalException(tr("OpenGL not enabled in libmpv."));
 
-  mpv_opengl_cb_set_update_callback(m_mpvGL, on_update, (void *)this);
-
   connect(player, &PlayerComponent::windowVisible, this, &QQuickItem::setVisible);
   window()->update();
 }

+ 1 - 2
src/player/PlayerQuickItem.h

@@ -31,6 +31,7 @@ public slots:
   void onPlaybackActive(bool active);
 
 private:
+  static void on_update(void *ctx);
   mpv::qt::Handle m_mpv;
   mpv_opengl_cb_context* m_mpvGL;
   QQuickWindow* m_window;
@@ -50,7 +51,6 @@ public:
     QString debugInfo() { return m_debugInfo; }
 
 signals:
-    void onUpdate();
     void onFatalError(QString message);
 
 private slots:
@@ -60,7 +60,6 @@ private slots:
     void onHandleFatalError(QString message);
 
 private:
-    static void on_update(void *ctx);
     mpv::qt::Handle m_mpv;
     mpv_opengl_cb_context* m_mpvGL;
     PlayerRenderer* m_renderer;