1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135
|
// Copyright (C) 2016 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include "qopenglcontextwindow.h"
#include <QtGui/QOpenGLFunctions>
#include <QtGui/QOffscreenSurface>
#include <QtGui/QGuiApplication>
#include <QtGui/QMatrix4x4>
#include <qpa/qplatformnativeinterface.h>
#include <QtEglSupport/private/qeglconvenience_p.h>
QOpenGLContextWindow::QOpenGLContextWindow()
: m_blitter(0)
{
setSurfaceType(OpenGLSurface);
m_context = new QOpenGLContext(this);
m_context->setFormat(requestedFormat());
m_context->create();
m_image = QImage(QStringLiteral("qticon64.png")).convertToFormat(QImage::Format_RGBA8888);
Q_ASSERT(!m_image.isNull());
create(); // to make sure format() returns something real
createForeignContext();
}
QOpenGLContextWindow::~QOpenGLContextWindow()
{
if (m_blitter) {
m_blitter->destroy(); // the dtor does not call this for some reason
delete m_blitter;
}
}
void QOpenGLContextWindow::render()
{
if (!m_context->makeCurrent(this))
qFatal("makeCurrent() failed");
QOpenGLFunctions *f = m_context->functions();
f->glViewport(0, 0, dWidth(), dHeight());
f->glClearColor(0, 0, 0, 1);
f->glClear(GL_COLOR_BUFFER_BIT);
if (!m_blitter) {
m_blitter = new QOpenGLTextureBlitter;
m_blitter->create();
}
// Draw the image. If nothing gets shown, then something went wrong with the context
// adoption or sharing was not successfully enabled.
m_blitter->bind();
QRectF r(0, 0, dWidth(), dHeight());
QMatrix4x4 target = QOpenGLTextureBlitter::targetTransform(QRectF(100, 100, 100, 100), r.toRect());
m_blitter->blit(m_textureId, target, QOpenGLTextureBlitter::OriginTopLeft);
m_blitter->release();
m_context->swapBuffers(this);
}
void QOpenGLContextWindow::exposeEvent(QExposeEvent *)
{
if (isExposed())
render();
}
void QOpenGLContextWindow::createForeignContext()
{
// Here a context will be created manually. This context will share with m_context's
// underlying native context. This way the texture, that belongs to the context
// created here, will be accessible from m_context too.
using namespace QNativeInterface;
auto *eglContext = m_context->nativeInterface<QEGLContext>();
if (!eglContext)
qFatal("Not running with EGL backend");
EGLContext shareCtx = eglContext->nativeContext();
Q_ASSERT(shareCtx != EGL_NO_CONTEXT);
EGLDisplay dpy = (EGLDisplay) qGuiApp->platformNativeInterface()->nativeResourceForWindow(
QByteArrayLiteral("egldisplay"), this);
Q_ASSERT(dpy != EGL_NO_DISPLAY);
QSurfaceFormat fmt = format();
EGLConfig config = q_configFromGLFormat(dpy, fmt);
QList<EGLint> contextAttrs;
contextAttrs.append(EGL_CONTEXT_CLIENT_VERSION);
contextAttrs.append(fmt.majorVersion());
contextAttrs.append(EGL_NONE);
switch (fmt.renderableType()) {
#ifdef EGL_VERSION_1_4
case QSurfaceFormat::OpenGL:
eglBindAPI(EGL_OPENGL_API);
break;
#endif // EGL_VERSION_1_4
default:
eglBindAPI(EGL_OPENGL_ES_API);
break;
}
EGLContext ctx = eglCreateContext(dpy, config, shareCtx, contextAttrs.constData());
Q_ASSERT(ctx != EGL_NO_CONTEXT);
// Wrap ctx into a QOpenGLContext.
QOpenGLContext *ctxWrap = QEGLContext::fromNative(ctx, dpy, m_context);
Q_ASSERT(ctxWrap->nativeInterface<QEGLContext>()->nativeContext() == ctx);
QOffscreenSurface surface;
surface.setFormat(fmt);
surface.create();
if (!ctxWrap->makeCurrent(&surface))
qFatal("Failed to make pbuffer surface current");
// Create the texture.
QOpenGLFunctions *f = QOpenGLContext::currentContext()->functions();
GLuint textureId = 0;
f->glGenTextures(1, &textureId);
f->glBindTexture(GL_TEXTURE_2D, textureId);
f->glTexParameterf(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
f->glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, m_image.width(), m_image.height(), 0, GL_RGBA, GL_UNSIGNED_BYTE,
m_image.constBits());
Q_ASSERT(f->glGetError() == GL_NO_ERROR);
ctxWrap->doneCurrent();
delete ctxWrap; // ctx is not destroyed
eglDestroyContext(dpy, ctx); // resources like the texture stay alive until any context on the share list is alive
Q_ASSERT(eglGetError() == EGL_SUCCESS);
m_textureId = textureId;
}
|