| OLD | NEW |
| 1 /* | 1 /* |
| 2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. | 2 * Copyright (c) 2016 The WebRTC project authors. All Rights Reserved. |
| 3 * | 3 * |
| 4 * Use of this source code is governed by a BSD-style license | 4 * Use of this source code is governed by a BSD-style license |
| 5 * that can be found in the LICENSE file in the root of the source | 5 * that can be found in the LICENSE file in the root of the source |
| 6 * tree. An additional intellectual property rights grant can be found | 6 * tree. An additional intellectual property rights grant can be found |
| 7 * in the file PATENTS. All contributing project authors may | 7 * in the file PATENTS. All contributing project authors may |
| 8 * be found in the AUTHORS file in the root of the source tree. | 8 * be found in the AUTHORS file in the root of the source tree. |
| 9 */ | 9 */ |
| 10 | 10 |
| 11 #include "webrtc/modules/video_coding/utility/ivf_file_writer.h" | 11 #include "webrtc/modules/video_coding/utility/ivf_file_writer.h" |
| 12 | 12 |
| 13 #include <memory> | 13 #include <memory> |
| 14 | 14 |
| 15 #include "testing/gtest/include/gtest/gtest.h" | 15 #include "testing/gtest/include/gtest/gtest.h" |
| 16 #include "webrtc/base/helpers.h" | 16 #include "webrtc/base/helpers.h" |
| 17 #include "webrtc/base/logging.h" | 17 #include "webrtc/base/logging.h" |
| 18 #include "webrtc/base/thread.h" | 18 #include "webrtc/base/thread.h" |
| 19 #include "webrtc/base/timeutils.h" | 19 #include "webrtc/base/timeutils.h" |
| 20 #include "webrtc/modules/rtp_rtcp/source/byte_io.h" | 20 #include "webrtc/modules/rtp_rtcp/source/byte_io.h" |
| 21 #include "webrtc/test/testsupport/fileutils.h" | 21 #include "webrtc/test/testsupport/fileutils.h" |
| 22 | 22 |
| 23 #if defined(WEBRTC_WIN) |
| 24 #include "webrtc/base/win32.h" |
| 25 #else |
| 26 #include <unistd.h> |
| 27 #endif |
| 28 |
| 23 namespace webrtc { | 29 namespace webrtc { |
| 24 | 30 |
| 25 namespace { | 31 namespace { |
| 26 static const int kHeaderSize = 32; | 32 static const int kHeaderSize = 32; |
| 27 static const int kFrameHeaderSize = 12; | 33 static const int kFrameHeaderSize = 12; |
| 28 static uint8_t dummy_payload[4] = {0, 1, 2, 3}; | 34 static uint8_t dummy_payload[4] = {0, 1, 2, 3}; |
| 29 static const int kMaxFileRetries = 5; | |
| 30 } // namespace | 35 } // namespace |
| 31 | 36 |
| 37 void RemoveFile(const std::string& path) { |
| 38 #if defined(WEBRTC_WIN) |
| 39 ::DeleteFile(rtc::ToUtf16(path).c_str()); |
| 40 #else |
| 41 ::unlink(path.c_str()); |
| 42 #endif |
| 43 } |
| 44 |
| 32 class IvfFileWriterTest : public ::testing::Test { | 45 class IvfFileWriterTest : public ::testing::Test { |
| 33 protected: | 46 protected: |
| 34 void SetUp() override { | 47 void SetUp() override { |
| 35 const uint64_t start_id = rtc::CreateRandomId64(); | 48 file_name_ = |
| 36 uint64_t id = start_id; | 49 webrtc::test::TempFilename(webrtc::test::OutputPath(), "test_file"); |
| 37 do { | |
| 38 std::ostringstream oss; | |
| 39 oss << test::OutputPath() << "ivf_test_file_" << id++ << ".ivf"; | |
| 40 file_name_ = oss.str(); | |
| 41 } while ((id - start_id) < 100u && FileExists(false)); | |
| 42 ASSERT_LT(id - start_id, 100u); | |
| 43 } | 50 } |
| 51 void TearDown() override { RemoveFile(file_name_); } |
| 44 | 52 |
| 45 bool WriteDummyTestFrames(int width, | 53 bool WriteDummyTestFrames(VideoCodecType codec_type, |
| 54 int width, |
| 46 int height, | 55 int height, |
| 47 int num_frames, | 56 int num_frames, |
| 48 bool use_capture_tims_ms) { | 57 bool use_capture_tims_ms) { |
| 49 EncodedImage frame; | 58 EncodedImage frame; |
| 50 frame._buffer = dummy_payload; | 59 frame._buffer = dummy_payload; |
| 51 frame._encodedWidth = width; | 60 frame._encodedWidth = width; |
| 52 frame._encodedHeight = height; | 61 frame._encodedHeight = height; |
| 53 for (int i = 1; i <= num_frames; ++i) { | 62 for (int i = 1; i <= num_frames; ++i) { |
| 54 frame._length = i % sizeof(dummy_payload); | 63 frame._length = i % sizeof(dummy_payload); |
| 55 if (use_capture_tims_ms) { | 64 if (use_capture_tims_ms) { |
| 56 frame.capture_time_ms_ = i; | 65 frame.capture_time_ms_ = i; |
| 57 } else { | 66 } else { |
| 58 frame._timeStamp = i; | 67 frame._timeStamp = i; |
| 59 } | 68 } |
| 60 if (!file_writer_->WriteFrame(frame)) | 69 if (!file_writer_->WriteFrame(frame, codec_type)) |
| 61 return false; | 70 return false; |
| 62 } | 71 } |
| 63 return true; | 72 return true; |
| 64 } | 73 } |
| 65 | 74 |
| 66 void VerifyIvfHeader(FileWrapper* file, | 75 void VerifyIvfHeader(rtc::File* file, |
| 67 const uint8_t fourcc[4], | 76 const uint8_t fourcc[4], |
| 68 int width, | 77 int width, |
| 69 int height, | 78 int height, |
| 70 uint32_t num_frames, | 79 uint32_t num_frames, |
| 71 bool use_capture_tims_ms) { | 80 bool use_capture_tims_ms) { |
| 72 ASSERT_TRUE(file->is_open()); | 81 ASSERT_TRUE(file->IsOpen()); |
| 73 uint8_t data[kHeaderSize]; | 82 uint8_t data[kHeaderSize]; |
| 74 ASSERT_EQ(kHeaderSize, file->Read(data, kHeaderSize)); | 83 ASSERT_EQ(static_cast<size_t>(kHeaderSize), file->Read(data, kHeaderSize)); |
| 75 | 84 |
| 76 uint8_t dkif[4] = {'D', 'K', 'I', 'F'}; | 85 uint8_t dkif[4] = {'D', 'K', 'I', 'F'}; |
| 77 EXPECT_EQ(0, memcmp(dkif, data, 4)); | 86 EXPECT_EQ(0, memcmp(dkif, data, 4)); |
| 78 EXPECT_EQ(0u, ByteReader<uint16_t>::ReadLittleEndian(&data[4])); | 87 EXPECT_EQ(0u, ByteReader<uint16_t>::ReadLittleEndian(&data[4])); |
| 79 EXPECT_EQ(32u, ByteReader<uint16_t>::ReadLittleEndian(&data[6])); | 88 EXPECT_EQ(32u, ByteReader<uint16_t>::ReadLittleEndian(&data[6])); |
| 80 EXPECT_EQ(0, memcmp(fourcc, &data[8], 4)); | 89 EXPECT_EQ(0, memcmp(fourcc, &data[8], 4)); |
| 81 EXPECT_EQ(width, ByteReader<uint16_t>::ReadLittleEndian(&data[12])); | 90 EXPECT_EQ(width, ByteReader<uint16_t>::ReadLittleEndian(&data[12])); |
| 82 EXPECT_EQ(height, ByteReader<uint16_t>::ReadLittleEndian(&data[14])); | 91 EXPECT_EQ(height, ByteReader<uint16_t>::ReadLittleEndian(&data[14])); |
| 83 EXPECT_EQ(use_capture_tims_ms ? 1000u : 90000u, | 92 EXPECT_EQ(use_capture_tims_ms ? 1000u : 90000u, |
| 84 ByteReader<uint32_t>::ReadLittleEndian(&data[16])); | 93 ByteReader<uint32_t>::ReadLittleEndian(&data[16])); |
| 85 EXPECT_EQ(1u, ByteReader<uint32_t>::ReadLittleEndian(&data[20])); | 94 EXPECT_EQ(1u, ByteReader<uint32_t>::ReadLittleEndian(&data[20])); |
| 86 EXPECT_EQ(num_frames, ByteReader<uint32_t>::ReadLittleEndian(&data[24])); | 95 EXPECT_EQ(num_frames, ByteReader<uint32_t>::ReadLittleEndian(&data[24])); |
| 87 EXPECT_EQ(0u, ByteReader<uint32_t>::ReadLittleEndian(&data[28])); | 96 EXPECT_EQ(0u, ByteReader<uint32_t>::ReadLittleEndian(&data[28])); |
| 88 } | 97 } |
| 89 | 98 |
| 90 void VerifyDummyTestFrames(FileWrapper* file, uint32_t num_frames) { | 99 void VerifyDummyTestFrames(rtc::File* file, uint32_t num_frames) { |
| 91 const int kMaxFrameSize = 4; | 100 const int kMaxFrameSize = 4; |
| 92 for (uint32_t i = 1; i <= num_frames; ++i) { | 101 for (uint32_t i = 1; i <= num_frames; ++i) { |
| 93 uint8_t frame_header[kFrameHeaderSize]; | 102 uint8_t frame_header[kFrameHeaderSize]; |
| 94 ASSERT_EQ(kFrameHeaderSize, file->Read(frame_header, kFrameHeaderSize)); | 103 ASSERT_EQ(static_cast<unsigned int>(kFrameHeaderSize), |
| 104 file->Read(frame_header, kFrameHeaderSize)); |
| 95 uint32_t frame_length = | 105 uint32_t frame_length = |
| 96 ByteReader<uint32_t>::ReadLittleEndian(&frame_header[0]); | 106 ByteReader<uint32_t>::ReadLittleEndian(&frame_header[0]); |
| 97 EXPECT_EQ(i % 4, frame_length); | 107 EXPECT_EQ(i % 4, frame_length); |
| 98 uint64_t timestamp = | 108 uint64_t timestamp = |
| 99 ByteReader<uint64_t>::ReadLittleEndian(&frame_header[4]); | 109 ByteReader<uint64_t>::ReadLittleEndian(&frame_header[4]); |
| 100 EXPECT_EQ(i, timestamp); | 110 EXPECT_EQ(i, timestamp); |
| 101 | 111 |
| 102 uint8_t data[kMaxFrameSize] = {}; | 112 uint8_t data[kMaxFrameSize] = {}; |
| 103 ASSERT_EQ(frame_length, | 113 ASSERT_EQ(frame_length, |
| 104 static_cast<uint32_t>(file->Read(data, frame_length))); | 114 static_cast<uint32_t>(file->Read(data, frame_length))); |
| 105 EXPECT_EQ(0, memcmp(data, dummy_payload, frame_length)); | 115 EXPECT_EQ(0, memcmp(data, dummy_payload, frame_length)); |
| 106 } | 116 } |
| 107 } | 117 } |
| 108 | 118 |
| 109 void RunBasicFileStructureTest(VideoCodecType codec_type, | 119 void RunBasicFileStructureTest(VideoCodecType codec_type, |
| 110 const uint8_t fourcc[4], | 120 const uint8_t fourcc[4], |
| 111 bool use_capture_tims_ms) { | 121 bool use_capture_tims_ms) { |
| 112 file_writer_ = IvfFileWriter::Open(file_name_, codec_type); | 122 file_writer_ = IvfFileWriter::Wrap(rtc::File::Open(file_name_), 0); |
| 113 ASSERT_TRUE(file_writer_.get()); | 123 ASSERT_TRUE(file_writer_.get()); |
| 114 const int kWidth = 320; | 124 const int kWidth = 320; |
| 115 const int kHeight = 240; | 125 const int kHeight = 240; |
| 116 const int kNumFrames = 257; | 126 const int kNumFrames = 257; |
| 117 EXPECT_TRUE( | 127 ASSERT_TRUE(WriteDummyTestFrames(codec_type, kWidth, kHeight, kNumFrames, |
| 118 WriteDummyTestFrames(kWidth, kHeight, kNumFrames, use_capture_tims_ms)); | 128 use_capture_tims_ms)); |
| 119 EXPECT_TRUE(file_writer_->Close()); | 129 EXPECT_TRUE(file_writer_->Close()); |
| 120 | 130 |
| 121 std::unique_ptr<FileWrapper> out_file(FileWrapper::Create()); | 131 rtc::File out_file = rtc::File::Open(file_name_); |
| 122 ASSERT_TRUE(out_file->OpenFile(file_name_.c_str(), true)); | 132 VerifyIvfHeader(&out_file, fourcc, kWidth, kHeight, kNumFrames, |
| 123 VerifyIvfHeader(out_file.get(), fourcc, kWidth, kHeight, kNumFrames, | |
| 124 use_capture_tims_ms); | 133 use_capture_tims_ms); |
| 125 VerifyDummyTestFrames(out_file.get(), kNumFrames); | 134 VerifyDummyTestFrames(&out_file, kNumFrames); |
| 126 | 135 |
| 127 out_file->CloseFile(); | 136 out_file.Close(); |
| 128 | |
| 129 bool file_removed = false; | |
| 130 for (int i = 0; i < kMaxFileRetries; ++i) { | |
| 131 file_removed = remove(file_name_.c_str()) == 0; | |
| 132 if (file_removed) | |
| 133 break; | |
| 134 | |
| 135 // Couldn't remove file for some reason, wait a sec and try again. | |
| 136 rtc::Thread::SleepMs(1000); | |
| 137 } | |
| 138 EXPECT_TRUE(file_removed); | |
| 139 } | |
| 140 | |
| 141 // Check whether file exists or not, and if it does not meet expectation, | |
| 142 // wait a bit and check again, up to kMaxFileRetries times. This is an ugly | |
| 143 // hack to avoid flakiness on certain operating systems where antivirus | |
| 144 // software may unexpectedly lock files and keep them from disappearing or | |
| 145 // being reused. | |
| 146 bool FileExists(bool expected) { | |
| 147 bool file_exists = expected; | |
| 148 std::unique_ptr<FileWrapper> file_wrapper; | |
| 149 int iterations = 0; | |
| 150 do { | |
| 151 if (file_wrapper.get() != nullptr) | |
| 152 rtc::Thread::SleepMs(1000); | |
| 153 file_wrapper.reset(FileWrapper::Create()); | |
| 154 file_exists = file_wrapper->OpenFile(file_name_.c_str(), true); | |
| 155 file_wrapper->CloseFile(); | |
| 156 } while (file_exists != expected && ++iterations < kMaxFileRetries); | |
| 157 return file_exists; | |
| 158 } | 137 } |
| 159 | 138 |
| 160 std::string file_name_; | 139 std::string file_name_; |
| 161 std::unique_ptr<IvfFileWriter> file_writer_; | 140 std::unique_ptr<IvfFileWriter> file_writer_; |
| 162 }; | 141 }; |
| 163 | 142 |
| 164 TEST_F(IvfFileWriterTest, RemovesUnusedFile) { | |
| 165 file_writer_ = IvfFileWriter::Open(file_name_, kVideoCodecVP8); | |
| 166 ASSERT_TRUE(file_writer_.get() != nullptr); | |
| 167 EXPECT_TRUE(FileExists(true)); | |
| 168 EXPECT_TRUE(file_writer_->Close()); | |
| 169 EXPECT_FALSE(FileExists(false)); | |
| 170 EXPECT_FALSE(file_writer_->Close()); // Can't close twice. | |
| 171 } | |
| 172 | |
| 173 TEST_F(IvfFileWriterTest, WritesBasicVP8FileNtpTimestamp) { | 143 TEST_F(IvfFileWriterTest, WritesBasicVP8FileNtpTimestamp) { |
| 174 const uint8_t fourcc[4] = {'V', 'P', '8', '0'}; | 144 const uint8_t fourcc[4] = {'V', 'P', '8', '0'}; |
| 175 RunBasicFileStructureTest(kVideoCodecVP8, fourcc, false); | 145 RunBasicFileStructureTest(kVideoCodecVP8, fourcc, false); |
| 176 } | 146 } |
| 177 | 147 |
| 178 TEST_F(IvfFileWriterTest, WritesBasicVP8FileMsTimestamp) { | 148 TEST_F(IvfFileWriterTest, WritesBasicVP8FileMsTimestamp) { |
| 179 const uint8_t fourcc[4] = {'V', 'P', '8', '0'}; | 149 const uint8_t fourcc[4] = {'V', 'P', '8', '0'}; |
| 180 RunBasicFileStructureTest(kVideoCodecVP8, fourcc, true); | 150 RunBasicFileStructureTest(kVideoCodecVP8, fourcc, true); |
| 181 } | 151 } |
| 182 | 152 |
| (...skipping 10 matching lines...) Expand all Loading... |
| 193 TEST_F(IvfFileWriterTest, WritesBasicH264FileNtpTimestamp) { | 163 TEST_F(IvfFileWriterTest, WritesBasicH264FileNtpTimestamp) { |
| 194 const uint8_t fourcc[4] = {'H', '2', '6', '4'}; | 164 const uint8_t fourcc[4] = {'H', '2', '6', '4'}; |
| 195 RunBasicFileStructureTest(kVideoCodecH264, fourcc, false); | 165 RunBasicFileStructureTest(kVideoCodecH264, fourcc, false); |
| 196 } | 166 } |
| 197 | 167 |
| 198 TEST_F(IvfFileWriterTest, WritesBasicH264FileMsTimestamp) { | 168 TEST_F(IvfFileWriterTest, WritesBasicH264FileMsTimestamp) { |
| 199 const uint8_t fourcc[4] = {'H', '2', '6', '4'}; | 169 const uint8_t fourcc[4] = {'H', '2', '6', '4'}; |
| 200 RunBasicFileStructureTest(kVideoCodecH264, fourcc, true); | 170 RunBasicFileStructureTest(kVideoCodecH264, fourcc, true); |
| 201 } | 171 } |
| 202 | 172 |
| 173 TEST_F(IvfFileWriterTest, ClosesWhenReachesLimit) { |
| 174 const uint8_t fourcc[4] = {'V', 'P', '8', '0'}; |
| 175 const int kWidth = 320; |
| 176 const int kHeight = 240; |
| 177 const int kNumFramesToWrite = 2; |
| 178 const int kNumFramesToFit = 1; |
| 179 |
| 180 file_writer_ = IvfFileWriter::Wrap( |
| 181 rtc::File::Open(file_name_), |
| 182 kHeaderSize + |
| 183 kNumFramesToFit * (kFrameHeaderSize + sizeof(dummy_payload))); |
| 184 ASSERT_TRUE(file_writer_.get()); |
| 185 |
| 186 ASSERT_FALSE(WriteDummyTestFrames(kVideoCodecVP8, kWidth, kHeight, |
| 187 kNumFramesToWrite, true)); |
| 188 ASSERT_FALSE(file_writer_->Close()); |
| 189 |
| 190 rtc::File out_file = rtc::File::Open(file_name_); |
| 191 VerifyIvfHeader(&out_file, fourcc, kWidth, kHeight, kNumFramesToFit, true); |
| 192 VerifyDummyTestFrames(&out_file, kNumFramesToFit); |
| 193 |
| 194 out_file.Close(); |
| 195 } |
| 196 |
| 203 } // namespace webrtc | 197 } // namespace webrtc |
| OLD | NEW |