/* * Copyright (c) 2018 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 #include #include "api/scoped_refptr.h" #include "api/video/i420_buffer.h" #include "rtc_base/logging.h" #include "rtc_base/strings/string_builder.h" #include "test/testsupport/file_utils.h" #include "test/testsupport/frame_reader.h" namespace webrtc { namespace test { namespace { // Size of header: "YUV4MPEG2 WXXXXXX HXXXXXX FXXX:1 C420\n" // We allocate up to 6 digits for width and height and up to 3 digits for FPS. const size_t kFileHeaderMaxSize = 38; // Size of header: "YUV4MPEG2 WX HX FX:1 C420\n" const size_t kFileHeaderMinSize = 26; // Size of header: "FRAME\n" const size_t kFrameHeaderSize = 6; std::string GetExpectedHeaderPrefix(int width, int height) { rtc::StringBuilder out; out << "YUV4MPEG2 W" << width << " H" << height << " F"; return out.str(); } } // namespace Y4mFrameReaderImpl::Y4mFrameReaderImpl(std::string input_filename, int width, int height) : YuvFrameReaderImpl(input_filename, width, height) { frame_length_in_bytes_ += kFrameHeaderSize; buffer_ = new char[kFileHeaderMaxSize]; } Y4mFrameReaderImpl::~Y4mFrameReaderImpl() { delete[] buffer_; } bool Y4mFrameReaderImpl::Init() { if (input_width_ <= 0 || input_height_ <= 0) { RTC_LOG(LS_ERROR) << "Frame width and height must be positive. Was: " << input_width_ << "x" << input_height_; return false; } input_file_ = fopen(input_filename_.c_str(), "rb"); if (input_file_ == nullptr) { RTC_LOG(LS_ERROR) << "Couldn't open input file: " << input_filename_; return false; } size_t source_file_size = GetFileSize(input_filename_); if (source_file_size <= 0u) { RTC_LOG(LS_ERROR) << "Input file " << input_filename_ << " is empty."; return false; } char* c_file_header = fgets(buffer_, kFileHeaderMaxSize, input_file_); std::string file_header(c_file_header); if (file_header.size() < kFileHeaderMinSize) { RTC_LOG(LS_ERROR) << "Couldn't read Y4M header from input file: " << input_filename_; return false; } if (file_header.find(GetExpectedHeaderPrefix(input_width_, input_height_)) != 0) { RTC_LOG(LS_ERROR) << "Couldn't read Y4M file: " << input_filename_ << ". Input file has different resolution, expected: " << GetExpectedHeaderPrefix(input_width_, input_height_) << "[0-9]?:1 C420; got: " << file_header; return false; } number_of_frames_ = static_cast((source_file_size - file_header.size()) / frame_length_in_bytes_); if (number_of_frames_ == 0) { RTC_LOG(LS_ERROR) << "Input file " << input_filename_ << " is too small."; } return true; } rtc::scoped_refptr Y4mFrameReaderImpl::ReadFrame() { if (input_file_ == nullptr) { RTC_LOG(LS_ERROR) << "Y4mFrameReaderImpl is not initialized."; return nullptr; } if (fread(buffer_, 1, kFrameHeaderSize, input_file_) < kFrameHeaderSize && ferror(input_file_)) { RTC_LOG(LS_ERROR) << "Couldn't read frame header from input file: " << input_filename_; return nullptr; } return YuvFrameReaderImpl::ReadFrame(); } } // namespace test } // namespace webrtc