Chromium Code Reviews| Index: cc/trees/layer_tree_host_unittest.cc |
| diff --git a/cc/trees/layer_tree_host_unittest.cc b/cc/trees/layer_tree_host_unittest.cc |
| index aedd0d937a43a69ffce015ea7507f522556a406a..a014ae795078f189911b3de9024835edad5bcaf7 100644 |
| --- a/cc/trees/layer_tree_host_unittest.cc |
| +++ b/cc/trees/layer_tree_host_unittest.cc |
| @@ -26,9 +26,11 @@ |
| #include "cc/output/output_surface.h" |
| #include "cc/quads/draw_quad.h" |
| #include "cc/quads/io_surface_draw_quad.h" |
| +#include "cc/quads/tile_draw_quad.h" |
| #include "cc/resources/prioritized_resource.h" |
| #include "cc/resources/prioritized_resource_manager.h" |
| #include "cc/resources/resource_update_queue.h" |
| +#include "cc/test/delayed_raster_trigger.h" |
| #include "cc/test/fake_content_layer.h" |
| #include "cc/test/fake_content_layer_client.h" |
| #include "cc/test/fake_content_layer_impl.h" |
| @@ -5205,4 +5207,342 @@ class LayerTreeHostAcceptsDeltasFromImplWithoutRootLayer |
| }; |
| MULTI_THREAD_TEST_F(LayerTreeHostAcceptsDeltasFromImplWithoutRootLayer); |
| + |
| +class LayerTreeHostTestCrispUpAfterPinchEnds : public LayerTreeHostTest { |
| + protected: |
| + void InitializeSettings(LayerTreeSettings* settings) override { |
| + settings->impl_side_painting = true; |
| + } |
| + |
| + void SetupTree() override { |
| + frame_ = 1; |
| + posted_ = false; |
| + client_.set_fill_with_nonsolid_color(true); |
| + |
| + scoped_refptr<Layer> root = Layer::Create(); |
| + root->SetBounds(gfx::Size(500, 500)); |
| + |
| + scoped_refptr<Layer> pinch = Layer::Create(); |
| + pinch->SetBounds(gfx::Size(500, 500)); |
| + pinch->SetScrollClipLayerId(root->id()); |
| + pinch->SetIsContainerForFixedPositionLayers(true); |
| + root->AddChild(pinch); |
| + |
| + scoped_refptr<FakePictureLayer> layer = FakePictureLayer::Create(&client_); |
| + layer->SetBounds(gfx::Size(500, 500)); |
| + layer->SetContentsOpaque(true); |
| + layer->SetUseDelayedRaster(&delayed_raster_trigger_); |
| + pinch->AddChild(layer); |
| + |
| + layer_tree_host()->RegisterViewportLayers(root, pinch, pinch); |
| + layer_tree_host()->SetPageScaleFactorAndLimits(1.f, 1.f, 4.f); |
| + layer_tree_host()->SetRootLayer(root); |
| + LayerTreeHostTest::SetupTree(); |
| + } |
| + |
| + // Returns the delta scale of all quads in the frame's root pass from their |
| + // ideal, or 0 if they are not all the same. |
| + float FrameQuadScaleDeltaFromIdeal(LayerTreeHostImpl::FrameData* frame_data) { |
| + if (frame_data->has_no_damage) |
| + return 0.f; |
| + float frame_scale = 0.f; |
| + RenderPass* root_pass = frame_data->render_passes.back(); |
| + for (const auto& draw_quad : root_pass->quad_list) { |
| + // Checkerboards mean an incomplete frame. |
| + if (draw_quad->material != DrawQuad::TILED_CONTENT) |
| + return 0.f; |
| + const TileDrawQuad* quad = TileDrawQuad::MaterialCast(draw_quad); |
| + float quad_scale = |
| + quad->tex_coord_rect.width() / static_cast<float>(quad->rect.width()); |
| + float transform_scale = |
| + SkMScalarToFloat(quad->quadTransform().matrix().get(0, 0)); |
| + float scale = quad_scale / transform_scale; |
| + if (frame_scale != 0.f && frame_scale != scale) |
| + return 0.f; |
| + frame_scale = scale; |
| + } |
| + return frame_scale; |
| + } |
| + |
| + void BeginTest() override { PostSetNeedsCommitToMainThread(); } |
| + |
| + DrawResult PrepareToDrawOnThread(LayerTreeHostImpl* host_impl, |
| + LayerTreeHostImpl::FrameData* frame_data, |
| + DrawResult draw_result) override { |
| + float quad_scale_delta = FrameQuadScaleDeltaFromIdeal(frame_data); |
| + switch (frame_) { |
| + case 1: |
| + // Drew at page scale 1 before any pinching. |
| + EXPECT_EQ(1.f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f, quad_scale_delta); |
| + PostNextAfterDraw(host_impl); |
| + break; |
| + case 2: |
| + if (quad_scale_delta != 1.f) |
| + break; |
| + // Drew at page scale 2.2 after pinching in. |
| + EXPECT_EQ(2.2f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f, quad_scale_delta); |
| + PostNextAfterDraw(host_impl); |
| + break; |
| + case 3: |
| + if (quad_scale_delta != 2.2f) |
| + break; |
| + // Drew at page scale 1 with the 2.2 tiling while pinching out. |
| + EXPECT_EQ(1.f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(2.2f, quad_scale_delta); |
| + PostNextAfterDraw(host_impl); |
| + break; |
| + case 4: |
| + // Drew at page scale 1 with the 2.2 tiling after pinching out completed |
| + // while waiting for texture uploads to complete. |
| + EXPECT_EQ(1.f, host_impl->active_tree()->total_page_scale_factor()); |
| + // This frame will not have any damage, since it's actually the same as |
| + // the last frame, and should contain no incomplete tiles. We just want |
| + // to make sure we drew here at least once after the pinch ended to be |
| + // sure that drawing after pinch doesn't leave us at the wrong scale |
| + // forever. |
| + EXPECT_TRUE(frame_data->has_no_damage); |
| + PostNextAfterDraw(host_impl); |
| + break; |
| + case 5: |
| + if (quad_scale_delta != 1.f) |
| + break; |
| + // Drew at scale 1 after texture uploads are done. |
| + EXPECT_EQ(1.f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f, quad_scale_delta); |
| + EndTest(); |
| + break; |
| + } |
| + return draw_result; |
| + } |
| + |
| + void PostNextAfterDraw(LayerTreeHostImpl* host_impl) { |
| + if (posted_) |
| + return; |
| + posted_ = true; |
| + ImplThreadTaskRunner()->PostDelayedTask( |
| + FROM_HERE, base::Bind(&LayerTreeHostTestCrispUpAfterPinchEnds::Next, |
| + base::Unretained(this), host_impl), |
| + // Use a delay to allow raster/upload to happen in between frames. This |
| + // should cause flakiness if we fail to block raster/upload when |
| + // desired. |
| + base::TimeDelta::FromMilliseconds(16 * 6)); |
| + } |
| + |
| + void Next(LayerTreeHostImpl* host_impl) { |
| + ++frame_; |
| + posted_ = false; |
| + switch (frame_) { |
| + case 2: |
| + // Pinch zoom in. |
| + host_impl->PinchGestureBegin(); |
| + host_impl->PinchGestureUpdate(2.2f, gfx::Point(100, 100)); |
| + host_impl->PinchGestureEnd(); |
| + break; |
| + case 3: |
| + // Pinch zoom back to 1.f but don't end it. |
| + host_impl->PinchGestureBegin(); |
| + host_impl->PinchGestureUpdate(1.f / 2.2f, gfx::Point(100, 100)); |
| + break; |
| + case 4: |
| + // End the pinch, but delay tile production. |
| + delayed_raster_trigger_.BeginDelay(); |
| + host_impl->PinchGestureEnd(); |
| + break; |
| + case 5: |
| + // Let tiles complete. |
| + delayed_raster_trigger_.EndDelay(); |
| + // TestContext()->set_query_result_available_ext(1); |
| + break; |
| + } |
| + } |
| + |
| + void NotifyTileStateChangedOnThread(LayerTreeHostImpl* host_impl, |
| + const Tile* tile) override { |
| + // On frame_ == 4, we are preventing texture uploads from completing, |
| + // so this verifies they are not completing before frame_ == 5. |
| + // Flaky failures here indicate we're failing to prevent uploads from |
| + // completing. |
| + EXPECT_NE(4, frame_); |
| + } |
| + |
| + void AfterTest() override {} |
| + |
| + FakeContentLayerClient client_; |
| + int frame_; |
| + bool posted_; |
| + DelayedRasterTrigger delayed_raster_trigger_; |
| +}; |
| + |
| +MULTI_THREAD_TEST_F(LayerTreeHostTestCrispUpAfterPinchEnds); |
| + |
| +class LayerTreeHostTestCrispUpAfterPinchEndsWithOneCopy |
| + : public LayerTreeHostTestCrispUpAfterPinchEnds { |
| + protected: |
| + void InitializeSettings(LayerTreeSettings* settings) override { |
| + settings->impl_side_painting = true; |
| + settings->use_one_copy = true; |
| + } |
| + |
| + scoped_ptr<FakeOutputSurface> CreateFakeOutputSurface( |
| + bool fallback) override { |
| + scoped_ptr<TestWebGraphicsContext3D> context3d = |
| + TestWebGraphicsContext3D::Create(); |
| + context3d->set_support_image(true); |
| + context3d->set_support_sync_query(true); |
| + |
| + if (delegating_renderer()) |
| + return FakeOutputSurface::CreateDelegating3d(context3d.Pass()); |
| + else |
| + return FakeOutputSurface::Create3d(context3d.Pass()); |
| + } |
| +}; |
| + |
| +MULTI_THREAD_TEST_F(LayerTreeHostTestCrispUpAfterPinchEndsWithOneCopy); |
| + |
| +class LayerTreeHostTestContinuousDrawWhenCreatingVisibleTiles |
| + : public LayerTreeHostTest { |
| + protected: |
| + void InitializeSettings(LayerTreeSettings* settings) override { |
| + settings->impl_side_painting = true; |
| + } |
| + |
| + void SetupTree() override { |
| + step_ = 1; |
| + continuous_draws_ = 0; |
| + client_.set_fill_with_nonsolid_color(true); |
| + |
| + scoped_refptr<Layer> root = Layer::Create(); |
| + root->SetBounds(gfx::Size(500, 500)); |
| + |
| + scoped_refptr<Layer> pinch = Layer::Create(); |
| + pinch->SetBounds(gfx::Size(500, 500)); |
| + pinch->SetScrollClipLayerId(root->id()); |
| + pinch->SetIsContainerForFixedPositionLayers(true); |
| + root->AddChild(pinch); |
| + |
| + scoped_refptr<FakePictureLayer> layer = FakePictureLayer::Create(&client_); |
| + layer->SetBounds(gfx::Size(500, 500)); |
| + layer->SetContentsOpaque(true); |
| + layer->SetUseDelayedRaster(&delayed_raster_trigger_); |
| + pinch->AddChild(layer); |
| + |
| + layer_tree_host()->RegisterViewportLayers(root, pinch, pinch); |
| + layer_tree_host()->SetPageScaleFactorAndLimits(1.f, 1.f, 4.f); |
| + layer_tree_host()->SetRootLayer(root); |
| + LayerTreeHostTest::SetupTree(); |
| + } |
| + |
| + // Returns the delta scale of all quads in the frame's root pass from their |
| + // ideal, or 0 if they are not all the same. |
| + float FrameQuadScaleDeltaFromIdeal(LayerTreeHostImpl::FrameData* frame_data) { |
| + if (frame_data->has_no_damage) |
| + return 0.f; |
| + float frame_scale = 0.f; |
| + RenderPass* root_pass = frame_data->render_passes.back(); |
| + for (const auto& draw_quad : root_pass->quad_list) { |
| + const TileDrawQuad* quad = TileDrawQuad::MaterialCast(draw_quad); |
| + float quad_scale = |
| + quad->tex_coord_rect.width() / static_cast<float>(quad->rect.width()); |
| + float transform_scale = |
| + SkMScalarToFloat(quad->quadTransform().matrix().get(0, 0)); |
| + float scale = quad_scale / transform_scale; |
| + if (frame_scale != 0.f && frame_scale != scale) |
| + return 0.f; |
| + frame_scale = scale; |
| + } |
| + return frame_scale; |
| + } |
| + |
| + void BeginTest() override { PostSetNeedsCommitToMainThread(); } |
| + |
| + DrawResult PrepareToDrawOnThread(LayerTreeHostImpl* host_impl, |
| + LayerTreeHostImpl::FrameData* frame_data, |
| + DrawResult draw_result) override { |
| + float quad_scale_delta = FrameQuadScaleDeltaFromIdeal(frame_data); |
| + switch (step_) { |
| + case 1: |
| + // Drew at scale 1 before any pinching. |
| + EXPECT_EQ(1.f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f, quad_scale_delta); |
| + break; |
| + case 2: |
| + if (quad_scale_delta != 1.f / 1.5f) |
| + break; |
| + // Drew at scale 1 still though the ideal is 1.5. |
| + EXPECT_EQ(1.5f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f / 1.5f, quad_scale_delta); |
| + break; |
| + case 3: |
| + // Continuous draws are attempted. |
| + EXPECT_EQ(1.5f, host_impl->active_tree()->total_page_scale_factor()); |
| + if (!frame_data->has_no_damage) |
| + EXPECT_EQ(1.f / 1.5f, quad_scale_delta); |
| + break; |
| + case 4: |
| + if (quad_scale_delta != 1.f) |
| + break; |
| + // Drew at scale 1.5 when all the tiles completed. |
| + EXPECT_EQ(1.5f, host_impl->active_tree()->total_page_scale_factor()); |
| + EXPECT_EQ(1.f, quad_scale_delta); |
| + |
| + // We should not continue to draw any more. End the test after a timeout |
| + // to watch for any extraneous draws. |
| + EndTestAfterDelayMs(16 * 4); |
|
brianderson
2014/11/10 22:12:31
If there is a failure, this could be flaky, but I
danakj
2014/11/12 20:19:40
Done.
danakj
2014/11/12 20:19:40
Done.
|
| + ++step_; |
| + break; |
| + case 5: |
| + // No draws should happen once we have a complete frame. |
| + EXPECT_TRUE(false); |
|
brianderson
2014/11/10 22:12:31
ADD_FAILURE() << "No draws should happen once we h
danakj
2014/11/12 20:19:40
Done.
|
| + break; |
| + } |
| + return draw_result; |
| + } |
| + |
| + void DrawLayersOnThread(LayerTreeHostImpl* host_impl) override { |
| + switch (step_) { |
| + case 1: |
| + // Delay tile production. |
| + delayed_raster_trigger_.BeginDelay(); |
| + // Pinch zoom in to cause new tiles to be required. |
| + host_impl->PinchGestureBegin(); |
| + host_impl->PinchGestureUpdate(1.5f, gfx::Point(100, 100)); |
| + host_impl->PinchGestureEnd(); |
| + ++step_; |
| + break; |
| + case 2: |
| + ++step_; |
| + break; |
| + case 3: |
| + // We should continue to try draw while there are incomplete visible |
| + // tiles. |
| + if (++continuous_draws_ > 5) { |
| + // Allow the tiles to complete. |
| + delayed_raster_trigger_.EndDelay(); |
| + ++step_; |
| + } |
| + break; |
| + } |
| + } |
| + |
| + void NotifyTileStateChangedOnThread(LayerTreeHostImpl* host_impl, |
| + const Tile* tile) override { |
| + // On step_ == 2, we are preventing texture uploads from completing, |
| + // so this verifies they are not completing before step_ == 3. |
| + // Flaky failures here indicate we're failing to prevent uploads from |
| + // completing. |
| + EXPECT_NE(2, step_); |
| + } |
| + |
| + void AfterTest() override { EXPECT_GT(continuous_draws_, 5); } |
| + |
| + FakeContentLayerClient client_; |
| + int step_; |
| + int continuous_draws_; |
| + DelayedRasterTrigger delayed_raster_trigger_; |
| +}; |
| + |
| +MULTI_THREAD_TEST_F(LayerTreeHostTestContinuousDrawWhenCreatingVisibleTiles); |
| + |
| } // namespace cc |