Chromium Code Reviews| Index: media/mpeg2/mpeg2ts_psi.cc |
| diff --git a/media/mpeg2/mpeg2ts_psi.cc b/media/mpeg2/mpeg2ts_psi.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..95800c59e8b837034355bd572491a26d5e5f7b31 |
| --- /dev/null |
| +++ b/media/mpeg2/mpeg2ts_psi.cc |
| @@ -0,0 +1,115 @@ |
| +// Copyright (c) 2013 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/mpeg2/mpeg2ts_psi.h" |
| + |
| +#include "base/logging.h" |
| +#include "media/base/bit_reader.h" |
| +#include "media/mpeg2/mpeg2ts_crc.h" |
| + |
| +namespace media { |
| +namespace mpeg2ts { |
| + |
| +Mpeg2TsPsiParser::Mpeg2TsPsiParser() |
| + : wait_for_pusi_(true), |
| + leading_bytes_to_discard_(0) { |
| +} |
| + |
| +Mpeg2TsPsiParser::~Mpeg2TsPsiParser() { |
| +} |
| + |
| +bool Mpeg2TsPsiParser::Parse(bool payload_unit_start_indicator, |
| + const uint8* buf, int size) { |
| + return ParseInternal(payload_unit_start_indicator, buf, size); |
| +} |
| + |
| +bool Mpeg2TsPsiParser::ParseInternal(bool payload_unit_start_indicator, |
|
damienv1
2013/09/10 04:10:02
Could remove ParseInternal function (since Parse i
damienv1
2013/09/10 21:03:48
Done.
|
| + const uint8* buf, int size) { |
| + // Ignore partial PSI. |
| + if (wait_for_pusi_ && !payload_unit_start_indicator) |
| + return true; |
| + |
| + if (payload_unit_start_indicator) { |
| + // Reset the state of the PSI section. |
| + ResetPsiState(); |
| + |
| + // Update the state. |
| + wait_for_pusi_ = false; |
| + DCHECK_GE(size, 1); |
| + int pointer_field = buf[0]; |
| + leading_bytes_to_discard_ = pointer_field; |
| + buf++; |
| + size--; |
| + } |
| + |
| + // Discard some leading bytes if needed. |
| + if (leading_bytes_to_discard_ > 0) { |
| + int nbytes_to_discard = std::min(leading_bytes_to_discard_, size); |
| + buf += nbytes_to_discard; |
| + size -= nbytes_to_discard; |
| + leading_bytes_to_discard_ -= nbytes_to_discard; |
| + } |
| + if (size == 0) |
| + return true; |
| + |
| + // Add the data to the parser state. |
| + psi_byte_queue_.Push(buf, size); |
| + int raw_psi_size; |
| + const uint8* raw_psi; |
| + psi_byte_queue_.Peek(&raw_psi, &raw_psi_size); |
| + |
| + // Check whether we have enough data to start parsing. |
| + if (raw_psi_size < 3) |
| + return true; |
| + int section_length = |
| + ((static_cast<int>(raw_psi[1]) << 8) | |
| + (static_cast<int>(raw_psi[2]))) & 0xfff; |
| + if (section_length >= 1021) |
| + return false; |
| + int psi_length = section_length + 3; |
| + if (raw_psi_size < psi_length) { |
| + // Don't throw an error when there is not enough data, |
| + // just wait for more data to come. |
| + return true; |
| + } |
| + |
| + // There should not be any trailing bytes after a PMT. |
| + // Instead, the pointer field should be used to stuff bytes. |
| + DVLOG_IF(1, raw_psi_size > psi_length) |
| + << "Trailing bytes after a PSI section: " |
| + << psi_length << " vs " << raw_psi_size; |
| + |
| + // Verify the CRC. |
| + Mpeg2TsCrc crc; |
| + for (int k = 0; k < psi_length; k++) |
| + crc.Update(raw_psi[k]); |
| + if (!crc.IsValid()) |
| + return false; |
| + |
| + // Parse the PSI section. |
| + BitReader bit_reader(raw_psi, raw_psi_size); |
| + bool status = ParsePsiSection(&bit_reader); |
| + if (status) |
| + ResetPsiState(); |
| + |
| + return status; |
| +} |
| + |
| +void Mpeg2TsPsiParser::Flush() { |
| +} |
| + |
| +void Mpeg2TsPsiParser::Reset() { |
| + ResetPsiSection(); |
| + ResetPsiState(); |
| +} |
| + |
| +void Mpeg2TsPsiParser::ResetPsiState() { |
| + wait_for_pusi_ = true; |
| + psi_byte_queue_.Reset(); |
| + leading_bytes_to_discard_ = 0; |
| +} |
| + |
| +} // namespace mpeg2ts |
| +} // namespace media |
| + |