/* -*- 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.*/ #if !defined(DASHRepDecoder_h_) #define DASHRepDecoder_h_ #include "Representation.h" #include "DASHDecoder.h" #include "WebMDecoder.h" #include "WebMReader.h" #include "MediaDecoder.h" namespace mozilla { namespace layers { class ImageContainer; } class DASHDecoder; class DASHRepReader; class DASHRepDecoder : public MediaDecoder { public: typedef mozilla::net::Representation Representation; typedef mozilla::net::SegmentBase SegmentBase; typedef mozilla::layers::ImageContainer ImageContainer; // Constructor takes a ptr to the main decoder. DASHRepDecoder(DASHDecoder* aMainDecoder) : mMainDecoder(aMainDecoder), mMPDRepresentation(nullptr), mMetadataChunkCount(0), mCurrentByteRange(), mSubsegmentIdx(-1), mReader(nullptr) { MOZ_COUNT_CTOR(DASHRepDecoder); } ~DASHRepDecoder() { MOZ_COUNT_DTOR(DASHRepDecoder); } // Clone not supported; just return nullptr. virtual MediaDecoder* Clone() { return nullptr; } // Called by the main decoder at creation time; points to the main state // machine managed by the main decoder. Called on the main thread only. nsresult SetStateMachine(MediaDecoderStateMachine* aSM); private: // Overridden to return the ptr set by SetStateMachine. Called on the main // thread only. MediaDecoderStateMachine* CreateStateMachine(); public: // Called by DASHDecoder at creation time; points to the media resource // for this decoder's |Representation|. Called on the main thread only. void SetResource(MediaResource* aResource); // Sets the |Representation| object for this decoder. Called on the main // thread. void SetMPDRepresentation(Representation const * aRep); // Called from DASHDecoder on main thread; Starts media stream download. virtual nsresult Load(nsIStreamListener** aListener = nullptr, MediaDecoder* aCloneDonor = nullptr) MOZ_OVERRIDE; // Loads the next byte range (or first one on first call). Called on the main // thread only. void LoadNextByteRange(); // Returns true if the subsegment is already in the media cache. bool IsSubsegmentCached(int32_t aSubsegmentIdx); // Calls from DASHRepDecoder. Called on the main thread only. void SetReader(WebMReader* aReader); // Called if the media file encounters a network error. Call on the main // thread only. void NetworkError(); // Called from reader during ReadMetadata. This should be ignored here, and // instead, duration should be set following MPD parsing. void SetMediaDuration(int64_t aDuration) MOZ_OVERRIDE { }; // Set the duration of the media resource in units of seconds. // This is called via a channel listener if it can pick up the duration // from a content header. Must be called from the main thread only. virtual void SetDuration(double aDuration); // Set media stream as infinite. Called on the main thread only. void SetInfinite(bool aInfinite); // Sets media stream as seekable. Called on main thread only. void SetMediaSeekable(bool aSeekable); // Fire progress events if needed according to the time and byte // constraints outlined in the specification. aTimer is true // if the method is called as a result of the progress timer rather // than the result of downloaded data. void Progress(bool aTimer); // Called as data arrives on the stream and is read into the cache. Called // on the main thread only. void NotifyDataArrived(const char* aBuffer, uint32_t aLength, int64_t aOffset); // Called by MediaResource when some data has been received. // Call on the main thread only. void NotifyBytesDownloaded(); // Notify that a byte range request has been completed by the media resource. // Called on the main thread only. void NotifyDownloadEnded(nsresult aStatus); // Called asynchronously by |LoadNextByteRange| if the data is already in the // media cache. This will call NotifyDownloadEnded on the main thread with // |aStatus| of NS_OK. void DoNotifyDownloadEnded(); // Called by MediaResource when the "cache suspended" status changes. // If MediaResource::IsSuspendedByCache returns true, then the decoder // should stop buffering or otherwise waiting for download progress and // start consuming data, if possible, because the cache is full. void NotifySuspendedStatusChanged(); // Increments the parsed and decoded frame counters by the passed in counts. // Can be called on any thread. void NotifyDecodedFrames(uint32_t aParsed, uint32_t aDecoded) MOZ_OVERRIDE { if (mMainDecoder) {mMainDecoder->NotifyDecodedFrames(aParsed, aDecoded); } } // Gets a byte range containing the byte offset. Call on main thread only. nsresult GetByteRangeForSeek(int64_t const aOffset, MediaByteRange& aByteRange); // Gets the number of data byte ranges (not inc. metadata). uint32_t GetNumDataByteRanges() { return mByteRanges.Length(); } // Notify that a switch is about to happen. Called on the main thread. void PrepareForSwitch(); // Returns true if the current thread is the state machine thread. bool OnStateMachineThread() const MOZ_OVERRIDE; // Returns true if the current thread is the decode thread. bool OnDecodeThread() const MOZ_OVERRIDE; // Returns main decoder's monitor for synchronised access. ReentrantMonitor& GetReentrantMonitor() MOZ_OVERRIDE; // Called on the decode thread from WebMReader. ImageContainer* GetImageContainer() MOZ_OVERRIDE; // Called when Metadata has been read; notifies that index data is read. // Called on the decode thread only. void OnReadMetadataCompleted() MOZ_OVERRIDE; // Stop updating the bytes downloaded for progress notifications. Called // when seeking to prevent wild changes to the progress notification. // Must be called with the decoder monitor held. void StopProgressUpdates() MOZ_OVERRIDE; // Allow updating the bytes downloaded for progress notifications. Must // be called with the decoder monitor held. void StartProgressUpdates() MOZ_OVERRIDE; // Overridden to cleanup ref to |DASHDecoder|. Called on main thread only. void Shutdown() { NS_ASSERTION(NS_IsMainThread(), "Should be on main thread."); // Remove ref to state machine before |MediaDecoder|::|Shutdown|, since // |DASHDecoder| is responsible for its shutdown. mDecoderStateMachine = nullptr; // Call parent class shutdown. MediaDecoder::Shutdown(); NS_ENSURE_TRUE_VOID(mShuttingDown); // Cleanup ref to main decoder. mMainDecoder = nullptr; } // Drop reference to state machine and mReader (owned by state machine). // Only called during shutdown dance. void ReleaseStateMachine(); // Notifies the element that decoding has failed. void DecodeError(); private: // Populates |mByteRanges| by calling |GetIndexByteRanges| from |mReader|. // Called on the main thread only. nsresult PopulateByteRanges(); // The main decoder. nsRefPtr mMainDecoder; // This decoder's MPD |Representation| object. Representation const * mMPDRepresentation; // Countdown var for loading metadata byte ranges. uint16_t mMetadataChunkCount; // All the byte ranges for this |Representation|. nsTArray mByteRanges; // Byte range for the init and index bytes. MediaByteRange mInitByteRange; MediaByteRange mIndexByteRange; // The current byte range being requested. MediaByteRange mCurrentByteRange; // Index of the current byte range. Initialized to -1. int32_t mSubsegmentIdx; // Ptr to the reader object for this |Representation|. Owned by state // machine. DASHRepReader* mReader; }; } // namespace mozilla #endif //DASHRepDecoder_h_