Files
tdesktop/Telegram/lib_lottie/lottie/details/lottie_cache.cpp
allhaileris afb81b8278
Some checks failed
Docker. / Ubuntu (push) Has been cancelled
User-agent updater. / User-agent (push) Failing after 15s
Lock Threads / lock (push) Failing after 10s
Waiting for answer. / waiting-for-answer (push) Failing after 22s
Close stale issues and PRs / stale (push) Successful in 13s
Needs user action. / needs-user-action (push) Failing after 8s
Can't reproduce. / cant-reproduce (push) Failing after 8s
init
2026-02-16 15:50:16 +03:00

355 lines
8.8 KiB
C++

// This file is part of Desktop App Toolkit,
// a set of libraries for developing nice desktop applications.
//
// For license and copyright information please follow this link:
// https://github.com/desktop-app/legal/blob/master/LEGAL
//
#include "lottie/details/lottie_cache.h"
#include "lottie/details/lottie_frame_renderer.h"
#include "ffmpeg/ffmpeg_utility.h"
#include "base/bytes.h"
#include "base/assertion.h"
#include <QDataStream>
#include <QIODevice>
#include <range/v3/numeric/accumulate.hpp>
namespace Lottie {
namespace {
// Must not exceed max database allowed entry size.
constexpr auto kMaxCacheSize = 10 * 1024 * 1024;
} // namespace
Cache::Cache(
const QByteArray &data,
const FrameRequest &request,
FnMut<void(QByteArray &&cached)> put)
: _data(data)
, _put(std::move(put)) {
if (!readHeader(request)) {
_framesReady = 0;
_framesInData = 0;
_data = QByteArray();
}
}
Cache::Cache(Cache &&) = default;
Cache &Cache::operator=(Cache&&) = default;
Cache::~Cache() {
finalizeEncoding();
}
void Cache::init(
QSize original,
int frameRate,
int framesCount,
const FrameRequest &request) {
_size = request.size(original, sizeRounding());
_original = original;
_frameRate = frameRate;
_framesCount = framesCount;
_framesReady = 0;
_framesInData = 0;
prepareBuffers();
}
int Cache::sizeRounding() const {
return 8;
}
int Cache::frameRate() const {
return _frameRate;
}
int Cache::framesReady() const {
return _framesReady;
}
int Cache::framesCount() const {
return _framesCount;
}
QSize Cache::originalSize() const {
return _original;
}
bool Cache::readHeader(const FrameRequest &request) {
if (_data.isEmpty()) {
return false;
}
QDataStream stream(&_data, QIODevice::ReadOnly);
auto encoder = qint32(0);
stream >> encoder;
if (static_cast<Encoder>(encoder) != Encoder::YUV420A4_LZ4) {
return false;
}
auto size = QSize();
auto original = QSize();
auto frameRate = qint32(0);
auto framesCount = qint32(0);
auto framesReady = qint32(0);
stream
>> size
>> original
>> frameRate
>> framesCount
>> framesReady;
if (stream.status() != QDataStream::Ok
|| original.isEmpty()
|| (original.width() > kMaxSize)
|| (original.height() > kMaxSize)
|| (frameRate <= 0)
|| (frameRate > kNormalFrameRate && frameRate != kMaxFrameRate)
|| (framesCount <= 0)
|| (framesCount > kMaxFramesCount)
|| (framesReady <= 0)
|| (framesReady > framesCount)
|| request.size(original, sizeRounding()) != size) {
return false;
}
_encoder = static_cast<Encoder>(encoder);
_size = size;
_original = original;
_frameRate = frameRate;
_framesCount = framesCount;
_framesReady = framesReady;
_framesInData = framesReady;
prepareBuffers();
return (renderFrame(_firstFrame, request, 0) == FrameRenderResult::Ok);
}
QImage Cache::takeFirstFrame() {
return std::move(_firstFrame);
}
FrameRenderResult Cache::renderFrame(
QImage &to,
const FrameRequest &request,
int index) {
const auto result = renderFrame(_readContext, to, request, index);
if (result == FrameRenderResult::Ok) {
if (index + 1 == _framesReady && _data.size() > _readContext.offset) {
_data.resize(_readContext.offset);
}
} else if (result == FrameRenderResult::BadCacheSize) {
_framesReady = 0;
_framesInData = 0;
_data.clear();
}
return result;
}
FrameRenderResult Cache::renderFrame(
CacheReadContext &context,
QImage &to,
const FrameRequest &request,
int index) const {
Expects(index >= _framesReady
|| index == context.offsetFrameIndex
|| index == 0);
Expects(index >= _framesReady || context.ready());
if (index >= _framesReady) {
return FrameRenderResult::NotReady;
} else if (request.size(_original, sizeRounding()) != _size) {
return FrameRenderResult::BadCacheSize;
} else if (index == 0) {
context.offsetFrameIndex = 0;
context.offset = headerSize();
}
const auto [ok, xored] = readCompressedFrame(context);
if (!ok || (xored && index == 0)) {
return FrameRenderResult::Failed;
}
if (xored) {
Xor(context.previous, context.uncompressed);
} else {
std::swap(context.uncompressed, context.previous);
}
Decode(to, context.previous, _size, context.decodeContext);
return FrameRenderResult::Ok;
}
void Cache::appendFrame(
const QImage &frame,
const FrameRequest &request,
int index) {
if (request.size(_original, sizeRounding()) != _size) {
_framesReady = 0;
_framesInData = 0;
_data = QByteArray();
}
if (index != _framesReady) {
return;
} else if (index == 0) {
_size = request.size(_original, sizeRounding());
_encode = EncodeFields();
_encode.compressedFrames.reserve(_framesCount);
prepareBuffers();
}
Assert(frame.size() == _size);
Assert(_readContext.ready());
Encode(_readContext.uncompressed, frame, _encode.cache, _encode.context);
CompressAndSwapFrame(
_encode.compressBuffer,
(index != 0) ? &_encode.xorCompressBuffer : nullptr,
_readContext.uncompressed,
_readContext.previous);
const auto compressed = _encode.compressBuffer;
const auto nowSize = (_data.isEmpty() ? headerSize() : _data.size())
+ _encode.totalSize;
const auto totalSize = nowSize + compressed.size();
if (nowSize <= kMaxCacheSize && totalSize > kMaxCacheSize) {
// Write to cache while we still can.
finalizeEncoding();
}
_encode.totalSize += compressed.size();
_encode.compressedFrames.push_back(compressed);
_encode.compressedFrames.back().detach();
++_readContext.offsetFrameIndex;
_readContext.offset += compressed.size();
if (++_framesReady == _framesCount) {
finalizeEncoding();
}
}
void Cache::finalizeEncoding() {
if (_encode.compressedFrames.empty() || !_put) {
return;
}
const auto size = (_data.isEmpty() ? headerSize() : _data.size())
+ _encode.totalSize;
if (_data.isEmpty()) {
_data.reserve(size);
writeHeader();
} else {
updateFramesReadyCount();
}
const auto offset = _data.size();
_data.resize(size);
auto to = _data.data() + offset;
for (const auto &block : _encode.compressedFrames) {
const auto amount = qint32(block.size());
memcpy(to, block.data(), amount);
to += amount;
}
_framesInData = _framesReady;
if (_data.size() <= kMaxCacheSize) {
_put(QByteArray(_data));
}
_encode = EncodeFields();
}
int Cache::headerSize() const {
return 8 * sizeof(qint32);
}
void Cache::writeHeader() {
Expects(_data.isEmpty());
QDataStream stream(&_data, QIODevice::WriteOnly);
stream
<< static_cast<qint32>(_encoder)
<< _size
<< _original
<< qint32(_frameRate)
<< qint32(_framesCount)
<< qint32(_framesReady);
}
void Cache::updateFramesReadyCount() {
Expects(_data.size() >= headerSize());
QDataStream stream(&_data, QIODevice::ReadWrite);
stream.device()->seek(headerSize() - sizeof(qint32));
stream << qint32(_framesReady);
}
void Cache::prepareBuffers() {
prepareBuffers(_readContext);
}
void Cache::prepareBuffers(CacheReadContext &context) const {
if (_size.isEmpty()) {
return;
}
// 12 bit per pixel in YUV420P.
const auto bytesPerLine = _size.width();
context.offset = headerSize();
context.offsetFrameIndex = 0;
context.uncompressed.allocate(bytesPerLine, _size.height());
context.previous.allocate(bytesPerLine, _size.height());
}
void Cache::keepUpContext(CacheReadContext &context) const {
Expects(!context.ready()
|| context.previous.size() == _readContext.previous.size());
Expects(!context.ready()
|| context.uncompressed.size() == _readContext.uncompressed.size());
Expects(&context != &_readContext);
if (!context.ready()) {
prepareBuffers(context);
}
context.offset = _readContext.offset;
context.offsetFrameIndex = _readContext.offsetFrameIndex;
memcpy(
context.previous.data(),
_readContext.previous.data(),
_readContext.previous.size());
memcpy(
context.uncompressed.data(),
_readContext.uncompressed.data(),
_readContext.uncompressed.size());
}
Cache::ReadResult Cache::readCompressedFrame(
CacheReadContext &context) const {
Expects(context.ready());
auto length = qint32(0);
const auto part = [&] {
if (context.offsetFrameIndex >= _framesInData) {
// One reader is still accumulating compressed frames,
// while second reader already started reading after the first.
const auto readyIndex = context.offsetFrameIndex - _framesInData;
Assert(readyIndex < _encode.compressedFrames.size());
return bytes::make_span(_encode.compressedFrames[readyIndex]);
} else if (_data.size() > context.offset) {
return bytes::make_span(_data).subspan(context.offset);
} else {
return bytes::const_span();
}
}();
if (part.size() < sizeof(length)) {
return { false };
}
bytes::copy(
bytes::object_as_span(&length),
part.subspan(0, sizeof(length)));
const auto bytes = part.subspan(sizeof(length));
const auto xored = (length < 0);
if (xored) {
length = -length;
}
const auto ok = (length <= bytes.size())
? UncompressToRaw(context.uncompressed, bytes.subspan(0, length))
: false;
if (ok) {
context.offset += sizeof(length) + length;
++context.offsetFrameIndex;
}
return { ok, xored };
}
} // namespace Lottie