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
355 lines
8.8 KiB
C++
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
|