mirror of https://github.com/procxx/kepka.git
Encode cached frames to YUV420P.
This commit is contained in:
parent
0b8aa880e5
commit
5628c1eee6
|
@ -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) {
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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() {
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
|
@ -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,
|
||||||
|
|
|
@ -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 {
|
||||||
|
|
Loading…
Reference in New Issue