blob: 58a808840dbf42901e938908b282c0e9a329f8cb [file] [log] [blame]
/*
* Copyright (c) 2019 The WebRTC project authors. All Rights Reserved.
*
* Use of this source code is governed by a BSD-style license
* that can be found in the LICENSE file in the root of the source
* tree. An additional intellectual property rights grant can be found
* in the file PATENTS. All contributing project authors may
* be found in the AUTHORS file in the root of the source tree.
*/
#include "modules/video_coding/utility/ivf_file_reader.h"
#include "modules/video_coding/utility/ivf_file_writer.h"
#include <memory>
#include <string>
#include "test/gtest.h"
#include "test/testsupport/file_utils.h"
namespace webrtc {
namespace {
constexpr int kWidth = 320;
constexpr int kHeight = 240;
constexpr int kNumFrames = 3;
constexpr uint8_t kDummyPayload[4] = {'0', '1', '2', '3'};
} // namespace
class IvfFileReaderTest : public ::testing::Test {
protected:
void SetUp() override {
file_name_ =
webrtc::test::TempFilename(webrtc::test::OutputPath(), "test_file.ivf");
}
void TearDown() override { webrtc::test::RemoveFile(file_name_); }
bool WriteDummyTestFrames(IvfFileWriter* file_writer,
VideoCodecType codec_type,
int width,
int height,
int num_frames,
bool use_capture_tims_ms,
int spatial_layers_count) {
EncodedImage frame;
frame.SetSpatialIndex(spatial_layers_count);
rtc::scoped_refptr<EncodedImageBuffer> payload = EncodedImageBuffer::Create(
sizeof(kDummyPayload) * spatial_layers_count);
for (int i = 0; i < spatial_layers_count; ++i) {
memcpy(&payload->data()[i * sizeof(kDummyPayload)], kDummyPayload,
sizeof(kDummyPayload));
frame.SetSpatialLayerFrameSize(i, sizeof(kDummyPayload));
}
frame.SetEncodedData(payload);
frame._encodedWidth = width;
frame._encodedHeight = height;
for (int i = 1; i <= num_frames; ++i) {
if (use_capture_tims_ms) {
frame.capture_time_ms_ = i;
} else {
frame.SetTimestamp(i);
}
if (!file_writer->WriteFrame(frame, codec_type))
return false;
}
return true;
}
void CreateTestFile(VideoCodecType codec_type,
bool use_capture_tims_ms,
int spatial_layers_count) {
std::unique_ptr<IvfFileWriter> file_writer =
IvfFileWriter::Wrap(FileWrapper::OpenWriteOnly(file_name_), 0);
ASSERT_TRUE(file_writer.get());
ASSERT_TRUE(WriteDummyTestFrames(file_writer.get(), codec_type, kWidth,
kHeight, kNumFrames, use_capture_tims_ms,
spatial_layers_count));
ASSERT_TRUE(file_writer->Close());
}
void ValidateFrame(absl::optional<EncodedImage> frame,
int frame_index,
bool use_capture_tims_ms,
int spatial_layers_count) {
ASSERT_TRUE(frame);
EXPECT_EQ(frame->SpatialIndex(), spatial_layers_count);
if (use_capture_tims_ms) {
EXPECT_EQ(frame->capture_time_ms_, static_cast<int64_t>(frame_index));
EXPECT_EQ(frame->Timestamp(), static_cast<int64_t>(90 * frame_index));
} else {
EXPECT_EQ(frame->Timestamp(), static_cast<int64_t>(frame_index));
}
ASSERT_EQ(frame->size(), sizeof(kDummyPayload) * spatial_layers_count);
for (int i = 0; i < spatial_layers_count; ++i) {
EXPECT_EQ(memcmp(&frame->data()[i * sizeof(kDummyPayload)], kDummyPayload,
sizeof(kDummyPayload)),
0)
<< std::string(reinterpret_cast<char const*>(
&frame->data()[i * sizeof(kDummyPayload)]),
sizeof(kDummyPayload));
}
}
void ValidateContent(VideoCodecType codec_type,
bool use_capture_tims_ms,
int spatial_layers_count) {
std::unique_ptr<IvfFileReader> reader =
IvfFileReader::Create(FileWrapper::OpenReadOnly(file_name_));
ASSERT_TRUE(reader.get());
EXPECT_EQ(reader->GetVideoCodecType(), codec_type);
EXPECT_EQ(reader->GetFramesCount(),
spatial_layers_count * static_cast<size_t>(kNumFrames));
for (int i = 1; i <= kNumFrames; ++i) {
ASSERT_TRUE(reader->HasMoreFrames());
ValidateFrame(reader->NextFrame(), i, use_capture_tims_ms,
spatial_layers_count);
EXPECT_FALSE(reader->HasError());
}
EXPECT_FALSE(reader->HasMoreFrames());
EXPECT_FALSE(reader->NextFrame());
EXPECT_FALSE(reader->HasError());
ASSERT_TRUE(reader->Close());
}
std::string file_name_;
};
TEST_F(IvfFileReaderTest, BasicVp8FileNtpTimestamp) {
CreateTestFile(kVideoCodecVP8, false, 1);
ValidateContent(kVideoCodecVP8, false, 1);
}
TEST_F(IvfFileReaderTest, BasicVP8FileMsTimestamp) {
CreateTestFile(kVideoCodecVP8, true, 1);
ValidateContent(kVideoCodecVP8, true, 1);
}
TEST_F(IvfFileReaderTest, BasicVP9FileNtpTimestamp) {
CreateTestFile(kVideoCodecVP9, false, 1);
ValidateContent(kVideoCodecVP9, false, 1);
}
TEST_F(IvfFileReaderTest, BasicVP9FileMsTimestamp) {
CreateTestFile(kVideoCodecVP9, true, 1);
ValidateContent(kVideoCodecVP9, true, 1);
}
TEST_F(IvfFileReaderTest, BasicAv1FileNtpTimestamp) {
CreateTestFile(kVideoCodecAV1, false, 1);
ValidateContent(kVideoCodecAV1, false, 1);
}
TEST_F(IvfFileReaderTest, BasicAv1FileMsTimestamp) {
CreateTestFile(kVideoCodecAV1, true, 1);
ValidateContent(kVideoCodecAV1, true, 1);
}
TEST_F(IvfFileReaderTest, BasicH264FileNtpTimestamp) {
CreateTestFile(kVideoCodecH264, false, 1);
ValidateContent(kVideoCodecH264, false, 1);
}
TEST_F(IvfFileReaderTest, BasicH264FileMsTimestamp) {
CreateTestFile(kVideoCodecH264, true, 1);
ValidateContent(kVideoCodecH264, true, 1);
}
TEST_F(IvfFileReaderTest, MultilayerVp8FileNtpTimestamp) {
CreateTestFile(kVideoCodecVP8, false, 3);
ValidateContent(kVideoCodecVP8, false, 3);
}
TEST_F(IvfFileReaderTest, MultilayerVP9FileNtpTimestamp) {
CreateTestFile(kVideoCodecVP9, false, 3);
ValidateContent(kVideoCodecVP9, false, 3);
}
TEST_F(IvfFileReaderTest, MultilayerAv1FileNtpTimestamp) {
CreateTestFile(kVideoCodecAV1, false, 3);
ValidateContent(kVideoCodecAV1, false, 3);
}
TEST_F(IvfFileReaderTest, MultilayerH264FileNtpTimestamp) {
CreateTestFile(kVideoCodecH264, false, 3);
ValidateContent(kVideoCodecH264, false, 3);
}
} // namespace webrtc