OLD | NEW |
1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "media/filters/chunk_demuxer.h" | 5 #include "media/filters/chunk_demuxer.h" |
6 | 6 |
7 #include <stddef.h> | 7 #include <stddef.h> |
8 #include <stdint.h> | 8 #include <stdint.h> |
9 #include <algorithm> | 9 #include <algorithm> |
10 #include <utility> | 10 #include <utility> |
(...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
118 } | 118 } |
119 } | 119 } |
120 | 120 |
121 MATCHER_P(HasTimestamp, timestamp_in_ms, "") { | 121 MATCHER_P(HasTimestamp, timestamp_in_ms, "") { |
122 return arg.get() && !arg->end_of_stream() && | 122 return arg.get() && !arg->end_of_stream() && |
123 arg->timestamp().InMilliseconds() == timestamp_in_ms; | 123 arg->timestamp().InMilliseconds() == timestamp_in_ms; |
124 } | 124 } |
125 | 125 |
126 MATCHER(IsEndOfStream, "") { return arg.get() && arg->end_of_stream(); } | 126 MATCHER(IsEndOfStream, "") { return arg.get() && arg->end_of_stream(); } |
127 | 127 |
128 MATCHER_P(SegmentMissingFrames, frame_types_string, "") { | 128 MATCHER_P(SegmentMissingFrames, track_id, "") { |
129 return CONTAINS_STRING(arg, "Media segment did not contain any " + | 129 return CONTAINS_STRING( |
130 std::string(frame_types_string) + | 130 arg, "Media segment did not contain any coded frames for track " + |
131 " coded frames, mismatching initialization " | 131 std::string(track_id)); |
132 "segment. Therefore, MSE coded frame " | |
133 "processing may not interoperably detect " | |
134 "discontinuities in appended media."); | |
135 } | 132 } |
136 | 133 |
137 MATCHER(StreamParsingFailed, "") { | 134 MATCHER(StreamParsingFailed, "") { |
138 return CONTAINS_STRING(arg, "Append: stream parsing failed."); | 135 return CONTAINS_STRING(arg, "Append: stream parsing failed."); |
139 } | 136 } |
140 | 137 |
141 MATCHER_P(FoundStream, stream_type_string, "") { | 138 MATCHER_P(FoundStream, stream_type_string, "") { |
142 return CONTAINS_STRING( | 139 return CONTAINS_STRING( |
143 arg, "found_" + std::string(stream_type_string) + "_stream") && | 140 arg, "found_" + std::string(stream_type_string) + "_stream") && |
144 CONTAINS_STRING(arg, "true"); | 141 CONTAINS_STRING(arg, "true"); |
145 } | 142 } |
146 | 143 |
147 MATCHER_P2(CodecName, stream_type_string, codec_string, "") { | 144 MATCHER_P2(CodecName, stream_type_string, codec_string, "") { |
148 return CONTAINS_STRING(arg, | 145 return CONTAINS_STRING(arg, |
149 std::string(stream_type_string) + "_codec_name") && | 146 std::string(stream_type_string) + "_codec_name") && |
150 CONTAINS_STRING(arg, std::string(codec_string)); | 147 CONTAINS_STRING(arg, std::string(codec_string)); |
151 } | 148 } |
152 | 149 |
153 MATCHER_P2(InitSegmentMismatchesMimeType, | 150 MATCHER_P2(InitSegmentMismatchesMimeType, stream_type, codec_name, "") { |
154 track_type_string_with_article, | 151 return CONTAINS_STRING(arg, std::string(stream_type) + " stream codec " + |
155 mime_missing_track_type_bool, | 152 std::string(codec_name) + |
156 "") { | 153 " doesn't match SourceBuffer codecs."); |
157 return CONTAINS_STRING( | 154 } |
158 arg, "Initialization segment " + | 155 |
159 std::string(mime_missing_track_type_bool ? "has " | 156 MATCHER_P(InitSegmentMissesExpectedTrack, missing_codec, "") { |
160 : "does not have ") + | 157 return CONTAINS_STRING(arg, "Initialization segment misses expected " + |
161 std::string(track_type_string_with_article) + | 158 std::string(missing_codec) + " track."); |
162 " track, but the mimetype " + | |
163 std::string(mime_missing_track_type_bool ? "does not specify " | |
164 : "specifies ") + | |
165 std::string(track_type_string_with_article) + " codec."); | |
166 } | 159 } |
167 | 160 |
168 MATCHER_P2(GeneratedSplice, duration_microseconds, time_microseconds, "") { | 161 MATCHER_P2(GeneratedSplice, duration_microseconds, time_microseconds, "") { |
169 return CONTAINS_STRING(arg, "Generated splice of overlap duration " + | 162 return CONTAINS_STRING(arg, "Generated splice of overlap duration " + |
170 base::IntToString(duration_microseconds) + | 163 base::IntToString(duration_microseconds) + |
171 "us into new buffer at " + | 164 "us into new buffer at " + |
172 base::IntToString(time_microseconds) + "us."); | 165 base::IntToString(time_microseconds) + "us."); |
173 } | 166 } |
174 | 167 |
175 MATCHER_P2(SkippingSpliceAtOrBefore, | 168 MATCHER_P2(SkippingSpliceAtOrBefore, |
(...skipping 205 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
381 | 374 |
382 memcpy(buf, info->data(), info->data_size()); | 375 memcpy(buf, info->data(), info->data_size()); |
383 buf += info->data_size(); | 376 buf += info->data_size(); |
384 | 377 |
385 memcpy(buf, kTracksHeader, kTracksHeaderSize); | 378 memcpy(buf, kTracksHeader, kTracksHeaderSize); |
386 WriteInt64(buf + kTracksSizeOffset, tracks_element_size); | 379 WriteInt64(buf + kTracksSizeOffset, tracks_element_size); |
387 buf += kTracksHeaderSize; | 380 buf += kTracksHeaderSize; |
388 | 381 |
389 // TODO(xhwang): Simplify this! Probably have test data files that contain | 382 // TODO(xhwang): Simplify this! Probably have test data files that contain |
390 // ContentEncodings directly instead of trying to create one at run-time. | 383 // ContentEncodings directly instead of trying to create one at run-time. |
391 if (has_audio) { | |
392 memcpy(buf, audio_track_entry->data(), | |
393 audio_track_entry->data_size()); | |
394 if (is_audio_encrypted) { | |
395 memcpy(buf + audio_track_entry->data_size(), | |
396 audio_content_encodings->data(), | |
397 audio_content_encodings->data_size()); | |
398 WriteInt64(buf + kAudioTrackSizeOffset, | |
399 audio_track_entry->data_size() + | |
400 audio_content_encodings->data_size() - | |
401 kAudioTrackEntryHeaderSize); | |
402 buf += audio_content_encodings->data_size(); | |
403 } | |
404 buf += audio_track_entry->data_size(); | |
405 } | |
406 | |
407 if (has_video) { | 384 if (has_video) { |
408 memcpy(buf, video_track_entry->data(), | 385 memcpy(buf, video_track_entry->data(), |
409 video_track_entry->data_size()); | 386 video_track_entry->data_size()); |
410 if (is_video_encrypted) { | 387 if (is_video_encrypted) { |
411 memcpy(buf + video_track_entry->data_size(), | 388 memcpy(buf + video_track_entry->data_size(), |
412 video_content_encodings->data(), | 389 video_content_encodings->data(), |
413 video_content_encodings->data_size()); | 390 video_content_encodings->data_size()); |
414 WriteInt64(buf + kVideoTrackSizeOffset, | 391 WriteInt64(buf + kVideoTrackSizeOffset, |
415 video_track_entry->data_size() + | 392 video_track_entry->data_size() + |
416 video_content_encodings->data_size() - | 393 video_content_encodings->data_size() - |
417 kVideoTrackEntryHeaderSize); | 394 kVideoTrackEntryHeaderSize); |
418 buf += video_content_encodings->data_size(); | 395 buf += video_content_encodings->data_size(); |
419 } | 396 } |
420 buf += video_track_entry->data_size(); | 397 buf += video_track_entry->data_size(); |
421 } | 398 } |
422 | 399 |
| 400 if (has_audio) { |
| 401 memcpy(buf, audio_track_entry->data(), audio_track_entry->data_size()); |
| 402 if (is_audio_encrypted) { |
| 403 memcpy(buf + audio_track_entry->data_size(), |
| 404 audio_content_encodings->data(), |
| 405 audio_content_encodings->data_size()); |
| 406 WriteInt64(buf + kAudioTrackSizeOffset, |
| 407 audio_track_entry->data_size() + |
| 408 audio_content_encodings->data_size() - |
| 409 kAudioTrackEntryHeaderSize); |
| 410 buf += audio_content_encodings->data_size(); |
| 411 } |
| 412 buf += audio_track_entry->data_size(); |
| 413 } |
| 414 |
423 if (has_text) { | 415 if (has_text) { |
424 memcpy(buf, text_track_entry->data(), | 416 memcpy(buf, text_track_entry->data(), |
425 text_track_entry->data_size()); | 417 text_track_entry->data_size()); |
426 buf += text_track_entry->data_size(); | 418 buf += text_track_entry->data_size(); |
427 } | 419 } |
428 } | 420 } |
429 | 421 |
430 ChunkDemuxer::Status AddId() { | 422 ChunkDemuxer::Status AddId() { |
431 return AddId(kSourceId, HAS_AUDIO | HAS_VIDEO); | 423 return AddId(kSourceId, HAS_AUDIO | HAS_VIDEO); |
432 } | 424 } |
433 | 425 |
434 ChunkDemuxer::Status AddId(const std::string& source_id, int stream_flags) { | 426 ChunkDemuxer::Status AddId(const std::string& source_id, int stream_flags) { |
435 bool has_audio = (stream_flags & HAS_AUDIO) != 0; | 427 bool has_audio = (stream_flags & HAS_AUDIO) != 0; |
436 bool has_video = (stream_flags & HAS_VIDEO) != 0; | 428 bool has_video = (stream_flags & HAS_VIDEO) != 0; |
437 std::vector<std::string> codecs; | 429 std::string codecs; |
438 std::string type; | 430 std::string type; |
439 | 431 |
440 if (has_audio) { | 432 if (has_audio) { |
441 codecs.push_back("vorbis"); | 433 codecs += "vorbis"; |
442 type = "audio/webm"; | 434 type = "audio/webm"; |
443 } | 435 } |
444 | 436 |
445 if (has_video) { | 437 if (has_video) { |
446 codecs.push_back("vp8"); | 438 if (codecs == "") |
| 439 codecs = "vp8"; |
| 440 else |
| 441 codecs += ",vp8"; |
447 type = "video/webm"; | 442 type = "video/webm"; |
448 } | 443 } |
449 | 444 |
450 if (!has_audio && !has_video) { | 445 if (!has_audio && !has_video) { |
451 return AddId(kSourceId, HAS_AUDIO | HAS_VIDEO); | 446 return AddId(kSourceId, HAS_AUDIO | HAS_VIDEO); |
452 } | 447 } |
453 | 448 |
454 ChunkDemuxer::Status status = demuxer_->AddId(source_id, type, codecs); | 449 return AddId(source_id, type, codecs); |
| 450 } |
| 451 |
| 452 ChunkDemuxer::Status AddId(const std::string& source_id, |
| 453 const std::string& mime_type, |
| 454 const std::string& codecs) { |
| 455 ChunkDemuxer::Status status = demuxer_->AddId(source_id, mime_type, codecs); |
455 if (status == ChunkDemuxer::kOk) | 456 if (status == ChunkDemuxer::kOk) |
456 demuxer_->SetTracksWatcher( | 457 demuxer_->SetTracksWatcher( |
457 source_id, base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | 458 source_id, base::Bind(&ChunkDemuxerTest::InitSegmentReceived, |
458 base::Unretained(this))); | 459 base::Unretained(this))); |
459 return status; | 460 return status; |
460 } | 461 } |
461 | 462 |
462 #if BUILDFLAG(ENABLE_MSE_MPEG2TS_STREAM_PARSER) | |
463 ChunkDemuxer::Status AddIdForMp2tSource(const std::string& source_id) { | |
464 std::vector<std::string> codecs; | |
465 std::string type = "video/mp2t"; | |
466 codecs.push_back("mp4a.40.2"); | |
467 codecs.push_back("avc1.640028"); | |
468 ChunkDemuxer::Status status = demuxer_->AddId(source_id, type, codecs); | |
469 if (status == ChunkDemuxer::kOk) | |
470 demuxer_->SetTracksWatcher( | |
471 source_id, base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | |
472 base::Unretained(this))); | |
473 return status; | |
474 } | |
475 #endif | |
476 | |
477 bool AppendData(const uint8_t* data, size_t length) { | 463 bool AppendData(const uint8_t* data, size_t length) { |
478 return AppendData(kSourceId, data, length); | 464 return AppendData(kSourceId, data, length); |
479 } | 465 } |
480 | 466 |
481 bool AppendCluster(const std::string& source_id, | 467 bool AppendCluster(const std::string& source_id, |
482 std::unique_ptr<Cluster> cluster) { | 468 std::unique_ptr<Cluster> cluster) { |
483 return AppendData(source_id, cluster->data(), cluster->size()); | 469 return AppendData(source_id, cluster->data(), cluster->size()); |
484 } | 470 } |
485 | 471 |
486 bool AppendCluster(std::unique_ptr<Cluster> cluster) { | 472 bool AppendCluster(std::unique_ptr<Cluster> cluster) { |
(...skipping 324 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
811 USE_ALTERNATE_AUDIO_TRACK_ID = 1 << 3, | 797 USE_ALTERNATE_AUDIO_TRACK_ID = 1 << 3, |
812 USE_ALTERNATE_VIDEO_TRACK_ID = 1 << 4, | 798 USE_ALTERNATE_VIDEO_TRACK_ID = 1 << 4, |
813 USE_ALTERNATE_TEXT_TRACK_ID = 1 << 5, | 799 USE_ALTERNATE_TEXT_TRACK_ID = 1 << 5, |
814 }; | 800 }; |
815 | 801 |
816 bool InitDemuxer(int stream_flags) { | 802 bool InitDemuxer(int stream_flags) { |
817 return InitDemuxerWithEncryptionInfo(stream_flags, false, false); | 803 return InitDemuxerWithEncryptionInfo(stream_flags, false, false); |
818 } | 804 } |
819 | 805 |
820 void ExpectInitMediaLogs(int stream_flags) { | 806 void ExpectInitMediaLogs(int stream_flags) { |
| 807 if (stream_flags & HAS_VIDEO) { |
| 808 EXPECT_MEDIA_LOG(FoundStream("video")); |
| 809 EXPECT_MEDIA_LOG(CodecName("video", "vp8")); |
| 810 } |
821 if (stream_flags & HAS_AUDIO) { | 811 if (stream_flags & HAS_AUDIO) { |
822 EXPECT_MEDIA_LOG(FoundStream("audio")); | 812 EXPECT_MEDIA_LOG(FoundStream("audio")); |
823 EXPECT_MEDIA_LOG(CodecName("audio", "vorbis")); | 813 EXPECT_MEDIA_LOG(CodecName("audio", "vorbis")); |
824 } | 814 } |
825 | |
826 if (stream_flags & HAS_VIDEO) { | |
827 EXPECT_MEDIA_LOG(FoundStream("video")); | |
828 EXPECT_MEDIA_LOG(CodecName("video", "vp8")); | |
829 } | |
830 } | 815 } |
831 | 816 |
832 bool InitDemuxerWithEncryptionInfo( | 817 bool InitDemuxerWithEncryptionInfo( |
833 int stream_flags, bool is_audio_encrypted, bool is_video_encrypted) { | 818 int stream_flags, bool is_audio_encrypted, bool is_video_encrypted) { |
834 PipelineStatus expected_status = | 819 PipelineStatus expected_status = |
835 (stream_flags != 0) ? PIPELINE_OK : CHUNK_DEMUXER_ERROR_APPEND_FAILED; | 820 (stream_flags != 0) ? PIPELINE_OK : CHUNK_DEMUXER_ERROR_APPEND_FAILED; |
836 | 821 |
837 base::TimeDelta expected_duration = kNoTimestamp; | 822 base::TimeDelta expected_duration = kNoTimestamp; |
838 if (expected_status == PIPELINE_OK) | 823 if (expected_status == PIPELINE_OK) |
839 expected_duration = kDefaultDuration(); | 824 expected_duration = kDefaultDuration(); |
(...skipping 17 matching lines...) Expand all Loading... |
857 | 842 |
858 // Adding expectations prior to CreateInitDoneCB() here because InSequence | 843 // Adding expectations prior to CreateInitDoneCB() here because InSequence |
859 // tests require init segment received before duration set. Also, only | 844 // tests require init segment received before duration set. Also, only |
860 // expect an init segment received callback if there is actually a track in | 845 // expect an init segment received callback if there is actually a track in |
861 // it. | 846 // it. |
862 if (stream_flags != 0) { | 847 if (stream_flags != 0) { |
863 ExpectInitMediaLogs(stream_flags); | 848 ExpectInitMediaLogs(stream_flags); |
864 EXPECT_CALL(*this, InitSegmentReceivedMock(_)); | 849 EXPECT_CALL(*this, InitSegmentReceivedMock(_)); |
865 } else { | 850 } else { |
866 // OnNewConfigs() requires at least one audio, video, or text track. | 851 // OnNewConfigs() requires at least one audio, video, or text track. |
| 852 EXPECT_MEDIA_LOG(InitSegmentMissesExpectedTrack("vorbis")); |
| 853 EXPECT_MEDIA_LOG(InitSegmentMissesExpectedTrack("vp8")); |
867 EXPECT_MEDIA_LOG(StreamParsingFailed()); | 854 EXPECT_MEDIA_LOG(StreamParsingFailed()); |
868 } | 855 } |
869 | 856 |
870 demuxer_->Initialize( | 857 demuxer_->Initialize( |
871 &host_, CreateInitDoneCB(expected_duration, expected_status), true); | 858 &host_, CreateInitDoneCB(expected_duration, expected_status), true); |
872 | 859 |
873 if (AddId(kSourceId, stream_flags) != ChunkDemuxer::kOk) | 860 if (AddId(kSourceId, stream_flags) != ChunkDemuxer::kOk) |
874 return false; | 861 return false; |
875 | 862 |
876 return AppendInitSegmentWithEncryptedInfo( | 863 return AppendInitSegmentWithEncryptedInfo( |
(...skipping 1526 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2403 kSourceId, &tmp, 1, append_window_start_for_next_append_, | 2390 kSourceId, &tmp, 1, append_window_start_for_next_append_, |
2404 append_window_end_for_next_append_, ×tamp_offset_map_[kSourceId])); | 2391 append_window_end_for_next_append_, ×tamp_offset_map_[kSourceId])); |
2405 } | 2392 } |
2406 | 2393 |
2407 TEST_F(ChunkDemuxerTest, AVHeadersWithAudioOnlyType) { | 2394 TEST_F(ChunkDemuxerTest, AVHeadersWithAudioOnlyType) { |
2408 EXPECT_CALL(*this, DemuxerOpened()); | 2395 EXPECT_CALL(*this, DemuxerOpened()); |
2409 demuxer_->Initialize( | 2396 demuxer_->Initialize( |
2410 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), | 2397 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), |
2411 true); | 2398 true); |
2412 | 2399 |
2413 std::vector<std::string> codecs(1); | 2400 ASSERT_EQ(demuxer_->AddId(kSourceId, "audio/webm", "vorbis"), |
2414 codecs[0] = "vorbis"; | |
2415 ASSERT_EQ(demuxer_->AddId(kSourceId, "audio/webm", codecs), | |
2416 ChunkDemuxer::kOk); | 2401 ChunkDemuxer::kOk); |
2417 demuxer_->SetTracksWatcher(kSourceId, | 2402 demuxer_->SetTracksWatcher(kSourceId, |
2418 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | 2403 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, |
2419 base::Unretained(this))); | 2404 base::Unretained(this))); |
2420 | 2405 |
2421 // Video track is unexpected per mimetype. | 2406 // Video track is unexpected per mimetype. |
2422 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("a video", true)); | 2407 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("Video", "vp8")); |
2423 EXPECT_MEDIA_LOG(StreamParsingFailed()); | 2408 EXPECT_MEDIA_LOG(StreamParsingFailed()); |
2424 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO | HAS_VIDEO)); | 2409 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO | HAS_VIDEO)); |
2425 } | 2410 } |
2426 | 2411 |
2427 TEST_F(ChunkDemuxerTest, AVHeadersWithVideoOnlyType) { | 2412 TEST_F(ChunkDemuxerTest, AVHeadersWithVideoOnlyType) { |
2428 EXPECT_CALL(*this, DemuxerOpened()); | 2413 EXPECT_CALL(*this, DemuxerOpened()); |
2429 demuxer_->Initialize( | 2414 demuxer_->Initialize( |
2430 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), | 2415 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), |
2431 true); | 2416 true); |
2432 | 2417 |
2433 std::vector<std::string> codecs(1); | 2418 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", "vp8"), ChunkDemuxer::kOk); |
2434 codecs[0] = "vp8"; | |
2435 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", codecs), | |
2436 ChunkDemuxer::kOk); | |
2437 demuxer_->SetTracksWatcher(kSourceId, | 2419 demuxer_->SetTracksWatcher(kSourceId, |
2438 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | 2420 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, |
2439 base::Unretained(this))); | 2421 base::Unretained(this))); |
2440 | 2422 |
2441 // Audio track is unexpected per mimetype. | 2423 // Audio track is unexpected per mimetype. |
2442 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("an audio", true)); | 2424 EXPECT_MEDIA_LOG(FoundStream("video")); |
| 2425 EXPECT_MEDIA_LOG(CodecName("video", "vp8")); |
| 2426 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("Audio", "vorbis")); |
2443 EXPECT_MEDIA_LOG(StreamParsingFailed()); | 2427 EXPECT_MEDIA_LOG(StreamParsingFailed()); |
2444 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO | HAS_VIDEO)); | 2428 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO | HAS_VIDEO)); |
2445 } | 2429 } |
2446 | 2430 |
2447 TEST_F(ChunkDemuxerTest, AudioOnlyHeaderWithAVType) { | 2431 TEST_F(ChunkDemuxerTest, AudioOnlyHeaderWithAVType) { |
2448 EXPECT_CALL(*this, DemuxerOpened()); | 2432 EXPECT_CALL(*this, DemuxerOpened()); |
2449 demuxer_->Initialize( | 2433 demuxer_->Initialize( |
2450 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), | 2434 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), |
2451 true); | 2435 true); |
2452 | 2436 |
2453 std::vector<std::string> codecs(2); | 2437 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", "vorbis,vp8"), |
2454 codecs[0] = "vorbis"; | |
2455 codecs[1] = "vp8"; | |
2456 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", codecs), | |
2457 ChunkDemuxer::kOk); | 2438 ChunkDemuxer::kOk); |
2458 demuxer_->SetTracksWatcher(kSourceId, | 2439 demuxer_->SetTracksWatcher(kSourceId, |
2459 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | 2440 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, |
2460 base::Unretained(this))); | 2441 base::Unretained(this))); |
2461 | 2442 |
2462 // Video track is also expected per mimetype. | 2443 // Video track is also expected per mimetype. |
2463 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("a video", false)); | 2444 EXPECT_MEDIA_LOG(FoundStream("audio")); |
| 2445 EXPECT_MEDIA_LOG(CodecName("audio", "vorbis")); |
| 2446 EXPECT_MEDIA_LOG(InitSegmentMissesExpectedTrack("vp8")); |
2464 EXPECT_MEDIA_LOG(StreamParsingFailed()); | 2447 EXPECT_MEDIA_LOG(StreamParsingFailed()); |
2465 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO)); | 2448 ASSERT_FALSE(AppendInitSegment(HAS_AUDIO)); |
2466 } | 2449 } |
2467 | 2450 |
2468 TEST_F(ChunkDemuxerTest, VideoOnlyHeaderWithAVType) { | 2451 TEST_F(ChunkDemuxerTest, VideoOnlyHeaderWithAVType) { |
2469 EXPECT_CALL(*this, DemuxerOpened()); | 2452 EXPECT_CALL(*this, DemuxerOpened()); |
2470 demuxer_->Initialize( | 2453 demuxer_->Initialize( |
2471 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), | 2454 &host_, CreateInitDoneCB(kNoTimestamp, CHUNK_DEMUXER_ERROR_APPEND_FAILED), |
2472 true); | 2455 true); |
2473 | 2456 |
2474 std::vector<std::string> codecs(2); | 2457 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", "vorbis,vp8"), |
2475 codecs[0] = "vorbis"; | |
2476 codecs[1] = "vp8"; | |
2477 ASSERT_EQ(demuxer_->AddId(kSourceId, "video/webm", codecs), | |
2478 ChunkDemuxer::kOk); | 2458 ChunkDemuxer::kOk); |
2479 demuxer_->SetTracksWatcher(kSourceId, | 2459 demuxer_->SetTracksWatcher(kSourceId, |
2480 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, | 2460 base::Bind(&ChunkDemuxerTest::InitSegmentReceived, |
2481 base::Unretained(this))); | 2461 base::Unretained(this))); |
2482 | 2462 |
2483 // Audio track is also expected per mimetype. | 2463 // Audio track is also expected per mimetype. |
2484 EXPECT_MEDIA_LOG(InitSegmentMismatchesMimeType("an audio", false)); | 2464 EXPECT_MEDIA_LOG(FoundStream("video")); |
| 2465 EXPECT_MEDIA_LOG(CodecName("video", "vp8")); |
| 2466 EXPECT_MEDIA_LOG(InitSegmentMissesExpectedTrack("vorbis")); |
2485 EXPECT_MEDIA_LOG(StreamParsingFailed()); | 2467 EXPECT_MEDIA_LOG(StreamParsingFailed()); |
2486 ASSERT_FALSE(AppendInitSegment(HAS_VIDEO)); | 2468 ASSERT_FALSE(AppendInitSegment(HAS_VIDEO)); |
2487 } | 2469 } |
2488 | 2470 |
2489 TEST_F(ChunkDemuxerTest, MultipleHeaders) { | 2471 TEST_F(ChunkDemuxerTest, MultipleHeaders) { |
2490 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 2472 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
2491 | 2473 |
2492 ASSERT_TRUE(AppendCluster(kDefaultFirstCluster())); | 2474 ASSERT_TRUE(AppendCluster(kDefaultFirstCluster())); |
2493 | 2475 |
2494 // Append another identical initialization segment. | 2476 // Append another identical initialization segment. |
(...skipping 46 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2541 TEST_F(ChunkDemuxerTest, AddIdFailures) { | 2523 TEST_F(ChunkDemuxerTest, AddIdFailures) { |
2542 EXPECT_CALL(*this, DemuxerOpened()); | 2524 EXPECT_CALL(*this, DemuxerOpened()); |
2543 demuxer_->Initialize( | 2525 demuxer_->Initialize( |
2544 &host_, CreateInitDoneCB(kDefaultDuration(), PIPELINE_OK), true); | 2526 &host_, CreateInitDoneCB(kDefaultDuration(), PIPELINE_OK), true); |
2545 | 2527 |
2546 std::string audio_id = "audio1"; | 2528 std::string audio_id = "audio1"; |
2547 std::string video_id = "video1"; | 2529 std::string video_id = "video1"; |
2548 | 2530 |
2549 ASSERT_EQ(AddId(audio_id, HAS_AUDIO), ChunkDemuxer::kOk); | 2531 ASSERT_EQ(AddId(audio_id, HAS_AUDIO), ChunkDemuxer::kOk); |
2550 | 2532 |
2551 // Adding an id with audio/video should fail because we already added audio. | |
2552 ASSERT_EQ(AddId(), ChunkDemuxer::kReachedIdLimit); | |
2553 | |
2554 ExpectInitMediaLogs(HAS_AUDIO); | 2533 ExpectInitMediaLogs(HAS_AUDIO); |
2555 EXPECT_CALL(*this, InitSegmentReceivedMock(_)); | 2534 EXPECT_CALL(*this, InitSegmentReceivedMock(_)); |
2556 ASSERT_TRUE(AppendInitSegmentWithSourceId(audio_id, HAS_AUDIO)); | 2535 ASSERT_TRUE(AppendInitSegmentWithSourceId(audio_id, HAS_AUDIO)); |
2557 | 2536 |
2558 // Adding an id after append should fail. | 2537 // Adding an id after append should fail. |
2559 ASSERT_EQ(AddId(video_id, HAS_VIDEO), ChunkDemuxer::kReachedIdLimit); | 2538 ASSERT_EQ(AddId(video_id, HAS_VIDEO), ChunkDemuxer::kReachedIdLimit); |
2560 } | 2539 } |
2561 | 2540 |
2562 // Test that Read() calls after a RemoveId() return "end of stream" buffers. | 2541 // Test that Read() calls after a RemoveId() return "end of stream" buffers. |
2563 TEST_F(ChunkDemuxerTest, RemoveId) { | 2542 TEST_F(ChunkDemuxerTest, RemoveId) { |
(...skipping 576 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3140 | 3119 |
3141 #if defined(USE_PROPRIETARY_CODECS) | 3120 #if defined(USE_PROPRIETARY_CODECS) |
3142 #if defined(OS_ANDROID) | 3121 #if defined(OS_ANDROID) |
3143 if (HasPlatformDecoderSupport()) | 3122 if (HasPlatformDecoderSupport()) |
3144 expected = ChunkDemuxer::kOk; | 3123 expected = ChunkDemuxer::kOk; |
3145 #else | 3124 #else |
3146 expected = ChunkDemuxer::kOk; | 3125 expected = ChunkDemuxer::kOk; |
3147 #endif | 3126 #endif |
3148 #endif | 3127 #endif |
3149 | 3128 |
3150 std::vector<std::string> codecs; | 3129 EXPECT_EQ(demuxer_->AddId("source_id", "video/mp4", "avc1.4D4041"), expected); |
3151 codecs.push_back("avc1.4D4041"); | |
3152 | |
3153 EXPECT_EQ(demuxer_->AddId("source_id", "video/mp4", codecs), expected); | |
3154 } | 3130 } |
3155 | 3131 |
3156 // Test codec ID's that are not compliant with RFC6381, but have been | 3132 // Test codec ID's that are not compliant with RFC6381, but have been |
3157 // seen in the wild. | 3133 // seen in the wild. |
3158 TEST_F(ChunkDemuxerTest, CodecIDsThatAreNotRFC6381Compliant) { | 3134 TEST_F(ChunkDemuxerTest, CodecIDsThatAreNotRFC6381Compliant) { |
3159 ChunkDemuxer::Status expected = ChunkDemuxer::kNotSupported; | 3135 ChunkDemuxer::Status expected = ChunkDemuxer::kNotSupported; |
3160 | 3136 |
3161 #if defined(USE_PROPRIETARY_CODECS) | 3137 #if defined(USE_PROPRIETARY_CODECS) |
3162 expected = ChunkDemuxer::kOk; | 3138 expected = ChunkDemuxer::kOk; |
3163 #endif | 3139 #endif |
3164 const char* codec_ids[] = { | 3140 const char* codec_ids[] = { |
3165 // GPAC places leading zeros on the audio object type. | 3141 // GPAC places leading zeros on the audio object type. |
3166 "mp4a.40.02", | 3142 "mp4a.40.02", |
3167 "mp4a.40.05" | 3143 "mp4a.40.05" |
3168 }; | 3144 }; |
3169 | 3145 |
3170 for (size_t i = 0; i < arraysize(codec_ids); ++i) { | 3146 for (size_t i = 0; i < arraysize(codec_ids); ++i) { |
3171 std::vector<std::string> codecs; | |
3172 codecs.push_back(codec_ids[i]); | |
3173 | |
3174 ChunkDemuxer::Status result = | 3147 ChunkDemuxer::Status result = |
3175 demuxer_->AddId("source_id", "audio/mp4", codecs); | 3148 demuxer_->AddId("source_id", "audio/mp4", codec_ids[i]); |
3176 | 3149 |
3177 EXPECT_EQ(result, expected) | 3150 EXPECT_EQ(result, expected) |
3178 << "Fail to add codec_id '" << codec_ids[i] << "'"; | 3151 << "Fail to add codec_id '" << codec_ids[i] << "'"; |
3179 | 3152 |
3180 if (result == ChunkDemuxer::kOk) | 3153 if (result == ChunkDemuxer::kOk) |
3181 demuxer_->RemoveId("source_id"); | 3154 demuxer_->RemoveId("source_id"); |
3182 } | 3155 } |
3183 } | 3156 } |
3184 | 3157 |
3185 TEST_F(ChunkDemuxerTest, EndOfStreamStillSetAfterSeek) { | 3158 TEST_F(ChunkDemuxerTest, EndOfStreamStillSetAfterSeek) { |
(...skipping 281 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3467 append_window_end_for_next_append_, | 3440 append_window_end_for_next_append_, |
3468 ×tamp_offset_map_[kSourceId]); | 3441 ×tamp_offset_map_[kSourceId]); |
3469 | 3442 |
3470 // After ResetParserState(), parsing should no longer be in the middle of a | 3443 // After ResetParserState(), parsing should no longer be in the middle of a |
3471 // media segment. | 3444 // media segment. |
3472 ASSERT_FALSE(demuxer_->IsParsingMediaSegment(kSourceId)); | 3445 ASSERT_FALSE(demuxer_->IsParsingMediaSegment(kSourceId)); |
3473 } | 3446 } |
3474 | 3447 |
3475 #if defined(USE_PROPRIETARY_CODECS) | 3448 #if defined(USE_PROPRIETARY_CODECS) |
3476 #if BUILDFLAG(ENABLE_MSE_MPEG2TS_STREAM_PARSER) | 3449 #if BUILDFLAG(ENABLE_MSE_MPEG2TS_STREAM_PARSER) |
| 3450 namespace { |
| 3451 const char* kMp2tMimeType = "video/mp2t"; |
| 3452 const char* kMp2tCodecs = "mp4a.40.2,avc1.640028"; |
| 3453 } |
| 3454 |
3477 TEST_F(ChunkDemuxerTest, EmitBuffersDuringAbort) { | 3455 TEST_F(ChunkDemuxerTest, EmitBuffersDuringAbort) { |
3478 EXPECT_CALL(*this, DemuxerOpened()); | 3456 EXPECT_CALL(*this, DemuxerOpened()); |
3479 EXPECT_MEDIA_LOG(FoundStream("audio")); | 3457 EXPECT_MEDIA_LOG(FoundStream("audio")); |
3480 EXPECT_MEDIA_LOG(CodecName("audio", "aac")); | 3458 EXPECT_MEDIA_LOG(CodecName("audio", "aac")); |
3481 EXPECT_MEDIA_LOG(FoundStream("video")); | 3459 EXPECT_MEDIA_LOG(FoundStream("video")); |
3482 EXPECT_MEDIA_LOG(CodecName("video", "h264")); | 3460 EXPECT_MEDIA_LOG(CodecName("video", "h264")); |
3483 demuxer_->Initialize(&host_, CreateInitDoneCB(kInfiniteDuration, PIPELINE_OK), | 3461 demuxer_->Initialize(&host_, CreateInitDoneCB(kInfiniteDuration, PIPELINE_OK), |
3484 true); | 3462 true); |
3485 EXPECT_EQ(ChunkDemuxer::kOk, AddIdForMp2tSource(kSourceId)); | 3463 EXPECT_EQ(ChunkDemuxer::kOk, AddId(kSourceId, kMp2tMimeType, kMp2tCodecs)); |
3486 | 3464 |
3487 // For info: | 3465 // For info: |
3488 // DTS/PTS derived using dvbsnoop -s ts -if bear-1280x720.ts -tssubdecode | 3466 // DTS/PTS derived using dvbsnoop -s ts -if bear-1280x720.ts -tssubdecode |
3489 // Video: first PES: | 3467 // Video: first PES: |
3490 // PTS: 126912 (0x0001efc0) [= 90 kHz-Timestamp: 0:00:01.4101] | 3468 // PTS: 126912 (0x0001efc0) [= 90 kHz-Timestamp: 0:00:01.4101] |
3491 // DTS: 123909 (0x0001e405) [= 90 kHz-Timestamp: 0:00:01.3767] | 3469 // DTS: 123909 (0x0001e405) [= 90 kHz-Timestamp: 0:00:01.3767] |
3492 // Audio: first PES: | 3470 // Audio: first PES: |
3493 // PTS: 126000 (0x0001ec30) [= 90 kHz-Timestamp: 0:00:01.4000] | 3471 // PTS: 126000 (0x0001ec30) [= 90 kHz-Timestamp: 0:00:01.4000] |
3494 // DTS: 123910 (0x0001e406) [= 90 kHz-Timestamp: 0:00:01.3767] | 3472 // DTS: 123910 (0x0001e406) [= 90 kHz-Timestamp: 0:00:01.3767] |
3495 // Video: last PES: | 3473 // Video: last PES: |
(...skipping 27 matching lines...) Expand all Loading... |
3523 } | 3501 } |
3524 | 3502 |
3525 TEST_F(ChunkDemuxerTest, SeekCompleteDuringAbort) { | 3503 TEST_F(ChunkDemuxerTest, SeekCompleteDuringAbort) { |
3526 EXPECT_CALL(*this, DemuxerOpened()); | 3504 EXPECT_CALL(*this, DemuxerOpened()); |
3527 EXPECT_MEDIA_LOG(FoundStream("audio")); | 3505 EXPECT_MEDIA_LOG(FoundStream("audio")); |
3528 EXPECT_MEDIA_LOG(CodecName("audio", "aac")); | 3506 EXPECT_MEDIA_LOG(CodecName("audio", "aac")); |
3529 EXPECT_MEDIA_LOG(FoundStream("video")); | 3507 EXPECT_MEDIA_LOG(FoundStream("video")); |
3530 EXPECT_MEDIA_LOG(CodecName("video", "h264")); | 3508 EXPECT_MEDIA_LOG(CodecName("video", "h264")); |
3531 demuxer_->Initialize(&host_, CreateInitDoneCB(kInfiniteDuration, PIPELINE_OK), | 3509 demuxer_->Initialize(&host_, CreateInitDoneCB(kInfiniteDuration, PIPELINE_OK), |
3532 true); | 3510 true); |
3533 EXPECT_EQ(ChunkDemuxer::kOk, AddIdForMp2tSource(kSourceId)); | 3511 EXPECT_EQ(ChunkDemuxer::kOk, AddId(kSourceId, kMp2tMimeType, kMp2tCodecs)); |
3534 | 3512 |
3535 // For info: | 3513 // For info: |
3536 // DTS/PTS derived using dvbsnoop -s ts -if bear-1280x720.ts -tssubdecode | 3514 // DTS/PTS derived using dvbsnoop -s ts -if bear-1280x720.ts -tssubdecode |
3537 // Video: first PES: | 3515 // Video: first PES: |
3538 // PTS: 126912 (0x0001efc0) [= 90 kHz-Timestamp: 0:00:01.4101] | 3516 // PTS: 126912 (0x0001efc0) [= 90 kHz-Timestamp: 0:00:01.4101] |
3539 // DTS: 123909 (0x0001e405) [= 90 kHz-Timestamp: 0:00:01.3767] | 3517 // DTS: 123909 (0x0001e405) [= 90 kHz-Timestamp: 0:00:01.3767] |
3540 // Audio: first PES: | 3518 // Audio: first PES: |
3541 // PTS: 126000 (0x0001ec30) [= 90 kHz-Timestamp: 0:00:01.4000] | 3519 // PTS: 126000 (0x0001ec30) [= 90 kHz-Timestamp: 0:00:01.4000] |
3542 // DTS: 123910 (0x0001e406) [= 90 kHz-Timestamp: 0:00:01.3767] | 3520 // DTS: 123910 (0x0001e406) [= 90 kHz-Timestamp: 0:00:01.3767] |
3543 // Video: last PES: | 3521 // Video: last PES: |
(...skipping 903 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4447 CheckExpectedBuffers(audio_stream, kAudioStreamInfo); | 4425 CheckExpectedBuffers(audio_stream, kAudioStreamInfo); |
4448 CheckExpectedBuffers(video_stream, kVideoStreamInfo); | 4426 CheckExpectedBuffers(video_stream, kVideoStreamInfo); |
4449 | 4427 |
4450 // But if we pretend that playback position has moved to 120ms, that allows | 4428 // But if we pretend that playback position has moved to 120ms, that allows |
4451 // EvictCodedFrames to garbage-collect enough data to succeed. | 4429 // EvictCodedFrames to garbage-collect enough data to succeed. |
4452 ASSERT_TRUE(demuxer_->EvictCodedFrames(kSourceId, | 4430 ASSERT_TRUE(demuxer_->EvictCodedFrames(kSourceId, |
4453 base::TimeDelta::FromMilliseconds(120), | 4431 base::TimeDelta::FromMilliseconds(120), |
4454 80)); | 4432 80)); |
4455 | 4433 |
4456 Seek(base::TimeDelta::FromMilliseconds(0)); | 4434 Seek(base::TimeDelta::FromMilliseconds(0)); |
4457 // Audio stream had 8 buffers, video stream had 15. We told EvictCodedFrames | |
4458 // that the new data size is 8 blocks muxed, i.e. 80 bytes. Given the current | |
4459 // ratio of video to the total data size (15 : (8+15) ~= 0.65) the estimated | |
4460 // sizes of video and audio data in the new 80 byte chunk are 52 bytes for | |
4461 // video (80*0.65 = 52) and 28 bytes for audio (80 - 52). | |
4462 // Given these numbers MSE GC will remove just one audio block (since current | |
4463 // audio size is 80 bytes, new data is 28 bytes, we need to remove just one 10 | |
4464 // byte block to stay under 100 bytes memory limit after append | |
4465 // 80 - 10 + 28 = 98). | |
4466 // For video stream 150 + 52 = 202. Video limit is 150 bytes. We need to | |
4467 // remove at least 6 blocks to stay under limit. | |
4468 CheckExpectedBuffers(audio_stream, "40K 80K 120K 160K 200K 240K 280K"); | |
4469 CheckExpectedBuffers(video_stream, "60K 70 80K 90 100K 110 120K 130 140K"); | |
4470 } | 4435 } |
4471 | 4436 |
4472 TEST_F(ChunkDemuxerTest, SegmentMissingAudioFrame_AudioOnly) { | 4437 TEST_F(ChunkDemuxerTest, SegmentMissingAudioFrame_AudioOnly) { |
4473 ASSERT_TRUE(InitDemuxer(HAS_AUDIO)); | 4438 ASSERT_TRUE(InitDemuxer(HAS_AUDIO)); |
4474 EXPECT_MEDIA_LOG(SegmentMissingFrames("audio")); | 4439 EXPECT_MEDIA_LOG(SegmentMissingFrames("2")); |
4475 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); | 4440 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); |
4476 } | 4441 } |
4477 | 4442 |
4478 TEST_F(ChunkDemuxerTest, SegmentMissingVideoFrame_VideoOnly) { | 4443 TEST_F(ChunkDemuxerTest, SegmentMissingVideoFrame_VideoOnly) { |
4479 ASSERT_TRUE(InitDemuxer(HAS_VIDEO)); | 4444 ASSERT_TRUE(InitDemuxer(HAS_VIDEO)); |
4480 EXPECT_MEDIA_LOG(SegmentMissingFrames("video")); | 4445 EXPECT_MEDIA_LOG(SegmentMissingFrames("1")); |
4481 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); | 4446 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); |
4482 } | 4447 } |
4483 | 4448 |
4484 TEST_F(ChunkDemuxerTest, SegmentMissingAudioFrame_AudioVideo) { | 4449 TEST_F(ChunkDemuxerTest, SegmentMissingAudioFrame_AudioVideo) { |
4485 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 4450 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
4486 EXPECT_MEDIA_LOG(SegmentMissingFrames("audio")); | 4451 EXPECT_MEDIA_LOG(SegmentMissingFrames("2")); |
4487 AppendSingleStreamCluster(kSourceId, kVideoTrackNum, 0, 10); | 4452 AppendSingleStreamCluster(kSourceId, kVideoTrackNum, 0, 10); |
4488 } | 4453 } |
4489 | 4454 |
4490 TEST_F(ChunkDemuxerTest, SegmentMissingVideoFrame_AudioVideo) { | 4455 TEST_F(ChunkDemuxerTest, SegmentMissingVideoFrame_AudioVideo) { |
4491 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 4456 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
4492 EXPECT_MEDIA_LOG(SegmentMissingFrames("video")); | 4457 EXPECT_MEDIA_LOG(SegmentMissingFrames("1")); |
4493 AppendSingleStreamCluster(kSourceId, kAudioTrackNum, 0, 10); | 4458 AppendSingleStreamCluster(kSourceId, kAudioTrackNum, 0, 10); |
4494 } | 4459 } |
4495 | 4460 |
4496 TEST_F(ChunkDemuxerTest, SegmentMissingAudioVideoFrames) { | 4461 TEST_F(ChunkDemuxerTest, SegmentMissingAudioVideoFrames) { |
4497 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 4462 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
4498 EXPECT_MEDIA_LOG(SegmentMissingFrames("audio or video")); | 4463 EXPECT_MEDIA_LOG(SegmentMissingFrames("1")); |
| 4464 EXPECT_MEDIA_LOG(SegmentMissingFrames("2")); |
4499 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); | 4465 ASSERT_TRUE(AppendCluster(GenerateEmptyCluster(0))); |
4500 } | 4466 } |
4501 | 4467 |
4502 TEST_F(ChunkDemuxerTest, RelaxedKeyframe_FirstSegmentMissingKeyframe) { | 4468 TEST_F(ChunkDemuxerTest, RelaxedKeyframe_FirstSegmentMissingKeyframe) { |
4503 // Append V:[n n n][n n K] | 4469 // Append V:[n n n][n n K] |
4504 // Expect V: [K] | 4470 // Expect V: [K] |
4505 ASSERT_TRUE(InitDemuxer(HAS_VIDEO)); | 4471 ASSERT_TRUE(InitDemuxer(HAS_VIDEO)); |
4506 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); | 4472 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); |
4507 | 4473 |
4508 EXPECT_MEDIA_LOG(WebMSimpleBlockDurationEstimated(10)).Times(2); | 4474 EXPECT_MEDIA_LOG(WebMSimpleBlockDurationEstimated(10)).Times(2); |
(...skipping 113 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4622 // V: [n n n][n K n] | 4588 // V: [n n n][n K n] |
4623 // Expect: | 4589 // Expect: |
4624 // A: [K K K][K K K] | 4590 // A: [K K K][K K K] |
4625 // V [................K n] (As would occur if there really were a | 4591 // V [................K n] (As would occur if there really were a |
4626 // jagged cluster start and not badly muxed clusters as used to | 4592 // jagged cluster start and not badly muxed clusters as used to |
4627 // simulate a jagged start in this test.) | 4593 // simulate a jagged start in this test.) |
4628 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 4594 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
4629 DemuxerStream* audio_stream = demuxer_->GetStream(DemuxerStream::AUDIO); | 4595 DemuxerStream* audio_stream = demuxer_->GetStream(DemuxerStream::AUDIO); |
4630 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); | 4596 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); |
4631 | 4597 |
4632 EXPECT_MEDIA_LOG(SegmentMissingFrames("video")); | 4598 EXPECT_MEDIA_LOG(SegmentMissingFrames("1")); |
4633 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "0K 10K 20D10K"), | 4599 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "0K 10K 20D10K"), |
4634 MuxedStreamInfo(kVideoTrackNum, "")); | 4600 MuxedStreamInfo(kVideoTrackNum, "")); |
4635 demuxer_->Remove(kSourceId, base::TimeDelta(), | 4601 demuxer_->Remove(kSourceId, base::TimeDelta(), |
4636 base::TimeDelta::FromMilliseconds(30)); | 4602 base::TimeDelta::FromMilliseconds(30)); |
4637 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "30K 40K 50D10K"), | 4603 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "30K 40K 50D10K"), |
4638 MuxedStreamInfo(kVideoTrackNum, "30 40 50", 10)); | 4604 MuxedStreamInfo(kVideoTrackNum, "30 40 50", 10)); |
4639 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "60K 70K 80D10K"), | 4605 AppendMuxedCluster(MuxedStreamInfo(kAudioTrackNum, "60K 70K 80D10K"), |
4640 MuxedStreamInfo(kVideoTrackNum, "60 70K 80", 10)); | 4606 MuxedStreamInfo(kVideoTrackNum, "60 70K 80", 10)); |
4641 CheckExpectedRanges(DemuxerStream::AUDIO, "{ [30,90) }"); | 4607 CheckExpectedRanges(DemuxerStream::AUDIO, "{ [30,90) }"); |
4642 CheckExpectedRanges(DemuxerStream::VIDEO, "{ [0,90) }"); | 4608 CheckExpectedRanges(DemuxerStream::VIDEO, "{ [0,90) }"); |
(...skipping 94 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
4737 TEST_F(ChunkDemuxerTest, StreamStatusNotifications) { | 4703 TEST_F(ChunkDemuxerTest, StreamStatusNotifications) { |
4738 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); | 4704 ASSERT_TRUE(InitDemuxer(HAS_AUDIO | HAS_VIDEO)); |
4739 DemuxerStream* audio_stream = demuxer_->GetStream(DemuxerStream::AUDIO); | 4705 DemuxerStream* audio_stream = demuxer_->GetStream(DemuxerStream::AUDIO); |
4740 EXPECT_NE(nullptr, audio_stream); | 4706 EXPECT_NE(nullptr, audio_stream); |
4741 CheckStreamStatusNotifications(audio_stream); | 4707 CheckStreamStatusNotifications(audio_stream); |
4742 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); | 4708 DemuxerStream* video_stream = demuxer_->GetStream(DemuxerStream::VIDEO); |
4743 EXPECT_NE(nullptr, video_stream); | 4709 EXPECT_NE(nullptr, video_stream); |
4744 CheckStreamStatusNotifications(video_stream); | 4710 CheckStreamStatusNotifications(video_stream); |
4745 } | 4711 } |
4746 | 4712 |
| 4713 TEST_F(ChunkDemuxerTest, MultipleIds) { |
| 4714 CreateNewDemuxer(); |
| 4715 EXPECT_CALL(*this, DemuxerOpened()); |
| 4716 EXPECT_CALL(host_, SetDuration(_)).Times(2); |
| 4717 demuxer_->Initialize(&host_, CreateInitDoneCB(kNoTimestamp, PIPELINE_OK), |
| 4718 true); |
| 4719 |
| 4720 const char* kId1 = "id1"; |
| 4721 const char* kId2 = "id2"; |
| 4722 EXPECT_EQ(AddId(kId1, "video/webm", "opus,vp9"), ChunkDemuxer::kOk); |
| 4723 EXPECT_EQ(AddId(kId2, "video/webm", "opus,vp9"), ChunkDemuxer::kOk); |
| 4724 scoped_refptr<DecoderBuffer> data1 = ReadTestDataFile("green-a300hz.webm"); |
| 4725 scoped_refptr<DecoderBuffer> data2 = ReadTestDataFile("red-a500hz.webm"); |
| 4726 |
| 4727 EXPECT_MEDIA_LOG(FoundStream("audio")).Times(2); |
| 4728 EXPECT_MEDIA_LOG(FoundStream("video")).Times(2); |
| 4729 EXPECT_MEDIA_LOG(CodecName("audio", "opus")).Times(2); |
| 4730 EXPECT_MEDIA_LOG(CodecName("video", "vp9")).Times(2); |
| 4731 EXPECT_CALL(*this, InitSegmentReceivedMock(_)).Times(2); |
| 4732 EXPECT_MEDIA_LOG(SegmentMissingFrames("1")).Times(AnyNumber()); |
| 4733 |
| 4734 EXPECT_TRUE(AppendData(kId1, data1->data(), data1->data_size())); |
| 4735 EXPECT_TRUE(AppendData(kId2, data2->data(), data2->data_size())); |
| 4736 CheckExpectedRanges(kId1, "{ [0,10007) }"); |
| 4737 CheckExpectedRanges(kId2, "{ [0,10007) }"); |
| 4738 } |
| 4739 |
| 4740 TEST_F(ChunkDemuxerTest, CompleteInitAfterIdRemoved) { |
| 4741 CreateNewDemuxer(); |
| 4742 EXPECT_CALL(*this, DemuxerOpened()); |
| 4743 demuxer_->Initialize(&host_, |
| 4744 CreateInitDoneCB(kDefaultDuration(), PIPELINE_OK), true); |
| 4745 |
| 4746 // Add two ids, then remove one of the ids and verify that adding data only |
| 4747 // for the remaining id still triggers the InitDoneCB. |
| 4748 const char* kId1 = "id1"; |
| 4749 const char* kId2 = "id2"; |
| 4750 EXPECT_EQ(AddId(kId1, "video/webm", "vp8"), ChunkDemuxer::kOk); |
| 4751 EXPECT_EQ(AddId(kId2, "video/mp4", "avc1.640028"), ChunkDemuxer::kOk); |
| 4752 demuxer_->RemoveId(kId2); |
| 4753 |
| 4754 EXPECT_CALL(*this, InitSegmentReceivedMock(_)); |
| 4755 EXPECT_MEDIA_LOG(WebMSimpleBlockDurationEstimated(30)); |
| 4756 EXPECT_MEDIA_LOG(FoundStream("video")); |
| 4757 EXPECT_MEDIA_LOG(CodecName("video", "vp8")); |
| 4758 |
| 4759 ASSERT_TRUE(AppendInitSegmentWithSourceId(kId1, HAS_VIDEO)); |
| 4760 AppendSingleStreamCluster(kId1, kVideoTrackNum, "0K 30 60 90"); |
| 4761 } |
| 4762 |
4747 } // namespace media | 4763 } // namespace media |
OLD | NEW |