| /* |
| * 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 "video/decode_synchronizer.h" |
| |
| #include <stddef.h> |
| |
| #include <memory> |
| #include <utility> |
| |
| #include "absl/functional/any_invocable.h" |
| #include "api/metronome/test/fake_metronome.h" |
| #include "api/units/time_delta.h" |
| #include "test/gmock.h" |
| #include "test/gtest.h" |
| #include "test/time_controller/simulated_time_controller.h" |
| #include "video/frame_decode_scheduler.h" |
| #include "video/frame_decode_timing.h" |
| |
| using ::testing::_; |
| using ::testing::Eq; |
| using ::testing::Invoke; |
| using ::testing::Mock; |
| using ::testing::MockFunction; |
| using ::testing::Return; |
| |
| namespace webrtc { |
| |
| class MockMetronome : public Metronome { |
| public: |
| MOCK_METHOD(void, |
| RequestCallOnNextTick, |
| (absl::AnyInvocable<void() &&> callback), |
| (override)); |
| MOCK_METHOD(TimeDelta, TickPeriod, (), (const, override)); |
| }; |
| |
| class DecodeSynchronizerTest : public ::testing::Test { |
| public: |
| static constexpr TimeDelta kTickPeriod = TimeDelta::Millis(33); |
| |
| DecodeSynchronizerTest() |
| : time_controller_(Timestamp::Millis(1337)), |
| clock_(time_controller_.GetClock()), |
| metronome_(kTickPeriod), |
| decode_synchronizer_(clock_, |
| &metronome_, |
| time_controller_.GetMainThread()) {} |
| |
| protected: |
| GlobalSimulatedTimeController time_controller_; |
| Clock* clock_; |
| test::ForcedTickMetronome metronome_; |
| DecodeSynchronizer decode_synchronizer_; |
| }; |
| |
| TEST_F(DecodeSynchronizerTest, AllFramesReadyBeforeNextTickDecoded) { |
| MockFunction<void(uint32_t, Timestamp)> mock_callback1; |
| auto scheduler1 = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| MockFunction<void(unsigned int, Timestamp)> mock_callback2; |
| auto scheduler2 = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| { |
| uint32_t frame_rtp = 90000; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = |
| clock_->CurrentTime() + kTickPeriod - TimeDelta::Millis(3), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(60)}; |
| scheduler1->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback1.AsStdFunction()); |
| EXPECT_CALL(mock_callback1, |
| Call(Eq(frame_rtp), Eq(frame_sched.render_time))); |
| } |
| { |
| uint32_t frame_rtp = 123456; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = |
| clock_->CurrentTime() + kTickPeriod - TimeDelta::Millis(2), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(70)}; |
| scheduler2->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback2.AsStdFunction()); |
| EXPECT_CALL(mock_callback2, |
| Call(Eq(frame_rtp), Eq(frame_sched.render_time))); |
| } |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // Cleanup |
| scheduler1->Stop(); |
| scheduler2->Stop(); |
| } |
| |
| TEST_F(DecodeSynchronizerTest, FramesNotDecodedIfDecodeTimeIsInNextInterval) { |
| MockFunction<void(unsigned int, Timestamp)> mock_callback; |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| uint32_t frame_rtp = 90000; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = |
| clock_->CurrentTime() + kTickPeriod + TimeDelta::Millis(10), |
| .render_time = |
| clock_->CurrentTime() + kTickPeriod + TimeDelta::Millis(30)}; |
| scheduler->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback.AsStdFunction()); |
| |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| // No decodes should have happened in this tick. |
| ::testing::Mock::VerifyAndClearExpectations(&mock_callback); |
| |
| // Decode should happen on next tick. |
| EXPECT_CALL(mock_callback, Call(Eq(frame_rtp), Eq(frame_sched.render_time))); |
| time_controller_.AdvanceTime(kTickPeriod); |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // Cleanup |
| scheduler->Stop(); |
| } |
| |
| TEST_F(DecodeSynchronizerTest, FrameDecodedOnce) { |
| MockFunction<void(unsigned int, Timestamp)> mock_callback; |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| uint32_t frame_rtp = 90000; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = clock_->CurrentTime() + TimeDelta::Millis(30), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(60)}; |
| scheduler->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback.AsStdFunction()); |
| EXPECT_CALL(mock_callback, Call(_, _)).Times(1); |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| ::testing::Mock::VerifyAndClearExpectations(&mock_callback); |
| |
| // Trigger tick again. No frame should be decoded now. |
| time_controller_.AdvanceTime(kTickPeriod); |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // Cleanup |
| scheduler->Stop(); |
| } |
| |
| TEST_F(DecodeSynchronizerTest, FrameWithDecodeTimeInPastDecodedImmediately) { |
| MockFunction<void(unsigned int, Timestamp)> mock_callback; |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| uint32_t frame_rtp = 90000; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = clock_->CurrentTime() - TimeDelta::Millis(5), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(30)}; |
| EXPECT_CALL(mock_callback, Call(Eq(90000u), _)).Times(1); |
| scheduler->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback.AsStdFunction()); |
| // Verify the callback was invoked already. |
| ::testing::Mock::VerifyAndClearExpectations(&mock_callback); |
| |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // Cleanup |
| scheduler->Stop(); |
| } |
| |
| TEST_F(DecodeSynchronizerTest, |
| FrameWithDecodeTimeFarBeforeNextTickDecodedImmediately) { |
| MockFunction<void(unsigned int, Timestamp)> mock_callback; |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| // Frame which would be behind by more than kMaxAllowedFrameDelay after |
| // the next tick. |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = clock_->CurrentTime() + kTickPeriod - |
| FrameDecodeTiming::kMaxAllowedFrameDelay - |
| TimeDelta::Millis(1), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(30)}; |
| EXPECT_CALL(mock_callback, Call(Eq(90000u), _)).Times(1); |
| scheduler->ScheduleFrame(90000, frame_sched, mock_callback.AsStdFunction()); |
| // Verify the callback was invoked already. |
| ::testing::Mock::VerifyAndClearExpectations(&mock_callback); |
| |
| time_controller_.AdvanceTime(kTickPeriod); |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // A frame that would be behind by exactly kMaxAllowedFrameDelay after next |
| // tick should decode at the next tick. |
| FrameDecodeTiming::FrameSchedule queued_frame{ |
| .latest_decode_time = clock_->CurrentTime() + kTickPeriod - |
| FrameDecodeTiming::kMaxAllowedFrameDelay, |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(30)}; |
| scheduler->ScheduleFrame(180000, queued_frame, mock_callback.AsStdFunction()); |
| // Verify the callback was invoked already. |
| ::testing::Mock::VerifyAndClearExpectations(&mock_callback); |
| |
| EXPECT_CALL(mock_callback, Call(Eq(180000u), _)).Times(1); |
| time_controller_.AdvanceTime(kTickPeriod); |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| |
| // Cleanup |
| scheduler->Stop(); |
| } |
| |
| TEST_F(DecodeSynchronizerTest, FramesNotReleasedAfterStop) { |
| MockFunction<void(unsigned int, Timestamp)> mock_callback; |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| |
| uint32_t frame_rtp = 90000; |
| FrameDecodeTiming::FrameSchedule frame_sched{ |
| .latest_decode_time = clock_->CurrentTime() + TimeDelta::Millis(30), |
| .render_time = clock_->CurrentTime() + TimeDelta::Millis(60)}; |
| scheduler->ScheduleFrame(frame_rtp, frame_sched, |
| mock_callback.AsStdFunction()); |
| // Cleanup |
| scheduler->Stop(); |
| |
| // No callback should occur on this tick since Stop() was called before. |
| metronome_.Tick(); |
| time_controller_.AdvanceTime(TimeDelta::Zero()); |
| } |
| |
| TEST(DecodeSynchronizerStandaloneTest, |
| MetronomeNotListenedWhenNoStreamsAreActive) { |
| GlobalSimulatedTimeController time_controller(Timestamp::Millis(4711)); |
| Clock* clock(time_controller.GetClock()); |
| MockMetronome metronome; |
| ON_CALL(metronome, TickPeriod).WillByDefault(Return(TimeDelta::Seconds(1))); |
| DecodeSynchronizer decode_synchronizer_(clock, &metronome, |
| time_controller.GetMainThread()); |
| absl::AnyInvocable<void() &&> callback; |
| EXPECT_CALL(metronome, RequestCallOnNextTick) |
| .WillOnce(Invoke([&callback](absl::AnyInvocable<void() &&> cb) { |
| callback = std::move(cb); |
| })); |
| auto scheduler = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| auto scheduler2 = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| scheduler->Stop(); |
| scheduler2->Stop(); |
| time_controller.AdvanceTime(TimeDelta::Seconds(1)); |
| ASSERT_TRUE(callback); |
| (std::move)(callback)(); |
| } |
| |
| TEST(DecodeSynchronizerStandaloneTest, |
| RegistersCallbackOnceDuringRepeatedRegistrations) { |
| GlobalSimulatedTimeController time_controller(Timestamp::Millis(4711)); |
| Clock* clock(time_controller.GetClock()); |
| MockMetronome metronome; |
| ON_CALL(metronome, TickPeriod).WillByDefault(Return(TimeDelta::Seconds(1))); |
| DecodeSynchronizer decode_synchronizer_(clock, &metronome, |
| time_controller.GetMainThread()); |
| // Expect at most 1 call to register a callback. |
| EXPECT_CALL(metronome, RequestCallOnNextTick); |
| auto scheduler1 = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| scheduler1->Stop(); |
| auto scheduler2 = decode_synchronizer_.CreateSynchronizedFrameScheduler(); |
| Mock::VerifyAndClearExpectations(&metronome); |
| scheduler2->Stop(); |
| } |
| |
| } // namespace webrtc |