910e62b5创建于 1月15日历史提交
// Copyright 2015 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "media/cdm/cenc_utils.h"

#include <stddef.h>
#include <stdint.h>

#include <limits>

#include "base/check.h"
#include "testing/gtest/include/gtest/gtest.h"

namespace media {

namespace {

// clang-format off
constexpr auto kKey1 = std::to_array<uint8_t>({
    0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,
    0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03
});

constexpr auto kKey2 = std::to_array<uint8_t>({
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,
});

constexpr auto kKey3 = std::to_array<uint8_t>({
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x05,
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x05,
});

constexpr auto kKey4 = std::to_array<uint8_t>({
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x06,
    0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x06,
});

constexpr auto kCommonSystemSystemId = std::to_array<uint8_t>({
    0x10, 0x77, 0xEF, 0xEC, 0xC0, 0xB2, 0x4D, 0x02,
    0xAC, 0xE3, 0x3C, 0x1E, 0x52, 0xE2, 0xFB, 0x4B
});
// clang-format on

// Initialize the start of the 'pssh' box (up to key_count)
void InitializePSSHBox(std::vector<uint8_t>* box,
                       uint8_t size,
                       uint8_t version) {
  DCHECK(box->size() == 0);

  box->reserve(size);
  // Add size.
  DCHECK(size < std::numeric_limits<uint8_t>::max());
  box->push_back(0);
  box->push_back(0);
  box->push_back(0);
  box->push_back(size);
  // Add 'pssh'.
  box->push_back('p');
  box->push_back('s');
  box->push_back('s');
  box->push_back('h');
  // Add version.
  box->push_back(version);
  // Add flags.
  box->push_back(0);
  box->push_back(0);
  box->push_back(0);
  // Add Common Encryption SystemID.
  box->insert(box->end(), kCommonSystemSystemId.begin(),
              kCommonSystemSystemId.end());
}

std::vector<uint8_t> MakePSSHBox(uint8_t version) {
  std::vector<uint8_t> box;
  uint8_t size = (version == 0) ? 32 : 36;
  InitializePSSHBox(&box, size, version);
  if (version > 0) {
    // Add key_count (= 0).
    box.push_back(0);
    box.push_back(0);
    box.push_back(0);
    box.push_back(0);
  }
  // Add data_size (= 0).
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  return box;
}

std::vector<uint8_t> MakePSSHBox(uint8_t version,
                                 base::span<const uint8_t> key1) {
  DCHECK(version > 0);
  DCHECK(key1.size() == 16);

  std::vector<uint8_t> box;
  uint8_t size = 52;
  InitializePSSHBox(&box, size, version);

  // Add key_count (= 1).
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  box.push_back(1);

  // Add key1.
  for (size_t i = 0; i < key1.size(); ++i) {
    box.push_back(key1[i]);
  }

  // Add data_size (= 0).
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  return box;
}

std::vector<uint8_t> MakePSSHBox(uint8_t version,
                                 base::span<const uint8_t> key1,
                                 base::span<const uint8_t> key2) {
  DCHECK(version > 0);
  DCHECK(key1.size() == 16);
  DCHECK(key2.size() == 16);

  std::vector<uint8_t> box;
  uint8_t size = 68;
  InitializePSSHBox(&box, size, version);

  // Add key_count (= 2).
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  box.push_back(2);

  // Add key1.
  for (size_t i = 0; i < key1.size(); ++i) {
    box.push_back(key1[i]);
  }

  // Add key2.
  for (size_t i = 0; i < key2.size(); ++i) {
    box.push_back(key2[i]);
  }

  // Add data_size (= 0).
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  box.push_back(0);
  return box;
}

void AppendData(std::vector<uint8_t>& pssh_box,
                base::span<const uint8_t> data) {
  // This assumes that |pssh_box| has been created using the routines above,
  // and simply appends the data to the end of it. It updates the box size
  // and sets the data size.
  DCHECK(data.size() < 100);
  pssh_box[3] += static_cast<uint8_t>(data.size());
  pssh_box.pop_back();
  pssh_box.push_back(static_cast<uint8_t>(data.size()));
  pssh_box.insert(pssh_box.end(), data.begin(), data.end());
}

TEST(CencUtilsTest, EmptyPSSH) {
  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(std::vector<uint8_t>()));
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(std::vector<uint8_t>(), &key_ids));
}

TEST(CencUtilsTest, PSSHVersion0) {
  std::vector<uint8_t> box = MakePSSHBox(0);
  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box));
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(box, &key_ids));
}

TEST(CencUtilsTest, PSSHVersion1WithNoKeys) {
  std::vector<uint8_t> box = MakePSSHBox(1);
  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box));
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(box, &key_ids));
}

TEST(CencUtilsTest, PSSHVersion1WithOneKey) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1);
  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(box, &key_ids));
  EXPECT_EQ(1u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey1);
}

TEST(CencUtilsTest, PSSHVersion1WithTwoKeys) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);
  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(box, &key_ids));
  EXPECT_EQ(2u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey1);
  EXPECT_EQ(base::as_byte_span(key_ids[1]), kKey2);
}

TEST(CencUtilsTest, PSSHVersion0Plus1) {
  std::vector<uint8_t> box0 = MakePSSHBox(0);
  std::vector<uint8_t> box1 = MakePSSHBox(1, kKey1);

  // Concatenate box1 onto end of box0.
  box0.insert(box0.end(), box1.begin(), box1.end());
  EXPECT_TRUE(ValidatePsshInput(box0));

  // No key IDs returned as only the first 'pssh' box is processed.
  KeyIdList key_ids;
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(box0, &key_ids));
}

TEST(CencUtilsTest, PSSHVersion1Plus0) {
  std::vector<uint8_t> box0 = MakePSSHBox(0);
  std::vector<uint8_t> box1 = MakePSSHBox(1, kKey1);

  // Concatenate box0 onto end of box1.
  box1.insert(box1.end(), box0.begin(), box0.end());

  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box1));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(box1, &key_ids));
  EXPECT_EQ(1u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey1);
}

TEST(CencUtilsTest, MultiplePSSHVersion1) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);
  std::vector<uint8_t> box1 = MakePSSHBox(1, kKey3);
  std::vector<uint8_t> box2 = MakePSSHBox(1, kKey4);

  // Concatenate box1 and box2 onto end of box.
  box.insert(box.end(), box1.begin(), box1.end());
  box.insert(box.end(), box2.begin(), box2.end());

  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(box));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(box, &key_ids));
  EXPECT_EQ(2u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey1);
  EXPECT_EQ(base::as_byte_span(key_ids[1]), kKey2);
}

TEST(CencUtilsTest, PsshBoxSmallerThanSize) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);
  KeyIdList key_ids;

  // Tries every buffer size less than the indicated 'pssh' box size.
  for (size_t i = 1; i < box.size(); ++i) {
    auto truncated = base::span(box).first(i);
    EXPECT_FALSE(ValidatePsshInput(truncated)) << "Failed for length " << i;
    EXPECT_FALSE(GetKeyIdsForCommonSystemId(truncated, &key_ids));
  }
}

TEST(CencUtilsTest, PsshBoxLargerThanSize) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);
  KeyIdList key_ids;

  // Add 20 additional bytes to |box|.
  size_t original_size = box.size();
  for (size_t i = 0; i < 20; ++i)
    box.push_back(static_cast<uint8_t>(i));

  // Tries every size greater than |original_size|.
  for (size_t i = original_size + 1; i < box.size(); ++i) {
    auto truncated = base::span(box).first(i);
    EXPECT_FALSE(ValidatePsshInput(truncated)) << "Failed for length " << i;
    EXPECT_FALSE(GetKeyIdsForCommonSystemId(truncated, &key_ids));
  }
}

TEST(CencUtilsTest, UnrecognizedSystemID) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);

  // Modify the System ID.
  ++box[20];

  KeyIdList key_ids;
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(box, &key_ids));
}

TEST(CencUtilsTest, InvalidFlags) {
  std::vector<uint8_t> box = MakePSSHBox(1, kKey1, kKey2);

  // Modify flags.
  box[10] = 3;

  KeyIdList key_ids;
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(box, &key_ids));
}

TEST(CencUtilsTest, LongSize) {
  constexpr auto data = std::to_array<uint8_t>({
      0x00, 0x00, 0x00, 0x01,                          // size = 1
      0x70, 0x73, 0x73, 0x68,                          // 'pssh'
      0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x4c,  // longsize
      0x01,                                            // version
      0x00, 0x00, 0x00,                                // flags
      0x10, 0x77, 0xEF, 0xEC, 0xC0, 0xB2, 0x4D, 0x02,  // SystemID
      0xAC, 0xE3, 0x3C, 0x1E, 0x52, 0xE2, 0xFB, 0x4B,
      0x00, 0x00, 0x00, 0x02,                          // key count
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,  // key1
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,  // key2
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,
      0x00, 0x00, 0x00, 0x00  // datasize
  });

  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(data));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(data, &key_ids));
  EXPECT_EQ(2u, key_ids.size());
}

TEST(CencUtilsTest, SizeIsZero) {
  constexpr auto data = std::to_array<uint8_t>({
      0x00, 0x00, 0x00, 0x00,                          // size = 0
      0x70, 0x73, 0x73, 0x68,                          // 'pssh'
      0x01,                                            // version
      0x00, 0x00, 0x00,                                // flags
      0x10, 0x77, 0xEF, 0xEC, 0xC0, 0xB2, 0x4D, 0x02,  // SystemID
      0xAC, 0xE3, 0x3C, 0x1E, 0x52, 0xE2, 0xFB, 0x4B,
      0x00, 0x00, 0x00, 0x02,                          // key count
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,  // key1
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,  // key2
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,
      0x00, 0x00, 0x00, 0x00  // datasize
  });

  KeyIdList key_ids;
  EXPECT_TRUE(ValidatePsshInput(data));
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(data, &key_ids));
  EXPECT_EQ(2u, key_ids.size());
}

TEST(CencUtilsTest, HugeSize) {
  constexpr auto data = std::to_array<uint8_t>({
      0x00, 0x00, 0x00, 0x01,                          // size = 1
      0x70, 0x73, 0x73, 0x68,                          // 'pssh'
      0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,  // longsize = big
      0x01,                                            // version
      0x00, 0x00, 0x00,                                // flags
      0x10, 0x77, 0xEF, 0xEC, 0xC0, 0xB2, 0x4D, 0x02,  // SystemID
      0xAC, 0xE3, 0x3C, 0x1E, 0x52, 0xE2, 0xFB, 0x4B,
      0x00, 0x00, 0x00, 0x02,                          // key count
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,  // key1
      0x7E, 0x57, 0x1D, 0x03, 0x7E, 0x57, 0x1D, 0x03,
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,  // key2
      0x7E, 0x57, 0x1D, 0x04, 0x7E, 0x57, 0x1D, 0x04,
      0x00, 0x00, 0x00, 0x00  // datasize
  });

  KeyIdList key_ids;
  // These calls fail as the box size is huge (0xffffffffffffffff) and there
  // is not enough bytes in |data|.
  EXPECT_FALSE(ValidatePsshInput(data));
  EXPECT_FALSE(GetKeyIdsForCommonSystemId(data, &key_ids));
}

TEST(CencUtilsTest, GetPsshData_Version0) {
  constexpr auto data = std::to_array<uint8_t>({0x01, 0x02, 0x03, 0x04});
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(0);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(0u, pssh_data.size());

  AppendData(box, data);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(data, base::as_byte_span(pssh_data));
}

TEST(CencUtilsTest, GetPsshData_Version1NoKeys) {
  constexpr auto data = std::to_array<uint8_t>({0x05, 0x06, 0x07, 0x08});
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(1);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(0u, pssh_data.size());

  AppendData(box, data);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(data, base::as_byte_span(pssh_data));
}

TEST(CencUtilsTest, GetPsshData_Version1WithKeys) {
  constexpr auto data =
      std::to_array<uint8_t>({0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08});
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(1, kKey1);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(0u, pssh_data.size());

  AppendData(box, data);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(data, base::as_byte_span(pssh_data));
}

TEST(CencUtilsTest, GetPsshData_Version2) {
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(1, kKey1);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));

  // Change the version manually, since we don't know what v2 will contain.
  box[8] = 2;
  EXPECT_FALSE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
}

TEST(CencUtilsTest, GetPsshData_Version2ThenVersion1) {
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box_v1 = MakePSSHBox(1, kKey1);
  std::vector<uint8_t> box_v2 = MakePSSHBox(2, kKey2, kKey3);

  // Concatenate the boxes together (v2 first).
  std::vector<uint8_t> boxes;
  boxes.insert(boxes.end(), box_v2.begin(), box_v2.end());
  boxes.insert(boxes.end(), box_v1.begin(), box_v1.end());
  EXPECT_TRUE(GetPsshData(boxes, kCommonSystemSystemId, &pssh_data));

  // GetKeyIdsForCommonSystemId() should return the single key from the v1
  // 'pssh' box.
  KeyIdList key_ids;
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(boxes, &key_ids));
  EXPECT_EQ(1u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey1);
}

TEST(CencUtilsTest, GetPsshData_Version1ThenVersion2) {
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box_v1 = MakePSSHBox(1, kKey3);
  std::vector<uint8_t> box_v2 = MakePSSHBox(2, kKey4);

  // Concatenate the boxes together (v1 first).
  std::vector<uint8_t> boxes;
  boxes.insert(boxes.end(), box_v1.begin(), box_v1.end());
  boxes.insert(boxes.end(), box_v2.begin(), box_v2.end());
  EXPECT_TRUE(GetPsshData(boxes, kCommonSystemSystemId, &pssh_data));

  // GetKeyIdsForCommonSystemId() should return the single key from the v1
  // 'pssh' box.
  KeyIdList key_ids;
  EXPECT_TRUE(GetKeyIdsForCommonSystemId(boxes, &key_ids));
  EXPECT_EQ(1u, key_ids.size());
  EXPECT_EQ(base::as_byte_span(key_ids[0]), kKey3);
}

TEST(CencUtilsTest, GetPsshData_DifferentSystemID) {
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(1, kKey1);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
  EXPECT_FALSE(GetPsshData(box, kKey1, &pssh_data));
}

TEST(CencUtilsTest, GetPsshData_MissingData) {
  constexpr auto data =
      std::to_array<uint8_t>({0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08});
  std::vector<uint8_t> pssh_data;

  std::vector<uint8_t> box = MakePSSHBox(1, kKey1);
  AppendData(box, data);
  EXPECT_TRUE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));

  // Remove some data from the end, so now the size is incorrect.
  box.pop_back();
  box.pop_back();
  EXPECT_FALSE(GetPsshData(box, kCommonSystemSystemId, &pssh_data));
}

TEST(CencUtilsTest, GetPsshData_MultiplePssh) {
  constexpr auto data1 =
      std::to_array<uint8_t>({0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07});
  constexpr auto data2 = std::to_array<uint8_t>({0xa1, 0xa2, 0xa3, 0xa4});

  std::vector<uint8_t> box1 = MakePSSHBox(1, kKey1);
  AppendData(box1, data1);
  std::vector<uint8_t> box2 = MakePSSHBox(0);
  AppendData(box2, data2);

  std::vector<uint8_t> pssh_data;
  box1.insert(box1.end(), box2.begin(), box2.end());
  EXPECT_TRUE(GetPsshData(box1, kCommonSystemSystemId, &pssh_data));
  EXPECT_EQ(data1, base::as_byte_span(pssh_data));
  EXPECT_NE(data2, base::as_byte_span(pssh_data));
}

TEST(CencUtilsTest, NonPsshData) {
  // Create a non-'pssh' box.
  constexpr auto data =
      std::to_array<uint8_t>({0x00, 0x00, 0x00, 0x08,  // size = 8
                              'p', 's', 's', 'g'});
  EXPECT_FALSE(ValidatePsshInput(data));

  // Make a valid 'pssh' box.
  std::vector<uint8_t> pssh_box = MakePSSHBox(1, kKey1);
  EXPECT_TRUE(ValidatePsshInput(pssh_box));

  // Concatenate the boxes together (|pssh_box| first).
  std::vector<uint8_t> boxes;
  boxes.insert(boxes.end(), pssh_box.begin(), pssh_box.end());
  boxes.insert(boxes.end(), data.begin(), data.end());
  EXPECT_FALSE(ValidatePsshInput(boxes));

  // Repeat with |non_pssh_box| first.
  boxes.clear();
  boxes.insert(boxes.end(), data.begin(), data.end());
  boxes.insert(boxes.end(), pssh_box.begin(), pssh_box.end());
  EXPECT_FALSE(ValidatePsshInput(boxes));
}

}  // namespace

}  // namespace media