kdelibs/kimgio/ico.cpp

441 lines
14 KiB
C++
Raw Normal View History

/* This file is part of the KDE libraries
Copyright (C) 2022 Ivailo Monev <xakepa10@gmail.com>
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Library General Public
License version 2, as published by the Free Software Foundation.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Library General Public License for more details.
You should have received a copy of the GNU Library General Public License
along with this library; see the file COPYING.LIB. If not, write to
the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
Boston, MA 02110-1301, USA.
*/
#include "ico.h"
#include <QImage>
#include <kdebug.h>
#include <limits.h>
static const char* const s_icopluginformat = "ico";
static const ushort s_peekbuffsize = 32;
static const uchar s_pngheader[] = { 0x89, 0x50, 0x4e, 0x47, 0x0d, 0x0a, 0x1a, 0x0a };
static const ushort s_pngheadersize = 8;
// for reference:
// https://en.wikipedia.org/wiki/List_of_file_signatures
static const uchar s_icoheader[] = { 0x0, 0x0, 0x1, 0x0, 0x0 };
static const uchar s_icoheader1[] = { 0x0, 0x0, 0x1, 0x0, 0x01 };
static const uchar s_icoheader2[] = { 0x0, 0x0, 0x1, 0x0, 0x02 };
static const uchar s_icoheader3[] = { 0x0, 0x0, 0x1, 0x0, 0x03 };
static const uchar s_icoheader4[] = { 0x0, 0x0, 0x1, 0x0, 0x04 };
static const uchar s_icoheader5[] = { 0x0, 0x0, 0x1, 0x0, 0x05 };
static const uchar s_icoheader6[] = { 0x0, 0x0, 0x1, 0x0, 0x06 };
static const uchar s_icoheader7[] = { 0x0, 0x0, 0x1, 0x0, 0x07 };
static const uchar s_icoheader8[] = { 0x0, 0x0, 0x1, 0x0, 0x08 };
static const uchar s_icoheader9[] = { 0x0, 0x0, 0x1, 0x0, 0x09 };
static const struct HeadersTblData {
const uchar *header;
const int headersize;
} HeadersTbl[] = {
{ s_icoheader, 5 },
{ s_icoheader1, 5 },
{ s_icoheader2, 5 },
{ s_icoheader3, 5 },
{ s_icoheader4, 5 },
{ s_icoheader5, 5 },
{ s_icoheader6, 5 },
{ s_icoheader7, 5 },
{ s_icoheader8, 5 },
{ s_icoheader9, 5 }
};
static const qint16 HeadersTblSize = sizeof(HeadersTbl) / sizeof(HeadersTblData);
// for reference:
// https://en.wikipedia.org/wiki/ICO_(file_format)
// https://en.wikipedia.org/wiki/BMP_file_format
enum ICOType {
TypeIcon = 1,
TypeCursor = 2
};
enum BMPCompression {
CompressionRGB = 0,
CompressionRLE8 = 1,
CompressionRLE4 = 2,
CompressionBitFields = 3,
CompressionJPEG = 4,
CompressionPNG = 5,
CompressionAlphaBitFields = 6,
CompressionCMYK = 7,
CompressionCMYKRLE8 = 8,
CompressionCMYKRLE4 = 9,
};
ICOHandler::ICOHandler()
{
}
ICOHandler::~ICOHandler()
{
}
bool ICOHandler::canRead() const
{
if (ICOHandler::canRead(device())) {
setFormat(s_icopluginformat);
return true;
}
return false;
}
bool ICOHandler::read(QImage *image)
{
Q_ASSERT(sizeof(uchar) == 1);
Q_ASSERT(sizeof(ushort) == 2);
Q_ASSERT(sizeof(uint) == 4);
QDataStream datastream(device());
datastream.setByteOrder(QDataStream::LittleEndian);
ushort icoreserved = 0;
ushort icotype = 0;
ushort iconimages = 0;
datastream >> icoreserved;
datastream >> icotype;
datastream >> iconimages;
if (icotype == ICOType::TypeCursor) {
kWarning() << "Cursor icons are not supported";
return false;
} else if (icotype != ICOType::TypeIcon) {
kWarning() << "Invalid icon type" << icotype;
return false;
} else if (iconimages < 1) {
kDebug() << "No images" << iconimages;
return false;
}
for (ushort ii = 0; ii < iconimages; ii++) {
uchar icowidth = 0;
uchar icoheight = 0;
uchar icocolors = 0;
uchar icoreserved2 = 0;
ushort icoplaneorhhs = 0;
ushort icobpporvhs = 0;
uint icoimagesize = 0;
uint icoimageoffset = 0;
datastream >> icowidth;
datastream >> icoheight;
datastream >> icocolors;
datastream >> icoreserved2;
datastream >> icoplaneorhhs;
datastream >> icobpporvhs;
datastream >> icoimagesize;
datastream >> icoimageoffset;
if (Q_UNLIKELY(icoimageoffset > datastream.device()->size())) {
kWarning() << "Invalid image offset" << icoimageoffset;
return false;
}
if (Q_UNLIKELY(icoimagesize >= INT_MAX)) {
kWarning() << "ICO image size is too big" << icoimagesize;
return false;
}
datastream.device()->seek(icoimageoffset);
QByteArray imagebytes(icoimagesize, char(0));
if (Q_UNLIKELY(datastream.readRawData(imagebytes.data(), icoimagesize) != icoimagesize)) {
kWarning() << "Could not read image data";
return false;
}
if (imagebytes.size() > s_pngheadersize &&
::memcmp(imagebytes.constData(), s_pngheader, s_pngheadersize) != 0) {
datastream.device()->seek(icoimageoffset);
uint bmpheadersize = 0;
uint bmpwidth = 0;
uint bmpheight = 0;
ushort bmpplanes = 0;
ushort bmpbpp = 0;
uint bmpcompression = 0;
uint bmpimagesize = 0;
uint bmphppm = 0;
uint bmpvppm = 0;
uint bmpncolors = 0;
uint bmpnimportantcolors = 0;
datastream >> bmpheadersize;
datastream >> bmpwidth;
datastream >> bmpheight;
datastream >> bmpplanes;
datastream >> bmpbpp;
datastream >> bmpcompression;
datastream >> bmpimagesize;
datastream >> bmphppm;
datastream >> bmpvppm;
datastream >> bmpncolors;
datastream >> bmpnimportantcolors;
if (bmpheadersize != 40) {
kWarning() << "Invalid BMP info header size" << bmpheadersize;
return false;
}
if (bmpcompression != BMPCompression::CompressionRGB) {
kWarning() << "Unsupported BMP compression" << bmpcompression;
return false;
}
// fallbacks
const int imagewidth = (icowidth ? icowidth : bmpwidth);
const int imageheight = (icoheight ? icoheight : bmpheight);
int imageboundary = 0;
QImage::Format imageformat = QImage::Format_ARGB32;
switch (bmpbpp) {
case 32: {
imageboundary = (imagewidth * imageheight * 4);
break;
}
case 24: {
imageformat = QImage::Format_RGB32;
imageboundary = (imagewidth * imageheight * 3);
break;
}
default: {
kWarning() << "Unsupported BMP bits per-pixel" << bmpbpp;
return false;
}
}
if (Q_UNLIKELY(bmpimagesize == 0)) {
kDebug() << "BMP image size is dummy" << bmpimagesize << imageboundary;
bmpimagesize = imageboundary;
} else if (Q_UNLIKELY(bmpimagesize >= INT_MAX)) {
kWarning() << "BMP image size is too big" << bmpimagesize;
return false;
}
imagebytes.resize(bmpimagesize);
if (Q_UNLIKELY(datastream.readRawData(imagebytes.data(), bmpimagesize) != bmpimagesize)) {
kWarning() << "Could not read BMP image data";
return false;
}
QImage bmpimage(imagewidth, imageheight, imageformat);
if (Q_UNLIKELY(bmpimage.isNull())) {
kWarning() << "Could not create BMP image" << imagewidth << imageheight << imageformat;
return false;
}
if (bmpimagesize != imageboundary) {
// data may be padded
kDebug() << "BMP and QImage bytes count mismatch" << bmpimagesize << imageboundary;
}
switch (bmpbpp) {
case 32: {
#if Q_BYTE_ORDER == Q_BIG_ENDIAN
QRgb* bmpimagebits = reinterpret_cast<QRgb*>(bmpimage.bits());
for (uint bi = 0; bi < bmpimagesize && bi < imageboundary; bi += 4) {
*bmpimagebits = qRgba(imagebytes.at(bi + 2), imagebytes.at(bi + 1), imagebytes.at(bi), imagebytes.at(bi + 3));
bmpimagebits++;
}
#else
char* bmpimagebits = reinterpret_cast<char*>(bmpimage.bits());
::memcpy(bmpimagebits, imagebytes.constData(), imageboundary * sizeof(char));
#endif
break;
}
case 24: {
QRgb* bmpimagebits = reinterpret_cast<QRgb*>(bmpimage.bits());
for (uint bi = 0; bi < bmpimagesize && bi < imageboundary; bi += 3) {
*bmpimagebits = qRgb(imagebytes.at(bi + 2), imagebytes.at(bi + 1), imagebytes.at(bi));
bmpimagebits++;
}
break;
}
default: {
Q_ASSERT(false);
break;
}
}
// image is flipped vertically
*image = bmpimage.mirrored(false, true);
kDebug() << "Valid BMP image" << ii;
return true;
}
const QImage pngimage = QImage::fromData(imagebytes.constData(), imagebytes.size(), "PNG");
if (Q_LIKELY(!pngimage.isNull())) {
kDebug() << "Valid PNG image" << ii;
*image = pngimage;
return true;
}
}
kWarning() << "No images could be loaded";
return false;
}
bool ICOHandler::write(const QImage &image)
{
if (image.isNull()) {
kWarning() << "Null image";
return false;
}
QImage bmpimage = image.convertToFormat(QImage::Format_ARGB32);
if (bmpimage.width() > 256 || bmpimage.height() > 256) {
kDebug() << "Scaling down image";
bmpimage = bmpimage.scaled(256, 256, Qt::KeepAspectRatio, Qt::SmoothTransformation);
}
bmpimage = bmpimage.mirrored(false, true);
QDataStream datastream(device());
datastream.setByteOrder(QDataStream::LittleEndian);
ushort icoreserved = 0;
ushort icotype = ICOType::TypeIcon;
ushort iconimages = 1;
datastream << icoreserved;
datastream << icotype;
datastream << iconimages;
uchar icowidth = bmpimage.width();
uchar icoheight = bmpimage.height();
uchar icocolors = 0;
uchar icoreserved2 = 0;
ushort icoplaneorhhs = 0;
ushort icobpporvhs = 0;
uint icoimagesize = bmpimage.byteCount();
uint icoimageoffset = 0;
datastream << icowidth;
datastream << icoheight;
datastream << icocolors;
datastream << icoreserved2;
datastream << icoplaneorhhs;
datastream << icobpporvhs;
datastream << icoimagesize;
icoimageoffset = (datastream.device()->pos() + sizeof(uint));
datastream << icoimageoffset;
uint bmpheadersize = 40;
uint bmpwidth = bmpimage.width();
uint bmpheight = bmpimage.height();
ushort bmpplanes = 0;
ushort bmpbpp = bmpimage.depth();
uint bmpcompression = BMPCompression::CompressionRGB;
uint bmpimagesize = bmpimage.byteCount();
uint bmphppm = 0;
uint bmpvppm = 0;
uint bmpncolors = 0;
uint bmpnimportantcolors = 0;
datastream << bmpheadersize;
datastream << bmpwidth;
datastream << bmpheight;
datastream << bmpplanes;
datastream << bmpbpp;
datastream << bmpcompression;
datastream << bmpimagesize;
datastream << bmphppm;
datastream << bmpvppm;
datastream << bmpncolors;
datastream << bmpnimportantcolors;
#if Q_BYTE_ORDER == Q_BIG_ENDIAN
const QRgb* bmpimagebits = reinterpret_cast<const QRgb*>(bmpimage.constBits());
for (uint bi = 0; bi < bmpimagesize; bi += 4) {
const uchar bmpb = qBlue(*bmpimagebits);
const uchar bmpg = qGreen(*bmpimagebits);
const uchar bmpr = qRed(*bmpimagebits);
const uchar bmpa = qAlpha(*bmpimagebits);
datastream << bmpb << bmpg << bmpr << bmpa;
bmpimagebits++;
}
#else
const char* bmpimagebits = reinterpret_cast<const char*>(bmpimage.constBits());
datastream.writeRawData(bmpimagebits, bmpimagesize);
#endif
return (datastream.status() == QDataStream::Ok);
}
QByteArray ICOHandler::name() const
{
return s_icopluginformat;
}
bool ICOHandler::canRead(QIODevice *device)
{
if (Q_UNLIKELY(!device)) {
kWarning() << "Called with no device";
return false;
}
const QByteArray data = device->peek(s_peekbuffsize);
// ICONDIR, ICONDIRENTRY and one bit of data
if (data.size() < 23) {
kDebug() << "Not enough data for ICO";
return false;
}
for (int i = 0; i < HeadersTblSize; i++) {
if (::memcmp(data.constData(), HeadersTbl[i].header, HeadersTbl[i].headersize) == 0) {
kDebug() << "Header detected";
return true;
}
}
return false;
}
QStringList ICOPlugin::keys() const
{
return QStringList() << s_icopluginformat;
}
QList<QByteArray> ICOPlugin::mimeTypes() const
{
static const QList<QByteArray> list = QList<QByteArray>()
<< "image/vnd.microsoft.icon";
return list;
}
QImageIOPlugin::Capabilities ICOPlugin::capabilities(QIODevice *device, const QByteArray &format) const
{
if (format == s_icopluginformat) {
return QImageIOPlugin::Capabilities(QImageIOPlugin::CanRead | QImageIOPlugin::CanWrite);
}
if (!device || !device->isOpen()) {
return 0;
}
QImageIOPlugin::Capabilities cap;
if (device->isReadable() && ICOHandler::canRead(device)) {
cap |= QImageIOPlugin::CanRead;
}
if (device->isWritable()) {
cap |= QImageIOPlugin::CanWrite;
}
return cap;
}
QImageIOHandler *ICOPlugin::create(QIODevice *device, const QByteArray &format) const
{
QImageIOHandler *handler = new ICOHandler();
handler->setDevice(device);
handler->setFormat(format);
return handler;
}
Q_EXPORT_PLUGIN(ICOPlugin)