// Copyright 2014 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 #include "base/basictypes.h" #include "base/logging.h" #include "base/memory/scoped_ptr.h" #include "media/formats/mp4/box_reader.h" #include "media/formats/mp4/rcheck.h" #include "testing/gtest/include/gtest/gtest.h" namespace media { namespace mp4 { static const uint8 kSkipBox[] = { // Top-level test box containing three children 0x00, 0x00, 0x00, 0x40, 's', 'k', 'i', 'p', 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0xf9, 0x0a, 0x0b, 0x0c, 0xfd, 0x0e, 0x0f, 0x10, // Ordinary (8-byte header) child box 0x00, 0x00, 0x00, 0x0c, 'p', 's', 's', 'h', 0xde, 0xad, 0xbe, 0xef, // Extended-size header child box 0x00, 0x00, 0x00, 0x01, 'p', 's', 's', 'h', 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x14, 0xfa, 0xce, 0xca, 0xfe, // Empty free box 0x00, 0x00, 0x00, 0x08, 'f', 'r', 'e', 'e', // Trailing garbage 0x00 }; struct FreeBox : Box { virtual bool Parse(BoxReader* reader) OVERRIDE { return true; } virtual FourCC BoxType() const OVERRIDE { return FOURCC_FREE; } }; struct PsshBox : Box { uint32 val; virtual bool Parse(BoxReader* reader) OVERRIDE { return reader->Read4(&val); } virtual FourCC BoxType() const OVERRIDE { return FOURCC_PSSH; } }; struct SkipBox : Box { uint8 a, b; uint16 c; int32 d; int64 e; std::vector kids; FreeBox mpty; virtual bool Parse(BoxReader* reader) OVERRIDE { RCHECK(reader->ReadFullBoxHeader() && reader->Read1(&a) && reader->Read1(&b) && reader->Read2(&c) && reader->Read4s(&d) && reader->Read4sInto8s(&e)); return reader->ScanChildren() && reader->ReadChildren(&kids) && reader->MaybeReadChild(&mpty); } virtual FourCC BoxType() const OVERRIDE { return FOURCC_SKIP; } SkipBox(); virtual ~SkipBox(); }; SkipBox::SkipBox() {} SkipBox::~SkipBox() {} class BoxReaderTest : public testing::Test { protected: std::vector GetBuf() { return std::vector(kSkipBox, kSkipBox + sizeof(kSkipBox)); } }; TEST_F(BoxReaderTest, ExpectedOperationTest) { std::vector buf = GetBuf(); bool err; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); EXPECT_FALSE(err); EXPECT_TRUE(reader.get()); SkipBox box; EXPECT_TRUE(box.Parse(reader.get())); EXPECT_EQ(0x01, reader->version()); EXPECT_EQ(0x020304u, reader->flags()); EXPECT_EQ(0x05, box.a); EXPECT_EQ(0x06, box.b); EXPECT_EQ(0x0708, box.c); EXPECT_EQ(static_cast(0xf90a0b0c), box.d); EXPECT_EQ(static_cast(0xfd0e0f10), box.e); EXPECT_EQ(2u, box.kids.size()); EXPECT_EQ(0xdeadbeef, box.kids[0].val); EXPECT_EQ(0xfacecafe, box.kids[1].val); // Accounting for the extra byte outside of the box above EXPECT_EQ(buf.size(), static_cast(reader->size() + 1)); } TEST_F(BoxReaderTest, OuterTooShortTest) { std::vector buf = GetBuf(); bool err; // Create a soft failure by truncating the outer box. scoped_ptr r( BoxReader::ReadTopLevelBox(&buf[0], buf.size() - 2, LogCB(), &err)); EXPECT_FALSE(err); EXPECT_FALSE(r.get()); } TEST_F(BoxReaderTest, InnerTooLongTest) { std::vector buf = GetBuf(); bool err; // Make an inner box too big for its outer box. buf[25] = 1; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); SkipBox box; EXPECT_FALSE(box.Parse(reader.get())); } TEST_F(BoxReaderTest, WrongFourCCTest) { std::vector buf = GetBuf(); bool err; // Set an unrecognized top-level FourCC. buf[5] = 1; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); EXPECT_FALSE(reader.get()); EXPECT_TRUE(err); } TEST_F(BoxReaderTest, ScanChildrenTest) { std::vector buf = GetBuf(); bool err; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); EXPECT_TRUE(reader->SkipBytes(16) && reader->ScanChildren()); FreeBox free; EXPECT_TRUE(reader->ReadChild(&free)); EXPECT_FALSE(reader->ReadChild(&free)); EXPECT_TRUE(reader->MaybeReadChild(&free)); std::vector kids; EXPECT_TRUE(reader->ReadChildren(&kids)); EXPECT_EQ(2u, kids.size()); kids.clear(); EXPECT_FALSE(reader->ReadChildren(&kids)); EXPECT_TRUE(reader->MaybeReadChildren(&kids)); } TEST_F(BoxReaderTest, ReadAllChildrenTest) { std::vector buf = GetBuf(); // Modify buffer to exclude its last 'free' box buf[3] = 0x38; bool err; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); std::vector kids; EXPECT_TRUE(reader->SkipBytes(16) && reader->ReadAllChildren(&kids)); EXPECT_EQ(2u, kids.size()); EXPECT_EQ(kids[0].val, 0xdeadbeef); // Ensure order is preserved } static void TestTopLevelBox(const uint8* data, int size, uint32 fourCC) { std::vector buf(data, data + size); bool err; scoped_ptr reader( BoxReader::ReadTopLevelBox(&buf[0], buf.size(), LogCB(), &err)); EXPECT_FALSE(err); EXPECT_TRUE(reader); EXPECT_EQ(fourCC, reader->type()); EXPECT_EQ(reader->size(), size); } TEST_F(BoxReaderTest, SkippingBloc) { static const uint8 kData[] = { 0x00, 0x00, 0x00, 0x09, 'b', 'l', 'o', 'c', 0x00 }; TestTopLevelBox(kData, sizeof(kData), FOURCC_BLOC); } TEST_F(BoxReaderTest, SkippingEmsg) { static const uint8 kData[] = { 0x00, 0x00, 0x00, 0x24, 'e', 'm', 's', 'g', 0x00, // version = 0 0x00, 0x00, 0x00, // flags = 0 0x61, 0x00, // scheme_id_uri = "a" 0x61, 0x00, // value = "a" 0x00, 0x00, 0x00, 0x01, // timescale = 1 0x00, 0x00, 0x00, 0x02, // presentation_time_delta = 2 0x00, 0x00, 0x00, 0x03, // event_duration = 3 0x00, 0x00, 0x00, 0x04, // id = 4 0x05, 0x06, 0x07, 0x08, // message_data[4] = 0x05060708 }; TestTopLevelBox(kData, sizeof(kData), FOURCC_EMSG); } TEST_F(BoxReaderTest, SkippingUuid) { static const uint8 kData[] = { 0x00, 0x00, 0x00, 0x19, 'u', 'u', 'i', 'd', 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F, 0x10, // usertype 0x00, }; TestTopLevelBox(kData, sizeof(kData), FOURCC_UUID); } } // namespace mp4 } // namespace media