|  | /* | 
|  | *  Copyright (c) 2022 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 "test/testsupport/fixed_fps_video_frame_writer_adapter.h" | 
|  |  | 
|  | #include <cmath> | 
|  | #include <utility> | 
|  |  | 
|  | #include "absl/types/optional.h" | 
|  | #include "api/units/time_delta.h" | 
|  | #include "api/video/video_sink_interface.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "test/testsupport/video_frame_writer.h" | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace test { | 
|  | namespace { | 
|  |  | 
|  | constexpr TimeDelta kOneSecond = TimeDelta::Seconds(1); | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | FixedFpsVideoFrameWriterAdapter::FixedFpsVideoFrameWriterAdapter( | 
|  | int fps, | 
|  | Clock* clock, | 
|  | std::unique_ptr<VideoFrameWriter> delegate) | 
|  | : inter_frame_interval_(kOneSecond / fps), | 
|  | clock_(clock), | 
|  | delegate_(std::move(delegate)) {} | 
|  |  | 
|  | FixedFpsVideoFrameWriterAdapter::~FixedFpsVideoFrameWriterAdapter() { | 
|  | Close(); | 
|  | } | 
|  |  | 
|  | void FixedFpsVideoFrameWriterAdapter::Close() { | 
|  | if (is_closed_) { | 
|  | return; | 
|  | } | 
|  | is_closed_ = true; | 
|  | if (!last_frame_.has_value()) { | 
|  | return; | 
|  | } | 
|  | Timestamp now = Now(); | 
|  | RTC_CHECK(WriteMissedSlotsExceptLast(now)); | 
|  | RTC_CHECK(delegate_->WriteFrame(*last_frame_)); | 
|  | delegate_->Close(); | 
|  | } | 
|  |  | 
|  | bool FixedFpsVideoFrameWriterAdapter::WriteFrame(const VideoFrame& frame) { | 
|  | RTC_CHECK(!is_closed_); | 
|  | Timestamp now = Now(); | 
|  | if (!last_frame_.has_value()) { | 
|  | RTC_CHECK(!last_frame_time_.IsFinite()); | 
|  | last_frame_ = frame; | 
|  | last_frame_time_ = now; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | RTC_CHECK(last_frame_time_.IsFinite()); | 
|  |  | 
|  | if (last_frame_time_ > now) { | 
|  | // New frame was recevied before expected time "slot" for current | 
|  | // `last_frame_` came => just replace current `last_frame_` with | 
|  | // received `frame`. | 
|  | RTC_CHECK_LE(last_frame_time_ - now, inter_frame_interval_ / 2); | 
|  | last_frame_ = frame; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (!WriteMissedSlotsExceptLast(now)) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | if (now - last_frame_time_ < inter_frame_interval_ / 2) { | 
|  | // New frame was received closer to the expected time "slot" for current | 
|  | // `last_frame_` than to the next "slot" => just replace current | 
|  | // `last_frame_` with received `frame`. | 
|  | last_frame_ = frame; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | if (!delegate_->WriteFrame(*last_frame_)) { | 
|  | return false; | 
|  | } | 
|  | last_frame_ = frame; | 
|  | last_frame_time_ = last_frame_time_ + inter_frame_interval_; | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool FixedFpsVideoFrameWriterAdapter::WriteMissedSlotsExceptLast( | 
|  | Timestamp now) { | 
|  | RTC_CHECK(last_frame_time_.IsFinite()); | 
|  | while (now - last_frame_time_ > inter_frame_interval_) { | 
|  | if (!delegate_->WriteFrame(*last_frame_)) { | 
|  | return false; | 
|  | } | 
|  | last_frame_time_ = last_frame_time_ + inter_frame_interval_; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | Timestamp FixedFpsVideoFrameWriterAdapter::Now() const { | 
|  | return clock_->CurrentTime(); | 
|  | } | 
|  |  | 
|  | }  // namespace test | 
|  | }  // namespace webrtc |