Index: media/mp4/es_descriptor.cc |
diff --git a/media/mp4/es_descriptor.cc b/media/mp4/es_descriptor.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..1cb4c65802a5deed339caf0fd6d08435ab1e8447 |
--- /dev/null |
+++ b/media/mp4/es_descriptor.cc |
@@ -0,0 +1,112 @@ |
+// Copyright (c) 2012 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "media/mp4/es_descriptor.h" |
+ |
+#include "media/base/bit_reader.h" |
+#include "media/mp4/rcheck.h" |
+ |
+// The elementary stream size is specific by up to 4 bytes. |
+// The MSB of a byte indicates if there are more bytes for the size. |
+static bool ReadESSize(media::BitReader* reader, uint32* size) { |
+ uint8 msb; |
+ uint8 byte; |
+ |
+ *size = 0; |
+ |
+ for (size_t i = 0; i < 4; ++i) { |
+ RCHECK(reader->ReadBits(1, &msb)); |
+ RCHECK(reader->ReadBits(7, &byte)); |
+ *size = (*size << 7) + byte; |
+ |
+ if (msb == 0) |
+ break; |
+ } |
+ |
+ return true; |
+} |
+ |
+namespace media { |
+ |
+namespace mp4 { |
+ |
+ESDescriptor::ESDescriptor() |
+ : object_type_(kForbidden) { |
+} |
+ |
+ESDescriptor::~ESDescriptor() {} |
+ |
+bool ESDescriptor::Parse(const std::vector<uint8>& data) { |
+ BitReader reader(&data[0], data.size()); |
+ uint8 tag; |
+ uint32 size; |
+ uint8 stream_dependency_flag; |
+ uint8 url_flag; |
+ uint8 ocr_stream_flag; |
+ |
+ RCHECK(reader.ReadBits(8, &tag)); |
+ RCHECK(tag == kESDescrTag); |
+ RCHECK(ReadESSize(&reader, &size)); |
+ RCHECK(static_cast<off_t>(size * CHAR_BIT) <= reader.NumBitsLeft()); |
+ |
+ RCHECK(reader.SkipBits(16)); // ES_ID |
+ RCHECK(reader.ReadBits(1, &stream_dependency_flag)); |
+ RCHECK(reader.ReadBits(1, &url_flag)); |
+ RCHECK(reader.ReadBits(1, &ocr_stream_flag)); |
+ RCHECK(reader.SkipBits(5)); // streamPriority |
+ |
+ if (stream_dependency_flag) |
+ reader.SkipBits(16); // dependsOn_ES_ID; |
+ RCHECK(!url_flag); // We don't support url flag |
+ if (ocr_stream_flag) |
+ reader.SkipBits(16); // OCR_ES_Id |
+ |
+ RCHECK(ParseDecoderConfigDescriptor(&reader)); |
+ |
+ return true; |
+} |
+ |
+uint8 ESDescriptor::object_type() const { |
+ return object_type_; |
+} |
+ |
+const std::vector<uint8>& ESDescriptor::decoder_specific_info() const { |
+ return decoder_specific_info_; |
+} |
+ |
+bool ESDescriptor::ParseDecoderConfigDescriptor(BitReader* reader) { |
+ uint8 tag; |
+ uint32 size; |
+ |
+ RCHECK(reader->ReadBits(8, &tag)); |
+ RCHECK(tag == kDecoderConfigDescrTag); |
+ RCHECK(ReadESSize(reader, &size)); |
+ RCHECK(static_cast<off_t>(size * CHAR_BIT) <= reader->NumBitsLeft()); |
+ |
+ RCHECK(reader->ReadBits(8, &object_type_)); |
+ RCHECK(reader->SkipBits(96)); |
+ RCHECK(ParseDecoderSpecificInfo(reader)); |
+ |
+ return true; |
+} |
+ |
+bool ESDescriptor::ParseDecoderSpecificInfo(BitReader* reader) { |
+ uint8 tag; |
+ uint32 size; |
+ |
+ RCHECK(reader->ReadBits(8, &tag)); |
+ RCHECK(tag == kDecoderSpecificInfoTag); |
+ RCHECK(ReadESSize(reader, &size)); |
+ RCHECK(static_cast<off_t>(size * CHAR_BIT) <= reader->NumBitsLeft()); |
+ |
+ decoder_specific_info_.resize(size); |
+ for (uint32 i = 0; i < size; ++i) |
+ RCHECK(reader->ReadBits(8, &decoder_specific_info_[i])); |
+ |
+ return true; |
+} |
+ |
+} // namespace mp4 |
+ |
+} // namespace media |