[drape] remove synchronization on Qt rendering. FR now render into texture, GUI thread blit this texture on screen

This commit is contained in:
ExMix 2015-10-24 20:05:56 +03:00 committed by r.kuznetsov
parent 4fe196ed84
commit ab3e1e12bd
7 changed files with 289 additions and 273 deletions

View file

@ -13,6 +13,10 @@
#include <QtGui/QMouseEvent>
#include <QtGui/QGuiApplication>
#include <QtGui/QOpenGLShaderProgram>
#include <QtGui/QOpenGLContextGroup>
#include <QtGui/QOpenGLFunctions>
#include <QtGui/QVector2D>
#include <QtWidgets/QMenu>
#include <QtWidgets/QApplication>
@ -21,6 +25,7 @@
#include <QtCore/QLocale>
#include <QtCore/QDateTime>
#include <QtCore/QThread>
#include <QtCore/QTimer>
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<UserMarkCopy> mark)
{
});
@ -88,6 +90,11 @@ bool IsLocationEmulation(QMouseEvent * e)
vector<storage::TIndex> 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<QtOGLContextFactory>(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<mutex> 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<mutex> 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<mutex> 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<mutex> 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<mutex> 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<mutex> 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());

View file

@ -12,15 +12,17 @@
#include "std/mutex.hpp"
#include "std/condition_variable.hpp"
#include <QtGui/QOpenGLWindow>
#include <QtWidgets/QOpenGLWidget>
class QQuickWindow;
namespace qt
{
class QScaleSlider;
class DrawWidget : public QOpenGLWindow
class DrawWidget : public QOpenGLWidget
{
using TBase = QOpenGLWindow;
using TBase = QOpenGLWidget;
drape_ptr<QtOGLContextFactory> m_contextFactory;
unique_ptr<Framework> 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);

View file

@ -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()));

View file

@ -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()

View file

@ -1,57 +1,53 @@
#pragma once
#include "drape/oglcontext.hpp"
#include "std/mutex.hpp"
#include "std/function.hpp"
#include "std/atomic.hpp"
#include <QtGui/QWindow>
#include <QtGui/QOffscreenSurface>
#include <QtGui/QOpenGLFramebufferObject>
#include <QtGui/QOpenGLContext>
#include <QtCore/QThread>
class QtRenderOGLContext : public dp::OGLContext
{
public:
using TRegisterThreadFn = function<void (QThread * thread)>;
using TSwapFn = function<void ()>;
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<bool> 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;
};

View file

@ -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;
}

View file

@ -3,19 +3,19 @@
#include "drape/oglcontextfactory.hpp"
#include "qt/qtoglcontext.hpp"
#include <QtGui/QOffscreenSurface>
#include <QtGui/QOpenGLContext>
#include <QtGui/QOpenGLFramebufferObject>
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;
};