7#include "private/qplatformaudiooutput_p.h"
8#include "private/qplatformvideosink_p.h"
9#include "private/qaudiobufferoutput_p.h"
11#include "playbackengine/qffmpegdemuxer_p.h"
12#include "playbackengine/qffmpegstreamdecoder_p.h"
13#include "playbackengine/qffmpegsubtitlerenderer_p.h"
14#include "playbackengine/qffmpegvideorenderer_p.h"
15#include "playbackengine/qffmpegaudiorenderer_p.h"
17#include <qloggingcategory.h>
20namespace ranges = QtMultimediaPrivate::ranges;
28template <
typename Array>
31 using T =
typename Array::value_type;
32 return { T{ {}, {} }, T{ {}, {} }, T{ {}, {} } };
37 m_streams(defaultObjectsArray<
decltype(m_streams)>()),
38 m_renderers(defaultObjectsArray<
decltype(m_renderers)>()),
41 qCDebug(qLcPlaybackEngine) <<
"Create PlaybackEngine";
42 qRegisterMetaType<QFFmpeg::Packet>();
43 qRegisterMetaType<QFFmpeg::Frame>();
44 qRegisterMetaType<QFFmpeg::TrackPosition>();
45 qRegisterMetaType<QFFmpeg::PlaybackEngineObjectID>();
49 qCDebug(qLcPlaybackEngine) <<
"Delete PlaybackEngine";
52 forEachExistingObject([](
auto &object) { object.reset(); });
56void PlaybackEngine::onRendererFinished(
const PlaybackEngineObjectID &id)
61 auto isAtEnd = [
this](
auto trackType) {
62 return !m_renderers[trackType] || m_renderers[trackType]->isAtEnd();
65 if (!isAtEnd(QPlatformMediaPlayer::VideoStream))
68 if (!isAtEnd(QPlatformMediaPlayer::AudioStream))
71 if (!isAtEnd(QPlatformMediaPlayer::SubtitleStream) && !hasMediaStream())
74 if (std::exchange(m_state, QMediaPlayer::StoppedState) == QMediaPlayer::StoppedState)
77 finilizeTime(duration().asTimePoint());
81 qCDebug(qLcPlaybackEngine) <<
"Playback engine end of stream";
86void PlaybackEngine::onRendererLoopChanged(
const PlaybackEngineObjectID &id, TrackPosition offset,
92 if (loopIndex > m_currentLoopOffset.loopIndex) {
93 m_currentLoopOffset = { offset, loopIndex };
95 }
else if (loopIndex == m_currentLoopOffset.loopIndex && offset != m_currentLoopOffset.loopStartTimeUs) {
96 qWarning() <<
"Unexpected offset for loop" << loopIndex <<
":" << offset.get() <<
"vs"
97 << m_currentLoopOffset.loopStartTimeUs.get();
98 m_currentLoopOffset.loopStartTimeUs = offset;
102void PlaybackEngine::onFirstPacketFound(
const PlaybackEngineObjectID &id, TrackPosition absSeekPos)
104 if (!checkObjectID(m_demuxer, id))
107 if (m_timeController.isStarted())
110 const SteadyClock::time_point now = SteadyClock::now();
111 const SteadyClock::time_point expectedTimePoint = m_timeController.timeFromPosition(absSeekPos);
113 std::chrono::round<std::chrono::microseconds>(now - expectedTimePoint);
114 qCDebug(qLcPlaybackEngine) <<
"Delay of demuxer initialization:" << delay;
115 m_timeController.sync(now, absSeekPos);
116 m_timeController.start();
118 forEachExistingObject<Renderer>(
119 [&](
auto &renderer) { renderer->setTimeController(m_timeController); });
122void PlaybackEngine::onRendererSynchronized(
const PlaybackEngineObjectID &id,
123 SteadyClock::time_point tp, TrackPosition pos)
125 if (!hasRenderer(id))
128 Q_ASSERT(checkObjectID(m_renderers[QPlatformMediaPlayer::AudioStream], id));
130 forEachExistingObject<Renderer>([&](
auto &renderer) {
131 if (id.objectID != renderer->objectID()) {
132 auto tc = m_timeController;
133 tc.syncSoft(tp, pos);
134 renderer->setTimeController(tc);
138 m_timeController.sync(tp, pos);
142 if (!m_media.avContext())
145 if (state == m_state)
148 const auto prevState = std::exchange(m_state, state);
150 if (m_state == QMediaPlayer::StoppedState) {
152 finilizeTime(TrackPosition(0));
155 if (prevState == QMediaPlayer::StoppedState || m_state == QMediaPlayer::StoppedState)
158 if (prevState == QMediaPlayer::StoppedState)
159 triggerStepIfNeeded();
161 updateObjectsPausedState();
166 const bool paused = m_state != QMediaPlayer::PlayingState;
167 m_timeController.setPaused(paused);
169 forEachExistingObject([&](
auto &object) {
170 if constexpr (std::is_same_v<
decltype(*object), Renderer &>)
171 object->setPaused(paused);
173 object->setPaused(
false);
180 if (!std::exchange(engine->m_threadsDirty,
true))
181 QMetaObject::invokeMethod(engine, &PlaybackEngine::deleteFreeThreads, Qt::QueuedConnection);
188 connect(&object, &PlaybackEngineObject::error,
this, &PlaybackEngine::errorOccured);
190 auto threadName = objectThreadName(object);
191 auto &thread = m_threads[threadName];
193 thread = std::make_unique<QThread>();
194 thread->setObjectName(threadName);
198 Q_ASSERT(object.thread() != thread.get());
199 object.moveToThread(thread.get());
206 case QPlatformMediaPlayer::VideoStream:
207 return m_videoSink ? createPlaybackEngineObject<VideoRenderer>(
208 m_timeController, m_videoSink, m_media.transformation())
209 : RendererPtr{ {}, {} };
210 case QPlatformMediaPlayer::AudioStream:
211 return m_audioOutput || m_audioBufferOutput
212 ? createPlaybackEngineObject<AudioRenderer>(
213 m_timeController, m_audioOutput, m_audioBufferOutput, m_pitchCompensation)
214 : RendererPtr{ {}, {} };
215 case QPlatformMediaPlayer::SubtitleStream:
217 ? createPlaybackEngineObject<SubtitleRenderer>(m_timeController, m_videoSink)
218 : RendererPtr{ {}, {} };
224template<
typename C,
typename Action>
227 auto handleNotNullObject = [&](
auto &object) {
228 if constexpr (std::is_base_of_v<C, std::remove_reference_t<
decltype(*object)>>)
235 std::for_each(m_renderers.begin(), m_renderers.end(), handleNotNullObject);
236 std::for_each(m_streams.begin(), m_streams.end(), handleNotNullObject);
237 handleNotNullObject(m_demuxer);
240template<
typename Action>
243 forEachExistingObject<PlaybackEngineObject>(std::forward<Action>(action));
248 pos = boundPosition(pos);
250 m_timeController.sync(m_currentLoopOffset.loopStartTimeUs.asDuration() + pos);
251 if (!m_demuxer || !m_media.avContext()) {
252 m_seekPending =
true;
256 m_seekPending =
false;
257 ++m_currentID.sessionID;
259 m_timeController.deactivate();
260 m_timeController.setPaused(m_state != QMediaPlayer::PlayingState);
262 forEachExistingObject([&](
auto &object) {
263 if constexpr (std::is_same_v<
decltype(*object), Renderer &>)
264 object->seek(m_currentID.sessionID, m_timeController, m_currentLoopOffset);
266 object->seek(m_currentID.sessionID, pos, m_currentLoopOffset);
269 triggerStepIfNeeded();
275 qWarning() <<
"Cannot set loops for non-seekable source";
279 if (std::exchange(m_loops, loops) == loops)
282 qCDebug(qLcPlaybackEngine) <<
"set playback engine loops:" << loops <<
"prev loops:" << m_loops
283 <<
"index:" << m_currentLoopOffset.loopIndex;
286 m_demuxer->setLoops(loops);
291 if (m_state != QMediaPlayer::PausedState)
294 if (m_renderers[QPlatformMediaPlayer::VideoStream])
295 m_renderers[QPlatformMediaPlayer::VideoStream]->doForceStep();
302QString
PlaybackEngine::objectThreadName(
const PlaybackEngineObject &object)
304 QString result = QString::fromLatin1(object.metaObject()->className());
305 if (
auto stream = qobject_cast<
const StreamDecoder *>(&object))
306 result += QString::number(stream->trackType());
315 m_timeController.setPlaybackRate(rate);
316 forEachExistingObject<Renderer>([rate](
auto &renderer) { renderer->setPlaybackRate(rate); });
320 return m_timeController.playbackRate();
325 m_timeController.deactivate();
327 forEachExistingObject([](
auto &object) { object.reset(); });
329 createObjectsIfNeeded();
334 if (m_state == QMediaPlayer::StoppedState || !m_media.avContext())
337 for (
int i = 0; i < QPlatformMediaPlayer::NTrackTypes; ++i)
338 createStreamAndRenderer(
static_cast<QPlatformMediaPlayer::TrackType>(i));
344 m_timeController.start();
350 triggerStepIfNeeded();
351 updateObjectsPausedState();
354void PlaybackEngine::createStreamAndRenderer(QPlatformMediaPlayer::TrackType trackType)
356 auto codecContext = codecContextForTrack(trackType);
358 auto &renderer = m_renderers[trackType];
364 renderer = createRenderer(trackType);
369 connect(renderer.get(), &Renderer::synchronized,
this,
370 &PlaybackEngine::onRendererSynchronized);
372 connect(renderer.get(), &Renderer::loopChanged,
this,
373 &PlaybackEngine::onRendererLoopChanged);
375 connect(renderer.get(), &PlaybackEngineObject::atEnd,
this,
376 &PlaybackEngine::onRendererFinished);
379 auto &stream = m_streams[trackType] =
380 createPlaybackEngineObject<StreamDecoder>(*codecContext, renderer->seekPosition());
382 Q_ASSERT(trackType == stream->trackType());
384 connect(stream.get(), &StreamDecoder::requestHandleFrame, renderer.get(), &Renderer::render);
385 connect(stream.get(), &PlaybackEngineObject::atEnd, renderer.get(),
386 &Renderer::onFinalFrameReceived);
387 connect(renderer.get(), &Renderer::frameProcessed, stream.get(),
388 &StreamDecoder::onFrameProcessed);
391std::optional<CodecContext>
PlaybackEngine::codecContextForTrack(QPlatformMediaPlayer::TrackType trackType)
393 const auto streamIndex = m_media.currentStreamIndex(trackType);
397 auto &codecContext = m_codecContexts[trackType];
401 <<
"Create codec for stream:" << streamIndex <<
"trackType:" << trackType;
402 auto maybeCodecContext = CodecContext::create(m_media.avContext()->streams[streamIndex],
403 m_media.avContext(), m_options);
405 if (!maybeCodecContext) {
406 emit errorOccured(QMediaPlayer::FormatError,
407 u"Cannot create codec," + maybeCodecContext.error());
411 codecContext = maybeCodecContext.value();
419 return m_renderers[QPlatformMediaPlayer::AudioStream]
420 || m_renderers[QPlatformMediaPlayer::VideoStream];
425 std::array<
int, QPlatformMediaPlayer::NTrackTypes> streamIndexes = { -1, -1, -1 };
427 bool hasStreams =
false;
430 const auto trackType = stream->trackType();
431 streamIndexes[trackType] = m_media.currentStreamIndex(trackType);
437 const TrackPosition currentLoopPosUs = currentPosition(
false);
439 m_demuxer = createPlaybackEngineObject<Demuxer>(m_media.avContext(), currentLoopPosUs,
440 m_seekPending, m_currentLoopOffset,
441 streamIndexes, m_loops);
443 m_seekPending =
false;
448 connect(m_demuxer.get(), Demuxer::signalByTrackType(stream->trackType()), stream.get(),
449 &StreamDecoder::decode);
450 connect(m_demuxer.get(), &PlaybackEngineObject::atEnd, stream.get(),
451 &StreamDecoder::onFinalPacketReceived);
452 connect(stream.get(), &StreamDecoder::packetProcessed, m_demuxer.get(),
453 &Demuxer::onPacketProcessed);
456 connect(m_demuxer.get(), &Demuxer::firstPacketFound,
this, &PlaybackEngine::onFirstPacketFound);
460 m_threadsDirty =
false;
461 auto freeThreads =
std::move(m_threads);
463 forEachExistingObject([&](
auto &object) {
464 m_threads.insert(freeThreads.extract(objectThreadName(*object)));
467 for (
auto &[name, thr] : freeThreads)
470 for (
auto &[name, thr] : freeThreads)
476 Q_ASSERT(!m_media.avContext());
477 Q_ASSERT(m_state == QMediaPlayer::StoppedState);
478 Q_ASSERT(m_threads.empty());
480 m_media = std::move(media);
481 updateVideoSinkSize();
486 auto prev = std::exchange(m_videoSink, sink);
490 updateVideoSinkSize(prev);
493 if (!sink || !prev) {
505 QAudioOutput *prev = std::exchange(m_audioOutput, output);
509 updateActiveAudioOutput(output);
511 if (!output || !prev) {
519 QAudioBufferOutput *prev = std::exchange(m_audioBufferOutput, output);
522 updateActiveAudioOutput(output);
527 std::optional<TrackPosition> pos;
529 if (m_timeController.isStarted()) {
530 for (size_t i = 0; i < m_renderers.size(); ++i) {
531 const auto &renderer = m_renderers[i];
536 if (!topPos && i == QPlatformMediaPlayer::SubtitleStream && hasMediaStream())
539 const auto rendererPos = renderer->lastPosition();
540 pos = !pos ? rendererPos
541 : topPos ?
std::max(*pos, rendererPos)
542 :
std::min(*pos, rendererPos);
548 pos = m_timeController.currentPosition();
550 return boundPosition(*pos - m_currentLoopOffset.loopStartTimeUs.asDuration());
555 return m_media.duration();
563 return m_media.streamInfo(trackType);
568 return m_media.metaData();
578 m_pitchCompensation = enabled;
585 if (!m_media.setActiveTrack(trackType, streamNumber))
588 m_codecContexts[trackType] = {};
590 m_renderers[trackType].reset();
591 m_streams = defaultObjectsArray<
decltype(m_streams)>();
600 updateVideoSinkSize();
601 createObjectsIfNeeded();
602 updateObjectsPausedState();
607 Q_ASSERT(pos >= TrackPosition(0) && pos <= duration().asTimePoint());
609 m_timeController.deactivate();
610 m_timeController.sync(pos);
611 m_currentLoopOffset = {};
616 if (m_audioBufferOutput)
617 updateActiveAudioOutput(
static_cast<QAudioBufferOutput *>(
nullptr));
619 updateActiveAudioOutput(
static_cast<QAudioOutput *>(
nullptr));
623bool PlaybackEngine::hasRenderer(
const PlaybackEngineObjectID &id)
const
625 return ranges::any_of(m_renderers, [&](
auto &renderer) {
626 return checkObjectID(renderer, id);
630template <
typename AudioOutput>
634 renderer->setOutput(output);
639 if (
auto renderer = qobject_cast<SubtitleRenderer *>(
640 m_renderers[QPlatformMediaPlayer::SubtitleStream].get()))
641 renderer->setOutput(sink, cleanOutput);
643 qobject_cast<VideoRenderer *>(m_renderers[QPlatformMediaPlayer::VideoStream].get()))
644 renderer->setOutput(sink, cleanOutput);
649 auto platformVideoSink = m_videoSink ? m_videoSink->platformVideoSink() :
nullptr;
650 if (!platformVideoSink)
653 if (prevSink && prevSink->platformVideoSink())
654 platformVideoSink->setNativeSize(prevSink->platformVideoSink()->nativeSize());
656 const auto streamIndex = m_media.currentStreamIndex(QPlatformMediaPlayer::VideoStream);
657 if (streamIndex >= 0) {
658 const auto context = m_media.avContext();
659 const auto stream = context->streams[streamIndex];
660 const AVRational pixelAspectRatio =
661 av_guess_sample_aspect_ratio(context, stream,
nullptr);
664 qCalculateFrameSize({ stream->codecpar->width, stream->codecpar->height },
665 { pixelAspectRatio.num, pixelAspectRatio.den });
667 platformVideoSink->setNativeSize(
668 qRotatedFrameSize(size, m_media.transformation().rotation));
673TrackPosition
PlaybackEngine::boundPosition(TrackPosition position)
const
675 position = qMax(position, TrackPosition(0));
676 return duration() > TrackDuration(0) ? qMin(position, duration().asTimePoint()) : position;
681 return qobject_cast<
AudioRenderer *>(m_renderers[QPlatformMediaPlayer::AudioStream].get());
688#include "moc_qffmpegplaybackengine_p.cpp"
void setPitchCompensation(bool enabled)
void seek(TrackPosition pos)
void setLoops(int loopsCount)
void setVideoSink(QVideoSink *sink)
void setState(QMediaPlayer::PlaybackState state)
void updateActiveAudioOutput(AudioOutput *output)
virtual RendererPtr createRenderer(QPlatformMediaPlayer::TrackType trackType)
void setAudioBufferOutput(QAudioBufferOutput *output)
void setPlaybackRate(float rate)
void setAudioSink(QAudioOutput *output)
void setPitchCompensation(bool enabled)
~PlaybackEngine() override
const QList< MediaDataHolder::StreamInfo > & streamInfo(QPlatformMediaPlayer::TrackType trackType) const
TrackDuration duration() const
void setActiveTrack(QPlatformMediaPlayer::TrackType type, int streamNumber)
void setMedia(MediaDataHolder media)
TrackPosition currentPosition(bool topPos=true) const
const QMediaMetaData & metaData() const
void updateActiveVideoOutput(QVideoSink *sink, bool cleanOutput=false)
float playbackRate() const
The QPlaybackOptions class enables low-level control of media playback options.
static Array defaultObjectsArray()
std::conditional_t< QT_FFMPEG_AVIO_WRITE_CONST, const uint8_t *, uint8_t * > AvioWriteBufferType
Combined button and popup list for selecting options.
#define qCDebug(category,...)
#define Q_STATIC_LOGGING_CATEGORY(name,...)
void operator()(PlaybackEngineObject *) const