blob: 940b9157643eed61f0ab45a67ac385ac74de105e [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "mojo/services/gfx/composition/cpp/frame_tracker.h"
#include "mojo/public/cpp/application/application_test_base.h"
#include "mojo/public/cpp/system/macros.h"
namespace test {
class FrameTrackerTest : public mojo::test::ApplicationTestBase {
public:
FrameTrackerTest() {}
~FrameTrackerTest() override {}
protected:
mojo::gfx::composition::FrameTracker frame_tracker_;
uint64_t Update(int64_t frame_time,
uint64_t frame_interval,
int64_t frame_deadline,
int64_t presentation_time,
MojoTimeTicks now) {
mojo::gfx::composition::FrameInfo frame_info;
frame_info.frame_time = frame_time;
frame_info.frame_interval = frame_interval;
frame_info.frame_deadline = frame_deadline;
frame_info.presentation_time = presentation_time;
return frame_tracker_.Update(frame_info, now);
}
private:
MOJO_DISALLOW_COPY_AND_ASSIGN(FrameTrackerTest);
};
namespace {
TEST_F(FrameTrackerTest, InitialState) {
EXPECT_EQ(0u, frame_tracker_.frame_count());
EXPECT_EQ(0, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(0u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(0, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(0, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, ClearResetsEverything) {
Update(10, 10u, 10, 10, 10);
frame_tracker_.Clear();
EXPECT_EQ(0u, frame_tracker_.frame_count());
EXPECT_EQ(0, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(0u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(0, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(0, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, TypicalUpdate) {
// Signalled right at frame time.
// No corrections.
EXPECT_EQ(0u, Update(12, 10u, 24, 28, 12));
EXPECT_EQ(1u, frame_tracker_.frame_count());
EXPECT_EQ(12, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(24, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(28, frame_tracker_.frame_info().presentation_time);
// Signalled 1 ms after frame time.
// No corrections.
EXPECT_EQ(10u, Update(22, 10u, 34, 38, 22 + 1));
EXPECT_EQ(2u, frame_tracker_.frame_count());
EXPECT_EQ(22, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(34, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(38, frame_tracker_.frame_info().presentation_time);
// Signalled 9 ms after frame time (frame interval is 10 ms).
// No corrections.
EXPECT_EQ(10u, Update(32, 10u, 44, 48, 32 + 9));
EXPECT_EQ(3u, frame_tracker_.frame_count());
EXPECT_EQ(32, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(44, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(48, frame_tracker_.frame_info().presentation_time);
// Frame interval changed.
// No corrections.
EXPECT_EQ(14u, Update(46, 15u, 59, 62, 46 + 2));
EXPECT_EQ(4u, frame_tracker_.frame_count());
EXPECT_EQ(46, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(15u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(59, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(62, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, LagCompensation) {
// Received signal exactly when next frame should begin.
// Skip 1 frame.
EXPECT_EQ(0u, Update(12, 10u, 24, 28, 12 + 10));
EXPECT_EQ(1u, frame_tracker_.frame_count());
EXPECT_EQ(22, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(34, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(38, frame_tracker_.frame_info().presentation_time);
// Received signal 2 ms after next frame should begin.
// Skip 1 frame.
EXPECT_EQ(20u, Update(32, 10u, 44, 48, 32 + 10 + 2));
EXPECT_EQ(2u, frame_tracker_.frame_count());
EXPECT_EQ(42, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(54, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(58, frame_tracker_.frame_info().presentation_time);
// Received signal 35 ms after next frame should begin.
// Skip 4 frames.
EXPECT_EQ(50u, Update(52, 10u, 64, 68, 52 + 10 + 35));
EXPECT_EQ(3u, frame_tracker_.frame_count());
EXPECT_EQ(92, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(104, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(108, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, FrameTimeInPast) {
// Frame time is in the future.
// Clamp frame time to present.
EXPECT_EQ(0u, Update(12, 10u, 24, 28, 12 - 1));
EXPECT_EQ(1u, frame_tracker_.frame_count());
EXPECT_EQ(11, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(24, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(28, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, FrameDeadlineBehindFrameTime) {
// Frame deadline is earlier than frame time.
// Clamp frame deadline time to frame time.
EXPECT_EQ(0u, Update(12, 10u, 12 - 1, 28, 12));
EXPECT_EQ(1u, frame_tracker_.frame_count());
EXPECT_EQ(12, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(12, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(28, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, PresentationTimeBehindFrameDeadline) {
// Presentation time is earlier than frame deadline.
// Clamp presentation time to frame deadline.
EXPECT_EQ(0u, Update(12, 10u, 24, 24 - 1, 12));
EXPECT_EQ(1u, frame_tracker_.frame_count());
EXPECT_EQ(12, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(24, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(24, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, NonMonotonicFrameTime) {
Update(12, 10u, 24, 28, 12);
// Frame time is going backwards.
// Clamp frame time to old frame time.
EXPECT_EQ(0u, Update(10, 10u, 24, 28, 13));
EXPECT_EQ(2u, frame_tracker_.frame_count());
EXPECT_EQ(12, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(24, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(28, frame_tracker_.frame_info().presentation_time);
}
TEST_F(FrameTrackerTest, NonMonotonicPresentationTime) {
Update(12, 10u, 24, 28, 12);
// Presentation time is going backwards.
// Clamp presentation time to old presentation time.
EXPECT_EQ(10u, Update(22, 10u, 26, 27, 22));
EXPECT_EQ(2u, frame_tracker_.frame_count());
EXPECT_EQ(22, frame_tracker_.frame_info().frame_time);
EXPECT_EQ(10u, frame_tracker_.frame_info().frame_interval);
EXPECT_EQ(26, frame_tracker_.frame_info().frame_deadline);
EXPECT_EQ(28, frame_tracker_.frame_info().presentation_time);
}
} // namespace
} // namespace mojo