mirror of
https://invent.kde.org/frameworks/kimageformats.git
synced 2025-05-28 00:30:23 -04:00
Where possible, QIODevice::peek has been used instead of transactions or instead of using ungetchar() for sequential access devices and seek() for random access devices. Furthermore: - RAS format gained the ability of read on sequential devices. - Removed unused code in XCF (still related to ungetchar and sequential devices). - These changes should prevent errors like the ones fixed by MR !258
486 lines
13 KiB
C++
486 lines
13 KiB
C++
/*
|
|
This file is part of the KDE project
|
|
SPDX-FileCopyrightText: 2023 Ernest Gupik <ernestgupik@wp.pl>
|
|
SPDX-FileCopyrightText: 2023 Mirco Miranda <mircomir@outlook.com>
|
|
|
|
SPDX-License-Identifier: LGPL-2.0-or-later
|
|
*/
|
|
|
|
#include "qoi_p.h"
|
|
#include "scanlineconverter_p.h"
|
|
#include "util_p.h"
|
|
|
|
#include <QColorSpace>
|
|
#include <QFile>
|
|
#include <QIODevice>
|
|
#include <QImage>
|
|
|
|
namespace // Private
|
|
{
|
|
|
|
#define QOI_OP_INDEX 0x00 /* 00xxxxxx */
|
|
#define QOI_OP_DIFF 0x40 /* 01xxxxxx */
|
|
#define QOI_OP_LUMA 0x80 /* 10xxxxxx */
|
|
#define QOI_OP_RUN 0xc0 /* 11xxxxxx */
|
|
#define QOI_OP_RGB 0xfe /* 11111110 */
|
|
#define QOI_OP_RGBA 0xff /* 11111111 */
|
|
#define QOI_MASK_2 0xc0 /* 11000000 */
|
|
|
|
#define QOI_MAGIC (((unsigned int)'q') << 24 | ((unsigned int)'o') << 16 | ((unsigned int)'i') << 8 | ((unsigned int)'f'))
|
|
#define QOI_HEADER_SIZE 14
|
|
#define QOI_END_STREAM_PAD 8
|
|
|
|
struct QoiHeader {
|
|
QoiHeader()
|
|
: MagicNumber(0)
|
|
, Width(0)
|
|
, Height(0)
|
|
, Channels(0)
|
|
, Colorspace(2)
|
|
{
|
|
}
|
|
|
|
QoiHeader(const QoiHeader&) = default;
|
|
QoiHeader& operator=(const QoiHeader&) = default;
|
|
|
|
quint32 MagicNumber;
|
|
quint32 Width;
|
|
quint32 Height;
|
|
quint8 Channels;
|
|
quint8 Colorspace;
|
|
};
|
|
|
|
struct Px {
|
|
bool operator==(const Px &other) const
|
|
{
|
|
return r == other.r && g == other.g && b == other.b && a == other.a;
|
|
}
|
|
quint8 r;
|
|
quint8 g;
|
|
quint8 b;
|
|
quint8 a;
|
|
};
|
|
|
|
static QDataStream &operator>>(QDataStream &s, QoiHeader &head)
|
|
{
|
|
s >> head.MagicNumber;
|
|
s >> head.Width;
|
|
s >> head.Height;
|
|
s >> head.Channels;
|
|
s >> head.Colorspace;
|
|
return s;
|
|
}
|
|
|
|
static QDataStream &operator<<(QDataStream &s, const QoiHeader &head)
|
|
{
|
|
s << head.MagicNumber;
|
|
s << head.Width;
|
|
s << head.Height;
|
|
s << head.Channels;
|
|
s << head.Colorspace;
|
|
return s;
|
|
}
|
|
|
|
static bool IsSupported(const QoiHeader &head)
|
|
{
|
|
// Check magic number
|
|
if (head.MagicNumber != QOI_MAGIC) {
|
|
return false;
|
|
}
|
|
// Check if the header is a valid QOI header
|
|
if (head.Width == 0 || head.Height == 0 || head.Channels < 3 || head.Colorspace > 1) {
|
|
return false;
|
|
}
|
|
// Set a reasonable upper limit
|
|
if (head.Width > 300000 || head.Height > 300000) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
static int QoiHash(const Px &px)
|
|
{
|
|
return px.r * 3 + px.g * 5 + px.b * 7 + px.a * 11;
|
|
}
|
|
|
|
static QImage::Format imageFormat(const QoiHeader &head)
|
|
{
|
|
if (IsSupported(head)) {
|
|
return (head.Channels == 3 ? QImage::Format_RGB32 : QImage::Format_ARGB32);
|
|
}
|
|
return QImage::Format_Invalid;
|
|
}
|
|
|
|
static bool LoadQOI(QIODevice *device, const QoiHeader &qoi, QImage &img)
|
|
{
|
|
Px index[64] = {Px{0, 0, 0, 0}};
|
|
Px px = Px{0, 0, 0, 255};
|
|
|
|
// The px_len should be enough to read a complete "compressed" row: an uncompressible row can become
|
|
// larger than the row itself. It should never be more than 1/3 (RGB) or 1/4 (RGBA) the length of the
|
|
// row itself (see test bnm_rgb*.qoi) so I set the extra data to 1/2.
|
|
// The minimum value is to ensure that enough bytes are read when the image is very small (e.g. 1x1px):
|
|
// it can be set as large as you like.
|
|
quint64 px_len = std::max(quint64(1024), quint64(qoi.Width) * qoi.Channels * 3 / 2);
|
|
if (px_len > kMaxQVectorSize) {
|
|
return false;
|
|
}
|
|
|
|
// Allocate image
|
|
img = imageAlloc(qoi.Width, qoi.Height, imageFormat(qoi));
|
|
if (img.isNull()) {
|
|
return false;
|
|
}
|
|
|
|
// Set the image colorspace based on the qoi.Colorspace value
|
|
// As per specification: 0 = sRGB with linear alpha, 1 = all channels linear
|
|
if (qoi.Colorspace) {
|
|
img.setColorSpace(QColorSpace(QColorSpace::SRgbLinear));
|
|
} else {
|
|
img.setColorSpace(QColorSpace(QColorSpace::SRgb));
|
|
}
|
|
|
|
// Handle the byte stream
|
|
QByteArray ba;
|
|
for (quint32 y = 0, run = 0; y < qoi.Height; ++y) {
|
|
if (quint64(ba.size()) < px_len) {
|
|
ba.append(device->read(px_len));
|
|
}
|
|
|
|
if (ba.size() < QOI_END_STREAM_PAD) {
|
|
return false;
|
|
}
|
|
|
|
quint64 chunks_len = ba.size() - QOI_END_STREAM_PAD;
|
|
quint64 p = 0;
|
|
QRgb *scanline = reinterpret_cast<QRgb *>(img.scanLine(y));
|
|
const quint8 *input = reinterpret_cast<const quint8 *>(ba.constData());
|
|
for (quint32 x = 0; x < qoi.Width; ++x) {
|
|
if (run > 0) {
|
|
run--;
|
|
} else if (p < chunks_len) {
|
|
quint32 b1 = input[p++];
|
|
|
|
if (b1 == QOI_OP_RGB) {
|
|
px.r = input[p++];
|
|
px.g = input[p++];
|
|
px.b = input[p++];
|
|
} else if (b1 == QOI_OP_RGBA) {
|
|
px.r = input[p++];
|
|
px.g = input[p++];
|
|
px.b = input[p++];
|
|
px.a = input[p++];
|
|
} else if ((b1 & QOI_MASK_2) == QOI_OP_INDEX) {
|
|
px = index[b1];
|
|
} else if ((b1 & QOI_MASK_2) == QOI_OP_DIFF) {
|
|
px.r += ((b1 >> 4) & 0x03) - 2;
|
|
px.g += ((b1 >> 2) & 0x03) - 2;
|
|
px.b += (b1 & 0x03) - 2;
|
|
} else if ((b1 & QOI_MASK_2) == QOI_OP_LUMA) {
|
|
quint32 b2 = input[p++];
|
|
quint32 vg = (b1 & 0x3f) - 32;
|
|
px.r += vg - 8 + ((b2 >> 4) & 0x0f);
|
|
px.g += vg;
|
|
px.b += vg - 8 + (b2 & 0x0f);
|
|
} else if ((b1 & QOI_MASK_2) == QOI_OP_RUN) {
|
|
run = (b1 & 0x3f);
|
|
}
|
|
index[QoiHash(px) & 0x3F] = px;
|
|
}
|
|
// Set the values for the pixel at (x, y)
|
|
scanline[x] = qRgba(px.r, px.g, px.b, px.a);
|
|
}
|
|
|
|
if (p) {
|
|
ba.remove(0, p);
|
|
}
|
|
}
|
|
|
|
// From specs the byte stream's end is marked with 7 0x00 bytes followed by a single 0x01 byte.
|
|
// NOTE: Instead of using "ba == QByteArray::fromRawData("\x00\x00\x00\x00\x00\x00\x00\x01", 8)"
|
|
// we preferred a generic check that allows data to exist after the end of the file.
|
|
return (ba.startsWith(QByteArray::fromRawData("\x00\x00\x00\x00\x00\x00\x00\x01", 8)));
|
|
}
|
|
|
|
static bool SaveQOI(QIODevice *device, const QoiHeader &qoi, const QImage &img)
|
|
{
|
|
Px index[64] = {Px{0, 0, 0, 0}};
|
|
Px px = Px{0, 0, 0, 255};
|
|
Px px_prev = px;
|
|
|
|
auto run = 0;
|
|
auto channels = qoi.Channels;
|
|
|
|
QByteArray ba;
|
|
ba.reserve(img.width() * channels * 3 / 2);
|
|
|
|
ScanLineConverter converter(channels == 3 ? QImage::Format_RGB888 : QImage::Format_RGBA8888);
|
|
converter.setTargetColorSpace(QColorSpace(qoi.Colorspace == 1 ? QColorSpace::SRgbLinear : QColorSpace::SRgb));
|
|
|
|
for (auto h = img.height(), y = 0; y < h; ++y) {
|
|
auto pixels = converter.convertedScanLine(img, y);
|
|
if (pixels == nullptr) {
|
|
return false;
|
|
}
|
|
|
|
for (auto w = img.width() * channels, px_pos = 0; px_pos < w; px_pos += channels) {
|
|
px.r = pixels[px_pos + 0];
|
|
px.g = pixels[px_pos + 1];
|
|
px.b = pixels[px_pos + 2];
|
|
|
|
if (channels == 4) {
|
|
px.a = pixels[px_pos + 3];
|
|
}
|
|
|
|
if (px == px_prev) {
|
|
run++;
|
|
if (run == 62 || (px_pos == w - channels && y == h - 1)) {
|
|
ba.append(QOI_OP_RUN | (run - 1));
|
|
run = 0;
|
|
}
|
|
} else {
|
|
int index_pos;
|
|
|
|
if (run > 0) {
|
|
ba.append(QOI_OP_RUN | (run - 1));
|
|
run = 0;
|
|
}
|
|
|
|
index_pos = QoiHash(px) & 0x3F;
|
|
|
|
if (index[index_pos] == px) {
|
|
ba.append(QOI_OP_INDEX | index_pos);
|
|
} else {
|
|
index[index_pos] = px;
|
|
|
|
if (px.a == px_prev.a) {
|
|
signed char vr = px.r - px_prev.r;
|
|
signed char vg = px.g - px_prev.g;
|
|
signed char vb = px.b - px_prev.b;
|
|
|
|
signed char vg_r = vr - vg;
|
|
signed char vg_b = vb - vg;
|
|
|
|
if (vr > -3 && vr < 2 && vg > -3 && vg < 2 && vb > -3 && vb < 2) {
|
|
ba.append(QOI_OP_DIFF | (vr + 2) << 4 | (vg + 2) << 2 | (vb + 2));
|
|
} else if (vg_r > -9 && vg_r < 8 && vg > -33 && vg < 32 && vg_b > -9 && vg_b < 8) {
|
|
ba.append(QOI_OP_LUMA | (vg + 32));
|
|
ba.append((vg_r + 8) << 4 | (vg_b + 8));
|
|
} else {
|
|
ba.append(char(QOI_OP_RGB));
|
|
ba.append(px.r);
|
|
ba.append(px.g);
|
|
ba.append(px.b);
|
|
}
|
|
} else {
|
|
ba.append(char(QOI_OP_RGBA));
|
|
ba.append(px.r);
|
|
ba.append(px.g);
|
|
ba.append(px.b);
|
|
ba.append(px.a);
|
|
}
|
|
}
|
|
}
|
|
px_prev = px;
|
|
}
|
|
|
|
auto written = device->write(ba);
|
|
if (written < 0) {
|
|
return false;
|
|
}
|
|
if (written) {
|
|
ba.remove(0, written);
|
|
}
|
|
}
|
|
|
|
// QOI end of stream
|
|
ba.append(QByteArray::fromRawData("\x00\x00\x00\x00\x00\x00\x00\x01", 8));
|
|
|
|
// write remaining data
|
|
for (qint64 w = 0, write = 0, size = ba.size(); write < size; write += w) {
|
|
w = device->write(ba.constData() + write, size - write);
|
|
if (w < 0) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
class QOIHandlerPrivate
|
|
{
|
|
public:
|
|
QOIHandlerPrivate() {}
|
|
~QOIHandlerPrivate() {}
|
|
|
|
QoiHeader m_header;
|
|
};
|
|
|
|
|
|
QOIHandler::QOIHandler()
|
|
: QImageIOHandler()
|
|
, d(new QOIHandlerPrivate)
|
|
{
|
|
}
|
|
|
|
bool QOIHandler::canRead() const
|
|
{
|
|
if (canRead(device())) {
|
|
setFormat("qoi");
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool QOIHandler::canRead(QIODevice *device)
|
|
{
|
|
if (!device) {
|
|
qWarning("QOIHandler::canRead() called with no device");
|
|
return false;
|
|
}
|
|
|
|
auto head = device->peek(QOI_HEADER_SIZE);
|
|
if (head.size() < QOI_HEADER_SIZE) {
|
|
return false;
|
|
}
|
|
|
|
QDataStream stream(head);
|
|
stream.setByteOrder(QDataStream::BigEndian);
|
|
QoiHeader qoi;
|
|
stream >> qoi;
|
|
|
|
return IsSupported(qoi);
|
|
}
|
|
|
|
bool QOIHandler::read(QImage *image)
|
|
{
|
|
QDataStream s(device());
|
|
s.setByteOrder(QDataStream::BigEndian);
|
|
|
|
// Read image header
|
|
auto&& qoi = d->m_header;
|
|
s >> qoi;
|
|
|
|
// Check if file is supported
|
|
if (!IsSupported(qoi)) {
|
|
return false;
|
|
}
|
|
|
|
QImage img;
|
|
bool result = LoadQOI(s.device(), qoi, img);
|
|
|
|
if (result == false) {
|
|
return false;
|
|
}
|
|
|
|
*image = img;
|
|
return true;
|
|
}
|
|
|
|
bool QOIHandler::write(const QImage &image)
|
|
{
|
|
if (image.isNull()) {
|
|
return false;
|
|
}
|
|
|
|
QoiHeader qoi;
|
|
qoi.MagicNumber = QOI_MAGIC;
|
|
qoi.Width = image.width();
|
|
qoi.Height = image.height();
|
|
qoi.Channels = image.hasAlphaChannel() ? 4 : 3;
|
|
qoi.Colorspace = image.colorSpace().transferFunction() == QColorSpace::TransferFunction::Linear ? 1 : 0;
|
|
|
|
if (!IsSupported(qoi)) {
|
|
return false;
|
|
}
|
|
|
|
QDataStream s(device());
|
|
s.setByteOrder(QDataStream::BigEndian);
|
|
s << qoi;
|
|
if (s.status() != QDataStream::Ok) {
|
|
return false;
|
|
}
|
|
|
|
return SaveQOI(s.device(), qoi, image);
|
|
}
|
|
|
|
bool QOIHandler::supportsOption(ImageOption option) const
|
|
{
|
|
if (option == QImageIOHandler::Size) {
|
|
return true;
|
|
}
|
|
if (option == QImageIOHandler::ImageFormat) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
QVariant QOIHandler::option(ImageOption option) const
|
|
{
|
|
QVariant v;
|
|
|
|
if (option == QImageIOHandler::Size) {
|
|
auto&& header = d->m_header;
|
|
if (IsSupported(header)) {
|
|
v = QVariant::fromValue(QSize(header.Width, header.Height));
|
|
} else if (auto d = device()) {
|
|
QDataStream s(d->peek(sizeof(QoiHeader)));
|
|
s.setByteOrder(QDataStream::BigEndian);
|
|
s >> header;
|
|
if (s.status() == QDataStream::Ok && IsSupported(header)) {
|
|
v = QVariant::fromValue(QSize(header.Width, header.Height));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (option == QImageIOHandler::ImageFormat) {
|
|
auto&& header = d->m_header;
|
|
if (IsSupported(header)) {
|
|
v = QVariant::fromValue(imageFormat(header));
|
|
} else if (auto d = device()) {
|
|
QDataStream s(d->peek(sizeof(QoiHeader)));
|
|
s.setByteOrder(QDataStream::BigEndian);
|
|
s >> header;
|
|
if (s.status() == QDataStream::Ok && IsSupported(header)) {
|
|
v = QVariant::fromValue(imageFormat(header));
|
|
}
|
|
}
|
|
}
|
|
|
|
return v;
|
|
}
|
|
|
|
QImageIOPlugin::Capabilities QOIPlugin::capabilities(QIODevice *device, const QByteArray &format) const
|
|
{
|
|
if (format == "qoi" || format == "QOI") {
|
|
return Capabilities(CanRead | CanWrite);
|
|
}
|
|
if (!format.isEmpty()) {
|
|
return {};
|
|
}
|
|
if (!device->isOpen()) {
|
|
return {};
|
|
}
|
|
|
|
Capabilities cap;
|
|
if (device->isReadable() && QOIHandler::canRead(device)) {
|
|
cap |= CanRead;
|
|
}
|
|
if (device->isWritable()) {
|
|
cap |= CanWrite;
|
|
}
|
|
return cap;
|
|
}
|
|
|
|
QImageIOHandler *QOIPlugin::create(QIODevice *device, const QByteArray &format) const
|
|
{
|
|
QImageIOHandler *handler = new QOIHandler;
|
|
handler->setDevice(device);
|
|
handler->setFormat(format);
|
|
return handler;
|
|
}
|
|
|
|
#include "moc_qoi_p.cpp"
|