349 lines
8.6 KiB
C++
349 lines
8.6 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "media/streaming/media_streaming_utility.h"
|
|
|
|
#include "media/streaming/media_streaming_common.h"
|
|
#include "ui/image/image_prepare.h"
|
|
#include "ffmpeg/ffmpeg_utility.h"
|
|
|
|
namespace Media {
|
|
namespace Streaming {
|
|
namespace {
|
|
|
|
constexpr auto kSkipInvalidDataPackets = 10;
|
|
|
|
} // namespace
|
|
|
|
crl::time FramePosition(const Stream &stream) {
|
|
const auto pts = !stream.decodedFrame
|
|
? AV_NOPTS_VALUE
|
|
: (stream.decodedFrame->best_effort_timestamp != AV_NOPTS_VALUE)
|
|
? stream.decodedFrame->best_effort_timestamp
|
|
: (stream.decodedFrame->pts != AV_NOPTS_VALUE)
|
|
? stream.decodedFrame->pts
|
|
: stream.decodedFrame->pkt_dts;
|
|
return FFmpeg::PtsToTime(pts, stream.timeBase);
|
|
}
|
|
|
|
FFmpeg::AvErrorWrap ProcessPacket(Stream &stream, FFmpeg::Packet &&packet) {
|
|
Expects(stream.codec != nullptr);
|
|
|
|
auto error = FFmpeg::AvErrorWrap();
|
|
|
|
const auto native = &packet.fields();
|
|
const auto guard = gsl::finally([
|
|
&,
|
|
size = native->size,
|
|
data = native->data
|
|
] {
|
|
native->size = size;
|
|
native->data = data;
|
|
packet = FFmpeg::Packet();
|
|
});
|
|
|
|
error = avcodec_send_packet(
|
|
stream.codec.get(),
|
|
native->data ? native : nullptr); // Drain on eof.
|
|
if (error) {
|
|
LogError(qstr("avcodec_send_packet"), error);
|
|
if (error.code() == AVERROR_INVALIDDATA
|
|
// There is a sample voice message where skipping such packet
|
|
// results in a crash (read_access to nullptr) in swr_convert().
|
|
&& stream.codec->codec_id != AV_CODEC_ID_OPUS) {
|
|
if (++stream.invalidDataPackets < kSkipInvalidDataPackets) {
|
|
return FFmpeg::AvErrorWrap(); // Try to skip a bad packet.
|
|
}
|
|
}
|
|
} else {
|
|
stream.invalidDataPackets = 0;
|
|
}
|
|
return error;
|
|
}
|
|
|
|
FFmpeg::AvErrorWrap ReadNextFrame(Stream &stream) {
|
|
Expects(stream.decodedFrame != nullptr);
|
|
|
|
auto error = FFmpeg::AvErrorWrap();
|
|
|
|
do {
|
|
error = avcodec_receive_frame(
|
|
stream.codec.get(),
|
|
stream.decodedFrame.get());
|
|
if (!error
|
|
|| error.code() != AVERROR(EAGAIN)
|
|
|| stream.queue.empty()) {
|
|
return error;
|
|
}
|
|
|
|
error = ProcessPacket(stream, std::move(stream.queue.front()));
|
|
stream.queue.pop_front();
|
|
} while (!error);
|
|
|
|
return error;
|
|
}
|
|
|
|
bool GoodForRequest(
|
|
const QImage &image,
|
|
bool hasAlpha,
|
|
int rotation,
|
|
const FrameRequest &request) {
|
|
if (image.isNull()
|
|
|| (hasAlpha && !request.keepAlpha)
|
|
|| request.colored.alpha() != 0) {
|
|
return false;
|
|
} else if (request.resize.isEmpty()) {
|
|
return true;
|
|
} else if (rotation != 0) {
|
|
return false;
|
|
} else if ((request.radius != ImageRoundRadius::None)
|
|
&& ((request.corners & RectPart::AllCorners) != 0)) {
|
|
return false;
|
|
}
|
|
return (request.resize == request.outer)
|
|
&& (request.resize == image.size());
|
|
}
|
|
|
|
bool TransferFrame(
|
|
Stream &stream,
|
|
not_null<AVFrame*> decodedFrame,
|
|
not_null<AVFrame*> transferredFrame) {
|
|
Expects(decodedFrame->hw_frames_ctx != nullptr);
|
|
|
|
const auto error = FFmpeg::AvErrorWrap(
|
|
av_hwframe_transfer_data(transferredFrame, decodedFrame, 0));
|
|
if (error) {
|
|
LogError(qstr("av_hwframe_transfer_data"), error);
|
|
return false;
|
|
}
|
|
FFmpeg::ClearFrameMemory(decodedFrame);
|
|
return true;
|
|
}
|
|
|
|
QImage ConvertFrame(
|
|
Stream &stream,
|
|
not_null<AVFrame*> frame,
|
|
QSize resize,
|
|
QImage storage) {
|
|
const auto frameSize = QSize(frame->width, frame->height);
|
|
if (frameSize.isEmpty()) {
|
|
LOG(("Streaming Error: Bad frame size %1,%2"
|
|
).arg(frameSize.width()
|
|
).arg(frameSize.height()));
|
|
return QImage();
|
|
} else if (!FFmpeg::FrameHasData(frame)) {
|
|
LOG(("Streaming Error: Bad frame data."));
|
|
return QImage();
|
|
}
|
|
if (resize.isEmpty()) {
|
|
resize = frameSize;
|
|
} else if (FFmpeg::RotationSwapWidthHeight(stream.rotation)) {
|
|
resize.transpose();
|
|
}
|
|
|
|
if (!FFmpeg::GoodStorageForFrame(storage, resize)) {
|
|
storage = FFmpeg::CreateFrameStorage(resize);
|
|
}
|
|
|
|
const auto format = AV_PIX_FMT_BGRA;
|
|
const auto hasDesiredFormat = (frame->format == format);
|
|
if (frameSize == storage.size() && hasDesiredFormat) {
|
|
static_assert(sizeof(uint32) == FFmpeg::kPixelBytesSize);
|
|
auto to = reinterpret_cast<uint32*>(storage.bits());
|
|
auto from = reinterpret_cast<const uint32*>(frame->data[0]);
|
|
const auto deltaTo = (storage.bytesPerLine() / sizeof(uint32))
|
|
- storage.width();
|
|
const auto deltaFrom = (frame->linesize[0] / sizeof(uint32))
|
|
- frame->width;
|
|
for ([[maybe_unused]] const auto y : ranges::views::ints(0, frame->height)) {
|
|
for ([[maybe_unused]] const auto x : ranges::views::ints(0, frame->width)) {
|
|
// Wipe out possible alpha values.
|
|
*to++ = 0xFF000000U | *from++;
|
|
}
|
|
to += deltaTo;
|
|
from += deltaFrom;
|
|
}
|
|
} else {
|
|
stream.swscale = MakeSwscalePointer(
|
|
frame,
|
|
resize,
|
|
&stream.swscale);
|
|
if (!stream.swscale) {
|
|
return QImage();
|
|
}
|
|
|
|
// AV_NUM_DATA_POINTERS defined in AVFrame struct
|
|
uint8_t *data[AV_NUM_DATA_POINTERS] = { storage.bits(), nullptr };
|
|
int linesize[AV_NUM_DATA_POINTERS] = { int(storage.bytesPerLine()), 0 };
|
|
|
|
sws_scale(
|
|
stream.swscale.get(),
|
|
frame->data,
|
|
frame->linesize,
|
|
0,
|
|
frame->height,
|
|
data,
|
|
linesize);
|
|
|
|
if (frame->format == AV_PIX_FMT_YUVA420P) {
|
|
FFmpeg::PremultiplyInplace(storage);
|
|
}
|
|
}
|
|
|
|
FFmpeg::ClearFrameMemory(frame);
|
|
return storage;
|
|
}
|
|
|
|
FrameYUV ExtractYUV(Stream &stream, AVFrame *frame) {
|
|
return {
|
|
.size = { frame->width, frame->height },
|
|
.chromaSize = {
|
|
AV_CEIL_RSHIFT(frame->width, 1), // SWScale does that.
|
|
AV_CEIL_RSHIFT(frame->height, 1)
|
|
},
|
|
.y = { .data = frame->data[0], .stride = frame->linesize[0] },
|
|
.u = { .data = frame->data[1], .stride = frame->linesize[1] },
|
|
.v = { .data = frame->data[2], .stride = frame->linesize[2] },
|
|
};
|
|
}
|
|
|
|
void PaintFrameOuter(QPainter &p, const QRect &inner, QSize outer) {
|
|
const auto left = inner.x();
|
|
const auto right = outer.width() - inner.width() - left;
|
|
const auto top = inner.y();
|
|
const auto bottom = outer.height() - inner.height() - top;
|
|
if (left > 0) {
|
|
p.fillRect(0, 0, left, outer.height(), st::imageBg);
|
|
}
|
|
if (right > 0) {
|
|
p.fillRect(
|
|
outer.width() - right,
|
|
0,
|
|
right,
|
|
outer.height(),
|
|
st::imageBg);
|
|
}
|
|
if (top > 0) {
|
|
p.fillRect(left, 0, inner.width(), top, st::imageBg);
|
|
}
|
|
if (bottom > 0) {
|
|
p.fillRect(
|
|
left,
|
|
outer.height() - bottom,
|
|
inner.width(),
|
|
bottom,
|
|
st::imageBg);
|
|
}
|
|
}
|
|
|
|
void PaintFrameInner(
|
|
QPainter &p,
|
|
QRect to,
|
|
const QImage &original,
|
|
bool alpha,
|
|
int rotation) {
|
|
const auto rotated = [](QRect rect, int rotation) {
|
|
switch (rotation) {
|
|
case 0: return rect;
|
|
case 90: return QRect(
|
|
rect.y(),
|
|
-rect.x() - rect.width(),
|
|
rect.height(),
|
|
rect.width());
|
|
case 180: return QRect(
|
|
-rect.x() - rect.width(),
|
|
-rect.y() - rect.height(),
|
|
rect.width(),
|
|
rect.height());
|
|
case 270: return QRect(
|
|
-rect.y() - rect.height(),
|
|
rect.x(),
|
|
rect.height(),
|
|
rect.width());
|
|
}
|
|
Unexpected("Rotation in PaintFrameInner.");
|
|
};
|
|
|
|
PainterHighQualityEnabler hq(p);
|
|
if (rotation) {
|
|
p.rotate(rotation);
|
|
}
|
|
const auto rect = rotated(to, rotation);
|
|
if (alpha) {
|
|
p.fillRect(rect, Qt::white);
|
|
}
|
|
p.drawImage(rect, original);
|
|
}
|
|
|
|
void PaintFrameContent(
|
|
QPainter &p,
|
|
const QImage &original,
|
|
bool alpha,
|
|
int rotation,
|
|
const FrameRequest &request) {
|
|
const auto full = request.outer.isEmpty()
|
|
? original.size()
|
|
: request.outer;
|
|
const auto size = request.resize.isEmpty()
|
|
? original.size()
|
|
: request.resize;
|
|
const auto to = QRect(
|
|
(full.width() - size.width()) / 2,
|
|
(full.height() - size.height()) / 2,
|
|
size.width(),
|
|
size.height());
|
|
if (!alpha || !request.keepAlpha) {
|
|
PaintFrameOuter(p, to, full);
|
|
}
|
|
const auto deAlpha = alpha && !request.keepAlpha;
|
|
PaintFrameInner(p, to, original, deAlpha, rotation);
|
|
}
|
|
|
|
void ApplyFrameRounding(QImage &storage, const FrameRequest &request) {
|
|
if (!(request.corners & RectPart::AllCorners)
|
|
|| (request.radius == ImageRoundRadius::None)) {
|
|
return;
|
|
}
|
|
storage = Images::Round(
|
|
std::move(storage),
|
|
request.radius,
|
|
request.corners);
|
|
}
|
|
|
|
QImage PrepareByRequest(
|
|
const QImage &original,
|
|
bool alpha,
|
|
int rotation,
|
|
const FrameRequest &request,
|
|
QImage storage) {
|
|
Expects(!request.outer.isEmpty() || alpha);
|
|
|
|
const auto outer = request.outer.isEmpty()
|
|
? original.size()
|
|
: request.outer;
|
|
if (!FFmpeg::GoodStorageForFrame(storage, outer)) {
|
|
storage = FFmpeg::CreateFrameStorage(outer);
|
|
}
|
|
|
|
if (alpha && request.keepAlpha) {
|
|
storage.fill(Qt::transparent);
|
|
}
|
|
|
|
QPainter p(&storage);
|
|
PaintFrameContent(p, original, alpha, rotation, request);
|
|
p.end();
|
|
|
|
ApplyFrameRounding(storage, request);
|
|
if (request.colored.alpha() != 0) {
|
|
storage = Images::Colored(std::move(storage), request.colored);
|
|
}
|
|
return storage;
|
|
}
|
|
|
|
} // namespace Streaming
|
|
} // namespace Media
|