// Copyright 2014 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 "base/memory/scoped_ptr.h" #include "base/message_loop/message_loop.h" #include "cc/output/software_frame_data.h" #include "content/browser/compositor/software_output_device_ozone.h" #include "testing/gtest/include/gtest/gtest.h" #include "third_party/skia/include/core/SkSurface.h" #include "ui/compositor/compositor.h" #include "ui/compositor/test/context_factories_for_test.h" #include "ui/gfx/ozone/surface_factory_ozone.h" #include "ui/gfx/ozone/surface_ozone_canvas.h" #include "ui/gfx/size.h" #include "ui/gfx/skia_util.h" #include "ui/gfx/vsync_provider.h" #include "ui/gl/gl_implementation.h" namespace { class MockSurfaceOzone : public gfx::SurfaceOzoneCanvas { public: MockSurfaceOzone() {} virtual ~MockSurfaceOzone() {} // gfx::SurfaceOzoneCanvas overrides: virtual void ResizeCanvas(const gfx::Size& size) OVERRIDE { surface_ = skia::AdoptRef(SkSurface::NewRaster( SkImageInfo::MakeN32Premul(size.width(), size.height()))); } virtual skia::RefPtr GetCanvas() OVERRIDE { return skia::SharePtr(surface_->getCanvas()); } virtual void PresentCanvas(const gfx::Rect& damage) OVERRIDE {} virtual scoped_ptr CreateVSyncProvider() OVERRIDE { return scoped_ptr(); } private: skia::RefPtr surface_; DISALLOW_COPY_AND_ASSIGN(MockSurfaceOzone); }; class MockSurfaceFactoryOzone : public gfx::SurfaceFactoryOzone { public: MockSurfaceFactoryOzone() {} virtual ~MockSurfaceFactoryOzone() {} virtual HardwareState InitializeHardware() OVERRIDE { return SurfaceFactoryOzone::INITIALIZED; } virtual void ShutdownHardware() OVERRIDE {} virtual gfx::AcceleratedWidget GetAcceleratedWidget() OVERRIDE { return 1; } virtual bool LoadEGLGLES2Bindings( AddGLLibraryCallback add_gl_library, SetGLGetProcAddressProcCallback set_gl_get_proc_address) OVERRIDE { return false; } virtual scoped_ptr CreateCanvasForWidget( gfx::AcceleratedWidget widget) OVERRIDE { return make_scoped_ptr(new MockSurfaceOzone()); } private: DISALLOW_COPY_AND_ASSIGN(MockSurfaceFactoryOzone); }; } // namespace class SoftwareOutputDeviceOzoneTest : public testing::Test { public: SoftwareOutputDeviceOzoneTest(); virtual ~SoftwareOutputDeviceOzoneTest(); virtual void SetUp() OVERRIDE; virtual void TearDown() OVERRIDE; protected: scoped_ptr output_device_; bool enable_pixel_output_; private: scoped_ptr compositor_; scoped_ptr message_loop_; scoped_ptr surface_factory_; DISALLOW_COPY_AND_ASSIGN(SoftwareOutputDeviceOzoneTest); }; SoftwareOutputDeviceOzoneTest::SoftwareOutputDeviceOzoneTest() : enable_pixel_output_(false) { message_loop_.reset(new base::MessageLoopForUI); } SoftwareOutputDeviceOzoneTest::~SoftwareOutputDeviceOzoneTest() { } void SoftwareOutputDeviceOzoneTest::SetUp() { ui::InitializeContextFactoryForTests(enable_pixel_output_); ui::Compositor::Initialize(); surface_factory_.reset(new MockSurfaceFactoryOzone()); gfx::SurfaceFactoryOzone::SetInstance(surface_factory_.get()); const gfx::Size size(500, 400); compositor_.reset(new ui::Compositor( gfx::SurfaceFactoryOzone::GetInstance()->GetAcceleratedWidget())); compositor_->SetScaleAndSize(1.0f, size); output_device_.reset(new content::SoftwareOutputDeviceOzone( compositor_.get())); output_device_->Resize(size); } void SoftwareOutputDeviceOzoneTest::TearDown() { output_device_.reset(); compositor_.reset(); surface_factory_.reset(); ui::TerminateContextFactoryForTests(); ui::Compositor::Terminate(); } class SoftwareOutputDeviceOzonePixelTest : public SoftwareOutputDeviceOzoneTest { protected: virtual void SetUp() OVERRIDE; }; void SoftwareOutputDeviceOzonePixelTest::SetUp() { enable_pixel_output_ = true; SoftwareOutputDeviceOzoneTest::SetUp(); } TEST_F(SoftwareOutputDeviceOzoneTest, CheckCorrectResizeBehavior) { gfx::Rect damage(0, 0, 100, 100); gfx::Size size(200, 100); // Reduce size. output_device_->Resize(size); SkCanvas* canvas = output_device_->BeginPaint(damage); gfx::Size canvas_size(canvas->getDeviceSize().width(), canvas->getDeviceSize().height()); EXPECT_EQ(size.ToString(), canvas_size.ToString()); size.SetSize(1000, 500); // Increase size. output_device_->Resize(size); canvas = output_device_->BeginPaint(damage); canvas_size.SetSize(canvas->getDeviceSize().width(), canvas->getDeviceSize().height()); EXPECT_EQ(size.ToString(), canvas_size.ToString()); } TEST_F(SoftwareOutputDeviceOzonePixelTest, CheckCopyToBitmap) { const int width = 6; const int height = 4; const gfx::Rect area(width, height); output_device_->Resize(area.size()); SkCanvas* canvas = output_device_->BeginPaint(area); // Clear the background to black. canvas->drawColor(SK_ColorBLACK); cc::SoftwareFrameData frame; output_device_->EndPaint(&frame); // Draw a white rectangle. gfx::Rect damage(area.width() / 2, area.height() / 2); canvas = output_device_->BeginPaint(damage); canvas->clipRect(gfx::RectToSkRect(damage), SkRegion::kReplace_Op); canvas->drawColor(SK_ColorWHITE); output_device_->EndPaint(&frame); SkPMColor pixels[width * height]; output_device_->CopyToPixels(area, pixels); // Check that the copied bitmap contains the same pixel values as what we // painted. const SkPMColor white = SkPreMultiplyColor(SK_ColorWHITE); const SkPMColor black = SkPreMultiplyColor(SK_ColorBLACK); for (int i = 0; i < area.height(); ++i) { for (int j = 0; j < area.width(); ++j) { if (j < damage.width() && i < damage.height()) EXPECT_EQ(white, pixels[i * area.width() + j]); else EXPECT_EQ(black, pixels[i * area.width() + j]); } } }