// Copyright 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "cc/output/software_renderer.h" #include "cc/layers/quad_sink.h" #include "cc/output/compositor_frame_metadata.h" #include "cc/output/software_output_device.h" #include "cc/quads/render_pass.h" #include "cc/quads/render_pass_draw_quad.h" #include "cc/quads/solid_color_draw_quad.h" #include "cc/quads/tile_draw_quad.h" #include "cc/test/animation_test_common.h" #include "cc/test/fake_output_surface.h" #include "cc/test/fake_output_surface_client.h" #include "cc/test/geometry_test_utils.h" #include "cc/test/render_pass_test_common.h" #include "cc/test/render_pass_test_utils.h" #include "testing/gmock/include/gmock/gmock.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/skia/include/core/SkCanvas.h" #include "third_party/skia/include/core/SkDevice.h" namespace cc { namespace { class SoftwareRendererTest : public testing::Test, public RendererClient { public: SoftwareRendererTest() : should_clear_root_render_pass_(true) {} void InitializeRenderer( scoped_ptr software_output_device) { output_surface_ = FakeOutputSurface::CreateSoftware( software_output_device.Pass()); CHECK(output_surface_->BindToClient(&output_surface_client_)); resource_provider_ = ResourceProvider::Create(output_surface_.get(), 0); renderer_ = SoftwareRenderer::Create( this, output_surface_.get(), resource_provider()); } ResourceProvider* resource_provider() const { return resource_provider_.get(); } SoftwareRenderer* renderer() const { return renderer_.get(); } void set_viewport(gfx::Rect viewport) { viewport_ = viewport; } void set_should_clear_root_render_pass(bool clear_root_render_pass) { should_clear_root_render_pass_ = clear_root_render_pass; } // RendererClient implementation. virtual gfx::Rect DeviceViewport() const OVERRIDE { return viewport_; } virtual float DeviceScaleFactor() const OVERRIDE { return 1.f; } virtual const LayerTreeSettings& Settings() const OVERRIDE { return settings_; } virtual void SetFullRootLayerDamage() OVERRIDE {} virtual bool HasImplThread() const OVERRIDE { return false; } virtual bool ShouldClearRootRenderPass() const OVERRIDE { return should_clear_root_render_pass_; } virtual CompositorFrameMetadata MakeCompositorFrameMetadata() const OVERRIDE { return CompositorFrameMetadata(); } virtual bool AllowPartialSwap() const OVERRIDE { return true; } virtual bool ExternalStencilTestEnabled() const OVERRIDE { return false; } protected: FakeOutputSurfaceClient output_surface_client_; scoped_ptr output_surface_; scoped_ptr resource_provider_; scoped_ptr renderer_; gfx::Rect viewport_; LayerTreeSettings settings_; bool should_clear_root_render_pass_; }; TEST_F(SoftwareRendererTest, SolidColorQuad) { gfx::Size outer_size(100, 100); gfx::Size inner_size(98, 98); gfx::Rect outer_rect(outer_size); gfx::Rect inner_rect(gfx::Point(1, 1), inner_size); gfx::Rect visible_rect(gfx::Point(1, 2), gfx::Size(98, 97)); set_viewport(gfx::Rect(outer_size)); InitializeRenderer(make_scoped_ptr(new SoftwareOutputDevice)); scoped_ptr shared_quad_state = SharedQuadState::Create(); shared_quad_state->SetAll( gfx::Transform(), outer_size, outer_rect, outer_rect, false, 1.0); RenderPass::Id root_render_pass_id = RenderPass::Id(1, 1); scoped_ptr root_render_pass = TestRenderPass::Create(); root_render_pass->SetNew( root_render_pass_id, outer_rect, outer_rect, gfx::Transform()); scoped_ptr outer_quad = SolidColorDrawQuad::Create(); outer_quad->SetNew( shared_quad_state.get(), outer_rect, SK_ColorYELLOW, false); scoped_ptr inner_quad = SolidColorDrawQuad::Create(); inner_quad->SetNew(shared_quad_state.get(), inner_rect, SK_ColorCYAN, false); inner_quad->visible_rect = visible_rect; root_render_pass->AppendQuad(inner_quad.PassAs()); root_render_pass->AppendQuad(outer_quad.PassAs()); RenderPassList list; list.push_back(root_render_pass.PassAs()); renderer()->DrawFrame(&list, NULL); SkBitmap output; output.setConfig(SkBitmap::kARGB_8888_Config, DeviceViewport().width(), DeviceViewport().height()); output.allocPixels(); renderer()->GetFramebufferPixels(output.getPixels(), outer_rect); EXPECT_EQ(SK_ColorYELLOW, output.getColor(0, 0)); EXPECT_EQ(SK_ColorYELLOW, output.getColor(outer_size.width() - 1, outer_size.height() - 1)); EXPECT_EQ(SK_ColorYELLOW, output.getColor(1, 1)); EXPECT_EQ(SK_ColorCYAN, output.getColor(1, 2)); EXPECT_EQ(SK_ColorCYAN, output.getColor(inner_size.width() - 1, inner_size.height() - 1)); } TEST_F(SoftwareRendererTest, TileQuad) { gfx::Size outer_size(100, 100); gfx::Size inner_size(98, 98); gfx::Rect outer_rect(outer_size); gfx::Rect inner_rect(gfx::Point(1, 1), inner_size); set_viewport(gfx::Rect(outer_size)); InitializeRenderer(make_scoped_ptr(new SoftwareOutputDevice)); ResourceProvider::ResourceId resource_yellow = resource_provider()->CreateResource( outer_size, GL_RGBA, ResourceProvider::TextureUsageAny); ResourceProvider::ResourceId resource_cyan = resource_provider()->CreateResource( inner_size, GL_RGBA, ResourceProvider::TextureUsageAny); SkBitmap yellow_tile; yellow_tile.setConfig( SkBitmap::kARGB_8888_Config, outer_size.width(), outer_size.height()); yellow_tile.allocPixels(); yellow_tile.eraseColor(SK_ColorYELLOW); SkBitmap cyan_tile; cyan_tile.setConfig( SkBitmap::kARGB_8888_Config, inner_size.width(), inner_size.height()); cyan_tile.allocPixels(); cyan_tile.eraseColor(SK_ColorCYAN); resource_provider()->SetPixels( resource_yellow, static_cast(yellow_tile.getPixels()), gfx::Rect(outer_size), gfx::Rect(outer_size), gfx::Vector2d()); resource_provider()->SetPixels(resource_cyan, static_cast(cyan_tile.getPixels()), gfx::Rect(inner_size), gfx::Rect(inner_size), gfx::Vector2d()); gfx::Rect root_rect = DeviceViewport(); scoped_ptr shared_quad_state = SharedQuadState::Create(); shared_quad_state->SetAll( gfx::Transform(), outer_size, outer_rect, outer_rect, false, 1.0); RenderPass::Id root_render_pass_id = RenderPass::Id(1, 1); scoped_ptr root_render_pass = TestRenderPass::Create(); root_render_pass->SetNew( root_render_pass_id, root_rect, root_rect, gfx::Transform()); scoped_ptr outer_quad = TileDrawQuad::Create(); outer_quad->SetNew(shared_quad_state.get(), outer_rect, outer_rect, resource_yellow, gfx::RectF(outer_size), outer_size, false); scoped_ptr inner_quad = TileDrawQuad::Create(); inner_quad->SetNew(shared_quad_state.get(), inner_rect, inner_rect, resource_cyan, gfx::RectF(inner_size), inner_size, false); root_render_pass->AppendQuad(inner_quad.PassAs()); root_render_pass->AppendQuad(outer_quad.PassAs()); RenderPassList list; list.push_back(root_render_pass.PassAs()); renderer()->DrawFrame(&list, NULL); SkBitmap output; output.setConfig(SkBitmap::kARGB_8888_Config, DeviceViewport().width(), DeviceViewport().height()); output.allocPixels(); renderer()->GetFramebufferPixels(output.getPixels(), outer_rect); EXPECT_EQ(SK_ColorYELLOW, output.getColor(0, 0)); EXPECT_EQ(SK_ColorYELLOW, output.getColor(outer_size.width() - 1, outer_size.height() - 1)); EXPECT_EQ(SK_ColorCYAN, output.getColor(1, 1)); EXPECT_EQ(SK_ColorCYAN, output.getColor(inner_size.width() - 1, inner_size.height() - 1)); } TEST_F(SoftwareRendererTest, TileQuadVisibleRect) { gfx::Size tile_size(100, 100); gfx::Rect tile_rect(tile_size); gfx::Rect visible_rect = tile_rect; visible_rect.Inset(1, 2, 3, 4); set_viewport(gfx::Rect(tile_size)); InitializeRenderer(make_scoped_ptr(new SoftwareOutputDevice)); ResourceProvider::ResourceId resource_cyan = resource_provider()->CreateResource( tile_size, GL_RGBA, ResourceProvider::TextureUsageAny); SkBitmap cyan_tile; // The lowest five rows are yellow. cyan_tile.setConfig( SkBitmap::kARGB_8888_Config, tile_size.width(), tile_size.height()); cyan_tile.allocPixels(); cyan_tile.eraseColor(SK_ColorCYAN); cyan_tile.eraseArea( SkIRect::MakeLTRB( 0, visible_rect.bottom() - 1, tile_rect.width(), tile_rect.bottom()), SK_ColorYELLOW); resource_provider()->SetPixels(resource_cyan, static_cast(cyan_tile.getPixels()), gfx::Rect(tile_size), gfx::Rect(tile_size), gfx::Vector2d()); gfx::Rect root_rect = DeviceViewport(); scoped_ptr shared_quad_state = SharedQuadState::Create(); shared_quad_state->SetAll( gfx::Transform(), tile_size, tile_rect, tile_rect, false, 1.0); RenderPass::Id root_render_pass_id = RenderPass::Id(1, 1); scoped_ptr root_render_pass = TestRenderPass::Create(); root_render_pass->SetNew( root_render_pass_id, root_rect, root_rect, gfx::Transform()); scoped_ptr quad = TileDrawQuad::Create(); quad->SetNew(shared_quad_state.get(), tile_rect, tile_rect, resource_cyan, gfx::RectF(tile_size), tile_size, false); quad->visible_rect = visible_rect; root_render_pass->AppendQuad(quad.PassAs()); RenderPassList list; list.push_back(root_render_pass.PassAs()); renderer()->DrawFrame(&list, NULL); SkBitmap output; output.setConfig(SkBitmap::kARGB_8888_Config, DeviceViewport().width(), DeviceViewport().height()); output.allocPixels(); renderer()->GetFramebufferPixels(output.getPixels(), tile_rect); // Check portion of tile not in visible rect isn't drawn. const unsigned int kTransparent = SK_ColorTRANSPARENT; EXPECT_EQ(kTransparent, output.getColor(0, 0)); EXPECT_EQ(kTransparent, output.getColor(tile_rect.width() - 1, tile_rect.height() - 1)); EXPECT_EQ(kTransparent, output.getColor(visible_rect.x() - 1, visible_rect.y() - 1)); EXPECT_EQ(kTransparent, output.getColor(visible_rect.right(), visible_rect.bottom())); // Ensure visible part is drawn correctly. EXPECT_EQ(SK_ColorCYAN, output.getColor(visible_rect.x(), visible_rect.y())); EXPECT_EQ( SK_ColorCYAN, output.getColor(visible_rect.right() - 2, visible_rect.bottom() - 2)); // Ensure last visible line is correct. EXPECT_EQ( SK_ColorYELLOW, output.getColor(visible_rect.right() - 1, visible_rect.bottom() - 1)); } TEST_F(SoftwareRendererTest, ShouldClearRootRenderPass) { gfx::Rect viewport_rect(0, 0, 100, 100); set_viewport(viewport_rect); set_should_clear_root_render_pass(false); InitializeRenderer(make_scoped_ptr(new SoftwareOutputDevice)); RenderPassList list; SkBitmap output; output.setConfig(SkBitmap::kARGB_8888_Config, viewport_rect.width(), viewport_rect.height()); output.allocPixels(); // Draw a fullscreen green quad in a first frame. RenderPass::Id root_clear_pass_id(1, 0); TestRenderPass* root_clear_pass = AddRenderPass( &list, root_clear_pass_id, viewport_rect, gfx::Transform()); AddQuad(root_clear_pass, viewport_rect, SK_ColorGREEN); renderer()->DecideRenderPassAllocationsForFrame(list); renderer()->DrawFrame(&list, NULL); renderer()->GetFramebufferPixels(output.getPixels(), viewport_rect); EXPECT_EQ(SK_ColorGREEN, output.getColor(0, 0)); EXPECT_EQ(SK_ColorGREEN, output.getColor(viewport_rect.width() - 1, viewport_rect.height() - 1)); list.clear(); // Draw a smaller magenta rect without filling the viewport in a separate // frame. gfx::Rect smaller_rect(20, 20, 60, 60); RenderPass::Id root_smaller_pass_id(2, 0); TestRenderPass* root_smaller_pass = AddRenderPass( &list, root_smaller_pass_id, viewport_rect, gfx::Transform()); AddQuad(root_smaller_pass, smaller_rect, SK_ColorMAGENTA); renderer()->DecideRenderPassAllocationsForFrame(list); renderer()->DrawFrame(&list, NULL); renderer()->GetFramebufferPixels(output.getPixels(), viewport_rect); // If we didn't clear, the borders should still be green. EXPECT_EQ(SK_ColorGREEN, output.getColor(0, 0)); EXPECT_EQ(SK_ColorGREEN, output.getColor(viewport_rect.width() - 1, viewport_rect.height() - 1)); EXPECT_EQ(SK_ColorMAGENTA, output.getColor(smaller_rect.x(), smaller_rect.y())); EXPECT_EQ(SK_ColorMAGENTA, output.getColor(smaller_rect.right() - 1, smaller_rect.bottom() - 1)); } TEST_F(SoftwareRendererTest, RenderPassVisibleRect) { gfx::Rect viewport_rect(0, 0, 100, 100); set_viewport(viewport_rect); InitializeRenderer(make_scoped_ptr(new SoftwareOutputDevice)); RenderPassList list; SkBitmap output; output.setConfig(SkBitmap::kARGB_8888_Config, viewport_rect.width(), viewport_rect.height()); output.allocPixels(); // Pass drawn as inner quad is magenta. gfx::Rect smaller_rect(20, 20, 60, 60); RenderPass::Id smaller_pass_id(2, 1); TestRenderPass* smaller_pass = AddRenderPass(&list, smaller_pass_id, smaller_rect, gfx::Transform()); AddQuad(smaller_pass, smaller_rect, SK_ColorMAGENTA); // Root pass is green. RenderPass::Id root_clear_pass_id(1, 0); TestRenderPass* root_clear_pass = AddRenderPass(&list, root_clear_pass_id, viewport_rect, gfx::Transform()); AddRenderPassQuad(root_clear_pass, smaller_pass); AddQuad(root_clear_pass, viewport_rect, SK_ColorGREEN); // Interior pass quad has smaller visible rect. gfx::Rect interior_visible_rect(30, 30, 40, 40); root_clear_pass->quad_list[0]->visible_rect = interior_visible_rect; renderer()->DecideRenderPassAllocationsForFrame(list); renderer()->DrawFrame(&list, NULL); renderer()->GetFramebufferPixels(output.getPixels(), viewport_rect); EXPECT_EQ(SK_ColorGREEN, output.getColor(0, 0)); EXPECT_EQ( SK_ColorGREEN, output.getColor(viewport_rect.width() - 1, viewport_rect.height() - 1)); // Part outside visible rect should remain green. EXPECT_EQ(SK_ColorGREEN, output.getColor(smaller_rect.x(), smaller_rect.y())); EXPECT_EQ( SK_ColorGREEN, output.getColor(smaller_rect.right() - 1, smaller_rect.bottom() - 1)); EXPECT_EQ( SK_ColorMAGENTA, output.getColor(interior_visible_rect.x(), interior_visible_rect.y())); EXPECT_EQ(SK_ColorMAGENTA, output.getColor(interior_visible_rect.right() - 1, interior_visible_rect.bottom() - 1)); } } // namespace } // namespace cc