blob: 222fa55612b403e51a1745c23824311a7db17cfb [file] [log] [blame]
// Copyright 2015 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 <stdint.h>
#include "base/bind.h"
#include "base/memory/scoped_ptr.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/gl/gl_context_stub_with_extensions.h"
#include "ui/gl/gl_implementation.h"
#include "ui/gl/gl_mock.h"
#include "ui/gl/gpu_preference.h"
#include "ui/gl/gpu_timing.h"
#include "ui/gl/gpu_timing_fake.h"
#include "ui/gl/test/gl_surface_test_support.h"
namespace gfx {
class GPUTimingTest : public testing::Test {
public:
void SetUp() override {
setup_ = false;
cpu_time_bounded_ = false;
}
void TearDown() override {
if (setup_) {
MockGLInterface::SetGLInterface(NULL);
gfx::ClearGLBindings();
}
setup_ = false;
cpu_time_bounded_ = false;
context_ = nullptr;
gl_.reset();
gpu_timing_fake_queries_.Reset();
}
void SetupGLContext(const char* gl_version, const char* gl_extensions) {
ASSERT_FALSE(setup_) << "Cannot setup GL context twice.";
SetGLGetProcAddressProc(MockGLInterface::GetGLProcAddress);
GLSurfaceTestSupport::InitializeOneOffWithMockBindings();
gl_.reset(new ::testing::StrictMock<MockGLInterface>());
MockGLInterface::SetGLInterface(gl_.get());
context_ = new GLContextStubWithExtensions;
context_->AddExtensionsString(gl_extensions);
context_->SetGLVersionString(gl_version);
gpu_timing_fake_queries_.Reset();
GLSurfaceTestSupport::InitializeDynamicMockBindings(context_.get());
setup_ = true;
}
scoped_refptr<GPUTimingClient> CreateGPUTimingClient() {
if (!setup_) {
SetupGLContext("2.0", "");
}
scoped_refptr<GPUTimingClient> client = context_->CreateGPUTimingClient();
if (!cpu_time_bounded_) {
client->SetCpuTimeForTesting(base::Bind(&GPUTimingFake::GetFakeCPUTime));
cpu_time_bounded_ = true;
}
return client;
}
protected:
bool setup_ = false;
bool cpu_time_bounded_ = false;
scoped_ptr< ::testing::StrictMock<MockGLInterface> > gl_;
scoped_refptr<GLContextStubWithExtensions> context_;
GPUTimingFake gpu_timing_fake_queries_;
};
TEST_F(GPUTimingTest, FakeTimerTest) {
scoped_refptr<GPUTimingClient> gpu_timing_client = CreateGPUTimingClient();
// Tests that we can properly set fake cpu times.
gpu_timing_fake_queries_.SetCurrentCPUTime(123);
EXPECT_EQ(123, gpu_timing_client->GetCurrentCPUTime());
base::Callback<int64_t(void)> empty;
gpu_timing_client->SetCpuTimeForTesting(empty);
EXPECT_NE(123, gpu_timing_client->GetCurrentCPUTime());
}
TEST_F(GPUTimingTest, ForceTimeElapsedQuery) {
// Test that forcing time elapsed query affects all clients.
scoped_refptr<GPUTimingClient> client1 = CreateGPUTimingClient();
EXPECT_FALSE(client1->IsForceTimeElapsedQuery());
scoped_refptr<GPUTimingClient> client_force = CreateGPUTimingClient();
EXPECT_FALSE(client1->IsForceTimeElapsedQuery());
client_force->ForceTimeElapsedQuery();
EXPECT_TRUE(client1->IsForceTimeElapsedQuery());
EXPECT_TRUE(client1->IsForceTimeElapsedQuery());
scoped_refptr<GPUTimingClient> client2 = CreateGPUTimingClient();
EXPECT_TRUE(client2->IsForceTimeElapsedQuery());
}
TEST_F(GPUTimingTest, QueryTimeStampTest) {
SetupGLContext("3.2", "GL_ARB_timer_query");
scoped_refptr<GPUTimingClient> client = CreateGPUTimingClient();
scoped_ptr<GPUTimer> gpu_timer = client->CreateGPUTimer(false);
const int64_t begin_cpu_time = 123;
const int64_t begin_gl_time = 10 * base::Time::kNanosecondsPerMicrosecond;
const int64_t cpu_gl_offset =
begin_gl_time / base::Time::kNanosecondsPerMicrosecond - begin_cpu_time;
gpu_timing_fake_queries_.SetCPUGLOffset(cpu_gl_offset);
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time);
gpu_timing_fake_queries_.ExpectGPUTimeStampQuery(*gl_, false);
gpu_timer->QueryTimeStamp();
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time - 1);
EXPECT_FALSE(gpu_timer->IsAvailable());
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time);
EXPECT_TRUE(gpu_timer->IsAvailable());
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time + 1);
EXPECT_TRUE(gpu_timer->IsAvailable());
EXPECT_EQ(0, gpu_timer->GetDeltaElapsed());
int64_t start, end;
gpu_timer->GetStartEndTimestamps(&start, &end);
EXPECT_EQ(begin_cpu_time, start);
EXPECT_EQ(begin_cpu_time, end);
}
TEST_F(GPUTimingTest, QueryTimeStampUsingElapsedTest) {
// Test timestamp queries using GL_EXT_timer_query which does not support
// timestamp queries. Internally we fall back to time elapsed queries.
SetupGLContext("3.2", "GL_EXT_timer_query");
scoped_refptr<GPUTimingClient> client = CreateGPUTimingClient();
scoped_ptr<GPUTimer> gpu_timer = client->CreateGPUTimer(false);
ASSERT_TRUE(client->IsForceTimeElapsedQuery());
const int64_t begin_cpu_time = 123;
const int64_t begin_gl_time = 10 * base::Time::kNanosecondsPerMicrosecond;
const int64_t cpu_gl_offset = begin_gl_time - begin_cpu_time;
gpu_timing_fake_queries_.SetCPUGLOffset(cpu_gl_offset);
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time);
gpu_timing_fake_queries_.ExpectGPUTimeStampQuery(*gl_, true);
gpu_timer->QueryTimeStamp();
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time - 1);
EXPECT_FALSE(gpu_timer->IsAvailable());
gpu_timing_fake_queries_.SetCurrentCPUTime(begin_cpu_time + 1);
EXPECT_TRUE(gpu_timer->IsAvailable());
EXPECT_EQ(0, gpu_timer->GetDeltaElapsed());
int64_t start, end;
gpu_timer->GetStartEndTimestamps(&start, &end);
EXPECT_EQ(begin_cpu_time, start);
EXPECT_EQ(begin_cpu_time, end);
}
} // namespace gpu