Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qffmpegresampler.cpp
Go to the documentation of this file.
1// Copyright (C) 2022 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
4#include "playbackengine/qffmpegcodeccontext_p.h"
6#include <qloggingcategory.h>
7
8Q_STATIC_LOGGING_CATEGORY(qLcResampler, "qt.multimedia.ffmpeg.resampler")
9Q_STATIC_LOGGING_CATEGORY(qLcResamplerTrace, "qt.multimedia.ffmpeg.resampler.trace")
10
11QT_BEGIN_NAMESPACE
12
13using namespace QFFmpeg;
14
15QFFmpegResampler::QFFmpegResampler(const QAudioFormat &inputFormat,
16 const QAudioFormat &outputFormat, qint64 startTime)
17 : m_inputFormat(inputFormat), m_outputFormat(outputFormat), m_startTime(startTime)
18{
19 Q_ASSERT(inputFormat.isValid());
20 Q_ASSERT(outputFormat.isValid());
21
22 const auto inputAvFormat = AVAudioFormat(m_inputFormat);
23 const auto outputAvFormat = AVAudioFormat(m_outputFormat);
24
25 m_resampler = createResampleContext(inputAvFormat, outputAvFormat);
26
27 qCDebug(qLcResampler) << "Created QFFmpegResampler with offset" << m_startTime
28 << "us. Converting from" << inputAvFormat << "to" << outputAvFormat;
29}
30
31QFFmpegResampler::QFFmpegResampler(const CodecContext *codecContext,
32 const QAudioFormat &outputFormat,
33 qint64 startTime)
34 : m_outputFormat(outputFormat), m_startTime(startTime)
35{
36 Q_ASSERT(codecContext);
37
38 const AVStream *audioStream = codecContext->stream();
39
40 if (!m_outputFormat.isValid())
41 // want the native format
42 m_outputFormat = QFFmpegMediaFormatInfo::audioFormatFromCodecParameters(*audioStream->codecpar);
43
44 const auto inputAvFormat = AVAudioFormat(audioStream->codecpar);
45 const auto outputAvFormat = AVAudioFormat(m_outputFormat);
46
47 m_resampler = createResampleContext(inputAvFormat, outputAvFormat);
48
49 qCDebug(qLcResampler).nospace() << "Created QFFmpegResampler. Offset: " << m_startTime
50 << "us. From: " << inputAvFormat << " to: " << outputAvFormat;
51}
52
54
55QAudioBuffer QFFmpegResampler::resample(const char* data, size_t size)
56{
57 if (!m_inputFormat.isValid())
58 return {};
59
60 return resample(reinterpret_cast<const uint8_t **>(&data),
61 m_inputFormat.framesForBytes(static_cast<qint32>(size)));
62}
63
65{
66 return resample(const_cast<const uint8_t **>(frame->extended_data), frame->nb_samples);
67}
68
69QAudioBuffer QFFmpegResampler::resample(const uint8_t **inputData, int inputSamplesCount)
70{
71 const int maxOutSamples = adjustMaxOutSamples(inputSamplesCount);
72
73 QByteArray samples(m_outputFormat.bytesForFrames(maxOutSamples), Qt::Uninitialized);
74 auto *out = reinterpret_cast<uint8_t *>(samples.data());
75 const int outSamples =
76 swr_convert(m_resampler.get(), &out, maxOutSamples, inputData, inputSamplesCount);
77
78 samples.resize(m_outputFormat.bytesForFrames(outSamples));
79
80 const qint64 startTime = m_outputFormat.durationForFrames(m_samplesProcessed) + m_startTime;
81 m_samplesProcessed += outSamples;
82
83 qCDebug(qLcResamplerTrace).nospace()
84 << "Created output buffer. Time stamp: " << startTime
85 << "us. Samples in: " << inputSamplesCount
86 << ", Samples out: " << outSamples << ", Max samples: " << maxOutSamples;
87 return QAudioBuffer(samples, m_outputFormat, startTime);
88}
89
90int QFFmpegResampler::adjustMaxOutSamples(int inputSamplesCount)
91{
92 int maxOutSamples = swr_get_out_samples(m_resampler.get(), inputSamplesCount);
93
94 const auto remainingCompensationDistance = m_endCompensationSample - m_samplesProcessed;
95
96 if (remainingCompensationDistance > 0 && maxOutSamples > remainingCompensationDistance) {
97 // If the remaining compensation distance less than output frame,
98 // the ffmpeg resampler bufferises the rest of frames that makes
99 // unexpected delays on large frames.
100 // The hack might cause some compensation bias on large frames,
101 // however it's not significant for our logic, in fact.
102 // TODO: probably, it will need some improvements
103 setSampleCompensation(0, 0);
104 maxOutSamples = swr_get_out_samples(m_resampler.get(), inputSamplesCount);
105 }
106
107 return maxOutSamples;
108}
109
110void QFFmpegResampler::setSampleCompensation(qint32 delta, quint32 distance)
111{
112 const int res = swr_set_compensation(m_resampler.get(), delta, static_cast<int>(distance));
113 if (res < 0)
114 qCWarning(qLcResampler) << "swr_set_compensation fail:" << res;
115 else {
116 m_sampleCompensationDelta = delta;
117 m_endCompensationSample = m_samplesProcessed + distance;
118 }
119}
120
122{
123 return m_samplesProcessed < m_endCompensationSample ? m_sampleCompensationDelta : 0;
124}
125
126QT_END_NAMESPACE
The QAudioFormat class stores audio stream parameter information.
constexpr bool isValid() const noexcept
Returns true if all of the parameters are valid.
QAudioBuffer resample(const char *data, size_t size) override
~QFFmpegResampler() override
void setSampleCompensation(qint32 delta, quint32 distance)
QAudioBuffer resample(const AVFrame *frame)
qint32 activeSampleCompensationDelta() const
#define qCWarning(category,...)
#define qCDebug(category,...)
#define Q_STATIC_LOGGING_CATEGORY(name,...)