From ab3e1e12bd3f1252cb005f1823340afba206b722 Mon Sep 17 00:00:00 2001 From: ExMix Date: Sat, 24 Oct 2015 20:05:56 +0300 Subject: [PATCH] [drape] remove synchronization on Qt rendering. FR now render into texture, GUI thread blit this texture on screen --- qt/draw_widget.cpp | 234 ++++++++++++++++--------------------- qt/draw_widget.hpp | 48 ++------ qt/mainwindow.cpp | 13 +-- qt/qtoglcontext.cpp | 113 ++++++++++++------ qt/qtoglcontext.hpp | 74 ++++++------ qt/qtoglcontextfactory.cpp | 60 ++++++++-- qt/qtoglcontextfactory.hpp | 20 ++-- 7 files changed, 289 insertions(+), 273 deletions(-) diff --git a/qt/draw_widget.cpp b/qt/draw_widget.cpp index 63229ca751..c967b73df5 100644 --- a/qt/draw_widget.cpp +++ b/qt/draw_widget.cpp @@ -13,6 +13,10 @@ #include #include +#include +#include +#include +#include #include #include @@ -21,6 +25,7 @@ #include #include #include +#include namespace qt { @@ -67,18 +72,15 @@ bool IsLocationEmulation(QMouseEvent * e) void DummyDismiss() {} - DrawWidget::DrawWidget() - : m_contextFactory(nullptr), + DrawWidget::DrawWidget(QWidget * parent) + : TBase(parent), + m_contextFactory(nullptr), m_framework(new Framework()), m_ratio(1.0), - m_rendererThread(nullptr), - m_state(NotInitialized), m_pScale(nullptr), m_enableScaleUpdate(true), m_emulatingLocation(false) { - setSurfaceType(QSurface::OpenGLSurface); - m_framework->SetUserMarkActivationListener([](unique_ptr mark) { }); @@ -88,6 +90,11 @@ bool IsLocationEmulation(QMouseEvent * e) vector const &) { }); + + QTimer * timer = new QTimer(this); + VERIFY(connect(timer, SIGNAL(timeout()), this, SLOT(update())), ()); + timer->setSingleShot(false); + timer->start(30); } DrawWidget::~DrawWidget() @@ -106,17 +113,6 @@ bool IsLocationEmulation(QMouseEvent * e) void DrawWidget::PrepareShutdown() { - if (!m_contextFactory) - return; - - // Discard current and all future Swap requests - m_contextFactory->shutDown(); - frameSwappedSlot(NotInitialized); - - // Shutdown engine. FR have ogl context in this moment and can delete OGL resources - // PrepareToShutdown make FR::join and after this call we can bind OGL context to gui thread - m_framework->PrepareToShutdown(); - m_contextFactory.reset(); } void DrawWidget::UpdateAfterSettingsChanged() @@ -197,27 +193,98 @@ bool IsLocationEmulation(QMouseEvent * e) p.m_widgetsInitInfo[gui::WIDGET_SCALE_LABEL] = gui::Position(dp::LeftBottom); - m_framework->CreateDrapeEngine(make_ref(m_contextFactory), move(p)); + m_framework->CreateDrapeEngine(make_ref(m_contextFactory), std::move(p)); m_framework->AddViewportListener(bind(&DrawWidget::OnViewportChanged, this, _1)); } void DrawWidget::initializeGL() { - Qt::ConnectionType swapType = Qt::QueuedConnection; - Qt::ConnectionType regType = Qt::BlockingQueuedConnection; - VERIFY(connect(this, SIGNAL(Swap()), SLOT(OnSwap()), swapType), ()); - VERIFY(connect(this, SIGNAL(RegRenderingThread(QThread *)), SLOT(OnRegRenderingThread(QThread *)), regType), ()); - VERIFY(connect(this, SIGNAL(frameSwapped()), SLOT(frameSwappedSlot())), ()); - - ASSERT(m_contextFactory == nullptr, ()); - m_ratio = devicePixelRatio(); - QtOGLContextFactory::TRegisterThreadFn regFn = bind(&DrawWidget::CallRegisterThread, this, _1); - QtOGLContextFactory::TSwapFn swapFn = bind(&DrawWidget::CallSwap, this); - m_contextFactory = make_unique_dp(context(), QThread::currentThread(), regFn, swapFn); + m_contextFactory.reset(new QtOGLContextFactory(context())); CreateEngine(); LoadState(); + } + + void DrawWidget::paintGL() + { + static QOpenGLShaderProgram * program = nullptr; + if (program == nullptr) + { + const char * vertexSrc = "\ + attribute vec2 a_position; \ + attribute vec2 a_texCoord; \ + uniform mat4 u_projection; \ + varying vec2 v_texCoord; \ + \ + void main(void) \ + { \ + gl_Position = u_projection * vec4(a_position, 0.0, 1.0);\ + v_texCoord = a_texCoord; \ + }"; + + const char * fragmentSrc = "\ + uniform sampler2D u_sampler; \ + varying vec2 v_texCoord; \ + \ + void main(void) \ + { \ + gl_FragColor = texture2D(u_sampler, v_texCoord); \ + }"; + + program = new QOpenGLShaderProgram(this); + program->addShaderFromSourceCode(QOpenGLShader::Vertex, vertexSrc); + program->addShaderFromSourceCode(QOpenGLShader::Fragment, fragmentSrc); + program->link(); + } + + if (m_contextFactory->LockFrame()) + { + QOpenGLFunctions * funcs = context()->functions(); + funcs->glActiveTexture(GL_TEXTURE0); + GLuint image = m_contextFactory->GetTextureHandle(); + funcs->glBindTexture(GL_TEXTURE_2D, image); + + int projectionLocation = program->uniformLocation("u_projection"); + int samplerLocation = program->uniformLocation("u_sampler"); + + QMatrix4x4 projection; + projection.ortho(rect()); + + program->bind(); + program->setUniformValue(projectionLocation, projection); + program->setUniformValue(samplerLocation, 0); + + float w = width(); + float h = height(); + + QVector2D positions[4] = + { + QVector2D(0.0, 0.0), + QVector2D(w, 0.0), + QVector2D(0.0, h), + QVector2D(w, h) + }; + + QRectF const & texRect = m_contextFactory->GetTexRect(); + QVector2D texCoords[4] = + { + QVector2D(texRect.bottomLeft()), + QVector2D(texRect.bottomRight()), + QVector2D(texRect.topLeft()), + QVector2D(texRect.topRight()) + }; + + program->enableAttributeArray("a_position"); + program->enableAttributeArray("a_texCoord"); + program->setAttributeArray("a_position", positions, 0); + program->setAttributeArray("a_texCoord", texCoords, 0); + + funcs->glClearColor(0.65, 0.65, 0.65, 1.0); + funcs->glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); + funcs->glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); + + m_contextFactory->UnlockFrame(); + } - emit EngineCreated(); } void DrawWidget::resizeGL(int width, int height) @@ -235,32 +302,10 @@ bool IsLocationEmulation(QMouseEvent * e) layout[w] = pos.m_pixelPivot; }); - m_framework->SetWidgetLayout(move(layout)); + m_framework->SetWidgetLayout(std::move(layout)); } } - void DrawWidget::paintGL() { /*Must be empty*/ } - - void DrawWidget::exposeEvent(QExposeEvent * event) - { - if (isExposed()) - { - m_swapMutex.lock(); - if (m_state == Render) - { - unique_lock waitContextLock(m_waitContextMutex); - m_state = WaitContext; - m_swapMutex.unlock(); - - m_waitContextCond.wait(waitContextLock, [this](){ return m_state != WaitContext; }); - } - else - m_swapMutex.unlock(); - } - - TBase::exposeEvent(event); - } - void DrawWidget::mousePressEvent(QMouseEvent * e) { TBase::mousePressEvent(e); @@ -339,87 +384,6 @@ bool IsLocationEmulation(QMouseEvent * e) m_emulatingLocation = false; } - void DrawWidget::CallSwap() - { - // Called on FR thread. In this point OGL context have already moved into GUI thread. - unique_lock lock(m_swapMutex); - if (m_state == NotInitialized) - { - // This can be in two cases if GUI thread in PrepareToShutDown - return; - } - - ASSERT(m_state != WaitSwap, ()); - if (m_state == WaitContext) - { - lock_guard waitContextLock(m_waitContextMutex); - m_state = Render; - m_waitContextCond.notify_one(); - } - - if (m_state == Render) - { - // We have to wait, while Qt on GUI thread finish composing widgets and make SwapBuffers - // After SwapBuffers Qt will call our SLOT(frameSwappedSlot) - m_state = WaitSwap; - emit Swap(); - m_swapCond.wait(lock, [this]() { return m_state != WaitSwap; }); - } - } - - void DrawWidget::CallRegisterThread(QThread * thread) - { - // Called on FR thread. SIGNAL(RegRenderingThread) and SLOT(OnRegRenderingThread) - // connected by through Qt::BlockingQueuedConnection and we don't need any synchronization - ASSERT(m_state == NotInitialized, ()); - emit RegRenderingThread(thread); - } - - void DrawWidget::OnSwap() - { - // Called on GUI thread. In this point FR thread must wait SwapBuffers signal - lock_guard lock(m_swapMutex); - if (m_state == WaitSwap) - { - context()->makeCurrent(this); - update(); - } - } - - void DrawWidget::OnRegRenderingThread(QThread * thread) - { - // Called on GUI thread. - // Here we register thread of FR, to return OGL context into it after SwapBuffers - // After this operation we can start rendering into back buffer on FR thread - lock_guard lock(m_swapMutex); - - ASSERT(m_state == NotInitialized, ()); - m_state = Render; - m_rendererThread = thread; - MoveContextToRenderThread(); - } - - void DrawWidget::frameSwappedSlot(RenderingState state) - { - // Qt call this slot on GUI thread after glSwapBuffers perfomed - // Here we move OGL context into FR thread and wake up FR - lock_guard lock(m_swapMutex); - - if (m_state == WaitSwap) - { - MoveContextToRenderThread(); - m_state = state; - m_swapCond.notify_all(); - } - } - - void DrawWidget::MoveContextToRenderThread() - { - QOpenGLContext * ctx = context(); - ctx->doneCurrent(); - ctx->moveToThread(m_rendererThread); - } - void DrawWidget::wheelEvent(QWheelEvent * e) { m_framework->Scale(exp(e->delta() / 360.0), m2::PointD(L2D(e->x()), L2D(e->y())), false); @@ -529,7 +493,7 @@ bool IsLocationEmulation(QMouseEvent * e) void DrawWidget::UpdateScaleControl() { - if (m_pScale && isExposed() && m_enableScaleUpdate) + if (m_pScale && m_enableScaleUpdate) { // don't send ScaleChanged m_pScale->SetPosWithBlockedSignals(m_framework->GetDrawScale()); diff --git a/qt/draw_widget.hpp b/qt/draw_widget.hpp index e71d3af89b..bd8e372f48 100644 --- a/qt/draw_widget.hpp +++ b/qt/draw_widget.hpp @@ -12,15 +12,17 @@ #include "std/mutex.hpp" #include "std/condition_variable.hpp" -#include +#include + +class QQuickWindow; namespace qt { class QScaleSlider; - class DrawWidget : public QOpenGLWindow + class DrawWidget : public QOpenGLWidget { - using TBase = QOpenGLWindow; + using TBase = QOpenGLWidget; drape_ptr m_contextFactory; unique_ptr m_framework; @@ -41,7 +43,7 @@ namespace qt void SliderReleased(); public: - DrawWidget(); + DrawWidget(QWidget * parent); ~DrawWidget(); void SetScaleControl(QScaleSlider * pScale); @@ -69,13 +71,11 @@ namespace qt void CreateEngine(); protected: + void initializeGL() override; + void paintGL() override; + void resizeGL(int width, int height) override; /// @name Overriden from QOpenGLWindow. //@{ - void initializeGL() override; - void resizeGL(int width, int height) override; - void paintGL() override; - - void exposeEvent(QExposeEvent * event) override; void mousePressEvent(QMouseEvent * e) override; void mouseDoubleClickEvent(QMouseEvent * e) override; void mouseMoveEvent(QMouseEvent * e) override; @@ -83,35 +83,7 @@ namespace qt void wheelEvent(QWheelEvent * e) override; void keyPressEvent(QKeyEvent * e) override; void keyReleaseEvent(QKeyEvent * e) override; - //@}a - - private: - enum RenderingState - { - NotInitialized, - WaitContext, - WaitSwap, - Render, - }; - - void CallSwap(); - void CallRegisterThread(QThread * thread); - Q_SIGNAL void Swap(); - Q_SIGNAL void RegRenderingThread(QThread * thread); - Q_SLOT void OnSwap(); - Q_SLOT void OnRegRenderingThread(QThread * thread); - Q_SLOT void frameSwappedSlot(RenderingState state = Render); - - void MoveContextToRenderThread(); - QThread * m_rendererThread; - - mutex m_swapMutex; - condition_variable m_swapCond; - - mutex m_waitContextMutex; - condition_variable m_waitContextCond; - - RenderingState m_state; + //@} private: void SubmitFakeLocationPoint(m2::PointD const & pt); diff --git a/qt/mainwindow.cpp b/qt/mainwindow.cpp index de45ee2cd4..29d59ce96e 100644 --- a/qt/mainwindow.cpp +++ b/qt/mainwindow.cpp @@ -55,11 +55,11 @@ MainWindow::MainWindow() : m_locationService(CreateDesktopLocationService(*this) QDesktopWidget const * desktop(QApplication::desktop()); setGeometry(desktop->screenGeometry(desktop->primaryScreen())); - m_pDrawWidget = new DrawWidget(); - QSurfaceFormat format = m_pDrawWidget->requestedFormat(); + m_pDrawWidget = new DrawWidget(this); + QSurfaceFormat format = m_pDrawWidget->format(); - format.setMajorVersion(3); - format.setMinorVersion(2); + format.setMajorVersion(2); + format.setMinorVersion(1); format.setAlphaBufferSize(0); format.setBlueBufferSize(8); @@ -74,9 +74,8 @@ MainWindow::MainWindow() : m_locationService(CreateDesktopLocationService(*this) format.setProfile(QSurfaceFormat::CompatibilityProfile); //format.setOption(QSurfaceFormat::DebugContext); m_pDrawWidget->setFormat(format); - QWidget * w = QWidget::createWindowContainer(m_pDrawWidget, this); - w->setMouseTracking(true); - setCentralWidget(w); + m_pDrawWidget->setMouseTracking(true); + setCentralWidget(m_pDrawWidget); QObject::connect(m_pDrawWidget, SIGNAL(EngineCreated()), this, SLOT(OnEngineCreated())); diff --git a/qt/qtoglcontext.cpp b/qt/qtoglcontext.cpp index a423e46756..0abb3c0a16 100644 --- a/qt/qtoglcontext.cpp +++ b/qt/qtoglcontext.cpp @@ -2,85 +2,122 @@ #include "base/assert.hpp" #include "base/logging.hpp" +#include "base/macros.hpp" +#include "base/math.hpp" #include "drape/glfunctions.hpp" -QtRenderOGLContext::QtRenderOGLContext(QOpenGLContext * nativeContext, QThread * guiThread, - TRegisterThreadFn const & regFn, TSwapFn const & swapFn) - : m_surface(nativeContext->surface()) - , m_ctx(nativeContext) - , m_guiThread(guiThread) - , m_regFn(regFn) - , m_swapFn(swapFn) - , m_isRegistered(false) - , m_shutedDown(false) +QtRenderOGLContext::QtRenderOGLContext(QOpenGLContext * rootContext, QOffscreenSurface * surface) + : m_surface(surface) { + m_ctx = new QOpenGLContext(); + m_ctx->setFormat(rootContext->format()); + m_ctx->setShareContext(rootContext); + m_ctx->create(); + ASSERT(m_ctx->isValid(), ()); +} + +QtRenderOGLContext::~QtRenderOGLContext() +{ + delete m_frontFrame; + delete m_backFrame; + delete m_ctx; } void QtRenderOGLContext::present() { - if (m_shutedDown) - return; + if (!m_resizeLock) + lockFrame(); - MoveContextOnGui(); - m_swapFn(); + m_resizeLock = false; + GLFunctions::glFinish(); - makeCurrent(); + swap(m_frontFrame, m_backFrame); + unlockFrame(); } void QtRenderOGLContext::makeCurrent() { - if (!m_isRegistered) - { - m_regFn(QThread::currentThread()); - m_isRegistered = true; - } - - m_ctx->makeCurrent(m_surface); + VERIFY(m_ctx->makeCurrent(m_surface), ()); } void QtRenderOGLContext::doneCurrent() { - MoveContextOnGui(); + m_ctx->doneCurrent(); } void QtRenderOGLContext::setDefaultFramebuffer() { - GLFunctions::glBindFramebuffer(GL_FRAMEBUFFER, m_ctx->defaultFramebufferObject()); + if (m_backFrame == nullptr) + { + LOG(LWARNING, ("Back framebuffer has't been created")); + return; + } + + m_backFrame->bind(); } -void QtRenderOGLContext::shutDown() +void QtRenderOGLContext::resize(int w, int h) { - m_shutedDown = true; + lockFrame(); + m_resizeLock = true; + + delete m_frontFrame; + delete m_backFrame; + + QSize size(my::NextPowOf2(w), my::NextPowOf2(h)); + m_texRect = QRectF(0.0, 0.0, w / (float)size.width(), h / (float)size.height()); + + m_frontFrame = new QOpenGLFramebufferObject(size, QOpenGLFramebufferObject::Depth); + m_backFrame = new QOpenGLFramebufferObject(size, QOpenGLFramebufferObject::Depth); } -void QtRenderOGLContext::MoveContextOnGui() +void QtRenderOGLContext::lockFrame() { - m_ctx->doneCurrent(); - m_ctx->moveToThread(m_guiThread); + m_lock.lock(); } -QtUploadOGLContext::QtUploadOGLContext(QSurface * surface, QOpenGLContext * contextToShareWith) +QRectF const & QtRenderOGLContext::getTexRect() const { - m_surface = surface; - m_nativeContext = new QOpenGLContext(); + return m_texRect; +} - ASSERT(contextToShareWith != nullptr, ()); - m_nativeContext->setShareContext(contextToShareWith); +GLuint QtRenderOGLContext::getTextureHandle() const +{ + if (m_frontFrame == nullptr) + return 0; - m_nativeContext->setFormat(contextToShareWith->format()); - VERIFY(m_nativeContext->create(), ()); + return m_frontFrame->texture(); +} + +void QtRenderOGLContext::unlockFrame() +{ + m_lock.unlock(); +} + +QtUploadOGLContext::QtUploadOGLContext(QOpenGLContext * rootContext, QOffscreenSurface * surface) + : m_surface(surface) +{ + m_ctx = new QOpenGLContext(); + m_ctx->setFormat(rootContext->format()); + m_ctx->setShareContext(rootContext); + m_ctx->create(); + ASSERT(m_ctx->isValid(), ()); } QtUploadOGLContext::~QtUploadOGLContext() { - delete m_nativeContext; + delete m_ctx; } void QtUploadOGLContext::makeCurrent() { - ASSERT(m_nativeContext->isValid(), ()); - m_nativeContext->makeCurrent(m_surface); + m_ctx->makeCurrent(m_surface); +} + +void QtUploadOGLContext::doneCurrent() +{ + m_ctx->doneCurrent(); } void QtUploadOGLContext::present() diff --git a/qt/qtoglcontext.hpp b/qt/qtoglcontext.hpp index feaaee10c6..1e4ac1dab4 100644 --- a/qt/qtoglcontext.hpp +++ b/qt/qtoglcontext.hpp @@ -1,57 +1,53 @@ #pragma once #include "drape/oglcontext.hpp" +#include "std/mutex.hpp" -#include "std/function.hpp" -#include "std/atomic.hpp" - -#include +#include +#include #include -#include - class QtRenderOGLContext : public dp::OGLContext { public: - using TRegisterThreadFn = function; - using TSwapFn = function; + QtRenderOGLContext(QOpenGLContext * rootContext, QOffscreenSurface * surface); + ~QtRenderOGLContext(); - QtRenderOGLContext(QOpenGLContext * nativeContext, QThread * guiThread, - TRegisterThreadFn const & regFn, TSwapFn const & swapFn); + void present() override; + void makeCurrent() override; + void doneCurrent() override; + void setDefaultFramebuffer() override; + void resize(int w, int h) override; + + void lockFrame(); + GLuint getTextureHandle() const; + QRectF const & getTexRect() const; + void unlockFrame(); + +private: + QOffscreenSurface * m_surface = nullptr; + QOpenGLContext * m_ctx = nullptr; + + QOpenGLFramebufferObject * m_frontFrame = nullptr; + QOpenGLFramebufferObject * m_backFrame = nullptr; + QRectF m_texRect = QRectF(0.0, 0.0, 0.0, 0.0); + + mutex m_lock; + bool m_resizeLock = false; +}; + +class QtUploadOGLContext: public dp::OGLContext +{ +public: + QtUploadOGLContext(QOpenGLContext * rootContext, QOffscreenSurface * surface); + ~QtUploadOGLContext(); void present() override; void makeCurrent() override; void doneCurrent() override; void setDefaultFramebuffer() override; - void shutDown(); - - QOpenGLContext * getNativeContext() { return m_ctx; } - private: - void MoveContextOnGui(); - -private: - QSurface * m_surface; - QOpenGLContext * m_ctx; - QThread * m_guiThread; - TRegisterThreadFn m_regFn; - TSwapFn m_swapFn; - bool m_isRegistered; - atomic m_shutedDown; -}; - -class QtUploadOGLContext: public dp::OGLContext -{ -public: - QtUploadOGLContext(QSurface * surface, QOpenGLContext * contextToShareWith); - ~QtUploadOGLContext(); - - virtual void present(); - virtual void makeCurrent(); - virtual void setDefaultFramebuffer(); - -private: - QOpenGLContext * m_nativeContext; - QSurface * m_surface; + QOpenGLContext * m_ctx = nullptr; + QOffscreenSurface * m_surface = nullptr; }; diff --git a/qt/qtoglcontextfactory.cpp b/qt/qtoglcontextfactory.cpp index d40caed765..2d8093da53 100644 --- a/qt/qtoglcontextfactory.cpp +++ b/qt/qtoglcontextfactory.cpp @@ -2,13 +2,13 @@ #include "base/assert.hpp" -QtOGLContextFactory::QtOGLContextFactory(QOpenGLContext * renderContext, QThread * thread, - TRegisterThreadFn const & regFn, TSwapFn const & swapFn) - : m_drawContext(new QtRenderOGLContext(renderContext, thread, regFn, swapFn)) +QtOGLContextFactory::QtOGLContextFactory(QOpenGLContext * rootContext) + : m_rootContext(rootContext) + , m_drawContext(nullptr) , m_uploadContext(nullptr) { - m_uploadThreadSurface = new QOffscreenSurface(renderContext->screen()); - m_uploadThreadSurface->create(); + m_uploadSurface = createSurface(); + m_drawSurface = createSurface(); } QtOGLContextFactory::~QtOGLContextFactory() @@ -16,19 +16,63 @@ QtOGLContextFactory::~QtOGLContextFactory() delete m_drawContext; delete m_uploadContext; - m_uploadThreadSurface->destroy(); - delete m_uploadThreadSurface; + m_drawSurface->destroy(); + m_uploadSurface->destroy(); + + delete m_drawSurface; + delete m_uploadSurface; +} + +bool QtOGLContextFactory::LockFrame() +{ + if (m_drawContext == nullptr) + return false; + + m_drawContext->lockFrame(); + return true; +} + +QRectF const & QtOGLContextFactory::GetTexRect() const +{ + ASSERT(m_drawContext != nullptr, ()); + return m_drawContext->getTexRect(); +} + +GLuint QtOGLContextFactory::GetTextureHandle() const +{ + ASSERT(m_drawContext != nullptr, ()); + return m_drawContext->getTextureHandle(); +} + +void QtOGLContextFactory::UnlockFrame() +{ + ASSERT(m_drawContext != nullptr, ()); + m_drawContext->unlockFrame(); } dp::OGLContext * QtOGLContextFactory::getDrawContext() { + if (m_drawContext == nullptr) + m_drawContext = new QtRenderOGLContext(m_rootContext, m_drawSurface); + return m_drawContext; } dp::OGLContext * QtOGLContextFactory::getResourcesUploadContext() { if (m_uploadContext == nullptr) - m_uploadContext = new QtUploadOGLContext(m_uploadThreadSurface, m_drawContext->getNativeContext()); + m_uploadContext = new QtUploadOGLContext(m_rootContext, m_uploadSurface); return m_uploadContext; } + +QOffscreenSurface * QtOGLContextFactory::createSurface() +{ + QSurfaceFormat format = m_rootContext->format(); + QOffscreenSurface * result = new QOffscreenSurface(m_rootContext->screen()); + result->setFormat(format); + result->create(); + ASSERT(result->isValid(), ()); + + return result; +} diff --git a/qt/qtoglcontextfactory.hpp b/qt/qtoglcontextfactory.hpp index ea9c4936cd..16bb0c7362 100644 --- a/qt/qtoglcontextfactory.hpp +++ b/qt/qtoglcontextfactory.hpp @@ -3,19 +3,19 @@ #include "drape/oglcontextfactory.hpp" #include "qt/qtoglcontext.hpp" -#include +#include +#include class QtOGLContextFactory : public dp::OGLContextFactory { public: - using TRegisterThreadFn = QtRenderOGLContext::TRegisterThreadFn; - using TSwapFn = QtRenderOGLContext::TSwapFn; - - QtOGLContextFactory(QOpenGLContext * renderContext, QThread * thread, - TRegisterThreadFn const & regFn, TSwapFn const & swapFn); + QtOGLContextFactory(QOpenGLContext * rootContext); ~QtOGLContextFactory(); - void shutDown() { m_drawContext->shutDown(); } + bool LockFrame(); + GLuint GetTextureHandle() const; + QRectF const & GetTexRect() const; + void UnlockFrame(); virtual dp::OGLContext * getDrawContext(); virtual dp::OGLContext * getResourcesUploadContext(); @@ -24,8 +24,12 @@ protected: virtual bool isDrawContextCreated() const { return m_drawContext != nullptr; } virtual bool isUploadContextCreated() const { return m_uploadContext != nullptr; } + QOffscreenSurface * createSurface(); + private: + QOpenGLContext * m_rootContext; QtRenderOGLContext * m_drawContext; + QOffscreenSurface * m_drawSurface; QtUploadOGLContext * m_uploadContext; - QOffscreenSurface * m_uploadThreadSurface; + QOffscreenSurface * m_uploadSurface; };