blob: 2f33475682035ba4b7ba9607316986f57eda7c3a [file] [log] [blame]
// Copyright (c) 2016 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.
#ifndef GL_GLEXT_PROTOTYPES
#define GL_GLEXT_PROTOTYPES
#endif
#include <GLES2/gl2.h>
#include <GLES2/gl2ext.h>
#include <GLES2/gl2extchromium.h>
#include <GLES3/gl3.h>
#include <stddef.h>
#include <stdint.h>
#include "base/stl_util.h"
#include "gpu/command_buffer/tests/gl_manager.h"
#include "gpu/command_buffer/tests/gl_test_utils.h"
#include "testing/gmock/include/gmock/gmock.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace gpu {
// A collection of tests that exercise the GL_CHROMIUM_copy_texture extension.
class GLApplyScreenSpaceAntialiasingCHROMIUMTest : public testing::Test {
protected:
void CreateAndBindDestinationTextureAndFBO(GLenum target) {
glGenTextures(1, &textures_);
glBindTexture(target, textures_);
// Some drivers (NVidia/SGX) require texture settings to be a certain way or
// they won't report FRAMEBUFFER_COMPLETE.
glTexParameterf(target, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameterf(target, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexParameteri(target, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(target, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glGenFramebuffers(1, &framebuffer_id_);
glBindFramebuffer(GL_FRAMEBUFFER, framebuffer_id_);
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, target,
textures_, 0);
}
void CheckStatus() {
available_ =
GLTestHelper::HasExtension("GL_CHROMIUM_screen_space_antialiasing");
if (!available_) {
LOG(INFO) << "GL_CHROMIUM_screen_space_antialiasing not supported. "
"Skipping test...";
return;
}
// Antialiasing will be enabled if framebuffer CMAA is enabled via GPU
// driver workarounds 'use_framebuffer_cmaa'.
available_ = gl_.workarounds().use_framebuffer_cmaa;
if (!available_) {
LOG(INFO) << "'use_framebuffer_cmaa' workaround not applied. "
"Skipping test...";
return;
}
CreateAndBindDestinationTextureAndFBO(GL_TEXTURE_2D);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE,
nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_FRAMEBUFFER_COMPLETE),
glCheckFramebufferStatus(GL_FRAMEBUFFER));
glClearColor(0, 1, 0, 1);
glClear(GL_COLOR_BUFFER_BIT);
glApplyScreenSpaceAntialiasingCHROMIUM();
if (glGetError() == GL_NO_ERROR)
return;
// For example, linux NVidia fails with this log.
// ApplyFramebufferAttachmentCMAAINTEL: shader compilation failed:
// GL_FRAGMENT_SHADER shader compilation failed: 0(98) : error C3013:
// input/output layout qualifiers supported above GL version 130
available_ = false;
LOG(ERROR) << "GL_CHROMIUM_screen_space_antialiasing maybe not supported "
"in non-Intel GPU.";
DeleteResources();
}
void SetUp() override {
GLManager::Options options;
gl_.Initialize(options);
CheckStatus();
}
void TearDown() override {
if (available_)
DeleteResources();
gl_.Destroy();
}
void DeleteResources() {
glDeleteTextures(1, &textures_);
glDeleteFramebuffers(1, &framebuffer_id_);
}
GLManager gl_;
GLuint textures_ = 0;
GLuint framebuffer_id_ = 0;
bool available_ = false;
};
class GLApplyScreenSpaceAntialiasingCHROMIUMES3Test
: public GLApplyScreenSpaceAntialiasingCHROMIUMTest {
protected:
void SetUp() override {
GLManager::Options options;
options.context_type = CONTEXT_TYPE_OPENGLES3;
gl_.Initialize(options);
CheckStatus();
}
void CheckStatus() {
// Not applicable for devices not supporting OpenGLES3.
if (!gl_.IsInitialized()) {
LOG(INFO) << "CONTEXT_TYPE_OPENGLES3 not supported. "
"Skipping test...";
return;
}
GLApplyScreenSpaceAntialiasingCHROMIUMTest::CheckStatus();
}
};
// TODO(dongseong.hwang): This test fails on the Nexus 9 GPU fyi bot.
// crbug.com/659638
#if defined(OS_ANDROID)
#define MAYBE_Basic DISABLED_Basic
#else
#define MAYBE_Basic Basic
#endif
// Test to ensure that the basic functionality of the extension works.
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, MAYBE_Basic) {
if (!available_)
return;
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
// Check the FB is still bound.
GLint value = 0;
glGetIntegerv(GL_FRAMEBUFFER_BINDING, &value);
GLuint fb_id = value;
EXPECT_EQ(framebuffer_id_, fb_id);
// Check that FB is complete.
EXPECT_EQ(static_cast<GLenum>(GL_FRAMEBUFFER_COMPLETE),
glCheckFramebufferStatus(GL_FRAMEBUFFER));
uint8_t pixels[1 * 4] = {0u, 255u, 0u, 255u};
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, pixels, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, DefaultFBO) {
if (!available_)
return;
glBindFramebuffer(GL_FRAMEBUFFER, 0);
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
}
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, InternalFormat) {
if (!available_)
return;
GLint formats[] = {GL_RGB, GL_RGBA, GL_ALPHA, GL_LUMINANCE,
GL_LUMINANCE_ALPHA};
for (size_t index = 0; index < base::size(formats); index++) {
glTexImage2D(GL_TEXTURE_2D, 0, formats[index], 1, 1, 0, formats[index],
GL_UNSIGNED_BYTE, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
textures_, 0);
// Only if the format is supported by FBO, supported by this extension.
if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
continue;
glClear(GL_COLOR_BUFFER_BIT);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError()) << "index:"
<< index;
uint8_t pixels[1 * 4] = {0u, 255u, 0u, 255u};
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, pixels, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
}
struct FormatType {
GLenum internal_format;
GLenum format;
GLenum type;
};
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMES3Test, InternalFormat) {
if (!available_)
return;
FormatType format_type = {GL_RGBA8, GL_RGBA, GL_UNSIGNED_BYTE};
uint8_t pixels[1 * 4] = {0u, 255u, 0u, 255u};
glTexImage2D(GL_TEXTURE_2D, 0, format_type.internal_format, 1, 1, 0,
format_type.format, format_type.type, pixels);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
textures_, 0);
// Only if the format is supported by FBO, supported by this extension.
if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
return;
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError())
<< "internal_format: "
<< gles2::GLES2Util::GetStringEnum(format_type.internal_format);
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, pixels, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMES3Test,
InternalFormatNotSupported) {
if (!available_)
return;
FormatType format_type = {GL_RGBA32UI, GL_RGBA_INTEGER, GL_UNSIGNED_INT};
uint32_t pixels[1 * 4] = {0u, 255u, 0u, 255u};
glTexImage2D(GL_TEXTURE_2D, 0, format_type.internal_format, 1, 1, 0,
format_type.format, format_type.type, pixels);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D,
textures_, 0);
// Only if the format is supported by FBO, supported by this extension.
if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
return;
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError())
<< "internal_format: "
<< gles2::GLES2Util::GetStringEnum(format_type.internal_format);
}
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, ImmutableTexture) {
if (!available_)
return;
if (!GLTestHelper::HasExtension("GL_EXT_texture_storage")) {
LOG(INFO) << "GL_EXT_texture_storage not supported. Skipping test...";
return;
}
GLenum internal_formats[] = {GL_RGB8_OES, GL_RGBA8_OES, GL_BGRA8_EXT};
for (auto internal_format : internal_formats) {
glDeleteTextures(1, &textures_);
glDeleteFramebuffers(1, &framebuffer_id_);
CreateAndBindDestinationTextureAndFBO(GL_TEXTURE_2D);
glTexStorage2DEXT(GL_TEXTURE_2D, 1, internal_format, 1, 1);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
EXPECT_EQ(static_cast<GLenum>(GL_FRAMEBUFFER_COMPLETE),
glCheckFramebufferStatus(GL_FRAMEBUFFER));
glClear(GL_COLOR_BUFFER_BIT);
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
uint8_t pixels[1 * 4] = {0u, 255u, 0u, 255u};
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, pixels, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
}
// Similar to webgl conformance test 'testAntialias(true)' in
// context-attributes-alpha-depth-stencil-antialias.html
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, AntiAliasing) {
if (!available_)
return;
// Fill colors in the FBO as follows
// +-----+
// |R|R| |
// +-----+
// |R| | |
// +-----+
// | | | |
// +-+-+-+
const int length = 3;
uint8_t rgba_pixels[4 * length * length] = {
255u, 0u, 0u, 255u, 255u, 0u, 0u, 255u, 0u, 0u, 0u, 0u,
255u, 0u, 0u, 255u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u,
0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u, 0u,
};
glBindTexture(GL_TEXTURE_2D, textures_);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, length, length, 0, GL_RGBA,
GL_UNSIGNED_BYTE, rgba_pixels);
uint8_t transparent[4] = {0u, 0u, 0u, 0u};
uint8_t red[4] = {255u, 0u, 0u, 255u};
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, red, nullptr);
GLTestHelper::CheckPixels(0, 1, 1, 1, 0, red, nullptr);
GLTestHelper::CheckPixels(0, 2, 1, 1, 0, transparent, nullptr);
GLTestHelper::CheckPixels(1, 0, 1, 1, 0, red, nullptr);
GLTestHelper::CheckPixels(1, 1, 1, 1, 0, transparent, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
GLTestHelper::CheckPixels(0, 0, 1, 1, 0, red, nullptr);
GLTestHelper::CheckPixels(2, 2, 1, 1, 0, transparent, nullptr);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
// Check if middle pixel is anti-aliased.
uint8_t pixels[4] = {0u};
glReadPixels(1, 1, 1, 1, GL_RGBA, GL_UNSIGNED_BYTE, &pixels);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
EXPECT_NE(transparent, pixels);
EXPECT_NE(red, pixels);
EXPECT_TRUE(pixels[0] > transparent[0] && pixels[0] < red[0]);
EXPECT_EQ(0u, pixels[1]);
EXPECT_EQ(0u, pixels[2]);
EXPECT_TRUE(pixels[3] > transparent[3] && pixels[3] < red[3]);
}
namespace {
void glEnableDisable(GLint param, GLboolean value) {
if (value)
glEnable(param);
else
glDisable(param);
}
} // unnamed namespace
// Validate that some basic GL state is not touched upon execution of
// the extension.
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, BasicStatePreservation) {
if (!available_)
return;
GLboolean reference_settings[2] = {GL_TRUE, GL_FALSE};
for (int x = 0; x < 2; ++x) {
GLboolean setting = reference_settings[x];
glEnableDisable(GL_DEPTH_TEST, setting);
glEnableDisable(GL_SCISSOR_TEST, setting);
glEnableDisable(GL_STENCIL_TEST, setting);
glEnableDisable(GL_CULL_FACE, setting);
glEnableDisable(GL_BLEND, setting);
glColorMask(setting, setting, setting, setting);
glDepthMask(setting);
glActiveTexture(GL_TEXTURE1 + x);
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
EXPECT_EQ(setting, glIsEnabled(GL_DEPTH_TEST));
EXPECT_EQ(setting, glIsEnabled(GL_SCISSOR_TEST));
EXPECT_EQ(setting, glIsEnabled(GL_STENCIL_TEST));
EXPECT_EQ(setting, glIsEnabled(GL_CULL_FACE));
EXPECT_EQ(setting, glIsEnabled(GL_BLEND));
GLboolean bool_array[4] = {GL_FALSE, GL_FALSE, GL_FALSE, GL_FALSE};
glGetBooleanv(GL_DEPTH_WRITEMASK, bool_array);
EXPECT_EQ(setting, bool_array[0]);
bool_array[0] = GL_FALSE;
glGetBooleanv(GL_COLOR_WRITEMASK, bool_array);
EXPECT_EQ(setting, bool_array[0]);
EXPECT_EQ(setting, bool_array[1]);
EXPECT_EQ(setting, bool_array[2]);
EXPECT_EQ(setting, bool_array[3]);
GLint active_texture = 0;
glGetIntegerv(GL_ACTIVE_TEXTURE, &active_texture);
EXPECT_EQ(GL_TEXTURE1 + x, active_texture);
}
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
// Verify that invocation of the extension does not modify the bound
// texture state.
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, TextureStatePreserved) {
if (!available_)
return;
GLuint texture_ids[2];
glGenTextures(2, texture_ids);
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D, texture_ids[0]);
glActiveTexture(GL_TEXTURE1);
glBindTexture(GL_TEXTURE_2D, texture_ids[1]);
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
GLint active_texture = 0;
glGetIntegerv(GL_ACTIVE_TEXTURE, &active_texture);
EXPECT_EQ(GL_TEXTURE1, active_texture);
GLint bound_texture = 0;
glGetIntegerv(GL_TEXTURE_BINDING_2D, &bound_texture);
EXPECT_EQ(texture_ids[1], static_cast<GLuint>(bound_texture));
glBindTexture(GL_TEXTURE_2D, 0);
bound_texture = 0;
glActiveTexture(GL_TEXTURE0);
glGetIntegerv(GL_TEXTURE_BINDING_2D, &bound_texture);
EXPECT_EQ(texture_ids[0], static_cast<GLuint>(bound_texture));
glBindTexture(GL_TEXTURE_2D, 0);
glDeleteTextures(2, texture_ids);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
}
TEST_F(GLApplyScreenSpaceAntialiasingCHROMIUMTest, ProgramStatePreservation) {
if (!available_)
return;
// unbind the one created in setup.
glBindFramebuffer(GL_FRAMEBUFFER, 0);
glBindTexture(GL_TEXTURE_2D, 0);
GLManager gl2;
GLManager::Options options;
options.size = gfx::Size(16, 16);
options.share_group_manager = &gl_;
gl2.Initialize(options);
gl_.MakeCurrent();
static const char* v_shader_str =
"attribute vec4 g_Position;\n"
"void main()\n"
"{\n"
" gl_Position = g_Position;\n"
"}\n";
static const char* f_shader_str =
"precision mediump float;\n"
"void main()\n"
"{\n"
" gl_FragColor = vec4(0,1,0,1);\n"
"}\n";
GLuint program = GLTestHelper::LoadProgram(v_shader_str, f_shader_str);
glUseProgram(program);
GLuint position_loc = glGetAttribLocation(program, "g_Position");
glFlush();
// Delete program from other context.
gl2.MakeCurrent();
glDeleteProgram(program);
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
glFlush();
// Program should still be usable on this context.
gl_.MakeCurrent();
GLTestHelper::SetupUnitQuad(position_loc);
// test using program before
uint8_t expected[] = {
0, 255, 0, 255,
};
uint8_t zero[] = {
0, 0, 0, 0,
};
glClearColor(0, 0, 0, 0);
glClear(GL_COLOR_BUFFER_BIT);
EXPECT_TRUE(GLTestHelper::CheckPixels(0, 0, 1, 1, 0, zero, nullptr));
glDrawArrays(GL_TRIANGLES, 0, 6);
EXPECT_TRUE(GLTestHelper::CheckPixels(0, 0, 1, 1, 0, expected, nullptr));
// Call copyTextureCHROMIUM
uint8_t pixels[1 * 4] = {255u, 0u, 0u, 255u};
glBindTexture(GL_TEXTURE_2D, textures_);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, 1, 1, 0, GL_RGBA, GL_UNSIGNED_BYTE,
pixels);
glBindFramebuffer(GL_FRAMEBUFFER, framebuffer_id_);
glApplyScreenSpaceAntialiasingCHROMIUM();
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
// test using program after
glClear(GL_COLOR_BUFFER_BIT);
EXPECT_TRUE(GLTestHelper::CheckPixels(0, 0, 1, 1, 0, zero, nullptr));
glDrawArrays(GL_TRIANGLES, 0, 6);
EXPECT_TRUE(GLTestHelper::CheckPixels(0, 0, 1, 1, 0, expected, nullptr));
EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
gl2.MakeCurrent();
gl2.Destroy();
gl_.MakeCurrent();
}
} // namespace gpu