mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
216fe28466
--HG-- rename : content/media/nsAudioAvailableEventManager.cpp => content/media/AudioAvailableEventManager.cpp rename : content/media/nsAudioAvailableEventManager.h => content/media/AudioAvailableEventManager.h rename : content/media/nsAudioStream.cpp => content/media/AudioStream.cpp rename : content/media/nsAudioStream.h => content/media/AudioStream.h rename : content/media/nsMediaCache.cpp => content/media/MediaCache.cpp rename : content/media/nsMediaCache.h => content/media/MediaCache.h rename : content/media/nsBuiltinDecoder.cpp => content/media/MediaDecoder.cpp rename : content/media/nsBuiltinDecoder.h => content/media/MediaDecoder.h rename : content/media/nsBuiltinDecoderReader.cpp => content/media/MediaDecoderReader.cpp rename : content/media/nsBuiltinDecoderReader.h => content/media/MediaDecoderReader.h rename : content/media/nsBuiltinDecoderStateMachine.cpp => content/media/MediaDecoderStateMachine.cpp rename : content/media/nsBuiltinDecoderStateMachine.h => content/media/MediaDecoderStateMachine.h rename : content/media/dash/nsDASHDecoder.cpp => content/media/dash/DASHDecoder.cpp rename : content/media/dash/nsDASHDecoder.h => content/media/dash/DASHDecoder.h rename : content/media/dash/nsDASHReader.cpp => content/media/dash/DASHReader.cpp rename : content/media/dash/nsDASHReader.h => content/media/dash/DASHReader.h rename : content/media/dash/nsDASHRepDecoder.cpp => content/media/dash/DASHRepDecoder.cpp rename : content/media/dash/nsDASHRepDecoder.h => content/media/dash/DASHRepDecoder.h rename : content/media/gstreamer/nsGStreamerDecoder.cpp => content/media/gstreamer/GStreamerDecoder.cpp rename : content/media/gstreamer/nsGStreamerDecoder.h => content/media/gstreamer/GStreamerDecoder.h rename : content/media/gstreamer/nsGStreamerReader.cpp => content/media/gstreamer/GStreamerReader.cpp rename : content/media/gstreamer/nsGStreamerReader.h => content/media/gstreamer/GStreamerReader.h rename : content/media/ogg/nsOggCodecState.cpp => content/media/ogg/OggCodecState.cpp rename : content/media/ogg/nsOggCodecState.h => content/media/ogg/OggCodecState.h rename : content/media/ogg/nsOggDecoder.cpp => content/media/ogg/OggDecoder.cpp rename : content/media/ogg/nsOggDecoder.h => content/media/ogg/OggDecoder.h rename : content/media/ogg/nsOggReader.cpp => content/media/ogg/OggReader.cpp rename : content/media/ogg/nsOggReader.h => content/media/ogg/OggReader.h rename : content/media/omx/nsMediaOmxDecoder.cpp => content/media/omx/MediaOmxDecoder.cpp rename : content/media/omx/nsMediaOmxDecoder.h => content/media/omx/MediaOmxDecoder.h rename : content/media/omx/nsMediaOmxReader.cpp => content/media/omx/MediaOmxReader.cpp rename : content/media/omx/nsMediaOmxReader.h => content/media/omx/MediaOmxReader.h rename : content/media/plugins/nsMediaPluginDecoder.cpp => content/media/plugins/MediaPluginDecoder.cpp rename : content/media/plugins/nsMediaPluginDecoder.h => content/media/plugins/MediaPluginDecoder.h rename : content/media/plugins/nsMediaPluginHost.cpp => content/media/plugins/MediaPluginHost.cpp rename : content/media/plugins/nsMediaPluginHost.h => content/media/plugins/MediaPluginHost.h rename : content/media/plugins/nsMediaPluginReader.cpp => content/media/plugins/MediaPluginReader.cpp rename : content/media/plugins/nsMediaPluginReader.h => content/media/plugins/MediaPluginReader.h rename : content/media/raw/nsRawDecoder.cpp => content/media/raw/RawDecoder.cpp rename : content/media/raw/nsRawDecoder.h => content/media/raw/RawDecoder.h rename : content/media/raw/nsRawReader.cpp => content/media/raw/RawReader.cpp rename : content/media/raw/nsRawReader.h => content/media/raw/RawReader.h rename : content/media/raw/nsRawStructs.h => content/media/raw/RawStructs.h rename : content/media/wave/nsWaveDecoder.cpp => content/media/wave/WaveDecoder.cpp rename : content/media/wave/nsWaveDecoder.h => content/media/wave/WaveDecoder.h rename : content/media/wave/nsWaveReader.cpp => content/media/wave/WaveReader.cpp rename : content/media/wave/nsWaveReader.h => content/media/wave/WaveReader.h rename : content/media/webm/nsWebMBufferedParser.cpp => content/media/webm/WebMBufferedParser.cpp rename : content/media/webm/nsWebMBufferedParser.h => content/media/webm/WebMBufferedParser.h rename : content/media/webm/nsWebMDecoder.cpp => content/media/webm/WebMDecoder.cpp rename : content/media/webm/nsWebMDecoder.h => content/media/webm/WebMDecoder.h rename : content/media/webm/nsWebMReader.cpp => content/media/webm/WebMReader.cpp rename : content/media/webm/nsWebMReader.h => content/media/webm/WebMReader.h
331 lines
10 KiB
C++
331 lines
10 KiB
C++
/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
/* vim:set ts=2 sw=2 sts=2 et cindent: */
|
|
|
|
/* This Source Code Form Is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
/* DASH - Dynamic Adaptive Streaming over HTTP
|
|
*
|
|
* DASH is an adaptive bitrate streaming technology where a multimedia file is
|
|
* partitioned into one or more segments and delivered to a client using HTTP.
|
|
*
|
|
* see DASHDecoder.cpp for info on DASH interaction with the media engine.*/
|
|
|
|
#include "nsTimeRanges.h"
|
|
#include "VideoFrameContainer.h"
|
|
#include "MediaDecoder.h"
|
|
#include "DASHReader.h"
|
|
|
|
namespace mozilla {
|
|
|
|
#ifdef PR_LOGGING
|
|
extern PRLogModuleInfo* gMediaDecoderLog;
|
|
#define LOG(msg, ...) PR_LOG(gMediaDecoderLog, PR_LOG_DEBUG, \
|
|
("%p [DASHReader] " msg, this, __VA_ARGS__))
|
|
#define LOG1(msg) PR_LOG(gMediaDecoderLog, PR_LOG_DEBUG, \
|
|
("%p [DASHReader] " msg, this))
|
|
#else
|
|
#define LOG(msg, ...)
|
|
#define LOG1(msg)
|
|
#endif
|
|
|
|
nsresult
|
|
DASHReader::Init(MediaDecoderReader* aCloneDonor)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Should be on main thread.");
|
|
|
|
ReentrantMonitorAutoEnter mon(mDecoder->GetReentrantMonitor());
|
|
NS_ASSERTION(mAudioReaders.Length() != 0 && mVideoReaders.Length() != 0,
|
|
"Audio and video readers should exist already.");
|
|
|
|
nsresult rv;
|
|
for (uint i = 0; i < mAudioReaders.Length(); i++) {
|
|
rv = mAudioReaders[i]->Init(nullptr);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
for (uint i = 0; i < mVideoReaders.Length(); i++) {
|
|
rv = mVideoReaders[i]->Init(nullptr);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
void
|
|
DASHReader::AddAudioReader(MediaDecoderReader* aAudioReader)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Should be on main thread.");
|
|
NS_ENSURE_TRUE(aAudioReader, );
|
|
|
|
ReentrantMonitorAutoEnter mon(mDecoder->GetReentrantMonitor());
|
|
|
|
mAudioReaders.AppendElement(aAudioReader);
|
|
// XXX For now, just pick the first reader to be default.
|
|
if (!mAudioReader)
|
|
mAudioReader = aAudioReader;
|
|
}
|
|
|
|
void
|
|
DASHReader::AddVideoReader(MediaDecoderReader* aVideoReader)
|
|
{
|
|
NS_ASSERTION(NS_IsMainThread(), "Should be on main thread.");
|
|
NS_ENSURE_TRUE(aVideoReader, );
|
|
|
|
ReentrantMonitorAutoEnter mon(mDecoder->GetReentrantMonitor());
|
|
|
|
mVideoReaders.AppendElement(aVideoReader);
|
|
// XXX For now, just pick the first reader to be default.
|
|
if (!mVideoReader)
|
|
mVideoReader = aVideoReader;
|
|
}
|
|
|
|
int64_t
|
|
DASHReader::VideoQueueMemoryInUse()
|
|
{
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
return (mVideoReader ? mVideoReader->VideoQueueMemoryInUse() : 0);
|
|
}
|
|
|
|
int64_t
|
|
DASHReader::AudioQueueMemoryInUse()
|
|
{
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
return (mAudioReader ? mAudioReader->AudioQueueMemoryInUse() : 0);
|
|
}
|
|
|
|
bool
|
|
DASHReader::DecodeVideoFrame(bool &aKeyframeSkip,
|
|
int64_t aTimeThreshold)
|
|
{
|
|
NS_ASSERTION(mDecoder->OnDecodeThread(), "Should be on decode thread.");
|
|
if (mVideoReader) {
|
|
return mVideoReader->DecodeVideoFrame(aKeyframeSkip, aTimeThreshold);
|
|
} else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool
|
|
DASHReader::DecodeAudioData()
|
|
{
|
|
NS_ASSERTION(mDecoder->OnDecodeThread(), "Should be on decode thread.");
|
|
return (mAudioReader ? mAudioReader->DecodeAudioData() : false);
|
|
}
|
|
|
|
nsresult
|
|
DASHReader::ReadMetadata(nsVideoInfo* aInfo,
|
|
MetadataTags** aTags)
|
|
{
|
|
NS_ASSERTION(mDecoder->OnDecodeThread(), "Should be on decode thread.");
|
|
|
|
// Wait for MPD to be parsed and child readers created.
|
|
LOG1("Waiting for metadata download.");
|
|
nsresult rv = WaitForMetadata();
|
|
// If we get an abort, return silently; the decoder is shutting down.
|
|
if (NS_ERROR_ABORT == rv) {
|
|
return NS_OK;
|
|
}
|
|
// Verify no other errors before continuing.
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// Get metadata from child readers.
|
|
nsVideoInfo audioInfo, videoInfo;
|
|
|
|
if (mVideoReader) {
|
|
rv = mVideoReader->ReadMetadata(&videoInfo, aTags);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
mInfo.mHasVideo = videoInfo.mHasVideo;
|
|
mInfo.mDisplay = videoInfo.mDisplay;
|
|
}
|
|
if (mAudioReader) {
|
|
rv = mAudioReader->ReadMetadata(&audioInfo, aTags);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
mInfo.mHasAudio = audioInfo.mHasAudio;
|
|
mInfo.mAudioRate = audioInfo.mAudioRate;
|
|
mInfo.mAudioChannels = audioInfo.mAudioChannels;
|
|
mInfo.mStereoMode = audioInfo.mStereoMode;
|
|
}
|
|
|
|
*aInfo = mInfo;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DASHReader::Seek(int64_t aTime,
|
|
int64_t aStartTime,
|
|
int64_t aEndTime,
|
|
int64_t aCurrentTime)
|
|
{
|
|
NS_ASSERTION(mDecoder->OnDecodeThread(), "Should be on decode thread.");
|
|
|
|
nsresult rv;
|
|
|
|
if (mAudioReader) {
|
|
rv = mAudioReader->Seek(aTime, aStartTime, aEndTime, aCurrentTime);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
if (mVideoReader) {
|
|
rv = mVideoReader->Seek(aTime, aStartTime, aEndTime, aCurrentTime);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsresult
|
|
DASHReader::GetBuffered(nsTimeRanges* aBuffered,
|
|
int64_t aStartTime)
|
|
{
|
|
NS_ENSURE_ARG(aBuffered);
|
|
|
|
MediaResource* resource = nullptr;
|
|
MediaDecoder* decoder = nullptr;
|
|
|
|
// Need to find intersect of |nsTimeRanges| for audio and video.
|
|
nsTimeRanges audioBuffered, videoBuffered;
|
|
uint32_t audioRangeCount, videoRangeCount;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
// First, get buffered ranges for sub-readers.
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
if (mAudioReader) {
|
|
decoder = mAudioReader->GetDecoder();
|
|
NS_ENSURE_TRUE(decoder, NS_ERROR_NULL_POINTER);
|
|
resource = decoder->GetResource();
|
|
NS_ENSURE_TRUE(resource, NS_ERROR_NULL_POINTER);
|
|
resource->Pin();
|
|
rv = mAudioReader->GetBuffered(&audioBuffered, aStartTime);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
resource->Unpin();
|
|
rv = audioBuffered.GetLength(&audioRangeCount);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
if (mVideoReader) {
|
|
decoder = mVideoReader->GetDecoder();
|
|
NS_ENSURE_TRUE(decoder, NS_ERROR_NULL_POINTER);
|
|
resource = decoder->GetResource();
|
|
NS_ENSURE_TRUE(resource, NS_ERROR_NULL_POINTER);
|
|
resource->Pin();
|
|
rv = mVideoReader->GetBuffered(&videoBuffered, aStartTime);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
resource->Unpin();
|
|
rv = videoBuffered.GetLength(&videoRangeCount);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
|
|
// Now determine buffered data for available sub-readers.
|
|
if (mAudioReader && mVideoReader) {
|
|
// Calculate intersecting ranges.
|
|
for (uint32_t i = 0; i < audioRangeCount; i++) {
|
|
// |A|udio, |V|ideo, |I|ntersect.
|
|
double startA, startV, startI;
|
|
double endA, endV, endI;
|
|
rv = audioBuffered.Start(i, &startA);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
rv = audioBuffered.End(i, &endA);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
for (uint32_t j = 0; j < videoRangeCount; j++) {
|
|
rv = videoBuffered.Start(i, &startV);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
rv = videoBuffered.End(i, &endV);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
// If video block is before audio block, compare next video block.
|
|
if (startA > endV) {
|
|
continue;
|
|
// If video block is after audio block, all of them are; compare next
|
|
// audio block.
|
|
} else if (endA < startV) {
|
|
break;
|
|
}
|
|
// Calculate intersections of current audio and video blocks.
|
|
startI = (startA > startV) ? startA : startV;
|
|
endI = (endA > endV) ? endV : endA;
|
|
aBuffered->Add(startI, endI);
|
|
}
|
|
}
|
|
} else if (mAudioReader) {
|
|
*aBuffered = audioBuffered;
|
|
} else if (mVideoReader) {
|
|
*aBuffered = videoBuffered;
|
|
} else {
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
VideoData*
|
|
DASHReader::FindStartTime(int64_t& aOutStartTime)
|
|
{
|
|
NS_ASSERTION(mDecoder->OnStateMachineThread() || mDecoder->OnDecodeThread(),
|
|
"Should be on state machine or decode thread.");
|
|
|
|
// Extract the start times of the bitstreams in order to calculate
|
|
// the duration.
|
|
int64_t videoStartTime = INT64_MAX;
|
|
int64_t audioStartTime = INT64_MAX;
|
|
VideoData* videoData = nullptr;
|
|
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
if (HasVideo()) {
|
|
// Forward to video reader.
|
|
videoData = mVideoReader->DecodeToFirstVideoData();
|
|
if (videoData) {
|
|
videoStartTime = videoData->mTime;
|
|
}
|
|
}
|
|
if (HasAudio()) {
|
|
// Forward to audio reader.
|
|
AudioData* audioData = mAudioReader->DecodeToFirstAudioData();
|
|
if (audioData) {
|
|
audioStartTime = audioData->mTime;
|
|
}
|
|
}
|
|
|
|
int64_t startTime = NS_MIN(videoStartTime, audioStartTime);
|
|
if (startTime != INT64_MAX) {
|
|
aOutStartTime = startTime;
|
|
}
|
|
|
|
return videoData;
|
|
}
|
|
|
|
MediaQueue<AudioData>&
|
|
DASHReader::AudioQueue()
|
|
{
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
NS_ASSERTION(mAudioReader, "mAudioReader is NULL!");
|
|
return mAudioReader->AudioQueue();
|
|
}
|
|
|
|
MediaQueue<VideoData>&
|
|
DASHReader::VideoQueue()
|
|
{
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
NS_ASSERTION(mVideoReader, "mVideoReader is NULL!");
|
|
return mVideoReader->VideoQueue();
|
|
}
|
|
|
|
bool
|
|
DASHReader::IsSeekableInBufferedRanges()
|
|
{
|
|
ReentrantMonitorConditionallyEnter mon(!mDecoder->OnDecodeThread(),
|
|
mDecoder->GetReentrantMonitor());
|
|
// At least one subreader must exist, and all subreaders must return true.
|
|
return (mVideoReader || mAudioReader) &&
|
|
!((mVideoReader && !mVideoReader->IsSeekableInBufferedRanges()) ||
|
|
(mAudioReader && !mAudioReader->IsSeekableInBufferedRanges()));
|
|
}
|
|
|
|
} // namespace mozilla
|
|
|