// Copyright 2013 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/callback.h" #include "base/memory/scoped_ptr.h" #include "cc/debug/micro_benchmark.h" #include "cc/debug/micro_benchmark_controller.h" #include "cc/layers/layer.h" #include "cc/resources/resource_update_queue.h" #include "cc/test/fake_layer_tree_host.h" #include "cc/test/fake_layer_tree_host_impl.h" #include "cc/test/fake_proxy.h" #include "testing/gtest/include/gtest/gtest.h" namespace cc { namespace { class MicroBenchmarkControllerTest : public testing::Test { public: virtual void SetUp() OVERRIDE { impl_proxy_ = make_scoped_ptr(new FakeImplProxy); layer_tree_host_impl_ = make_scoped_ptr(new FakeLayerTreeHostImpl(impl_proxy_.get())); layer_tree_host_ = FakeLayerTreeHost::Create(); layer_tree_host_->SetRootLayer(Layer::Create()); layer_tree_host_->InitializeForTesting(scoped_ptr(new FakeProxy)); } virtual void TearDown() OVERRIDE { layer_tree_host_impl_.reset(); layer_tree_host_.reset(); impl_proxy_.reset(); } scoped_ptr layer_tree_host_; scoped_ptr layer_tree_host_impl_; scoped_ptr impl_proxy_; }; void Noop(scoped_ptr value) { } void IncrementCallCount(int* count, scoped_ptr value) { ++(*count); } TEST_F(MicroBenchmarkControllerTest, ScheduleFail) { bool result = layer_tree_host_->ScheduleMicroBenchmark( "non_existant_benchmark", scoped_ptr(), base::Bind(&Noop)); EXPECT_FALSE(result); } TEST_F(MicroBenchmarkControllerTest, CommitScheduled) { EXPECT_FALSE(layer_tree_host_->needs_commit()); bool result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&Noop)); EXPECT_TRUE(result); EXPECT_TRUE(layer_tree_host_->needs_commit()); } TEST_F(MicroBenchmarkControllerTest, BenchmarkRan) { int run_count = 0; bool result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); scoped_ptr queue(new ResourceUpdateQueue); layer_tree_host_->SetOutputSurfaceLostForTesting(false); layer_tree_host_->UpdateLayers(queue.get()); EXPECT_EQ(1, run_count); } TEST_F(MicroBenchmarkControllerTest, MultipleBenchmarkRan) { int run_count = 0; bool result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); scoped_ptr queue(new ResourceUpdateQueue); layer_tree_host_->SetOutputSurfaceLostForTesting(false); layer_tree_host_->UpdateLayers(queue.get()); EXPECT_EQ(2, run_count); result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", scoped_ptr(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); layer_tree_host_->UpdateLayers(queue.get()); EXPECT_EQ(4, run_count); layer_tree_host_->UpdateLayers(queue.get()); EXPECT_EQ(4, run_count); } TEST_F(MicroBenchmarkControllerTest, BenchmarkImplRan) { int run_count = 0; scoped_ptr settings(new base::DictionaryValue); settings->SetBoolean("run_benchmark_impl", true); // Schedule a main thread benchmark. bool result = layer_tree_host_->ScheduleMicroBenchmark( "unittest_only_benchmark", settings.PassAs(), base::Bind(&IncrementCallCount, base::Unretained(&run_count))); EXPECT_TRUE(result); // Schedule impl benchmarks. In production code, this is run in commit. layer_tree_host_->GetMicroBenchmarkController()->ScheduleImplBenchmarks( layer_tree_host_impl_.get()); // Now complete the commit (as if on the impl thread). layer_tree_host_impl_->CommitComplete(); // Make sure all posted messages run. base::MessageLoop::current()->RunUntilIdle(); EXPECT_EQ(1, run_count); } } // namespace } // namespace cc