| /* | 
 |  *  Copyright 2021 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 "common_video/framerate_controller.h" | 
 |  | 
 | #include <limits> | 
 |  | 
 | #include "rtc_base/time_utils.h" | 
 | #include "test/gtest.h" | 
 |  | 
 | namespace webrtc { | 
 | namespace { | 
 | constexpr int kInputFps = 30; | 
 | constexpr int kNumFrames = 60; | 
 | }  // namespace | 
 |  | 
 | class FramerateControllerTest : public ::testing::Test { | 
 |  protected: | 
 |   int64_t GetNextTimestampNs() { | 
 |     int64_t interval_us = rtc::kNumMicrosecsPerSec / kInputFps; | 
 |     next_timestamp_us_ += interval_us; | 
 |     return next_timestamp_us_ * rtc::kNumNanosecsPerMicrosec; | 
 |   } | 
 |  | 
 |   int64_t next_timestamp_us_ = rtc::TimeMicros(); | 
 |   FramerateController controller_; | 
 | }; | 
 |  | 
 | TEST_F(FramerateControllerTest, NoFramesDroppedIfNothingRequested) { | 
 |   // Default max framerate is maxdouble. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_FALSE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, AllFramesDroppedIfZeroRequested) { | 
 |   controller_.SetMaxFramerate(0); | 
 |  | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_TRUE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, AllFramesDroppedIfNegativeRequested) { | 
 |   controller_.SetMaxFramerate(-1); | 
 |  | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_TRUE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, EverySecondFrameDroppedIfHalfRequested) { | 
 |   controller_.SetMaxFramerate(kInputFps / 2); | 
 |  | 
 |   // The first frame should not be dropped. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_EQ(i % 2 == 0, controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, EveryThirdFrameDroppedIfTwoThirdsRequested) { | 
 |   controller_.SetMaxFramerate(kInputFps * 2 / 3); | 
 |  | 
 |   // The first frame should not be dropped. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_EQ(i % 3 == 0, controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, NoFrameDroppedIfTwiceRequested) { | 
 |   controller_.SetMaxFramerate(kInputFps * 2); | 
 |  | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_FALSE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, TestAverageFramerate) { | 
 |   const double kMaxFps = 18.2; | 
 |   controller_.SetMaxFramerate(kMaxFps); | 
 |  | 
 |   const int kNumSec = 10; | 
 |   int frames_kept = 0; | 
 |   for (int i = 0; i < kInputFps * kNumSec; ++i) { | 
 |     if (!controller_.ShouldDropFrame(GetNextTimestampNs())) | 
 |       ++frames_kept; | 
 |   } | 
 |   double average_fps = static_cast<double>(frames_kept) / kNumSec; | 
 |   EXPECT_NEAR(kMaxFps, average_fps, 0.01); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, NoFrameDroppedForLargeTimestampOffset) { | 
 |   controller_.SetMaxFramerate(kInputFps); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(0)); | 
 |  | 
 |   const int64_t kLargeOffsetNs = -987654321LL * 1000; | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(kLargeOffsetNs)); | 
 |  | 
 |   int64_t input_interval_ns = rtc::kNumNanosecsPerSec / kInputFps; | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(kLargeOffsetNs + input_interval_ns)); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, NoFrameDroppedIfInputWithJitterRequested) { | 
 |   controller_.SetMaxFramerate(kInputFps); | 
 |  | 
 |   // Input fps with jitter. | 
 |   int64_t input_interval_ns = rtc::kNumNanosecsPerSec / kInputFps; | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 0 / 10)); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 10 / 10 - 1)); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 25 / 10)); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 30 / 10)); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 35 / 10)); | 
 |   EXPECT_FALSE(controller_.ShouldDropFrame(input_interval_ns * 50 / 10)); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, FrameDroppedWhenReductionRequested) { | 
 |   controller_.SetMaxFramerate(kInputFps); | 
 |  | 
 |   // Expect no frame drop. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_FALSE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 |  | 
 |   // Reduce max frame rate. | 
 |   controller_.SetMaxFramerate(kInputFps / 2); | 
 |  | 
 |   // Verify that every other frame is dropped. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_EQ(i % 2 == 0, controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, NoFramesDroppedAfterReset) { | 
 |   controller_.SetMaxFramerate(0); | 
 |  | 
 |   // All frames dropped. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_TRUE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 |  | 
 |   controller_.Reset(); | 
 |  | 
 |   // Expect no frame drop after reset. | 
 |   for (int i = 1; i < kNumFrames; ++i) | 
 |     EXPECT_FALSE(controller_.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | TEST_F(FramerateControllerTest, TestKeepFrame) { | 
 |   FramerateController controller(kInputFps / 2); | 
 |  | 
 |   EXPECT_FALSE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 |   EXPECT_TRUE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 |   EXPECT_FALSE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 |   EXPECT_TRUE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 |   EXPECT_FALSE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 |  | 
 |   // Next frame should be dropped. | 
 |   // Keep this frame (e.g. in case of a key frame). | 
 |   controller.KeepFrame(GetNextTimestampNs()); | 
 |   // Expect next frame to be dropped instead. | 
 |   EXPECT_TRUE(controller.ShouldDropFrame(GetNextTimestampNs())); | 
 | } | 
 |  | 
 | }  // namespace webrtc |