mirror of https://github.com/procxx/kepka.git
Render first frame when starting streaming.
This commit is contained in:
parent
473e30e594
commit
64f2f330f6
|
@ -369,7 +369,7 @@ public:
|
||||||
if (!_implementation && !init()) {
|
if (!_implementation && !init()) {
|
||||||
return error();
|
return error();
|
||||||
}
|
}
|
||||||
if (frame() && frame()->original.isNull()) {
|
if (frame()->original.isNull()) {
|
||||||
auto readResult = _implementation->readFramesTill(-1, ms);
|
auto readResult = _implementation->readFramesTill(-1, ms);
|
||||||
if (readResult == internal::ReaderImplementation::ReadResult::EndOfFile && _seekPositionMs > 0) {
|
if (readResult == internal::ReaderImplementation::ReadResult::EndOfFile && _seekPositionMs > 0) {
|
||||||
// If seek was done to the end: try to read the first frame,
|
// If seek was done to the end: try to read the first frame,
|
||||||
|
@ -460,7 +460,8 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
bool renderFrame() {
|
bool renderFrame() {
|
||||||
Assert(frame() != 0 && _request.valid());
|
Expects(_request.valid());
|
||||||
|
|
||||||
if (!_implementation->renderFrame(frame()->original, frame()->alpha, QSize(_request.framew, _request.frameh))) {
|
if (!_implementation->renderFrame(frame()->original, frame()->alpha, QSize(_request.framew, _request.frameh))) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -573,7 +574,7 @@ private:
|
||||||
};
|
};
|
||||||
Frame _frames[3];
|
Frame _frames[3];
|
||||||
int _frame = 0;
|
int _frame = 0;
|
||||||
Frame *frame() {
|
not_null<Frame*> frame() {
|
||||||
return _frames + _frame;
|
return _frames + _frame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -15,109 +15,58 @@ namespace Media {
|
||||||
namespace Streaming {
|
namespace Streaming {
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr int kSkipInvalidDataPackets = 10;
|
constexpr auto kSkipInvalidDataPackets = 10;
|
||||||
|
constexpr auto kAlignImageBy = 16;
|
||||||
|
constexpr auto kPixelBytesSize = 4;
|
||||||
|
|
||||||
|
void AlignedImageBufferCleanupHandler(void* data) {
|
||||||
|
const auto buffer = static_cast<uchar*>(data);
|
||||||
|
delete[] buffer;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create a QImage of desired size where all the data is properly aligned.
|
||||||
|
QImage CreateAlignedImage(QSize size) {
|
||||||
|
const auto width = size.width();
|
||||||
|
const auto height = size.height();
|
||||||
|
const auto widthAlign = kAlignImageBy / kPixelBytesSize;
|
||||||
|
const auto neededWidth = width + ((width % widthAlign)
|
||||||
|
? (widthAlign - (width % widthAlign))
|
||||||
|
: 0);
|
||||||
|
const auto perLine = neededWidth * kPixelBytesSize;
|
||||||
|
const auto buffer = new uchar[perLine * height + kAlignImageBy];
|
||||||
|
const auto cleanupData = static_cast<void*>(buffer);
|
||||||
|
const auto address = reinterpret_cast<uintptr_t>(buffer);
|
||||||
|
const auto alignedBuffer = buffer + ((address % kAlignImageBy)
|
||||||
|
? (kAlignImageBy - (address % kAlignImageBy))
|
||||||
|
: 0);
|
||||||
|
return QImage(
|
||||||
|
alignedBuffer,
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
perLine,
|
||||||
|
QImage::Format_ARGB32_Premultiplied,
|
||||||
|
AlignedImageBufferCleanupHandler,
|
||||||
|
cleanupData);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsAlignedImage(const QImage &image) {
|
||||||
|
return !(reinterpret_cast<uintptr_t>(image.bits()) % kAlignImageBy)
|
||||||
|
&& !(image.bytesPerLine() % kAlignImageBy);
|
||||||
|
}
|
||||||
|
|
||||||
|
void ClearFrameMemory(AVFrame *frame) {
|
||||||
|
if (frame && frame->data[0]) {
|
||||||
|
av_frame_unref(frame);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
|
||||||
void LogError(QLatin1String method) {
|
CodecPointer MakeCodecPointer(not_null<AVStream*> stream) {
|
||||||
LOG(("Streaming Error: Error in %1.").arg(method));
|
|
||||||
}
|
|
||||||
|
|
||||||
void LogError(QLatin1String method, AvErrorWrap error) {
|
|
||||||
LOG(("Streaming Error: Error in %1 (code: %2, text: %3)."
|
|
||||||
).arg(method
|
|
||||||
).arg(error.code()
|
|
||||||
).arg(error.text()));
|
|
||||||
}
|
|
||||||
|
|
||||||
crl::time PtsToTime(int64_t pts, const AVRational &timeBase) {
|
|
||||||
return (pts == AV_NOPTS_VALUE)
|
|
||||||
? Information::kDurationUnknown
|
|
||||||
: ((pts * 1000LL * timeBase.num) / timeBase.den);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::optional<AvErrorWrap> ReadNextFrame(Stream &stream) {
|
|
||||||
Expects(stream.frame != nullptr);
|
|
||||||
|
|
||||||
auto error = AvErrorWrap();
|
auto error = AvErrorWrap();
|
||||||
|
|
||||||
if (stream.frame->data) {
|
auto result = CodecPointer(avcodec_alloc_context3(nullptr));
|
||||||
av_frame_unref(stream.frame.get());
|
const auto context = result.get();
|
||||||
}
|
|
||||||
do {
|
|
||||||
error = avcodec_receive_frame(stream.codec, stream.frame.get());
|
|
||||||
if (!error) {
|
|
||||||
//processReadFrame(); // #TODO streaming
|
|
||||||
return std::nullopt;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (error.code() != AVERROR(EAGAIN) || stream.queue.empty()) {
|
|
||||||
return error;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto packet = &stream.queue.front().fields();
|
|
||||||
const auto guard = gsl::finally([
|
|
||||||
&,
|
|
||||||
size = packet->size,
|
|
||||||
data = packet->data
|
|
||||||
] {
|
|
||||||
packet->size = size;
|
|
||||||
packet->data = data;
|
|
||||||
stream.queue.pop_front();
|
|
||||||
});
|
|
||||||
|
|
||||||
error = avcodec_send_packet(
|
|
||||||
stream.codec,
|
|
||||||
packet->data ? packet : nullptr); // Drain on eof.
|
|
||||||
if (!error) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
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) {
|
|
||||||
continue; // Try to skip a bad packet.
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return error;
|
|
||||||
} while (true);
|
|
||||||
|
|
||||||
[[unreachable]];
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer::CodecPointer(std::nullptr_t) {
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer::CodecPointer(CodecPointer &&other)
|
|
||||||
: _context(base::take(other._context)) {
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer &CodecPointer::operator=(CodecPointer &&other) {
|
|
||||||
if (this != &other) {
|
|
||||||
destroy();
|
|
||||||
_context = base::take(other._context);
|
|
||||||
}
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer &CodecPointer::operator=(std::nullptr_t) {
|
|
||||||
destroy();
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
void CodecPointer::destroy() {
|
|
||||||
if (_context) {
|
|
||||||
avcodec_free_context(&_context);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer CodecPointer::FromStream(not_null<AVStream*> stream) {
|
|
||||||
auto error = AvErrorWrap();
|
|
||||||
|
|
||||||
auto result = CodecPointer();
|
|
||||||
const auto context = result._context = avcodec_alloc_context3(nullptr);
|
|
||||||
if (!context) {
|
if (!context) {
|
||||||
LogError(qstr("avcodec_alloc_context3"));
|
LogError(qstr("avcodec_alloc_context3"));
|
||||||
return {};
|
return {};
|
||||||
|
@ -141,35 +90,230 @@ CodecPointer CodecPointer::FromStream(not_null<AVStream*> stream) {
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
AVCodecContext *CodecPointer::get() const {
|
void CodecDeleter::operator()(AVCodecContext *value) {
|
||||||
return _context;
|
if (value) {
|
||||||
|
avcodec_free_context(&value);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
AVCodecContext *CodecPointer::operator->() const {
|
FramePointer MakeFramePointer() {
|
||||||
Expects(_context != nullptr);
|
return FramePointer(av_frame_alloc());
|
||||||
|
|
||||||
return get();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CodecPointer::operator AVCodecContext*() const {
|
void FrameDeleter::operator()(AVFrame *value) {
|
||||||
return get();
|
|
||||||
}
|
|
||||||
|
|
||||||
AVCodecContext* CodecPointer::release() {
|
|
||||||
return base::take(_context);
|
|
||||||
}
|
|
||||||
|
|
||||||
CodecPointer::~CodecPointer() {
|
|
||||||
destroy();
|
|
||||||
}
|
|
||||||
|
|
||||||
FrameDeleter::pointer FrameDeleter::create() {
|
|
||||||
return av_frame_alloc();
|
|
||||||
}
|
|
||||||
|
|
||||||
void FrameDeleter::operator()(pointer value) {
|
|
||||||
av_frame_free(&value);
|
av_frame_free(&value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SwsContextPointer MakeSwsContextPointer(
|
||||||
|
not_null<AVFrame*> frame,
|
||||||
|
QSize resize,
|
||||||
|
SwsContextPointer *existing) {
|
||||||
|
const auto result = sws_getCachedContext(
|
||||||
|
existing ? existing->release() : nullptr,
|
||||||
|
frame->width,
|
||||||
|
frame->height,
|
||||||
|
AVPixelFormat(frame->format),
|
||||||
|
resize.width(),
|
||||||
|
resize.height(),
|
||||||
|
AV_PIX_FMT_BGRA,
|
||||||
|
0,
|
||||||
|
nullptr,
|
||||||
|
nullptr,
|
||||||
|
nullptr);
|
||||||
|
if (!result) {
|
||||||
|
LogError(qstr("sws_getCachedContext"));
|
||||||
|
}
|
||||||
|
return SwsContextPointer(result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SwsContextDeleter::operator()(SwsContext *value) {
|
||||||
|
if (value) {
|
||||||
|
sws_freeContext(value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void LogError(QLatin1String method) {
|
||||||
|
LOG(("Streaming Error: Error in %1.").arg(method));
|
||||||
|
}
|
||||||
|
|
||||||
|
void LogError(QLatin1String method, AvErrorWrap error) {
|
||||||
|
LOG(("Streaming Error: Error in %1 (code: %2, text: %3)."
|
||||||
|
).arg(method
|
||||||
|
).arg(error.code()
|
||||||
|
).arg(error.text()));
|
||||||
|
}
|
||||||
|
|
||||||
|
crl::time PtsToTime(int64_t pts, const AVRational &timeBase) {
|
||||||
|
return (pts == AV_NOPTS_VALUE)
|
||||||
|
? Information::kDurationUnknown
|
||||||
|
: ((pts * 1000LL * timeBase.num) / timeBase.den);
|
||||||
|
}
|
||||||
|
|
||||||
|
int ReadRotationFromMetadata(not_null<AVStream*> stream) {
|
||||||
|
const auto tag = av_dict_get(stream->metadata, "rotate", nullptr, 0);
|
||||||
|
if (tag && *tag->value) {
|
||||||
|
const auto string = QString::fromUtf8(tag->value);
|
||||||
|
auto ok = false;
|
||||||
|
const auto degrees = string.toInt(&ok);
|
||||||
|
if (ok && (degrees == 90 || degrees == 180 || degrees == 270)) {
|
||||||
|
return degrees;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 90;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool RotationSwapWidthHeight(int rotation) {
|
||||||
|
return (rotation == 90 || rotation == 270);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<AvErrorWrap> ReadNextFrame(Stream &stream) {
|
||||||
|
Expects(stream.frame != nullptr);
|
||||||
|
|
||||||
|
auto error = AvErrorWrap();
|
||||||
|
|
||||||
|
ClearFrameMemory(stream.frame.get());
|
||||||
|
do {
|
||||||
|
error = avcodec_receive_frame(stream.codec.get(), stream.frame.get());
|
||||||
|
if (!error) {
|
||||||
|
//processReadFrame(); // #TODO streaming
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (error.code() != AVERROR(EAGAIN) || stream.queue.empty()) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto packet = &stream.queue.front().fields();
|
||||||
|
const auto guard = gsl::finally([
|
||||||
|
&,
|
||||||
|
size = packet->size,
|
||||||
|
data = packet->data
|
||||||
|
] {
|
||||||
|
packet->size = size;
|
||||||
|
packet->data = data;
|
||||||
|
stream.queue.pop_front();
|
||||||
|
});
|
||||||
|
|
||||||
|
error = avcodec_send_packet(
|
||||||
|
stream.codec.get(),
|
||||||
|
packet->data ? packet : nullptr); // Drain on eof.
|
||||||
|
if (!error) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
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) {
|
||||||
|
continue; // Try to skip a bad packet.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return error;
|
||||||
|
} while (true);
|
||||||
|
|
||||||
|
[[unreachable]];
|
||||||
|
}
|
||||||
|
|
||||||
|
QImage ConvertFrame(
|
||||||
|
Stream &stream,
|
||||||
|
QSize resize,
|
||||||
|
QImage storage) {
|
||||||
|
Expects(stream.frame != nullptr);
|
||||||
|
|
||||||
|
const auto frame = stream.frame.get();
|
||||||
|
const auto frameSize = QSize(frame->width, frame->height);
|
||||||
|
if (frameSize.isEmpty()) {
|
||||||
|
LOG(("Streaming Error: Bad frame size %1,%2"
|
||||||
|
).arg(resize.width()
|
||||||
|
).arg(resize.height()));
|
||||||
|
return QImage();
|
||||||
|
} else if (!frame->data[0]) {
|
||||||
|
LOG(("Streaming Error: Bad frame data."));
|
||||||
|
return QImage();
|
||||||
|
}
|
||||||
|
if (resize.isEmpty()) {
|
||||||
|
resize = frameSize;
|
||||||
|
} else if (RotationSwapWidthHeight(stream.rotation)) {
|
||||||
|
resize.transpose();
|
||||||
|
}
|
||||||
|
if (storage.isNull()
|
||||||
|
|| storage.size() != resize
|
||||||
|
|| !storage.isDetached()
|
||||||
|
|| !IsAlignedImage(storage)) {
|
||||||
|
storage = CreateAlignedImage(resize);
|
||||||
|
}
|
||||||
|
const auto format = AV_PIX_FMT_BGRA;
|
||||||
|
const auto hasDesiredFormat = (frame->format == format)
|
||||||
|
|| ((frame->format == -1) && (stream.codec->pix_fmt == format));
|
||||||
|
if (frameSize == storage.size() && hasDesiredFormat) {
|
||||||
|
static_assert(sizeof(uint32) == kPixelBytesSize);
|
||||||
|
auto to = reinterpret_cast<uint32*>(storage.bits());
|
||||||
|
auto from = reinterpret_cast<const uint32*>(frame->data[0]);
|
||||||
|
const auto deltaTo = (storage.bytesPerLine() / kPixelBytesSize)
|
||||||
|
- storage.width();
|
||||||
|
const auto deltaFrom = (frame->linesize[0] / kPixelBytesSize)
|
||||||
|
- frame->width;
|
||||||
|
for (const auto y : ranges::view::ints(0, frame->height)) {
|
||||||
|
for (const auto x : ranges::view::ints(0, frame->width)) {
|
||||||
|
// Wipe out possible alpha values.
|
||||||
|
*to++ = 0x000000FFU | *from++;
|
||||||
|
}
|
||||||
|
to += deltaTo;
|
||||||
|
from += deltaFrom;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
stream.swsContext = MakeSwsContextPointer(
|
||||||
|
frame,
|
||||||
|
resize,
|
||||||
|
&stream.swsContext);
|
||||||
|
if (!stream.swsContext) {
|
||||||
|
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] = { storage.bytesPerLine(), 0 };
|
||||||
|
|
||||||
|
const auto lines = sws_scale(
|
||||||
|
stream.swsContext.get(),
|
||||||
|
frame->data,
|
||||||
|
frame->linesize,
|
||||||
|
0,
|
||||||
|
frame->height,
|
||||||
|
data,
|
||||||
|
linesize);
|
||||||
|
if (lines != resize.height()) {
|
||||||
|
LOG(("Streaming Error: "
|
||||||
|
"Unable to sws_scale to good size %1, got %2."
|
||||||
|
).arg(resize.height()
|
||||||
|
).arg(lines));
|
||||||
|
return QImage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (stream.rotation == 180) {
|
||||||
|
storage = std::move(storage).mirrored(true, true);
|
||||||
|
} else if (stream.rotation != 0) {
|
||||||
|
auto transform = QTransform();
|
||||||
|
transform.rotate(stream.rotation);
|
||||||
|
storage = storage.transformed(transform);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read some future packets for audio stream.
|
||||||
|
//if (_audioStreamId >= 0) {
|
||||||
|
// while (_frameMs + 5000 > _lastReadAudioMs
|
||||||
|
// && _frameMs + 15000 > _lastReadVideoMs) {
|
||||||
|
// auto packetResult = readAndProcessPacket();
|
||||||
|
// if (packetResult != PacketResult::Ok) {
|
||||||
|
// break;
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
//}
|
||||||
|
// #TODO streaming
|
||||||
|
|
||||||
|
ClearFrameMemory(stream.frame.get());
|
||||||
|
return std::move(storage);
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace Streaming
|
} // namespace Streaming
|
||||||
} // namespace Media
|
} // namespace Media
|
||||||
|
|
|
@ -10,13 +10,12 @@ https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#include <libavcodec/avcodec.h>
|
#include <libavcodec/avcodec.h>
|
||||||
#include <libavformat/avformat.h>
|
#include <libavformat/avformat.h>
|
||||||
|
#include <libswscale/swscale.h>
|
||||||
} // extern "C"
|
} // extern "C"
|
||||||
|
|
||||||
namespace Media {
|
namespace Media {
|
||||||
namespace Streaming {
|
namespace Streaming {
|
||||||
|
|
||||||
[[nodiscard]] crl::time PtsToTime(int64_t pts, const AVRational &timeBase);
|
|
||||||
|
|
||||||
enum class Mode {
|
enum class Mode {
|
||||||
Both,
|
Both,
|
||||||
Audio,
|
Audio,
|
||||||
|
@ -33,6 +32,7 @@ struct Information {
|
||||||
|
|
||||||
crl::time started = 0;
|
crl::time started = 0;
|
||||||
QImage cover;
|
QImage cover;
|
||||||
|
int rotation = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class AvErrorWrap {
|
class AvErrorWrap {
|
||||||
|
@ -61,9 +61,6 @@ private:
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void LogError(QLatin1String method);
|
|
||||||
void LogError(QLatin1String method, AvErrorWrap error);
|
|
||||||
|
|
||||||
class Packet {
|
class Packet {
|
||||||
public:
|
public:
|
||||||
Packet() {
|
Packet() {
|
||||||
|
@ -118,35 +115,26 @@ private:
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class CodecPointer {
|
struct CodecDeleter {
|
||||||
public:
|
void operator()(AVCodecContext *value);
|
||||||
CodecPointer(std::nullptr_t = nullptr);
|
|
||||||
CodecPointer(CodecPointer &&other);
|
|
||||||
CodecPointer &operator=(CodecPointer &&other);
|
|
||||||
CodecPointer &operator=(std::nullptr_t);
|
|
||||||
~CodecPointer();
|
|
||||||
|
|
||||||
[[nodiscard]] static CodecPointer FromStream(
|
|
||||||
not_null<AVStream*> stream);
|
|
||||||
|
|
||||||
[[nodiscard]] AVCodecContext *get() const;
|
|
||||||
[[nodiscard]] AVCodecContext *operator->() const;
|
|
||||||
[[nodiscard]] operator AVCodecContext*() const;
|
|
||||||
[[nodiscard]] AVCodecContext* release();
|
|
||||||
|
|
||||||
private:
|
|
||||||
void destroy();
|
|
||||||
|
|
||||||
AVCodecContext *_context = nullptr;
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
using CodecPointer = std::unique_ptr<AVCodecContext, CodecDeleter>;
|
||||||
|
CodecPointer MakeCodecPointer(not_null<AVStream*> stream);
|
||||||
|
|
||||||
struct FrameDeleter {
|
struct FrameDeleter {
|
||||||
using pointer = AVFrame*;
|
void operator()(AVFrame *value);
|
||||||
[[nodiscard]] static pointer create();
|
|
||||||
void operator()(pointer value);
|
|
||||||
};
|
};
|
||||||
using FramePointer = std::unique_ptr<FrameDeleter::pointer, FrameDeleter>;
|
using FramePointer = std::unique_ptr<AVFrame, FrameDeleter>;
|
||||||
|
FramePointer MakeFramePointer();
|
||||||
|
|
||||||
|
struct SwsContextDeleter {
|
||||||
|
void operator()(SwsContext *value);
|
||||||
|
};
|
||||||
|
using SwsContextPointer = std::unique_ptr<SwsContext, SwsContextDeleter>;
|
||||||
|
SwsContextPointer MakeSwsContextPointer(
|
||||||
|
not_null<AVFrame*> frame,
|
||||||
|
QSize resize,
|
||||||
|
SwsContextPointer *existing = nullptr);
|
||||||
|
|
||||||
struct Stream {
|
struct Stream {
|
||||||
CodecPointer codec;
|
CodecPointer codec;
|
||||||
|
@ -154,9 +142,23 @@ struct Stream {
|
||||||
std::deque<Packet> queue;
|
std::deque<Packet> queue;
|
||||||
crl::time lastReadPositionTime = 0;
|
crl::time lastReadPositionTime = 0;
|
||||||
int invalidDataPackets = 0;
|
int invalidDataPackets = 0;
|
||||||
|
|
||||||
|
// Video only.
|
||||||
|
int rotation = 0;
|
||||||
|
SwsContextPointer swsContext;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
void LogError(QLatin1String method);
|
||||||
|
void LogError(QLatin1String method, AvErrorWrap error);
|
||||||
|
|
||||||
|
[[nodiscard]] crl::time PtsToTime(int64_t pts, const AVRational &timeBase);
|
||||||
|
[[nodiscard]] int ReadRotationFromMetadata(not_null<AVStream*> stream);
|
||||||
|
[[nodiscard]] bool RotationSwapWidthHeight(int rotation);
|
||||||
[[nodiscard]] std::optional<AvErrorWrap> ReadNextFrame(Stream &stream);
|
[[nodiscard]] std::optional<AvErrorWrap> ReadNextFrame(Stream &stream);
|
||||||
|
[[nodiscard]] QImage ConvertFrame(
|
||||||
|
Stream& stream,
|
||||||
|
QSize resize,
|
||||||
|
QImage storage);
|
||||||
|
|
||||||
} // namespace Streaming
|
} // namespace Streaming
|
||||||
} // namespace Media
|
} // namespace Media
|
||||||
|
|
|
@ -100,7 +100,7 @@ void File::Context::initStream(StreamWrap &wrap, AVMediaType type) {
|
||||||
type,
|
type,
|
||||||
-1,
|
-1,
|
||||||
-1,
|
-1,
|
||||||
0,
|
nullptr,
|
||||||
0);
|
0);
|
||||||
if (wrap.id < 0) {
|
if (wrap.id < 0) {
|
||||||
return;
|
return;
|
||||||
|
@ -108,27 +108,15 @@ void File::Context::initStream(StreamWrap &wrap, AVMediaType type) {
|
||||||
|
|
||||||
wrap.info = _formatContext->streams[wrap.id];
|
wrap.info = _formatContext->streams[wrap.id];
|
||||||
if (type == AVMEDIA_TYPE_VIDEO) {
|
if (type == AVMEDIA_TYPE_VIDEO) {
|
||||||
const auto rotateTag = av_dict_get(
|
wrap.stream.rotation = ReadRotationFromMetadata(wrap.info);
|
||||||
wrap.info->metadata,
|
|
||||||
"rotate",
|
|
||||||
nullptr,
|
|
||||||
0);
|
|
||||||
if (rotateTag && *rotateTag->value) {
|
|
||||||
const auto stringRotateTag = QString::fromUtf8(rotateTag->value);
|
|
||||||
auto toIntSucceeded = false;
|
|
||||||
const auto rotateDegrees = stringRotateTag.toInt(&toIntSucceeded);
|
|
||||||
if (toIntSucceeded) {
|
|
||||||
//_rotation = rotationFromDegrees(rotateDegrees); // #TODO streaming
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
wrap.stream.codec = CodecPointer::FromStream(wrap.info);
|
wrap.stream.codec = MakeCodecPointer(wrap.info);
|
||||||
if (!wrap.stream.codec) {
|
if (!wrap.stream.codec) {
|
||||||
ClearStream(wrap);
|
ClearStream(wrap);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
wrap.stream.frame.reset(FrameDeleter::create());
|
wrap.stream.frame = MakeFramePointer();
|
||||||
if (!wrap.stream.frame) {
|
if (!wrap.stream.frame) {
|
||||||
ClearStream(wrap);
|
ClearStream(wrap);
|
||||||
return;
|
return;
|
||||||
|
@ -289,6 +277,12 @@ void File::Context::readInformation(crl::time positionTime) {
|
||||||
information.cover = readFirstVideoFrame();
|
information.cover = readFirstVideoFrame();
|
||||||
if (unroll()) {
|
if (unroll()) {
|
||||||
return;
|
return;
|
||||||
|
} else if (!information.cover.isNull()) {
|
||||||
|
information.video = information.cover.size();
|
||||||
|
information.rotation = _video.stream.rotation;
|
||||||
|
if (RotationSwapWidthHeight(information.rotation)) {
|
||||||
|
information.video.transpose();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
information.audio = (_audio.info != nullptr);
|
information.audio = (_audio.info != nullptr);
|
||||||
|
@ -342,7 +336,7 @@ base::variant<QImage, AvErrorWrap> File::Context::tryReadFirstVideoFrame() {
|
||||||
}
|
}
|
||||||
return *error;
|
return *error;
|
||||||
}
|
}
|
||||||
return QImage(); // #TODO streaming decode frame
|
return ConvertFrame(_video.stream, QSize(), QImage());
|
||||||
}
|
}
|
||||||
|
|
||||||
void File::Context::enqueueEofPackets() {
|
void File::Context::enqueueEofPackets() {
|
||||||
|
@ -373,7 +367,7 @@ void File::Context::processPacket(Packet &&packet) {
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
check(_audio) && check(_video);
|
check(_audio) || check(_video);
|
||||||
}
|
}
|
||||||
|
|
||||||
void File::Context::readNextPacket() {
|
void File::Context::readNextPacket() {
|
||||||
|
@ -445,9 +439,6 @@ File::Context::~Context() {
|
||||||
ClearStream(_audio);
|
ClearStream(_audio);
|
||||||
ClearStream(_video);
|
ClearStream(_video);
|
||||||
|
|
||||||
//if (_swsContext) {
|
|
||||||
// sws_freeContext(_swsContext);
|
|
||||||
//}
|
|
||||||
if (_opened) {
|
if (_opened) {
|
||||||
avformat_close_input(&_formatContext);
|
avformat_close_input(&_formatContext);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue