mirror of https://github.com/procxx/kepka.git
Decide if we XOR frames for each frame.
This commit is contained in:
parent
059a24bcdf
commit
0b8aa880e5
|
@ -10,9 +10,11 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
#include "lottie/lottie_frame_renderer.h"
|
#include "lottie/lottie_frame_renderer.h"
|
||||||
#include "ffmpeg/ffmpeg_utility.h"
|
#include "ffmpeg/ffmpeg_utility.h"
|
||||||
#include "base/bytes.h"
|
#include "base/bytes.h"
|
||||||
|
#include "logs.h"
|
||||||
|
|
||||||
#include <QDataStream>
|
#include <QDataStream>
|
||||||
#include <lz4.h>
|
#include <lz4.h>
|
||||||
|
#include <lz4hc.h>
|
||||||
#include <range/v3/numeric/accumulate.hpp>
|
#include <range/v3/numeric/accumulate.hpp>
|
||||||
|
|
||||||
namespace Lottie {
|
namespace Lottie {
|
||||||
|
@ -20,6 +22,33 @@ namespace {
|
||||||
|
|
||||||
constexpr auto kAlignStorage = 16;
|
constexpr auto kAlignStorage = 16;
|
||||||
|
|
||||||
|
void Xor(AlignedStorage &to, const AlignedStorage &from) {
|
||||||
|
Expects(to.rawSize() == from.rawSize());
|
||||||
|
|
||||||
|
using Block = std::conditional_t<
|
||||||
|
sizeof(void*) == sizeof(uint64),
|
||||||
|
uint64,
|
||||||
|
uint32>;
|
||||||
|
constexpr auto kBlockSize = sizeof(Block);
|
||||||
|
const auto amount = from.rawSize();
|
||||||
|
const auto fromBytes = reinterpret_cast<const uchar*>(from.raw());
|
||||||
|
const auto toBytes = reinterpret_cast<uchar*>(to.raw());
|
||||||
|
const auto skip = reinterpret_cast<quintptr>(toBytes) % kBlockSize;
|
||||||
|
const auto blocks = (amount - skip) / kBlockSize;
|
||||||
|
for (auto i = 0; i != skip; ++i) {
|
||||||
|
toBytes[i] ^= fromBytes[i];
|
||||||
|
}
|
||||||
|
const auto fromBlocks = reinterpret_cast<const Block*>(fromBytes + skip);
|
||||||
|
const auto toBlocks = reinterpret_cast<Block*>(toBytes + skip);
|
||||||
|
for (auto i = 0; i != blocks; ++i) {
|
||||||
|
toBlocks[i] ^= fromBlocks[i];
|
||||||
|
}
|
||||||
|
const auto left = amount - skip - (blocks * kBlockSize);
|
||||||
|
for (auto i = amount - left; i != amount; ++i) {
|
||||||
|
toBytes[i] ^= fromBytes[i];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
bool UncompressToRaw(AlignedStorage &to, bytes::const_span from) {
|
bool UncompressToRaw(AlignedStorage &to, bytes::const_span from) {
|
||||||
if (from.empty() || from.size() > to.rawSize()) {
|
if (from.empty() || from.size() > to.rawSize()) {
|
||||||
return false;
|
return false;
|
||||||
|
@ -37,21 +66,51 @@ bool UncompressToRaw(AlignedStorage &to, bytes::const_span from) {
|
||||||
|
|
||||||
void CompressFromRaw(QByteArray &to, const AlignedStorage &from) {
|
void CompressFromRaw(QByteArray &to, const AlignedStorage &from) {
|
||||||
const auto size = from.rawSize();
|
const auto size = from.rawSize();
|
||||||
const auto max = LZ4_compressBound(size);
|
const auto max = sizeof(qint32) + LZ4_compressBound(size);
|
||||||
to.reserve(max);
|
to.reserve(max);
|
||||||
to.resize(max);
|
to.resize(max);
|
||||||
const auto compressed = LZ4_compress_default(
|
const auto compressed = LZ4_compress_default(
|
||||||
static_cast<const char*>(from.raw()),
|
static_cast<const char*>(from.raw()),
|
||||||
to.data(),
|
to.data() + sizeof(qint32),
|
||||||
size,
|
size,
|
||||||
to.size());
|
to.size() - sizeof(qint32));
|
||||||
Assert(compressed > 0);
|
Assert(compressed > 0);
|
||||||
if (compressed >= size) {
|
if (compressed >= size + sizeof(qint32)) {
|
||||||
to.resize(size);
|
to.resize(size + sizeof(qint32));
|
||||||
memcpy(to.data(), from.raw(), size);
|
memcpy(to.data() + sizeof(qint32), from.raw(), size);
|
||||||
} else {
|
} else {
|
||||||
to.resize(compressed);
|
to.resize(compressed + sizeof(qint32));
|
||||||
}
|
}
|
||||||
|
const auto length = qint32(to.size() - sizeof(qint32));
|
||||||
|
bytes::copy(
|
||||||
|
bytes::make_detached_span(to),
|
||||||
|
bytes::object_as_span(&length));
|
||||||
|
}
|
||||||
|
|
||||||
|
void CompressAndSwapFrame(
|
||||||
|
QByteArray &to,
|
||||||
|
QByteArray *additional,
|
||||||
|
AlignedStorage &frame,
|
||||||
|
AlignedStorage &previous) {
|
||||||
|
CompressFromRaw(to, frame);
|
||||||
|
std::swap(frame, previous);
|
||||||
|
if (!additional) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if XOR-d delta compresses better.
|
||||||
|
Xor(frame, previous);
|
||||||
|
CompressFromRaw(*additional, frame);
|
||||||
|
if (additional->size() >= to.size()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
std::swap(to, *additional);
|
||||||
|
|
||||||
|
// Negative length means we XOR-d with the previous frame.
|
||||||
|
const auto negativeLength = -qint32(to.size() - sizeof(qint32));
|
||||||
|
bytes::copy(
|
||||||
|
bytes::make_detached_span(to),
|
||||||
|
bytes::object_as_span(&negativeLength));
|
||||||
}
|
}
|
||||||
|
|
||||||
void Decode(QImage &to, AlignedStorage &from, const QSize &fromSize) {
|
void Decode(QImage &to, AlignedStorage &from, const QSize &fromSize) {
|
||||||
|
@ -83,33 +142,6 @@ void Encode(AlignedStorage &to, const QImage &from, const QSize &toSize) {
|
||||||
to.copyAlignedToRaw();
|
to.copyAlignedToRaw();
|
||||||
}
|
}
|
||||||
|
|
||||||
void Xor(AlignedStorage &to, const AlignedStorage &from) {
|
|
||||||
Expects(to.rawSize() == from.rawSize());
|
|
||||||
|
|
||||||
using Block = std::conditional_t<
|
|
||||||
sizeof(void*) == sizeof(uint64),
|
|
||||||
uint64,
|
|
||||||
uint32>;
|
|
||||||
constexpr auto kBlockSize = sizeof(Block);
|
|
||||||
const auto amount = from.rawSize();
|
|
||||||
const auto fromBytes = reinterpret_cast<const uchar*>(from.raw());
|
|
||||||
const auto toBytes = reinterpret_cast<uchar*>(to.raw());
|
|
||||||
const auto skip = reinterpret_cast<quintptr>(toBytes) % kBlockSize;
|
|
||||||
const auto blocks = (amount - skip) / kBlockSize;
|
|
||||||
for (auto i = 0; i != skip; ++i) {
|
|
||||||
toBytes[i] ^= fromBytes[i];
|
|
||||||
}
|
|
||||||
const auto fromBlocks = reinterpret_cast<const Block*>(fromBytes + skip);
|
|
||||||
const auto toBlocks = reinterpret_cast<Block*>(toBytes + skip);
|
|
||||||
for (auto i = 0; i != blocks; ++i) {
|
|
||||||
toBlocks[i] ^= fromBlocks[i];
|
|
||||||
}
|
|
||||||
const auto left = amount - skip - (blocks * kBlockSize);
|
|
||||||
for (auto i = amount - left; i != amount; ++i) {
|
|
||||||
toBytes[i] ^= fromBytes[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
void AlignedStorage::allocate(int packedBytesPerLine, int lines) {
|
void AlignedStorage::allocate(int packedBytesPerLine, int lines) {
|
||||||
|
@ -295,15 +327,16 @@ bool CacheState::renderFrame(
|
||||||
_offset = headerSize();
|
_offset = headerSize();
|
||||||
_offsetFrameIndex = 0;
|
_offsetFrameIndex = 0;
|
||||||
}
|
}
|
||||||
if (!readCompressedDelta()) {
|
const auto [ok, xored] = readCompressedFrame();
|
||||||
|
if (!ok || (xored && index == 0)) {
|
||||||
_framesReady = 0;
|
_framesReady = 0;
|
||||||
_data = QByteArray();
|
_data = QByteArray();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (index == 0) {
|
if (xored) {
|
||||||
std::swap(_uncompressed, _previous);
|
|
||||||
} else {
|
|
||||||
Xor(_previous, _uncompressed);
|
Xor(_previous, _uncompressed);
|
||||||
|
} else {
|
||||||
|
std::swap(_uncompressed, _previous);
|
||||||
}
|
}
|
||||||
Decode(to, _previous, _size);
|
Decode(to, _previous, _size);
|
||||||
return true;
|
return true;
|
||||||
|
@ -326,14 +359,13 @@ void CacheState::appendFrame(
|
||||||
prepareBuffers();
|
prepareBuffers();
|
||||||
}
|
}
|
||||||
Encode(_uncompressed, frame, _size);
|
Encode(_uncompressed, frame, _size);
|
||||||
if (index == 0) {
|
CompressAndSwapFrame(
|
||||||
writeCompressedDelta();
|
_compressBuffer,
|
||||||
std::swap(_uncompressed, _previous);
|
(index != 0) ? &_xorCompressBuffer : nullptr,
|
||||||
} else {
|
_uncompressed,
|
||||||
std::swap(_uncompressed, _previous);
|
_previous);
|
||||||
Xor(_uncompressed, _previous);
|
_compressedFrames.push_back(_compressBuffer);
|
||||||
writeCompressedDelta();
|
_compressedFrames.back().detach();
|
||||||
}
|
|
||||||
if (++_framesReady == _framesCount) {
|
if (++_framesReady == _framesCount) {
|
||||||
finalizeEncoding();
|
finalizeEncoding();
|
||||||
}
|
}
|
||||||
|
@ -341,7 +373,6 @@ void CacheState::appendFrame(
|
||||||
|
|
||||||
void CacheState::finalizeEncoding() {
|
void CacheState::finalizeEncoding() {
|
||||||
const auto size = (_data.isEmpty() ? headerSize() : _data.size())
|
const auto size = (_data.isEmpty() ? headerSize() : _data.size())
|
||||||
+ (_compressedFrames.size() * sizeof(qint32))
|
|
||||||
+ ranges::accumulate(
|
+ ranges::accumulate(
|
||||||
_compressedFrames,
|
_compressedFrames,
|
||||||
0,
|
0,
|
||||||
|
@ -351,19 +382,24 @@ void CacheState::finalizeEncoding() {
|
||||||
_data.reserve(size);
|
_data.reserve(size);
|
||||||
writeHeader();
|
writeHeader();
|
||||||
}
|
}
|
||||||
|
auto xored = 0;
|
||||||
const auto offset = _data.size();
|
const auto offset = _data.size();
|
||||||
_data.resize(size);
|
_data.resize(size);
|
||||||
auto to = _data.data() + offset;
|
auto to = _data.data() + offset;
|
||||||
for (const auto &block : _compressedFrames) {
|
for (const auto &block : _compressedFrames) {
|
||||||
const auto amount = qint32(block.size());
|
const auto amount = qint32(block.size());
|
||||||
memcpy(to, &amount, sizeof(qint32));
|
|
||||||
to += sizeof(qint32);
|
|
||||||
memcpy(to, block.data(), amount);
|
memcpy(to, block.data(), amount);
|
||||||
|
if (*reinterpret_cast<const qint32*>(block.data()) < 0) {
|
||||||
|
++xored;
|
||||||
|
}
|
||||||
to += amount;
|
to += amount;
|
||||||
}
|
}
|
||||||
_compressedFrames.clear();
|
_compressedFrames.clear();
|
||||||
_compressedFrames.shrink_to_fit();
|
_compressedFrames.shrink_to_fit();
|
||||||
_compressBuffer = QByteArray();
|
_compressBuffer.clear();
|
||||||
|
_compressBuffer.squeeze();
|
||||||
|
|
||||||
|
LOG(("SIZE: %1 (%2x%3, %4 frames, %5 xored)").arg(_data.size()).arg(_size.width()).arg(_size.height()).arg(_framesCount).arg(xored));
|
||||||
}
|
}
|
||||||
|
|
||||||
int CacheState::headerSize() const {
|
int CacheState::headerSize() const {
|
||||||
|
@ -384,33 +420,32 @@ void CacheState::writeHeader() {
|
||||||
<< qint32(_framesReady);
|
<< qint32(_framesReady);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CacheState::writeCompressedDelta() {
|
|
||||||
CompressFromRaw(_compressBuffer, _uncompressed);
|
|
||||||
_compressedFrames.push_back(_compressBuffer);
|
|
||||||
_compressedFrames.back().detach();
|
|
||||||
}
|
|
||||||
|
|
||||||
void CacheState::prepareBuffers() {
|
void CacheState::prepareBuffers() {
|
||||||
_uncompressed.allocate(_size.width() * 4, _size.height());
|
_uncompressed.allocate(_size.width() * 4, _size.height());
|
||||||
_previous.allocate(_size.width() * 4, _size.height());
|
_previous.allocate(_size.width() * 4, _size.height());
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CacheState::readCompressedDelta() {
|
CacheState::ReadResult CacheState::readCompressedFrame() {
|
||||||
auto length = qint32(0);
|
auto length = qint32(0);
|
||||||
const auto part = bytes::make_span(_data).subspan(_offset);
|
const auto part = bytes::make_span(_data).subspan(_offset);
|
||||||
if (part.size() < sizeof(length)) {
|
if (part.size() < sizeof(length)) {
|
||||||
return false;
|
return { false };
|
||||||
}
|
}
|
||||||
bytes::copy(
|
bytes::copy(
|
||||||
bytes::object_as_span(&length),
|
bytes::object_as_span(&length),
|
||||||
part.subspan(0, sizeof(length)));
|
part.subspan(0, sizeof(length)));
|
||||||
const auto bytes = part.subspan(sizeof(length));
|
const auto bytes = part.subspan(sizeof(length));
|
||||||
|
|
||||||
|
const auto xored = (length < 0);
|
||||||
|
if (xored) {
|
||||||
|
length = -length;
|
||||||
|
}
|
||||||
_offset += sizeof(length) + length;
|
_offset += sizeof(length) + length;
|
||||||
++_offsetFrameIndex;
|
++_offsetFrameIndex;
|
||||||
return (length <= bytes.size())
|
const auto ok = (length <= bytes.size())
|
||||||
? UncompressToRaw(_uncompressed, bytes.subspan(0, length))
|
? UncompressToRaw(_uncompressed, bytes.subspan(0, length))
|
||||||
: false;
|
: false;
|
||||||
|
return { ok, xored };
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Lottie
|
} // namespace Lottie
|
||||||
|
|
|
@ -75,18 +75,22 @@ public:
|
||||||
int index);
|
int index);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
struct ReadResult {
|
||||||
|
bool ok = false;
|
||||||
|
bool xored = false;
|
||||||
|
};
|
||||||
int headerSize() const;
|
int headerSize() const;
|
||||||
void prepareBuffers();
|
void prepareBuffers();
|
||||||
void finalizeEncoding();
|
void finalizeEncoding();
|
||||||
|
|
||||||
void writeHeader();
|
void writeHeader();
|
||||||
[[nodiscard]] bool readHeader(const FrameRequest &request);
|
[[nodiscard]] bool readHeader(const FrameRequest &request);
|
||||||
void writeCompressedDelta();
|
[[nodiscard]] ReadResult readCompressedFrame();
|
||||||
[[nodiscard]] bool readCompressedDelta();
|
|
||||||
|
|
||||||
QByteArray _data;
|
QByteArray _data;
|
||||||
std::vector<QByteArray> _compressedFrames;
|
std::vector<QByteArray> _compressedFrames;
|
||||||
QByteArray _compressBuffer;
|
QByteArray _compressBuffer;
|
||||||
|
QByteArray _xorCompressBuffer;
|
||||||
QSize _size;
|
QSize _size;
|
||||||
QSize _original;
|
QSize _original;
|
||||||
AlignedStorage _uncompressed;
|
AlignedStorage _uncompressed;
|
||||||
|
|
Loading…
Reference in New Issue