418 lines
15 KiB
C++
418 lines
15 KiB
C++
// 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 "base/basictypes.h"
|
|
#include "base/logging.h"
|
|
#include "base/memory/scoped_ptr.h"
|
|
#include "media/mp4/box_definitions.h"
|
|
#include "media/mp4/rcheck.h"
|
|
#include "media/mp4/track_run_iterator.h"
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
// The sum of the elements in a vector initialized with SumAscending,
|
|
// less the value of the last element.
|
|
static const int kSumAscending1 = 45;
|
|
|
|
static const int kAudioScale = 48000;
|
|
static const int kVideoScale = 25;
|
|
|
|
static const uint8 kAuxInfo[] = {
|
|
// Sample 1: IV (no subsumples).
|
|
0x41, 0x54, 0x65, 0x73, 0x74, 0x49, 0x76, 0x31,
|
|
// Sample 2: IV.
|
|
0x41, 0x54, 0x65, 0x73, 0x74, 0x49, 0x76, 0x32,
|
|
// Sample 2: Subsample count.
|
|
0x00, 0x02,
|
|
// Sample 2: Subsample 1.
|
|
0x00, 0x01, 0x00, 0x00, 0x00, 0x02,
|
|
// Sample 2: Subsample 2.
|
|
0x00, 0x03, 0x00, 0x00, 0x00, 0x04};
|
|
|
|
static const char kIv1[] = {0x41, 0x54, 0x65, 0x73, 0x74, 0x49, 0x76, 0x31, };
|
|
|
|
static const uint8 kKeyId[] = {0x41, 0x47, 0x6f, 0x6f, 0x67, 0x6c, 0x65, 0x54,
|
|
0x65, 0x73, 0x74, 0x4b, 0x65, 0x79, 0x49, 0x44};
|
|
|
|
namespace media {
|
|
namespace mp4 {
|
|
|
|
class TrackRunIteratorTest : public testing::Test {
|
|
public:
|
|
TrackRunIteratorTest() { CreateMovie(); }
|
|
|
|
protected:
|
|
Movie moov_;
|
|
scoped_ptr<TrackRunIterator> iter_;
|
|
|
|
void CreateMovie() {
|
|
moov_.header.timescale = 1000;
|
|
moov_.tracks.resize(3);
|
|
moov_.extends.tracks.resize(2);
|
|
moov_.tracks[0].header.track_id = 1;
|
|
moov_.tracks[0].media.header.timescale = kAudioScale;
|
|
SampleDescription& desc1 =
|
|
moov_.tracks[0].media.information.sample_table.description;
|
|
AudioSampleEntry aud_desc;
|
|
aud_desc.format = FOURCC_MP4A;
|
|
aud_desc.sinf.info.track_encryption.is_encrypted = false;
|
|
desc1.type = kAudio;
|
|
desc1.audio_entries.push_back(aud_desc);
|
|
moov_.extends.tracks[0].track_id = 1;
|
|
moov_.extends.tracks[0].default_sample_description_index = 1;
|
|
|
|
moov_.tracks[1].header.track_id = 2;
|
|
moov_.tracks[1].media.header.timescale = kVideoScale;
|
|
SampleDescription& desc2 =
|
|
moov_.tracks[1].media.information.sample_table.description;
|
|
VideoSampleEntry vid_desc;
|
|
vid_desc.format = FOURCC_AVC1;
|
|
vid_desc.sinf.info.track_encryption.is_encrypted = false;
|
|
desc2.type = kVideo;
|
|
desc2.video_entries.push_back(vid_desc);
|
|
moov_.extends.tracks[1].track_id = 2;
|
|
moov_.extends.tracks[1].default_sample_description_index = 1;
|
|
|
|
moov_.tracks[2].header.track_id = 3;
|
|
moov_.tracks[2].media.information.sample_table.description.type = kHint;
|
|
}
|
|
|
|
MovieFragment CreateFragment() {
|
|
MovieFragment moof;
|
|
moof.tracks.resize(2);
|
|
moof.tracks[0].decode_time.decode_time = 0;
|
|
moof.tracks[0].header.track_id = 1;
|
|
moof.tracks[0].header.flags = kDefaultSampleFlagsPresentMask;
|
|
moof.tracks[0].header.default_sample_duration = 1024;
|
|
moof.tracks[0].header.default_sample_size = 4;
|
|
moof.tracks[0].runs.resize(2);
|
|
moof.tracks[0].runs[0].sample_count = 10;
|
|
moof.tracks[0].runs[0].data_offset = 100;
|
|
SetAscending(&moof.tracks[0].runs[0].sample_sizes);
|
|
|
|
moof.tracks[0].runs[1].sample_count = 10;
|
|
moof.tracks[0].runs[1].data_offset = 10000;
|
|
|
|
moof.tracks[1].header.track_id = 2;
|
|
moof.tracks[1].header.flags = 0;
|
|
moof.tracks[1].decode_time.decode_time = 10;
|
|
moof.tracks[1].runs.resize(1);
|
|
moof.tracks[1].runs[0].sample_count = 10;
|
|
moof.tracks[1].runs[0].data_offset = 200;
|
|
SetAscending(&moof.tracks[1].runs[0].sample_sizes);
|
|
SetAscending(&moof.tracks[1].runs[0].sample_durations);
|
|
moof.tracks[1].runs[0].sample_flags.resize(10);
|
|
for (size_t i = 1; i < moof.tracks[1].runs[0].sample_flags.size(); i++) {
|
|
moof.tracks[1].runs[0].sample_flags[i] = kNonKeySampleMask;
|
|
}
|
|
|
|
return moof;
|
|
}
|
|
|
|
// Update the first sample description of a Track to indicate encryption
|
|
void AddEncryption(Track* track) {
|
|
SampleDescription* stsd =
|
|
&track->media.information.sample_table.description;
|
|
ProtectionSchemeInfo* sinf;
|
|
if (!stsd->video_entries.empty()) {
|
|
sinf = &stsd->video_entries[0].sinf;
|
|
} else {
|
|
sinf = &stsd->audio_entries[0].sinf;
|
|
}
|
|
|
|
sinf->type.type = FOURCC_CENC;
|
|
sinf->info.track_encryption.is_encrypted = true;
|
|
sinf->info.track_encryption.default_iv_size = 8;
|
|
sinf->info.track_encryption.default_kid.assign(kKeyId,
|
|
kKeyId + arraysize(kKeyId));
|
|
}
|
|
|
|
// Add aux info covering the first track run to a TrackFragment, and update
|
|
// the run to ensure it matches length and subsample information.
|
|
void AddAuxInfoHeaders(int offset, TrackFragment* frag) {
|
|
frag->auxiliary_offset.offsets.push_back(offset);
|
|
frag->auxiliary_size.sample_count = 2;
|
|
frag->auxiliary_size.sample_info_sizes.push_back(8);
|
|
frag->auxiliary_size.sample_info_sizes.push_back(22);
|
|
frag->runs[0].sample_count = 2;
|
|
frag->runs[0].sample_sizes[1] = 10;
|
|
}
|
|
|
|
void SetAscending(std::vector<uint32>* vec) {
|
|
vec->resize(10);
|
|
for (size_t i = 0; i < vec->size(); i++)
|
|
(*vec)[i] = i + 1;
|
|
}
|
|
};
|
|
|
|
TEST_F(TrackRunIteratorTest, NoRunsTest) {
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
ASSERT_TRUE(iter_->Init(MovieFragment()));
|
|
EXPECT_FALSE(iter_->IsRunValid());
|
|
EXPECT_FALSE(iter_->IsSampleValid());
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, BasicOperationTest) {
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
MovieFragment moof = CreateFragment();
|
|
|
|
// Test that runs are sorted correctly, and that properties of the initial
|
|
// sample of the first run are correct
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
EXPECT_TRUE(iter_->IsRunValid());
|
|
EXPECT_FALSE(iter_->is_encrypted());
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->sample_offset(), 100);
|
|
EXPECT_EQ(iter_->sample_size(), 1);
|
|
EXPECT_EQ(iter_->dts(), 0);
|
|
EXPECT_EQ(iter_->cts(), 0);
|
|
EXPECT_EQ(iter_->duration(), 1024);
|
|
EXPECT_TRUE(iter_->is_keyframe());
|
|
|
|
// Advance to the last sample in the current run, and test its properties
|
|
for (int i = 0; i < 9; i++)
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->sample_offset(), 100 + kSumAscending1);
|
|
EXPECT_EQ(iter_->sample_size(), 10);
|
|
EXPECT_EQ(iter_->dts(), 1024 * 9);
|
|
EXPECT_EQ(iter_->duration(), 1024);
|
|
EXPECT_TRUE(iter_->is_keyframe());
|
|
|
|
// Test end-of-run
|
|
iter_->AdvanceSample();
|
|
EXPECT_FALSE(iter_->IsSampleValid());
|
|
|
|
// Test last sample of next run
|
|
iter_->AdvanceRun();
|
|
EXPECT_TRUE(iter_->is_keyframe());
|
|
for (int i = 0; i < 9; i++)
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->track_id(), 2u);
|
|
EXPECT_EQ(iter_->sample_offset(), 200 + kSumAscending1);
|
|
EXPECT_EQ(iter_->sample_size(), 10);
|
|
int64 base_dts = kSumAscending1 + moof.tracks[1].decode_time.decode_time;
|
|
EXPECT_EQ(iter_->dts(), base_dts);
|
|
EXPECT_EQ(iter_->duration(), 10);
|
|
EXPECT_FALSE(iter_->is_keyframe());
|
|
|
|
// Test final run
|
|
iter_->AdvanceRun();
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->dts(), 1024 * 10);
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(moof.tracks[0].runs[1].data_offset +
|
|
moof.tracks[0].header.default_sample_size,
|
|
iter_->sample_offset());
|
|
iter_->AdvanceRun();
|
|
EXPECT_FALSE(iter_->IsRunValid());
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, TrackExtendsDefaultsTest) {
|
|
moov_.extends.tracks[0].default_sample_duration = 50;
|
|
moov_.extends.tracks[0].default_sample_size = 3;
|
|
moov_.extends.tracks[0].default_sample_flags = kNonKeySampleMask;
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
MovieFragment moof = CreateFragment();
|
|
moof.tracks[0].header.flags = 0;
|
|
moof.tracks[0].header.default_sample_size = 0;
|
|
moof.tracks[0].header.default_sample_duration = 0;
|
|
moof.tracks[0].runs[0].sample_sizes.clear();
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
iter_->AdvanceSample();
|
|
EXPECT_FALSE(iter_->is_keyframe());
|
|
EXPECT_EQ(iter_->sample_size(), 3);
|
|
EXPECT_EQ(iter_->sample_offset(), moof.tracks[0].runs[0].data_offset + 3);
|
|
EXPECT_EQ(iter_->duration(), 50);
|
|
EXPECT_EQ(iter_->dts(), 50);
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, FirstSampleFlagTest) {
|
|
// Ensure that keyframes are flagged correctly in the face of BMFF boxes which
|
|
// explicitly specify the flags for the first sample in a run and rely on
|
|
// defaults for all subsequent samples
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
MovieFragment moof = CreateFragment();
|
|
moof.tracks[1].header.flags = kDefaultSampleFlagsPresentMask;
|
|
moof.tracks[1].header.default_sample_flags = kNonKeySampleMask;
|
|
moof.tracks[1].runs[0].sample_flags.resize(1);
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
iter_->AdvanceRun();
|
|
EXPECT_TRUE(iter_->is_keyframe());
|
|
iter_->AdvanceSample();
|
|
EXPECT_FALSE(iter_->is_keyframe());
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, ReorderingTest) {
|
|
// Test frame reordering. The frames have the following
|
|
// decode timestamps:
|
|
//
|
|
// 0ms 40ms 120ms 240ms
|
|
// | 0 | 1 - | 2 - - |
|
|
//
|
|
// ...and these composition timestamps, after edit list adjustment:
|
|
//
|
|
// 0ms 40ms 160ms 240ms
|
|
// | 0 | 2 - - | 1 - |
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
|
|
// Add CTS offsets. Without bias, the CTS offsets for the first three frames
|
|
// would simply be [0, 3, -2]. Since CTS offsets should be non-negative for
|
|
// maximum compatibility, these values are biased up to [2, 5, 0].
|
|
MovieFragment moof = CreateFragment();
|
|
std::vector<int32>& cts_offsets =
|
|
moof.tracks[1].runs[0].sample_composition_time_offsets;
|
|
cts_offsets.resize(10);
|
|
cts_offsets[0] = 2;
|
|
cts_offsets[1] = 5;
|
|
cts_offsets[2] = 0;
|
|
moof.tracks[1].decode_time.decode_time = 0;
|
|
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
iter_->AdvanceRun();
|
|
EXPECT_EQ(iter_->dts(), 0);
|
|
EXPECT_EQ(iter_->cts(), 2);
|
|
EXPECT_EQ(iter_->duration(), 1);
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->dts(), 1);
|
|
EXPECT_EQ(iter_->cts(), 6);
|
|
EXPECT_EQ(iter_->duration(), 2);
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->dts(), 3);
|
|
EXPECT_EQ(iter_->cts(), 3);
|
|
EXPECT_EQ(iter_->duration(), 3);
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, IgnoreUnknownAuxInfoTest) {
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
MovieFragment moof = CreateFragment();
|
|
moof.tracks[1].auxiliary_offset.offsets.push_back(50);
|
|
moof.tracks[1].auxiliary_size.default_sample_info_size = 2;
|
|
moof.tracks[1].auxiliary_size.sample_count = 2;
|
|
moof.tracks[1].runs[0].sample_count = 2;
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
iter_->AdvanceRun();
|
|
EXPECT_FALSE(iter_->AuxInfoNeedsToBeCached());
|
|
}
|
|
|
|
TEST_F(TrackRunIteratorTest, DecryptConfigTest) {
|
|
AddEncryption(&moov_.tracks[1]);
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
|
|
MovieFragment moof = CreateFragment();
|
|
AddAuxInfoHeaders(50, &moof.tracks[1]);
|
|
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
|
|
// The run for track 2 will be first, since its aux info offset is the first
|
|
// element in the file.
|
|
EXPECT_EQ(iter_->track_id(), 2u);
|
|
EXPECT_TRUE(iter_->is_encrypted());
|
|
EXPECT_TRUE(iter_->AuxInfoNeedsToBeCached());
|
|
EXPECT_EQ(static_cast<uint32>(iter_->aux_info_size()), arraysize(kAuxInfo));
|
|
EXPECT_EQ(iter_->aux_info_offset(), 50);
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 50);
|
|
EXPECT_FALSE(iter_->CacheAuxInfo(NULL, 0));
|
|
EXPECT_FALSE(iter_->CacheAuxInfo(kAuxInfo, 3));
|
|
EXPECT_TRUE(iter_->AuxInfoNeedsToBeCached());
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
EXPECT_FALSE(iter_->AuxInfoNeedsToBeCached());
|
|
EXPECT_EQ(iter_->sample_offset(), 200);
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), moof.tracks[0].runs[0].data_offset);
|
|
scoped_ptr<DecryptConfig> config = iter_->GetDecryptConfig();
|
|
ASSERT_EQ(arraysize(kKeyId), config->key_id().size());
|
|
EXPECT_TRUE(
|
|
!memcmp(kKeyId, config->key_id().data(), config->key_id().size()));
|
|
ASSERT_EQ(arraysize(kIv1), config->iv().size());
|
|
EXPECT_TRUE(!memcmp(kIv1, config->iv().data(), config->iv().size()));
|
|
EXPECT_TRUE(config->subsamples().empty());
|
|
iter_->AdvanceSample();
|
|
config = iter_->GetDecryptConfig();
|
|
EXPECT_EQ(config->subsamples().size(), 2u);
|
|
EXPECT_EQ(config->subsamples()[0].clear_bytes, 1u);
|
|
EXPECT_EQ(config->subsamples()[1].cypher_bytes, 4u);
|
|
}
|
|
|
|
// It is legal for aux info blocks to be shared among multiple formats.
|
|
TEST_F(TrackRunIteratorTest, SharedAuxInfoTest) {
|
|
AddEncryption(&moov_.tracks[0]);
|
|
AddEncryption(&moov_.tracks[1]);
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
|
|
MovieFragment moof = CreateFragment();
|
|
moof.tracks[0].runs.resize(1);
|
|
AddAuxInfoHeaders(50, &moof.tracks[0]);
|
|
AddAuxInfoHeaders(50, &moof.tracks[1]);
|
|
moof.tracks[0].auxiliary_size.default_sample_info_size = 8;
|
|
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->aux_info_offset(), 50);
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
scoped_ptr<DecryptConfig> config = iter_->GetDecryptConfig();
|
|
ASSERT_EQ(arraysize(kIv1), config->iv().size());
|
|
EXPECT_TRUE(!memcmp(kIv1, config->iv().data(), config->iv().size()));
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 50);
|
|
iter_->AdvanceRun();
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 50);
|
|
EXPECT_EQ(iter_->aux_info_offset(), 50);
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 200);
|
|
ASSERT_EQ(arraysize(kIv1), config->iv().size());
|
|
EXPECT_TRUE(!memcmp(kIv1, config->iv().data(), config->iv().size()));
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 201);
|
|
}
|
|
|
|
// Sensible files are expected to place auxiliary information for a run
|
|
// immediately before the main data for that run. Alternative schemes are
|
|
// possible, however, including the somewhat reasonable behavior of placing all
|
|
// aux info at the head of the 'mdat' box together, and the completely
|
|
// unreasonable behavior demonstrated here:
|
|
// byte 50: track 2, run 1 aux info
|
|
// byte 100: track 1, run 1 data
|
|
// byte 200: track 2, run 1 data
|
|
// byte 201: track 1, run 2 aux info (*inside* track 2, run 1 data)
|
|
// byte 10000: track 1, run 2 data
|
|
// byte 20000: track 1, run 1 aux info
|
|
TEST_F(TrackRunIteratorTest, UnexpectedOrderingTest) {
|
|
AddEncryption(&moov_.tracks[0]);
|
|
AddEncryption(&moov_.tracks[1]);
|
|
iter_.reset(new TrackRunIterator(&moov_));
|
|
|
|
MovieFragment moof = CreateFragment();
|
|
AddAuxInfoHeaders(20000, &moof.tracks[0]);
|
|
moof.tracks[0].auxiliary_offset.offsets.push_back(201);
|
|
moof.tracks[0].auxiliary_size.sample_count += 2;
|
|
moof.tracks[0].auxiliary_size.default_sample_info_size = 8;
|
|
moof.tracks[0].runs[1].sample_count = 2;
|
|
AddAuxInfoHeaders(50, &moof.tracks[1]);
|
|
moof.tracks[1].runs[0].sample_sizes[0] = 5;
|
|
|
|
ASSERT_TRUE(iter_->Init(moof));
|
|
EXPECT_EQ(iter_->track_id(), 2u);
|
|
EXPECT_EQ(iter_->aux_info_offset(), 50);
|
|
EXPECT_EQ(iter_->sample_offset(), 200);
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 100);
|
|
iter_->AdvanceRun();
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->aux_info_offset(), 20000);
|
|
EXPECT_EQ(iter_->sample_offset(), 100);
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 100);
|
|
iter_->AdvanceSample();
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 101);
|
|
iter_->AdvanceRun();
|
|
EXPECT_EQ(iter_->track_id(), 1u);
|
|
EXPECT_EQ(iter_->aux_info_offset(), 201);
|
|
EXPECT_EQ(iter_->sample_offset(), 10000);
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 201);
|
|
EXPECT_TRUE(iter_->CacheAuxInfo(kAuxInfo, arraysize(kAuxInfo)));
|
|
EXPECT_EQ(iter_->GetMaxClearOffset(), 10000);
|
|
}
|
|
|
|
} // namespace mp4
|
|
} // namespace media
|