/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "gtest/gtest.h" #include "MediaData.h" #include "mozilla/ArrayUtils.h" #include "mp4_demuxer/BufferStream.h" #include "mp4_demuxer/MP4Metadata.h" #include "mp4_demuxer/MoofParser.h" using namespace mozilla; using namespace mp4_demuxer; class TestStream : public Stream { public: TestStream(const uint8_t* aBuffer, size_t aSize) : mHighestSuccessfulEndOffset(0) , mBuffer(aBuffer) , mSize(aSize) { } bool ReadAt(int64_t aOffset, void* aData, size_t aLength, size_t* aBytesRead) override { if (aOffset < 0 || aOffset > static_cast(mSize)) { return false; } // After the test, 0 <= aOffset <= mSize <= SIZE_MAX, so it's safe to cast to size_t. size_t offset = static_cast(aOffset); // Don't read past the end (but it's not an error to try). if (aLength > mSize - offset) { aLength = mSize - offset; } // Now, 0 <= offset <= offset + aLength <= mSize <= SIZE_MAX. *aBytesRead = aLength; memcpy(aData, mBuffer + offset, aLength); if (mHighestSuccessfulEndOffset < offset + aLength) { mHighestSuccessfulEndOffset = offset + aLength; } return true; } bool CachedReadAt(int64_t aOffset, void* aData, size_t aLength, size_t* aBytesRead) override { return ReadAt(aOffset, aData, aLength, aBytesRead); } bool Length(int64_t* aLength) override { *aLength = mSize; return true; } void DiscardBefore(int64_t aOffset) override { } // Offset past the last character ever read. 0 when nothing read yet. size_t mHighestSuccessfulEndOffset; protected: virtual ~TestStream() { } const uint8_t* mBuffer; size_t mSize; }; TEST(stagefright_MP4Metadata, EmptyStream) { RefPtr stream = new TestStream(nullptr, 0); EXPECT_FALSE(MP4Metadata::HasCompleteMetadata(stream)); RefPtr metadataBuffer = MP4Metadata::Metadata(stream); EXPECT_FALSE(metadataBuffer); MP4Metadata metadata(stream); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kUndefinedTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kAudioTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kVideoTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kTextTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(static_cast(-1))); EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kUndefinedTrack, 0)); EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kAudioTrack, 0)); EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kVideoTrack, 0)); EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kTextTrack, 0)); EXPECT_FALSE(metadata.GetTrackInfo(static_cast(-1), 0)); // We can seek anywhere in any MPEG4. EXPECT_TRUE(metadata.CanSeek()); EXPECT_FALSE(metadata.Crypto().valid); } TEST(stagefright_MoofParser, EmptyStream) { RefPtr stream = new TestStream(nullptr, 0); MoofParser parser(stream, 0, false); EXPECT_EQ(0u, parser.mOffset); EXPECT_TRUE(parser.ReachedEnd()); MediaByteRangeSet byteRanges; EXPECT_FALSE(parser.RebuildFragmentedIndex(byteRanges)); EXPECT_TRUE(parser.GetCompositionRange(byteRanges).IsNull()); EXPECT_TRUE(parser.mInitRange.IsEmpty()); EXPECT_EQ(0u, parser.mOffset); EXPECT_TRUE(parser.ReachedEnd()); EXPECT_FALSE(parser.HasMetadata()); RefPtr metadataBuffer = parser.Metadata(); EXPECT_FALSE(metadataBuffer); EXPECT_TRUE(parser.FirstCompleteMediaSegment().IsEmpty()); EXPECT_TRUE(parser.FirstCompleteMediaHeader().IsEmpty()); } nsTArray ReadTestFile(const char* aFilename) { if (!aFilename) { return {}; } FILE* f = fopen(aFilename, "rb"); if (!f) { return {}; } if (fseek(f, 0, SEEK_END) != 0) { fclose(f); return {}; } long position = ftell(f); // I know EOF==-1, so this test is made obsolete by '<0', but I don't want // the code to rely on that. if (position == 0 || position == EOF || position < 0) { fclose(f); return {}; } if (fseek(f, 0, SEEK_SET) != 0) { fclose(f); return {}; } size_t len = static_cast(position); nsTArray buffer(len); buffer.SetLength(len); size_t read = fread(buffer.Elements(), 1, len, f); fclose(f); if (read != len) { return {}; } return buffer; } struct TestFileData { const char* mFilename; uint32_t mNumberVideoTracks; int64_t mVideoDuration; // For first video track, -1 if N/A. int32_t mWidth; int32_t mHeight; uint32_t mNumberAudioTracks; int64_t mAudioDuration; // For first audio track, -1 if N/A. bool mHasCrypto; uint64_t mMoofReachedOffset; // or 0 for the end. bool mValidMoof; bool mHeader; }; static const TestFileData testFiles[] = { // filename #V dur w h #A dur crypt off moof headr { "test_case_1156505.mp4", 0, -1, 0, 0, 0, -1, false, 152, false, false }, { "test_case_1181213.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1181215.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1181220.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1181223.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1181719.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1185230.mp4", 1, 416666, 320, 240, 1, 5, false, 0, false, false }, { "test_case_1187067.mp4", 1, 80000, 160, 90, 0, -1, false, 0, false, false }, { "test_case_1200326.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1204580.mp4", 1, 502500, 320, 180, 0, -1, false, 0, false, false }, { "test_case_1216748.mp4", 0, -1, 0, 0, 0, -1, false, 152, false, false }, { "test_case_1296473.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1296532.mp4", 1, 5589333, 560, 320, 1, 5589333, true, 0, true, true }, { "test_case_1301065.mp4", 0, -1, 0, 0, 1, 100079991719000000, false, 0, false, false }, { "test_case_1301065-u32max.mp4", 0, -1, 0, 0, 1, 97391548639, false, 0, false, false }, { "test_case_1301065-max-ez.mp4", 0, -1, 0, 0, 1, 209146758205306, false, 0, false, false }, { "test_case_1301065-harder.mp4", 0, -1, 0, 0, 1, 209146758205328, false, 0, false, false }, { "test_case_1301065-max-ok.mp4", 0, -1, 0, 0, 1, 9223372036854775804, false, 0, false, false }, { "test_case_1301065-overfl.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1301065-i64max.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1301065-i64min.mp4", 0, -1, 0, 0, 0, -1, false, 0, false, false }, { "test_case_1301065-u64max.mp4", 0, -1, 0, 0, 1, 0, false, 0, false, false }, { "test_case_1351094.mp4", 0, -1, 0, 0, 0, -1, false, 0, true, true }, }; TEST(stagefright_MPEG4Metadata, test_case_mp4) { for (size_t test = 0; test < ArrayLength(testFiles); ++test) { nsTArray buffer = ReadTestFile(testFiles[test].mFilename); ASSERT_FALSE(buffer.IsEmpty()); RefPtr stream = new TestStream(buffer.Elements(), buffer.Length()); EXPECT_TRUE(MP4Metadata::HasCompleteMetadata(stream)); RefPtr metadataBuffer = MP4Metadata::Metadata(stream); EXPECT_TRUE(metadataBuffer); MP4Metadata metadata(stream); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kUndefinedTrack)); EXPECT_EQ(testFiles[test].mNumberAudioTracks, metadata.GetNumberTracks(TrackInfo::kAudioTrack)); EXPECT_EQ(testFiles[test].mNumberVideoTracks, metadata.GetNumberTracks(TrackInfo::kVideoTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(TrackInfo::kTextTrack)); EXPECT_EQ(0u, metadata.GetNumberTracks(static_cast(-1))); EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kUndefinedTrack, 0)); UniquePtr trackInfo = metadata.GetTrackInfo(TrackInfo::kVideoTrack, 0); if (testFiles[test].mNumberVideoTracks == 0) { EXPECT_TRUE(!trackInfo); } else { ASSERT_TRUE(!!trackInfo); const VideoInfo* videoInfo = trackInfo->GetAsVideoInfo(); ASSERT_TRUE(!!videoInfo); EXPECT_TRUE(videoInfo->IsValid()); EXPECT_TRUE(videoInfo->IsVideo()); EXPECT_EQ(testFiles[test].mVideoDuration, videoInfo->mDuration); EXPECT_EQ(testFiles[test].mWidth, videoInfo->mDisplay.width); EXPECT_EQ(testFiles[test].mHeight, videoInfo->mDisplay.height); FallibleTArray indices; EXPECT_TRUE(metadata.ReadTrackIndex(indices, videoInfo->mTrackId)); for (const mp4_demuxer::Index::Indice& indice : indices) { EXPECT_TRUE(indice.start_offset <= indice.end_offset); EXPECT_TRUE(indice.start_composition <= indice.end_composition); } } trackInfo = metadata.GetTrackInfo(TrackInfo::kAudioTrack, 0); if (testFiles[test].mNumberAudioTracks == 0) { EXPECT_TRUE(!trackInfo); } else { ASSERT_TRUE(!!trackInfo); const AudioInfo* audioInfo = trackInfo->GetAsAudioInfo(); ASSERT_TRUE(!!audioInfo); EXPECT_TRUE(audioInfo->IsValid()); EXPECT_TRUE(audioInfo->IsAudio()); EXPECT_EQ(testFiles[test].mAudioDuration, audioInfo->mDuration); FallibleTArray indices; EXPECT_TRUE(metadata.ReadTrackIndex(indices, audioInfo->mTrackId)); for (const mp4_demuxer::Index::Indice& indice : indices) { EXPECT_TRUE(indice.start_offset <= indice.end_offset); EXPECT_TRUE(indice.start_composition <= indice.end_composition); } } EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kTextTrack, 0)); EXPECT_FALSE(metadata.GetTrackInfo(static_cast(-1), 0)); // We can see anywhere in any MPEG4. EXPECT_TRUE(metadata.CanSeek()); EXPECT_EQ(testFiles[test].mHasCrypto, metadata.Crypto().valid); } } // Bug 1224019: This test produces way to much output, disabling for now. #if 0 TEST(stagefright_MPEG4Metadata, test_case_mp4_subsets) { static const size_t step = 1u; for (size_t test = 0; test < ArrayLength(testFiles); ++test) { nsTArray buffer = ReadTestFile(testFiles[test].mFilename); ASSERT_FALSE(buffer.IsEmpty()); ASSERT_LE(step, buffer.Length()); // Just exercizing the parser starting at different points through the file, // making sure it doesn't crash. // No checks because results would differ for each position. for (size_t offset = 0; offset < buffer.Length() - step; offset += step) { size_t size = buffer.Length() - offset; while (size > 0) { RefPtr stream = new TestStream(buffer.Elements() + offset, size); MP4Metadata::HasCompleteMetadata(stream); RefPtr metadataBuffer = MP4Metadata::Metadata(stream); MP4Metadata metadata(stream); if (stream->mHighestSuccessfulEndOffset <= 0) { // No successful reads -> Cutting down the size won't change anything. break; } if (stream->mHighestSuccessfulEndOffset < size) { // Read up to a point before the end -> Resize down to that point. size = stream->mHighestSuccessfulEndOffset; } else { // Read up to the end (or after?!) -> Just cut 1 byte. size -= 1; } } } } } #endif TEST(stagefright_MoofParser, test_case_mp4) { for (size_t test = 0; test < ArrayLength(testFiles); ++test) { nsTArray buffer = ReadTestFile(testFiles[test].mFilename); ASSERT_FALSE(buffer.IsEmpty()); RefPtr stream = new TestStream(buffer.Elements(), buffer.Length()); MoofParser parser(stream, 0, false); EXPECT_EQ(0u, parser.mOffset); EXPECT_FALSE(parser.ReachedEnd()); EXPECT_TRUE(parser.mInitRange.IsEmpty()); EXPECT_TRUE(parser.HasMetadata()); RefPtr metadataBuffer = parser.Metadata(); EXPECT_TRUE(metadataBuffer); EXPECT_FALSE(parser.mInitRange.IsEmpty()); const MediaByteRangeSet byteRanges( MediaByteRange(0, int64_t(buffer.Length()))); EXPECT_EQ(testFiles[test].mValidMoof, parser.RebuildFragmentedIndex(byteRanges)); if (testFiles[test].mMoofReachedOffset == 0) { EXPECT_EQ(buffer.Length(), parser.mOffset); EXPECT_TRUE(parser.ReachedEnd()); } else { EXPECT_EQ(testFiles[test].mMoofReachedOffset, parser.mOffset); EXPECT_FALSE(parser.ReachedEnd()); } EXPECT_FALSE(parser.mInitRange.IsEmpty()); EXPECT_TRUE(parser.GetCompositionRange(byteRanges).IsNull()); EXPECT_TRUE(parser.FirstCompleteMediaSegment().IsEmpty()); EXPECT_EQ(testFiles[test].mHeader, !parser.FirstCompleteMediaHeader().IsEmpty()); } } // Bug 1224019: This test produces way to much output, disabling for now. #if 0 TEST(stagefright_MoofParser, test_case_mp4_subsets) { const size_t step = 1u; for (size_t test = 0; test < ArrayLength(testFiles); ++test) { nsTArray buffer = ReadTestFile(testFiles[test].mFilename); ASSERT_FALSE(buffer.IsEmpty()); ASSERT_LE(step, buffer.Length()); // Just exercizing the parser starting at different points through the file, // making sure it doesn't crash. // No checks because results would differ for each position. for (size_t offset = 0; offset < buffer.Length() - step; offset += step) { size_t size = buffer.Length() - offset; while (size > 0) { RefPtr stream = new TestStream(buffer.Elements() + offset, size); MoofParser parser(stream, 0, false); MediaByteRangeSet byteRanges; EXPECT_FALSE(parser.RebuildFragmentedIndex(byteRanges)); parser.GetCompositionRange(byteRanges); parser.HasMetadata(); RefPtr metadataBuffer = parser.Metadata(); parser.FirstCompleteMediaSegment(); parser.FirstCompleteMediaHeader(); if (stream->mHighestSuccessfulEndOffset <= 0) { // No successful reads -> Cutting down the size won't change anything. break; } if (stream->mHighestSuccessfulEndOffset < size) { // Read up to a point before the end -> Resize down to that point. size = stream->mHighestSuccessfulEndOffset; } else { // Read up to the end (or after?!) -> Just cut 1 byte. size -= 1; } } } } } #endif uint8_t media_libstagefright_gtest_video_init_mp4[] = { 0x00, 0x00, 0x00, 0x18, 0x66, 0x74, 0x79, 0x70, 0x69, 0x73, 0x6f, 0x6d, 0x00, 0x00, 0x00, 0x01, 0x69, 0x73, 0x6f, 0x6d, 0x61, 0x76, 0x63, 0x31, 0x00, 0x00, 0x02, 0xd1, 0x6d, 0x6f, 0x6f, 0x76, 0x00, 0x00, 0x00, 0x6c, 0x6d, 0x76, 0x68, 0x64, 0x00, 0x00, 0x00, 0x00, 0xc8, 0x49, 0x73, 0xf8, 0xc8, 0x4a, 0xc5, 0x7a, 0x00, 0x00, 0x02, 0x58, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x03, 0x00, 0x00, 0x00, 0x18, 0x69, 0x6f, 0x64, 0x73, 0x00, 0x00, 0x00, 0x00, 0x10, 0x80, 0x80, 0x80, 0x07, 0x00, 0x4f, 0xff, 0xff, 0x29, 0x15, 0xff, 0x00, 0x00, 0x02, 0x0d, 0x74, 0x72, 0x61, 0x6b, 0x00, 0x00, 0x00, 0x5c, 0x74, 0x6b, 0x68, 0x64, 0x00, 0x00, 0x00, 0x01, 0xc8, 0x49, 0x73, 0xf8, 0xc8, 0x49, 0x73, 0xf9, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x00, 0x00, 0x00, 0x02, 0x80, 0x00, 0x00, 0x01, 0x68, 0x00, 0x00, 0x00, 0x00, 0x01, 0xa9, 0x6d, 0x64, 0x69, 0x61, 0x00, 0x00, 0x00, 0x20, 0x6d, 0x64, 0x68, 0x64, 0x00, 0x00, 0x00, 0x00, 0xc8, 0x49, 0x73, 0xf8, 0xc8, 0x49, 0x73, 0xf9, 0x00, 0x00, 0x75, 0x30, 0x00, 0x00, 0x00, 0x00, 0x55, 0xc4, 0x00, 0x00, 0x00, 0x00, 0x00, 0x38, 0x68, 0x64, 0x6c, 0x72, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x76, 0x69, 0x64, 0x65, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x47, 0x50, 0x41, 0x43, 0x20, 0x49, 0x53, 0x4f, 0x20, 0x56, 0x69, 0x64, 0x65, 0x6f, 0x20, 0x48, 0x61, 0x6e, 0x64, 0x6c, 0x65, 0x72, 0x00, 0x00, 0x00, 0x00, 0x01, 0x49, 0x6d, 0x69, 0x6e, 0x66, 0x00, 0x00, 0x00, 0x14, 0x76, 0x6d, 0x68, 0x64, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x24, 0x64, 0x69, 0x6e, 0x66, 0x00, 0x00, 0x00, 0x1c, 0x64, 0x72, 0x65, 0x66, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x0c, 0x75, 0x72, 0x6c, 0x20, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x09, 0x73, 0x74, 0x62, 0x6c, 0x00, 0x00, 0x00, 0xad, 0x73, 0x74, 0x73, 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x9d, 0x61, 0x76, 0x63, 0x31, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0x80, 0x01, 0x68, 0x00, 0x48, 0x00, 0x00, 0x00, 0x48, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x18, 0xff, 0xff, 0x00, 0x00, 0x00, 0x33, 0x61, 0x76, 0x63, 0x43, 0x01, 0x64, 0x00, 0x1f, 0xff, 0xe1, 0x00, 0x1b, 0x67, 0x64, 0x00, 0x1f, 0xac, 0x2c, 0xc5, 0x02, 0x80, 0xbf, 0xe5, 0xc0, 0x44, 0x00, 0x00, 0x03, 0x00, 0x04, 0x00, 0x00, 0x03, 0x00, 0xf2, 0x3c, 0x60, 0xc6, 0x58, 0x01, 0x00, 0x05, 0x68, 0xe9, 0x2b, 0x2c, 0x8b, 0x00, 0x00, 0x00, 0x14, 0x62, 0x74, 0x72, 0x74, 0x00, 0x01, 0x5a, 0xc2, 0x00, 0x24, 0x74, 0x38, 0x00, 0x09, 0x22, 0x00, 0x00, 0x00, 0x00, 0x10, 0x73, 0x74, 0x74, 0x73, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x63, 0x74, 0x74, 0x73, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x73, 0x74, 0x73, 0x63, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x14, 0x73, 0x74, 0x73, 0x7a, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x73, 0x74, 0x63, 0x6f, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x38, 0x6d, 0x76, 0x65, 0x78, 0x00, 0x00, 0x00, 0x10, 0x6d, 0x65, 0x68, 0x64, 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, 0x76, 0x18, 0x00, 0x00, 0x00, 0x20, 0x74, 0x72, 0x65, 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00, 0x03, 0xe8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x00, 0x00 }; const uint32_t media_libstagefright_gtest_video_init_mp4_len = 745; TEST(stagefright_MP4Metadata, EmptyCTTS) { RefPtr buffer = new MediaByteBuffer(media_libstagefright_gtest_video_init_mp4_len); buffer->AppendElements(media_libstagefright_gtest_video_init_mp4, media_libstagefright_gtest_video_init_mp4_len); RefPtr stream = new BufferStream(buffer); EXPECT_TRUE(MP4Metadata::HasCompleteMetadata(stream)); RefPtr metadataBuffer = MP4Metadata::Metadata(stream); EXPECT_TRUE(metadataBuffer); MP4Metadata metadata(stream); EXPECT_EQ(1u, metadata.GetNumberTracks(TrackInfo::kVideoTrack)); mozilla::UniquePtr track = metadata.GetTrackInfo(TrackInfo::kVideoTrack, 0); EXPECT_TRUE(track != nullptr); // We can seek anywhere in any MPEG4. EXPECT_TRUE(metadata.CanSeek()); EXPECT_FALSE(metadata.Crypto().valid); }