6#include "private/qwindowsmultimediautils_p.h"
8#include <qmediadevices.h>
9#include <qaudiodevice.h>
10#include <private/qmemoryvideobuffer_p.h>
11#include <private/qvideoframe_p.h>
12#include <QtCore/private/qcomptr_p.h>
13#include <QtCore/qdebug.h>
15#include <mmdeviceapi.h>
26 m_durationTimer.setInterval(100);
27 connect(&m_durationTimer, &QTimer::timeout,
this, &QWindowsMediaDeviceReader::updateDuration);
43 IMFAttributes *sourceAttributes =
nullptr;
45 HRESULT hr = MFCreateAttributes(&sourceAttributes, 2);
48 hr = sourceAttributes->SetGUID(MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE,
49 video ? MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_GUID
50 : MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_AUDCAP_GUID);
53 hr = sourceAttributes->SetString(video ? MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_VIDCAP_SYMBOLIC_LINK
54 : MF_DEVSOURCE_ATTRIBUTE_SOURCE_TYPE_AUDCAP_ENDPOINT_ID,
55 reinterpret_cast<LPCWSTR>(deviceId.utf16()));
58 hr = MFCreateDeviceSource(sourceAttributes, source);
61 sourceAttributes->Release();
70 IMFMediaSource *secondSource,
71 IMFMediaSource **aggregateSource,
72 IMFSourceReader **sourceReader)
74 if ((!firstSource && !secondSource) || !aggregateSource || !sourceReader)
77 *aggregateSource =
nullptr;
78 *sourceReader =
nullptr;
80 IMFCollection *sourceCollection =
nullptr;
82 HRESULT hr = MFCreateCollection(&sourceCollection);
86 sourceCollection->AddElement(firstSource);
89 sourceCollection->AddElement(secondSource);
91 hr = MFCreateAggregateSource(sourceCollection, aggregateSource);
94 IMFAttributes *readerAttributes =
nullptr;
96 hr = MFCreateAttributes(&readerAttributes, 1);
100 hr = readerAttributes->SetUnknown(MF_SOURCE_READER_ASYNC_CALLBACK,
101 static_cast<IMFSourceReaderCallback*>(
this));
104 hr = MFCreateSourceReaderFromMediaSource(*aggregateSource, readerAttributes, sourceReader);
106 readerAttributes->Release();
109 sourceCollection->Release();
120 if (m_sourceReader && m_videoSource) {
123 IMFMediaType *mediaType =
nullptr;
126 float currFrameRate = 0.0f;
128 while (SUCCEEDED(m_sourceReader->GetNativeMediaType(DWORD(MF_SOURCE_READER_FIRST_VIDEO_STREAM),
129 index, &mediaType))) {
131 GUID subtype = GUID_NULL;
132 if (SUCCEEDED(mediaType->GetGUID(MF_MT_SUBTYPE, &subtype))) {
134 auto pixelFormat = QWindowsMultimediaUtils::pixelFormatFromMediaSubtype(subtype);
135 if (pixelFormat != QVideoFrameFormat::Format_Invalid) {
137 UINT32 width, height;
138 if (SUCCEEDED(MFGetAttributeSize(mediaType, MF_MT_FRAME_SIZE, &width, &height))) {
141 if (SUCCEEDED(MFGetAttributeRatio(mediaType, MF_MT_FRAME_RATE, &num, &den))) {
143 UINT32 area = width * height;
144 float frameRate =
float(num) / den;
146 if (!reqFormat.isNull()
147 && UINT32(reqFormat.resolution().width()) == width
148 && UINT32(reqFormat.resolution().height()) == height
149 && qFuzzyCompare(reqFormat.maxFrameRate(), frameRate)
150 && reqFormat.pixelFormat() == pixelFormat) {
151 mediaType->Release();
155 if ((currFrameRate < 29.9 && currFrameRate < frameRate) ||
156 (currFrameRate == frameRate && currArea < area)) {
158 currFrameRate = frameRate;
165 mediaType->Release();
186 hr = m_sourceReader->GetCurrentMediaType(DWORD(MF_SOURCE_READER_FIRST_VIDEO_STREAM),
189 hr = m_sourceReader->GetNativeMediaType(DWORD(MF_SOURCE_READER_FIRST_VIDEO_STREAM),
190 mediaTypeIndex, &m_videoMediaType);
192 hr = m_sourceReader->SetCurrentMediaType(DWORD(MF_SOURCE_READER_FIRST_VIDEO_STREAM),
193 nullptr, m_videoMediaType);
198 GUID subtype = GUID_NULL;
199 hr = m_videoMediaType->GetGUID(MF_MT_SUBTYPE, &subtype);
202 m_pixelFormat = QWindowsMultimediaUtils::pixelFormatFromMediaSubtype(subtype);
204 if (m_pixelFormat == QVideoFrameFormat::Format_Invalid) {
209 hr = MFGetAttributeSize(m_videoMediaType, MF_MT_FRAME_SIZE, &m_frameWidth, &m_frameHeight);
213 hr = MFGetStrideForBitmapInfoHeader(subtype.Data1, m_frameWidth, &m_stride);
215 m_stride = qAbs(m_stride);
216 UINT32 frameRateNum, frameRateDen;
217 hr = MFGetAttributeRatio(m_videoMediaType, MF_MT_FRAME_RATE, &frameRateNum, &frameRateDen);
220 m_frameRate = qreal(frameRateNum) / frameRateDen;
222 hr = m_sourceReader->SetStreamSelection(DWORD(MF_SOURCE_READER_FIRST_VIDEO_STREAM), TRUE);
233HRESULT
QWindowsMediaDeviceReader::initAudioType(IMFMediaType *mediaType, UINT32 channels, UINT32 samplesPerSec,
bool flt)
238 HRESULT hr = mediaType->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Audio);
240 hr = mediaType->SetGUID(MF_MT_SUBTYPE, flt ? MFAudioFormat_Float : MFAudioFormat_PCM);
242 hr = mediaType->SetUINT32(MF_MT_AUDIO_NUM_CHANNELS, channels);
244 hr = mediaType->SetUINT32(MF_MT_AUDIO_CHANNEL_MASK,
245 (channels == 1) ? SPEAKER_FRONT_CENTER : (SPEAKER_FRONT_LEFT | SPEAKER_FRONT_RIGHT ));
247 hr = mediaType->SetUINT32(MF_MT_AUDIO_SAMPLES_PER_SECOND, samplesPerSec);
249 UINT32 bitsPerSample = flt ? 32 : 16;
250 UINT32 bytesPerFrame = channels * bitsPerSample/8;
251 hr = mediaType->SetUINT32(MF_MT_AUDIO_BITS_PER_SAMPLE, bitsPerSample);
253 hr = mediaType->SetUINT32(MF_MT_AUDIO_BLOCK_ALIGNMENT, bytesPerFrame);
255 hr = mediaType->SetUINT32(MF_MT_AUDIO_AVG_BYTES_PER_SECOND, bytesPerFrame * samplesPerSec);
276 HRESULT hr = m_sourceReader->GetCurrentMediaType(DWORD(MF_SOURCE_READER_FIRST_AUDIO_STREAM),
279 hr = initAudioType(m_audioMediaType, 2, 48000,
true);
281 hr = m_sourceReader->SetCurrentMediaType(DWORD(MF_SOURCE_READER_FIRST_AUDIO_STREAM),
282 nullptr, m_audioMediaType);
284 hr = m_sourceReader->SetStreamSelection(DWORD(MF_SOURCE_READER_FIRST_AUDIO_STREAM), TRUE);
297 m_sourceVideoStreamIndex = MF_SOURCE_READER_INVALID_STREAM_INDEX;
298 m_sourceAudioStreamIndex = MF_SOURCE_READER_INVALID_STREAM_INDEX;
301 BOOL selected = FALSE;
303 while (m_sourceReader->GetStreamSelection(index, &selected) == S_OK) {
305 IMFMediaType *mediaType =
nullptr;
306 if (SUCCEEDED(m_sourceReader->GetCurrentMediaType(index, &mediaType))) {
307 GUID majorType = GUID_NULL;
308 if (SUCCEEDED(mediaType->GetGUID(MF_MT_MAJOR_TYPE, &majorType))) {
309 if (majorType == MFMediaType_Video)
310 m_sourceVideoStreamIndex = index;
311 else if (majorType == MFMediaType_Audio)
312 m_sourceAudioStreamIndex = index;
314 mediaType->Release();
319 if ((m_videoSource && m_sourceVideoStreamIndex == MF_SOURCE_READER_INVALID_STREAM_INDEX) ||
320 (m_audioSource && m_sourceAudioStreamIndex == MF_SOURCE_READER_INVALID_STREAM_INDEX))
327 QMutexLocker locker(&m_mutex);
331 m_audioOutputId = audioOutputId;
333 if (!m_active || m_audioOutputId.isEmpty())
336 HRESULT hr = startMonitoring();
338 return SUCCEEDED(hr);
343 if (m_audioOutputId.isEmpty())
346 IMFAttributes *sinkAttributes =
nullptr;
348 HRESULT hr = MFCreateAttributes(&sinkAttributes, 1);
351 hr = sinkAttributes->SetString(MF_AUDIO_RENDERER_ATTRIBUTE_ENDPOINT_ID,
352 reinterpret_cast<LPCWSTR>(m_audioOutputId.utf16()));
355 IMFMediaSink *mediaSink =
nullptr;
356 hr = MFCreateAudioRenderer(sinkAttributes, &mediaSink);
359 IMFStreamSink *streamSink =
nullptr;
360 hr = mediaSink->GetStreamSinkByIndex(0, &streamSink);
363 IMFMediaTypeHandler *typeHandler =
nullptr;
364 hr = streamSink->GetMediaTypeHandler(&typeHandler);
367 hr = typeHandler->IsMediaTypeSupported(m_audioMediaType,
nullptr);
370 hr = typeHandler->SetCurrentMediaType(m_audioMediaType);
373 IMFAttributes *writerAttributes =
nullptr;
375 HRESULT hr = MFCreateAttributes(&writerAttributes, 1);
378 hr = writerAttributes->SetUINT32(MF_SINK_WRITER_DISABLE_THROTTLING, TRUE);
381 IMFSinkWriter *sinkWriter =
nullptr;
382 hr = MFCreateSinkWriterFromMediaSink(mediaSink, writerAttributes, &sinkWriter);
385 hr = sinkWriter->SetInputMediaType(0, m_audioMediaType,
nullptr);
388 IMFSimpleAudioVolume *audioVolume =
nullptr;
390 if (SUCCEEDED(MFGetService(
391 mediaSink, MR_POLICY_VOLUME_SERVICE,
392 IID_PPV_ARGS(&audioVolume)))) {
393 audioVolume->SetMasterVolume(
float(m_outputVolume));
394 audioVolume->SetMute(m_outputMuted);
395 audioVolume->Release();
398 hr = sinkWriter->BeginWriting();
400 m_monitorSink = mediaSink;
401 m_monitorSink->AddRef();
402 m_monitorWriter = sinkWriter;
403 m_monitorWriter->AddRef();
406 sinkWriter->Release();
409 writerAttributes->Release();
413 typeHandler->Release();
415 streamSink->Release();
417 mediaSink->Release();
420 sinkAttributes->Release();
428 if (m_monitorWriter) {
429 m_monitorWriter->Release();
430 m_monitorWriter =
nullptr;
433 m_monitorSink->Shutdown();
434 m_monitorSink->Release();
435 m_monitorSink =
nullptr;
442 const QCameraFormat &cameraFormat,
443 const QString µphoneId)
445 QMutexLocker locker(&m_mutex);
447 if (cameraId.isEmpty() && microphoneId.isEmpty())
456 if (!cameraId.isEmpty()) {
457 if (!SUCCEEDED(createSource(cameraId,
true, &m_videoSource))) {
463 if (!microphoneId.isEmpty()) {
464 if (!SUCCEEDED(createSource(microphoneId,
false, &m_audioSource))) {
470 if (!SUCCEEDED(createAggregateReader(m_videoSource, m_audioSource, &m_aggregateSource, &m_sourceReader))) {
475 DWORD mediaTypeIndex = findMediaTypeIndex(cameraFormat);
477 if (!SUCCEEDED(prepareVideoStream(mediaTypeIndex))) {
482 if (!SUCCEEDED(prepareAudioStream())) {
487 if (!SUCCEEDED(initSourceIndexes())) {
492 updateSinkInputMediaTypes();
496 if (!SUCCEEDED(m_sourceReader->ReadSample(MF_SOURCE_READER_ANY_STREAM, 0,
nullptr,
nullptr,
nullptr,
nullptr))) {
515 QMutexLocker locker(&m_mutex);
522 if (m_videoMediaType) {
523 m_videoMediaType->Release();
524 m_videoMediaType =
nullptr;
526 if (m_audioMediaType) {
527 m_audioMediaType->Release();
528 m_audioMediaType =
nullptr;
530 if (m_sourceReader) {
531 m_sourceReader->Release();
532 m_sourceReader =
nullptr;
534 if (m_aggregateSource) {
535 m_aggregateSource->Release();
536 m_aggregateSource =
nullptr;
539 m_videoSource->Release();
540 m_videoSource =
nullptr;
543 m_audioSource->Release();
544 m_audioSource =
nullptr;
549 UINT32 height, qreal frameRate, IMFMediaType **mediaType)
554 *mediaType =
nullptr;
555 IMFMediaType *targetMediaType =
nullptr;
557 if (SUCCEEDED(MFCreateMediaType(&targetMediaType))) {
559 if (SUCCEEDED(targetMediaType->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Video))) {
561 if (SUCCEEDED(targetMediaType->SetGUID(MF_MT_SUBTYPE, format))) {
563 if (SUCCEEDED(targetMediaType->SetUINT32(MF_MT_AVG_BITRATE, bitRate))) {
565 if (SUCCEEDED(MFSetAttributeSize(targetMediaType, MF_MT_FRAME_SIZE, width, height))) {
567 if (SUCCEEDED(MFSetAttributeRatio(targetMediaType, MF_MT_FRAME_RATE,
568 UINT32(frameRate * 1000), 1000))) {
570 if (SUCCEEDED(MFGetAttributeRatio(m_videoMediaType, MF_MT_PIXEL_ASPECT_RATIO, &t1, &t2))) {
572 if (SUCCEEDED(MFSetAttributeRatio(targetMediaType, MF_MT_PIXEL_ASPECT_RATIO, t1, t2))) {
574 if (SUCCEEDED(m_videoMediaType->GetUINT32(MF_MT_INTERLACE_MODE, &t1))) {
576 if (SUCCEEDED(targetMediaType->SetUINT32(MF_MT_INTERLACE_MODE, t1))) {
578 *mediaType = targetMediaType;
589 targetMediaType->Release();
599 *mediaType =
nullptr;
600 IMFMediaType *targetMediaType =
nullptr;
602 if (SUCCEEDED(MFCreateMediaType(&targetMediaType))) {
604 if (SUCCEEDED(targetMediaType->SetGUID(MF_MT_MAJOR_TYPE, MFMediaType_Audio))) {
606 if (SUCCEEDED(targetMediaType->SetGUID(MF_MT_SUBTYPE, format))) {
608 if (bitRate == 0 || SUCCEEDED(targetMediaType->SetUINT32(MF_MT_AUDIO_AVG_BYTES_PER_SECOND, bitRate / 8))) {
610 *mediaType = targetMediaType;
615 targetMediaType->Release();
624 if (m_videoSource && m_videoMediaType && m_sinkVideoStreamIndex != MF_SINK_WRITER_INVALID_STREAM_INDEX) {
625 hr = m_sinkWriter->SetInputMediaType(m_sinkVideoStreamIndex, m_videoMediaType,
nullptr);
628 if (m_audioSource && m_audioMediaType && m_sinkAudioStreamIndex != MF_SINK_WRITER_INVALID_STREAM_INDEX) {
629 hr = m_sinkWriter->SetInputMediaType(m_sinkAudioStreamIndex, m_audioMediaType,
nullptr);
637 const QString &fileName,
const GUID &container,
const GUID &videoFormat, UINT32 videoBitRate,
638 UINT32 width, UINT32 height, qreal frameRate,
const GUID &audioFormat, UINT32 audioBitRate)
640 QMutexLocker locker(&m_mutex);
642 if (!m_active || m_recording || (videoFormat == GUID_NULL && audioFormat == GUID_NULL))
643 return QMediaRecorder::ResourceError;
645 ComPtr<IMFAttributes> writerAttributes;
647 HRESULT hr = MFCreateAttributes(writerAttributes.GetAddressOf(), 2);
649 return QMediaRecorder::ResourceError;
652 hr = writerAttributes->SetUnknown(MF_SINK_WRITER_ASYNC_CALLBACK,
653 static_cast<IMFSinkWriterCallback*>(
this));
655 return QMediaRecorder::ResourceError;
657 hr = writerAttributes->SetGUID(MF_TRANSCODE_CONTAINERTYPE, container);
659 return QMediaRecorder::ResourceError;
661 ComPtr<IMFSinkWriter> sinkWriter;
662 hr = MFCreateSinkWriterFromURL(
reinterpret_cast<LPCWSTR>(fileName.utf16()),
663 nullptr, writerAttributes.Get(), sinkWriter.GetAddressOf());
665 return QMediaRecorder::LocationNotWritable;
667 m_sinkVideoStreamIndex = MF_SINK_WRITER_INVALID_STREAM_INDEX;
668 m_sinkAudioStreamIndex = MF_SINK_WRITER_INVALID_STREAM_INDEX;
670 if (m_videoSource && videoFormat != GUID_NULL) {
671 IMFMediaType *targetMediaType =
nullptr;
673 hr = createVideoMediaType(videoFormat, videoBitRate, width, height, frameRate, &targetMediaType);
676 hr = sinkWriter->AddStream(targetMediaType, &m_sinkVideoStreamIndex);
679 hr = sinkWriter->SetInputMediaType(m_sinkVideoStreamIndex, m_videoMediaType,
nullptr);
681 targetMediaType->Release();
686 if (m_audioSource && audioFormat != GUID_NULL) {
687 IMFMediaType *targetMediaType =
nullptr;
689 hr = createAudioMediaType(audioFormat, audioBitRate, &targetMediaType);
692 hr = sinkWriter->AddStream(targetMediaType, &m_sinkAudioStreamIndex);
695 hr = sinkWriter->SetInputMediaType(m_sinkAudioStreamIndex, m_audioMediaType,
nullptr);
697 targetMediaType->Release();
703 return QMediaRecorder::FormatError;
705 hr = sinkWriter->BeginWriting();
707 return QMediaRecorder::ResourceError;
709 m_sinkWriter = sinkWriter.Detach();
711 m_currentDuration = 0;
713 m_durationTimer.start();
717 m_pauseChanging =
false;
719 return QMediaRecorder::NoError;
724 QMutexLocker locker(&m_mutex);
726 if (m_sinkWriter && m_recording) {
728 HRESULT hr = m_sinkWriter->Finalize();
731 m_hasFinalized.wait(&m_mutex);
733 m_sinkWriter->Release();
734 m_sinkWriter =
nullptr;
736 QMetaObject::invokeMethod(
this,
"recordingError",
737 Qt::QueuedConnection, Q_ARG(
int, hr));
743 m_pauseChanging =
false;
745 m_durationTimer.stop();
747 m_currentDuration = -1;
752 if (!m_recording || m_paused)
754 m_pauseTime = m_lastTimestamp;
756 m_pauseChanging =
true;
762 if (!m_recording || !m_paused)
765 m_pauseChanging =
true;
774 if (riid == IID_IMFSourceReaderCallback) {
775 *ppvObject =
static_cast<IMFSourceReaderCallback*>(
this);
776 }
else if (riid == IID_IMFSinkWriterCallback) {
777 *ppvObject =
static_cast<IMFSinkWriterCallback*>(
this);
778 }
else if (riid == IID_IUnknown) {
779 *ppvObject =
static_cast<IUnknown*>(
static_cast<IMFSourceReaderCallback*>(
this));
781 *ppvObject =
nullptr;
782 return E_NOINTERFACE;
790 return InterlockedIncrement(&m_cRef);
795 LONG cRef = InterlockedDecrement(&m_cRef);
809 return m_frameHeight;
819 m_inputMuted = muted;
824 m_inputVolume = qBound(0.0, volume, 1.0);
829 QMutexLocker locker(&m_mutex);
831 m_outputMuted = muted;
833 if (m_active && m_monitorSink) {
834 IMFSimpleAudioVolume *audioVolume =
nullptr;
835 if (SUCCEEDED(MFGetService(m_monitorSink, MR_POLICY_VOLUME_SERVICE,
836 IID_PPV_ARGS(&audioVolume)))) {
837 audioVolume->SetMute(m_outputMuted);
838 audioVolume->Release();
845 QMutexLocker locker(&m_mutex);
847 m_outputVolume = qBound(0.0, volume, 1.0);
849 if (m_active && m_monitorSink) {
850 IMFSimpleAudioVolume *audioVolume =
nullptr;
851 if (SUCCEEDED(MFGetService(m_monitorSink, MR_POLICY_VOLUME_SERVICE,
852 IID_PPV_ARGS(&audioVolume)))) {
853 audioVolume->SetMasterVolume(
float(m_outputVolume));
854 audioVolume->Release();
861 if (m_currentDuration >= 0 && m_lastDuration != m_currentDuration) {
862 m_lastDuration = m_currentDuration;
863 emit durationChanged(m_currentDuration);
869 DWORD dwStreamFlags, LONGLONG llTimestamp,
872 QMutexLocker locker(&m_mutex);
874 if (FAILED(hrStatus)) {
875 emit streamingError(
int(hrStatus));
879 m_lastTimestamp = llTimestamp;
881 if ((dwStreamFlags & MF_SOURCE_READERF_ENDOFSTREAM) == MF_SOURCE_READERF_ENDOFSTREAM) {
883 emit streamingStopped();
888 emit streamingStarted();
892 if (m_monitorWriter && dwStreamIndex == m_sourceAudioStreamIndex)
893 m_monitorWriter->WriteSample(0, pSample);
898 m_timeOffset = llTimestamp;
899 m_firstFrame =
false;
900 emit recordingStarted();
903 if (m_pauseChanging) {
906 m_pauseTime = llTimestamp;
908 m_timeOffset += llTimestamp - m_pauseTime;
909 m_pauseChanging =
false;
913 if (m_sinkWriter && !m_paused) {
915 pSample->SetSampleTime(llTimestamp - m_timeOffset);
917 if (dwStreamIndex == m_sourceVideoStreamIndex) {
919 m_sinkWriter->WriteSample(m_sinkVideoStreamIndex, pSample);
921 }
else if (dwStreamIndex == m_sourceAudioStreamIndex) {
923 float volume = m_inputMuted ? 0.0f :
float(m_inputVolume);
926 if (volume != 1.0f) {
927 IMFMediaBuffer *mediaBuffer =
nullptr;
928 if (SUCCEEDED(pSample->ConvertToContiguousBuffer(&mediaBuffer))) {
931 BYTE *buffer =
nullptr;
933 if (SUCCEEDED(mediaBuffer->Lock(&buffer,
nullptr, &bufLen))) {
935 float *floatBuffer =
reinterpret_cast<
float*>(buffer);
937 for (DWORD i = 0; i < bufLen/4; ++i)
938 floatBuffer[i] *= volume;
940 mediaBuffer->Unlock();
942 mediaBuffer->Release();
946 m_sinkWriter->WriteSample(m_sinkAudioStreamIndex, pSample);
948 m_currentDuration = (llTimestamp - m_timeOffset) / 10000;
953 if (dwStreamIndex == m_sourceVideoStreamIndex) {
954 IMFMediaBuffer *mediaBuffer =
nullptr;
955 if (SUCCEEDED(pSample->ConvertToContiguousBuffer(&mediaBuffer))) {
958 BYTE *buffer =
nullptr;
960 if (SUCCEEDED(mediaBuffer->Lock(&buffer,
nullptr, &bufLen))) {
961 auto bytes = QByteArray(
reinterpret_cast<
char*>(buffer), bufLen);
962 QVideoFrameFormat format(QSize(m_frameWidth, m_frameHeight), m_pixelFormat);
964 QVideoFrame frame = QVideoFramePrivate::createFrame(
965 std::make_unique<QMemoryVideoBuffer>(std::move(bytes), m_stride),
969 frame.setStartTime(llTimestamp * 0.1);
971 LONGLONG duration = -1;
972 if (SUCCEEDED(pSample->GetSampleDuration(&duration)))
973 frame.setEndTime((llTimestamp + duration) * 0.1);
975 emit videoFrameChanged(frame);
977 mediaBuffer->Unlock();
979 mediaBuffer->Release();
985 m_sourceReader->ReadSample(MF_SOURCE_READER_ANY_STREAM,
986 0,
nullptr,
nullptr,
nullptr,
nullptr);
1005 QMutexLocker locker(&m_mutex);
1007 m_sinkWriter->Release();
1008 m_sinkWriter =
nullptr;
1010 emit recordingStopped();
1011 m_hasFinalized.notify_one();
1022#include "moc_qwindowsmediadevicereader_p.cpp"