6#include <QtCore/qiodevice.h>
7#include <QtCore/qloggingcategory.h>
8#include <QtCore/qmimedatabase.h>
9#include <QtCore/qmutex.h>
10#include <QtCore/qthread.h>
11#include <QtCore/private/qcore_mac_p.h>
13#include "private/qcoreaudioutils_p.h"
15#import <AVFoundation/AVFoundation.h>
19Q_STATIC_LOGGING_CATEGORY(qLcAVFAudioDecoder,
"qt.multimedia.darwin.AVFAudioDecoder");
21using namespace Qt::Literals;
24 const QCFType<CMSampleBufferRef> &sampleBuffer)
30 auto validateFormat = [&] {
31 CMFormatDescriptionRef formatDescription = CMSampleBufferGetFormatDescription(sampleBuffer);
32 if (!formatDescription)
34 const AudioStreamBasicDescription *
const asbd =
35 CMAudioFormatDescriptionGetStreamBasicDescription(formatDescription);
38 return qtFormat == QCoreAudioUtils::toPreferredQAudioFormat(*asbd);
41 Q_ASSERT(validateFormat());
44 size_t audioBufferListSize = 0;
45 OSStatus err = CMSampleBufferGetAudioBufferListWithRetainedBlockBuffer(sampleBuffer,
51 kCMSampleBufferFlag_AudioBufferList_Assure16ByteAlignment,
56 QCFType<CMBlockBufferRef> blockBuffer;
57 AudioBufferList* audioBufferList = (AudioBufferList*) malloc(audioBufferListSize);
59 err = CMSampleBufferGetAudioBufferListWithRetainedBlockBuffer(sampleBuffer,
65 kCMSampleBufferFlag_AudioBufferList_Assure16ByteAlignment,
68 free(audioBufferList);
73 for (UInt32 i = 0; i < audioBufferList->mNumberBuffers; i++)
75 AudioBuffer audioBuffer = audioBufferList->mBuffers[i];
76 abuf.push_back(QByteArray((
const char*)audioBuffer.mData, audioBuffer.mDataByteSize));
79 free(audioBufferList);
81 CMTime sampleStartTime = (CMSampleBufferGetPresentationTimeStamp(sampleBuffer));
82 float sampleStartTimeSecs = CMTimeGetSeconds(sampleStartTime);
84 return QAudioBuffer(abuf, qtFormat, qint64(sampleStartTimeSecs * 1000000));
87@interface AVFResourceReaderDelegate : NSObject <AVAssetResourceLoaderDelegate> {
88 AVFAudioDecoder *m_decoder;
92- (BOOL)resourceLoader:(AVAssetResourceLoader *)resourceLoader
93 shouldWaitForLoadingOfRequestedResource:(AVAssetResourceLoadingRequest *)loadingRequest;
99@implementation AVFResourceReaderDelegate
101- (id)initWithDecoder:(AVFAudioDecoder *)decoder
103 if (!(self = [super init]))
111-(BOOL)resourceLoader:(AVAssetResourceLoader *)resourceLoader
112 shouldWaitForLoadingOfRequestedResource:(AVAssetResourceLoadingRequest *)loadingRequest
114 Q_UNUSED(resourceLoader);
116 if (![loadingRequest.request.URL.scheme isEqualToString:@
"iodevice"])
119 std::lock_guard locker(m_mutex);
121 QIODevice *device = m_decoder ? m_decoder->sourceDevice() :
nullptr;
125 device->seek(loadingRequest.dataRequest.requestedOffset);
126 if (loadingRequest.contentInformationRequest) {
127 loadingRequest.contentInformationRequest.contentLength = device->size();
128 loadingRequest.contentInformationRequest.byteRangeAccessSupported = YES;
131 if (loadingRequest.dataRequest) {
132 NSInteger requestedLength = loadingRequest.dataRequest.requestedLength;
133 int maxBytes = qMin(32 * 1024,
int(requestedLength));
135 buffer.resize(maxBytes);
136 NSInteger submitted = 0;
137 while (submitted < requestedLength) {
138 qint64 len = device->read(buffer.data(), maxBytes);
142 [loadingRequest.dataRequest respondWithData:[NSData dataWithBytes:buffer length:len]];
147 [loadingRequest finishLoading];
155 std::lock_guard locker(m_mutex);
163NSDictionary *av_audio_settings_for_format(
const QAudioFormat &format)
165 float sampleRate = format.sampleRate();
166 int nChannels = format.channelCount();
167 int sampleSize = format.bytesPerSample() * 8;
168 BOOL isFloat = format.sampleFormat() == QAudioFormat::Float;
170 NSDictionary *audioSettings = [NSDictionary dictionaryWithObjectsAndKeys:
171 [NSNumber numberWithInt:kAudioFormatLinearPCM], AVFormatIDKey,
172 [NSNumber numberWithFloat:sampleRate], AVSampleRateKey,
173 [NSNumber numberWithInt:nChannels], AVNumberOfChannelsKey,
174 [NSNumber numberWithInt:sampleSize], AVLinearPCMBitDepthKey,
175 [NSNumber numberWithBool:isFloat], AVLinearPCMIsFloatKey,
176 [NSNumber numberWithBool:NO], AVLinearPCMIsNonInterleaved,
177 [NSNumber numberWithBool:NO], AVLinearPCMIsBigEndianKey,
180 return audioSettings;
183QAudioFormat qt_format_for_audio_track(AVAssetTrack *track)
185 CMFormatDescriptionRef desc = (__bridge CMFormatDescriptionRef)track.formatDescriptions[0];
186 const AudioStreamBasicDescription*
const asbd =
187 CMAudioFormatDescriptionGetStreamBasicDescription(desc);
188 return QCoreAudioUtils::toPreferredQAudioFormat(*asbd);
201 [m_reader cancelReading];
205 [m_readerOutput release];
209AVFAudioDecoder::AVFAudioDecoder(QAudioDecoder *parent)
210 : QPlatformAudioDecoder(parent)
212 m_readingQueue = dispatch_queue_create(
"reader_queue", DISPATCH_QUEUE_SERIAL);
213 m_decodingQueue = dispatch_queue_create(
"decoder_queue", DISPATCH_QUEUE_SERIAL);
215 m_readerDelegate = [[AVFResourceReaderDelegate alloc] initWithDecoder:
this];
216 Q_ASSERT(m_readerDelegate);
219AVFAudioDecoder::~AVFAudioDecoder()
223 [m_readerDelegate clearDecoder];
224 [m_readerDelegate release];
228 dispatch_release(m_readingQueue);
229 dispatch_release(m_decodingQueue);
232QUrl AVFAudioDecoder::source()
const
237void AVFAudioDecoder::setSource(
const QUrl &fileName)
239 if (!m_device && m_source == fileName)
249 if (!m_source.isEmpty()) {
250 NSURL *nsURL = m_source.toNSURL();
251 m_asset = [[AVURLAsset alloc] initWithURL:nsURL options:nil];
257QIODevice *AVFAudioDecoder::sourceDevice()
const
262void AVFAudioDecoder::setSourceDevice(QIODevice *device)
264 if (m_device == device && m_source.isEmpty())
275 const QString ext = QMimeDatabase().mimeTypeForData(m_device).preferredSuffix();
276 const QString url = u"iodevice:///iodevice."_s + ext;
277 NSString *
_Nonnull urlString = url.toNSString();
278 NSURL *nsURL = [NSURL URLWithString:urlString];
281 processInvalidMedia(QAudioDecoder::FormatError,
282 tr(
"Failed to create URL for the device"));
285 m_asset = [[AVURLAsset alloc] initWithURL:nsURL options:nil];
289 [m_asset.resourceLoader setDelegate:m_readerDelegate queue:m_decodingQueue];
295void AVFAudioDecoder::start()
297 if (m_decodingContext) {
298 qCDebug(qLcAVFAudioDecoder()) <<
"AVFAudioDecoder has been already started";
304 if (m_device && (!m_device->isOpen() || !m_device->isReadable())) {
305 processInvalidMedia(QAudioDecoder::ResourceError, tr(
"Unable to read from specified device"));
309 m_decodingContext = std::make_shared<DecodingContext>();
310 std::weak_ptr<DecodingContext> weakContext(m_decodingContext);
312 auto handleLoadingResult = [=,
this]() {
313 NSError *error = nil;
314 AVKeyValueStatus status = [m_asset statusOfValueForKey:@
"tracks" error:&error];
316 if (status == AVKeyValueStatusFailed) {
317 if (error.domain == NSURLErrorDomain)
318 processInvalidMedia(QAudioDecoder::ResourceError,
319 QString::fromNSString(error.localizedDescription));
321 processInvalidMedia(QAudioDecoder::FormatError,
322 tr(
"Could not load media source's tracks"));
323 }
else if (status != AVKeyValueStatusLoaded) {
324 qWarning() <<
"Unexpected AVKeyValueStatus:" << status;
332 [m_asset loadValuesAsynchronouslyForKeys:@[ @
"tracks" ]
333 completionHandler:[=,
this]() {
334 invokeWithDecodingContext(weakContext, handleLoadingResult);
338void AVFAudioDecoder::decBuffersCounter(uint val)
341 QMutexLocker locker(&m_buffersCounterMutex);
342 m_buffersCounter -= val;
345 Q_ASSERT(m_buffersCounter >= 0);
347 m_buffersCounterCondition.wakeAll();
350void AVFAudioDecoder::stop()
352 qCDebug(qLcAVFAudioDecoder()) <<
"stop decoding";
354 m_decodingContext.reset();
355 decBuffersCounter(m_cachedBuffers.size());
356 m_cachedBuffers.clear();
358 bufferAvailableChanged(
false);
365QAudioFormat AVFAudioDecoder::audioFormat()
const
370void AVFAudioDecoder::setAudioFormat(
const QAudioFormat &format)
372 if (m_format != format) {
374 formatChanged(m_format);
378QAudioBuffer AVFAudioDecoder::read()
380 if (m_cachedBuffers.empty())
381 return QAudioBuffer();
383 Q_ASSERT(m_cachedBuffers.size() > 0);
384 QAudioBuffer buffer = m_cachedBuffers.dequeue();
385 decBuffersCounter(1);
387 positionChanged(buffer.startTime() / 1000);
388 bufferAvailableChanged(!m_cachedBuffers.empty());
392void AVFAudioDecoder::processInvalidMedia(QAudioDecoder::Error errorCode,
393 const QString &errorString)
395 qCDebug(qLcAVFAudioDecoder()) <<
"Invalid media. Error code:" << errorCode
396 <<
"Description:" << errorString;
398 Q_ASSERT(QThread::currentThread() == thread());
400 error(
int(errorCode), errorString);
409void AVFAudioDecoder::onFinished()
411 m_decodingContext.reset();
417void AVFAudioDecoder::initAssetReaderImpl(AVAssetTrack *track, NSError *error)
419 Q_ASSERT(track !=
nullptr);
422 processInvalidMedia(QAudioDecoder::ResourceError, QString::fromNSString(error.localizedDescription));
426 QAudioFormat format = m_format.isValid() ? m_format : qt_format_for_audio_track(track);
427 if (!format.isValid()) {
428 processInvalidMedia(QAudioDecoder::FormatError, tr(
"Unsupported source format"));
432 durationChanged(CMTimeGetSeconds(track.timeRange.duration) * 1000);
434 NSDictionary *audioSettings = av_audio_settings_for_format(format);
436 AVAssetReaderTrackOutput *readerOutput =
437 [[AVAssetReaderTrackOutput alloc] initWithTrack:track outputSettings:audioSettings];
438 AVAssetReader *reader = [[AVAssetReader alloc] initWithAsset:m_asset error:&error];
439 auto cleanup = qScopeGuard([&] {
440 [readerOutput release];
445 processInvalidMedia(QAudioDecoder::ResourceError, QString::fromNSString(error.localizedDescription));
448 if (![reader canAddOutput:readerOutput]) {
449 processInvalidMedia(QAudioDecoder::ResourceError, tr(
"Failed to add asset reader output"));
453 [reader addOutput:readerOutput];
455 Q_ASSERT(m_decodingContext);
458 m_decodingContext->m_reader = reader;
459 m_decodingContext->m_readerOutput = readerOutput;
464void AVFAudioDecoder::initAssetReader()
466 qCDebug(qLcAVFAudioDecoder()) <<
"Init asset reader";
469 Q_ASSERT(QThread::currentThread() == thread());
471#if defined(Q_OS_VISIONOS)
472 [m_asset loadTracksWithMediaType:AVMediaTypeAudio completionHandler:[=](NSArray<AVAssetTrack *> *tracks, NSError *error) {
473 if (tracks && tracks.count > 0) {
474 if (AVAssetTrack *track = [tracks objectAtIndex:0])
475 QMetaObject::invokeMethod(
this, &AVFAudioDecoder::initAssetReaderImpl, Qt::QueuedConnection, track, error);
479 NSArray<AVAssetTrack *> *tracks = [m_asset tracksWithMediaType:AVMediaTypeAudio];
480 if (tracks && tracks.count > 0) {
481 if (AVAssetTrack *track = [tracks objectAtIndex:0])
482 initAssetReaderImpl(track,
nullptr );
488void AVFAudioDecoder::startReading(QAudioFormat format)
490 Q_ASSERT(m_decodingContext);
491 Q_ASSERT(m_decodingContext->m_reader);
492 Q_ASSERT(QThread::currentThread() == thread());
495 if (![m_decodingContext->m_reader startReading]) {
496 processInvalidMedia(QAudioDecoder::ResourceError, tr(
"Could not start reading"));
502 std::weak_ptr<DecodingContext> weakContext = m_decodingContext;
507 auto copyNextSampleBuffer = [=,
this]() {
508 auto decodingContext = weakContext.lock();
509 if (!decodingContext)
512 QCFType<CMSampleBufferRef> sampleBuffer{
513 [decodingContext->m_readerOutput copyNextSampleBuffer],
518 dispatch_async(m_decodingQueue, [=,
this]() {
519 if (!weakContext.expired() && CMSampleBufferDataIsReady(sampleBuffer)) {
520 auto audioBuffer = handleNextSampleBuffer(format, sampleBuffer);
522 if (audioBuffer.isValid())
523 invokeWithDecodingContext(weakContext, [=,
this]() {
524 handleNewAudioBuffer(audioBuffer);
532 dispatch_async(m_readingQueue, [=,
this]() {
533 qCDebug(qLcAVFAudioDecoder()) <<
"start reading thread";
539 waitUntilBuffersCounterLessMax();
540 }
while (copyNextSampleBuffer());
543 invokeWithDecodingContext(weakContext, [
this]() { onFinished(); });
547void AVFAudioDecoder::waitUntilBuffersCounterLessMax()
549 if (m_buffersCounter >= MAX_BUFFERS_IN_QUEUE) {
552 QMutexLocker locker(&m_buffersCounterMutex);
554 while (m_buffersCounter >= MAX_BUFFERS_IN_QUEUE)
555 m_buffersCounterCondition.wait(&m_buffersCounterMutex);
559void AVFAudioDecoder::handleNewAudioBuffer(QAudioBuffer buffer)
561 m_cachedBuffers.enqueue(std::move(buffer));
564 Q_ASSERT(m_cachedBuffers.size() == m_buffersCounter);
566 bufferAvailableChanged(
true);
571
572
573
574
576void AVFAudioDecoder::invokeWithDecodingContext(std::weak_ptr<DecodingContext> weakContext, F &&f)
578 if (!weakContext.expired())
579 QMetaObject::invokeMethod(
580 this, [
this, f = std::forward<F>(f), weakContext = std::move(weakContext)]() {
583 if (
auto context = weakContext.lock(); context && context == m_decodingContext)
588#include "moc_avfaudiodecoder_p.cpp"
static constexpr int MAX_BUFFERS_IN_QUEUE
static QAudioBuffer handleNextSampleBuffer(QAudioFormat qtFormat, const QCFType< CMSampleBufferRef > &sampleBuffer)
AVAssetReaderTrackOutput * m_readerOutput