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

Side by Side Diff: cc/trees/layer_tree_host_unittest.cc

Issue 12502026: Reformat to match chromium style. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: addressed comments and some more Created 7 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 | Annotate | Revision Log
« no previous file with comments | « no previous file | 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
1 // Copyright 2011 The Chromium Authors. All rights reserved. 1 // Copyright 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be 2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file. 3 // found in the LICENSE file.
4 4
5 #include "cc/trees/layer_tree_host.h" 5 #include "cc/trees/layer_tree_host.h"
6 6
7 #include "base/synchronization/lock.h" 7 #include "base/synchronization/lock.h"
8 #include "cc/animation/timing_function.h" 8 #include "cc/animation/timing_function.h"
9 #include "cc/layers/content_layer.h" 9 #include "cc/layers/content_layer.h"
10 #include "cc/layers/content_layer_client.h" 10 #include "cc/layers/content_layer_client.h"
(...skipping 24 matching lines...) Expand all
35 #include "third_party/khronos/GLES2/gl2.h" 35 #include "third_party/khronos/GLES2/gl2.h"
36 #include "third_party/khronos/GLES2/gl2ext.h" 36 #include "third_party/khronos/GLES2/gl2ext.h"
37 #include "third_party/skia/include/core/SkPicture.h" 37 #include "third_party/skia/include/core/SkPicture.h"
38 #include "ui/gfx/point_conversions.h" 38 #include "ui/gfx/point_conversions.h"
39 #include "ui/gfx/size_conversions.h" 39 #include "ui/gfx/size_conversions.h"
40 #include "ui/gfx/vector2d_conversions.h" 40 #include "ui/gfx/vector2d_conversions.h"
41 41
42 namespace cc { 42 namespace cc {
43 namespace { 43 namespace {
44 44
45 class LayerTreeHostTest : public LayerTreeTest {}; 45 class LayerTreeHostTest : public LayerTreeTest {
46 };
46 47
47 // Test interleaving of redraws and commits 48 // Test interleaving of redraws and commits
48 class LayerTreeHostTestCommitingWithContinuousRedraw : public LayerTreeHostTest { 49 class LayerTreeHostTestCommitingWithContinuousRedraw
49 public: 50 : public LayerTreeHostTest {
50 LayerTreeHostTestCommitingWithContinuousRedraw() 51 public:
51 : m_numCompleteCommits(0) 52 LayerTreeHostTestCommitingWithContinuousRedraw()
52 , m_numDraws(0) 53 : num_complete_commits_(0), num_draws_(0) {}
53 { 54
54 } 55 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
55 56
56 virtual void BeginTest() OVERRIDE 57 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
57 { 58 num_complete_commits_++;
58 PostSetNeedsCommitToMainThread(); 59 if (num_complete_commits_ == 2)
59 } 60 EndTest();
60 61 }
61 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 62
62 { 63 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
63 m_numCompleteCommits++; 64 if (num_draws_ == 1)
64 if (m_numCompleteCommits == 2) 65 PostSetNeedsCommitToMainThread();
65 EndTest(); 66 num_draws_++;
66 } 67 PostSetNeedsRedrawToMainThread();
67 68 }
68 virtual void DrawLayersOnThread(LayerTreeHostImpl*) OVERRIDE 69
69 { 70 virtual void AfterTest() OVERRIDE {}
70 if (m_numDraws == 1) 71
71 PostSetNeedsCommitToMainThread(); 72 private:
72 m_numDraws++; 73 int num_complete_commits_;
73 PostSetNeedsRedrawToMainThread(); 74 int num_draws_;
74 } 75 };
75 76
76 virtual void AfterTest() OVERRIDE 77 MULTI_THREAD_TEST_F(LayerTreeHostTestCommitingWithContinuousRedraw);
77 {
78 }
79
80 private:
81 int m_numCompleteCommits;
82 int m_numDraws;
83 };
84
85 MULTI_THREAD_TEST_F(LayerTreeHostTestCommitingWithContinuousRedraw)
86 78
87 // Two setNeedsCommits in a row should lead to at least 1 commit and at least 1 79 // Two setNeedsCommits in a row should lead to at least 1 commit and at least 1
88 // draw with frame 0. 80 // draw with frame 0.
89 class LayerTreeHostTestSetNeedsCommit1 : public LayerTreeHostTest { 81 class LayerTreeHostTestSetNeedsCommit1 : public LayerTreeHostTest {
90 public: 82 public:
91 LayerTreeHostTestSetNeedsCommit1() 83 LayerTreeHostTestSetNeedsCommit1() : num_commits_(0), num_draws_(0) {}
92 : m_numCommits(0) 84
93 , m_numDraws(0) 85 virtual void BeginTest() OVERRIDE {
94 { 86 PostSetNeedsCommitToMainThread();
95 } 87 PostSetNeedsCommitToMainThread();
96 88 }
97 virtual void BeginTest() OVERRIDE 89
98 { 90 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
99 PostSetNeedsCommitToMainThread(); 91 num_draws_++;
100 PostSetNeedsCommitToMainThread(); 92 if (!impl->active_tree()->source_frame_number())
101 } 93 EndTest();
102 94 }
103 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 95
104 { 96 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
105 m_numDraws++; 97 num_commits_++;
106 if (!impl->active_tree()->source_frame_number()) 98 }
107 EndTest(); 99
108 } 100 virtual void AfterTest() OVERRIDE {
109 101 EXPECT_GE(1, num_commits_);
110 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 102 EXPECT_GE(1, num_draws_);
111 { 103 }
112 m_numCommits++; 104
113 } 105 private:
114 106 int num_commits_;
115 virtual void AfterTest() OVERRIDE 107 int num_draws_;
116 { 108 };
117 EXPECT_GE(1, m_numCommits); 109
118 EXPECT_GE(1, m_numDraws); 110 // MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsCommit1);
119 }
120
121 private:
122 int m_numCommits;
123 int m_numDraws;
124 };
125
126 //MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsCommit1)
127 111
128 // A setNeedsCommit should lead to 1 commit. Issuing a second commit after that 112 // A setNeedsCommit should lead to 1 commit. Issuing a second commit after that
129 // first committed frame draws should lead to another commit. 113 // first committed frame draws should lead to another commit.
130 class LayerTreeHostTestSetNeedsCommit2 : public LayerTreeHostTest { 114 class LayerTreeHostTestSetNeedsCommit2 : public LayerTreeHostTest {
131 public: 115 public:
132 LayerTreeHostTestSetNeedsCommit2() 116 LayerTreeHostTestSetNeedsCommit2() : num_commits_(0), num_draws_(0) {}
133 : m_numCommits(0) 117
134 , m_numDraws(0) 118 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
135 { 119
136 } 120 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
137 121 if (impl->active_tree()->source_frame_number() == 0)
138 virtual void BeginTest() OVERRIDE 122 PostSetNeedsCommitToMainThread();
139 { 123 else if (impl->active_tree()->source_frame_number() == 1)
140 PostSetNeedsCommitToMainThread(); 124 EndTest();
141 } 125 }
142 126
143 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 127 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
144 { 128 num_commits_++;
145 if (impl->active_tree()->source_frame_number() == 0) 129 }
146 PostSetNeedsCommitToMainThread(); 130
147 else if (impl->active_tree()->source_frame_number() == 1) 131 virtual void AfterTest() OVERRIDE {
148 EndTest(); 132 EXPECT_EQ(2, num_commits_);
149 } 133 EXPECT_GE(2, num_draws_);
150 134 }
151 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 135
152 { 136 private:
153 m_numCommits++; 137 int num_commits_;
154 } 138 int num_draws_;
155 139 };
156 virtual void AfterTest() OVERRIDE 140
157 { 141 MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsCommit2);
158 EXPECT_EQ(2, m_numCommits);
159 EXPECT_GE(2, m_numDraws);
160 }
161
162 private:
163 int m_numCommits;
164 int m_numDraws;
165 };
166
167 MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsCommit2)
168 142
169 // 1 setNeedsRedraw after the first commit has completed should lead to 1 143 // 1 setNeedsRedraw after the first commit has completed should lead to 1
170 // additional draw. 144 // additional draw.
171 class LayerTreeHostTestSetNeedsRedraw : public LayerTreeHostTest { 145 class LayerTreeHostTestSetNeedsRedraw : public LayerTreeHostTest {
172 public: 146 public:
173 LayerTreeHostTestSetNeedsRedraw() 147 LayerTreeHostTestSetNeedsRedraw() : num_commits_(0), num_draws_(0) {}
174 : m_numCommits(0) 148
175 , m_numDraws(0) 149 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
176 { 150
177 } 151 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
178 152 EXPECT_EQ(0, impl->active_tree()->source_frame_number());
179 virtual void BeginTest() OVERRIDE 153 if (!num_draws_) {
180 { 154 // Redraw again to verify that the second redraw doesn't commit.
181 PostSetNeedsCommitToMainThread(); 155 PostSetNeedsRedrawToMainThread();
182 } 156 } else {
183 157 EndTest();
184 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 158 }
185 { 159 num_draws_++;
186 EXPECT_EQ(0, impl->active_tree()->source_frame_number()); 160 }
187 if (!m_numDraws) 161
188 PostSetNeedsRedrawToMainThread(); // Redraw again to verify that the second redraw doesn't commit. 162 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
189 else 163 EXPECT_EQ(0, num_draws_);
190 EndTest(); 164 num_commits_++;
191 m_numDraws++; 165 }
192 } 166
193 167 virtual void AfterTest() OVERRIDE {
194 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 168 EXPECT_GE(2, num_draws_);
195 { 169 EXPECT_EQ(1, num_commits_);
196 EXPECT_EQ(0, m_numDraws); 170 }
197 m_numCommits++; 171
198 } 172 private:
199 173 int num_commits_;
200 virtual void AfterTest() OVERRIDE 174 int num_draws_;
201 { 175 };
202 EXPECT_GE(2, m_numDraws); 176
203 EXPECT_EQ(1, m_numCommits); 177 MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsRedraw);
204 }
205
206 private:
207 int m_numCommits;
208 int m_numDraws;
209 };
210
211 MULTI_THREAD_TEST_F(LayerTreeHostTestSetNeedsRedraw)
212 178
213 class LayerTreeHostTestNoExtraCommitFromInvalidate : public LayerTreeHostTest { 179 class LayerTreeHostTestNoExtraCommitFromInvalidate : public LayerTreeHostTest {
214 public: 180 public:
215 LayerTreeHostTestNoExtraCommitFromInvalidate() 181 LayerTreeHostTestNoExtraCommitFromInvalidate()
216 : m_rootLayer(ContentLayer::Create(&client_)) 182 : root_layer_(ContentLayer::Create(&client_)) {}
217 { 183
218 } 184 virtual void BeginTest() OVERRIDE {
219 185 root_layer_->SetAutomaticallyComputeRasterScale(false);
220 virtual void BeginTest() OVERRIDE 186 root_layer_->SetIsDrawable(true);
221 { 187 root_layer_->SetBounds(gfx::Size(1, 1));
222 m_rootLayer->SetAutomaticallyComputeRasterScale(false); 188 layer_tree_host()->SetRootLayer(root_layer_);
223 m_rootLayer->SetIsDrawable(true); 189 PostSetNeedsCommitToMainThread();
224 m_rootLayer->SetBounds(gfx::Size(1, 1)); 190 }
225 layer_tree_host()->SetRootLayer(m_rootLayer); 191
226 PostSetNeedsCommitToMainThread(); 192 virtual void DidCommit() OVERRIDE {
227 } 193 switch (layer_tree_host()->commit_number()) {
228 194 case 1:
229 virtual void DidCommit() OVERRIDE 195 // Changing the content bounds will cause a single commit!
230 { 196 root_layer_->SetRasterScale(4.f);
231 switch (layer_tree_host()->commit_number()) { 197 break;
232 case 1: 198 default:
233 // Changing the content bounds will cause a single commit! 199 // No extra commits.
234 m_rootLayer->SetRasterScale(4.0f); 200 EXPECT_EQ(2, layer_tree_host()->commit_number());
235 break; 201 EndTest();
236 default: 202 }
237 // No extra commits. 203 }
238 EXPECT_EQ(2, layer_tree_host()->commit_number()); 204
239 EndTest(); 205 virtual void AfterTest() OVERRIDE {}
240 } 206
241 } 207 private:
242 208 FakeContentLayerClient client_;
243 virtual void AfterTest() OVERRIDE 209 scoped_refptr<ContentLayer> root_layer_;
244 { 210 };
245 } 211
246 212 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestNoExtraCommitFromInvalidate);
247 private:
248 FakeContentLayerClient client_;
249 scoped_refptr<ContentLayer> m_rootLayer;
250 };
251
252 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestNoExtraCommitFromInvalidate)
253 213
254 class LayerTreeHostTestCompositeAndReadback : public LayerTreeHostTest { 214 class LayerTreeHostTestCompositeAndReadback : public LayerTreeHostTest {
255 public: 215 public:
256 LayerTreeHostTestCompositeAndReadback() 216 LayerTreeHostTestCompositeAndReadback() : num_commits_(0) {}
257 : m_numCommits(0) 217
258 { 218 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
259 } 219
260 220 virtual void DidCommit() OVERRIDE {
261 virtual void BeginTest() OVERRIDE 221 num_commits_++;
262 { 222 if (num_commits_ == 1) {
263 PostSetNeedsCommitToMainThread(); 223 char pixels[4];
264 } 224 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1));
265 225 } else if (num_commits_ == 2) {
266 virtual void DidCommit() OVERRIDE 226 // This is inside the readback. We should get another commit after it.
267 { 227 } else if (num_commits_ == 3) {
268 m_numCommits++; 228 EndTest();
269 if (m_numCommits == 1) { 229 } else {
270 char pixels[4]; 230 NOTREACHED();
271 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1)); 231 }
272 } else if (m_numCommits == 2) { 232 }
273 // This is inside the readback. We should get another commit after i t. 233
274 } else if (m_numCommits == 3) { 234 virtual void AfterTest() OVERRIDE {}
275 EndTest(); 235
276 } else { 236 private:
277 NOTREACHED(); 237 int num_commits_;
278 } 238 };
279 } 239
280 240 MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadback);
281 virtual void AfterTest() OVERRIDE 241
282 { 242 class LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDraws
283 } 243 : public LayerTreeHostTest {
284 244 public:
285 private: 245 LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDraws()
286 int m_numCommits; 246 : num_commits_(0) {}
287 }; 247
288 248 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
289 MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadback) 249
290 250 virtual void DidCommit() OVERRIDE {
291 class LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDraws : public La yerTreeHostTest { 251 num_commits_++;
292 public: 252 if (num_commits_ == 1) {
293 LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDraws() 253 layer_tree_host()->SetNeedsCommit();
294 : m_numCommits(0) 254 } else if (num_commits_ == 2) {
295 { 255 char pixels[4];
296 } 256 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1));
297 257 } else if (num_commits_ == 3) {
298 virtual void BeginTest() OVERRIDE 258 // This is inside the readback. We should get another commit after it.
299 { 259 } else if (num_commits_ == 4) {
300 PostSetNeedsCommitToMainThread(); 260 EndTest();
301 } 261 } else {
302 262 NOTREACHED();
303 virtual void DidCommit() OVERRIDE 263 }
304 { 264 }
305 m_numCommits++; 265
306 if (m_numCommits == 1) { 266 virtual void AfterTest() OVERRIDE {}
307 layer_tree_host()->SetNeedsCommit(); 267
308 } else if (m_numCommits == 2) { 268 private:
309 char pixels[4]; 269 int num_commits_;
310 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1)); 270 };
311 } else if (m_numCommits == 3) { 271
312 // This is inside the readback. We should get another commit after i t. 272 MULTI_THREAD_TEST_F(
313 } else if (m_numCommits == 4) { 273 LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDraws);
314 EndTest(); 274
315 } else { 275 // If the layerTreeHost says it can't draw, Then we should not try to draw.
316 NOTREACHED();
317 }
318 }
319
320 virtual void AfterTest() OVERRIDE
321 {
322 }
323
324 private:
325 int m_numCommits;
326 };
327
328 MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadbackBeforePreviousCommitDra ws)
329
330 // If the layerTreeHost says it can't draw, then we should not try to draw.
331 class LayerTreeHostTestCanDrawBlocksDrawing : public LayerTreeHostTest { 276 class LayerTreeHostTestCanDrawBlocksDrawing : public LayerTreeHostTest {
332 public: 277 public:
333 LayerTreeHostTestCanDrawBlocksDrawing() 278 LayerTreeHostTestCanDrawBlocksDrawing() : num_commits_(0), done_(false) {}
334 : m_numCommits(0) 279
335 , m_done(false) 280 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
336 { 281
337 } 282 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
338 283 if (done_)
339 virtual void BeginTest() OVERRIDE 284 return;
340 { 285 // Only the initial draw should bring us here.
341 PostSetNeedsCommitToMainThread(); 286 EXPECT_TRUE(impl->CanDraw());
342 } 287 EXPECT_EQ(0, impl->active_tree()->source_frame_number());
343 288 }
344 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 289
345 { 290 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
346 if (m_done) 291 if (done_)
347 return; 292 return;
348 // Only the initial draw should bring us here. 293 if (num_commits_ >= 1) {
349 EXPECT_TRUE(impl->CanDraw()); 294 // After the first commit, we should not be able to draw.
350 EXPECT_EQ(0, impl->active_tree()->source_frame_number()); 295 EXPECT_FALSE(impl->CanDraw());
351 } 296 }
352 297 }
353 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 298
354 { 299 virtual void DidCommit() OVERRIDE {
355 if (m_done) 300 num_commits_++;
356 return; 301 if (num_commits_ == 1) {
357 if (m_numCommits >= 1) { 302 // Make the viewport empty so the host says it can't draw.
358 // After the first commit, we should not be able to draw. 303 layer_tree_host()->SetViewportSize(gfx::Size(0, 0), gfx::Size(0, 0));
359 EXPECT_FALSE(impl->CanDraw()); 304 } else if (num_commits_ == 2) {
360 } 305 char pixels[4];
361 } 306 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1));
362 307 } else if (num_commits_ == 3) {
363 virtual void DidCommit() OVERRIDE 308 // Let it draw so we go idle and end the test.
364 { 309 layer_tree_host()->SetViewportSize(gfx::Size(1, 1), gfx::Size(1, 1));
365 m_numCommits++; 310 done_ = true;
366 if (m_numCommits == 1) { 311 EndTest();
367 // Make the viewport empty so the host says it can't draw. 312 }
368 layer_tree_host()->SetViewportSize(gfx::Size(0, 0), gfx::Size(0, 0)) ; 313 }
369 } else if (m_numCommits == 2) { 314
370 char pixels[4]; 315 virtual void AfterTest() OVERRIDE {}
371 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1)); 316
372 } else if (m_numCommits == 3) { 317 private:
373 // Let it draw so we go idle and end the test. 318 int num_commits_;
374 layer_tree_host()->SetViewportSize(gfx::Size(1, 1), gfx::Size(1, 1)) ; 319 bool done_;
375 m_done = true; 320 };
376 EndTest(); 321
377 } 322 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestCanDrawBlocksDrawing);
378 }
379
380 virtual void AfterTest() OVERRIDE
381 {
382 }
383
384 private:
385 int m_numCommits;
386 bool m_done;
387 };
388
389 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestCanDrawBlocksDrawing)
390 323
391 // beginLayerWrite should prevent draws from executing until a commit occurs 324 // beginLayerWrite should prevent draws from executing until a commit occurs
392 class LayerTreeHostTestWriteLayersRedraw : public LayerTreeHostTest { 325 class LayerTreeHostTestWriteLayersRedraw : public LayerTreeHostTest {
393 public: 326 public:
394 LayerTreeHostTestWriteLayersRedraw() 327 LayerTreeHostTestWriteLayersRedraw() : num_commits_(0), num_draws_(0) {}
395 : m_numCommits(0) 328
396 , m_numDraws(0) 329 virtual void BeginTest() OVERRIDE {
397 { 330 PostAcquireLayerTextures();
398 } 331 PostSetNeedsRedrawToMainThread(); // should be inhibited without blocking
399 332 PostSetNeedsCommitToMainThread();
400 virtual void BeginTest() OVERRIDE 333 }
401 { 334
402 PostAcquireLayerTextures(); 335 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
403 PostSetNeedsRedrawToMainThread(); // should be inhibited without blockin g 336 num_draws_++;
404 PostSetNeedsCommitToMainThread(); 337 EXPECT_EQ(num_draws_, num_commits_);
405 } 338 }
406 339
407 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 340 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
408 { 341 num_commits_++;
409 m_numDraws++; 342 EndTest();
410 EXPECT_EQ(m_numDraws, m_numCommits); 343 }
411 } 344
412 345 virtual void AfterTest() OVERRIDE { EXPECT_EQ(1, num_commits_); }
413 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 346
414 { 347 private:
415 m_numCommits++; 348 int num_commits_;
416 EndTest(); 349 int num_draws_;
417 } 350 };
418 351
419 virtual void AfterTest() OVERRIDE 352 MULTI_THREAD_TEST_F(LayerTreeHostTestWriteLayersRedraw);
420 { 353
421 EXPECT_EQ(1, m_numCommits); 354 // Verify that when resuming visibility, Requesting layer write permission
422 }
423
424 private:
425 int m_numCommits;
426 int m_numDraws;
427 };
428
429 MULTI_THREAD_TEST_F(LayerTreeHostTestWriteLayersRedraw)
430
431 // Verify that when resuming visibility, requesting layer write permission
432 // will not deadlock the main thread even though there are not yet any 355 // will not deadlock the main thread even though there are not yet any
433 // scheduled redraws. This behavior is critical for reliably surviving tab 356 // scheduled redraws. This behavior is critical for reliably surviving tab
434 // switching. There are no failure conditions to this test, it just passes 357 // switching. There are no failure conditions to this test, it just passes
435 // by not timing out. 358 // by not timing out.
436 class LayerTreeHostTestWriteLayersAfterVisible : public LayerTreeHostTest { 359 class LayerTreeHostTestWriteLayersAfterVisible : public LayerTreeHostTest {
437 public: 360 public:
438 LayerTreeHostTestWriteLayersAfterVisible() 361 LayerTreeHostTestWriteLayersAfterVisible() : num_commits_(0) {}
439 : m_numCommits(0) 362
440 { 363 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
441 } 364
442 365 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
443 virtual void BeginTest() OVERRIDE 366 num_commits_++;
444 { 367 if (num_commits_ == 2)
445 PostSetNeedsCommitToMainThread(); 368 EndTest();
446 } 369 else if (num_commits_ < 2) {
447 370 PostSetVisibleToMainThread(false);
448 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 371 PostSetVisibleToMainThread(true);
449 { 372 PostAcquireLayerTextures();
450 m_numCommits++; 373 PostSetNeedsCommitToMainThread();
451 if (m_numCommits == 2) 374 }
452 EndTest(); 375 }
453 else if (m_numCommits < 2) { 376
454 PostSetVisibleToMainThread(false); 377 virtual void AfterTest() OVERRIDE {}
455 PostSetVisibleToMainThread(true); 378
456 PostAcquireLayerTextures(); 379 private:
457 PostSetNeedsCommitToMainThread(); 380 int num_commits_;
458 } 381 };
459 } 382
460 383 MULTI_THREAD_TEST_F(LayerTreeHostTestWriteLayersAfterVisible);
461 virtual void AfterTest() OVERRIDE 384
462 { 385 // A compositeAndReadback while invisible should force a normal commit without
463 } 386 // assertion.
464 387 class LayerTreeHostTestCompositeAndReadbackWhileInvisible
465 private: 388 : public LayerTreeHostTest {
466 int m_numCommits; 389 public:
467 }; 390 LayerTreeHostTestCompositeAndReadbackWhileInvisible() : num_commits_(0) {}
468 391
469 MULTI_THREAD_TEST_F(LayerTreeHostTestWriteLayersAfterVisible) 392 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
470 393
471 // A compositeAndReadback while invisible should force a normal commit without a ssertion. 394 virtual void DidCommitAndDrawFrame() OVERRIDE {
472 class LayerTreeHostTestCompositeAndReadbackWhileInvisible : public LayerTreeHost Test { 395 num_commits_++;
473 public: 396 if (num_commits_ == 1) {
474 LayerTreeHostTestCompositeAndReadbackWhileInvisible() 397 layer_tree_host()->SetVisible(false);
475 : m_numCommits(0) 398 layer_tree_host()->SetNeedsCommit();
476 { 399 layer_tree_host()->SetNeedsCommit();
477 } 400 char pixels[4];
478 401 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1));
479 virtual void BeginTest() OVERRIDE 402 } else {
480 { 403 EndTest();
481 PostSetNeedsCommitToMainThread(); 404 }
482 } 405 }
483 406
484 virtual void DidCommitAndDrawFrame() OVERRIDE 407 virtual void AfterTest() OVERRIDE {}
485 { 408
486 m_numCommits++; 409 private:
487 if (m_numCommits == 1) { 410 int num_commits_;
488 layer_tree_host()->SetVisible(false); 411 };
489 layer_tree_host()->SetNeedsCommit(); 412
490 layer_tree_host()->SetNeedsCommit(); 413 MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadbackWhileInvisible);
491 char pixels[4];
492 layer_tree_host()->CompositeAndReadback(&pixels, gfx::Rect(0, 0, 1, 1));
493 } else
494 EndTest();
495
496 }
497
498 virtual void AfterTest() OVERRIDE
499 {
500 }
501
502 private:
503 int m_numCommits;
504 };
505
506 MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadbackWhileInvisible)
507 414
508 class LayerTreeHostTestAbortFrameWhenInvisible : public LayerTreeHostTest { 415 class LayerTreeHostTestAbortFrameWhenInvisible : public LayerTreeHostTest {
509 public: 416 public:
510 LayerTreeHostTestAbortFrameWhenInvisible() 417 LayerTreeHostTestAbortFrameWhenInvisible() {}
511 { 418
512 } 419 virtual void BeginTest() OVERRIDE {
513 420 // Request a commit (from the main thread), Which will trigger the commit
514 virtual void BeginTest() OVERRIDE 421 // flow from the impl side.
515 { 422 layer_tree_host()->SetNeedsCommit();
516 // Request a commit (from the main thread), which will trigger the commi t flow from the impl side. 423 // Then mark ourselves as not visible before processing any more messages
517 layer_tree_host()->SetNeedsCommit(); 424 // on the main thread.
518 // Then mark ourselves as not visible before processing any more message s on the main thread. 425 layer_tree_host()->SetVisible(false);
519 layer_tree_host()->SetVisible(false); 426 // If we make it without kicking a frame, we pass!
520 // If we make it without kicking a frame, we pass! 427 EndTestAfterDelay(1);
521 EndTestAfterDelay(1); 428 }
522 } 429
523 430 virtual void Layout() OVERRIDE {
524 virtual void Layout() OVERRIDE 431 ASSERT_FALSE(true);
525 { 432 EndTest();
526 ASSERT_FALSE(true); 433 }
527 EndTest(); 434
528 } 435 virtual void AfterTest() OVERRIDE {}
529 436
530 virtual void AfterTest() OVERRIDE 437 private:
531 { 438 };
532 } 439
533 440 MULTI_THREAD_TEST_F(LayerTreeHostTestAbortFrameWhenInvisible);
534 private: 441
535 }; 442 // This test verifies that properties on the layer tree host are commited
536 443 // to the impl side.
537 MULTI_THREAD_TEST_F(LayerTreeHostTestAbortFrameWhenInvisible)
538
539 // This test verifies that properties on the layer tree host are commited to the impl side.
540 class LayerTreeHostTestCommit : public LayerTreeHostTest { 444 class LayerTreeHostTestCommit : public LayerTreeHostTest {
541 public: 445 public:
542 446
543 LayerTreeHostTestCommit() { } 447 LayerTreeHostTestCommit() {}
544 448
545 virtual void BeginTest() OVERRIDE 449 virtual void BeginTest() OVERRIDE {
546 { 450 layer_tree_host()->SetViewportSize(gfx::Size(20, 20), gfx::Size(20, 20));
547 layer_tree_host()->SetViewportSize(gfx::Size(20, 20), gfx::Size(20, 20)) ; 451 layer_tree_host()->set_background_color(SK_ColorGRAY);
548 layer_tree_host()->set_background_color(SK_ColorGRAY); 452 layer_tree_host()->SetPageScaleFactorAndLimits(5.f, 5.f, 5.f);
549 layer_tree_host()->SetPageScaleFactorAndLimits(5, 5, 5); 453
550 454 PostSetNeedsCommitToMainThread();
551 PostSetNeedsCommitToMainThread(); 455 }
552 } 456
553 457 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
554 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 458 EXPECT_EQ(gfx::Size(20, 20), impl->layout_viewport_size());
555 { 459 EXPECT_EQ(SK_ColorGRAY, impl->active_tree()->background_color());
556 EXPECT_EQ(gfx::Size(20, 20), impl->layout_viewport_size()); 460 EXPECT_EQ(5.f, impl->active_tree()->page_scale_factor());
557 EXPECT_EQ(SK_ColorGRAY, impl->active_tree()->background_color()); 461
558 EXPECT_EQ(5, impl->active_tree()->page_scale_factor()); 462 EndTest();
559 463 }
560 EndTest(); 464
561 } 465 virtual void AfterTest() OVERRIDE {}
562 466 };
563 virtual void AfterTest() OVERRIDE { } 467
564 }; 468 MULTI_THREAD_TEST_F(LayerTreeHostTestCommit);
565 469
566 MULTI_THREAD_TEST_F(LayerTreeHostTestCommit) 470 // Verifies that startPageScaleAnimation events propagate correctly
567 471 // from LayerTreeHost to LayerTreeHostImpl in the MT compositor.
568 // Verifies that startPageScaleAnimation events propagate correctly from LayerTr eeHost to
569 // LayerTreeHostImpl in the MT compositor.
570 class LayerTreeHostTestStartPageScaleAnimation : public LayerTreeHostTest { 472 class LayerTreeHostTestStartPageScaleAnimation : public LayerTreeHostTest {
571 public: 473 public:
572 474 LayerTreeHostTestStartPageScaleAnimation() : animation_requested_(false) {}
573 LayerTreeHostTestStartPageScaleAnimation() 475
574 : m_animationRequested(false) 476 virtual void BeginTest() OVERRIDE {
575 { 477 layer_tree_host()->root_layer()->SetScrollable(true);
576 } 478 layer_tree_host()->root_layer()->SetScrollOffset(gfx::Vector2d());
577 479 PostSetNeedsCommitToMainThread();
578 virtual void BeginTest() OVERRIDE 480 PostSetNeedsRedrawToMainThread();
579 { 481 }
580 layer_tree_host()->root_layer()->SetScrollable(true); 482
581 layer_tree_host()->root_layer()->SetScrollOffset(gfx::Vector2d()); 483 void RequestStartPageScaleAnimation() {
582 PostSetNeedsCommitToMainThread(); 484 layer_tree_host()->StartPageScaleAnimation(
583 PostSetNeedsRedrawToMainThread(); 485 gfx::Vector2d(), false, 1.25f, base::TimeDelta());
584 } 486 }
585 487
586 void requestStartPageScaleAnimation() 488 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
587 { 489 impl->active_tree()->root_layer()->SetScrollable(true);
588 layer_tree_host()->StartPageScaleAnimation(gfx::Vector2d(), false, 1.25, base::TimeDelta()); 490 impl->active_tree()->root_layer()->SetScrollOffset(gfx::Vector2d());
589 } 491 impl->active_tree()->SetPageScaleFactorAndLimits(
590 492 impl->active_tree()->page_scale_factor(), 0.5f, 2.f);
591 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 493
592 { 494 // We request animation only once.
593 impl->active_tree()->root_layer()->SetScrollable(true); 495 if (!animation_requested_) {
594 impl->active_tree()->root_layer()->SetScrollOffset(gfx::Vector2d()); 496 impl->proxy()->MainThread()->PostTask(
595 impl->active_tree()->SetPageScaleFactorAndLimits(impl->active_tree()->pa ge_scale_factor(), 0.5, 2); 497 base::Bind(&LayerTreeHostTestStartPageScaleAnimation::
596 498 RequestStartPageScaleAnimation,
597 // We request animation only once. 499 base::Unretained(this)));
598 if (!m_animationRequested) { 500 animation_requested_ = true;
599 impl->proxy()->MainThread()->PostTask(base::Bind(&LayerTreeHostTestS tartPageScaleAnimation::requestStartPageScaleAnimation, base::Unretained(this))) ; 501 }
600 m_animationRequested = true; 502 }
601 } 503
602 } 504 virtual void ApplyScrollAndScale(gfx::Vector2d scroll_delta, float scale)
603 505 OVERRIDE {
604 virtual void ApplyScrollAndScale(gfx::Vector2d scrollDelta, float scale) OVE RRIDE 506 gfx::Vector2d offset = layer_tree_host()->root_layer()->scroll_offset();
605 { 507 layer_tree_host()->root_layer()->SetScrollOffset(offset + scroll_delta);
606 gfx::Vector2d offset = layer_tree_host()->root_layer()->scroll_offset(); 508 layer_tree_host()->SetPageScaleFactorAndLimits(scale, 0.5f, 2.f);
607 layer_tree_host()->root_layer()->SetScrollOffset(offset + scrollDelta); 509 }
608 layer_tree_host()->SetPageScaleFactorAndLimits(scale, 0.5, 2); 510
609 } 511 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
610 512 impl->ProcessScrollDeltas();
611 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 513 // We get one commit before the first draw, and the animation doesn't
612 { 514 // happen until the second draw.
613 impl->ProcessScrollDeltas(); 515 if (impl->active_tree()->source_frame_number() == 1) {
614 // We get one commit before the first draw, and the animation doesn't ha ppen until the second draw. 516 EXPECT_EQ(1.25f, impl->active_tree()->page_scale_factor());
615 if (impl->active_tree()->source_frame_number() == 1) { 517 EndTest();
616 EXPECT_EQ(1.25, impl->active_tree()->page_scale_factor()); 518 } else {
617 EndTest(); 519 PostSetNeedsRedrawToMainThread();
618 } else 520 }
619 PostSetNeedsRedrawToMainThread(); 521 }
620 } 522
621 523 virtual void AfterTest() OVERRIDE {}
622 virtual void AfterTest() OVERRIDE 524
623 { 525 private:
624 } 526 bool animation_requested_;
625
626 private:
627 bool m_animationRequested;
628 }; 527 };
629 528
630 // TODO(aelias): This test is currently broken: http://crbug.com/178295 529 // TODO(aelias): This test is currently broken: http://crbug.com/178295
631 //MULTI_THREAD_TEST_F(LayerTreeHostTestStartPageScaleAnimation) 530 // MULTI_THREAD_TEST_F(LayerTreeHostTestStartPageScaleAnimation);
632 531
633 class LayerTreeHostTestSetVisible : public LayerTreeHostTest { 532 class LayerTreeHostTestSetVisible : public LayerTreeHostTest {
634 public: 533 public:
635 534
636 LayerTreeHostTestSetVisible() 535 LayerTreeHostTestSetVisible() : num_draws_(0) {}
637 : m_numDraws(0) 536
638 { 537 virtual void BeginTest() OVERRIDE {
639 } 538 PostSetNeedsCommitToMainThread();
640 539 PostSetVisibleToMainThread(false);
641 virtual void BeginTest() OVERRIDE 540 // This is suppressed while we're invisible.
642 { 541 PostSetNeedsRedrawToMainThread();
643 PostSetNeedsCommitToMainThread(); 542 // Triggers the redraw.
644 PostSetVisibleToMainThread(false); 543 PostSetVisibleToMainThread(true);
645 PostSetNeedsRedrawToMainThread(); // This is suppressed while we're invi sible. 544 }
646 PostSetVisibleToMainThread(true); // Triggers the redraw. 545
647 } 546 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
648 547 EXPECT_TRUE(impl->visible());
649 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 548 ++num_draws_;
650 { 549 EndTest();
651 EXPECT_TRUE(impl->visible()); 550 }
652 ++m_numDraws; 551
653 EndTest(); 552 virtual void AfterTest() OVERRIDE { EXPECT_EQ(1, num_draws_); }
654 } 553
655 554 private:
656 virtual void AfterTest() OVERRIDE 555 int num_draws_;
657 { 556 };
658 EXPECT_EQ(1, m_numDraws); 557
659 } 558 MULTI_THREAD_TEST_F(LayerTreeHostTestSetVisible);
660
661 private:
662 int m_numDraws;
663 };
664
665 MULTI_THREAD_TEST_F(LayerTreeHostTestSetVisible)
666 559
667 class TestOpacityChangeLayerDelegate : public ContentLayerClient { 560 class TestOpacityChangeLayerDelegate : public ContentLayerClient {
668 public: 561 public:
669 TestOpacityChangeLayerDelegate() 562 TestOpacityChangeLayerDelegate() : test_layer_(0) {}
670 : m_testLayer(0) 563
671 { 564 void SetTestLayer(Layer* test_layer) { test_layer_ = test_layer; }
672 } 565
673 566 virtual void PaintContents(SkCanvas*, gfx::Rect, gfx::RectF*) OVERRIDE {
674 void setTestLayer(Layer* testLayer) 567 // Set layer opacity to 0.
675 { 568 if (test_layer_)
676 m_testLayer = testLayer; 569 test_layer_->SetOpacity(0.f);
677 } 570 }
678 571 virtual void DidChangeLayerCanUseLCDText() OVERRIDE {}
679 virtual void PaintContents(SkCanvas*, gfx::Rect, gfx::RectF*) OVERRIDE 572
680 { 573 private:
681 // Set layer opacity to 0. 574 Layer* test_layer_;
682 if (m_testLayer)
683 m_testLayer->SetOpacity(0);
684 }
685 virtual void DidChangeLayerCanUseLCDText() OVERRIDE {}
686
687 private:
688 Layer* m_testLayer;
689 }; 575 };
690 576
691 class ContentLayerWithUpdateTracking : public ContentLayer { 577 class ContentLayerWithUpdateTracking : public ContentLayer {
692 public: 578 public:
693 static scoped_refptr<ContentLayerWithUpdateTracking> Create(ContentLayerClie nt* client) { return make_scoped_refptr(new ContentLayerWithUpdateTracking(clien t)); } 579 static scoped_refptr<ContentLayerWithUpdateTracking> Create(
694 580 ContentLayerClient* client) {
695 int paintContentsCount() { return m_paintContentsCount; } 581 return make_scoped_refptr(new ContentLayerWithUpdateTracking(client));
696 void resetPaintContentsCount() { m_paintContentsCount = 0; } 582 }
697 583
698 virtual void Update(ResourceUpdateQueue* queue, const OcclusionTracker* occl usion, RenderingStats* stats) OVERRIDE 584 int PaintContentsCount() { return paint_contents_count_; }
699 { 585 void ResetPaintContentsCount() { paint_contents_count_ = 0; }
700 ContentLayer::Update(queue, occlusion, stats); 586
701 m_paintContentsCount++; 587 virtual void Update(ResourceUpdateQueue* queue,
702 } 588 const OcclusionTracker* occlusion,
703 589 RenderingStats* stats) OVERRIDE {
704 private: 590 ContentLayer::Update(queue, occlusion, stats);
705 explicit ContentLayerWithUpdateTracking(ContentLayerClient* client) 591 paint_contents_count_++;
706 : ContentLayer(client) 592 }
707 , m_paintContentsCount(0) 593
708 { 594 private:
709 SetAnchorPoint(gfx::PointF(0, 0)); 595 explicit ContentLayerWithUpdateTracking(ContentLayerClient* client)
710 SetBounds(gfx::Size(10, 10)); 596 : ContentLayer(client), paint_contents_count_(0) {
711 SetIsDrawable(true); 597 SetAnchorPoint(gfx::PointF(0.f, 0.f));
712 } 598 SetBounds(gfx::Size(10, 10));
713 virtual ~ContentLayerWithUpdateTracking() 599 SetIsDrawable(true);
714 { 600 }
715 } 601 virtual ~ContentLayerWithUpdateTracking() {}
716 602
717 int m_paintContentsCount; 603 int paint_contents_count_;
718 }; 604 };
719 605
720 // Layer opacity change during paint should not prevent compositor resources fro m being updated during commit. 606 // Layer opacity change during paint should not prevent compositor resources
607 // from being updated during commit.
721 class LayerTreeHostTestOpacityChange : public LayerTreeHostTest { 608 class LayerTreeHostTestOpacityChange : public LayerTreeHostTest {
722 public: 609 public:
723 LayerTreeHostTestOpacityChange() 610 LayerTreeHostTestOpacityChange()
724 : m_testOpacityChangeDelegate() 611 : test_opacity_change_delegate_(),
725 , m_updateCheckLayer(ContentLayerWithUpdateTracking::Create(&m_testOpaci tyChangeDelegate)) 612 update_check_layer_(ContentLayerWithUpdateTracking::Create(
726 { 613 &test_opacity_change_delegate_)) {
727 m_testOpacityChangeDelegate.setTestLayer(m_updateCheckLayer.get()); 614 test_opacity_change_delegate_.SetTestLayer(update_check_layer_.get());
728 } 615 }
729 616
730 virtual void BeginTest() OVERRIDE 617 virtual void BeginTest() OVERRIDE {
731 { 618 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10));
732 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10)) ; 619 layer_tree_host()->root_layer()->AddChild(update_check_layer_);
733 layer_tree_host()->root_layer()->AddChild(m_updateCheckLayer); 620
734 621 PostSetNeedsCommitToMainThread();
735 PostSetNeedsCommitToMainThread(); 622 }
736 } 623
737 624 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
738 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE 625 EndTest();
739 { 626 }
740 EndTest(); 627
741 } 628 virtual void AfterTest() OVERRIDE {
742 629 // Update() should have been called once.
743 virtual void AfterTest() OVERRIDE 630 EXPECT_EQ(1, update_check_layer_->PaintContentsCount());
744 { 631
745 // Update() should have been called once. 632 // clear update_check_layer_ so LayerTreeHost dies.
746 EXPECT_EQ(1, m_updateCheckLayer->paintContentsCount()); 633 update_check_layer_ = NULL;
747 634 }
748 // clear m_updateCheckLayer so LayerTreeHost dies. 635
749 m_updateCheckLayer = NULL; 636 private:
750 } 637 TestOpacityChangeLayerDelegate test_opacity_change_delegate_;
751 638 scoped_refptr<ContentLayerWithUpdateTracking> update_check_layer_;
752 private: 639 };
753 TestOpacityChangeLayerDelegate m_testOpacityChangeDelegate; 640
754 scoped_refptr<ContentLayerWithUpdateTracking> m_updateCheckLayer; 641 MULTI_THREAD_TEST_F(LayerTreeHostTestOpacityChange);
755 };
756
757 MULTI_THREAD_TEST_F(LayerTreeHostTestOpacityChange)
758 642
759 class NoScaleContentLayer : public ContentLayer { 643 class NoScaleContentLayer : public ContentLayer {
760 public: 644 public:
761 static scoped_refptr<NoScaleContentLayer> Create(ContentLayerClient* client) { return make_scoped_refptr(new NoScaleContentLayer(client)); } 645 static scoped_refptr<NoScaleContentLayer> Create(ContentLayerClient* client) {
762 646 return make_scoped_refptr(new NoScaleContentLayer(client));
763 virtual void CalculateContentsScale( 647 }
764 float ideal_contents_scale, 648
765 bool animating_transform_to_screen, 649 virtual void CalculateContentsScale(float ideal_contents_scale,
766 float* contents_scale_x, 650 bool animating_transform_to_screen,
767 float* contents_scale_y, 651 float* contents_scale_x,
768 gfx::Size* contentBounds) OVERRIDE 652 float* contents_scale_y,
769 { 653 gfx::Size* contentBounds) OVERRIDE {
770 // Skip over the ContentLayer's method to the base Layer class. 654 // Skip over the ContentLayer's method to the base Layer class.
771 Layer::CalculateContentsScale( 655 Layer::CalculateContentsScale(ideal_contents_scale,
772 ideal_contents_scale, 656 animating_transform_to_screen,
773 animating_transform_to_screen, 657 contents_scale_x,
774 contents_scale_x, 658 contents_scale_y,
775 contents_scale_y, 659 contentBounds);
776 contentBounds); 660 }
777 } 661
778 662 private:
779 private: 663 explicit NoScaleContentLayer(ContentLayerClient* client)
780 explicit NoScaleContentLayer(ContentLayerClient* client) 664 : ContentLayer(client) {}
781 : ContentLayer(client) { } 665 virtual ~NoScaleContentLayer() {}
782 virtual ~NoScaleContentLayer() { } 666 };
783 }; 667
784 668 class LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers
785 class LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers : public LayerTr eeHostTest { 669 : public LayerTreeHostTest {
786 public: 670 public:
787 671 LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers()
788 LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers() 672 : root_layer_(NoScaleContentLayer::Create(&client_)),
789 : m_rootLayer(NoScaleContentLayer::Create(&client_)) 673 child_layer_(ContentLayer::Create(&client_)) {}
790 , m_childLayer(ContentLayer::Create(&client_)) 674
791 { 675 virtual void BeginTest() OVERRIDE {
792 } 676 layer_tree_host()->SetViewportSize(gfx::Size(40, 40), gfx::Size(60, 60));
793 677 layer_tree_host()->SetDeviceScaleFactor(1.5);
794 virtual void BeginTest() OVERRIDE 678 EXPECT_EQ(gfx::Size(40, 40), layer_tree_host()->layout_viewport_size());
795 { 679 EXPECT_EQ(gfx::Size(60, 60), layer_tree_host()->device_viewport_size());
796 layer_tree_host()->SetViewportSize(gfx::Size(40, 40), gfx::Size(60, 60)) ; 680
797 layer_tree_host()->SetDeviceScaleFactor(1.5); 681 root_layer_->AddChild(child_layer_);
798 EXPECT_EQ(gfx::Size(40, 40), layer_tree_host()->layout_viewport_size()); 682
799 EXPECT_EQ(gfx::Size(60, 60), layer_tree_host()->device_viewport_size()); 683 root_layer_->SetIsDrawable(true);
800 684 root_layer_->SetBounds(gfx::Size(30, 30));
801 m_rootLayer->AddChild(m_childLayer); 685 root_layer_->SetAnchorPoint(gfx::PointF(0.f, 0.f));
802 686
803 m_rootLayer->SetIsDrawable(true); 687 child_layer_->SetIsDrawable(true);
804 m_rootLayer->SetBounds(gfx::Size(30, 30)); 688 child_layer_->SetPosition(gfx::Point(2, 2));
805 m_rootLayer->SetAnchorPoint(gfx::PointF(0, 0)); 689 child_layer_->SetBounds(gfx::Size(10, 10));
806 690 child_layer_->SetAnchorPoint(gfx::PointF(0.f, 0.f));
807 m_childLayer->SetIsDrawable(true); 691
808 m_childLayer->SetPosition(gfx::Point(2, 2)); 692 layer_tree_host()->SetRootLayer(root_layer_);
809 m_childLayer->SetBounds(gfx::Size(10, 10)); 693
810 m_childLayer->SetAnchorPoint(gfx::PointF(0, 0)); 694 ASSERT_TRUE(layer_tree_host()->InitializeRendererIfNeeded());
811 695 ResourceUpdateQueue queue;
812 layer_tree_host()->SetRootLayer(m_rootLayer); 696 layer_tree_host()->UpdateLayers(&queue, std::numeric_limits<size_t>::max());
813 697 PostSetNeedsCommitToMainThread();
814 ASSERT_TRUE(layer_tree_host()->InitializeRendererIfNeeded()); 698 }
815 ResourceUpdateQueue queue; 699
816 layer_tree_host()->UpdateLayers(&queue, std::numeric_limits<size_t>::max ()); 700 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
817 PostSetNeedsCommitToMainThread(); 701 // Should only do one commit.
818 } 702 EXPECT_EQ(0, impl->active_tree()->source_frame_number());
819 703 // Device scale factor should come over to impl.
820 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 704 EXPECT_NEAR(impl->device_scale_factor(), 1.5f, 0.00001f);
821 { 705
822 // Should only do one commit. 706 // Both layers are on impl.
823 EXPECT_EQ(0, impl->active_tree()->source_frame_number()); 707 ASSERT_EQ(1u, impl->active_tree()->root_layer()->children().size());
824 // Device scale factor should come over to impl. 708
825 EXPECT_NEAR(impl->device_scale_factor(), 1.5, 0.00001); 709 // Device viewport is scaled.
826 710 EXPECT_EQ(gfx::Size(40, 40), impl->layout_viewport_size());
827 // Both layers are on impl. 711 EXPECT_EQ(gfx::Size(60, 60), impl->device_viewport_size());
828 ASSERT_EQ(1u, impl->active_tree()->root_layer()->children().size()); 712
829 713 LayerImpl* root = impl->active_tree()->root_layer();
830 // Device viewport is scaled. 714 LayerImpl* child = impl->active_tree()->root_layer()->children()[0];
831 EXPECT_EQ(gfx::Size(40, 40), impl->layout_viewport_size()); 715
832 EXPECT_EQ(gfx::Size(60, 60), impl->device_viewport_size()); 716 // Positions remain in layout pixels.
833 717 EXPECT_EQ(gfx::Point(0, 0), root->position());
834 LayerImpl* root = impl->active_tree()->root_layer(); 718 EXPECT_EQ(gfx::Point(2, 2), child->position());
835 LayerImpl* child = impl->active_tree()->root_layer()->children()[0]; 719
836 720 // Compute all the layer transforms for the frame.
837 // Positions remain in layout pixels. 721 LayerTreeHostImpl::FrameData frame_data;
838 EXPECT_EQ(gfx::Point(0, 0), root->position()); 722 impl->PrepareToDraw(&frame_data);
839 EXPECT_EQ(gfx::Point(2, 2), child->position()); 723 impl->DidDrawAllLayers(frame_data);
840 724
841 // Compute all the layer transforms for the frame. 725 const LayerTreeHostImpl::LayerList& render_surface_layer_list =
842 LayerTreeHostImpl::FrameData frameData; 726 *frame_data.render_surface_layer_list;
843 impl->PrepareToDraw(&frameData); 727
844 impl->DidDrawAllLayers(frameData); 728 // Both layers should be drawing into the root render surface.
845 729 ASSERT_EQ(1u, render_surface_layer_list.size());
846 const LayerTreeHostImpl::LayerList& renderSurfaceLayerList = 730 ASSERT_EQ(root->render_surface(),
847 *frameData.render_surface_layer_list; 731 render_surface_layer_list[0]->render_surface());
848 732 ASSERT_EQ(2u, root->render_surface()->layer_list().size());
849 // Both layers should be drawing into the root render surface. 733
850 ASSERT_EQ(1u, renderSurfaceLayerList.size()); 734 // The root render surface is the size of the viewport.
851 ASSERT_EQ(root->render_surface(), renderSurfaceLayerList[0]->render_surf ace()); 735 EXPECT_RECT_EQ(gfx::Rect(0, 0, 60, 60),
852 ASSERT_EQ(2u, root->render_surface()->layer_list().size()); 736 root->render_surface()->content_rect());
853 737
854 // The root render surface is the size of the viewport. 738 // The content bounds of the child should be scaled.
855 EXPECT_RECT_EQ(gfx::Rect(0, 0, 60, 60), root->render_surface()->content_ rect()); 739 gfx::Size child_bounds_scaled =
856 740 gfx::ToCeiledSize(gfx::ScaleSize(child->bounds(), 1.5));
857 // The content bounds of the child should be scaled. 741 EXPECT_EQ(child_bounds_scaled, child->content_bounds());
858 gfx::Size childBoundsScaled = gfx::ToCeiledSize(gfx::ScaleSize(child->bo unds(), 1.5)); 742
859 EXPECT_EQ(childBoundsScaled, child->content_bounds()); 743 gfx::Transform scale_transform;
860 744 scale_transform.Scale(impl->device_scale_factor(),
861 gfx::Transform scaleTransform; 745 impl->device_scale_factor());
862 scaleTransform.Scale(impl->device_scale_factor(), impl->device_scale_fac tor()); 746
863 747 // The root layer is scaled by 2x.
864 // The root layer is scaled by 2x. 748 gfx::Transform root_screen_space_transform = scale_transform;
865 gfx::Transform rootScreenSpaceTransform = scaleTransform; 749 gfx::Transform root_draw_transform = scale_transform;
866 gfx::Transform rootDrawTransform = scaleTransform; 750
867 751 EXPECT_EQ(root_draw_transform, root->draw_transform());
868 EXPECT_EQ(rootDrawTransform, root->draw_transform()); 752 EXPECT_EQ(root_screen_space_transform, root->screen_space_transform());
869 EXPECT_EQ(rootScreenSpaceTransform, root->screen_space_transform()); 753
870 754 // The child is at position 2,2, which is transformed to 3,3 after the scale
871 // The child is at position 2,2, which is transformed to 3,3 after the s cale 755 gfx::Transform child_screen_space_transform;
872 gfx::Transform childScreenSpaceTransform; 756 child_screen_space_transform.Translate(3.f, 3.f);
873 childScreenSpaceTransform.Translate(3, 3); 757 gfx::Transform child_draw_transform = child_screen_space_transform;
874 gfx::Transform childDrawTransform = childScreenSpaceTransform; 758
875 759 EXPECT_TRANSFORMATION_MATRIX_EQ(child_draw_transform,
876 EXPECT_TRANSFORMATION_MATRIX_EQ(childDrawTransform, child->draw_transfor m()); 760 child->draw_transform());
877 EXPECT_TRANSFORMATION_MATRIX_EQ(childScreenSpaceTransform, child->screen _space_transform()); 761 EXPECT_TRANSFORMATION_MATRIX_EQ(child_screen_space_transform,
878 762 child->screen_space_transform());
879 EndTest(); 763
880 } 764 EndTest();
881 765 }
882 virtual void AfterTest() OVERRIDE 766
883 { 767 virtual void AfterTest() OVERRIDE {
884 m_rootLayer = NULL; 768 root_layer_ = NULL;
885 m_childLayer = NULL; 769 child_layer_ = NULL;
886 } 770 }
887 771
888 private: 772 private:
889 FakeContentLayerClient client_; 773 FakeContentLayerClient client_;
890 scoped_refptr<NoScaleContentLayer> m_rootLayer; 774 scoped_refptr<NoScaleContentLayer> root_layer_;
891 scoped_refptr<ContentLayer> m_childLayer; 775 scoped_refptr<ContentLayer> child_layer_;
892 }; 776 };
893 777
894 MULTI_THREAD_TEST_F(LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers) 778 MULTI_THREAD_TEST_F(LayerTreeHostTestDeviceScaleFactorScalesViewportAndLayers);
895 779
896 // Verify atomicity of commits and reuse of textures. 780 // Verify atomicity of commits and reuse of textures.
897 class LayerTreeHostTestAtomicCommit : public LayerTreeHostTest { 781 class LayerTreeHostTestAtomicCommit : public LayerTreeHostTest {
898 public: 782 public:
899 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE 783 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
784 // Make sure partial texture updates are turned off.
785 settings->max_partial_texture_updates = 0;
786 // Linear fade animator prevents scrollbars from drawing immediately.
787 settings->use_linear_fade_scrollbar_animator = false;
788 }
789
790 virtual void SetupTree() OVERRIDE {
791 layer_ = FakeContentLayer::Create(&client_);
792 layer_->SetBounds(gfx::Size(10, 20));
793
794 bool paint_scrollbar = true;
795 bool has_thumb = false;
796 scrollbar_ =
797 FakeScrollbarLayer::Create(paint_scrollbar, has_thumb, layer_->id());
798 scrollbar_->SetPosition(gfx::Point(0, 10));
799 scrollbar_->SetBounds(gfx::Size(10, 10));
800
801 layer_->AddChild(scrollbar_);
802
803 layer_tree_host()->SetRootLayer(layer_);
804 LayerTreeHostTest::SetupTree();
805 }
806
807 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
808
809 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
810 ASSERT_EQ(0u, layer_tree_host()->settings().max_partial_texture_updates);
811
812 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D*>(
813 impl->output_surface()->context3d());
814
815 switch (impl->active_tree()->source_frame_number()) {
816 case 0:
817 // Number of textures should be one for each layer
818 ASSERT_EQ(2, context->NumTextures());
819 // Number of textures used for commit should be one for each layer.
820 EXPECT_EQ(2, context->NumUsedTextures());
821 // Verify that used texture is correct.
822 EXPECT_TRUE(context->UsedTexture(context->TextureAt(0)));
823 EXPECT_TRUE(context->UsedTexture(context->TextureAt(1)));
824
825 context->ResetUsedTextures();
826 PostSetNeedsCommitToMainThread();
827 break;
828 case 1:
829 // Number of textures should be doubled as the first textures
830 // are used by impl thread and cannot by used for update.
831 ASSERT_EQ(4, context->NumTextures());
832 // Number of textures used for commit should still be
833 // one for each layer.
834 EXPECT_EQ(2, context->NumUsedTextures());
835 // First textures should not have been used.
836 EXPECT_FALSE(context->UsedTexture(context->TextureAt(0)));
837 EXPECT_FALSE(context->UsedTexture(context->TextureAt(1)));
838 // New textures should have been used.
839 EXPECT_TRUE(context->UsedTexture(context->TextureAt(2)));
840 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
841
842 context->ResetUsedTextures();
843 PostSetNeedsCommitToMainThread();
844 break;
845 case 2:
846 EndTest();
847 break;
848 default:
849 NOTREACHED();
850 break;
851 }
852 }
853
854 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
855 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D*>(
856 impl->output_surface()->context3d());
857
858 // Number of textures used for draw should always be one for each layer.
859 EXPECT_EQ(2, context->NumUsedTextures());
860 context->ResetUsedTextures();
861 }
862
863 virtual void Layout() OVERRIDE {
864 layer_->SetNeedsDisplay();
865 scrollbar_->SetNeedsDisplay();
866 }
867
868 virtual void AfterTest() OVERRIDE {}
869
870 private:
871 FakeContentLayerClient client_;
872 scoped_refptr<FakeContentLayer> layer_;
873 scoped_refptr<FakeScrollbarLayer> scrollbar_;
874 };
875
876 MULTI_THREAD_TEST_F(LayerTreeHostTestAtomicCommit);
877
878 static void SetLayerPropertiesForTesting(Layer* layer,
879 Layer* parent,
880 const gfx::Transform& transform,
881 gfx::PointF anchor,
882 gfx::PointF position,
883 gfx::Size bounds,
884 bool opaque) {
885 layer->RemoveAllChildren();
886 if (parent)
887 parent->AddChild(layer);
888 layer->SetTransform(transform);
889 layer->SetAnchorPoint(anchor);
890 layer->SetPosition(position);
891 layer->SetBounds(bounds);
892 layer->SetContentsOpaque(opaque);
893 }
894
895 class LayerTreeHostTestAtomicCommitWithPartialUpdate
896 : public LayerTreeHostTest {
897 public:
898 LayerTreeHostTestAtomicCommitWithPartialUpdate() : num_commits_(0) {}
899
900 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
901 // Allow one partial texture update.
902 settings->max_partial_texture_updates = 1;
903 // Linear fade animator prevents scrollbars from drawing immediately.
904 settings->use_linear_fade_scrollbar_animator = false;
905 }
906
907 virtual void SetupTree() OVERRIDE {
908 parent_ = FakeContentLayer::Create(&client_);
909 parent_->SetBounds(gfx::Size(10, 20));
910
911 child_ = FakeContentLayer::Create(&client_);
912 child_->SetPosition(gfx::Point(0, 10));
913 child_->SetBounds(gfx::Size(3, 10));
914
915 bool paint_scrollbar = true;
916 bool has_thumb = false;
917 scrollbar_with_paints_ =
918 FakeScrollbarLayer::Create(paint_scrollbar, has_thumb, parent_->id());
919 scrollbar_with_paints_->SetPosition(gfx::Point(3, 10));
920 scrollbar_with_paints_->SetBounds(gfx::Size(3, 10));
921
922 paint_scrollbar = false;
923 scrollbar_without_paints_ =
924 FakeScrollbarLayer::Create(paint_scrollbar, has_thumb, parent_->id());
925 scrollbar_without_paints_->SetPosition(gfx::Point(6, 10));
926 scrollbar_without_paints_->SetBounds(gfx::Size(3, 10));
927
928 parent_->AddChild(child_);
929 parent_->AddChild(scrollbar_with_paints_);
930 parent_->AddChild(scrollbar_without_paints_);
931
932 layer_tree_host()->SetRootLayer(parent_);
933 LayerTreeHostTest::SetupTree();
934 }
935
936 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
937
938 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
939 ASSERT_EQ(1u, layer_tree_host()->settings().max_partial_texture_updates);
940
941 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D*>(
942 impl->output_surface()->context3d());
943
944 switch (impl->active_tree()->source_frame_number()) {
945 case 0:
946 // Number of textures should be one for each layer.
947 ASSERT_EQ(4, context->NumTextures());
948 // Number of textures used for commit should be one for each layer.
949 EXPECT_EQ(4, context->NumUsedTextures());
950 // Verify that used textures are correct.
951 EXPECT_TRUE(context->UsedTexture(context->TextureAt(0)));
952 EXPECT_TRUE(context->UsedTexture(context->TextureAt(1)));
953 EXPECT_TRUE(context->UsedTexture(context->TextureAt(2)));
954 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
955
956 context->ResetUsedTextures();
957 PostSetNeedsCommitToMainThread();
958 break;
959 case 1:
960 // Number of textures should be two for each content layer and one
961 // for each scrollbar, since they always do a partial update.
962 ASSERT_EQ(6, context->NumTextures());
963 // Number of textures used for commit should be one for each content
964 // layer, and one for the scrollbar layer that paints.
965 EXPECT_EQ(3, context->NumUsedTextures());
966
967 // First content textures should not have been used.
968 EXPECT_FALSE(context->UsedTexture(context->TextureAt(0)));
969 EXPECT_FALSE(context->UsedTexture(context->TextureAt(1)));
970 // The non-painting scrollbar's texture wasn't updated.
971 EXPECT_FALSE(context->UsedTexture(context->TextureAt(2)));
972 // The painting scrollbar's partial update texture was used.
973 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
974 // New textures should have been used.
975 EXPECT_TRUE(context->UsedTexture(context->TextureAt(4)));
976 EXPECT_TRUE(context->UsedTexture(context->TextureAt(5)));
977
978 context->ResetUsedTextures();
979 PostSetNeedsCommitToMainThread();
980 break;
981 case 2:
982 // Number of textures should be two for each content layer and one
983 // for each scrollbar, since they always do a partial update.
984 ASSERT_EQ(6, context->NumTextures());
985 // Number of textures used for commit should be one for each content
986 // layer, and one for the scrollbar layer that paints.
987 EXPECT_EQ(3, context->NumUsedTextures());
988
989 // The non-painting scrollbar's texture wasn't updated.
990 EXPECT_FALSE(context->UsedTexture(context->TextureAt(2)));
991 // The painting scrollbar does a partial update.
992 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
993 // One content layer does a partial update also.
994 EXPECT_TRUE(context->UsedTexture(context->TextureAt(4)));
995 EXPECT_FALSE(context->UsedTexture(context->TextureAt(5)));
996
997 context->ResetUsedTextures();
998 PostSetNeedsCommitToMainThread();
999 break;
1000 case 3:
1001 // No textures should be used for commit.
1002 EXPECT_EQ(0, context->NumUsedTextures());
1003
1004 context->ResetUsedTextures();
1005 PostSetNeedsCommitToMainThread();
1006 break;
1007 case 4:
1008 // Number of textures used for commit should be two. One for the
1009 // content layer, and one for the painting scrollbar. The
1010 // non-painting scrollbar doesn't update its texture.
1011 EXPECT_EQ(2, context->NumUsedTextures());
1012
1013 context->ResetUsedTextures();
1014 PostSetNeedsCommitToMainThread();
1015 break;
1016 case 5:
1017 EndTest();
1018 break;
1019 default:
1020 NOTREACHED();
1021 break;
1022 }
1023 }
1024
1025 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1026 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D*>(
1027 impl->output_surface()->context3d());
1028
1029 // Number of textures used for drawing should one per layer except for
1030 // frame 3 where the viewport only contains one layer.
1031 if (impl->active_tree()->source_frame_number() == 3)
1032 EXPECT_EQ(1, context->NumUsedTextures());
1033 else
1034 EXPECT_EQ(4, context->NumUsedTextures());
1035
1036 context->ResetUsedTextures();
1037 }
1038
1039 virtual void Layout() OVERRIDE {
1040 switch (num_commits_++) {
1041 case 0:
1042 case 1:
1043 parent_->SetNeedsDisplay();
1044 child_->SetNeedsDisplay();
1045 scrollbar_with_paints_->SetNeedsDisplay();
1046 scrollbar_without_paints_->SetNeedsDisplay();
1047 break;
1048 case 2:
1049 // Damage part of layers.
1050 parent_->SetNeedsDisplayRect(gfx::RectF(0.f, 0.f, 5.f, 5.f));
1051 child_->SetNeedsDisplayRect(gfx::RectF(0.f, 0.f, 5.f, 5.f));
1052 scrollbar_with_paints_->SetNeedsDisplayRect(
1053 gfx::RectF(0.f, 0.f, 5.f, 5.f));
1054 scrollbar_without_paints_->SetNeedsDisplayRect(
1055 gfx::RectF(0.f, 0.f, 5.f, 5.f));
1056 break;
1057 case 3:
1058 child_->SetNeedsDisplay();
1059 scrollbar_with_paints_->SetNeedsDisplay();
1060 scrollbar_without_paints_->SetNeedsDisplay();
1061 layer_tree_host()->SetViewportSize(gfx::Size(10, 10),
1062 gfx::Size(10, 10));
1063 break;
1064 case 4:
1065 layer_tree_host()->SetViewportSize(gfx::Size(10, 20),
1066 gfx::Size(10, 20));
1067 break;
1068 case 5:
1069 break;
1070 default:
1071 NOTREACHED();
1072 break;
1073 }
1074 }
1075
1076 virtual void AfterTest() OVERRIDE {}
1077
1078 private:
1079 FakeContentLayerClient client_;
1080 scoped_refptr<FakeContentLayer> parent_;
1081 scoped_refptr<FakeContentLayer> child_;
1082 scoped_refptr<FakeScrollbarLayer> scrollbar_with_paints_;
1083 scoped_refptr<FakeScrollbarLayer> scrollbar_without_paints_;
1084 int num_commits_;
1085 };
1086
1087 MULTI_THREAD_TEST_F(LayerTreeHostTestAtomicCommitWithPartialUpdate);
1088
1089 class LayerTreeHostTestFinishAllRendering : public LayerTreeHostTest {
1090 public:
1091 LayerTreeHostTestFinishAllRendering() : once_(false), draw_count_(0) {}
1092
1093 virtual void BeginTest() OVERRIDE {
1094 layer_tree_host()->SetNeedsRedraw();
1095 PostSetNeedsCommitToMainThread();
1096 }
1097
1098 virtual void DidCommitAndDrawFrame() OVERRIDE {
1099 if (once_)
1100 return;
1101 once_ = true;
1102 layer_tree_host()->SetNeedsRedraw();
1103 layer_tree_host()->AcquireLayerTextures();
900 { 1104 {
901 // Make sure partial texture updates are turned off. 1105 base::AutoLock lock(lock_);
902 settings->max_partial_texture_updates = 0; 1106 draw_count_ = 0;
903 // Linear fade animator prevents scrollbars from drawing immediately. 1107 }
904 settings->use_linear_fade_scrollbar_animator = false; 1108 layer_tree_host()->FinishAllRendering();
905 }
906
907 virtual void SetupTree() OVERRIDE
908 { 1109 {
909 m_layer = FakeContentLayer::Create(&client_); 1110 base::AutoLock lock(lock_);
910 m_layer->SetBounds(gfx::Size(10, 20)); 1111 EXPECT_EQ(0, draw_count_);
911 1112 }
912 bool paint_scrollbar = true; 1113 EndTest();
913 bool has_thumb = false; 1114 }
914 m_scrollbar = FakeScrollbarLayer::Create( 1115
915 paint_scrollbar, has_thumb, m_layer->id()); 1116 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
916 m_scrollbar->SetPosition(gfx::Point(0, 10)); 1117 base::AutoLock lock(lock_);
917 m_scrollbar->SetBounds(gfx::Size(10, 10)); 1118 ++draw_count_;
918 1119 }
919 m_layer->AddChild(m_scrollbar); 1120
920 1121 virtual void AfterTest() OVERRIDE {}
921 layer_tree_host()->SetRootLayer(m_layer); 1122
922 LayerTreeHostTest::SetupTree(); 1123 private:
923 } 1124 bool once_;
924 1125 base::Lock lock_;
925 virtual void BeginTest() OVERRIDE 1126 int draw_count_;
926 { 1127 };
1128
1129 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestFinishAllRendering);
1130
1131 class LayerTreeHostTestCompositeAndReadbackCleanup : public LayerTreeHostTest {
1132 public:
1133 LayerTreeHostTestCompositeAndReadbackCleanup() {}
1134
1135 virtual void BeginTest() OVERRIDE {
1136 Layer* root_layer = layer_tree_host()->root_layer();
1137
1138 char pixels[4];
1139 layer_tree_host()->CompositeAndReadback(static_cast<void*>(&pixels),
1140 gfx::Rect(0, 0, 1, 1));
1141 EXPECT_FALSE(root_layer->render_surface());
1142
1143 EndTest();
1144 }
1145
1146 virtual void AfterTest() OVERRIDE {}
1147 };
1148
1149 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadbackCleanup);
1150
1151 class LayerTreeHostTestSurfaceNotAllocatedForLayersOutsideMemoryLimit
1152 : public LayerTreeHostTest {
1153 public:
1154 LayerTreeHostTestSurfaceNotAllocatedForLayersOutsideMemoryLimit()
1155 : root_layer_(ContentLayerWithUpdateTracking::Create(&fake_delegate_)),
1156 surface_layer1_(
1157 ContentLayerWithUpdateTracking::Create(&fake_delegate_)),
1158 replica_layer1_(
1159 ContentLayerWithUpdateTracking::Create(&fake_delegate_)),
1160 surface_layer2_(
1161 ContentLayerWithUpdateTracking::Create(&fake_delegate_)),
1162 replica_layer2_(
1163 ContentLayerWithUpdateTracking::Create(&fake_delegate_)) {}
1164
1165 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
1166 settings->cache_render_pass_contents = true;
1167 }
1168
1169 virtual void BeginTest() OVERRIDE {
1170 layer_tree_host()->SetViewportSize(gfx::Size(100, 100),
1171 gfx::Size(100, 100));
1172
1173 root_layer_->SetBounds(gfx::Size(100, 100));
1174 surface_layer1_->SetBounds(gfx::Size(100, 100));
1175 surface_layer1_->SetForceRenderSurface(true);
1176 surface_layer1_->SetOpacity(0.5f);
1177 surface_layer2_->SetBounds(gfx::Size(100, 100));
1178 surface_layer2_->SetForceRenderSurface(true);
1179 surface_layer2_->SetOpacity(0.5f);
1180
1181 surface_layer1_->SetReplicaLayer(replica_layer1_.get());
1182 surface_layer2_->SetReplicaLayer(replica_layer2_.get());
1183
1184 root_layer_->AddChild(surface_layer1_);
1185 surface_layer1_->AddChild(surface_layer2_);
1186 layer_tree_host()->SetRootLayer(root_layer_);
1187
1188 PostSetNeedsCommitToMainThread();
1189 }
1190
1191 virtual void DrawLayersOnThread(LayerTreeHostImpl* host_impl) OVERRIDE {
1192 Renderer* renderer = host_impl->renderer();
1193 RenderPass::Id surface1_render_pass_id = host_impl->active_tree()
1194 ->root_layer()->children()[0]->render_surface()->RenderPassId();
1195 RenderPass::Id surface2_render_pass_id =
1196 host_impl->active_tree()->root_layer()->children()[0]->children()[0]
1197 ->render_surface()->RenderPassId();
1198
1199 switch (host_impl->active_tree()->source_frame_number()) {
1200 case 0:
1201 EXPECT_TRUE(renderer->HaveCachedResourcesForRenderPassId(
1202 surface1_render_pass_id));
1203 EXPECT_TRUE(renderer->HaveCachedResourcesForRenderPassId(
1204 surface2_render_pass_id));
1205
1206 // Reduce the memory limit to only fit the root layer and one render
1207 // surface. This prevents any contents drawing into surfaces
1208 // from being allocated.
1209 host_impl->SetManagedMemoryPolicy(
1210 ManagedMemoryPolicy(100 * 100 * 4 * 2));
1211 break;
1212 case 1:
1213 EXPECT_FALSE(renderer->HaveCachedResourcesForRenderPassId(
1214 surface1_render_pass_id));
1215 EXPECT_FALSE(renderer->HaveCachedResourcesForRenderPassId(
1216 surface2_render_pass_id));
1217
1218 EndTest();
1219 break;
1220 }
1221 }
1222
1223 virtual void AfterTest() OVERRIDE {
1224 EXPECT_EQ(2, root_layer_->PaintContentsCount());
1225 EXPECT_EQ(2, surface_layer1_->PaintContentsCount());
1226 EXPECT_EQ(2, surface_layer2_->PaintContentsCount());
1227
1228 // Clear layer references so LayerTreeHost dies.
1229 root_layer_ = NULL;
1230 surface_layer1_ = NULL;
1231 replica_layer1_ = NULL;
1232 surface_layer2_ = NULL;
1233 replica_layer2_ = NULL;
1234 }
1235
1236 private:
1237 FakeContentLayerClient fake_delegate_;
1238 scoped_refptr<ContentLayerWithUpdateTracking> root_layer_;
1239 scoped_refptr<ContentLayerWithUpdateTracking> surface_layer1_;
1240 scoped_refptr<ContentLayerWithUpdateTracking> replica_layer1_;
1241 scoped_refptr<ContentLayerWithUpdateTracking> surface_layer2_;
1242 scoped_refptr<ContentLayerWithUpdateTracking> replica_layer2_;
1243 };
1244
1245 SINGLE_AND_MULTI_THREAD_TEST_F(
1246 LayerTreeHostTestSurfaceNotAllocatedForLayersOutsideMemoryLimit);
1247
1248 class EvictionTestLayer : public Layer {
1249 public:
1250 static scoped_refptr<EvictionTestLayer> Create() {
1251 return make_scoped_refptr(new EvictionTestLayer());
1252 }
1253
1254 virtual void Update(ResourceUpdateQueue*,
1255 const OcclusionTracker*,
1256 RenderingStats*) OVERRIDE;
1257 virtual bool DrawsContent() const OVERRIDE { return true; }
1258
1259 virtual scoped_ptr<LayerImpl> CreateLayerImpl(LayerTreeImpl* tree_impl)
1260 OVERRIDE;
1261 virtual void PushPropertiesTo(LayerImpl* impl) OVERRIDE;
1262 virtual void SetTexturePriorities(const PriorityCalculator&) OVERRIDE;
1263
1264 bool HaveBackingTexture() const {
1265 return texture_.get() ? texture_->have_backing_texture() : false;
1266 }
1267
1268 private:
1269 EvictionTestLayer() : Layer() {}
1270 virtual ~EvictionTestLayer() {}
1271
1272 void CreateTextureIfNeeded() {
1273 if (texture_.get())
1274 return;
1275 texture_ = PrioritizedResource::Create(
1276 layer_tree_host()->contents_texture_manager());
1277 texture_->SetDimensions(gfx::Size(10, 10), GL_RGBA);
1278 bitmap_.setConfig(SkBitmap::kARGB_8888_Config, 10, 10);
1279 }
1280
1281 scoped_ptr<PrioritizedResource> texture_;
1282 SkBitmap bitmap_;
1283 };
1284
1285 class EvictionTestLayerImpl : public LayerImpl {
1286 public:
1287 static scoped_ptr<EvictionTestLayerImpl> Create(LayerTreeImpl* tree_impl,
1288 int id) {
1289 return make_scoped_ptr(new EvictionTestLayerImpl(tree_impl, id));
1290 }
1291 virtual ~EvictionTestLayerImpl() {}
1292
1293 virtual void AppendQuads(QuadSink* quad_sink,
1294 AppendQuadsData* append_quads_data) OVERRIDE {
1295 ASSERT_TRUE(has_texture_);
1296 ASSERT_NE(0u, layer_tree_impl()->resource_provider()->num_resources());
1297 }
1298
1299 void SetHasTexture(bool has_texture) { has_texture_ = has_texture; }
1300
1301 private:
1302 EvictionTestLayerImpl(LayerTreeImpl* tree_impl, int id)
1303 : LayerImpl(tree_impl, id), has_texture_(false) {}
1304
1305 bool has_texture_;
1306 };
1307
1308 void EvictionTestLayer::SetTexturePriorities(const PriorityCalculator&) {
1309 CreateTextureIfNeeded();
1310 if (!texture_.get())
1311 return;
1312 texture_->set_request_priority(PriorityCalculator::UIPriority(true));
1313 }
1314
1315 void EvictionTestLayer::Update(ResourceUpdateQueue* queue,
1316 const OcclusionTracker*,
1317 RenderingStats*) {
1318 CreateTextureIfNeeded();
1319 if (!texture_.get())
1320 return;
1321
1322 gfx::Rect full_rect(0, 0, 10, 10);
1323 ResourceUpdate upload = ResourceUpdate::Create(
1324 texture_.get(), &bitmap_, full_rect, full_rect, gfx::Vector2d());
1325 queue->AppendFullUpload(upload);
1326 }
1327
1328 scoped_ptr<LayerImpl> EvictionTestLayer::CreateLayerImpl(
1329 LayerTreeImpl* tree_impl) {
1330 return EvictionTestLayerImpl::Create(tree_impl, layer_id_)
1331 .PassAs<LayerImpl>();
1332 }
1333
1334 void EvictionTestLayer::PushPropertiesTo(LayerImpl* layer_impl) {
1335 Layer::PushPropertiesTo(layer_impl);
1336
1337 EvictionTestLayerImpl* test_layer_impl =
1338 static_cast<EvictionTestLayerImpl*>(layer_impl);
1339 test_layer_impl->SetHasTexture(texture_->have_backing_texture());
1340 }
1341
1342 class LayerTreeHostTestEvictTextures : public LayerTreeHostTest {
1343 public:
1344 LayerTreeHostTestEvictTextures()
1345 : layer_(EvictionTestLayer::Create()),
1346 impl_for_evict_textures_(0),
1347 num_commits_(0) {}
1348
1349 virtual void BeginTest() OVERRIDE {
1350 layer_tree_host()->SetRootLayer(layer_);
1351 layer_tree_host()->SetViewportSize(gfx::Size(10, 20), gfx::Size(10, 20));
1352
1353 gfx::Transform identity_matrix;
1354 SetLayerPropertiesForTesting(layer_.get(),
1355 0,
1356 identity_matrix,
1357 gfx::PointF(0.f, 0.f),
1358 gfx::PointF(0.f, 0.f),
1359 gfx::Size(10, 20),
1360 true);
1361
1362 PostSetNeedsCommitToMainThread();
1363 }
1364
1365 void PostEvictTextures() {
1366 DCHECK(ImplThread());
1367 ImplThread()->PostTask(
1368 base::Bind(&LayerTreeHostTestEvictTextures::EvictTexturesOnImplThread,
1369 base::Unretained(this)));
1370 }
1371
1372 void EvictTexturesOnImplThread() {
1373 DCHECK(impl_for_evict_textures_);
1374 impl_for_evict_textures_->EnforceManagedMemoryPolicy(
1375 ManagedMemoryPolicy(0));
1376 }
1377
1378 // Commit 1: Just commit and draw normally, then post an eviction at the end
1379 // that will trigger a commit.
1380 // Commit 2: Triggered by the eviction, let it go through and then set
1381 // needsCommit.
1382 // Commit 3: Triggered by the setNeedsCommit. In Layout(), post an eviction
1383 // task, which will be handled before the commit. Don't set needsCommit, it
1384 // should have been posted. A frame should not be drawn (note,
1385 // didCommitAndDrawFrame may be called anyway).
1386 // Commit 4: Triggered by the eviction, let it go through and then set
1387 // needsCommit.
1388 // Commit 5: Triggered by the setNeedsCommit, post an eviction task in
1389 // Layout(), a frame should not be drawn but a commit will be posted.
1390 // Commit 6: Triggered by the eviction, post an eviction task in
1391 // Layout(), which will be a noop, letting the commit (which recreates the
1392 // textures) go through and draw a frame, then end the test.
1393 //
1394 // Commits 1+2 test the eviction recovery path where eviction happens outside
1395 // of the beginFrame/commit pair.
1396 // Commits 3+4 test the eviction recovery path where eviction happens inside
1397 // the beginFrame/commit pair.
1398 // Commits 5+6 test the path where an eviction happens during the eviction
1399 // recovery path.
1400 virtual void DidCommitAndDrawFrame() OVERRIDE {
1401 switch (num_commits_) {
1402 case 1:
1403 EXPECT_TRUE(layer_->HaveBackingTexture());
1404 PostEvictTextures();
1405 break;
1406 case 2:
1407 EXPECT_TRUE(layer_->HaveBackingTexture());
1408 layer_tree_host()->SetNeedsCommit();
1409 break;
1410 case 3:
1411 break;
1412 case 4:
1413 EXPECT_TRUE(layer_->HaveBackingTexture());
1414 layer_tree_host()->SetNeedsCommit();
1415 break;
1416 case 5:
1417 break;
1418 case 6:
1419 EXPECT_TRUE(layer_->HaveBackingTexture());
1420 EndTest();
1421 break;
1422 default:
1423 NOTREACHED();
1424 break;
1425 }
1426 }
1427
1428 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1429 impl_for_evict_textures_ = impl;
1430 }
1431
1432 virtual void Layout() OVERRIDE {
1433 ++num_commits_;
1434 switch (num_commits_) {
1435 case 1:
1436 case 2:
1437 break;
1438 case 3:
1439 PostEvictTextures();
1440 break;
1441 case 4:
1442 // We couldn't check in didCommitAndDrawFrame on commit 3,
1443 // so check here.
1444 EXPECT_FALSE(layer_->HaveBackingTexture());
1445 break;
1446 case 5:
1447 PostEvictTextures();
1448 break;
1449 case 6:
1450 // We couldn't check in didCommitAndDrawFrame on commit 5,
1451 // so check here.
1452 EXPECT_FALSE(layer_->HaveBackingTexture());
1453 PostEvictTextures();
1454 break;
1455 default:
1456 NOTREACHED();
1457 break;
1458 }
1459 }
1460
1461 virtual void AfterTest() OVERRIDE {}
1462
1463 private:
1464 FakeContentLayerClient client_;
1465 scoped_refptr<EvictionTestLayer> layer_;
1466 LayerTreeHostImpl* impl_for_evict_textures_;
1467 int num_commits_;
1468 };
1469
1470 MULTI_THREAD_TEST_F(LayerTreeHostTestEvictTextures);
1471
1472 class LayerTreeHostTestContinuousCommit : public LayerTreeHostTest {
1473 public:
1474 LayerTreeHostTestContinuousCommit()
1475 : num_commit_complete_(0), num_draw_layers_(0) {}
1476
1477 virtual void BeginTest() OVERRIDE {
1478 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10));
1479 layer_tree_host()->root_layer()->SetBounds(gfx::Size(10, 10));
1480
1481 PostSetNeedsCommitToMainThread();
1482 }
1483
1484 virtual void DidCommit() OVERRIDE {
1485 if (num_draw_layers_ == 2)
1486 return;
1487 PostSetNeedsCommitToMainThread();
1488 }
1489
1490 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1491 if (num_draw_layers_ == 1)
1492 num_commit_complete_++;
1493 }
1494
1495 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1496 num_draw_layers_++;
1497 if (num_draw_layers_ == 2)
1498 EndTest();
1499 }
1500
1501 virtual void AfterTest() OVERRIDE {
1502 // Check that we didn't commit twice between first and second draw.
1503 EXPECT_EQ(1, num_commit_complete_);
1504 }
1505
1506 private:
1507 int num_commit_complete_;
1508 int num_draw_layers_;
1509 };
1510
1511 MULTI_THREAD_TEST_F(LayerTreeHostTestContinuousCommit);
1512
1513 class LayerTreeHostTestContinuousInvalidate : public LayerTreeHostTest {
1514 public:
1515 LayerTreeHostTestContinuousInvalidate()
1516 : num_commit_complete_(0), num_draw_layers_(0) {}
1517
1518 virtual void BeginTest() OVERRIDE {
1519 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10));
1520 layer_tree_host()->root_layer()->SetBounds(gfx::Size(10, 10));
1521
1522 content_layer_ = ContentLayer::Create(&fake_delegate_);
1523 content_layer_->SetBounds(gfx::Size(10, 10));
1524 content_layer_->SetPosition(gfx::PointF(0.f, 0.f));
1525 content_layer_->SetAnchorPoint(gfx::PointF(0.f, 0.f));
1526 content_layer_->SetIsDrawable(true);
1527 layer_tree_host()->root_layer()->AddChild(content_layer_);
1528
1529 PostSetNeedsCommitToMainThread();
1530 }
1531
1532 virtual void DidCommit() OVERRIDE {
1533 if (num_draw_layers_ == 2)
1534 return;
1535 content_layer_->SetNeedsDisplay();
1536 }
1537
1538 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1539 if (num_draw_layers_ == 1)
1540 num_commit_complete_++;
1541 }
1542
1543 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE {
1544 num_draw_layers_++;
1545 if (num_draw_layers_ == 2)
1546 EndTest();
1547 }
1548
1549 virtual void AfterTest() OVERRIDE {
1550 // Check that we didn't commit twice between first and second draw.
1551 EXPECT_EQ(1, num_commit_complete_);
1552
1553 // Clear layer references so LayerTreeHost dies.
1554 content_layer_ = NULL;
1555 }
1556
1557 private:
1558 FakeContentLayerClient fake_delegate_;
1559 scoped_refptr<Layer> content_layer_;
1560 int num_commit_complete_;
1561 int num_draw_layers_;
1562 };
1563
1564 MULTI_THREAD_TEST_F(LayerTreeHostTestContinuousInvalidate);
1565
1566 class LayerTreeHostTestDeferCommits : public LayerTreeHostTest {
1567 public:
1568 LayerTreeHostTestDeferCommits()
1569 : num_commits_deferred_(0), num_complete_commits_(0) {}
1570
1571 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
1572
1573 virtual void DidDeferCommit() OVERRIDE {
1574 num_commits_deferred_++;
1575 layer_tree_host()->SetDeferCommits(false);
1576 }
1577
1578 virtual void DidCommit() OVERRIDE {
1579 num_complete_commits_++;
1580 switch (num_complete_commits_) {
1581 case 1:
1582 EXPECT_EQ(0, num_commits_deferred_);
1583 layer_tree_host()->SetDeferCommits(true);
927 PostSetNeedsCommitToMainThread(); 1584 PostSetNeedsCommitToMainThread();
928 } 1585 break;
929 1586 case 2:
930 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 1587 EndTest();
931 { 1588 break;
932 ASSERT_EQ(0u, 1589 default:
933 layer_tree_host()->settings().max_partial_texture_updates); 1590 NOTREACHED();
934 1591 break;
935 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D *>(impl->output_surface()->context3d()); 1592 }
936 1593 }
937 switch (impl->active_tree()->source_frame_number()) { 1594
938 case 0: 1595 virtual void AfterTest() OVERRIDE {
939 // Number of textures should be one for each layer 1596 EXPECT_EQ(1, num_commits_deferred_);
940 ASSERT_EQ(2, context->NumTextures()); 1597 EXPECT_EQ(2, num_complete_commits_);
941 // Number of textures used for commit should be one for each layer. 1598 }
942 EXPECT_EQ(2, context->NumUsedTextures()); 1599
943 // Verify that used texture is correct. 1600 private:
944 EXPECT_TRUE(context->UsedTexture(context->TextureAt(0))); 1601 int num_commits_deferred_;
945 EXPECT_TRUE(context->UsedTexture(context->TextureAt(1))); 1602 int num_complete_commits_;
946 1603 };
947 context->ResetUsedTextures(); 1604
948 PostSetNeedsCommitToMainThread(); 1605 MULTI_THREAD_TEST_F(LayerTreeHostTestDeferCommits);
949 break; 1606
950 case 1: 1607 class LayerTreeHostWithProxy : public LayerTreeHost {
951 // Number of textures should be doubled as the first textures 1608 public:
952 // are used by impl thread and cannot by used for update. 1609 LayerTreeHostWithProxy(FakeLayerTreeHostClient* client,
953 ASSERT_EQ(4, context->NumTextures()); 1610 const LayerTreeSettings& settings,
954 // Number of textures used for commit should still be one for each l ayer. 1611 scoped_ptr<Proxy> proxy)
955 EXPECT_EQ(2, context->NumUsedTextures()); 1612 : LayerTreeHost(client, settings) {
956 // First textures should not have been used. 1613 EXPECT_TRUE(InitializeForTesting(proxy.Pass()));
957 EXPECT_FALSE(context->UsedTexture(context->TextureAt(0))); 1614 }
958 EXPECT_FALSE(context->UsedTexture(context->TextureAt(1))); 1615 };
959 // New textures should have been used. 1616
960 EXPECT_TRUE(context->UsedTexture(context->TextureAt(2))); 1617 TEST(LayerTreeHostTest, LimitPartialUpdates) {
961 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3))); 1618 // When partial updates are not allowed, max updates should be 0.
962 1619 {
963 context->ResetUsedTextures(); 1620 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
964 PostSetNeedsCommitToMainThread(); 1621
965 break; 1622 scoped_ptr<FakeProxy> proxy =
966 case 2: 1623 make_scoped_ptr(new FakeProxy(scoped_ptr<Thread>()));
967 EndTest(); 1624 proxy->GetRendererCapabilities().allow_partial_texture_updates = false;
968 break; 1625 proxy->SetMaxPartialTextureUpdates(5);
969 default: 1626
970 NOTREACHED(); 1627 LayerTreeSettings settings;
971 break; 1628 settings.max_partial_texture_updates = 10;
972 } 1629
973 } 1630 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
974 1631 EXPECT_TRUE(host.InitializeRendererIfNeeded());
975 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 1632
976 { 1633 EXPECT_EQ(0u, host.settings().max_partial_texture_updates);
977 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D *>(impl->output_surface()->context3d()); 1634 }
978 1635
979 // Number of textures used for draw should always be one for each layer. 1636 // When partial updates are allowed,
980 EXPECT_EQ(2, context->NumUsedTextures()); 1637 // max updates should be limited by the proxy.
981 context->ResetUsedTextures(); 1638 {
982 } 1639 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
983 1640
984 virtual void Layout() OVERRIDE 1641 scoped_ptr<FakeProxy> proxy =
985 { 1642 make_scoped_ptr(new FakeProxy(scoped_ptr<Thread>()));
986 m_layer->SetNeedsDisplay(); 1643 proxy->GetRendererCapabilities().allow_partial_texture_updates = true;
987 m_scrollbar->SetNeedsDisplay(); 1644 proxy->SetMaxPartialTextureUpdates(5);
988 } 1645
989 1646 LayerTreeSettings settings;
990 virtual void AfterTest() OVERRIDE 1647 settings.max_partial_texture_updates = 10;
991 { 1648
992 } 1649 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
993 1650 EXPECT_TRUE(host.InitializeRendererIfNeeded());
994 private: 1651
995 FakeContentLayerClient client_; 1652 EXPECT_EQ(5u, host.settings().max_partial_texture_updates);
996 scoped_refptr<FakeContentLayer> m_layer; 1653 }
997 scoped_refptr<FakeScrollbarLayer> m_scrollbar; 1654
998 }; 1655 // When partial updates are allowed,
999 1656 // max updates should also be limited by the settings.
1000 MULTI_THREAD_TEST_F(LayerTreeHostTestAtomicCommit) 1657 {
1001 1658 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1002 static void setLayerPropertiesForTesting(Layer* layer, Layer* parent, const gfx: :Transform& transform, const gfx::PointF& anchor, const gfx::PointF& position, c onst gfx::Size& bounds, bool opaque) 1659
1003 { 1660 scoped_ptr<FakeProxy> proxy =
1004 layer->RemoveAllChildren(); 1661 make_scoped_ptr(new FakeProxy(scoped_ptr<Thread>()));
1005 if (parent) 1662 proxy->GetRendererCapabilities().allow_partial_texture_updates = true;
1006 parent->AddChild(layer); 1663 proxy->SetMaxPartialTextureUpdates(20);
1007 layer->SetTransform(transform); 1664
1008 layer->SetAnchorPoint(anchor); 1665 LayerTreeSettings settings;
1009 layer->SetPosition(position); 1666 settings.max_partial_texture_updates = 10;
1010 layer->SetBounds(bounds); 1667
1011 layer->SetContentsOpaque(opaque); 1668 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
1669 EXPECT_TRUE(host.InitializeRendererIfNeeded());
1670
1671 EXPECT_EQ(10u, host.settings().max_partial_texture_updates);
1672 }
1012 } 1673 }
1013 1674
1014 class LayerTreeHostTestAtomicCommitWithPartialUpdate : public LayerTreeHostTest { 1675 TEST(LayerTreeHostTest, PartialUpdatesWithGLRenderer) {
1015 public: 1676 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1016 LayerTreeHostTestAtomicCommitWithPartialUpdate() 1677
1017 : m_numCommits(0) 1678 LayerTreeSettings settings;
1018 { 1679 settings.max_partial_texture_updates = 4;
1019 } 1680
1020 1681 scoped_ptr<LayerTreeHost> host =
1021 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE 1682 LayerTreeHost::Create(&client, settings, scoped_ptr<Thread>());
1022 { 1683 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1023 // Allow one partial texture update. 1684 EXPECT_EQ(4u, host->settings().max_partial_texture_updates);
1024 settings->max_partial_texture_updates = 1; 1685 }
1025 // Linear fade animator prevents scrollbars from drawing immediately. 1686
1026 settings->use_linear_fade_scrollbar_animator = false; 1687 TEST(LayerTreeHostTest, PartialUpdatesWithSoftwareRenderer) {
1027 } 1688 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_SOFTWARE);
1028 1689
1029 virtual void SetupTree() OVERRIDE 1690 LayerTreeSettings settings;
1030 { 1691 settings.max_partial_texture_updates = 4;
1031 parent_ = FakeContentLayer::Create(&client_); 1692
1032 parent_->SetBounds(gfx::Size(10, 20)); 1693 scoped_ptr<LayerTreeHost> host =
1033 1694 LayerTreeHost::Create(&client, settings, scoped_ptr<Thread>());
1034 m_child = FakeContentLayer::Create(&client_); 1695 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1035 m_child->SetPosition(gfx::Point(0, 10)); 1696 EXPECT_EQ(4u, host->settings().max_partial_texture_updates);
1036 m_child->SetBounds(gfx::Size(3, 10)); 1697 }
1037 1698
1038 bool paint_scrollbar = true; 1699 TEST(LayerTreeHostTest, PartialUpdatesWithDelegatingRendererAndGLContent) {
1039 bool has_thumb = false; 1700 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DELEGATED_3D);
1040 m_scrollbarWithPaints = FakeScrollbarLayer::Create( 1701
1041 paint_scrollbar, has_thumb, parent_->id()); 1702 LayerTreeSettings settings;
1042 m_scrollbarWithPaints->SetPosition(gfx::Point(3, 10)); 1703 settings.max_partial_texture_updates = 4;
1043 m_scrollbarWithPaints->SetBounds(gfx::Size(3, 10)); 1704
1044 1705 scoped_ptr<LayerTreeHost> host =
1045 paint_scrollbar = false; 1706 LayerTreeHost::Create(&client, settings, scoped_ptr<Thread>());
1046 m_scrollbarWithoutPaints = FakeScrollbarLayer::Create( 1707 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1047 paint_scrollbar, has_thumb, parent_->id()); 1708 EXPECT_EQ(0u, host->settings().max_partial_texture_updates);
1048 m_scrollbarWithoutPaints->SetPosition(gfx::Point(6, 10)); 1709 }
1049 m_scrollbarWithoutPaints->SetBounds(gfx::Size(3, 10)); 1710
1050 1711 TEST(LayerTreeHostTest,
1051 parent_->AddChild(m_child); 1712 PartialUpdatesWithDelegatingRendererAndSoftwareContent) {
1052 parent_->AddChild(m_scrollbarWithPaints); 1713 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DELEGATED_SOFTWARE);
1053 parent_->AddChild(m_scrollbarWithoutPaints); 1714
1054 1715 LayerTreeSettings settings;
1055 layer_tree_host()->SetRootLayer(parent_); 1716 settings.max_partial_texture_updates = 4;
1056 LayerTreeHostTest::SetupTree(); 1717
1057 } 1718 scoped_ptr<LayerTreeHost> host =
1058 1719 LayerTreeHost::Create(&client, settings, scoped_ptr<Thread>());
1059 virtual void BeginTest() OVERRIDE 1720 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1060 { 1721 EXPECT_EQ(0u, host->settings().max_partial_texture_updates);
1722 }
1723
1724 class LayerTreeHostTestCapturePicture : public LayerTreeHostTest {
1725 public:
1726 LayerTreeHostTestCapturePicture()
1727 : bounds_(gfx::Size(100, 100)),
1728 layer_(PictureLayer::Create(&content_client_)) {}
1729
1730 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
1731 settings->impl_side_painting = true;
1732 }
1733
1734 class FillRectContentLayerClient : public ContentLayerClient {
1735 public:
1736 virtual void PaintContents(SkCanvas* canvas,
1737 gfx::Rect clip,
1738 gfx::RectF* opaque) OVERRIDE {
1739 SkPaint paint;
1740 paint.setColor(SK_ColorGREEN);
1741
1742 SkRect rect = SkRect::MakeWH(canvas->getDeviceSize().width(),
1743 canvas->getDeviceSize().height());
1744 *opaque = gfx::RectF(rect.width(), rect.height());
1745 canvas->drawRect(rect, paint);
1746 }
1747 virtual void DidChangeLayerCanUseLCDText() OVERRIDE {}
1748 };
1749
1750 virtual void BeginTest() OVERRIDE {
1751 layer_->SetIsDrawable(true);
1752 layer_->SetBounds(bounds_);
1753 layer_tree_host()->SetViewportSize(bounds_, bounds_);
1754 layer_tree_host()->SetRootLayer(layer_);
1755
1756 EXPECT_TRUE(layer_tree_host()->InitializeRendererIfNeeded());
1757 PostSetNeedsCommitToMainThread();
1758 }
1759
1760 virtual void DidCommitAndDrawFrame() OVERRIDE {
1761 picture_ = layer_tree_host()->CapturePicture();
1762 EndTest();
1763 }
1764
1765 virtual void AfterTest() OVERRIDE {
1766 EXPECT_EQ(bounds_, gfx::Size(picture_->width(), picture_->height()));
1767
1768 SkBitmap bitmap;
1769 bitmap.setConfig(
1770 SkBitmap::kARGB_8888_Config, bounds_.width(), bounds_.height());
1771 bitmap.allocPixels();
1772 bitmap.eraseARGB(0, 0, 0, 0);
1773 SkCanvas canvas(bitmap);
1774
1775 picture_->draw(&canvas);
1776
1777 bitmap.lockPixels();
1778 SkColor* pixels = reinterpret_cast<SkColor*>(bitmap.getPixels());
1779 EXPECT_EQ(SK_ColorGREEN, pixels[0]);
1780 bitmap.unlockPixels();
1781 }
1782
1783 private:
1784 gfx::Size bounds_;
1785 FillRectContentLayerClient content_client_;
1786 scoped_refptr<PictureLayer> layer_;
1787 skia::RefPtr<SkPicture> picture_;
1788 };
1789
1790 MULTI_THREAD_TEST_F(LayerTreeHostTestCapturePicture);
1791
1792 class LayerTreeHostTestMaxPendingFrames : public LayerTreeHostTest {
1793 public:
1794 LayerTreeHostTestMaxPendingFrames() : LayerTreeHostTest() {}
1795
1796 virtual scoped_ptr<OutputSurface> CreateOutputSurface() OVERRIDE {
1797 if (delegating_renderer_)
1798 return FakeOutputSurface::CreateDelegating3d().PassAs<OutputSurface>();
1799 return FakeOutputSurface::Create3d().PassAs<OutputSurface>();
1800 }
1801
1802 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
1803
1804 virtual void DrawLayersOnThread(LayerTreeHostImpl* host_impl) OVERRIDE {
1805 DCHECK(host_impl->proxy()->HasImplThread());
1806
1807 const ThreadProxy* proxy = static_cast<ThreadProxy*>(host_impl->proxy());
1808 if (delegating_renderer_) {
1809 EXPECT_EQ(1, proxy->MaxFramesPendingForTesting());
1810 } else {
1811 EXPECT_EQ(FrameRateController::DEFAULT_MAX_FRAMES_PENDING,
1812 proxy->MaxFramesPendingForTesting());
1813 }
1814 EndTest();
1815 }
1816
1817 virtual void AfterTest() OVERRIDE {}
1818
1819 protected:
1820 bool delegating_renderer_;
1821 };
1822
1823 TEST_F(LayerTreeHostTestMaxPendingFrames, DelegatingRenderer) {
1824 delegating_renderer_ = true;
1825 RunTest(true);
1826 }
1827
1828 TEST_F(LayerTreeHostTestMaxPendingFrames, GLRenderer) {
1829 delegating_renderer_ = false;
1830 RunTest(true);
1831 }
1832
1833 class LayerTreeHostTestShutdownWithOnlySomeResourcesEvicted
1834 : public LayerTreeHostTest {
1835 public:
1836 LayerTreeHostTestShutdownWithOnlySomeResourcesEvicted()
1837 : root_layer_(FakeContentLayer::Create(&client_)),
1838 child_layer1_(FakeContentLayer::Create(&client_)),
1839 child_layer2_(FakeContentLayer::Create(&client_)),
1840 num_commits_(0) {}
1841
1842 virtual void BeginTest() OVERRIDE {
1843 layer_tree_host()->SetViewportSize(gfx::Size(100, 100),
1844 gfx::Size(100, 100));
1845 root_layer_->SetBounds(gfx::Size(100, 100));
1846 child_layer1_->SetBounds(gfx::Size(100, 100));
1847 child_layer2_->SetBounds(gfx::Size(100, 100));
1848 root_layer_->AddChild(child_layer1_);
1849 root_layer_->AddChild(child_layer2_);
1850 layer_tree_host()->SetRootLayer(root_layer_);
1851 PostSetNeedsCommitToMainThread();
1852 }
1853
1854 virtual void DidSetVisibleOnImplTree(LayerTreeHostImpl* host_impl,
1855 bool visible) OVERRIDE {
1856 // One backing should remain unevicted.
1857 EXPECT_EQ(100 * 100 * 4 * 1,
1858 layer_tree_host()->contents_texture_manager()->MemoryUseBytes());
1859 // Make sure that contents textures are marked as having been
1860 // purged.
1861 EXPECT_TRUE(host_impl->active_tree()->ContentsTexturesPurged());
1862 // End the test in this state.
1863 EndTest();
1864 }
1865
1866 virtual void CommitCompleteOnThread(LayerTreeHostImpl* host_impl) OVERRIDE {
1867 ++num_commits_;
1868 switch (num_commits_) {
1869 case 1:
1870 // All three backings should have memory.
1871 EXPECT_EQ(
1872 100 * 100 * 4 * 3,
1873 layer_tree_host()->contents_texture_manager()->MemoryUseBytes());
1874 // Set a new policy that will kick out 1 of the 3 resources.
1875 // Because a resource was evicted, a commit will be kicked off.
1876 host_impl->SetManagedMemoryPolicy(
1877 ManagedMemoryPolicy(100 * 100 * 4 * 2,
1878 ManagedMemoryPolicy::CUTOFF_ALLOW_EVERYTHING,
1879 100 * 100 * 4 * 1,
1880 ManagedMemoryPolicy::CUTOFF_ALLOW_EVERYTHING));
1881 break;
1882 case 2:
1883 // Only two backings should have memory.
1884 EXPECT_EQ(
1885 100 * 100 * 4 * 2,
1886 layer_tree_host()->contents_texture_manager()->MemoryUseBytes());
1887 // Become backgrounded, which will cause 1 more resource to be
1888 // evicted.
1889 PostSetVisibleToMainThread(false);
1890 break;
1891 default:
1892 // No further commits should happen because this is not visible
1893 // anymore.
1894 NOTREACHED();
1895 break;
1896 }
1897 }
1898
1899 virtual void AfterTest() OVERRIDE {}
1900
1901 private:
1902 FakeContentLayerClient client_;
1903 scoped_refptr<FakeContentLayer> root_layer_;
1904 scoped_refptr<FakeContentLayer> child_layer1_;
1905 scoped_refptr<FakeContentLayer> child_layer2_;
1906 int num_commits_;
1907 };
1908
1909 SINGLE_AND_MULTI_THREAD_TEST_F(
1910 LayerTreeHostTestShutdownWithOnlySomeResourcesEvicted);
1911
1912 class LayerTreeHostTestPinchZoomScrollbarCreation : public LayerTreeHostTest {
1913 public:
1914 LayerTreeHostTestPinchZoomScrollbarCreation()
1915 : root_layer_(ContentLayer::Create(&client_)) {}
1916
1917 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
1918 settings->use_pinch_zoom_scrollbars = true;
1919 }
1920
1921 virtual void BeginTest() OVERRIDE {
1922 root_layer_->SetIsDrawable(true);
1923 root_layer_->SetBounds(gfx::Size(100, 100));
1924 layer_tree_host()->SetRootLayer(root_layer_);
1925 PostSetNeedsCommitToMainThread();
1926 }
1927
1928 virtual void DidCommit() OVERRIDE {
1929 // We always expect two pinch-zoom scrollbar layers.
1930 ASSERT_EQ(2, root_layer_->children().size());
1931
1932 // Pinch-zoom scrollbar layers always have invalid scrollLayerIds.
1933 ScrollbarLayer* layer1 = root_layer_->children()[0]->ToScrollbarLayer();
1934 ASSERT_TRUE(layer1);
1935 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID,
1936 layer1->scroll_layer_id());
1937 EXPECT_EQ(0.f, layer1->opacity());
1938 EXPECT_TRUE(layer1->OpacityCanAnimateOnImplThread());
1939 EXPECT_TRUE(layer1->DrawsContent());
1940
1941 ScrollbarLayer* layer2 = root_layer_->children()[1]->ToScrollbarLayer();
1942 ASSERT_TRUE(layer2);
1943 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID,
1944 layer2->scroll_layer_id());
1945 EXPECT_EQ(0.f, layer2->opacity());
1946 EXPECT_TRUE(layer2->OpacityCanAnimateOnImplThread());
1947 EXPECT_TRUE(layer2->DrawsContent());
1948
1949 EndTest();
1950 }
1951
1952 virtual void AfterTest() OVERRIDE {}
1953
1954 private:
1955 FakeContentLayerClient client_;
1956 scoped_refptr<ContentLayer> root_layer_;
1957 };
1958
1959 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarCreation);
1960
1961 class LayerTreeHostTestPinchZoomScrollbarResize : public LayerTreeHostTest {
1962 public:
1963 LayerTreeHostTestPinchZoomScrollbarResize()
1964 : root_layer_(ContentLayer::Create(&client_)), num_commits_(0) {}
1965
1966 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
1967 settings->use_pinch_zoom_scrollbars = true;
1968 }
1969
1970 virtual void BeginTest() OVERRIDE {
1971 root_layer_->SetIsDrawable(true);
1972 root_layer_->SetBounds(gfx::Size(100, 100));
1973 layer_tree_host()->SetRootLayer(root_layer_);
1974 layer_tree_host()->SetViewportSize(gfx::Size(100, 100),
1975 gfx::Size(100, 100));
1976 PostSetNeedsCommitToMainThread();
1977 }
1978
1979 virtual void DidCommit() OVERRIDE {
1980 num_commits_++;
1981
1982 ScrollbarLayer* layer1 = root_layer_->children()[0]->ToScrollbarLayer();
1983 ASSERT_TRUE(layer1);
1984 ScrollbarLayer* layer2 = root_layer_->children()[1]->ToScrollbarLayer();
1985 ASSERT_TRUE(layer2);
1986
1987 // Get scrollbar thickness from horizontal scrollbar's height.
1988 int thickness = layer1->bounds().height();
1989
1990 if (!layer1->Orientation() == WebKit::WebScrollbar::Horizontal)
1991 std::swap(layer1, layer2);
1992
1993 gfx::Size viewport_size = layer_tree_host()->layout_viewport_size();
1994 EXPECT_EQ(viewport_size.width() - thickness, layer1->bounds().width());
1995 EXPECT_EQ(viewport_size.height() - thickness, layer2->bounds().height());
1996
1997 switch (num_commits_) {
1998 case 1:
1999 // Resizing the viewport should also resize the pinch-zoom scrollbars.
2000 layer_tree_host()->SetViewportSize(gfx::Size(120, 150),
2001 gfx::Size(120, 150));
2002 break;
2003 default:
2004 EndTest();
2005 }
2006 }
2007
2008 virtual void AfterTest() OVERRIDE {}
2009
2010 private:
2011 FakeContentLayerClient client_;
2012 scoped_refptr<ContentLayer> root_layer_;
2013 int num_commits_;
2014 };
2015
2016 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarResize);
2017
2018 class LayerTreeHostTestPinchZoomScrollbarNewRootLayer
2019 : public LayerTreeHostTest {
2020 public:
2021 LayerTreeHostTestPinchZoomScrollbarNewRootLayer()
2022 : root_layer_(ContentLayer::Create(&client_)), num_commits_(0) {}
2023
2024 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE {
2025 settings->use_pinch_zoom_scrollbars = true;
2026 }
2027
2028 virtual void BeginTest() OVERRIDE {
2029 root_layer_->SetIsDrawable(true);
2030 root_layer_->SetBounds(gfx::Size(100, 100));
2031 layer_tree_host()->SetRootLayer(root_layer_);
2032 PostSetNeedsCommitToMainThread();
2033 }
2034
2035 virtual void DidCommit() OVERRIDE {
2036 num_commits_++;
2037
2038 // We always expect two pinch-zoom scrollbar layers.
2039 ASSERT_EQ(2, root_layer_->children().size());
2040
2041 // Pinch-zoom scrollbar layers always have invalid scrollLayerIds.
2042 ScrollbarLayer* layer1 = root_layer_->children()[0]->ToScrollbarLayer();
2043 ASSERT_TRUE(layer1);
2044 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID,
2045 layer1->scroll_layer_id());
2046 EXPECT_EQ(0.f, layer1->opacity());
2047 EXPECT_TRUE(layer1->DrawsContent());
2048
2049 ScrollbarLayer* layer2 = root_layer_->children()[1]->ToScrollbarLayer();
2050 ASSERT_TRUE(layer2);
2051 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID,
2052 layer2->scroll_layer_id());
2053 EXPECT_EQ(0.f, layer2->opacity());
2054 EXPECT_TRUE(layer2->DrawsContent());
2055
2056 if (num_commits_ == 1) {
2057 // Create a new root layer and attach to tree to verify the pinch
2058 // zoom scrollbars get correctly re-attached.
2059 root_layer_ = ContentLayer::Create(&client_);
2060 root_layer_->SetIsDrawable(true);
2061 root_layer_->SetBounds(gfx::Size(100, 100));
2062 layer_tree_host()->SetRootLayer(root_layer_);
2063 PostSetNeedsCommitToMainThread();
2064 } else {
2065 EndTest();
2066 }
2067 }
2068
2069 virtual void AfterTest() OVERRIDE {}
2070
2071 private:
2072 FakeContentLayerClient client_;
2073 scoped_refptr<ContentLayer> root_layer_;
2074 int num_commits_;
2075 };
2076
2077 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarNewRootLayer);
2078
2079 class LayerTreeHostTestLCDNotification : public LayerTreeHostTest {
2080 public:
2081 class NotificationClient : public ContentLayerClient {
2082 public:
2083 NotificationClient()
2084 : layer_(0), paint_count_(0), lcd_notification_count_(0) {}
2085
2086 void set_layer(Layer* layer) { layer_ = layer; }
2087 int paint_count() const { return paint_count_; }
2088 int lcd_notification_count() const { return lcd_notification_count_; }
2089
2090 virtual void PaintContents(SkCanvas* canvas,
2091 gfx::Rect clip,
2092 gfx::RectF* opaque) OVERRIDE {
2093 ++paint_count_;
2094 }
2095 virtual void DidChangeLayerCanUseLCDText() OVERRIDE {
2096 ++lcd_notification_count_;
2097 layer_->SetNeedsDisplay();
2098 }
2099
2100 private:
2101 Layer* layer_;
2102 int paint_count_;
2103 int lcd_notification_count_;
2104 };
2105
2106 virtual void SetupTree() OVERRIDE {
2107 scoped_refptr<ContentLayer> root_layer = ContentLayer::Create(&client_);
2108 root_layer->SetIsDrawable(true);
2109 root_layer->SetBounds(gfx::Size(1, 1));
2110
2111 layer_tree_host()->SetRootLayer(root_layer);
2112 client_.set_layer(root_layer.get());
2113
2114 // The expecations are based on the assumption that the default
2115 // LCD settings are:
2116 EXPECT_TRUE(layer_tree_host()->settings().can_use_lcd_text);
2117 EXPECT_FALSE(root_layer->can_use_lcd_text());
2118
2119 LayerTreeHostTest::SetupTree();
2120 }
2121
2122 virtual void BeginTest() OVERRIDE { PostSetNeedsCommitToMainThread(); }
2123 virtual void AfterTest() OVERRIDE {}
2124
2125 virtual void DidCommit() OVERRIDE {
2126 switch (layer_tree_host()->commit_number()) {
2127 case 1:
2128 // The first update consists one LCD notification and one paint.
2129 EXPECT_EQ(1, client_.lcd_notification_count());
2130 EXPECT_EQ(1, client_.paint_count());
2131 // LCD text must have been enabled on the layer.
2132 EXPECT_TRUE(layer_tree_host()->root_layer()->can_use_lcd_text());
1061 PostSetNeedsCommitToMainThread(); 2133 PostSetNeedsCommitToMainThread();
1062 } 2134 break;
1063 2135 case 2:
1064 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE 2136 // Since nothing changed on layer, there should be no notification
1065 { 2137 // or paint on the second update.
1066 ASSERT_EQ(1u, 2138 EXPECT_EQ(1, client_.lcd_notification_count());
1067 layer_tree_host()->settings().max_partial_texture_updates); 2139 EXPECT_EQ(1, client_.paint_count());
1068 2140 // LCD text must not have changed.
1069 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D *>(impl->output_surface()->context3d()); 2141 EXPECT_TRUE(layer_tree_host()->root_layer()->can_use_lcd_text());
1070 2142 // Change layer opacity that should trigger lcd notification.
1071 switch (impl->active_tree()->source_frame_number()) { 2143 layer_tree_host()->root_layer()->SetOpacity(.5f);
1072 case 0: 2144 // No need to request a commit - setting opacity will do it.
1073 // Number of textures should be one for each layer. 2145 break;
1074 ASSERT_EQ(4, context->NumTextures()); 2146 default:
1075 // Number of textures used for commit should be one for each layer. 2147 // Verify that there is not extra commit due to layer invalidation.
1076 EXPECT_EQ(4, context->NumUsedTextures()); 2148 EXPECT_EQ(3, layer_tree_host()->commit_number());
1077 // Verify that used textures are correct. 2149 // LCD notification count should have incremented due to
1078 EXPECT_TRUE(context->UsedTexture(context->TextureAt(0))); 2150 // change in layer opacity.
1079 EXPECT_TRUE(context->UsedTexture(context->TextureAt(1))); 2151 EXPECT_EQ(2, client_.lcd_notification_count());
1080 EXPECT_TRUE(context->UsedTexture(context->TextureAt(2))); 2152 // Paint count should be incremented due to invalidation.
1081 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3))); 2153 EXPECT_EQ(2, client_.paint_count());
1082 2154 // LCD text must have been disabled on the layer due to opacity.
1083 context->ResetUsedTextures(); 2155 EXPECT_FALSE(layer_tree_host()->root_layer()->can_use_lcd_text());
1084 PostSetNeedsCommitToMainThread();
1085 break;
1086 case 1:
1087 // Number of textures should be two for each content layer and one
1088 // for each scrollbar, since they always do a partial update.
1089 ASSERT_EQ(6, context->NumTextures());
1090 // Number of textures used for commit should be one for each content
1091 // layer, and one for the scrollbar layer that paints.
1092 EXPECT_EQ(3, context->NumUsedTextures());
1093
1094 // First content textures should not have been used.
1095 EXPECT_FALSE(context->UsedTexture(context->TextureAt(0)));
1096 EXPECT_FALSE(context->UsedTexture(context->TextureAt(1)));
1097 // The non-painting scrollbar's texture wasn't updated.
1098 EXPECT_FALSE(context->UsedTexture(context->TextureAt(2)));
1099 // The painting scrollbar's partial update texture was used.
1100 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
1101 // New textures should have been used.
1102 EXPECT_TRUE(context->UsedTexture(context->TextureAt(4)));
1103 EXPECT_TRUE(context->UsedTexture(context->TextureAt(5)));
1104
1105 context->ResetUsedTextures();
1106 PostSetNeedsCommitToMainThread();
1107 break;
1108 case 2:
1109 // Number of textures should be two for each content layer and one
1110 // for each scrollbar, since they always do a partial update.
1111 ASSERT_EQ(6, context->NumTextures());
1112 // Number of textures used for commit should be one for each content
1113 // layer, and one for the scrollbar layer that paints.
1114 EXPECT_EQ(3, context->NumUsedTextures());
1115
1116 // The non-painting scrollbar's texture wasn't updated.
1117 EXPECT_FALSE(context->UsedTexture(context->TextureAt(2)));
1118 // The painting scrollbar does a partial update.
1119 EXPECT_TRUE(context->UsedTexture(context->TextureAt(3)));
1120 // One content layer does a partial update also.
1121 EXPECT_TRUE(context->UsedTexture(context->TextureAt(4)));
1122 EXPECT_FALSE(context->UsedTexture(context->TextureAt(5)));
1123
1124 context->ResetUsedTextures();
1125 PostSetNeedsCommitToMainThread();
1126 break;
1127 case 3:
1128 // No textures should be used for commit.
1129 EXPECT_EQ(0, context->NumUsedTextures());
1130
1131 context->ResetUsedTextures();
1132 PostSetNeedsCommitToMainThread();
1133 break;
1134 case 4:
1135 // Number of textures used for commit should be two. One for the
1136 // content layer, and one for the painting scrollbar. The
1137 // non-painting scrollbar doesn't update its texture.
1138 EXPECT_EQ(2, context->NumUsedTextures());
1139
1140 context->ResetUsedTextures();
1141 PostSetNeedsCommitToMainThread();
1142 break;
1143 case 5:
1144 EndTest();
1145 break;
1146 default:
1147 NOTREACHED();
1148 break;
1149 }
1150 }
1151
1152 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE
1153 {
1154 TestWebGraphicsContext3D* context = static_cast<TestWebGraphicsContext3D *>(impl->output_surface()->context3d());
1155
1156 // Number of textures used for drawing should one per layer except for
1157 // frame 3 where the viewport only contains one layer.
1158 if (impl->active_tree()->source_frame_number() == 3)
1159 EXPECT_EQ(1, context->NumUsedTextures());
1160 else
1161 EXPECT_EQ(4, context->NumUsedTextures());
1162
1163 context->ResetUsedTextures();
1164 }
1165
1166 virtual void Layout() OVERRIDE
1167 {
1168 switch (m_numCommits++) {
1169 case 0:
1170 case 1:
1171 parent_->SetNeedsDisplay();
1172 m_child->SetNeedsDisplay();
1173 m_scrollbarWithPaints->SetNeedsDisplay();
1174 m_scrollbarWithoutPaints->SetNeedsDisplay();
1175 break;
1176 case 2:
1177 // Damage part of layers.
1178 parent_->SetNeedsDisplayRect(gfx::RectF(0, 0, 5, 5));
1179 m_child->SetNeedsDisplayRect(gfx::RectF(0, 0, 5, 5));
1180 m_scrollbarWithPaints->SetNeedsDisplayRect(gfx::RectF(0, 0, 5, 5));
1181 m_scrollbarWithoutPaints->SetNeedsDisplayRect(gfx::RectF(0, 0, 5, 5) );
1182 break;
1183 case 3:
1184 m_child->SetNeedsDisplay();
1185 m_scrollbarWithPaints->SetNeedsDisplay();
1186 m_scrollbarWithoutPaints->SetNeedsDisplay();
1187 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10));
1188 break;
1189 case 4:
1190 layer_tree_host()->SetViewportSize(gfx::Size(10, 20), gfx::Size(10, 20));
1191 break;
1192 case 5:
1193 break;
1194 default:
1195 NOTREACHED();
1196 break;
1197 }
1198 }
1199
1200 virtual void AfterTest() OVERRIDE
1201 {
1202 }
1203
1204 private:
1205 FakeContentLayerClient client_;
1206 scoped_refptr<FakeContentLayer> parent_;
1207 scoped_refptr<FakeContentLayer> m_child;
1208 scoped_refptr<FakeScrollbarLayer> m_scrollbarWithPaints;
1209 scoped_refptr<FakeScrollbarLayer> m_scrollbarWithoutPaints;
1210 int m_numCommits;
1211 };
1212
1213 MULTI_THREAD_TEST_F(LayerTreeHostTestAtomicCommitWithPartialUpdate)
1214
1215 class LayerTreeHostTestFinishAllRendering : public LayerTreeHostTest {
1216 public:
1217 LayerTreeHostTestFinishAllRendering()
1218 : m_once(false)
1219 , m_drawCount(0)
1220 {
1221 }
1222
1223 virtual void BeginTest() OVERRIDE
1224 {
1225 layer_tree_host()->SetNeedsRedraw();
1226 PostSetNeedsCommitToMainThread();
1227 }
1228
1229 virtual void DidCommitAndDrawFrame() OVERRIDE
1230 {
1231 if (m_once)
1232 return;
1233 m_once = true;
1234 layer_tree_host()->SetNeedsRedraw();
1235 layer_tree_host()->AcquireLayerTextures();
1236 {
1237 base::AutoLock lock(m_lock);
1238 m_drawCount = 0;
1239 }
1240 layer_tree_host()->FinishAllRendering();
1241 {
1242 base::AutoLock lock(m_lock);
1243 EXPECT_EQ(0, m_drawCount);
1244 }
1245 EndTest(); 2156 EndTest();
1246 } 2157 break;
1247 2158 }
1248 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE 2159 }
1249 { 2160
1250 base::AutoLock lock(m_lock); 2161 private:
1251 ++m_drawCount; 2162 NotificationClient client_;
1252 } 2163 };
1253 2164
1254 virtual void AfterTest() OVERRIDE 2165 SINGLE_THREAD_TEST_F(LayerTreeHostTestLCDNotification);
1255 {
1256 }
1257 private:
1258
1259 bool m_once;
1260 base::Lock m_lock;
1261 int m_drawCount;
1262 };
1263
1264 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestFinishAllRendering)
1265
1266 class LayerTreeHostTestCompositeAndReadbackCleanup : public LayerTreeHostTest {
1267 public:
1268 LayerTreeHostTestCompositeAndReadbackCleanup() { }
1269
1270 virtual void BeginTest() OVERRIDE
1271 {
1272 Layer* rootLayer = layer_tree_host()->root_layer();
1273
1274 char pixels[4];
1275 layer_tree_host()->CompositeAndReadback(static_cast<void*>(&pixels), gfx ::Rect(0, 0, 1, 1));
1276 EXPECT_FALSE(rootLayer->render_surface());
1277
1278 EndTest();
1279 }
1280
1281 virtual void AfterTest() OVERRIDE
1282 {
1283 }
1284 };
1285
1286 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestCompositeAndReadbackCleanup)
1287
1288 class LayerTreeHostTestSurfaceNotAllocatedForLayersOutsideMemoryLimit : public L ayerTreeHostTest {
1289 public:
1290 LayerTreeHostTestSurfaceNotAllocatedForLayersOutsideMemoryLimit()
1291 : m_rootLayer(ContentLayerWithUpdateTracking::Create(&m_fakeDelegate))
1292 , m_surfaceLayer1(ContentLayerWithUpdateTracking::Create(&m_fakeDelegate ))
1293 , m_replicaLayer1(ContentLayerWithUpdateTracking::Create(&m_fakeDelegate ))
1294 , m_surfaceLayer2(ContentLayerWithUpdateTracking::Create(&m_fakeDelegate ))
1295 , m_replicaLayer2(ContentLayerWithUpdateTracking::Create(&m_fakeDelegate ))
1296 {
1297 }
1298
1299 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE
1300 {
1301 settings->cache_render_pass_contents = true;
1302 }
1303
1304 virtual void BeginTest() OVERRIDE
1305 {
1306 layer_tree_host()->SetViewportSize(gfx::Size(100, 100), gfx::Size(100, 1 00));
1307
1308 m_rootLayer->SetBounds(gfx::Size(100, 100));
1309 m_surfaceLayer1->SetBounds(gfx::Size(100, 100));
1310 m_surfaceLayer1->SetForceRenderSurface(true);
1311 m_surfaceLayer1->SetOpacity(0.5);
1312 m_surfaceLayer2->SetBounds(gfx::Size(100, 100));
1313 m_surfaceLayer2->SetForceRenderSurface(true);
1314 m_surfaceLayer2->SetOpacity(0.5);
1315
1316 m_surfaceLayer1->SetReplicaLayer(m_replicaLayer1.get());
1317 m_surfaceLayer2->SetReplicaLayer(m_replicaLayer2.get());
1318
1319 m_rootLayer->AddChild(m_surfaceLayer1);
1320 m_surfaceLayer1->AddChild(m_surfaceLayer2);
1321 layer_tree_host()->SetRootLayer(m_rootLayer);
1322
1323 PostSetNeedsCommitToMainThread();
1324 }
1325
1326 virtual void DrawLayersOnThread(LayerTreeHostImpl* hostImpl) OVERRIDE
1327 {
1328 Renderer* renderer = hostImpl->renderer();
1329 RenderPass::Id surface1RenderPassId = hostImpl->active_tree()->root_laye r()->children()[0]->render_surface()->RenderPassId();
1330 RenderPass::Id surface2RenderPassId = hostImpl->active_tree()->root_laye r()->children()[0]->children()[0]->render_surface()->RenderPassId();
1331
1332 switch (hostImpl->active_tree()->source_frame_number()) {
1333 case 0:
1334 EXPECT_TRUE(renderer->HaveCachedResourcesForRenderPassId(surface1Ren derPassId));
1335 EXPECT_TRUE(renderer->HaveCachedResourcesForRenderPassId(surface2Ren derPassId));
1336
1337 // Reduce the memory limit to only fit the root layer and one render surface. This
1338 // prevents any contents drawing into surfaces from being allocated.
1339 hostImpl->SetManagedMemoryPolicy(ManagedMemoryPolicy(100 * 100 * 4 * 2));
1340 break;
1341 case 1:
1342 EXPECT_FALSE(renderer->HaveCachedResourcesForRenderPassId(surface1Re nderPassId));
1343 EXPECT_FALSE(renderer->HaveCachedResourcesForRenderPassId(surface2Re nderPassId));
1344
1345 EndTest();
1346 break;
1347 }
1348 }
1349
1350 virtual void AfterTest() OVERRIDE
1351 {
1352 EXPECT_EQ(2, m_rootLayer->paintContentsCount());
1353 EXPECT_EQ(2, m_surfaceLayer1->paintContentsCount());
1354 EXPECT_EQ(2, m_surfaceLayer2->paintContentsCount());
1355
1356 // Clear layer references so LayerTreeHost dies.
1357 m_rootLayer = NULL;
1358 m_surfaceLayer1 = NULL;
1359 m_replicaLayer1 = NULL;
1360 m_surfaceLayer2 = NULL;
1361 m_replicaLayer2 = NULL;
1362 }
1363
1364 private:
1365 FakeContentLayerClient m_fakeDelegate;
1366 scoped_refptr<ContentLayerWithUpdateTracking> m_rootLayer;
1367 scoped_refptr<ContentLayerWithUpdateTracking> m_surfaceLayer1;
1368 scoped_refptr<ContentLayerWithUpdateTracking> m_replicaLayer1;
1369 scoped_refptr<ContentLayerWithUpdateTracking> m_surfaceLayer2;
1370 scoped_refptr<ContentLayerWithUpdateTracking> m_replicaLayer2;
1371 };
1372
1373 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestSurfaceNotAllocatedForLayersOuts ideMemoryLimit)
1374
1375 class EvictionTestLayer : public Layer {
1376 public:
1377 static scoped_refptr<EvictionTestLayer> Create() { return make_scoped_refptr (new EvictionTestLayer()); }
1378
1379 virtual void Update(ResourceUpdateQueue*, const OcclusionTracker*, Rendering Stats*) OVERRIDE;
1380 virtual bool DrawsContent() const OVERRIDE { return true; }
1381
1382 virtual scoped_ptr<LayerImpl> CreateLayerImpl(LayerTreeImpl* treeImpl) OVERR IDE;
1383 virtual void PushPropertiesTo(LayerImpl*) OVERRIDE;
1384 virtual void SetTexturePriorities(const PriorityCalculator&) OVERRIDE;
1385
1386 bool haveBackingTexture() const { return m_texture.get() ? m_texture->have_b acking_texture() : false; }
1387
1388 private:
1389 EvictionTestLayer() : Layer() { }
1390 virtual ~EvictionTestLayer() { }
1391
1392 void createTextureIfNeeded()
1393 {
1394 if (m_texture.get())
1395 return;
1396 m_texture = PrioritizedResource::Create(layer_tree_host()->contents_text ure_manager());
1397 m_texture->SetDimensions(gfx::Size(10, 10), GL_RGBA);
1398 m_bitmap.setConfig(SkBitmap::kARGB_8888_Config, 10, 10);
1399 }
1400
1401 scoped_ptr<PrioritizedResource> m_texture;
1402 SkBitmap m_bitmap;
1403 };
1404
1405 class EvictionTestLayerImpl : public LayerImpl {
1406 public:
1407 static scoped_ptr<EvictionTestLayerImpl> Create(LayerTreeImpl* treeImpl, int id)
1408 {
1409 return make_scoped_ptr(new EvictionTestLayerImpl(treeImpl, id));
1410 }
1411 virtual ~EvictionTestLayerImpl() { }
1412
1413 virtual void AppendQuads(QuadSink* quad_sink,
1414 AppendQuadsData* append_quads_data) OVERRIDE
1415 {
1416 ASSERT_TRUE(m_hasTexture);
1417 ASSERT_NE(0u, layer_tree_impl()->resource_provider()->num_resources());
1418 }
1419
1420 void setHasTexture(bool hasTexture) { m_hasTexture = hasTexture; }
1421
1422 private:
1423 EvictionTestLayerImpl(LayerTreeImpl* treeImpl, int id)
1424 : LayerImpl(treeImpl, id)
1425 , m_hasTexture(false) { }
1426
1427 bool m_hasTexture;
1428 };
1429
1430 void EvictionTestLayer::SetTexturePriorities(const PriorityCalculator&)
1431 {
1432 createTextureIfNeeded();
1433 if (!m_texture.get())
1434 return;
1435 m_texture->set_request_priority(PriorityCalculator::UIPriority(true));
1436 }
1437
1438 void EvictionTestLayer::Update(ResourceUpdateQueue* queue, const OcclusionTracke r*, RenderingStats*)
1439 {
1440 createTextureIfNeeded();
1441 if (!m_texture.get())
1442 return;
1443
1444 gfx::Rect fullRect(0, 0, 10, 10);
1445 ResourceUpdate upload = ResourceUpdate::Create(
1446 m_texture.get(), &m_bitmap, fullRect, fullRect, gfx::Vector2d());
1447 queue->AppendFullUpload(upload);
1448 }
1449
1450 scoped_ptr<LayerImpl> EvictionTestLayer::CreateLayerImpl(LayerTreeImpl* treeImpl )
1451 {
1452 return EvictionTestLayerImpl::Create(treeImpl, layer_id_).PassAs<LayerImpl>( );
1453 }
1454
1455 void EvictionTestLayer::PushPropertiesTo(LayerImpl* layerImpl)
1456 {
1457 Layer::PushPropertiesTo(layerImpl);
1458
1459 EvictionTestLayerImpl* testLayerImpl = static_cast<EvictionTestLayerImpl*>(l ayerImpl);
1460 testLayerImpl->setHasTexture(m_texture->have_backing_texture());
1461 }
1462
1463 class LayerTreeHostTestEvictTextures : public LayerTreeHostTest {
1464 public:
1465 LayerTreeHostTestEvictTextures()
1466 : m_layer(EvictionTestLayer::Create())
1467 , m_implForEvictTextures(0)
1468 , m_numCommits(0)
1469 {
1470 }
1471
1472 virtual void BeginTest() OVERRIDE
1473 {
1474 layer_tree_host()->SetRootLayer(m_layer);
1475 layer_tree_host()->SetViewportSize(gfx::Size(10, 20), gfx::Size(10, 20)) ;
1476
1477 gfx::Transform identityMatrix;
1478 setLayerPropertiesForTesting(m_layer.get(), 0, identityMatrix, gfx::Poin tF(0, 0), gfx::PointF(0, 0), gfx::Size(10, 20), true);
1479
1480 PostSetNeedsCommitToMainThread();
1481 }
1482
1483 void postEvictTextures()
1484 {
1485 DCHECK(ImplThread());
1486 ImplThread()->PostTask(base::Bind(&LayerTreeHostTestEvictTextures::evict TexturesOnImplThread,
1487 base::Unretained(this)));
1488 }
1489
1490 void evictTexturesOnImplThread()
1491 {
1492 DCHECK(m_implForEvictTextures);
1493 m_implForEvictTextures->EnforceManagedMemoryPolicy(ManagedMemoryPolicy(0 ));
1494 }
1495
1496 // Commit 1: Just commit and draw normally, then post an eviction at the end
1497 // that will trigger a commit.
1498 // Commit 2: Triggered by the eviction, let it go through and then set
1499 // needsCommit.
1500 // Commit 3: Triggered by the setNeedsCommit. In Layout(), post an eviction
1501 // task, which will be handled before the commit. Don't set needsCommit, it
1502 // should have been posted. A frame should not be drawn (note,
1503 // didCommitAndDrawFrame may be called anyway).
1504 // Commit 4: Triggered by the eviction, let it go through and then set
1505 // needsCommit.
1506 // Commit 5: Triggered by the setNeedsCommit, post an eviction task in
1507 // Layout(), a frame should not be drawn but a commit will be posted.
1508 // Commit 6: Triggered by the eviction, post an eviction task in
1509 // Layout(), which will be a noop, letting the commit (which recreates the
1510 // textures) go through and draw a frame, then end the test.
1511 //
1512 // Commits 1+2 test the eviction recovery path where eviction happens outsid e
1513 // of the beginFrame/commit pair.
1514 // Commits 3+4 test the eviction recovery path where eviction happens inside
1515 // the beginFrame/commit pair.
1516 // Commits 5+6 test the path where an eviction happens during the eviction
1517 // recovery path.
1518 virtual void DidCommitAndDrawFrame() OVERRIDE
1519 {
1520 switch (m_numCommits) {
1521 case 1:
1522 EXPECT_TRUE(m_layer->haveBackingTexture());
1523 postEvictTextures();
1524 break;
1525 case 2:
1526 EXPECT_TRUE(m_layer->haveBackingTexture());
1527 layer_tree_host()->SetNeedsCommit();
1528 break;
1529 case 3:
1530 break;
1531 case 4:
1532 EXPECT_TRUE(m_layer->haveBackingTexture());
1533 layer_tree_host()->SetNeedsCommit();
1534 break;
1535 case 5:
1536 break;
1537 case 6:
1538 EXPECT_TRUE(m_layer->haveBackingTexture());
1539 EndTest();
1540 break;
1541 default:
1542 NOTREACHED();
1543 break;
1544 }
1545 }
1546
1547 virtual void CommitCompleteOnThread(LayerTreeHostImpl* impl) OVERRIDE
1548 {
1549 m_implForEvictTextures = impl;
1550 }
1551
1552 virtual void Layout() OVERRIDE
1553 {
1554 ++m_numCommits;
1555 switch (m_numCommits) {
1556 case 1:
1557 case 2:
1558 break;
1559 case 3:
1560 postEvictTextures();
1561 break;
1562 case 4:
1563 // We couldn't check in didCommitAndDrawFrame on commit 3, so check here.
1564 EXPECT_FALSE(m_layer->haveBackingTexture());
1565 break;
1566 case 5:
1567 postEvictTextures();
1568 break;
1569 case 6:
1570 // We couldn't check in didCommitAndDrawFrame on commit 5, so check here.
1571 EXPECT_FALSE(m_layer->haveBackingTexture());
1572 postEvictTextures();
1573 break;
1574 default:
1575 NOTREACHED();
1576 break;
1577 }
1578 }
1579
1580 virtual void AfterTest() OVERRIDE
1581 {
1582 }
1583
1584 private:
1585 FakeContentLayerClient client_;
1586 scoped_refptr<EvictionTestLayer> m_layer;
1587 LayerTreeHostImpl* m_implForEvictTextures;
1588 int m_numCommits;
1589 };
1590
1591 MULTI_THREAD_TEST_F(LayerTreeHostTestEvictTextures)
1592
1593 class LayerTreeHostTestContinuousCommit : public LayerTreeHostTest {
1594 public:
1595 LayerTreeHostTestContinuousCommit()
1596 : m_numCommitComplete(0)
1597 , m_numDrawLayers(0)
1598 {
1599 }
1600
1601 virtual void BeginTest() OVERRIDE
1602 {
1603 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10)) ;
1604 layer_tree_host()->root_layer()->SetBounds(gfx::Size(10, 10));
1605
1606 PostSetNeedsCommitToMainThread();
1607 }
1608
1609 virtual void DidCommit() OVERRIDE
1610 {
1611 if (m_numDrawLayers == 2)
1612 return;
1613 PostSetNeedsCommitToMainThread();
1614 }
1615
1616 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE
1617 {
1618 if (m_numDrawLayers == 1)
1619 m_numCommitComplete++;
1620 }
1621
1622 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE
1623 {
1624 m_numDrawLayers++;
1625 if (m_numDrawLayers == 2)
1626 EndTest();
1627 }
1628
1629 virtual void AfterTest() OVERRIDE
1630 {
1631 // Check that we didn't commit twice between first and second draw.
1632 EXPECT_EQ(1, m_numCommitComplete);
1633 }
1634
1635 private:
1636 int m_numCommitComplete;
1637 int m_numDrawLayers;
1638 };
1639
1640 MULTI_THREAD_TEST_F(LayerTreeHostTestContinuousCommit)
1641
1642 class LayerTreeHostTestContinuousInvalidate : public LayerTreeHostTest {
1643 public:
1644 LayerTreeHostTestContinuousInvalidate()
1645 : m_numCommitComplete(0)
1646 , m_numDrawLayers(0)
1647 {
1648 }
1649
1650 virtual void BeginTest() OVERRIDE
1651 {
1652 layer_tree_host()->SetViewportSize(gfx::Size(10, 10), gfx::Size(10, 10)) ;
1653 layer_tree_host()->root_layer()->SetBounds(gfx::Size(10, 10));
1654
1655 m_contentLayer = ContentLayer::Create(&m_fakeDelegate);
1656 m_contentLayer->SetBounds(gfx::Size(10, 10));
1657 m_contentLayer->SetPosition(gfx::PointF(0, 0));
1658 m_contentLayer->SetAnchorPoint(gfx::PointF(0, 0));
1659 m_contentLayer->SetIsDrawable(true);
1660 layer_tree_host()->root_layer()->AddChild(m_contentLayer);
1661
1662 PostSetNeedsCommitToMainThread();
1663 }
1664
1665 virtual void DidCommit() OVERRIDE
1666 {
1667 if (m_numDrawLayers == 2)
1668 return;
1669 m_contentLayer->SetNeedsDisplay();
1670 }
1671
1672 virtual void CommitCompleteOnThread(LayerTreeHostImpl*) OVERRIDE
1673 {
1674 if (m_numDrawLayers == 1)
1675 m_numCommitComplete++;
1676 }
1677
1678 virtual void DrawLayersOnThread(LayerTreeHostImpl* impl) OVERRIDE
1679 {
1680 m_numDrawLayers++;
1681 if (m_numDrawLayers == 2)
1682 EndTest();
1683 }
1684
1685 virtual void AfterTest() OVERRIDE
1686 {
1687 // Check that we didn't commit twice between first and second draw.
1688 EXPECT_EQ(1, m_numCommitComplete);
1689
1690 // Clear layer references so LayerTreeHost dies.
1691 m_contentLayer = NULL;
1692 }
1693
1694 private:
1695 FakeContentLayerClient m_fakeDelegate;
1696 scoped_refptr<Layer> m_contentLayer;
1697 int m_numCommitComplete;
1698 int m_numDrawLayers;
1699 };
1700
1701 MULTI_THREAD_TEST_F(LayerTreeHostTestContinuousInvalidate)
1702
1703 class LayerTreeHostTestDeferCommits : public LayerTreeHostTest {
1704 public:
1705 LayerTreeHostTestDeferCommits()
1706 : m_numCommitsDeferred(0)
1707 , m_numCompleteCommits(0)
1708 {
1709 }
1710
1711 virtual void BeginTest() OVERRIDE
1712 {
1713 PostSetNeedsCommitToMainThread();
1714 }
1715
1716 virtual void DidDeferCommit() OVERRIDE
1717 {
1718 m_numCommitsDeferred++;
1719 layer_tree_host()->SetDeferCommits(false);
1720 }
1721
1722 virtual void DidCommit() OVERRIDE
1723 {
1724 m_numCompleteCommits++;
1725 switch (m_numCompleteCommits) {
1726 case 1:
1727 EXPECT_EQ(0, m_numCommitsDeferred);
1728 layer_tree_host()->SetDeferCommits(true);
1729 PostSetNeedsCommitToMainThread();
1730 break;
1731 case 2:
1732 EndTest();
1733 break;
1734 default:
1735 NOTREACHED();
1736 break;
1737 }
1738 }
1739
1740 virtual void AfterTest() OVERRIDE
1741 {
1742 EXPECT_EQ(1, m_numCommitsDeferred);
1743 EXPECT_EQ(2, m_numCompleteCommits);
1744 }
1745
1746 private:
1747 int m_numCommitsDeferred;
1748 int m_numCompleteCommits;
1749 };
1750
1751 MULTI_THREAD_TEST_F(LayerTreeHostTestDeferCommits)
1752
1753 class LayerTreeHostWithProxy : public LayerTreeHost {
1754 public:
1755 LayerTreeHostWithProxy(FakeLayerTreeHostClient* client, const LayerTreeSetti ngs& settings, scoped_ptr<Proxy> proxy)
1756 : LayerTreeHost(client, settings)
1757 {
1758 EXPECT_TRUE(InitializeForTesting(proxy.Pass()));
1759 }
1760 };
1761
1762 TEST(LayerTreeHostTest, LimitPartialUpdates)
1763 {
1764 // When partial updates are not allowed, max updates should be 0.
1765 {
1766 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1767
1768 scoped_ptr<FakeProxy> proxy = make_scoped_ptr(new FakeProxy(scoped_ptr<T hread>()));
1769 proxy->GetRendererCapabilities().allow_partial_texture_updates = false;
1770 proxy->SetMaxPartialTextureUpdates(5);
1771
1772 LayerTreeSettings settings;
1773 settings.max_partial_texture_updates = 10;
1774
1775 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
1776 EXPECT_TRUE(host.InitializeRendererIfNeeded());
1777
1778 EXPECT_EQ(0u, host.settings().max_partial_texture_updates);
1779 }
1780
1781 // When partial updates are allowed, max updates should be limited by the pr oxy.
1782 {
1783 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1784
1785 scoped_ptr<FakeProxy> proxy = make_scoped_ptr(new FakeProxy(scoped_ptr<T hread>()));
1786 proxy->GetRendererCapabilities().allow_partial_texture_updates = true;
1787 proxy->SetMaxPartialTextureUpdates(5);
1788
1789 LayerTreeSettings settings;
1790 settings.max_partial_texture_updates = 10;
1791
1792 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
1793 EXPECT_TRUE(host.InitializeRendererIfNeeded());
1794
1795 EXPECT_EQ(5u, host.settings().max_partial_texture_updates);
1796 }
1797
1798 // When partial updates are allowed, max updates should also be limited by t he settings.
1799 {
1800 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1801
1802 scoped_ptr<FakeProxy> proxy = make_scoped_ptr(new FakeProxy(scoped_ptr<T hread>()));
1803 proxy->GetRendererCapabilities().allow_partial_texture_updates = true;
1804 proxy->SetMaxPartialTextureUpdates(20);
1805
1806 LayerTreeSettings settings;
1807 settings.max_partial_texture_updates = 10;
1808
1809 LayerTreeHostWithProxy host(&client, settings, proxy.PassAs<Proxy>());
1810 EXPECT_TRUE(host.InitializeRendererIfNeeded());
1811
1812 EXPECT_EQ(10u, host.settings().max_partial_texture_updates);
1813 }
1814 }
1815
1816 TEST(LayerTreeHostTest, PartialUpdatesWithGLRenderer)
1817 {
1818 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_3D);
1819
1820 LayerTreeSettings settings;
1821 settings.max_partial_texture_updates = 4;
1822
1823 scoped_ptr<LayerTreeHost> host = LayerTreeHost::Create(&client, settings, sc oped_ptr<Thread>());
1824 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1825 EXPECT_EQ(4u, host->settings().max_partial_texture_updates);
1826 }
1827
1828 TEST(LayerTreeHostTest, PartialUpdatesWithSoftwareRenderer)
1829 {
1830 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DIRECT_SOFTWARE);
1831
1832 LayerTreeSettings settings;
1833 settings.max_partial_texture_updates = 4;
1834
1835 scoped_ptr<LayerTreeHost> host = LayerTreeHost::Create(&client, settings, sc oped_ptr<Thread>());
1836 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1837 EXPECT_EQ(4u, host->settings().max_partial_texture_updates);
1838 }
1839
1840 TEST(LayerTreeHostTest, PartialUpdatesWithDelegatingRendererAndGLContent)
1841 {
1842 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DELEGATED_3D);
1843
1844 LayerTreeSettings settings;
1845 settings.max_partial_texture_updates = 4;
1846
1847 scoped_ptr<LayerTreeHost> host = LayerTreeHost::Create(&client, settings, sc oped_ptr<Thread>());
1848 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1849 EXPECT_EQ(0u, host->settings().max_partial_texture_updates);
1850 }
1851
1852 TEST(LayerTreeHostTest, PartialUpdatesWithDelegatingRendererAndSoftwareContent)
1853 {
1854 FakeLayerTreeHostClient client(FakeLayerTreeHostClient::DELEGATED_SOFTWARE);
1855
1856 LayerTreeSettings settings;
1857 settings.max_partial_texture_updates = 4;
1858
1859 scoped_ptr<LayerTreeHost> host = LayerTreeHost::Create(&client, settings, sc oped_ptr<Thread>());
1860 EXPECT_TRUE(host->InitializeRendererIfNeeded());
1861 EXPECT_EQ(0u, host->settings().max_partial_texture_updates);
1862 }
1863
1864 class LayerTreeHostTestCapturePicture : public LayerTreeHostTest {
1865 public:
1866 LayerTreeHostTestCapturePicture()
1867 : bounds_(gfx::Size(100, 100))
1868 , m_layer(PictureLayer::Create(&m_contentClient))
1869 {
1870 }
1871
1872 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE
1873 {
1874 settings->impl_side_painting = true;
1875 }
1876
1877 class FillRectContentLayerClient : public ContentLayerClient {
1878 public:
1879 virtual void PaintContents(SkCanvas* canvas, gfx::Rect clip, gfx::RectF* opaque) OVERRIDE
1880 {
1881 SkPaint paint;
1882 paint.setColor(SK_ColorGREEN);
1883
1884 SkRect rect = SkRect::MakeWH(canvas->getDeviceSize().width(), canvas ->getDeviceSize().height());
1885 *opaque = gfx::RectF(rect.width(), rect.height());
1886 canvas->drawRect(rect, paint);
1887 }
1888 virtual void DidChangeLayerCanUseLCDText() OVERRIDE {}
1889 };
1890
1891 virtual void BeginTest() OVERRIDE
1892 {
1893 m_layer->SetIsDrawable(true);
1894 m_layer->SetBounds(bounds_);
1895 layer_tree_host()->SetViewportSize(bounds_, bounds_);
1896 layer_tree_host()->SetRootLayer(m_layer);
1897
1898 EXPECT_TRUE(layer_tree_host()->InitializeRendererIfNeeded());
1899 PostSetNeedsCommitToMainThread();
1900 }
1901
1902 virtual void DidCommitAndDrawFrame() OVERRIDE
1903 {
1904 m_picture = layer_tree_host()->CapturePicture();
1905 EndTest();
1906 }
1907
1908 virtual void AfterTest() OVERRIDE
1909 {
1910 EXPECT_EQ(bounds_, gfx::Size(m_picture->width(), m_picture->height()));
1911
1912 SkBitmap bitmap;
1913 bitmap.setConfig(SkBitmap::kARGB_8888_Config, bounds_.width(), bounds_.h eight());
1914 bitmap.allocPixels();
1915 bitmap.eraseARGB(0, 0, 0, 0);
1916 SkCanvas canvas(bitmap);
1917
1918 m_picture->draw(&canvas);
1919
1920 bitmap.lockPixels();
1921 SkColor* pixels = reinterpret_cast<SkColor*>(bitmap.getPixels());
1922 EXPECT_EQ(SK_ColorGREEN, pixels[0]);
1923 bitmap.unlockPixels();
1924 }
1925
1926 private:
1927 gfx::Size bounds_;
1928 FillRectContentLayerClient m_contentClient;
1929 scoped_refptr<PictureLayer> m_layer;
1930 skia::RefPtr<SkPicture> m_picture;
1931 };
1932
1933 MULTI_THREAD_TEST_F(LayerTreeHostTestCapturePicture);
1934
1935 class LayerTreeHostTestMaxPendingFrames : public LayerTreeHostTest {
1936 public:
1937 LayerTreeHostTestMaxPendingFrames()
1938 : LayerTreeHostTest()
1939 {
1940 }
1941
1942 virtual scoped_ptr<OutputSurface> CreateOutputSurface() OVERRIDE
1943 {
1944 if (m_delegatingRenderer)
1945 return FakeOutputSurface::CreateDelegating3d().PassAs<OutputSurface> ();
1946 return FakeOutputSurface::Create3d().PassAs<OutputSurface>();
1947 }
1948
1949 virtual void BeginTest() OVERRIDE
1950 {
1951 PostSetNeedsCommitToMainThread();
1952 }
1953
1954 virtual void DrawLayersOnThread(LayerTreeHostImpl* hostImpl) OVERRIDE
1955 {
1956 DCHECK(hostImpl->proxy()->HasImplThread());
1957
1958 const ThreadProxy* proxy = static_cast<ThreadProxy*>(hostImpl->proxy());
1959 if (m_delegatingRenderer)
1960 EXPECT_EQ(1, proxy->MaxFramesPendingForTesting());
1961 else
1962 EXPECT_EQ(FrameRateController::DEFAULT_MAX_FRAMES_PENDING, proxy->Ma xFramesPendingForTesting());
1963 EndTest();
1964 }
1965
1966 virtual void AfterTest() OVERRIDE
1967 {
1968 }
1969
1970 protected:
1971 bool m_delegatingRenderer;
1972 };
1973
1974 TEST_F(LayerTreeHostTestMaxPendingFrames, DelegatingRenderer)
1975 {
1976 m_delegatingRenderer = true;
1977 RunTest(true);
1978 }
1979
1980 TEST_F(LayerTreeHostTestMaxPendingFrames, GLRenderer)
1981 {
1982 m_delegatingRenderer = false;
1983 RunTest(true);
1984 }
1985
1986 class LayerTreeHostTestShutdownWithOnlySomeResourcesEvicted : public LayerTreeHo stTest {
1987 public:
1988 LayerTreeHostTestShutdownWithOnlySomeResourcesEvicted()
1989 : m_rootLayer(FakeContentLayer::Create(&client_))
1990 , m_childLayer1(FakeContentLayer::Create(&client_))
1991 , m_childLayer2(FakeContentLayer::Create(&client_))
1992 , m_numCommits(0)
1993 {
1994 }
1995
1996 virtual void BeginTest() OVERRIDE
1997 {
1998 layer_tree_host()->SetViewportSize(gfx::Size(100, 100), gfx::Size(100, 1 00));
1999 m_rootLayer->SetBounds(gfx::Size(100, 100));
2000 m_childLayer1->SetBounds(gfx::Size(100, 100));
2001 m_childLayer2->SetBounds(gfx::Size(100, 100));
2002 m_rootLayer->AddChild(m_childLayer1);
2003 m_rootLayer->AddChild(m_childLayer2);
2004 layer_tree_host()->SetRootLayer(m_rootLayer);
2005 PostSetNeedsCommitToMainThread();
2006 }
2007
2008 virtual void DidSetVisibleOnImplTree(LayerTreeHostImpl* hostImpl, bool visib le) OVERRIDE
2009 {
2010 // One backing should remain unevicted.
2011 EXPECT_EQ(
2012 100 * 100 * 4 * 1,
2013 layer_tree_host()->contents_texture_manager()->MemoryUseBytes());
2014 // Make sure that contents textures are marked as having been
2015 // purged.
2016 EXPECT_TRUE(hostImpl->active_tree()->ContentsTexturesPurged());
2017 // End the test in this state.
2018 EndTest();
2019 }
2020
2021 virtual void CommitCompleteOnThread(LayerTreeHostImpl* hostImpl) OVERRIDE
2022 {
2023 ++m_numCommits;
2024 switch(m_numCommits) {
2025 case 1:
2026 // All three backings should have memory.
2027 EXPECT_EQ(
2028 100 * 100 * 4 * 3,
2029 layer_tree_host()->contents_texture_manager()->MemoryUseBytes()) ;
2030 // Set a new policy that will kick out 1 of the 3 resources.
2031 // Because a resource was evicted, a commit will be kicked off.
2032 hostImpl->SetManagedMemoryPolicy(ManagedMemoryPolicy(
2033 100 * 100 * 4 * 2,
2034 ManagedMemoryPolicy::CUTOFF_ALLOW_EVERYTHING,
2035 100 * 100 * 4 * 1,
2036 ManagedMemoryPolicy::CUTOFF_ALLOW_EVERYTHING));
2037 break;
2038 case 2:
2039 // Only two backings should have memory.
2040 EXPECT_EQ(
2041 100 * 100 * 4 * 2,
2042 layer_tree_host()->contents_texture_manager()->MemoryUseBytes()) ;
2043 // Become backgrounded, which will cause 1 more resource to be
2044 // evicted.
2045 PostSetVisibleToMainThread(false);
2046 break;
2047 default:
2048 // No further commits should happen because this is not visible
2049 // anymore.
2050 NOTREACHED();
2051 break;
2052 }
2053 }
2054
2055 virtual void AfterTest() OVERRIDE
2056 {
2057 }
2058
2059 private:
2060 FakeContentLayerClient client_;
2061 scoped_refptr<FakeContentLayer> m_rootLayer;
2062 scoped_refptr<FakeContentLayer> m_childLayer1;
2063 scoped_refptr<FakeContentLayer> m_childLayer2;
2064 int m_numCommits;
2065 };
2066
2067 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestShutdownWithOnlySomeResourcesEvi cted)
2068
2069 class LayerTreeHostTestPinchZoomScrollbarCreation : public LayerTreeHostTest {
2070 public:
2071 LayerTreeHostTestPinchZoomScrollbarCreation()
2072 : m_rootLayer(ContentLayer::Create(&client_))
2073 {
2074 }
2075
2076 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE
2077 {
2078 settings->use_pinch_zoom_scrollbars = true;
2079 }
2080
2081 virtual void BeginTest() OVERRIDE
2082 {
2083 m_rootLayer->SetIsDrawable(true);
2084 m_rootLayer->SetBounds(gfx::Size(100, 100));
2085 layer_tree_host()->SetRootLayer(m_rootLayer);
2086 PostSetNeedsCommitToMainThread();
2087 }
2088
2089 virtual void DidCommit() OVERRIDE
2090 {
2091 // We always expect two pinch-zoom scrollbar layers.
2092 ASSERT_TRUE(2 == m_rootLayer->children().size());
2093
2094 // Pinch-zoom scrollbar layers always have invalid scrollLayerIds.
2095 ScrollbarLayer* layer1 = m_rootLayer->children()[0]->ToScrollbarLayer();
2096 ASSERT_TRUE(layer1);
2097 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID, layer1->scroll_layer_i d());
2098 EXPECT_EQ(0, layer1->opacity());
2099 EXPECT_TRUE(layer1->OpacityCanAnimateOnImplThread());
2100 EXPECT_TRUE(layer1->DrawsContent());
2101
2102 ScrollbarLayer* layer2 = m_rootLayer->children()[1]->ToScrollbarLayer();
2103 ASSERT_TRUE(layer2);
2104 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID, layer2->scroll_layer_i d());
2105 EXPECT_EQ(0, layer2->opacity());
2106 EXPECT_TRUE(layer2->OpacityCanAnimateOnImplThread());
2107 EXPECT_TRUE(layer2->DrawsContent());
2108
2109 EndTest();
2110 }
2111
2112 virtual void AfterTest() OVERRIDE
2113 {
2114 }
2115
2116 private:
2117 FakeContentLayerClient client_;
2118 scoped_refptr<ContentLayer> m_rootLayer;
2119 };
2120
2121 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarCreation)
2122
2123 class LayerTreeHostTestPinchZoomScrollbarResize : public LayerTreeHostTest {
2124 public:
2125 LayerTreeHostTestPinchZoomScrollbarResize()
2126 : m_rootLayer(ContentLayer::Create(&client_))
2127 , m_numCommits(0)
2128 {
2129 }
2130
2131 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE
2132 {
2133 settings->use_pinch_zoom_scrollbars = true;
2134 }
2135
2136 virtual void BeginTest() OVERRIDE
2137 {
2138 m_rootLayer->SetIsDrawable(true);
2139 m_rootLayer->SetBounds(gfx::Size(100, 100));
2140 layer_tree_host()->SetRootLayer(m_rootLayer);
2141 layer_tree_host()->SetViewportSize(gfx::Size(100, 100),
2142 gfx::Size(100, 100));
2143 PostSetNeedsCommitToMainThread();
2144 }
2145
2146 virtual void DidCommit() OVERRIDE
2147 {
2148 m_numCommits++;
2149
2150 ScrollbarLayer* layer1 = m_rootLayer->children()[0]->ToScrollbarLayer();
2151 ASSERT_TRUE(layer1);
2152 ScrollbarLayer* layer2 = m_rootLayer->children()[1]->ToScrollbarLayer();
2153 ASSERT_TRUE(layer2);
2154
2155 // Get scrollbar thickness from horizontal scrollbar's height.
2156 int thickness = layer1->bounds().height();
2157
2158 if (!layer1->Orientation() == WebKit::WebScrollbar::Horizontal)
2159 std::swap(layer1, layer2);
2160
2161 gfx::Size viewportSize = layer_tree_host()->layout_viewport_size();
2162 EXPECT_EQ(viewportSize.width() - thickness, layer1->bounds().width());
2163 EXPECT_EQ(viewportSize.height() - thickness, layer2->bounds().height());
2164
2165 switch (m_numCommits) {
2166 case 1:
2167 // Resizing the viewport should also resize the pinch-zoom scrollbars.
2168 layer_tree_host()->SetViewportSize(gfx::Size(120, 150),
2169 gfx::Size(120, 150));
2170 break;
2171 default:
2172 EndTest();
2173 }
2174 }
2175
2176 virtual void AfterTest() OVERRIDE
2177 {
2178 }
2179
2180 private:
2181 FakeContentLayerClient client_;
2182 scoped_refptr<ContentLayer> m_rootLayer;
2183 int m_numCommits;
2184 };
2185
2186 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarResize)
2187
2188 class LayerTreeHostTestPinchZoomScrollbarNewRootLayer : public LayerTreeHostTest {
2189 public:
2190 LayerTreeHostTestPinchZoomScrollbarNewRootLayer()
2191 : m_rootLayer(ContentLayer::Create(&client_))
2192 , m_numCommits(0)
2193 {
2194 }
2195
2196 virtual void InitializeSettings(LayerTreeSettings* settings) OVERRIDE
2197 {
2198 settings->use_pinch_zoom_scrollbars = true;
2199 }
2200
2201 virtual void BeginTest() OVERRIDE
2202 {
2203 m_rootLayer->SetIsDrawable(true);
2204 m_rootLayer->SetBounds(gfx::Size(100, 100));
2205 layer_tree_host()->SetRootLayer(m_rootLayer);
2206 PostSetNeedsCommitToMainThread();
2207 }
2208
2209 virtual void DidCommit() OVERRIDE
2210 {
2211 m_numCommits++;
2212
2213 // We always expect two pinch-zoom scrollbar layers.
2214 ASSERT_TRUE(2 == m_rootLayer->children().size());
2215
2216 // Pinch-zoom scrollbar layers always have invalid scrollLayerIds.
2217 ScrollbarLayer* layer1 = m_rootLayer->children()[0]->ToScrollbarLayer();
2218 ASSERT_TRUE(layer1);
2219 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID, layer1->scroll_layer_i d());
2220 EXPECT_EQ(0, layer1->opacity());
2221 EXPECT_TRUE(layer1->DrawsContent());
2222
2223 ScrollbarLayer* layer2 = m_rootLayer->children()[1]->ToScrollbarLayer();
2224 ASSERT_TRUE(layer2);
2225 EXPECT_EQ(Layer::PINCH_ZOOM_ROOT_SCROLL_LAYER_ID, layer2->scroll_layer_i d());
2226 EXPECT_EQ(0, layer2->opacity());
2227 EXPECT_TRUE(layer2->DrawsContent());
2228
2229 if (m_numCommits == 1) {
2230 // Create a new root layer and attach to tree to verify the pinch
2231 // zoom scrollbars get correctly re-attached.
2232 m_rootLayer = ContentLayer::Create(&client_);
2233 m_rootLayer->SetIsDrawable(true);
2234 m_rootLayer->SetBounds(gfx::Size(100, 100));
2235 layer_tree_host()->SetRootLayer(m_rootLayer);
2236 PostSetNeedsCommitToMainThread();
2237 } else
2238 EndTest();
2239 }
2240
2241 virtual void AfterTest() OVERRIDE
2242 {
2243 }
2244
2245 private:
2246 FakeContentLayerClient client_;
2247 scoped_refptr<ContentLayer> m_rootLayer;
2248 int m_numCommits;
2249 };
2250
2251 SINGLE_AND_MULTI_THREAD_TEST_F(LayerTreeHostTestPinchZoomScrollbarNewRootLayer)
2252
2253 class LayerTreeHostTestLCDNotification : public LayerTreeHostTest {
2254 public:
2255 class NotificationClient : public ContentLayerClient {
2256 public:
2257 NotificationClient()
2258 : layer_(0)
2259 , paint_count_(0)
2260 , lcd_notification_count_(0)
2261 {
2262 }
2263
2264 void set_layer(Layer* layer) { layer_ = layer; }
2265 int paint_count() const { return paint_count_; }
2266 int lcd_notification_count() const { return lcd_notification_count_; }
2267
2268 virtual void PaintContents(SkCanvas* canvas,
2269 gfx::Rect clip,
2270 gfx::RectF* opaque) OVERRIDE
2271 {
2272 ++paint_count_;
2273 }
2274 virtual void DidChangeLayerCanUseLCDText() OVERRIDE
2275 {
2276 ++lcd_notification_count_;
2277 layer_->SetNeedsDisplay();
2278 }
2279
2280 private:
2281 Layer* layer_;
2282 int paint_count_;
2283 int lcd_notification_count_;
2284 };
2285
2286 virtual void SetupTree() OVERRIDE
2287 {
2288 scoped_refptr<ContentLayer> root_layer = ContentLayer::Create(&client_);
2289 root_layer->SetIsDrawable(true);
2290 root_layer->SetBounds(gfx::Size(1, 1));
2291
2292 layer_tree_host()->SetRootLayer(root_layer);
2293 client_.set_layer(root_layer.get());
2294
2295 // The expecations are based on the assumption that the default
2296 // LCD settings are:
2297 EXPECT_TRUE(layer_tree_host()->settings().can_use_lcd_text);
2298 EXPECT_FALSE(root_layer->can_use_lcd_text());
2299
2300 LayerTreeHostTest::SetupTree();
2301 }
2302
2303 virtual void BeginTest() OVERRIDE
2304 {
2305 PostSetNeedsCommitToMainThread();
2306 }
2307 virtual void AfterTest() OVERRIDE { }
2308
2309 virtual void DidCommit() OVERRIDE
2310 {
2311 switch (layer_tree_host()->commit_number()) {
2312 case 1:
2313 // The first update consists one LCD notification and one paint.
2314 EXPECT_EQ(1, client_.lcd_notification_count());
2315 EXPECT_EQ(1, client_.paint_count());
2316 // LCD text must have been enabled on the layer.
2317 EXPECT_TRUE(layer_tree_host()->root_layer()->can_use_lcd_text());
2318 PostSetNeedsCommitToMainThread();
2319 break;
2320 case 2:
2321 // Since nothing changed on layer, there should be no notification
2322 // or paint on the second update.
2323 EXPECT_EQ(1, client_.lcd_notification_count());
2324 EXPECT_EQ(1, client_.paint_count());
2325 // LCD text must not have changed.
2326 EXPECT_TRUE(layer_tree_host()->root_layer()->can_use_lcd_text());
2327 // Change layer opacity that should trigger lcd notification.
2328 layer_tree_host()->root_layer()->SetOpacity(0.5);
2329 // No need to request a commit - setting opacity will do it.
2330 break;
2331 default:
2332 // Verify that there is not extra commit due to layer invalidation.
2333 EXPECT_EQ(3, layer_tree_host()->commit_number());
2334 // LCD notification count should have incremented due to
2335 // change in layer opacity.
2336 EXPECT_EQ(2, client_.lcd_notification_count());
2337 // Paint count should be incremented due to invalidation.
2338 EXPECT_EQ(2, client_.paint_count());
2339 // LCD text must have been disabled on the layer due to opacity.
2340 EXPECT_FALSE(layer_tree_host()->root_layer()->can_use_lcd_text());
2341 EndTest();
2342 break;
2343 }
2344 }
2345
2346 private:
2347 NotificationClient client_;
2348 };
2349
2350 SINGLE_THREAD_TEST_F(LayerTreeHostTestLCDNotification)
2351 2166
2352 } // namespace 2167 } // namespace
2353 } // namespace cc 2168 } // namespace cc
OLDNEW
« no previous file with comments | « no previous file | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698