aboutsummaryrefslogtreecommitdiffstats
path: root/src/quick/scenegraph/compressedtexture
diff options
context:
space:
mode:
Diffstat (limited to 'src/quick/scenegraph/compressedtexture')
-rw-r--r--src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture.cpp16
-rw-r--r--src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture_p.h6
-rw-r--r--src/quick/scenegraph/compressedtexture/qsgcompressedtexture.cpp179
-rw-r--r--src/quick/scenegraph/compressedtexture/qsgcompressedtexture_p.h23
4 files changed, 212 insertions, 12 deletions
diff --git a/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture.cpp b/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture.cpp
index 796bc870de..46b2c6386c 100644
--- a/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture.cpp
+++ b/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture.cpp
@@ -65,7 +65,7 @@ namespace QSGCompressedAtlasTexture
{
Atlas::Atlas(const QSize &size, uint format)
- : QSGAtlasTexture::AtlasBase(size)
+ : QSGOpenGLAtlasTexture::AtlasBase(size)
, m_format(format)
{
}
@@ -88,10 +88,20 @@ Texture *Atlas::create(const QByteArray &data, int dataLength, int dataOffset, c
void Atlas::generateTexture()
{
+ int bytesPerBlock = 8;
+ switch (m_format) {
+ case QOpenGLTexture::RGBA8_ETC2_EAC:
+ case QOpenGLTexture::RGBA_DXT3:
+ case QOpenGLTexture::RGBA_DXT5:
+ bytesPerBlock = 16;
+ default:
+ break;
+ }
+
QOpenGLFunctions *funcs = QOpenGLContext::currentContext()->functions();
funcs->glCompressedTexImage2D(GL_TEXTURE_2D, 0, m_format,
m_size.width(), m_size.height(), 0,
- (m_size.width() * m_size.height()) / 2,
+ (m_size.width() / 4 * m_size.height() / 4) * bytesPerBlock,
nullptr);
}
@@ -117,7 +127,7 @@ void Atlas::uploadPendingTexture(int i)
}
Texture::Texture(Atlas *atlas, const QRect &textureRect, const QByteArray &data, int dataLength, int dataOffset, const QSize &size)
- : QSGAtlasTexture::TextureBase(atlas, textureRect)
+ : QSGOpenGLAtlasTexture::TextureBase(atlas, textureRect)
, m_nonatlas_texture(nullptr)
, m_data(data)
, m_size(size)
diff --git a/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture_p.h b/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture_p.h
index 59e935b623..78051778f5 100644
--- a/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture_p.h
+++ b/src/quick/scenegraph/compressedtexture/qsgcompressedatlastexture_p.h
@@ -57,7 +57,7 @@
#include <QtQuick/QSGTexture>
#include <QtQuick/private/qsgareaallocator_p.h>
-#include <QtQuick/private/qsgatlastexture_p.h>
+#include <QtQuick/private/qsgopenglatlastexture_p.h>
QT_BEGIN_NAMESPACE
@@ -67,7 +67,7 @@ namespace QSGCompressedAtlasTexture {
class Texture;
-class Atlas : public QSGAtlasTexture::AtlasBase
+class Atlas : public QSGOpenGLAtlasTexture::AtlasBase
{
public:
Atlas(const QSize &size, uint format);
@@ -84,7 +84,7 @@ private:
uint m_format;
};
-class Texture : public QSGAtlasTexture::TextureBase
+class Texture : public QSGOpenGLAtlasTexture::TextureBase
{
Q_OBJECT
public:
diff --git a/src/quick/scenegraph/compressedtexture/qsgcompressedtexture.cpp b/src/quick/scenegraph/compressedtexture/qsgcompressedtexture.cpp
index d3310bc11c..1a8bddaa6e 100644
--- a/src/quick/scenegraph/compressedtexture/qsgcompressedtexture.cpp
+++ b/src/quick/scenegraph/compressedtexture/qsgcompressedtexture.cpp
@@ -44,13 +44,15 @@
#include <QOpenGLFunctions>
#include <QDebug>
#include <QtQuick/private/qquickwindow_p.h>
+#include <QtGui/private/qrhi_p.h>
QT_BEGIN_NAMESPACE
Q_LOGGING_CATEGORY(QSG_LOG_TEXTUREIO, "qt.scenegraph.textureio");
QSGCompressedTexture::QSGCompressedTexture(const QTextureFileData &texData)
- : m_textureData(texData)
+ : QSGTexture(*(new QSGCompressedTexturePrivate)),
+ m_textureData(texData)
{
m_size = m_textureData.size();
m_hasAlpha = !formatIsOpaque(m_textureData.glInternalFormat());
@@ -68,6 +70,8 @@ QSGCompressedTexture::~QSGCompressedTexture()
funcs->glDeleteTextures(1, &m_textureId);
}
#endif
+
+ delete m_texture;
}
int QSGCompressedTexture::textureId() const
@@ -85,6 +89,20 @@ int QSGCompressedTexture::textureId() const
return m_textureId;
}
+int QSGCompressedTexturePrivate::comparisonKey() const
+{
+ Q_Q(const QSGCompressedTexture);
+ // not textureId() as that would create an id when not yet done - that's not wanted here
+ if (q->m_textureId)
+ return q->m_textureId;
+
+ if (q->m_texture)
+ return int(qintptr(q->m_texture));
+
+ // two textures (and so materials) with not-yet-created texture underneath are never equal
+ return int(qintptr(q));
+}
+
QSize QSGCompressedTexture::textureSize() const
{
return m_size;
@@ -145,6 +163,165 @@ void QSGCompressedTexture::bind()
#endif // QT_CONFIG(opengl)
}
+static QPair<QRhiTexture::Format, bool> toRhiCompressedFormat(uint glinternalformat)
+{
+ switch (glinternalformat) {
+ case QOpenGLTexture::RGB_DXT1:
+ return { QRhiTexture::BC1, false };
+ case QOpenGLTexture::SRGB_DXT1:
+ return { QRhiTexture::BC1, true };
+
+ case QOpenGLTexture::RGBA_DXT3:
+ return { QRhiTexture::BC3, false };
+ case QOpenGLTexture::SRGB_Alpha_DXT3:
+ return { QRhiTexture::BC3, true };
+
+ case QOpenGLTexture::RGBA_DXT5:
+ return { QRhiTexture::BC5, false };
+ case QOpenGLTexture::SRGB_Alpha_DXT5:
+ return { QRhiTexture::BC5, true };
+
+ case QOpenGLTexture::RGB8_ETC2:
+ return { QRhiTexture::ETC2_RGB8, false };
+ case QOpenGLTexture::SRGB8_ETC2:
+ return { QRhiTexture::ETC2_RGB8, true };
+
+ case QOpenGLTexture::RGB8_PunchThrough_Alpha1_ETC2:
+ return { QRhiTexture::ETC2_RGB8A1, false };
+ case QOpenGLTexture::SRGB8_PunchThrough_Alpha1_ETC2:
+ return { QRhiTexture::ETC2_RGB8A1, true };
+
+ case QOpenGLTexture::RGBA8_ETC2_EAC:
+ return { QRhiTexture::ETC2_RGBA8, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ETC2_EAC:
+ return { QRhiTexture::ETC2_RGBA8, true };
+
+ case QOpenGLTexture::RGBA_ASTC_4x4:
+ return { QRhiTexture::ASTC_4x4, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_4x4:
+ return { QRhiTexture::ASTC_4x4, true };
+
+ case QOpenGLTexture::RGBA_ASTC_5x4:
+ return { QRhiTexture::ASTC_5x4, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_5x4:
+ return { QRhiTexture::ASTC_5x4, true };
+
+ case QOpenGLTexture::RGBA_ASTC_5x5:
+ return { QRhiTexture::ASTC_5x5, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_5x5:
+ return { QRhiTexture::ASTC_5x5, true };
+
+ case QOpenGLTexture::RGBA_ASTC_6x5:
+ return { QRhiTexture::ASTC_6x5, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_6x5:
+ return { QRhiTexture::ASTC_6x5, true };
+
+ case QOpenGLTexture::RGBA_ASTC_6x6:
+ return { QRhiTexture::ASTC_6x6, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_6x6:
+ return { QRhiTexture::ASTC_6x6, true };
+
+ case QOpenGLTexture::RGBA_ASTC_8x5:
+ return { QRhiTexture::ASTC_8x5, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_8x5:
+ return { QRhiTexture::ASTC_8x5, true };
+
+ case QOpenGLTexture::RGBA_ASTC_8x6:
+ return { QRhiTexture::ASTC_8x6, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_8x6:
+ return { QRhiTexture::ASTC_8x6, true };
+
+ case QOpenGLTexture::RGBA_ASTC_8x8:
+ return { QRhiTexture::ASTC_8x8, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_8x8:
+ return { QRhiTexture::ASTC_8x8, true };
+
+ case QOpenGLTexture::RGBA_ASTC_10x5:
+ return { QRhiTexture::ASTC_10x5, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_10x5:
+ return { QRhiTexture::ASTC_10x5, true };
+
+ case QOpenGLTexture::RGBA_ASTC_10x6:
+ return { QRhiTexture::ASTC_10x6, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_10x6:
+ return { QRhiTexture::ASTC_10x6, true };
+
+ case QOpenGLTexture::RGBA_ASTC_10x8:
+ return { QRhiTexture::ASTC_10x8, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_10x8:
+ return { QRhiTexture::ASTC_10x8, true };
+
+ case QOpenGLTexture::RGBA_ASTC_10x10:
+ return { QRhiTexture::ASTC_10x10, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_10x10:
+ return { QRhiTexture::ASTC_10x10, true };
+
+ case QOpenGLTexture::RGBA_ASTC_12x10:
+ return { QRhiTexture::ASTC_12x10, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_12x10:
+ return { QRhiTexture::ASTC_12x10, true };
+
+ case QOpenGLTexture::RGBA_ASTC_12x12:
+ return { QRhiTexture::ASTC_12x12, false };
+ case QOpenGLTexture::SRGB8_Alpha8_ASTC_12x12:
+ return { QRhiTexture::ASTC_12x12, true };
+
+ default:
+ return { QRhiTexture::UnknownFormat, false };
+ }
+}
+
+QRhiTexture *QSGCompressedTexturePrivate::rhiTexture() const
+{
+ Q_Q(const QSGCompressedTexture);
+ return q->m_texture;
+}
+
+void QSGCompressedTexturePrivate::updateRhiTexture(QRhi *rhi, QRhiResourceUpdateBatch *resourceUpdates)
+{
+ Q_Q(QSGCompressedTexture);
+ if (q->m_uploaded)
+ return;
+
+ q->m_uploaded = true; // even if fails, no point in trying again
+
+ if (!q->m_textureData.isValid()) {
+ qCDebug(QSG_LOG_TEXTUREIO, "Invalid texture data for %s", q->m_textureData.logName().constData());
+ return;
+ }
+
+ const QPair<QRhiTexture::Format, bool> fmt = toRhiCompressedFormat(q->m_textureData.glInternalFormat());
+ if (fmt.first == QRhiTexture::UnknownFormat) {
+ qWarning("Unknown compressed format 0x%x", q->m_textureData.glInternalFormat());
+ return;
+ }
+
+ QRhiTexture::Flags texFlags = 0;
+ if (fmt.second)
+ texFlags |= QRhiTexture::sRGB;
+
+ if (!rhi->isTextureFormatSupported(fmt.first, texFlags)) {
+ qWarning("Unsupported compressed format 0x%x", q->m_textureData.glInternalFormat());
+ return;
+ }
+
+ if (!q->m_texture) {
+ q->m_texture = rhi->newTexture(fmt.first, q->m_size, 1, texFlags);
+ if (!q->m_texture->build()) {
+ qWarning("Failed to create QRhiTexture for compressed data");
+ delete q->m_texture;
+ q->m_texture = nullptr;
+ return;
+ }
+ }
+
+ // only upload mip level 0 since we never do mipmapping for compressed textures (for now?)
+ resourceUpdates->uploadTexture(q->m_texture, QRhiTextureUploadEntry(0, 0,
+ { q->m_textureData.data().constData() + q->m_textureData.dataOffset(), q->m_textureData.dataLength() }));
+
+ q->m_textureData = QTextureFileData(); // Release this memory, not needed anymore
+}
+
QTextureFileData QSGCompressedTexture::textureData() const
{
return m_textureData;
diff --git a/src/quick/scenegraph/compressedtexture/qsgcompressedtexture_p.h b/src/quick/scenegraph/compressedtexture/qsgcompressedtexture_p.h
index c3b58a2389..b858d8ddee 100644
--- a/src/quick/scenegraph/compressedtexture/qsgcompressedtexture_p.h
+++ b/src/quick/scenegraph/compressedtexture/qsgcompressedtexture_p.h
@@ -52,25 +52,28 @@
//
#include <private/qtexturefiledata_p.h>
-#include <QSGTexture>
-#include <QtQuick/private/qsgcontext_p.h>
+#include <private/qsgcontext_p.h>
+#include <private/qsgtexture_p.h>
#include <QQuickTextureFactory>
#include <QOpenGLFunctions>
QT_BEGIN_NAMESPACE
+class QSGCompressedTexturePrivate;
+
class Q_QUICK_PRIVATE_EXPORT QSGCompressedTexture : public QSGTexture
{
+ Q_DECLARE_PRIVATE(QSGCompressedTexture)
Q_OBJECT
public:
QSGCompressedTexture(const QTextureFileData& texData);
virtual ~QSGCompressedTexture();
- int textureId() const override;
QSize textureSize() const override;
bool hasAlphaChannel() const override;
bool hasMipmaps() const override;
+ int textureId() const override;
void bind() override;
QTextureFileData textureData() const;
@@ -81,14 +84,24 @@ protected:
QTextureFileData m_textureData;
QSize m_size;
mutable uint m_textureId = 0;
+ QRhiTexture *m_texture = nullptr;
bool m_hasAlpha = false;
bool m_uploaded = false;
};
-namespace QSGAtlasTexture {
+namespace QSGOpenGLAtlasTexture {
class Manager;
}
+class QSGCompressedTexturePrivate : public QSGTexturePrivate
+{
+ Q_DECLARE_PUBLIC(QSGCompressedTexture)
+public:
+ int comparisonKey() const override;
+ QRhiTexture *rhiTexture() const override;
+ void updateRhiTexture(QRhi *rhi, QRhiResourceUpdateBatch *resourceUpdates) override;
+};
+
class Q_QUICK_PRIVATE_EXPORT QSGCompressedTextureFactory : public QQuickTextureFactory
{
public:
@@ -101,7 +114,7 @@ protected:
QTextureFileData m_textureData;
private:
- friend class QSGAtlasTexture::Manager;
+ friend class QSGOpenGLAtlasTexture::Manager;
};
QT_END_NAMESPACE