From 81c1d8415c95f10bb92a580442df731a59db522f Mon Sep 17 00:00:00 2001 From: Tom Finegan Date: Wed, 8 Jan 2014 17:29:44 -0800 Subject: [PATCH] mkvparser: Add support for CodecDelay, DiscardPadding, and SeekPreRoll elements. Change-Id: Ic9e6ebcc2ba533f8cc1ab933d0bf55782ccab932 --- mkvparser.cpp | 82 +++++++++++++++++++++++++++++++++++++++++---------- mkvparser.hpp | 27 +++++++++++------ sample.cpp | 13 ++++++-- 3 files changed, 94 insertions(+), 28 deletions(-) diff --git a/mkvparser.cpp b/mkvparser.cpp index 3569acc..a63c2fa 100644 --- a/mkvparser.cpp +++ b/mkvparser.cpp @@ -5517,7 +5517,9 @@ Track::Info::Info(): codecId(NULL), codecNameAsUTF8(NULL), codecPrivate(NULL), - codecPrivateSize(0) + codecPrivateSize(0), + codecDelay(0), + seekPreRoll(0) { } @@ -5581,6 +5583,8 @@ int Track::Info::Copy(Info& dst) const dst.type = type; dst.number = number; dst.defaultDuration = defaultDuration; + dst.codecDelay = codecDelay; + dst.seekPreRoll = seekPreRoll; dst.uid = uid; dst.lacing = lacing; dst.settings = settings; @@ -5684,6 +5688,16 @@ unsigned long long Track::GetDefaultDuration() const return m_info.defaultDuration; } +unsigned long long Track::GetCodecDelay() const +{ + return m_info.codecDelay; +} + +unsigned long long Track::GetSeekPreRoll() const +{ + return m_info.seekPreRoll; +} + long Track::GetFirst(const BlockEntry*& pBlockEntry) const { const Cluster* pCluster = m_pSegment->GetFirst(); @@ -6766,6 +6780,15 @@ long Tracks::ParseTrackEntry( if (status) return status; } + else if (id == 0x16AA) //Codec Delay + { + info.codecDelay = UnserializeUInt(pReader, pos, size); + + } + else if (id == 0x16BB) //Seek Pre Roll + { + info.seekPreRoll = UnserializeUInt(pReader, pos, size); + } pos += size; //consume payload assert(pos <= track_stop); @@ -7519,7 +7542,9 @@ long Cluster::ParseSimpleBlock( return E_BUFFER_NOT_FULL; } - status = CreateBlock(0x23, block_start, block_size); //simple block id + status = CreateBlock(0x23, //simple block id + block_start, block_size, + 0); //DiscardPadding if (status != 0) return status; @@ -7558,6 +7583,8 @@ long Cluster::ParseBlockGroup( return E_BUFFER_NOT_FULL; } + long long discard_padding = 0; + while (pos < payload_stop) { //parse sub-block element ID @@ -7634,6 +7661,19 @@ long Cluster::ParseBlockGroup( if (size == unknown_size) return E_FILE_FORMAT_INVALID; + if (id == 0x35A2) //DiscardPadding + { + result = GetUIntLength(pReader, pos, len); + + if (result < 0) //error + return static_cast(result); + + status = UnserializeInt(pReader, pos, len, discard_padding); + + if (status < 0) //error + return status; + } + if (id != 0x21) //sub-part of BlockGroup is not a Block { pos += size; //consume sub-part of block group @@ -7761,8 +7801,9 @@ long Cluster::ParseBlockGroup( assert(pos == payload_stop); - status = CreateBlock(0x20, payload_start, payload_size); //BlockGroup ID - + status = CreateBlock(0x20, //BlockGroup ID + payload_start, payload_size, + discard_padding); if (status != 0) return status; @@ -8265,7 +8306,8 @@ long long Cluster::GetLastTime() const long Cluster::CreateBlock( long long id, long long pos, //absolute pos of payload - long long size) + long long size, + long long discard_padding) { assert((id == 0x20) || (id == 0x23)); //BlockGroup or SimpleBlock @@ -8308,15 +8350,16 @@ long Cluster::CreateBlock( } if (id == 0x20) //BlockGroup ID - return CreateBlockGroup(pos, size); + return CreateBlockGroup(pos, size, discard_padding); else //SimpleBlock ID return CreateSimpleBlock(pos, size); } long Cluster::CreateBlockGroup( - long long st, - long long sz) + long long start_offset, + long long size, + long long discard_padding) { assert(m_entries); assert(m_entries_size > 0); @@ -8325,8 +8368,8 @@ long Cluster::CreateBlockGroup( IMkvReader* const pReader = m_pSegment->m_pReader; - long long pos = st; - const long long stop = st + sz; + long long pos = start_offset; + const long long stop = start_offset + size; //For WebM files, there is a bias towards previous reference times //(in order to support alt-ref frames, which refer back to the previous @@ -8407,7 +8450,8 @@ long Cluster::CreateBlockGroup( bsize, prev, next, - duration); + duration, + discard_padding); if (pEntry == NULL) return -1; //generic error @@ -8984,7 +9028,7 @@ SimpleBlock::SimpleBlock( long long start, long long size) : BlockEntry(pCluster, idx), - m_block(start, size) + m_block(start, size, 0) { } @@ -9014,9 +9058,10 @@ BlockGroup::BlockGroup( long long block_size, long long prev, long long next, - long long duration) : + long long duration, + long long discard_padding) : BlockEntry(pCluster, idx), - m_block(block_start, block_size), + m_block(block_start, block_size, discard_padding), m_prev(prev), m_next(next), m_duration(duration) @@ -9082,14 +9127,15 @@ long long BlockGroup::GetDurationTimeCode() const return m_duration; } -Block::Block(long long start, long long size_) : +Block::Block(long long start, long long size_, long long discard_padding) : m_start(start), m_size(size_), m_track(0), m_timecode(-1), m_flags(0), m_frames(NULL), - m_frame_count(-1) + m_frame_count(-1), + m_discard_padding(discard_padding) { } @@ -9535,5 +9581,9 @@ long Block::Frame::Read(IMkvReader* pReader, unsigned char* buf) const return status; } +long long Block::GetDiscardPadding() const +{ + return m_discard_padding; +} } //end namespace mkvparser diff --git a/mkvparser.hpp b/mkvparser.hpp index cb3311c..d9c02b8 100644 --- a/mkvparser.hpp +++ b/mkvparser.hpp @@ -83,7 +83,7 @@ public: const long long m_start; const long long m_size; - Block(long long start, long long size); + Block(long long start, long long size, long long discard_padding); ~Block(); long Parse(const Cluster*); @@ -110,6 +110,8 @@ public: const Frame& GetFrame(int frame_index) const; + long long GetDiscardPadding() const; + private: long long m_track; //Track::Number() short m_timecode; //relative to cluster @@ -118,6 +120,8 @@ private: Frame* m_frames; int m_frame_count; +protected: + const long long m_discard_padding; }; @@ -178,7 +182,8 @@ public: long long block_size, //size of block's payload long long prev, long long next, - long long duration); + long long duration, + long long discard_padding); long Parse(); @@ -194,7 +199,6 @@ private: const long long m_prev; const long long m_next; const long long m_duration; - }; /////////////////////////////////////////////////////////////// @@ -355,6 +359,8 @@ public: const unsigned char* GetCodecPrivate(size_t&) const; bool GetLacing() const; unsigned long long GetDefaultDuration() const; + unsigned long long GetCodecDelay() const; + unsigned long long GetSeekPreRoll() const; const BlockEntry* GetEOS() const; @@ -371,14 +377,12 @@ public: ~Info(); int Copy(Info&) const; void Clear(); - private: - Info(const Info&); - Info& operator=(const Info&); - public: long type; long number; unsigned long long uid; unsigned long long defaultDuration; + unsigned long long codecDelay; + unsigned long long seekPreRoll; char* nameAsUTF8; char* language; char* codecId; @@ -387,7 +391,10 @@ public: size_t codecPrivateSize; bool lacing; Settings settings; + private: + Info(const Info&); + Info& operator=(const Info&); int CopyStr(char* Info::*str, Info&) const; }; @@ -944,8 +951,10 @@ private: long ParseSimpleBlock(long long, long long&, long&); long ParseBlockGroup(long long, long long&, long&); - long CreateBlock(long long id, long long pos, long long size); - long CreateBlockGroup(long long, long long); + long CreateBlock(long long id, long long pos, long long size, + long long discard_padding); + long CreateBlockGroup(long long start_offset, long long size, + long long discard_padding); long CreateSimpleBlock(long long, long long); }; diff --git a/sample.cpp b/sample.cpp index 710fb60..b68764e 100644 --- a/sample.cpp +++ b/sample.cpp @@ -225,7 +225,7 @@ int main(int argc, char* argv[]) const AudioTrack* const pAudioTrack = static_cast(pTrack); - const long long channels = pAudioTrack->GetChannels(); + const long long channels = pAudioTrack->GetChannels(); printf("\t\tAudio Channels\t\t: %lld\n", channels); const long long bitDepth = pAudioTrack->GetBitDepth(); @@ -233,6 +233,12 @@ int main(int argc, char* argv[]) const double sampleRate = pAudioTrack->GetSamplingRate(); printf("\t\tAddio Sample Rate\t: %.3f\n", sampleRate); + + const long long codecDelay = pAudioTrack->GetCodecDelay(); + printf("\t\tAudio Codec Delay\t\t: %lld\n", codecDelay); + + const long long seekPreRoll = pAudioTrack->GetSeekPreRoll(); + printf("\t\tAudio Seek Pre Roll\t\t: %lld\n", seekPreRoll); } } @@ -282,11 +288,12 @@ int main(int argc, char* argv[]) const long long trackType = pTrack->GetType(); const int frameCount = pBlock->GetFrameCount(); const long long time_ns = pBlock->GetTime(pCluster); + const long long discard_padding = pBlock->GetDiscardPadding(); - printf("\t\t\tBlock\t\t:%s,%s,%15lld\n", + printf("\t\t\tBlock\t\t:%s,%s,%15lld,%lld\n", (trackType == mkvparser::Track::kVideo) ? "V" : "A", pBlock->IsKey() ? "I" : "P", - time_ns); + time_ns, discard_padding); for (int i = 0; i < frameCount; ++i) {