/* Run-Length Encoding utilities. SPDX-FileCopyrightText: 2014-2015 Alex Merry SPDX-License-Identifier: LGPL-2.0-or-later */ #ifndef KIMAGEFORMATS_RLE_P_H #define KIMAGEFORMATS_RLE_P_H #include #include /** * The RLEVariant to use. * * This mostly concerns what to do values >= 128. */ enum class RLEVariant { /** * PackBits-style RLE * * Value 128 is ignored, 129 indicates a repetition * of size 2, 130 of size 3, up to 255 of size 128. */ PackBits, /** * Same as PackBits, but treat unpacked data as * 16-bit integers. */ PackBits16, /** * PIC-style RLE * * Value 128 indicates a 16-bit repetition count * follows, while 129 indicates a repetition * of size 128, 130 of size 127, down to 255 of * size 2. */ PIC, }; /** * Decodes data written in run-length encoding format. * * This is intended to be used with lambda functions. * * Note that this functions expects that, at the current location in @p stream, * exactly @p length items have been encoded as a unit (and so it will not be * partway through a run when it has decoded @p length items). If this is not * the case, it will return @c false. * * @param variant The RLE variant to decode. * @param stream The stream to read the data from. * @param buf The location to write the decoded data. * @param length The number of items to read. * @param readData A function that takes a QDataStream reference and reads a * single value. * @param updateItem A function that takes an item from @p buf and the result * of a readData call, and produces the item that should be * written to @p buf. * * @returns @c true if @p length items in mixed RLE were successfully read * into @p buf, @c false otherwise. */ template static inline bool decodeRLEData(RLEVariant variant, QDataStream &stream, Item *dest, quint32 length, Func1 readData, Func2 updateItem) { unsigned offset = 0; // in dest bool is_msb = true; // only used for 16-bit PackBits, data is big-endian quint16 temp_data = 0; while (offset < length) { unsigned remaining = length - offset; quint8 count1; stream >> count1; if (count1 >= 128u) { unsigned length = 0; if (variant == RLEVariant::PIC) { if (count1 == 128u) { // If the value is exactly 128, it means that it is more than // 127 repetitions quint16 count2; stream >> count2; length = count2; } else { // 2 to 128 repetitions length = count1 - 127u; } } else if (variant == RLEVariant::PackBits || variant == RLEVariant::PackBits16) { if (count1 == 128u) { // Ignore value 128 continue; } else { // 128 to 2 repetitions length = 257u - count1; } } else { Q_ASSERT(false); } if (length > remaining) { qDebug() << "Row overrun:" << length << ">" << remaining; return false; } auto datum = readData(stream); for (unsigned i = offset; i < offset + length; ++i) { if (variant == RLEVariant::PackBits16) { if (is_msb) { temp_data = datum << 8; is_msb = false; } else { temp_data |= datum; dest[i >> 1] = updateItem(dest[i >> 1], temp_data); is_msb = true; } } else { dest[i] = updateItem(dest[i], datum); } } offset += length; } else { // No repetitions unsigned length = count1 + 1u; if (length > remaining) { qDebug() << "Row overrun:" << length << ">" << remaining; return false; } for (unsigned i = offset; i < offset + length; ++i) { auto datum = readData(stream); if (variant == RLEVariant::PackBits16) { if (is_msb) { temp_data = datum << 8; is_msb = false; } else { temp_data |= datum; dest[i >> 1] = updateItem(dest[i >> 1], temp_data); is_msb = true; } } else { dest[i] = updateItem(dest[i], datum); } } offset += length; } } if (stream.status() != QDataStream::Ok) { qDebug() << "DataStream status was" << stream.status(); } return stream.status() == QDataStream::Ok; } /** * Encodes data in run-length encoding format. * * This is intended to be used with lambda functions. * * @param variant The RLE variant to encode in. * @param stream The stream to write the data to. * @param data The data to be written. * @param length The number of items to write. * @param itemsEqual A function that takes two items and returns whether * @p writeItem would write them identically. * @param writeItem A function that takes a QDataStream reference and an item * and writes the item to the data stream. */ template static inline void encodeRLEData(RLEVariant variant, QDataStream &stream, const Item *data, unsigned length, Func1 itemsEqual, Func2 writeItem) { unsigned offset = 0; const unsigned maxEncodableChunk = (variant == RLEVariant::PIC) ? 65535u : 128; while (offset < length) { const Item *chunkStart = data + offset; unsigned maxChunk = qMin(length - offset, maxEncodableChunk); const Item *chunkEnd = chunkStart + 1; quint16 chunkLength = 1; while (chunkLength < maxChunk && itemsEqual(*chunkStart, *chunkEnd)) { ++chunkEnd; ++chunkLength; } if (chunkLength > 128) { // Sequence of > 128 identical pixels Q_ASSERT(variant == RLEVariant::PIC); stream << quint8(128); stream << quint16(chunkLength); writeItem(stream, *chunkStart); } else if (chunkLength > 1) { // Sequence of <= 128 identical pixels quint8 encodedLength; if (variant == RLEVariant::PIC) { encodedLength = quint8(chunkLength + 127); } else if (variant == RLEVariant::PackBits) { encodedLength = quint8(257 - chunkLength); } else { Q_ASSERT(false); encodedLength = 0; } stream << encodedLength; writeItem(stream, *chunkStart); } else { // find a string of up to 128 values, each different from the one // that follows it if (maxChunk > 128) { maxChunk = 128; } chunkLength = 1; chunkEnd = chunkStart + 1; while (chunkLength < maxChunk && (chunkLength + 1u == maxChunk || !itemsEqual(*chunkEnd, *(chunkEnd+1)))) { ++chunkEnd; ++chunkLength; } stream << quint8(chunkLength - 1); for (unsigned i = 0; i < chunkLength; ++i) { writeItem(stream, *(chunkStart + i)); } } offset += chunkLength; } } #endif // KIMAGEFORMATS_RLE_P_H