diff options
Diffstat (limited to 'src/Runtime/ogl-runtime/src/runtimerender/Qt3DSDistanceFieldGlyphCache.cpp')
m--------- | src/Runtime/ogl-runtime | 0 | ||||
-rw-r--r-- | src/Runtime/ogl-runtime/src/runtimerender/Qt3DSDistanceFieldGlyphCache.cpp | 518 |
2 files changed, 0 insertions, 518 deletions
diff --git a/src/Runtime/ogl-runtime b/src/Runtime/ogl-runtime new file mode 160000 +Subproject 2025912174c4cf99270b7439ec3b021e1d089ae diff --git a/src/Runtime/ogl-runtime/src/runtimerender/Qt3DSDistanceFieldGlyphCache.cpp b/src/Runtime/ogl-runtime/src/runtimerender/Qt3DSDistanceFieldGlyphCache.cpp deleted file mode 100644 index e9453350..00000000 --- a/src/Runtime/ogl-runtime/src/runtimerender/Qt3DSDistanceFieldGlyphCache.cpp +++ /dev/null @@ -1,518 +0,0 @@ -/**************************************************************************** -** -** Copyright (C) 2019 The Qt Company Ltd. -** Contact: http://www.qt.io/licensing/ -** -** This file is part of Qt 3D Studio. -** -** $QT_BEGIN_LICENSE:GPL$ -** Commercial License Usage -** Licensees holding valid commercial Qt licenses may use this file in -** accordance with the commercial license agreement provided with the -** Software or, alternatively, in accordance with the terms contained in -** a written agreement between you and The Qt Company. For licensing terms -** and conditions see https://www.qt.io/terms-conditions. For further -** information use the contact form at https://www.qt.io/contact-us. -** -** GNU General Public License Usage -** Alternatively, this file may be used under the terms of the GNU -** General Public License version 3 or (at your option) any later version -** approved by the KDE Free Qt Foundation. The licenses are as published by -** the Free Software Foundation and appearing in the file LICENSE.GPL3 -** included in the packaging of this file. Please review the following -** information to ensure the GNU General Public License requirements will -** be met: https://www.gnu.org/licenses/gpl-3.0.html. -** -** $QT_END_LICENSE$ -** -****************************************************************************/ - -#include "Qt3DSDistanceFieldGlyphCache_p.h" - -#include <QtQuick/private/qsgareaallocator_p.h> - -#include <QtCore/qmath.h> -#include <QtCore/qendian.h> -#include <QtGui/qimage.h> - -#include "render/Qt3DSRenderContext.h" -#include "render/Qt3DSRenderBaseTypes.h" -#include "Qt3DSRenderResourceManager.h" -#include "foundation/Qt3DSAllocator.h" - -#if QT_VERSION >= QT_VERSION_CHECK(5,12,2) - -QT_BEGIN_NAMESPACE - -// Should work on most hardware. Used as stop gap until Qt 3D provides a -// way to retrieve the system value -#ifndef Q3DSDISTANCEFIELDGLYPHCACHE_MAXIMUM_TEXURE_SIZE -# define Q3DSDISTANCEFIELDGLYPHCACHE_MAXIMUM_TEXURE_SIZE 2048 -#endif - -#if !defined(Q3DSDISTANCEFIELDGLYPHCACHE_PADDING) -# define Q3DSDISTANCEFIELDGLYPHCACHE_PADDING 2 -#endif - -Q3DSDistanceFieldGlyphCache::Q3DSDistanceFieldGlyphCache( - const QRawFont &font, qt3ds::render::IQt3DSRenderContext &context) - : QSGDistanceFieldGlyphCache(font) - , m_context(context) -{ - m_maxTextureSize = Q3DSDISTANCEFIELDGLYPHCACHE_MAXIMUM_TEXURE_SIZE; - - loadPregeneratedCache(font); -} - -Q3DSDistanceFieldGlyphCache::~Q3DSDistanceFieldGlyphCache() -{ - for (auto &texture : m_textures) - qt3ds::foundation::NVDelete(m_context.GetAllocator(), texture.texture); - delete m_areaAllocator; -} - -int Q3DSDistanceFieldGlyphCache::maxTextureSize() const -{ - return m_maxTextureSize; -} - -Q3DSDistanceFieldGlyphCache::TextureInfo *Q3DSDistanceFieldGlyphCache::textureInfo(int index) const -{ - while (index >= m_textures.size()) - m_textures.append(TextureInfo()); - return &m_textures[index]; -} - -void Q3DSDistanceFieldGlyphCache::referenceGlyphs(const QSet<glyph_t> &glyphs) -{ - m_unusedGlyphs -= glyphs; -} - -void Q3DSDistanceFieldGlyphCache::releaseGlyphs(const QSet<glyph_t> &glyphs) -{ - m_unusedGlyphs += glyphs; -} - -void Q3DSDistanceFieldGlyphCache::setTextureData(qt3ds::render::NVRenderTexture2D *texture, - QImage &image) -{ - bool isGLES2 = m_context.GetRenderContext().GetRenderContextType() - == qt3ds::render::NVRenderContextValues::GLES2; - texture->SetTextureData(qt3ds::render::toU8DataRef(image.bits(), image.byteCount()), - 0, image.width(), image.height(), - isGLES2 ? qt3ds::render::NVRenderTextureFormats::Alpha8 - : qt3ds::render::NVRenderTextureFormats::R8); -} - -void Q3DSDistanceFieldGlyphCache::resizeTexture(TextureInfo *info, int width, int height) -{ - QImage &image = info->copy; - if (info->texture == nullptr) { - info->texture = m_context.GetRenderContext().CreateTexture2D(); - info->texture->SetMinFilter(qt3ds::render::NVRenderTextureMinifyingOp::Enum::Linear); - info->texture->SetMagFilter(qt3ds::render::NVRenderTextureMagnifyingOp::Enum::Linear); - setTextureData(info->texture, image); - } - - qt3ds::render::STextureDetails textureDetails = info->texture->GetTextureDetails(); - if (int(textureDetails.m_Width) != width || int(textureDetails.m_Height) != height) - setTextureData(info->texture, image); - - if (info->copy.width() != width || info->copy.height() != height) { - QImage newImage(width, height, QImage::Format_Alpha8); - - for (int y = 0; y < info->copy.height(); ++y) { - uchar *dest = newImage.scanLine(y); - const uchar *src = info->copy.scanLine(y); - ::memcpy(dest, src, size_t(info->copy.width())); - } - - info->copy = newImage; - - setTextureData(info->texture, image); - } -} - -void Q3DSDistanceFieldGlyphCache::storeGlyphs(const QList<QDistanceField> &glyphs) -{ - using GlyphTextureHash = QHash<TextureInfo *, QVector<glyph_t> >; - using GlyphTextureHashConstIt = GlyphTextureHash::const_iterator; - - GlyphTextureHash glyphTextures; - for (int i = 0; i < glyphs.size(); ++i) { - QDistanceField glyph = glyphs.at(i); - glyph_t glyphIndex = glyph.glyph(); - TexCoord c = glyphTexCoord(glyphIndex); - TextureInfo *texInfo = m_glyphsTexture.value(glyphIndex); - - resizeTexture(texInfo, maxTextureSize(), texInfo->allocatedArea.height()); - - Q_ASSERT(!glyphTextures[texInfo].contains(glyphIndex)); - glyphTextures[texInfo].append(glyphIndex); - - int padding = texInfo->padding; - int expectedWidth = qCeil(c.width + c.xMargin * 2); - glyph = glyph.copy(-padding, -padding, - expectedWidth + padding * 2, glyph.height() + padding * 2); - - for (int y = 0; y < glyph.height(); ++y) { - const uchar *src = glyph.scanLine(y); - uchar *dest = texInfo->copy.scanLine(int(c.y) + y - padding) + (int(c.x) - padding); - ::memcpy(dest, src, size_t(glyph.width())); - } - } - - for (GlyphTextureHashConstIt i = glyphTextures.constBegin(), - cend = glyphTextures.constEnd(); i != cend; ++i) { - Texture t; - // 0 == empty texture, (i - 1) == index into m_textures - t.textureId = uint(i.key() - m_textures.constData()) + 1; - qt3ds::render::STextureDetails textureDetails = i.key()->texture->GetTextureDetails(); - t.size = QSize(textureDetails.m_Width, textureDetails.m_Height); - setGlyphsTexture(i.value(), t); - - QImage &image = i.key()->copy; - - setTextureData(i.key()->texture, image); - } -} - -void Q3DSDistanceFieldGlyphCache::requestGlyphs(const QSet<glyph_t> &glyphs) -{ - // Note: Most of this is copy-pasted from QSGDefaultDistanceFieldGlyphCache in Qt Quick. - // All of this can probably be shared as a default implementation, since it does not - // actually create any textures, but it might have to be either templated or based - // on void*. For now we will just live with the duplication. - - QList<GlyphPosition> glyphPositions; - QVector<glyph_t> glyphsToRender; - - if (m_areaAllocator == nullptr) { - m_areaAllocator = new QSGAreaAllocator(QSize(maxTextureSize(), - m_maxTextureCount * maxTextureSize())); - } - - for (QSet<glyph_t>::const_iterator it = glyphs.constBegin(); it != glyphs.constEnd() ; ++it) { - glyph_t glyphIndex = *it; - - int padding = Q3DSDISTANCEFIELDGLYPHCACHE_PADDING; - QRectF boundingRect = glyphData(glyphIndex).boundingRect; - int glyphWidth = qCeil(boundingRect.width()) + distanceFieldRadius() * 2; - int glyphHeight = qCeil(boundingRect.height()) + distanceFieldRadius() * 2; - QSize glyphSize(glyphWidth + padding * 2, glyphHeight + padding * 2); - QRect alloc = m_areaAllocator->allocate(glyphSize); - - if (alloc.isNull()) { - // Unallocate unused glyphs until we can allocated the new glyph - while (alloc.isNull() && !m_unusedGlyphs.isEmpty()) { - glyph_t unusedGlyph = *m_unusedGlyphs.constBegin(); - - TexCoord unusedCoord = glyphTexCoord(unusedGlyph); - QRectF unusedGlyphBoundingRect = glyphData(unusedGlyph).boundingRect; - int unusedGlyphWidth = qCeil(unusedGlyphBoundingRect.width()) - + distanceFieldRadius() * 2; - int unusedGlyphHeight = qCeil(unusedGlyphBoundingRect.height()) - + distanceFieldRadius() * 2; - m_areaAllocator->deallocate(QRect(int(unusedCoord.x) - padding, - int(unusedCoord.y) - padding, - padding * 2 + unusedGlyphWidth, - padding * 2 + unusedGlyphHeight)); - - m_unusedGlyphs.remove(unusedGlyph); - m_glyphsTexture.remove(unusedGlyph); - removeGlyph(unusedGlyph); - - alloc = m_areaAllocator->allocate(glyphSize); - } - - // Not enough space left for this glyph... skip to the next one - if (alloc.isNull()) - continue; - } - - TextureInfo *tex = textureInfo(alloc.y() / maxTextureSize()); - alloc = QRect(alloc.x(), alloc.y() % maxTextureSize(), alloc.width(), alloc.height()); - - tex->allocatedArea |= alloc; - Q_ASSERT(tex->padding == padding || tex->padding < 0); - tex->padding = padding; - - GlyphPosition p; - p.glyph = glyphIndex; - p.position = alloc.topLeft() + QPoint(padding, padding); - - glyphPositions.append(p); - glyphsToRender.append(glyphIndex); - m_glyphsTexture.insert(glyphIndex, tex); - } - - setGlyphsPosition(glyphPositions); - markGlyphsToRender(glyphsToRender); -} - -void Q3DSDistanceFieldGlyphCache::processPendingGlyphs() -{ - update(); -} - -Q3DSDistanceFieldGlyphCache::TextureInfo *Q3DSDistanceFieldGlyphCache::textureInfoById( - uint id) const -{ - Q_ASSERT(id > 0); - return textureInfo(id - 1); -} - -// This is all copy-pasted from Qt Quick, as sharing it would require some refactoring, and we -// need to work with Qt 5.12.2 at the moment. -namespace { - struct Qtdf { - // We need these structs to be tightly packed, but some compilers we use do not - // support #pragma pack(1), so we need to hardcode the offsets/sizes in the - // file format - enum TableSize { - HeaderSize = 14, - GlyphRecordSize = 46, - TextureRecordSize = 17 - }; - - enum Offset { - // Header - majorVersion = 0, - minorVersion = 1, - pixelSize = 2, - textureSize = 4, - flags = 8, - headerPadding = 9, - numGlyphs = 10, - - // Glyph record - glyphIndex = 0, - textureOffsetX = 4, - textureOffsetY = 8, - textureWidth = 12, - textureHeight = 16, - xMargin = 20, - yMargin = 24, - boundingRectX = 28, - boundingRectY = 32, - boundingRectWidth = 36, - boundingRectHeight = 40, - textureIndex = 44, - - // Texture record - allocatedX = 0, - allocatedY = 4, - allocatedWidth = 8, - allocatedHeight = 12, - texturePadding = 16 - - }; - - template <typename T> - static inline T fetch(const char *data, Offset offset) - { - return qFromBigEndian<T>(data + int(offset)); - } - }; -} - -qreal Q3DSDistanceFieldGlyphCache::fontSize() const -{ - return QT_DISTANCEFIELD_BASEFONTSIZE(m_doubleGlyphResolution); -} - -bool Q3DSDistanceFieldGlyphCache::loadPregeneratedCache(const QRawFont &font) -{ - // The pregenerated data must be loaded first, otherwise the area allocator - // will be wrong - if (m_areaAllocator != nullptr) { - qWarning("Font cache must be loaded before cache is used"); - return false; - } - - QByteArray qtdfTable = font.fontTable("qtdf"); - if (qtdfTable.isEmpty()) - return false; - - using GlyphTextureHash = QHash<TextureInfo *, QVector<glyph_t> >; - - GlyphTextureHash glyphTextures; - - if (uint(qtdfTable.size()) < Qtdf::HeaderSize) { - qWarning("Invalid qtdf table in font '%s'", - qPrintable(font.familyName())); - return false; - } - - const char *qtdfTableStart = qtdfTable.constData(); - const char *qtdfTableEnd = qtdfTableStart + qtdfTable.size(); - - int padding = 0; - int textureCount = 0; - { - quint8 majorVersion = Qtdf::fetch<quint8>(qtdfTableStart, Qtdf::majorVersion); - quint8 minorVersion = Qtdf::fetch<quint8>(qtdfTableStart, Qtdf::minorVersion); - if (majorVersion != 5 || minorVersion != 12) { - qWarning("Invalid version of qtdf table %d.%d in font '%s'", - majorVersion, - minorVersion, - qPrintable(font.familyName())); - return false; - } - - qreal pixelSize = qreal(Qtdf::fetch<quint16>(qtdfTableStart, Qtdf::pixelSize)); - m_maxTextureSize = int(Qtdf::fetch<quint32>(qtdfTableStart, Qtdf::textureSize)); - m_doubleGlyphResolution = Qtdf::fetch<quint8>(qtdfTableStart, Qtdf::flags) == 1; - padding = Qtdf::fetch<quint8>(qtdfTableStart, Qtdf::headerPadding); - - if (pixelSize <= 0.0) { - qWarning("Invalid pixel size in '%s'", qPrintable(font.familyName())); - return false; - } - - if (m_maxTextureSize <= 0) { - qWarning("Invalid texture size in '%s'", qPrintable(font.familyName())); - return false; - } - - if (padding != Q3DSDISTANCEFIELDGLYPHCACHE_PADDING) { - qWarning("Padding mismatch in '%s'. Font requires %d, but Qt is compiled with %d.", - qPrintable(font.familyName()), - padding, - Q3DSDISTANCEFIELDGLYPHCACHE_PADDING); - } - - m_referenceFont.setPixelSize(pixelSize); - - quint32 glyphCount = Qtdf::fetch<quint32>(qtdfTableStart, Qtdf::numGlyphs); - m_unusedGlyphs.reserve(int(glyphCount)); - - const char *allocatorData = qtdfTableStart + Qtdf::HeaderSize; - { - m_areaAllocator = new QSGAreaAllocator(QSize(0, 0)); - allocatorData = m_areaAllocator->deserialize(allocatorData, - qtdfTableEnd - allocatorData); - if (allocatorData == nullptr) - return false; - } - - if (m_areaAllocator->size().height() % m_maxTextureSize != 0) { - qWarning("Area allocator size mismatch in '%s'", qPrintable(font.familyName())); - return false; - } - - textureCount = m_areaAllocator->size().height() / m_maxTextureSize; - m_maxTextureCount = qMax(m_maxTextureCount, textureCount); - - const char *textureRecord = allocatorData; - for (int i = 0; i < textureCount; ++i, textureRecord += Qtdf::TextureRecordSize) { - if (textureRecord + Qtdf::TextureRecordSize > qtdfTableEnd) { - qWarning("qtdf table too small in font '%s'.", - qPrintable(font.familyName())); - return false; - } - - TextureInfo *tex = textureInfo(i); - tex->allocatedArea.setX(int(Qtdf::fetch<quint32>(textureRecord, Qtdf::allocatedX))); - tex->allocatedArea.setY(int(Qtdf::fetch<quint32>(textureRecord, Qtdf::allocatedY))); - tex->allocatedArea.setWidth(int(Qtdf::fetch<quint32>(textureRecord, - Qtdf::allocatedWidth))); - tex->allocatedArea.setHeight(int(Qtdf::fetch<quint32>(textureRecord, - Qtdf::allocatedHeight))); - tex->padding = Qtdf::fetch<quint8>(textureRecord, Qtdf::texturePadding); - } - - const char *glyphRecord = textureRecord; - for (quint32 i = 0; i < glyphCount; ++i, glyphRecord += Qtdf::GlyphRecordSize) { - if (glyphRecord + Qtdf::GlyphRecordSize > qtdfTableEnd) { - qWarning("qtdf table too small in font '%s'.", - qPrintable(font.familyName())); - return false; - } - - glyph_t glyph = Qtdf::fetch<quint32>(glyphRecord, Qtdf::glyphIndex); - m_unusedGlyphs.insert(glyph); - - GlyphData &glyphData = emptyData(glyph); - -#define FROM_FIXED_POINT(value) (qreal(value)/qreal(65536)) - - glyphData.texCoord.x - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::textureOffsetX)); - glyphData.texCoord.y - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::textureOffsetY)); - glyphData.texCoord.width - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::textureWidth)); - glyphData.texCoord.height - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::textureHeight)); - glyphData.texCoord.xMargin - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::xMargin)); - glyphData.texCoord.yMargin - = FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, Qtdf::yMargin)); - glyphData.boundingRect.setX( - FROM_FIXED_POINT(Qtdf::fetch<qint32>(glyphRecord, Qtdf::boundingRectX))); - glyphData.boundingRect.setY( - FROM_FIXED_POINT(Qtdf::fetch<qint32>(glyphRecord, Qtdf::boundingRectY))); - glyphData.boundingRect.setWidth( - FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, - Qtdf::boundingRectWidth))); - glyphData.boundingRect.setHeight( - FROM_FIXED_POINT(Qtdf::fetch<quint32>(glyphRecord, - Qtdf::boundingRectHeight))); - -#undef FROM_FIXED_POINT - - int textureIndex = Qtdf::fetch<quint16>(glyphRecord, Qtdf::textureIndex); - if (textureIndex < 0 || textureIndex >= textureCount) { - qWarning("Invalid texture index %d (texture count == %d) in '%s'", - textureIndex, - textureCount, - qPrintable(font.familyName())); - return false; - } - - - TextureInfo *texInfo = textureInfo(textureIndex); - m_glyphsTexture.insert(glyph, texInfo); - - glyphTextures[texInfo].append(glyph); - } - - const uchar *textureData = reinterpret_cast<const uchar *>(glyphRecord); - for (int i = 0; i < textureCount; ++i) { - - TextureInfo *texInfo = textureInfo(i); - - int width = texInfo->allocatedArea.width(); - int height = texInfo->allocatedArea.height(); - qint64 size = width * height; - if (reinterpret_cast<const char *>(textureData + size) > qtdfTableEnd) { - qWarning("qtdf table too small in font '%s'.", - qPrintable(font.familyName())); - return false; - } - - resizeTexture(texInfo, width, height); - - memcpy(texInfo->copy.bits(), textureData, size); - textureData += size; - - QImage &image = texInfo->copy; - setTextureData(texInfo->texture, image); - - QVector<glyph_t> glyphs = glyphTextures.value(texInfo); - - Texture t; - t.textureId = uint(i + 1); - t.size = texInfo->copy.size(); - - setGlyphsTexture(glyphs, t); - } - } - - return true; -} - -QT_END_NAMESPACE - -#endif // QT_VERSION >= QT_VERSION_CHECK(5,12,2) |