6#include <QtQuick3DRuntimeRender/private/qssgrenderloadedtexture_p.h>
7#include <QtQuick3DRuntimeRender/private/qssgrendererutil_p.h>
8#include <QtQuick3DRuntimeRender/private/qssgruntimerenderlogging_p.h>
9#include <QtQuick3DRuntimeRender/private/qssgrendertexturedata_p.h>
10#include <QtGui/QImageReader>
11#include <QtGui/QColorSpace>
14#include <QtQuick3DUtils/private/qssgutils_p.h>
15#include <QtQuick3DUtils/private/qssgassert_p.h>
17#include <private/qtexturefilereader_p.h>
19#define TINYEXR_IMPLEMENTATION
20#define TINYEXR_USE_MINIZ 0
21#define TINYEXR_USE_THREAD 1
24QT_WARNING_DISABLE_CLANG(
"-Wunused-parameter")
25QT_WARNING_DISABLE_CLANG(
"-Wunused-function")
26QT_WARNING_DISABLE_GCC(
"-Wunused-parameter")
27QT_WARNING_DISABLE_GCC(
"-Wunused-function")
35QSharedPointer<QIODevice> QSSGInputUtil::getStreamForFile(
const QString &inPath,
bool inQuiet, QString *outPath)
37 QFile *file =
nullptr;
38 QString tryPath = inPath;
39 if (tryPath.startsWith(QLatin1String(
"qrc:/"))) {
40 tryPath = inPath.mid(3);
41 }
else if (tryPath.startsWith(QLatin1String(
"file://"))) {
42 tryPath = inPath.mid(7);
45 QFileInfo fi(tryPath);
46 bool found = fi.exists();
47 if (!found && fi.isNativePath()) {
48 tryPath.prepend(QLatin1String(
":/"));
53 QString filePath = fi.canonicalFilePath();
54 file =
new QFile(filePath);
55 if (file->open(QIODevice::ReadOnly)) {
63 if (!file && !inQuiet)
64 qCWarning(WARNING,
"Failed to find file: %s", qPrintable(inPath));
65 return QSharedPointer<QIODevice>(file);
68QSharedPointer<QIODevice> QSSGInputUtil::getStreamForTextureFile(
const QString &inPath,
bool inQuiet,
69 QString *outPath, FileType *outFileType)
71 static const QList<QByteArray> hdrFormats = QList<QByteArray>({
"hdr",
"exr" });
72 static const QList<QByteArray> textureFormats = QTextureFileReader::supportedFileFormats();
73 static const QList<QByteArray> imageFormats = QImageReader::supportedImageFormats();
74 static const QList<QByteArray> allFormats = textureFormats + hdrFormats + imageFormats;
78 QSharedPointer<QIODevice> stream = getStreamForFile(inPath,
true, &filePath);
80 ext = QFileInfo(filePath).suffix().toLatin1().toLower();
82 for (
const QByteArray &format : allFormats) {
83 QString tryName = inPath + QLatin1Char(
'.') + QLatin1String(format);
84 stream = getStreamForFile(tryName,
true, &filePath);
95 FileType type = UnknownFile;
96 if (hdrFormats.contains(ext))
98 else if (textureFormats.contains(ext))
100 else if (imageFormats.contains(ext))
104 }
else if (!inQuiet) {
105 qCWarning(WARNING,
"Failed to find texture file for: %s", qPrintable(inPath));
112 switch (internalFormat) {
114 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R8);
116 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R16);
118 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R16F);
120 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R32I);
122 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R32UI);
124 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R32F);
126 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RG8);
128 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA8);
130 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB8);
132 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8);
134 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8A8);
136 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB565);
138 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Alpha8);
140 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Luminance8);
142 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Luminance16);
144 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::LuminanceAlpha8);
146 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA16F);
148 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RG16F);
150 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RG32F);
152 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB32F);
154 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA32F);
156 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R11G11B10);
158 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB9E5);
160 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB10_A2);
162 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB16F);
164 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA32UI);
166 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB32UI);
168 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA16UI);
170 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB16UI);
172 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA8UI);
174 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB8UI);
176 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA32I);
178 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB32I);
180 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA16I);
182 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB16I);
184 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA8I);
186 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB8I);
188 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_DXT1);
190 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB_DXT1);
192 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_DXT3);
194 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_DXT5);
196 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R11_EAC_UNorm);
198 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::R11_EAC_SNorm);
200 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RG11_EAC_UNorm);
202 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RG11_EAC_SNorm);
204 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB8_ETC2);
206 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_ETC2);
208 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGB8_PunchThrough_Alpha1_ETC2);
210 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_PunchThrough_Alpha1_ETC2);
212 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA8_ETC2_EAC);
214 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ETC2_EAC);
216 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_4x4);
218 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_5x4);
220 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_5x5);
222 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_6x5);
224 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_6x6);
226 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_8x5);
228 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_8x6);
230 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_8x8);
232 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_10x5);
234 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_10x6);
236 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_10x8);
238 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_10x10);
240 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_12x10);
242 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::RGBA_ASTC_12x12);
244 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_4x4);
246 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_5x4);
248 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_5x5);
250 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_6x5);
252 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_6x6);
254 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_8x5);
256 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_8x6);
258 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_8x8);
260 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_10x5);
262 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_10x6);
264 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_10x8);
266 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_10x10);
268 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_12x10);
270 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_12x12);
272 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Depth16);
274 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Depth24);
276 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Depth32);
278 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Depth24Stencil8);
280 return QSSGRenderTextureFormat(QSSGRenderTextureFormat::Unknown);
286 QImage image(inPath);
289 const QPixelFormat pixFormat = image.pixelFormat();
290 QImage::Format targetFormat = QImage::Format_RGBA8888_Premultiplied;
291 if (image.colorCount())
292 targetFormat = QImage::Format_RGBA8888;
293 else if (pixFormat.channelCount() == 1)
294 targetFormat = QImage::Format_Grayscale8;
295 else if (pixFormat.alphaUsage() == QPixelFormat::IgnoresAlpha)
296 targetFormat = QImage::Format_RGBX8888;
297 else if (pixFormat.premultiplied() == QPixelFormat::NotPremultiplied)
298 targetFormat = QImage::Format_RGBA8888;
300 image.convertTo(targetFormat);
306QSSGLoadedTexture *QSSGLoadedTexture::loadQImage(
const QString &inPath, qint32 flipVertical)
308 QImage image = loadImage(inPath, flipVertical);
311 QSSGLoadedTexture *retval =
new QSSGLoadedTexture;
312 retval->width = image.width();
313 retval->height = image.height();
314 retval->components = image.pixelFormat().channelCount();
315 retval->image = image;
316 retval->data = (
void *)retval->image.bits();
317 retval->dataSizeInBytes = image.sizeInBytes();
318 retval->setFormatFromComponents();
320 retval->isSRGB = image.colorSpace().transferFunction() != QColorSpace::TransferFunction::Linear;
325QSSGLoadedTexture *QSSGLoadedTexture::loadCompressedImage(
const QString &inPath)
327 QSSGLoadedTexture *retval =
nullptr;
330 QFile imageFile(inPath);
331 if (!imageFile.open(QIODevice::ReadOnly)) {
332 qWarning() <<
"Could not open image file: " << inPath;
335 auto reader =
new QTextureFileReader(&imageFile, inPath);
337 if (!reader->canRead()) {
338 qWarning() <<
"Unable to read image file: " << inPath;
342 retval =
new QSSGLoadedTexture;
343 retval->textureFileData = reader->read();
346 retval->width = retval->textureFileData.size().width();
347 retval->height = retval->textureFileData.size().height();
348 auto glFormat = retval->textureFileData.glInternalFormat()
349 ? retval->textureFileData.glInternalFormat()
350 : retval->textureFileData.glFormat();
351 retval->format = fromGLtoTextureFormat(glFormat);
354 if ((retval->format.format >= QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_4x4) &&
355 (retval->format.format <= QSSGRenderTextureFormat::SRGB8_Alpha8_ASTC_12x12))
356 retval->isSRGB =
true;
366typedef unsigned char RGBE[4];
373#define MAXELEN 0x7fff
377inline int calculateLine(
int width,
int bitdepth) {
return ((width * bitdepth) + 7) / 8; }
379inline int calculatePitch(
int line) {
return (line + 3) & ~3; }
381float convertComponent(
int exponent,
int val)
383 float v = val / (256.0f);
384 float d = powf(2.0f, (
float)exponent - 128.0f);
388void decrunchScanline(
const char *&p,
const char *pEnd, RGBE *scanline,
int w)
390 scanline[0][
R] = *p++;
391 scanline[0][
G] = *p++;
392 scanline[0][
B] = *p++;
393 scanline[0][
E] = *p++;
395 if (scanline[0][
R] == 2 && scanline[0][
G] == 2 && scanline[0][
B] < 128) {
397 for (
int channel = 0; channel < 4; ++channel) {
398 for (
int x = 0; x < w && p < pEnd; ) {
399 unsigned char c = *p++;
402 int repCount = c & 127;
405 scanline[x++][channel] = c;
408 while (c-- && p < pEnd)
409 scanline[x++][channel] = *p++;
418 while (x < w && pEnd - p >= 4) {
419 scanline[x][
R] = *p++;
420 scanline[x][
G] = *p++;
421 scanline[x][
B] = *p++;
422 scanline[x][
E] = *p++;
424 if (scanline[x][
R] == 1 && scanline[x][
G] == 1 && scanline[x][
B] == 1) {
425 int repCount = scanline[x][3] << bitshift;
427 memcpy(scanline[x], scanline[x - 1], 4);
439void decodeScanlineToTexture(RGBE *scanline,
int width,
void *outBuf, quint32 offset, QSSGRenderTextureFormat inFormat)
441 quint8 *target =
reinterpret_cast<quint8 *>(outBuf);
444 if (inFormat == QSSGRenderTextureFormat::RGBE8) {
445 memcpy(target, scanline, size_t(width) * 4);
448 for (
int i = 0; i < width; ++i) {
449 rgbaF32[
R] = convertComponent(scanline[i][
E], scanline[i][
R]);
450 rgbaF32[
G] = convertComponent(scanline[i][
E], scanline[i][
G]);
451 rgbaF32[
B] = convertComponent(scanline[i][
E], scanline[i][
B]);
454 inFormat.encodeToPixel(rgbaF32, target, i * inFormat.getSizeofFormat());
459QSSGLoadedTexture *loadRadianceHdr(
const QSharedPointer<QIODevice> &source,
const QSSGRenderTextureFormat &format)
461 QSSGLoadedTexture *imageData =
nullptr;
465 source->read(sig, 11);
466 if (!strncmp(sig,
"#?RADIANCE\n", 11)) {
467 QByteArray buf = source->readAll();
468 const char *p = buf.constData();
469 const char *pEnd = p + buf.size();
478 if (line.startsWith(QByteArrayLiteral(
"FORMAT="))) {
479 const QByteArray format = line.mid(7).trimmed();
480 if (format != QByteArrayLiteral(
"32-bit_rle_rgbe")) {
481 qWarning(
"HDR format '%s' is not supported", format.constData());
491 qWarning(
"Malformed HDR image data at property strings");
503 qWarning(
"Malformed HDR image data at resolution string");
511 if (!sscanf_s(line.constData(),
"-Y %d +X %d", &height, &width)) {
513 if (!sscanf(line.constData(),
"-Y %d +X %d", &height, &width)) {
515 qWarning(
"Unsupported HDR resolution string '%s'", line.constData());
518 if (width <= 0 || height <= 0) {
519 qWarning(
"Invalid HDR resolution");
523 const int bytesPerPixel = format.getSizeofFormat();
524 const int bitCount = bytesPerPixel * 8;
525 const int pitch = calculatePitch(calculateLine(width, bitCount));
526 const size_t dataSize = height * pitch;
527 QSSG_CHECK_X(dataSize <= std::numeric_limits<quint32>::max(),
"Requested data size exceeds 4GB limit!");
528 imageData =
new QSSGLoadedTexture;
529 imageData->dataSizeInBytes = quint32(dataSize);
530 imageData->data = ::malloc(dataSize);
531 imageData->width = width;
532 imageData->height = height;
533 imageData->format = format;
534 imageData->components = format.getNumberOfComponent();
535 imageData->isSRGB =
false;
538 RGBE *scanline =
new RGBE[width];
542 for (
int y = 0; y < height; ++y) {
543 quint32 byteOffset = quint32((height - 1 - y) * width * bytesPerPixel);
545 qWarning(
"Unexpected end of HDR data");
549 decrunchScanline(p, pEnd, scanline, width);
550 decodeScanlineToTexture(scanline, width, imageData->data, byteOffset, format);
559QSSGLoadedTexture *loadExr(
const QSharedPointer<QIODevice> &source,
const QSSGRenderTextureFormat format)
561 QSSGLoadedTexture *imageData =
nullptr;
563 char versionBuffer[tinyexr::kEXRVersionSize];
565 auto size = source->read(versionBuffer, tinyexr::kEXRVersionSize);
567 if (size != tinyexr::kEXRVersionSize)
570 EXRVersion exrVersion;
571 if (ParseEXRVersionFromMemory(&exrVersion,
reinterpret_cast<
unsigned char *>(versionBuffer), tinyexr::kEXRVersionSize) != TINYEXR_SUCCESS)
575 if (exrVersion.multipart)
580 QByteArray buf = source->readAll();
581 const char *err =
nullptr;
584 InitEXRHeader(&exrHeader);
585 if (ParseEXRHeaderFromMemory(&exrHeader,
587 reinterpret_cast<
const unsigned char *>(buf.constData()),
589 &err) != TINYEXR_SUCCESS) {
590 qWarning(
"Failed to parse EXR Header with error: '%s'", err);
591 FreeEXRErrorMessage(err);
596 for (
int i = 0; i < exrHeader.num_channels; i++) {
597 if (exrHeader.pixel_types[i] == TINYEXR_PIXELTYPE_HALF)
598 exrHeader.requested_pixel_types[i] = TINYEXR_PIXELTYPE_FLOAT;
604 InitEXRImage(&exrImage);
605 if (LoadEXRImageFromMemory(&exrImage,
607 reinterpret_cast<
const unsigned char *>(buf.constData()),
609 &err) != TINYEXR_SUCCESS) {
610 qWarning(
"Failed to load EXR Image with error: '%s'", err);
611 FreeEXRHeader(&exrHeader);
612 FreeEXRErrorMessage(err);
617 const int bytesPerPixel = format.getSizeofFormat();
618 const int bitCount = bytesPerPixel * 8;
619 const int pitch = calculatePitch(calculateLine(exrImage.width, bitCount));
620 const size_t dataSize = exrImage.height * pitch;
621 QSSG_CHECK_X(dataSize <= std::numeric_limits<quint32>::max(),
"Requested data size exceeds 4GB limit!");
622 imageData =
new QSSGLoadedTexture;
623 imageData->dataSizeInBytes = quint32(dataSize);
624 imageData->data = ::malloc(imageData->dataSizeInBytes);
625 imageData->width = exrImage.width;
626 imageData->height = exrImage.height;
627 imageData->format = format;
628 imageData->components = format.getNumberOfComponent();
629 imageData->isSRGB =
false;
631 quint8 *target =
reinterpret_cast<quint8 *>(imageData->data);
639 for (
int c = 0; c < exrHeader.num_channels; c++) {
640 if (strcmp(exrHeader.channels[c].name,
"R") == 0)
642 else if (strcmp(exrHeader.channels[c].name,
"G") == 0)
644 else if (strcmp(exrHeader.channels[c].name,
"B") == 0)
646 else if (strcmp(exrHeader.channels[c].name,
"A") == 0)
649 const bool isSingleChannel = exrHeader.num_channels == 1;
652 if (exrHeader.tiled) {
653 for (
int it = 0; it < exrImage.num_tiles; it++) {
654 for (
int j = 0; j < exrHeader.tile_size_y; j++)
655 for (
int i = 0; i < exrHeader.tile_size_x; i++) {
657 exrImage.tiles[it].offset_x * exrHeader.tile_size_x + i;
659 exrImage.tiles[it].offset_y * exrHeader.tile_size_y + j;
660 const int inverseJJ = std::abs(jj - (exrImage.height - 1));
661 const int idx = ii + inverseJJ * exrImage.width;
664 if (ii >= exrImage.width) {
667 if (jj >= exrImage.height) {
670 const int srcIdx = i + j * exrHeader.tile_size_x;
671 unsigned char **src = exrImage.tiles[it].images;
672 if (isSingleChannel) {
673 rgbaF32[
R] =
reinterpret_cast<
float **>(src)[0][srcIdx];
674 rgbaF32[
G] = rgbaF32[
R];
675 rgbaF32[
B] = rgbaF32[
R];
676 rgbaF32[3] = rgbaF32[
R];
678 rgbaF32[
R] =
reinterpret_cast<
float **>(src)[idxR][srcIdx];
679 rgbaF32[
G] =
reinterpret_cast<
float **>(src)[idxG][srcIdx];
680 rgbaF32[
B] =
reinterpret_cast<
float **>(src)[idxB][srcIdx];
682 rgbaF32[3] =
reinterpret_cast<
float **>(src)[idxA][srcIdx];
686 format.encodeToPixel(rgbaF32, target, idx * bytesPerPixel);
691 for (
int y = exrImage.height - 1; y >= 0; --y) {
692 for (
int x = 0; x < exrImage.width; x++) {
693 const int i = y * exrImage.width + x;
694 if (isSingleChannel) {
695 rgbaF32[
R] =
reinterpret_cast<
float **>(exrImage.images)[0][i];
696 rgbaF32[
G] = rgbaF32[
R];
697 rgbaF32[
B] = rgbaF32[
R];
698 rgbaF32[3] = rgbaF32[
R];
700 rgbaF32[
R] =
reinterpret_cast<
float **>(exrImage.images)[idxR][i];
701 rgbaF32[
G] =
reinterpret_cast<
float **>(exrImage.images)[idxG][i];
702 rgbaF32[
B] =
reinterpret_cast<
float **>(exrImage.images)[idxB][i];
704 rgbaF32[3] =
reinterpret_cast<
float **>(exrImage.images)[idxA][i];
708 format.encodeToPixel(rgbaF32, target, idx * bytesPerPixel);
715 FreeEXRImage(&exrImage);
716 FreeEXRHeader(&exrHeader);
723QSSGLoadedTexture *QSSGLoadedTexture::loadLightmapImage(
const QString &inPath,
const QSSGRenderTextureFormat &inFormat,
const QString &key)
725 if (
auto stream = QSSGInputUtil::getStreamForFile(inPath))
726 return QSSGLightmapLoader::createTexture(stream, inFormat, key);
730QSSGLoadedTexture *QSSGLoadedTexture::loadHdrImage(
const QSharedPointer<QIODevice> &source,
const QSSGRenderTextureFormat &inFormat)
732 QSSGLoadedTexture *imageData =
nullptr;
734 QSSGRenderTextureFormat format = inFormat;
735 if (format.format == QSSGRenderTextureFormat::Unknown) {
740 format = QSSGRenderTextureFormat::RGBA16F;
744 imageData = loadRadianceHdr(source, format);
748 imageData = loadExr(source, format);
753QSSGLoadedTexture *QSSGLoadedTexture::loadTextureData(QSSGRenderTextureData *textureData)
755 QSSGLoadedTexture *imageData =
new QSSGLoadedTexture;
757 if (!textureData->format().isCompressedTextureFormat()) {
758 const int bytesPerPixel = textureData->format().getSizeofFormat();
759 const int bitCount = bytesPerPixel * 8;
760 const int pitch = calculatePitch(calculateLine(textureData->size().width(), bitCount));
761 size_t dataSize = size_t(textureData->size().height()) * pitch;
762 if (textureData->depth() > 0)
763 dataSize *= textureData->depth();
764 QSSG_CHECK_X(dataSize <= std::numeric_limits<quint32>::max(),
"Requested data size exceeds 4GB limit!");
765 imageData->dataSizeInBytes = quint32(dataSize);
767 imageData->data =
const_cast<
void*>(
reinterpret_cast<
const void*>(textureData->textureData().data()));
768 imageData->width = textureData->size().width();
769 imageData->height = textureData->size().height();
770 imageData->depth = textureData->depth();
771 imageData->format = textureData->format();
772 imageData->components = textureData->format().getNumberOfComponent();
776 imageData->data =
const_cast<
void*>(
reinterpret_cast<
const void*>(textureData->textureData().data()));
777 const size_t dataSize = textureData->textureData().size();
778 QSSG_CHECK_X(dataSize <= std::numeric_limits<quint32>::max(),
"Requested data size exceeds 4GB limit!");
779 imageData->dataSizeInBytes = quint32(dataSize);
783 imageData->width = textureData->size().width();
784 imageData->height = textureData->size().height();
785 imageData->format = textureData->format();
792 if (imageData->format == QSSGRenderTextureFormat::RGBE8 ||
793 imageData->format == QSSGRenderTextureFormat::RGBA16F ||
794 imageData->format == QSSGRenderTextureFormat::RGBA32F ||
795 imageData->format == QSSGRenderTextureFormat::BC6H)
796 imageData->isSRGB =
false;
798 imageData->isSRGB =
true;
805bool scanImageForAlpha(
const void *inData, quint32 inWidth, quint32 inHeight, quint32 inPixelSizeInBytes, quint8 inAlphaSizeInBits)
807 const quint8 *rowPtr =
reinterpret_cast<
const quint8 *>(inData);
808 bool hasAlpha =
false;
809 if (inAlphaSizeInBits == 0)
811 if (inPixelSizeInBytes != 2 && inPixelSizeInBytes != 4) {
815 if (inAlphaSizeInBits > 8) {
820 quint32 alphaRightShift = inPixelSizeInBytes * 8 - inAlphaSizeInBits;
821 quint32 maxAlphaValue = (1 << inAlphaSizeInBits) - 1;
823 for (quint32 rowIdx = 0; rowIdx < inHeight && !hasAlpha; ++rowIdx) {
824 for (quint32 idx = 0; idx < inWidth && !hasAlpha; ++idx, rowPtr += inPixelSizeInBytes) {
825 quint32 pixelValue = 0;
826 if (inPixelSizeInBytes == 2)
827 pixelValue = *(
reinterpret_cast<
const quint16 *>(rowPtr));
829 pixelValue = *(
reinterpret_cast<
const quint32 *>(rowPtr));
830 pixelValue = pixelValue >> alphaRightShift;
831 if (pixelValue < maxAlphaValue)
839QSSGLoadedTexture::~QSSGLoadedTexture()
841 if (data && image.sizeInBytes() <= 0 && ownsData)
845bool QSSGLoadedTexture::scanForTransparency()
const
847 switch (format.format) {
848 case QSSGRenderTextureFormat::SRGB8A8:
849 case QSSGRenderTextureFormat::RGBA8:
853 return scanImageForAlpha(data, width, height, 4, 8);
855 case QSSGRenderTextureFormat::SRGB8:
856 case QSSGRenderTextureFormat::RGB8:
857 case QSSGRenderTextureFormat::RGBE8:
859 case QSSGRenderTextureFormat::RGB565:
861 case QSSGRenderTextureFormat::RGBA5551:
865 return scanImageForAlpha(data, width, height, 2, 1);
867 case QSSGRenderTextureFormat::Alpha8:
869 case QSSGRenderTextureFormat::R8:
870 case QSSGRenderTextureFormat::Luminance8:
871 case QSSGRenderTextureFormat::RG8:
873 case QSSGRenderTextureFormat::LuminanceAlpha8:
877 return scanImageForAlpha(data, width, height, 2, 8);
878 case QSSGRenderTextureFormat::RGB_DXT1:
880 case QSSGRenderTextureFormat::RGBA_DXT3:
881 case QSSGRenderTextureFormat::RGBA_DXT1:
882 case QSSGRenderTextureFormat::RGBA_DXT5:
884 case QSSGRenderTextureFormat::RGB9E5:
886 case QSSGRenderTextureFormat::RG32F:
887 case QSSGRenderTextureFormat::RGB32F:
888 case QSSGRenderTextureFormat::RGBA16F:
889 case QSSGRenderTextureFormat::RGBA32F:
903 if (img1.size() != img2.size())
905 if (img1.pixelFormat().channelCount() != img2.pixelFormat().channelCount())
913 QStringList fileNames;
914 if (inPath.contains(QStringLiteral(
"%p"))) {
915 fileNames.reserve(6);
916 const char *faces[6] = {
"posx",
"negx",
"posy",
"negy",
"posz",
"negz" };
917 for (
const auto face : faces) {
918 QString fileName = inPath;
919 fileName.replace(QStringLiteral(
"%p"), QLatin1StringView(face));
920 fileNames << fileName;
923 }
else if (inPath.contains(QStringLiteral(
";"))) {
924 fileNames = inPath.split(QChar(u';'));
926 if (fileNames.size() != 6)
928 std::unique_ptr<QTextureFileData> textureFileData = std::make_unique<QTextureFileData>(QTextureFileData::ImageMode);
929 textureFileData->setNumFaces(6);
930 textureFileData->setNumLevels(1);
931 textureFileData->setLogName(inPath.toUtf8());
933 for (
int i = 0; i < 6; ++i) {
934 QString searchName = fileNames[i];
936 auto stream = QSSGInputUtil::getStreamForFile(searchName,
true, &filePath);
940 QImage face = loadImage(filePath, !flipY);
941 if (face.isNull() || (!prevImage.isNull() && !isCompatible(prevImage, face))) {
944 textureFileData->setData(face, 0, i);
945 textureFileData->setSize(face.size());
949 QSSGLoadedTexture *retval =
new QSSGLoadedTexture;
951 retval->textureFileData = *textureFileData;
953 retval->width = prevImage.width();
954 retval->height = prevImage.height();
955 retval->components = prevImage.pixelFormat().channelCount();
956 retval->image = prevImage;
957 retval->data = (
void *)retval->image.bits();
958 const size_t dataSize = prevImage.sizeInBytes();
959 QSSG_CHECK_X(dataSize <= std::numeric_limits<quint32>::max(),
"Requested data size exceeds 4GB limit!");
960 retval->dataSizeInBytes = quint32(dataSize);
961 retval->setFormatFromComponents();
963 retval->isSRGB = prevImage.colorSpace().transferFunction() != QColorSpace::TransferFunction::Linear;
968QSSGLoadedTexture *QSSGLoadedTexture::load(
const QString &inPath,
969 const QSSGRenderTextureFormat &inFormat,
972 if (inPath.isEmpty())
975 QSSGLoadedTexture *theLoadedImage =
nullptr;
977 QSSGInputUtil::FileType fileType = QSSGInputUtil::UnknownFile;
978 QSharedPointer<QIODevice> theStream =
979 QSSGInputUtil::getStreamForTextureFile(inPath,
true, &fileName, &fileType);
983 case QSSGInputUtil::HdrFile:
986 theLoadedImage = loadHdrImage(theStream, inFormat);
988 case QSSGInputUtil::TextureFile:
989 theLoadedImage = loadCompressedImage(fileName);
992 theLoadedImage = loadQImage(fileName, inFlipY);
997 return loadCubeMap(inPath, inFlipY);
999 return theLoadedImage;
Combined button and popup list for selecting options.
static bool isCompatible(const QImage &img1, const QImage &img2)
static QSSGRenderTextureFormat fromGLtoTextureFormat(quint32 internalFormat)
static QImage loadImage(const QString &inPath, bool flipVertical)
static QSSGLoadedTexture * loadCubeMap(const QString &inPath, bool flipY)