Encode cached frames to YUV420P.

This commit is contained in:
John Preston 2019-06-27 15:27:01 +02:00
parent 0b8aa880e5
commit 5628c1eee6
8 changed files with 384 additions and 174 deletions

View File

@ -174,8 +174,10 @@ void FrameDeleter::operator()(AVFrame *value) {
} }
SwscalePointer MakeSwscalePointer( SwscalePointer MakeSwscalePointer(
not_null<AVFrame*> frame, QSize srcSize,
QSize resize, int srcFormat,
QSize dstSize,
int dstFormat,
SwscalePointer *existing) { SwscalePointer *existing) {
// We have to use custom caching for SwsContext, because // We have to use custom caching for SwsContext, because
// sws_getCachedContext checks passed flags with existing context flags, // sws_getCachedContext checks passed flags with existing context flags,
@ -184,25 +186,26 @@ SwscalePointer MakeSwscalePointer(
// to the resulting context, so the caching doesn't work. // to the resulting context, so the caching doesn't work.
if (existing && (*existing) != nullptr) { if (existing && (*existing) != nullptr) {
const auto &deleter = existing->get_deleter(); const auto &deleter = existing->get_deleter();
if (deleter.resize == resize if (deleter.srcSize == srcSize
&& deleter.frameSize == QSize(frame->width, frame->height) && deleter.srcFormat == srcFormat
&& deleter.frameFormat == frame->format) { && deleter.dstSize == dstSize
&& deleter.dstFormat == dstFormat) {
return std::move(*existing); return std::move(*existing);
} }
} }
if (frame->format <= AV_PIX_FMT_NONE || frame->format >= AV_PIX_FMT_NB) { if (srcFormat <= AV_PIX_FMT_NONE || srcFormat >= AV_PIX_FMT_NB) {
LogError(qstr("frame->format")); LogError(qstr("frame->format"));
return SwscalePointer(); return SwscalePointer();
} }
const auto result = sws_getCachedContext( const auto result = sws_getCachedContext(
existing ? existing->release() : nullptr, existing ? existing->release() : nullptr,
frame->width, srcSize.width(),
frame->height, srcSize.height(),
AVPixelFormat(frame->format), AVPixelFormat(srcFormat),
resize.width(), dstSize.width(),
resize.height(), dstSize.height(),
AV_PIX_FMT_BGRA, AVPixelFormat(dstFormat),
0, 0,
nullptr, nullptr,
nullptr, nullptr,
@ -212,7 +215,19 @@ SwscalePointer MakeSwscalePointer(
} }
return SwscalePointer( return SwscalePointer(
result, result,
{ resize, QSize{ frame->width, frame->height }, frame->format }); { srcSize, srcFormat, dstSize, dstFormat });
}
SwscalePointer MakeSwscalePointer(
not_null<AVFrame*> frame,
QSize resize,
SwscalePointer *existing) {
return MakeSwscalePointer(
QSize(frame->width, frame->height),
frame->format,
resize,
AV_PIX_FMT_BGRA,
existing);
} }
void SwscaleDeleter::operator()(SwsContext *value) { void SwscaleDeleter::operator()(SwsContext *value) {

View File

@ -146,13 +146,20 @@ using FramePointer = std::unique_ptr<AVFrame, FrameDeleter>;
void ClearFrameMemory(AVFrame *frame); void ClearFrameMemory(AVFrame *frame);
struct SwscaleDeleter { struct SwscaleDeleter {
QSize resize; QSize srcSize;
QSize frameSize; int srcFormat = int(AV_PIX_FMT_NONE);
int frameFormat = int(AV_PIX_FMT_NONE); QSize dstSize;
int dstFormat = int(AV_PIX_FMT_NONE);
void operator()(SwsContext *value); void operator()(SwsContext *value);
}; };
using SwscalePointer = std::unique_ptr<SwsContext, SwscaleDeleter>; using SwscalePointer = std::unique_ptr<SwsContext, SwscaleDeleter>;
[[nodiscard]] SwscalePointer MakeSwscalePointer(
QSize srcSize,
int srcFormat,
QSize dstSize,
int dstFormat, // This field doesn't take part in caching!
SwscalePointer *existing = nullptr);
[[nodiscard]] SwscalePointer MakeSwscalePointer( [[nodiscard]] SwscalePointer MakeSwscalePointer(
not_null<AVFrame*> frame, not_null<AVFrame*> frame,
QSize resize, QSize resize,

View File

@ -13,6 +13,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "logs.h" #include "logs.h"
#include <QDataStream> #include <QDataStream>
#include <private/qdrawhelper_p.h>
#include <lz4.h> #include <lz4.h>
#include <lz4hc.h> #include <lz4hc.h>
#include <range/v3/numeric/accumulate.hpp> #include <range/v3/numeric/accumulate.hpp>
@ -22,62 +23,119 @@ namespace {
constexpr auto kAlignStorage = 16; constexpr auto kAlignStorage = 16;
void Xor(AlignedStorage &to, const AlignedStorage &from) { void Xor(EncodedStorage &to, const EncodedStorage &from) {
Expects(to.rawSize() == from.rawSize()); Expects(to.size() == from.size());
using Block = std::conditional_t< using Block = std::conditional_t<
sizeof(void*) == sizeof(uint64), sizeof(void*) == sizeof(uint64),
uint64, uint64,
uint32>; uint32>;
constexpr auto kBlockSize = sizeof(Block); constexpr auto kBlockSize = sizeof(Block);
const auto amount = from.rawSize(); const auto amount = from.size();
const auto fromBytes = reinterpret_cast<const uchar*>(from.raw()); const auto fromBytes = reinterpret_cast<const uchar*>(from.data());
const auto toBytes = reinterpret_cast<uchar*>(to.raw()); const auto toBytes = reinterpret_cast<uchar*>(to.data());
const auto skip = reinterpret_cast<quintptr>(toBytes) % kBlockSize; const auto blocks = amount / kBlockSize;
const auto blocks = (amount - skip) / kBlockSize; const auto fromBlocks = reinterpret_cast<const Block*>(fromBytes);
for (auto i = 0; i != skip; ++i) { const auto toBlocks = reinterpret_cast<Block*>(toBytes);
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) { for (auto i = 0; i != blocks; ++i) {
toBlocks[i] ^= fromBlocks[i]; toBlocks[i] ^= fromBlocks[i];
} }
const auto left = amount - skip - (blocks * kBlockSize); const auto left = amount - (blocks * kBlockSize);
for (auto i = amount - left; i != amount; ++i) { for (auto i = amount - left; i != amount; ++i) {
toBytes[i] ^= fromBytes[i]; toBytes[i] ^= fromBytes[i];
} }
} }
bool UncompressToRaw(AlignedStorage &to, bytes::const_span from) { void UnPremultiply(QImage &to, const QImage &from) {
if (from.empty() || from.size() > to.rawSize()) { // This creates QImage::Format_ARGB32_Premultiplied, but we use it
// as an image in QImage::Format_ARGB32 format.
if (!FFmpeg::GoodStorageForFrame(to, from.size())) {
to = FFmpeg::CreateFrameStorage(from.size());
}
const auto layout = &qPixelLayouts[QImage::Format_ARGB32];
const auto convert = layout->convertFromARGB32PM;
const auto fromPerLine = from.bytesPerLine();
const auto toPerLine = to.bytesPerLine();
const auto width = from.width();
if (fromPerLine != width * 4 || toPerLine != width * 4) {
auto fromBytes = from.bits();
auto toBytes = to.bits();
for (auto i = 0; i != to.height(); ++i) {
convert(
reinterpret_cast<uint*>(toBytes),
reinterpret_cast<const uint*>(fromBytes),
width,
layout,
nullptr);
fromBytes += fromPerLine;
toBytes += toPerLine;
}
} else {
convert(
reinterpret_cast<uint*>(to.bits()),
reinterpret_cast<const uint*>(from.bits()),
from.width() * from.height(),
layout,
nullptr);
}
}
void PremultiplyInplace(QImage &image) {
const auto layout = &qPixelLayouts[QImage::Format_ARGB32];
const auto convert = layout->convertToARGB32PM;
const auto perLine = image.bytesPerLine();
const auto width = image.width();
if (perLine != width * 4) {
auto bytes = image.bits();
for (auto i = 0; i != image.height(); ++i) {
convert(
reinterpret_cast<uint*>(bytes),
reinterpret_cast<const uint*>(bytes),
width,
layout,
nullptr);
bytes += perLine;
}
} else {
convert(
reinterpret_cast<uint*>(image.bits()),
reinterpret_cast<const uint*>(image.bits()),
image.width() * image.height(),
layout,
nullptr);
}
}
bool UncompressToRaw(EncodedStorage &to, bytes::const_span from) {
if (from.empty() || from.size() > to.size()) {
return false; return false;
} else if (from.size() == to.rawSize()) { } else if (from.size() == to.size()) {
memcpy(to.raw(), from.data(), from.size()); memcpy(to.data(), from.data(), from.size());
return true; return true;
} }
const auto result = LZ4_decompress_safe( const auto result = LZ4_decompress_safe(
reinterpret_cast<const char*>(from.data()), reinterpret_cast<const char*>(from.data()),
static_cast<char*>(to.raw()), to.data(),
from.size(), from.size(),
to.rawSize()); to.size());
return (result == to.rawSize()); return (result == to.size());
} }
void CompressFromRaw(QByteArray &to, const AlignedStorage &from) { void CompressFromRaw(QByteArray &to, const EncodedStorage &from) {
const auto size = from.rawSize(); const auto size = from.size();
const auto max = sizeof(qint32) + 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()), from.data(),
to.data() + sizeof(qint32), to.data() + sizeof(qint32),
size, size,
to.size() - sizeof(qint32)); to.size() - sizeof(qint32));
Assert(compressed > 0); Assert(compressed > 0);
if (compressed >= size + sizeof(qint32)) { if (compressed >= size + sizeof(qint32)) {
to.resize(size + sizeof(qint32)); to.resize(size + sizeof(qint32));
memcpy(to.data() + sizeof(qint32), from.raw(), size); memcpy(to.data() + sizeof(qint32), from.data(), size);
} else { } else {
to.resize(compressed + sizeof(qint32)); to.resize(compressed + sizeof(qint32));
} }
@ -90,8 +148,8 @@ void CompressFromRaw(QByteArray &to, const AlignedStorage &from) {
void CompressAndSwapFrame( void CompressAndSwapFrame(
QByteArray &to, QByteArray &to,
QByteArray *additional, QByteArray *additional,
AlignedStorage &frame, EncodedStorage &frame,
AlignedStorage &previous) { EncodedStorage &previous) {
CompressFromRaw(to, frame); CompressFromRaw(to, frame);
std::swap(frame, previous); std::swap(frame, previous);
if (!additional) { if (!additional) {
@ -113,116 +171,232 @@ void CompressAndSwapFrame(
bytes::object_as_span(&negativeLength)); bytes::object_as_span(&negativeLength));
} }
void Decode(QImage &to, AlignedStorage &from, const QSize &fromSize) { void DecodeYUV2RGB(
from.copyRawToAligned(); QImage &to,
if (!FFmpeg::GoodStorageForFrame(to, fromSize)) { const EncodedStorage &from,
to = FFmpeg::CreateFrameStorage(fromSize); FFmpeg::SwscalePointer &context) {
} context = FFmpeg::MakeSwscalePointer(
auto fromBytes = static_cast<const char*>(from.aligned()); to.size(),
auto toBytes = to.bits(); AV_PIX_FMT_YUV420P,
const auto fromPerLine = from.bytesPerLine(); to.size(),
const auto toPerLine = to.bytesPerLine(); AV_PIX_FMT_BGRA,
for (auto i = 0; i != to.height(); ++i) { &context);
memcpy(toBytes, fromBytes, to.width() * 4); Assert(context != nullptr);
fromBytes += fromPerLine;
toBytes += toPerLine; // AV_NUM_DATA_POINTERS defined in AVFrame struct
const uint8_t *src[AV_NUM_DATA_POINTERS] = {
from.yData(),
from.uData(),
from.vData(),
nullptr
};
int srcLineSize[AV_NUM_DATA_POINTERS] = {
from.yBytesPerLine(),
from.uBytesPerLine(),
from.vBytesPerLine(),
0
};
uint8_t *dst[AV_NUM_DATA_POINTERS] = { to.bits(), nullptr };
int dstLineSize[AV_NUM_DATA_POINTERS] = { to.bytesPerLine(), 0 };
const auto lines = sws_scale(
context.get(),
src,
srcLineSize,
0,
to.height(),
dst,
dstLineSize);
Ensures(lines == to.height());
}
void DecodeAlpha(QImage &to, const EncodedStorage &from) {
auto bytes = to.bits();
auto alpha = from.aData();
const auto perLine = to.bytesPerLine();
const auto width = to.width();
const auto height = to.height();
for (auto i = 0; i != height; ++i) {
auto ints = reinterpret_cast<uint32*>(bytes);
const auto till = ints + width;
while (ints != till) {
const auto value = uint32(*alpha++);
*ints = (*ints & 0x00FFFFFFU) | ((value & 0xF0U) << 24);
++ints;
*ints = (*ints & 0x00FFFFFFU) | (value << 28);
++ints;
}
bytes += perLine;
} }
} }
void Encode(AlignedStorage &to, const QImage &from, const QSize &toSize) { void Decode(
auto fromBytes = from.bits(); QImage &to,
auto toBytes = static_cast<char*>(to.aligned()); const EncodedStorage &from,
const auto fromPerLine = from.bytesPerLine(); const QSize &fromSize,
const auto toPerLine = to.bytesPerLine(); FFmpeg::SwscalePointer &context) {
for (auto i = 0; i != to.lines(); ++i) { if (!FFmpeg::GoodStorageForFrame(to, fromSize)) {
memcpy(toBytes, fromBytes, from.width() * 4); to = FFmpeg::CreateFrameStorage(fromSize);
fromBytes += fromPerLine;
toBytes += toPerLine;
} }
to.copyAlignedToRaw(); DecodeYUV2RGB(to, from, context);
DecodeAlpha(to, from);
PremultiplyInplace(to);
}
void EncodeRGB2YUV(
EncodedStorage &to,
const QImage &from,
FFmpeg::SwscalePointer &context) {
context = FFmpeg::MakeSwscalePointer(
from.size(),
AV_PIX_FMT_BGRA,
from.size(),
AV_PIX_FMT_YUV420P,
&context);
Assert(context != nullptr);
// AV_NUM_DATA_POINTERS defined in AVFrame struct
const uint8_t *src[AV_NUM_DATA_POINTERS] = { from.bits(), nullptr };
int srcLineSize[AV_NUM_DATA_POINTERS] = { from.bytesPerLine(), 0 };
uint8_t *dst[AV_NUM_DATA_POINTERS] = {
to.yData(),
to.uData(),
to.vData(),
nullptr
};
int dstLineSize[AV_NUM_DATA_POINTERS] = {
to.yBytesPerLine(),
to.uBytesPerLine(),
to.vBytesPerLine(),
0
};
const auto lines = sws_scale(
context.get(),
src,
srcLineSize,
0,
from.height(),
dst,
dstLineSize);
Ensures(lines == from.height());
}
void EncodeAlpha(EncodedStorage &to, const QImage &from) {
auto bytes = from.bits();
auto alpha = to.aData();
const auto perLine = from.bytesPerLine();
const auto width = from.width();
const auto height = from.height();
for (auto i = 0; i != height; ++i) {
auto ints = reinterpret_cast<const uint32*>(bytes);
const auto till = ints + width;
for (; ints != till; ints += 2) {
*alpha++ = (((*ints) >> 24) & 0xF0U) | ((*(ints + 1)) >> 28);
}
bytes += perLine;
}
}
void Encode(
EncodedStorage &to,
const QImage &from,
QImage &cache,
FFmpeg::SwscalePointer &context) {
UnPremultiply(cache, from);
EncodeRGB2YUV(to, cache, context);
EncodeAlpha(to, cache);
} }
} // namespace } // namespace
void AlignedStorage::allocate(int packedBytesPerLine, int lines) { void EncodedStorage::allocate(int width, int height) {
Expects(packedBytesPerLine >= 0); Expects((width % 2) == 0 && (height % 2) == 0);
Expects(lines >= 0);
_packedBytesPerLine = packedBytesPerLine; if (_width != width || _height != height) {
_lines = lines; _width = width;
reallocate(); _height = height;
reallocate();
}
} }
void AlignedStorage::reallocate() { void EncodedStorage::reallocate() {
const auto perLine = bytesPerLine(); const auto total = _width * _height * 2;
const auto total = perLine * _lines; _data = QByteArray(total + kAlignStorage - 1, Qt::Uninitialized);
_buffer = QByteArray(total + kAlignStorage - 1, Qt::Uninitialized);
_raw = (perLine != _packedBytesPerLine)
? QByteArray(_packedBytesPerLine * _lines, Qt::Uninitialized)
: QByteArray();
} }
int AlignedStorage::lines() const { int EncodedStorage::width() const {
return _lines; return _width;
} }
int AlignedStorage::rawSize() const { int EncodedStorage::height() const {
return _lines * _packedBytesPerLine; return _height;
} }
void *AlignedStorage::raw() { int EncodedStorage::size() const {
return (bytesPerLine() == _packedBytesPerLine) ? aligned() : _raw.data(); return _width * _height * 2;
} }
const void *AlignedStorage::raw() const { char *EncodedStorage::data() {
return (bytesPerLine() == _packedBytesPerLine) ? aligned() : _raw.data(); const auto result = reinterpret_cast<quintptr>(_data.data());
} return reinterpret_cast<char*>(kAlignStorage
int AlignedStorage::bytesPerLine() const {
return kAlignStorage
* ((_packedBytesPerLine + kAlignStorage - 1) / kAlignStorage);
}
void *AlignedStorage::aligned() {
const auto result = reinterpret_cast<quintptr>(_buffer.data());
return reinterpret_cast<void*>(kAlignStorage
* ((result + kAlignStorage - 1) / kAlignStorage)); * ((result + kAlignStorage - 1) / kAlignStorage));
} }
const void *AlignedStorage::aligned() const { const char *EncodedStorage::data() const {
const auto result = reinterpret_cast<quintptr>(_buffer.data()); const auto result = reinterpret_cast<quintptr>(_data.data());
return reinterpret_cast<void*>(kAlignStorage return reinterpret_cast<const char*>(kAlignStorage
* ((result + kAlignStorage - 1) / kAlignStorage)); * ((result + kAlignStorage - 1) / kAlignStorage));
} }
void AlignedStorage::copyRawToAligned() { uint8_t *EncodedStorage::yData() {
const auto fromPerLine = _packedBytesPerLine; return reinterpret_cast<uint8_t*>(data());
const auto toPerLine = bytesPerLine();
if (fromPerLine == toPerLine) {
return;
}
auto from = static_cast<const char*>(raw());
auto to = static_cast<char*>(aligned());
for (auto i = 0; i != _lines; ++i) {
memcpy(to, from, fromPerLine);
from += fromPerLine;
to += toPerLine;
}
} }
void AlignedStorage::copyAlignedToRaw() { const uint8_t *EncodedStorage::yData() const {
const auto fromPerLine = bytesPerLine(); return reinterpret_cast<const uint8_t*>(data());
const auto toPerLine = _packedBytesPerLine; }
if (fromPerLine == toPerLine) {
return; int EncodedStorage::yBytesPerLine() const {
} return _width;
auto from = static_cast<const char*>(aligned()); }
auto to = static_cast<char*>(raw());
for (auto i = 0; i != _lines; ++i) { uint8_t *EncodedStorage::uData() {
memcpy(to, from, toPerLine); return yData() + (_width * _height);
from += fromPerLine; }
to += toPerLine;
} const uint8_t *EncodedStorage::uData() const {
return yData() + (_width * _height);
}
int EncodedStorage::uBytesPerLine() const {
return _width / 2;
}
uint8_t *EncodedStorage::vData() {
return uData() + (_width * _height / 4);
}
const uint8_t *EncodedStorage::vData() const {
return uData() + (_width * _height / 4);
}
int EncodedStorage::vBytesPerLine() const {
return _width / 2;
}
uint8_t *EncodedStorage::aData() {
return uData() + (_width * _height) / 2;
}
const uint8_t *EncodedStorage::aData() const {
return uData() + (_width * _height) / 2;
}
int EncodedStorage::aBytesPerLine() const {
return _width / 2;
} }
CacheState::CacheState(const QByteArray &data, const FrameRequest &request) CacheState::CacheState(const QByteArray &data, const FrameRequest &request)
@ -338,7 +512,7 @@ bool CacheState::renderFrame(
} else { } else {
std::swap(_uncompressed, _previous); std::swap(_uncompressed, _previous);
} }
Decode(to, _previous, _size); Decode(to, _previous, _size, _decodeContext);
return true; return true;
} }
@ -355,17 +529,19 @@ void CacheState::appendFrame(
} }
if (index == 0) { if (index == 0) {
_size = request.size(_original); _size = request.size(_original);
_compressedFrames.reserve(_framesCount); _encode = EncodeFields();
_encode.compressedFrames.reserve(_framesCount);
prepareBuffers(); prepareBuffers();
} }
Encode(_uncompressed, frame, _size); Assert(frame.size() == _size);
Encode(_uncompressed, frame, _encode.cache, _encode.context);
CompressAndSwapFrame( CompressAndSwapFrame(
_compressBuffer, _encode.compressBuffer,
(index != 0) ? &_xorCompressBuffer : nullptr, (index != 0) ? &_encode.xorCompressBuffer : nullptr,
_uncompressed, _uncompressed,
_previous); _previous);
_compressedFrames.push_back(_compressBuffer); _encode.compressedFrames.push_back(_encode.compressBuffer);
_compressedFrames.back().detach(); _encode.compressedFrames.back().detach();
if (++_framesReady == _framesCount) { if (++_framesReady == _framesCount) {
finalizeEncoding(); finalizeEncoding();
} }
@ -374,7 +550,7 @@ 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())
+ ranges::accumulate( + ranges::accumulate(
_compressedFrames, _encode.compressedFrames,
0, 0,
std::plus(), std::plus(),
&QByteArray::size); &QByteArray::size);
@ -386,7 +562,7 @@ void CacheState::finalizeEncoding() {
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 : _encode.compressedFrames) {
const auto amount = qint32(block.size()); const auto amount = qint32(block.size());
memcpy(to, block.data(), amount); memcpy(to, block.data(), amount);
if (*reinterpret_cast<const qint32*>(block.data()) < 0) { if (*reinterpret_cast<const qint32*>(block.data()) < 0) {
@ -394,11 +570,8 @@ void CacheState::finalizeEncoding() {
} }
to += amount; to += amount;
} }
_compressedFrames.clear(); _encode = EncodeFields();
_compressedFrames.shrink_to_fit(); constexpr auto test = sizeof(_encode);
_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)); LOG(("SIZE: %1 (%2x%3, %4 frames, %5 xored)").arg(_data.size()).arg(_size.width()).arg(_size.height()).arg(_framesCount).arg(xored));
} }
@ -421,8 +594,11 @@ void CacheState::writeHeader() {
} }
void CacheState::prepareBuffers() { void CacheState::prepareBuffers() {
_uncompressed.allocate(_size.width() * 4, _size.height()); // 12 bit per pixel in YUV420P.
_previous.allocate(_size.width() * 4, _size.height()); const auto bytesPerLine = _size.width();
_uncompressed.allocate(bytesPerLine, _size.height());
_previous.allocate(bytesPerLine, _size.height());
} }
CacheState::ReadResult CacheState::readCompressedFrame() { CacheState::ReadResult CacheState::readCompressedFrame() {

View File

@ -7,6 +7,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#pragma once #pragma once
#include "ffmpeg/ffmpeg_utility.h"
#include <QImage> #include <QImage>
#include <QSize> #include <QSize>
#include <QByteArray> #include <QByteArray>
@ -15,34 +17,36 @@ namespace Lottie {
struct FrameRequest; struct FrameRequest;
class AlignedStorage { class EncodedStorage {
public: public:
void allocate(int packedBytesPerLine, int lines); void allocate(int width, int height);
int lines() const; int width() const;
int rawSize() const; int height() const;
// Gives a pointer to packedBytesPerLine * lines bytes of memory. char *data();
void *raw(); const char *data() const;
const void *raw() const; int size() const;
// Gives a stride value in the aligned storage (% 16 == 0). uint8_t *yData();
int bytesPerLine() const; const uint8_t *yData() const;
int yBytesPerLine() const;
// Gives a pointer to the aligned memory (% 16 == 0). uint8_t *uData();
void *aligned(); const uint8_t *uData() const;
const void *aligned() const; int uBytesPerLine() const;
uint8_t *vData();
void copyRawToAligned(); const uint8_t *vData() const;
void copyAlignedToRaw(); int vBytesPerLine() const;
uint8_t *aData();
const uint8_t *aData() const;
int aBytesPerLine() const;
private: private:
void reallocate(); void reallocate();
int _packedBytesPerLine = 0; int _width = 0;
int _lines = 0; int _height = 0;
QByteArray _raw; QByteArray _data;
QByteArray _buffer;
}; };
@ -79,6 +83,13 @@ private:
bool ok = false; bool ok = false;
bool xored = false; bool xored = false;
}; };
struct EncodeFields {
std::vector<QByteArray> compressedFrames;
QByteArray compressBuffer;
QByteArray xorCompressBuffer;
QImage cache;
FFmpeg::SwscalePointer context;
};
int headerSize() const; int headerSize() const;
void prepareBuffers(); void prepareBuffers();
void finalizeEncoding(); void finalizeEncoding();
@ -88,13 +99,12 @@ private:
[[nodiscard]] ReadResult readCompressedFrame(); [[nodiscard]] ReadResult readCompressedFrame();
QByteArray _data; QByteArray _data;
std::vector<QByteArray> _compressedFrames; EncodeFields _encode;
QByteArray _compressBuffer;
QByteArray _xorCompressBuffer;
QSize _size; QSize _size;
QSize _original; QSize _original;
AlignedStorage _uncompressed; EncodedStorage _uncompressed;
AlignedStorage _previous; EncodedStorage _previous;
FFmpeg::SwscalePointer _decodeContext;
QImage _firstFrame; QImage _firstFrame;
int _frameRate = 0; int _frameRate = 0;
int _framesCount = 0; int _framesCount = 0;

View File

@ -54,12 +54,14 @@ struct FrameRequest {
return box.isEmpty(); return box.isEmpty();
} }
[[nodiscard]] QSize size(const QSize &original) const { [[nodiscard]] QSize size(const QSize &original) const {
Expects(!box.isEmpty()); Expects(!empty());
const auto result = original.scaled(box, Qt::KeepAspectRatio); const auto result = original.scaled(box, Qt::KeepAspectRatio);
const auto skipw = result.width() % 2;
const auto skiph = result.height() % 2;
return QSize( return QSize(
std::max(result.width(), 1), std::max(result.width() - skipw, 2),
std::max(result.height(), 1)); std::max(result.height() - skiph, 2));
} }
[[nodiscard]] bool operator==(const FrameRequest &other) const { [[nodiscard]] bool operator==(const FrameRequest &other) const {

View File

@ -7,8 +7,6 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#include "ui/text/text.h" #include "ui/text/text.h"
#include <private/qharfbuzz_p.h>
#include "core/click_handler_types.h" #include "core/click_handler_types.h"
#include "core/crash_reports.h" #include "core/crash_reports.h"
#include "ui/text/text_block.h" #include "ui/text/text_block.h"
@ -18,6 +16,8 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
#include "mainwindow.h" #include "mainwindow.h"
#include <private/qharfbuzz_p.h>
namespace Ui { namespace Ui {
namespace Text { namespace Text {
namespace { namespace {

View File

@ -7,12 +7,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#pragma once #pragma once
#include "private/qfontengine_p.h"
#include "core/click_handler.h" #include "core/click_handler.h"
#include "ui/text/text_entity.h" #include "ui/text/text_entity.h"
#include "base/flags.h" #include "base/flags.h"
#include <private/qfontengine_p.h>
static const QChar TextCommand(0x0010); static const QChar TextCommand(0x0010);
enum TextCommands { enum TextCommands {
TextCommandBold = 0x01, TextCommandBold = 0x01,

View File

@ -7,7 +7,7 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
*/ */
#pragma once #pragma once
#include "private/qfontengine_p.h" #include <private/qfontengine_p.h>
namespace Ui { namespace Ui {
namespace Text { namespace Text {