Enable right clicking on the applist doodle web contents and log the data.
[chromium-blink-merge.git] / media / formats / mp2t / mp2t_stream_parser_unittest.cc
blob866c25ec42186f5aa5d674671a4434d7fe3f6cdc
1 // Copyright 2014 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include <algorithm>
6 #include <string>
8 #include "base/bind.h"
9 #include "base/bind_helpers.h"
10 #include "base/logging.h"
11 #include "base/memory/ref_counted.h"
12 #include "base/time/time.h"
13 #include "media/base/audio_decoder_config.h"
14 #include "media/base/decoder_buffer.h"
15 #include "media/base/stream_parser_buffer.h"
16 #include "media/base/test_data_util.h"
17 #include "media/base/text_track_config.h"
18 #include "media/base/video_decoder_config.h"
19 #include "media/formats/mp2t/mp2t_stream_parser.h"
20 #include "testing/gtest/include/gtest/gtest.h"
22 namespace media {
23 namespace mp2t {
25 namespace {
27 bool IsMonotonic(const StreamParser::BufferQueue& buffers) {
28 if (buffers.empty())
29 return true;
31 StreamParser::BufferQueue::const_iterator it1 = buffers.begin();
32 StreamParser::BufferQueue::const_iterator it2 = ++it1;
33 for ( ; it2 != buffers.end(); ++it1, ++it2) {
34 if ((*it2)->GetDecodeTimestamp() < (*it1)->GetDecodeTimestamp())
35 return false;
37 return true;
40 bool IsAlmostEqual(DecodeTimestamp t0, DecodeTimestamp t1) {
41 base::TimeDelta kMaxDeviation = base::TimeDelta::FromMilliseconds(5);
42 base::TimeDelta diff = t1 - t0;
43 return (diff >= -kMaxDeviation && diff <= kMaxDeviation);
46 } // namespace
48 class Mp2tStreamParserTest : public testing::Test {
49 public:
50 Mp2tStreamParserTest()
51 : segment_count_(0),
52 config_count_(0),
53 audio_frame_count_(0),
54 video_frame_count_(0),
55 has_video_(true),
56 audio_min_dts_(kNoDecodeTimestamp()),
57 audio_max_dts_(kNoDecodeTimestamp()),
58 video_min_dts_(kNoDecodeTimestamp()),
59 video_max_dts_(kNoDecodeTimestamp()) {
60 bool has_sbr = false;
61 parser_.reset(new Mp2tStreamParser(has_sbr));
64 protected:
65 scoped_ptr<Mp2tStreamParser> parser_;
66 int segment_count_;
67 int config_count_;
68 int audio_frame_count_;
69 int video_frame_count_;
70 bool has_video_;
71 DecodeTimestamp audio_min_dts_;
72 DecodeTimestamp audio_max_dts_;
73 DecodeTimestamp video_min_dts_;
74 DecodeTimestamp video_max_dts_;
76 void ResetStats() {
77 segment_count_ = 0;
78 config_count_ = 0;
79 audio_frame_count_ = 0;
80 video_frame_count_ = 0;
81 audio_min_dts_ = kNoDecodeTimestamp();
82 audio_max_dts_ = kNoDecodeTimestamp();
83 video_min_dts_ = kNoDecodeTimestamp();
84 video_max_dts_ = kNoDecodeTimestamp();
87 bool AppendData(const uint8* data, size_t length) {
88 return parser_->Parse(data, length);
91 bool AppendDataInPieces(const uint8* data, size_t length, size_t piece_size) {
92 const uint8* start = data;
93 const uint8* end = data + length;
94 while (start < end) {
95 size_t append_size = std::min(piece_size,
96 static_cast<size_t>(end - start));
97 if (!AppendData(start, append_size))
98 return false;
99 start += append_size;
101 return true;
104 void OnInit(const StreamParser::InitParameters& params) {
105 DVLOG(1) << "OnInit: dur=" << params.duration.InMilliseconds()
106 << ", autoTimestampOffset=" << params.auto_update_timestamp_offset;
109 bool OnNewConfig(const AudioDecoderConfig& ac,
110 const VideoDecoderConfig& vc,
111 const StreamParser::TextTrackConfigMap& tc) {
112 DVLOG(1) << "OnNewConfig: audio=" << ac.IsValidConfig()
113 << ", video=" << vc.IsValidConfig();
114 config_count_++;
115 EXPECT_TRUE(ac.IsValidConfig());
116 EXPECT_EQ(vc.IsValidConfig(), has_video_);
117 return true;
121 void DumpBuffers(const std::string& label,
122 const StreamParser::BufferQueue& buffers) {
123 DVLOG(2) << "DumpBuffers: " << label << " size " << buffers.size();
124 for (StreamParser::BufferQueue::const_iterator buf = buffers.begin();
125 buf != buffers.end(); buf++) {
126 DVLOG(3) << " n=" << buf - buffers.begin()
127 << ", size=" << (*buf)->data_size()
128 << ", dur=" << (*buf)->duration().InMilliseconds();
132 bool OnNewBuffers(const StreamParser::BufferQueue& audio_buffers,
133 const StreamParser::BufferQueue& video_buffers,
134 const StreamParser::TextBufferQueueMap& text_map) {
135 EXPECT_GT(config_count_, 0);
136 DumpBuffers("audio_buffers", audio_buffers);
137 DumpBuffers("video_buffers", video_buffers);
139 // TODO(wolenetz/acolwell): Add text track support to more MSE parsers. See
140 // http://crbug.com/336926.
141 if (!text_map.empty())
142 return false;
144 // Verify monotonicity.
145 if (!IsMonotonic(video_buffers))
146 return false;
147 if (!IsMonotonic(audio_buffers))
148 return false;
150 if (!video_buffers.empty()) {
151 DecodeTimestamp first_dts = video_buffers.front()->GetDecodeTimestamp();
152 DecodeTimestamp last_dts = video_buffers.back()->GetDecodeTimestamp();
153 if (video_max_dts_ != kNoDecodeTimestamp() && first_dts < video_max_dts_)
154 return false;
155 if (video_min_dts_ == kNoDecodeTimestamp())
156 video_min_dts_ = first_dts;
157 video_max_dts_ = last_dts;
159 if (!audio_buffers.empty()) {
160 DecodeTimestamp first_dts = audio_buffers.front()->GetDecodeTimestamp();
161 DecodeTimestamp last_dts = audio_buffers.back()->GetDecodeTimestamp();
162 if (audio_max_dts_ != kNoDecodeTimestamp() && first_dts < audio_max_dts_)
163 return false;
164 if (audio_min_dts_ == kNoDecodeTimestamp())
165 audio_min_dts_ = first_dts;
166 audio_max_dts_ = last_dts;
169 audio_frame_count_ += audio_buffers.size();
170 video_frame_count_ += video_buffers.size();
171 return true;
174 void OnKeyNeeded(const std::string& type,
175 const std::vector<uint8>& init_data) {
176 NOTREACHED() << "OnKeyNeeded not expected in the Mpeg2 TS parser";
179 void OnNewSegment() {
180 DVLOG(1) << "OnNewSegment";
181 segment_count_++;
184 void OnEndOfSegment() {
185 NOTREACHED() << "OnEndOfSegment not expected in the Mpeg2 TS parser";
188 void InitializeParser() {
189 parser_->Init(
190 base::Bind(&Mp2tStreamParserTest::OnInit,
191 base::Unretained(this)),
192 base::Bind(&Mp2tStreamParserTest::OnNewConfig,
193 base::Unretained(this)),
194 base::Bind(&Mp2tStreamParserTest::OnNewBuffers,
195 base::Unretained(this)),
196 true,
197 base::Bind(&Mp2tStreamParserTest::OnKeyNeeded,
198 base::Unretained(this)),
199 base::Bind(&Mp2tStreamParserTest::OnNewSegment,
200 base::Unretained(this)),
201 base::Bind(&Mp2tStreamParserTest::OnEndOfSegment,
202 base::Unretained(this)),
203 LogCB());
206 bool ParseMpeg2TsFile(const std::string& filename, int append_bytes) {
207 scoped_refptr<DecoderBuffer> buffer = ReadTestDataFile(filename);
208 EXPECT_TRUE(AppendDataInPieces(buffer->data(),
209 buffer->data_size(),
210 append_bytes));
211 return true;
215 TEST_F(Mp2tStreamParserTest, UnalignedAppend17) {
216 // Test small, non-segment-aligned appends.
217 InitializeParser();
218 ParseMpeg2TsFile("bear-1280x720.ts", 17);
219 parser_->Flush();
220 EXPECT_EQ(video_frame_count_, 82);
221 // This stream has no mid-stream configuration change.
222 EXPECT_EQ(config_count_, 1);
223 EXPECT_EQ(segment_count_, 1);
226 TEST_F(Mp2tStreamParserTest, UnalignedAppend512) {
227 // Test small, non-segment-aligned appends.
228 InitializeParser();
229 ParseMpeg2TsFile("bear-1280x720.ts", 512);
230 parser_->Flush();
231 EXPECT_EQ(video_frame_count_, 82);
232 // This stream has no mid-stream configuration change.
233 EXPECT_EQ(config_count_, 1);
234 EXPECT_EQ(segment_count_, 1);
237 TEST_F(Mp2tStreamParserTest, AppendAfterFlush512) {
238 InitializeParser();
239 ParseMpeg2TsFile("bear-1280x720.ts", 512);
240 parser_->Flush();
241 EXPECT_EQ(video_frame_count_, 82);
242 EXPECT_EQ(config_count_, 1);
243 EXPECT_EQ(segment_count_, 1);
245 ResetStats();
246 ParseMpeg2TsFile("bear-1280x720.ts", 512);
247 parser_->Flush();
248 EXPECT_EQ(video_frame_count_, 82);
249 EXPECT_EQ(config_count_, 1);
250 EXPECT_EQ(segment_count_, 1);
253 TEST_F(Mp2tStreamParserTest, TimestampWrapAround) {
254 // "bear-1280x720_ptswraparound.ts" has been transcoded
255 // from bear-1280x720.mp4 by applying a time offset of 95442s
256 // (close to 2^33 / 90000) which results in timestamps wrap around
257 // in the Mpeg2 TS stream.
258 InitializeParser();
259 ParseMpeg2TsFile("bear-1280x720_ptswraparound.ts", 512);
260 parser_->Flush();
261 EXPECT_EQ(video_frame_count_, 82);
263 EXPECT_TRUE(IsAlmostEqual(video_min_dts_,
264 DecodeTimestamp::FromSecondsD(95443.376)));
265 EXPECT_TRUE(IsAlmostEqual(video_max_dts_,
266 DecodeTimestamp::FromSecondsD(95446.079)));
268 // Note: for audio, AdtsStreamParser considers only the PTS (which is then
269 // used as the DTS).
270 // TODO(damienv): most of the time, audio streams just have PTS. Here, only
271 // the first PES packet has a DTS, all the other PES packets have PTS only.
272 // Reconsider the expected value for |audio_min_dts_| if DTS are used as part
273 // of the ADTS stream parser.
275 // Note: the last pts for audio is 95445.931 but this PES packet includes
276 // 9 ADTS frames with 1 AAC frame in each ADTS frame.
277 // So the PTS of the last AAC frame is:
278 // 95445.931 + 8 * (1024 / 44100) = 95446.117
279 EXPECT_TRUE(IsAlmostEqual(audio_min_dts_,
280 DecodeTimestamp::FromSecondsD(95443.400)));
281 EXPECT_TRUE(IsAlmostEqual(audio_max_dts_,
282 DecodeTimestamp::FromSecondsD(95446.117)));
285 TEST_F(Mp2tStreamParserTest, AudioInPrivateStream1) {
286 // Test small, non-segment-aligned appends.
287 InitializeParser();
288 has_video_ = false;
289 ParseMpeg2TsFile("bear_adts_in_private_stream_1.ts", 512);
290 parser_->Flush();
291 EXPECT_EQ(audio_frame_count_, 40);
292 EXPECT_EQ(video_frame_count_, 0);
293 // This stream has no mid-stream configuration change.
294 EXPECT_EQ(config_count_, 1);
295 EXPECT_EQ(segment_count_, 1);
298 } // namespace mp2t
299 } // namespace media