Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(26)

Side by Side Diff: services/ui/ws/frame_generator_unittest.cc

Issue 2755573002: Add FrameGenerator Unit Tests (Closed)
Patch Set: Addressed comments Created 3 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
« no previous file with comments | « services/ui/ws/frame_generator.cc ('k') | no next file » | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright 2017 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "services/ui/ws/frame_generator.h"
6
7 #include "base/macros.h"
8 #include "cc/output/compositor_frame_sink.h"
9 #include "cc/scheduler/begin_frame_source.h"
10 #include "cc/test/begin_frame_args_test.cc"
11 #include "cc/test/fake_external_begin_frame_source.h"
12 #include "services/ui/ws/frame_generator_delegate.h"
13 #include "services/ui/ws/server_window.h"
14 #include "services/ui/ws/server_window_delegate.h"
15 #include "testing/gtest/include/gtest/gtest.h"
16
17 namespace ui {
18 namespace ws {
19 namespace test {
20
21 // TestFrameGeneratorDelegate implements FrameGeneratorDelegate and returns
22 // false when IsInHighContrastMode() is called.
23 class TestFrameGeneratorDelegate : public FrameGeneratorDelegate {
24 public:
25 TestFrameGeneratorDelegate() {}
26 ~TestFrameGeneratorDelegate() override {}
27
28 // FrameGeneratorDelegate implementation:
29 bool IsInHighContrastMode() override { return false; }
30
31 private:
32 DISALLOW_COPY_AND_ASSIGN(TestFrameGeneratorDelegate);
33 };
34
35 // TestServerWindowDelegate implements ServerWindowDelegate and returns nullptrs
36 // when either of the methods from the interface is called.
37 class TestServerWindowDelegate : public ServerWindowDelegate {
38 public:
39 TestServerWindowDelegate() {}
40 ~TestServerWindowDelegate() override {}
41
42 // ServerWindowDelegate implementation:
43 cc::mojom::DisplayCompositor* GetDisplayCompositor() override {
44 return nullptr;
45 }
46
47 ServerWindow* GetRootWindow(const ServerWindow* window) override {
48 return nullptr;
49 }
50
51 private:
52 DISALLOW_COPY_AND_ASSIGN(TestServerWindowDelegate);
53 };
54
55 // FakeCompositorFrameSink observes a FakeExternalBeginFrameSource and receives
56 // CompositorFrames from a FrameGenerator.
57 class FakeCompositorFrameSink : public cc::CompositorFrameSink,
58 public cc::BeginFrameObserver,
59 public cc::ExternalBeginFrameSourceClient {
60 public:
61 FakeCompositorFrameSink()
62 : cc::CompositorFrameSink(nullptr, nullptr, nullptr, nullptr) {}
63
64 // cc::CompositorFrameSink implementation:
65 bool BindToClient(cc::CompositorFrameSinkClient* client) override {
66 if (!cc::CompositorFrameSink::BindToClient(client))
67 return false;
68
69 external_begin_frame_source_ =
70 base::MakeUnique<cc::ExternalBeginFrameSource>(this);
71 client_->SetBeginFrameSource(external_begin_frame_source_.get());
72 return true;
73 }
74
75 void DetachFromClient() override {
76 cc::CompositorFrameSink::DetachFromClient();
77 }
78
79 void SubmitCompositorFrame(cc::CompositorFrame frame) override {
80 ++number_frames_received_;
81 last_frame_ = std::move(frame);
82 last_referenced_surfaces_ = &last_frame_.metadata.referenced_surfaces;
83 last_metadata_ = &last_frame_.metadata;
Fady Samuel 2017/03/17 19:22:39 No need for these two lines: last_referenced_surfa
Alex Z. 2017/03/17 19:33:54 Done.
84 }
85
86 // cc::BeginFrameObserver implementation.
87 void OnBeginFrame(const cc::BeginFrameArgs& args) override {
88 external_begin_frame_source_->OnBeginFrame(args);
89 last_begin_frame_args_ = args;
90 }
91
92 const cc::BeginFrameArgs& LastUsedBeginFrameArgs() const override {
93 return last_begin_frame_args_;
94 }
95
96 void OnBeginFrameSourcePausedChanged(bool paused) override {}
97
98 // cc::ExternalBeginFrameSourceClient implementation:
99 void OnNeedsBeginFrames(bool needs_begin_frames) override {
100 needs_begin_frames_ = needs_begin_frames;
101 UpdateNeedsBeginFramesInternal();
102 }
103
104 void OnDidFinishFrame(const cc::BeginFrameAck& ack) override {
105 external_begin_frame_source_->DidFinishFrame(this, ack);
106 }
107
108 void SetBeginFrameSource(cc::BeginFrameSource* source) {
109 if (begin_frame_source_ && observing_begin_frames_) {
110 begin_frame_source_->RemoveObserver(this);
111 observing_begin_frames_ = false;
112 }
113 begin_frame_source_ = source;
114 UpdateNeedsBeginFramesInternal();
115 }
116
117 const std::vector<cc::SurfaceId>* last_referenced_surfaces() {
118 return last_referenced_surfaces_;
Fady Samuel 2017/03/17 19:22:39 I'm not entirely sure why this is returning a poin
Alex Z. 2017/03/17 19:33:54 Done.
119 }
120
121 const cc::CompositorFrameMetadata* last_metadata() { return last_metadata_; }
Fady Samuel 2017/03/17 19:22:39 I'm not entirely sure why this is returning a poin
122
123 int number_frames_received() { return number_frames_received_; }
124
125 private:
126 void UpdateNeedsBeginFramesInternal() {
127 if (!begin_frame_source_)
128 return;
129
130 if (needs_begin_frames_ == observing_begin_frames_)
131 return;
132
133 observing_begin_frames_ = needs_begin_frames_;
134 if (needs_begin_frames_) {
135 begin_frame_source_->AddObserver(this);
136 } else {
137 begin_frame_source_->RemoveObserver(this);
138 }
139 }
140
141 int number_frames_received_ = 0;
142 std::unique_ptr<cc::ExternalBeginFrameSource> external_begin_frame_source_;
143 cc::BeginFrameSource* begin_frame_source_ = nullptr;
144 cc::BeginFrameArgs last_begin_frame_args_;
145 bool observing_begin_frames_ = false;
146 bool needs_begin_frames_ = false;
147 cc::CompositorFrame last_frame_;
148 const std::vector<cc::SurfaceId>* last_referenced_surfaces_ = nullptr;
Fady Samuel 2017/03/17 19:22:39 remove this?
Alex Z. 2017/03/17 19:33:54 Done.
149 const cc::CompositorFrameMetadata* last_metadata_ = nullptr;
Fady Samuel 2017/03/17 19:22:39 remove this?
Alex Z. 2017/03/17 19:33:54 Done.
150
151 DISALLOW_COPY_AND_ASSIGN(FakeCompositorFrameSink);
152 };
153
154 class FrameGeneratorTest : public testing::Test {
155 public:
156 FrameGeneratorTest() {}
157 ~FrameGeneratorTest() override {}
158
159 // testing::Test overrides:
160 void SetUp() override {
161 testing::Test::SetUp();
162
163 std::unique_ptr<FakeCompositorFrameSink> compositor_frame_sink =
164 base::MakeUnique<FakeCompositorFrameSink>();
165 compositor_frame_sink_ = compositor_frame_sink.get();
166
167 constexpr float kRefreshRate = 0.f;
168 constexpr bool kTickAutomatically = false;
169 begin_frame_source_ = base::MakeUnique<cc::FakeExternalBeginFrameSource>(
170 kRefreshRate, kTickAutomatically);
171 compositor_frame_sink_->SetBeginFrameSource(begin_frame_source_.get());
172 delegate_ = base::MakeUnique<TestFrameGeneratorDelegate>();
173 server_window_delegate_ = base::MakeUnique<TestServerWindowDelegate>();
174 root_window_ = base::MakeUnique<ServerWindow>(server_window_delegate_.get(),
175 WindowId());
176 root_window_->SetVisible(true);
177 frame_generator_ = base::MakeUnique<FrameGenerator>(
178 delegate_.get(), root_window_.get(), std::move(compositor_frame_sink));
179 };
180
181 int NumberOfFramesReceived() {
182 return compositor_frame_sink_->number_frames_received();
183 }
184
185 void IssueBeginFrame() {
186 begin_frame_source_->TestOnBeginFrame(cc::CreateBeginFrameArgsForTesting(
187 BEGINFRAME_FROM_HERE, 0, next_sequence_number_));
188 ++next_sequence_number_;
189 }
190
191 FrameGenerator* frame_generator() { return frame_generator_.get(); }
192
193 const std::vector<cc::SurfaceId>* ReferencedSurfaces() {
194 return compositor_frame_sink_->last_referenced_surfaces();
195 }
196
197 const cc::CompositorFrameMetadata* LastMetadata() {
198 return compositor_frame_sink_->last_metadata();
199 }
200
201 private:
202 FakeCompositorFrameSink* compositor_frame_sink_ = nullptr;
203 std::unique_ptr<cc::FakeExternalBeginFrameSource> begin_frame_source_;
204 std::unique_ptr<TestFrameGeneratorDelegate> delegate_;
205 std::unique_ptr<TestServerWindowDelegate> server_window_delegate_;
206 std::unique_ptr<ServerWindow> root_window_;
207 std::unique_ptr<FrameGenerator> frame_generator_;
208 int next_sequence_number_ = 1;
209
210 DISALLOW_COPY_AND_ASSIGN(FrameGeneratorTest);
211 };
212
213 TEST_F(FrameGeneratorTest, OnSurfaceCreated) {
214 EXPECT_EQ(0, NumberOfFramesReceived());
215 EXPECT_EQ(nullptr, ReferencedSurfaces());
216
217 // FrameGenerator does not request BeginFrames upon creation.
218 IssueBeginFrame();
219 EXPECT_EQ(0, NumberOfFramesReceived());
220
221 const cc::SurfaceId kArbitrarySurfaceId(
222 cc::FrameSinkId(1, 1),
223 cc::LocalSurfaceId(1, base::UnguessableToken::Create()));
224 const cc::SurfaceInfo kArbitrarySurfaceInfo(kArbitrarySurfaceId, 1.0f,
225 gfx::Size(100, 100));
226 frame_generator()->OnSurfaceCreated(kArbitrarySurfaceInfo);
227 EXPECT_EQ(0, NumberOfFramesReceived());
228
229 IssueBeginFrame();
230 EXPECT_EQ(1, NumberOfFramesReceived());
231
232 // Verify that the CompositorFrame refers to the window manager's surface via
233 // referenced_surfaces.
234 const std::vector<cc::SurfaceId>* referenced_surfaces = ReferencedSurfaces();
235 EXPECT_TRUE(referenced_surfaces);
236 EXPECT_EQ(1lu, referenced_surfaces->size());
237 EXPECT_EQ(kArbitrarySurfaceId, referenced_surfaces->front());
238
239 // FrameGenerator stops requesting BeginFrames after submitting a
240 // CompositorFrame.
241 IssueBeginFrame();
242 EXPECT_EQ(1, NumberOfFramesReceived());
243 }
244
245 TEST_F(FrameGeneratorTest, SetDeviceScaleFactor) {
246 EXPECT_EQ(0, NumberOfFramesReceived());
247 const cc::SurfaceId kArbitrarySurfaceId(
248 cc::FrameSinkId(1, 1),
249 cc::LocalSurfaceId(1, base::UnguessableToken::Create()));
250 const cc::SurfaceInfo kArbitrarySurfaceInfo(kArbitrarySurfaceId, 1.0f,
251 gfx::Size(100, 100));
252 constexpr float kDefaultScaleFactor = 1.0f;
253 constexpr float kArbitraryScaleFactor = 0.5f;
254
255 // A valid SurfaceInfo is required before setting device scale factor.
256 frame_generator()->OnSurfaceCreated(kArbitrarySurfaceInfo);
257 IssueBeginFrame();
258 EXPECT_EQ(1, NumberOfFramesReceived());
259
260 // FrameGenerator stops requesting BeginFrames after receiving one.
261 IssueBeginFrame();
262 EXPECT_EQ(1, NumberOfFramesReceived());
263
264 // FrameGenerator does not request BeginFrames if its device scale factor
265 // remains unchanged.
266 frame_generator()->SetDeviceScaleFactor(kDefaultScaleFactor);
267 IssueBeginFrame();
268 EXPECT_EQ(1, NumberOfFramesReceived());
269 const cc::CompositorFrameMetadata* last_metadata = LastMetadata();
270 EXPECT_TRUE(last_metadata);
271 EXPECT_EQ(kDefaultScaleFactor, last_metadata->device_scale_factor);
272
273 frame_generator()->SetDeviceScaleFactor(kArbitraryScaleFactor);
274 IssueBeginFrame();
275 EXPECT_EQ(2, NumberOfFramesReceived());
276 last_metadata = LastMetadata();
277 EXPECT_EQ(kArbitraryScaleFactor, last_metadata->device_scale_factor);
278 }
279
280 } // namespace test
281 } // namespace ws
282 } // namespace ui
OLDNEW
« no previous file with comments | « services/ui/ws/frame_generator.cc ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698