| // Copyright 2012 The Chromium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef MEDIA_BASE_STREAM_PARSER_BUFFER_H_ |
| #define MEDIA_BASE_STREAM_PARSER_BUFFER_H_ |
| |
| #include <stddef.h> |
| #include <stdint.h> |
| |
| #include "base/time/time.h" |
| #include "media/base/decoder_buffer.h" |
| #include "media/base/demuxer_stream.h" |
| #include "media/base/media_export.h" |
| #include "media/base/stream_parser.h" |
| #include "media/base/timestamp_constants.h" |
| |
| namespace media { |
| |
| // Simple wrapper around base::TimeDelta that represents a decode timestamp. |
| // Making DecodeTimestamp a different type makes it easier to determine whether |
| // code is operating on presentation or decode timestamps and makes conversions |
| // between the two types explicit and easy to spot. |
| class DecodeTimestamp { |
| public: |
| constexpr DecodeTimestamp() = default; |
| |
| // Only operators that are actually used by the code have been defined. |
| // Reviewers should pay close attention to the addition of new operators. |
| constexpr bool operator<(const DecodeTimestamp& rhs) const { |
| return ts_ < rhs.ts_; |
| } |
| constexpr bool operator>(const DecodeTimestamp& rhs) const { |
| return ts_ > rhs.ts_; |
| } |
| constexpr bool operator==(const DecodeTimestamp& rhs) const { |
| return ts_ == rhs.ts_; |
| } |
| constexpr bool operator!=(const DecodeTimestamp& rhs) const { |
| return ts_ != rhs.ts_; |
| } |
| constexpr bool operator>=(const DecodeTimestamp& rhs) const { |
| return ts_ >= rhs.ts_; |
| } |
| constexpr bool operator<=(const DecodeTimestamp& rhs) const { |
| return ts_ <= rhs.ts_; |
| } |
| |
| constexpr base::TimeDelta operator-(const DecodeTimestamp& rhs) const { |
| return ts_ - rhs.ts_; |
| } |
| |
| constexpr DecodeTimestamp& operator+=(base::TimeDelta rhs) { |
| ts_ += rhs; |
| return *this; |
| } |
| |
| constexpr DecodeTimestamp& operator-=(base::TimeDelta rhs) { |
| ts_ -= rhs; |
| return *this; |
| } |
| |
| constexpr DecodeTimestamp operator+(base::TimeDelta rhs) const { |
| return DecodeTimestamp(ts_ + rhs); |
| } |
| |
| constexpr DecodeTimestamp operator-(base::TimeDelta rhs) const { |
| return DecodeTimestamp(ts_ - rhs); |
| } |
| |
| constexpr double operator/(base::TimeDelta rhs) const { return ts_ / rhs; } |
| constexpr int64_t IntDiv(base::TimeDelta rhs) const { |
| return ts_.IntDiv(rhs); |
| } |
| |
| static constexpr DecodeTimestamp FromSecondsD(double seconds) { |
| return DecodeTimestamp(base::Seconds(seconds)); |
| } |
| |
| static constexpr DecodeTimestamp FromMilliseconds(int64_t milliseconds) { |
| return DecodeTimestamp(base::Milliseconds(milliseconds)); |
| } |
| |
| static constexpr DecodeTimestamp FromMicroseconds(int64_t microseconds) { |
| return DecodeTimestamp(base::Microseconds(microseconds)); |
| } |
| |
| // This method is used to explicitly call out when presentation timestamps |
| // are being converted to a decode timestamp. |
| static constexpr DecodeTimestamp FromPresentationTime( |
| base::TimeDelta timestamp) { |
| return DecodeTimestamp(timestamp); |
| } |
| |
| constexpr double InSecondsF() const { return ts_.InSecondsF(); } |
| int64_t InMilliseconds() const { return ts_.InMilliseconds(); } |
| constexpr int64_t InMicroseconds() const { return ts_.InMicroseconds(); } |
| |
| constexpr bool is_inf() const { return ts_.is_inf(); } |
| |
| // TODO(acolwell): Remove once all the hacks are gone. This method is called |
| // by hacks where a decode time is being used as a presentation time. |
| constexpr base::TimeDelta ToPresentationTime() const { return ts_; } |
| |
| private: |
| constexpr explicit DecodeTimestamp(base::TimeDelta timestamp) |
| : ts_(timestamp) {} |
| |
| base::TimeDelta ts_; |
| }; |
| |
| // Assert assumptions necessary for DecodeTimestamp analogues of |
| // base::TimeDelta::is_inf(), media::kNoTimestamp and media::kInfiniteDuration. |
| static_assert(kNoTimestamp.is_min() && kNoTimestamp.is_inf()); |
| static_assert(kInfiniteDuration.is_max() && kInfiniteDuration.is_inf()); |
| |
| // Indicates an invalid or missing decode timestamp. |
| constexpr DecodeTimestamp kNoDecodeTimestamp = |
| DecodeTimestamp::FromPresentationTime(kNoTimestamp); |
| |
| // Similar to media::kInfiniteDuration, indicates a decode timestamp of positive |
| // infinity. |
| constexpr DecodeTimestamp kMaxDecodeTimestamp = |
| DecodeTimestamp::FromPresentationTime(kInfiniteDuration); |
| |
| class MEDIA_EXPORT StreamParserBuffer : public DecoderBuffer { |
| public: |
| // Value used to signal an invalid decoder config ID. |
| enum { kInvalidConfigId = -1 }; |
| |
| typedef DemuxerStream::Type Type; |
| typedef StreamParser::TrackId TrackId; |
| |
| static scoped_refptr<StreamParserBuffer> CreateEOSBuffer(); |
| |
| static scoped_refptr<StreamParserBuffer> CopyFrom(const uint8_t* data, |
| int data_size, |
| bool is_key_frame, |
| Type type, |
| TrackId track_id); |
| static scoped_refptr<StreamParserBuffer> CopyFrom(const uint8_t* data, |
| int data_size, |
| const uint8_t* side_data, |
| int side_data_size, |
| bool is_key_frame, |
| Type type, |
| TrackId track_id); |
| #if !defined(STARBOARD) |
| static scoped_refptr<StreamParserBuffer> FromExternalMemory( |
| std::unique_ptr<ExternalMemory> external_memory, |
| bool is_key_frame, |
| Type type, |
| TrackId track_id); |
| #endif // !defined(STARBOARD) |
| |
| StreamParserBuffer(const StreamParserBuffer&) = delete; |
| StreamParserBuffer& operator=(const StreamParserBuffer&) = delete; |
| |
| // Decode timestamp. If not explicitly set, or set to kNoTimestamp, the |
| // value will be taken from the normal timestamp. |
| DecodeTimestamp GetDecodeTimestamp() const; |
| void SetDecodeTimestamp(DecodeTimestamp timestamp); |
| |
| // Gets/sets the ID of the decoder config associated with this buffer. |
| int GetConfigId() const; |
| void SetConfigId(int config_id); |
| |
| // Gets the parser's media type associated with this buffer. Value is |
| // meaningless for EOS buffers. |
| Type type() const { return type_; } |
| const char* GetTypeName() const; |
| |
| // Gets the parser's track ID associated with this buffer. Value is |
| // meaningless for EOS buffers. |
| TrackId track_id() const { return track_id_; } |
| |
| // Specifies a buffer which must be decoded prior to this one to ensure this |
| // buffer can be accurately decoded. The given buffer must be of the same |
| // type, must not have any discard padding, and must not be an end of stream |
| // buffer. |preroll| is not copied. |
| // |
| // It's expected that this preroll buffer will be discarded entirely post |
| // decoding. As such it's discard_padding() will be set to kInfiniteDuration. |
| // |
| // All future timestamp, decode timestamp, config id, or track id changes to |
| // this buffer will be applied to the preroll buffer as well. |
| void SetPrerollBuffer(scoped_refptr<StreamParserBuffer> preroll); |
| scoped_refptr<StreamParserBuffer> preroll_buffer() { return preroll_buffer_; } |
| |
| void set_timestamp(base::TimeDelta timestamp) override; |
| |
| bool is_duration_estimated() const { return is_duration_estimated_; } |
| |
| void set_is_duration_estimated(bool is_estimated) { |
| is_duration_estimated_ = is_estimated; |
| } |
| |
| private: |
| #if !defined(STARBOARD) |
| StreamParserBuffer(std::unique_ptr<ExternalMemory> external_memory, |
| bool is_key_frame, |
| Type type, |
| TrackId track_id); |
| #endif // !defined(STARBOARD) |
| StreamParserBuffer(const uint8_t* data, |
| int data_size, |
| const uint8_t* side_data, |
| int side_data_size, |
| bool is_key_frame, |
| Type type, |
| TrackId track_id); |
| ~StreamParserBuffer() override; |
| |
| DecodeTimestamp decode_timestamp_ = kNoDecodeTimestamp; |
| int config_id_ = kInvalidConfigId; |
| Type type_; |
| TrackId track_id_; |
| scoped_refptr<StreamParserBuffer> preroll_buffer_; |
| bool is_duration_estimated_ = false; |
| }; |
| |
| } // namespace media |
| |
| #endif // MEDIA_BASE_STREAM_PARSER_BUFFER_H_ |