summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorAllan Sandfeld Jensen <allan.jensen@qt.io>2017-06-07 17:28:25 +0200
committerAllan Sandfeld Jensen <allan.jensen@qt.io>2017-07-03 07:41:56 +0000
commit989b9da3007ff06c79ab7ee553acfce9177df5b6 (patch)
tree1fc65836ff484409c74348f3283a5370d1b90829
parented1373124a55d41a8e7b5c123428f2a088d25896 (diff)
Fix BMP ImageFormat for semi-transparent files
Reads the v4 and v5 info-header together with the rest of the info- headers, and use that to report the correct image format before decoding. Change-Id: I69e2bcc54367b7f14820815ae2ae1fa2d8d5dc8c Reviewed-by: Eirik Aavitsland <eirik.aavitsland@qt.io>
-rw-r--r--src/gui/image/qbmphandler.cpp61
-rw-r--r--src/gui/image/qbmphandler_p.h15
-rw-r--r--tests/auto/gui/image/qimagereader/tst_qimagereader.cpp3
3 files changed, 36 insertions, 43 deletions
diff --git a/src/gui/image/qbmphandler.cpp b/src/gui/image/qbmphandler.cpp
index 324fe01414..4350a5c192 100644
--- a/src/gui/image/qbmphandler.cpp
+++ b/src/gui/image/qbmphandler.cpp
@@ -114,6 +114,15 @@ static QDataStream &operator>>(QDataStream &s, BMP_INFOHDR &bi)
s >> bi.biCompression >> bi.biSizeImage;
s >> bi.biXPelsPerMeter >> bi.biYPelsPerMeter;
s >> bi.biClrUsed >> bi.biClrImportant;
+ if (bi.biSize >= BMP_WIN4) {
+ s >> bi.biRedMask >> bi.biGreenMask >> bi.biBlueMask >> bi.biAlphaMask;
+ s >> bi.biCSType;
+ for (int i = 0; i < 9; ++i)
+ s >> bi.biEndpoints[i];
+ s >> bi.biGammaRed >> bi.biGammaGreen >> bi.biGammaBlue;
+ if (bi.biSize == BMP_WIN5)
+ s >> bi.biIntent >> bi.biProfileData >> bi.biProfileSize >> bi.biReserved;
+ }
}
else { // probably old Windows format
qint16 w, h;
@@ -219,53 +228,20 @@ static bool read_dib_body(QDataStream &s, const BMP_INFOHDR &bi, qint64 offset,
int alpha_scale = 0;
if (!d->isSequential())
- d->seek(startpos + BMP_FILEHDR_SIZE + (bi.biSize >= BMP_WIN4 ? BMP_WIN : bi.biSize)); // goto start of colormap or masks
+ d->seek(startpos + BMP_FILEHDR_SIZE + bi.biSize); // goto start of colormap or masks
- if (bi.biSize >= BMP_WIN4 || (comp == BMP_BITFIELDS && (nbits == 16 || nbits == 32))) {
+ if (bi.biSize >= BMP_WIN4) {
+ red_mask = bi.biRedMask;
+ green_mask = bi.biGreenMask;
+ blue_mask = bi.biBlueMask;
+ alpha_mask = bi.biAlphaMask;
+ } else if (comp == BMP_BITFIELDS && (nbits == 16 || nbits == 32)) {
if (d->read((char *)&red_mask, sizeof(red_mask)) != sizeof(red_mask))
return false;
if (d->read((char *)&green_mask, sizeof(green_mask)) != sizeof(green_mask))
return false;
if (d->read((char *)&blue_mask, sizeof(blue_mask)) != sizeof(blue_mask))
return false;
-
- // Read BMP v4+ header
- if (bi.biSize >= BMP_WIN4) {
- int CSType = 0;
- int gamma_red = 0;
- int gamma_green = 0;
- int gamma_blue = 0;
- int endpoints[9];
-
- if (d->read((char *)&alpha_mask, sizeof(alpha_mask)) != sizeof(alpha_mask))
- return false;
- if (d->read((char *)&CSType, sizeof(CSType)) != sizeof(CSType))
- return false;
- if (d->read((char *)&endpoints, sizeof(endpoints)) != sizeof(endpoints))
- return false;
- if (d->read((char *)&gamma_red, sizeof(gamma_red)) != sizeof(gamma_red))
- return false;
- if (d->read((char *)&gamma_green, sizeof(gamma_green)) != sizeof(gamma_green))
- return false;
- if (d->read((char *)&gamma_blue, sizeof(gamma_blue)) != sizeof(gamma_blue))
- return false;
-
- if (bi.biSize == BMP_WIN5) {
- qint32 intent = 0;
- qint32 profileData = 0;
- qint32 profileSize = 0;
- qint32 reserved = 0;
-
- if (d->read((char *)&intent, sizeof(intent)) != sizeof(intent))
- return false;
- if (d->read((char *)&profileData, sizeof(profileData)) != sizeof(profileData))
- return false;
- if (d->read((char *)&profileSize, sizeof(profileSize)) != sizeof(profileSize))
- return false;
- if (d->read((char *)&reserved, sizeof(reserved)) != sizeof(reserved) || reserved != 0)
- return false;
- }
- }
}
bool transp = (comp == BMP_BITFIELDS) && alpha_mask;
@@ -876,7 +852,10 @@ QVariant QBmpHandler::option(ImageOption option) const
case 32:
case 24:
case 16:
- format = QImage::Format_RGB32;
+ if (infoHeader.biCompression == BMP_BITFIELDS && infoHeader.biSize >= BMP_WIN4 && infoHeader.biAlphaMask)
+ format = QImage::Format_ARGB32;
+ else
+ format = QImage::Format_RGB32;
break;
case 8:
case 4:
diff --git a/src/gui/image/qbmphandler_p.h b/src/gui/image/qbmphandler_p.h
index 3e1fc3d511..56b39dd0f0 100644
--- a/src/gui/image/qbmphandler_p.h
+++ b/src/gui/image/qbmphandler_p.h
@@ -78,6 +78,21 @@ struct BMP_INFOHDR { // BMP information header
qint32 biYPelsPerMeter; // vertical resolution
qint32 biClrUsed; // number of colors used
qint32 biClrImportant; // number of important colors
+ // V4:
+ quint32 biRedMask;
+ quint32 biGreenMask;
+ quint32 biBlueMask;
+ quint32 biAlphaMask;
+ qint32 biCSType;
+ qint32 biEndpoints[9];
+ qint32 biGammaRed;
+ qint32 biGammaGreen;
+ qint32 biGammaBlue;
+ // V5:
+ qint32 biIntent;
+ qint32 biProfileData;
+ qint32 biProfileSize;
+ qint32 biReserved;
};
// BMP-Handler, which is also able to read and write the DIB
diff --git a/tests/auto/gui/image/qimagereader/tst_qimagereader.cpp b/tests/auto/gui/image/qimagereader/tst_qimagereader.cpp
index 574ad805ca..a908e06498 100644
--- a/tests/auto/gui/image/qimagereader/tst_qimagereader.cpp
+++ b/tests/auto/gui/image/qimagereader/tst_qimagereader.cpp
@@ -519,7 +519,7 @@ void tst_QImageReader::imageFormat_data()
QTest::newRow("xpm") << QString("marble.xpm") << QByteArray("xpm") << QImage::Format_Indexed8;
QTest::newRow("bmp-1") << QString("colorful.bmp") << QByteArray("bmp") << QImage::Format_Indexed8;
QTest::newRow("bmp-2") << QString("font.bmp") << QByteArray("bmp") << QImage::Format_Indexed8;
- QTest::newRow("bmp-3") << QString("test32bfv4.bmp") << QByteArray("bmp") << QImage::Format_RGB32;
+ QTest::newRow("bmp-3") << QString("test32bfv4.bmp") << QByteArray("bmp") << QImage::Format_ARGB32;
QTest::newRow("bmp-4") << QString("test32v5.bmp") << QByteArray("bmp") << QImage::Format_RGB32;
QTest::newRow("png") << QString("kollada.png") << QByteArray("png") << QImage::Format_ARGB32;
QTest::newRow("png-2") << QString("YCbCr_cmyk.png") << QByteArray("png") << QImage::Format_RGB32;
@@ -736,7 +736,6 @@ void tst_QImageReader::imageFormatBeforeRead()
QSize size = reader.size();
QImage image(size, fileFormat);
QVERIFY(reader.read(&image));
- QEXPECT_FAIL("bmp-3", "Semi-transparent BMPs not predicted", Continue);
QCOMPARE(image.format(), fileFormat);
}
}