| Index: gpu/command_buffer/tests/gl_chromium_path_rendering_unittest.cc
|
| diff --git a/gpu/command_buffer/tests/gl_chromium_path_rendering_unittest.cc b/gpu/command_buffer/tests/gl_chromium_path_rendering_unittest.cc
|
| index 0d17b7c6f3bd3575a36ed59411ea580530415e10..57f5dd02ea7993eeb64602c590762d63426d4a23 100644
|
| --- a/gpu/command_buffer/tests/gl_chromium_path_rendering_unittest.cc
|
| +++ b/gpu/command_buffer/tests/gl_chromium_path_rendering_unittest.cc
|
| @@ -7,11 +7,15 @@
|
| #include <GLES2/gl2extchromium.h>
|
| #include <cmath>
|
|
|
| +#include "base/command_line.h"
|
| +#include "gpu/command_buffer/service/gpu_switches.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"
|
|
|
| +#define SHADER(Src) #Src
|
| +
|
| namespace gpu {
|
|
|
| class CHROMIUMPathRenderingTest : public testing::Test {
|
| @@ -22,7 +26,9 @@ class CHROMIUMPathRenderingTest : public testing::Test {
|
| void SetUp() override {
|
| GLManager::Options options;
|
| options.size = gfx::Size(kResolution, kResolution);
|
| - gl_.Initialize(options);
|
| + base::CommandLine command_line(*base::CommandLine::ForCurrentProcess());
|
| + command_line.AppendSwitch(switches::kEnableGLPathRendering);
|
| + gl_.InitializeWithCommandLine(options, &command_line);
|
| }
|
|
|
| void TearDown() override { gl_.Destroy(); }
|
| @@ -37,13 +43,156 @@ class CHROMIUMPathRenderingTest : public testing::Test {
|
| EXPECT_EQ(static_cast<GLint>(round(expected[i])), actual[i]);
|
| }
|
| }
|
| +
|
| + void SetupStateForTestPattern() {
|
| + glViewport(0, 0, kResolution, kResolution);
|
| + glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
|
| + glStencilMask(0xffffffff);
|
| + glClearStencil(0);
|
| + glClear(GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
|
| +
|
| + static const char* kVertexShaderSource =
|
| + SHADER(void main() { gl_Position = vec4(1); });
|
| + static const char* kFragmentShaderSource =
|
| + SHADER(precision mediump float; uniform vec4 color;
|
| + void main() { gl_FragColor = color; });
|
| +
|
| + GLuint program =
|
| + GLTestHelper::LoadProgram(kVertexShaderSource, kFragmentShaderSource);
|
| + glUseProgram(program);
|
| + color_loc_ = glGetUniformLocation(program, "color");
|
| + glDeleteProgram(program);
|
| +
|
| + // Set up orthogonal projection with near/far plane distance of 2.
|
| + static GLfloat matrix[16] = {2.0f / (kResolution - 1),
|
| + 0.0f,
|
| + 0.0f,
|
| + 0.0f,
|
| + 0.0f,
|
| + 2.0f / (kResolution - 1),
|
| + 0.0f,
|
| + 0.0f,
|
| + 0.0f,
|
| + 0.0f,
|
| + -1.0f,
|
| + 0.0f,
|
| + -1.0f,
|
| + -1.0f,
|
| + 0.0f,
|
| + 1.0f};
|
| + glMatrixLoadfCHROMIUM(GL_PATH_PROJECTION_CHROMIUM, matrix);
|
| + glMatrixLoadIdentityCHROMIUM(GL_PATH_MODELVIEW_CHROMIUM);
|
| +
|
| + glEnable(GL_STENCIL_TEST);
|
| +
|
| + GLTestHelper::CheckGLError("no errors at state setup", __LINE__);
|
| + }
|
| +
|
| + void SetupPathStateForTestPattern(GLuint path) {
|
| + static const GLubyte kCommands[] = {GL_MOVE_TO_CHROMIUM,
|
| + GL_LINE_TO_CHROMIUM,
|
| + GL_QUADRATIC_CURVE_TO_CHROMIUM,
|
| + GL_CUBIC_CURVE_TO_CHROMIUM,
|
| + GL_CLOSE_PATH_CHROMIUM};
|
| +
|
| + static const GLfloat kCoords[] = {50.0f,
|
| + 50.0f,
|
| + 75.0f,
|
| + 75.0f,
|
| + 100.0f,
|
| + 62.5f,
|
| + 50.0f,
|
| + 25.5f,
|
| + 0.0f,
|
| + 62.5f,
|
| + 50.0f,
|
| + 50.0f,
|
| + 25.0f,
|
| + 75.0f};
|
| +
|
| + glPathCommandsCHROMIUM(path, arraysize(kCommands), kCommands,
|
| + arraysize(kCoords), GL_FLOAT, kCoords);
|
| +
|
| + glPathParameterfCHROMIUM(path, GL_PATH_STROKE_WIDTH_CHROMIUM, 5.0f);
|
| + glPathParameterfCHROMIUM(path, GL_PATH_MITER_LIMIT_CHROMIUM, 1.0f);
|
| + glPathParameterfCHROMIUM(path, GL_PATH_STROKE_BOUND_CHROMIUM, .02f);
|
| + glPathParameteriCHROMIUM(path, GL_PATH_JOIN_STYLE_CHROMIUM,
|
| + GL_ROUND_CHROMIUM);
|
| + glPathParameteriCHROMIUM(path, GL_PATH_END_CAPS_CHROMIUM,
|
| + GL_SQUARE_CHROMIUM);
|
| + }
|
| +
|
| + void VerifyTestPatternFill(float x, float y) {
|
| + static const float kFillCoords[] = {
|
| + 55.0f, 55.0f, 50.0f, 28.0f, 66.0f, 63.0f};
|
| + static const uint8 kBlue[] = {0, 0, 255, 255};
|
| +
|
| + for (size_t i = 0; i < arraysize(kFillCoords); i += 2) {
|
| + float fx = kFillCoords[i];
|
| + float fy = kFillCoords[i + 1];
|
| +
|
| + EXPECT_TRUE(GLTestHelper::CheckPixels(x + fx, y + fy, 1, 1, 0, kBlue));
|
| + }
|
| + }
|
| +
|
| + void VerifyTestPatternBg(float x, float y) {
|
| + const float kBackgroundCoords[] = {80.0f, 80.0f, 20.0f, 20.0f, 90.0f, 1.0f};
|
| + const uint8 kExpectedColor[] = {0, 0, 0, 0};
|
| +
|
| + for (size_t i = 0; i < arraysize(kBackgroundCoords); i += 2) {
|
| + float bx = kBackgroundCoords[i];
|
| + float by = kBackgroundCoords[i + 1];
|
| +
|
| + EXPECT_TRUE(
|
| + GLTestHelper::CheckPixels(x + bx, y + by, 1, 1, 0, kExpectedColor));
|
| + }
|
| + }
|
| +
|
| + void VerifyTestPatternStroke(float x, float y) {
|
| + // Inside the stroke we should have green.
|
| + const uint8 kGreen[] = {0, 255, 0, 255};
|
| + EXPECT_TRUE(GLTestHelper::CheckPixels(x + 50, y + 53, 1, 1, 0, kGreen));
|
| + EXPECT_TRUE(GLTestHelper::CheckPixels(x + 26, y + 76, 1, 1, 0, kGreen));
|
| +
|
| + // Outside the path we should have black.
|
| + const uint8 black[] = {0, 0, 0, 0};
|
| + EXPECT_TRUE(GLTestHelper::CheckPixels(x + 10, y + 10, 1, 1, 0, black));
|
| + EXPECT_TRUE(GLTestHelper::CheckPixels(x + 80, y + 80, 1, 1, 0, black));
|
| + }
|
| +
|
| + void TryAllDrawFunctions(GLuint path, GLenum expected_error) {
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glStencilStrokePathCHROMIUM(path, 0x80, 0x80);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glCoverFillPathCHROMIUM(path, GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glCoverStrokePathCHROMIUM(path, GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glStencilThenCoverStrokePathCHROMIUM(path, 0x80, 0x80,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| +
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(expected_error, glGetError());
|
| + }
|
| +
|
| GLManager gl_;
|
| + GLint color_loc_;
|
| };
|
|
|
| TEST_F(CHROMIUMPathRenderingTest, TestMatrix) {
|
| - if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering")) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| return;
|
| - }
|
| +
|
| static const GLfloat kIdentityMatrix[16] = {
|
| 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,
|
| 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, 0.0f, 0.0f, 1.0f};
|
| @@ -86,25 +235,388 @@ TEST_F(CHROMIUMPathRenderingTest, TestMatrix) {
|
| }
|
|
|
| TEST_F(CHROMIUMPathRenderingTest, TestMatrixErrors) {
|
| - if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering")) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| return;
|
| - }
|
| +
|
| GLfloat mf[16];
|
| memset(mf, 0, sizeof(mf));
|
|
|
| - // This should fail.
|
| + glMatrixLoadfCHROMIUM(GL_PATH_MODELVIEW_CHROMIUM, mf);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + glMatrixLoadIdentityCHROMIUM(GL_PATH_PROJECTION_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + // Test that invalid matrix targets fail.
|
| glMatrixLoadfCHROMIUM(GL_PATH_MODELVIEW_CHROMIUM - 1, mf);
|
| EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
|
|
| - glMatrixLoadfCHROMIUM(GL_PATH_MODELVIEW_CHROMIUM, mf);
|
| + // Test that invalid matrix targets fail.
|
| + glMatrixLoadIdentityCHROMIUM(GL_PATH_PROJECTION_CHROMIUM + 1);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestSimpleCalls) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + // This is unspecified in NV_path_rendering.
|
| + EXPECT_EQ(0u, glGenPathsCHROMIUM(0));
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + EXPECT_NE(path, 0u);
|
| + glDeletePathsCHROMIUM(path, 1);
|
| EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
|
|
| - // This should fail.
|
| - glMatrixLoadIdentityCHROMIUM(GL_PATH_PROJECTION_CHROMIUM + 1);
|
| + GLuint first_path = glGenPathsCHROMIUM(5);
|
| + EXPECT_NE(first_path, 0u);
|
| + glDeletePathsCHROMIUM(first_path, 5);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + // Test deleting paths that are not actually allocated:
|
| + // "unused names in /paths/ are silently ignored".
|
| + first_path = glGenPathsCHROMIUM(5);
|
| + EXPECT_NE(first_path, 0u);
|
| + glDeletePathsCHROMIUM(first_path, 6);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + GLsizei big_range = 0xffff;
|
| + // Setting big_range = std::numeric_limits<GLsizei>::max() should go through
|
| + // too, as far as NV_path_rendering is concerned. Current chromium side id
|
| + // allocator will use too much memory.
|
| + first_path = glGenPathsCHROMIUM(big_range);
|
| + EXPECT_NE(first_path, 0u);
|
| + glDeletePathsCHROMIUM(first_path, big_range);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + // Test glIsPathCHROMIUM().
|
| + path = glGenPathsCHROMIUM(1);
|
| + EXPECT_FALSE(glIsPathCHROMIUM(path));
|
| + GLubyte commands[] = {GL_MOVE_TO_CHROMIUM, GL_CLOSE_PATH_CHROMIUM};
|
| + GLfloat coords[] = {50.0f, 50.0f};
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands, arraysize(coords),
|
| + GL_FLOAT, coords);
|
| + EXPECT_TRUE(glIsPathCHROMIUM(path));
|
| + glDeletePathsCHROMIUM(path, 1);
|
| + EXPECT_FALSE(glIsPathCHROMIUM(path));
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestGenDeleteErrors) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + // GenPaths / DeletePaths tests.
|
| + // std::numeric_limits<GLuint>::max() is wrong for GLsizei.
|
| + GLuint first_path = glGenPathsCHROMIUM(std::numeric_limits<GLuint>::max());
|
| + EXPECT_EQ(first_path, 0u);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + first_path = glGenPathsCHROMIUM(-1);
|
| + EXPECT_EQ(first_path, 0u);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + glDeletePathsCHROMIUM(1, -5);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + first_path = glGenPathsCHROMIUM(-1);
|
| + EXPECT_EQ(first_path, 0u);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + // Test that delete with first_id and range such that first_id + range
|
| + // overflows the GLuint. Example:
|
| + // Range is 0x7fffffff. First id is X. Last id will be X + 0x7ffffffe.
|
| + // X = 0x80000001 would succeed, where as X = 0x80000002 would fail.
|
| + // To get 0x80000002, we need to allocate first 0x7fffffff and then
|
| + // 3 (0x80000000, 0x80000001 and 0x80000002).
|
| + // While not guaranteed by the API, we expect the implementation
|
| + // hands us deterministic ids.
|
| + first_path = glGenPathsCHROMIUM(std::numeric_limits<GLsizei>::max());
|
| + EXPECT_EQ(first_path, 1u);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + GLuint additional_paths = glGenPathsCHROMIUM(3);
|
| + EXPECT_EQ(additional_paths,
|
| + static_cast<GLuint>(std::numeric_limits<GLsizei>::max()) + 1u);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + // Test that passing a range so big that it would overflow client_id
|
| + // + range - 1 check causes an error.
|
| + glDeletePathsCHROMIUM(additional_paths + 2u,
|
| + std::numeric_limits<GLsizei>::max());
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
|
| +
|
| + // Cleanup the above allocations. Also test that passing max value still
|
| + // works.
|
| + glDeletePathsCHROMIUM(1, std::numeric_limits<GLsizei>::max());
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + glDeletePathsCHROMIUM(std::numeric_limits<GLsizei>::max(),
|
| + std::numeric_limits<GLsizei>::max());
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathParameterErrors) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + // PathParameter*: Wrong value for the pname should fail.
|
| + glPathParameteriCHROMIUM(path, GL_PATH_JOIN_STYLE_CHROMIUM, GL_FLAT_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| + glPathParameterfCHROMIUM(path, GL_PATH_END_CAPS_CHROMIUM,
|
| + GL_MITER_REVERT_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + // PathParameter*: Wrong floating-point value should fail.
|
| + glPathParameterfCHROMIUM(path, GL_PATH_STROKE_WIDTH_CHROMIUM, -0.1f);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| + glPathParameterfCHROMIUM(path, GL_PATH_MITER_LIMIT_CHROMIUM,
|
| + std::numeric_limits<float>::quiet_NaN());
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| + glPathParameterfCHROMIUM(path, GL_PATH_MITER_LIMIT_CHROMIUM,
|
| + std::numeric_limits<float>::infinity());
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + // PathParameter*: Wrong pname should fail.
|
| + glPathParameteriCHROMIUM(path, GL_PATH_STROKE_WIDTH_CHROMIUM - 1, 5);
|
| EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +}
|
|
|
| - glMatrixLoadIdentityCHROMIUM(GL_PATH_PROJECTION_CHROMIUM);
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathObjectState) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + glViewport(0, 0, kResolution, kResolution);
|
| + glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
|
| + glStencilMask(0xffffffff);
|
| + glClearStencil(0);
|
| + glClear(GL_COLOR_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
|
| + glPathStencilFuncCHROMIUM(GL_ALWAYS, 0, 0xFF);
|
| + glStencilOp(GL_KEEP, GL_KEEP, GL_ZERO);
|
| +
|
| + // Test that trying to draw non-existing paths does not produce errors or
|
| + // results.
|
| + GLuint non_existing_paths[] = {0, 55, 74744};
|
| + for (auto& p : non_existing_paths) {
|
| + EXPECT_FALSE(glIsPathCHROMIUM(p));
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + TryAllDrawFunctions(p, GL_NO_ERROR);
|
| + }
|
| +
|
| + // Path name marked as used but without path object state causes
|
| + // a GL error upon any draw command.
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + EXPECT_FALSE(glIsPathCHROMIUM(path));
|
| + TryAllDrawFunctions(path, GL_INVALID_OPERATION);
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +
|
| + // Document a bit of an inconsistency: path name marked as used but without
|
| + // path object state causes a GL error upon any draw command (tested above).
|
| + // Path name that had path object state, but then was "cleared", still has a
|
| + // path object state, even though the state is empty.
|
| + path = glGenPathsCHROMIUM(1);
|
| + EXPECT_FALSE(glIsPathCHROMIUM(path));
|
| + GLubyte commands[] = {GL_MOVE_TO_CHROMIUM, GL_CLOSE_PATH_CHROMIUM};
|
| + GLfloat coords[] = {50.0f, 50.0f};
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands, arraysize(coords),
|
| + GL_FLOAT, coords);
|
| + EXPECT_TRUE(glIsPathCHROMIUM(path));
|
| + glPathCommandsCHROMIUM(path, 0, NULL, 0, GL_FLOAT, NULL);
|
| + EXPECT_TRUE(glIsPathCHROMIUM(path)); // The surprise.
|
| + TryAllDrawFunctions(path, GL_NO_ERROR);
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +
|
| + // Document a bit of an inconsistency: "clearing" a used path name causes
|
| + // path to acquire state.
|
| + path = glGenPathsCHROMIUM(1);
|
| + EXPECT_FALSE(glIsPathCHROMIUM(path));
|
| + glPathCommandsCHROMIUM(path, 0, NULL, 0, GL_FLOAT, NULL);
|
| + EXPECT_TRUE(glIsPathCHROMIUM(path)); // The surprise.
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +
|
| + // Make sure nothing got drawn by the drawing commands that should not produce
|
| + // anything.
|
| + const uint8 black[] = {0, 0, 0, 0};
|
| + EXPECT_TRUE(
|
| + GLTestHelper::CheckPixels(0, 0, kResolution, kResolution, 0, black));
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestUnnamedPathsErrors) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + // Unnamed paths: Trying to create a path object with non-existing path name
|
| + // produces error. (Not a error in real NV_path_rendering).
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + GLubyte commands[] = {GL_MOVE_TO_CHROMIUM, GL_CLOSE_PATH_CHROMIUM};
|
| + GLfloat coords[] = {50.0f, 50.0f};
|
| + glPathCommandsCHROMIUM(555, arraysize(commands), commands, arraysize(coords),
|
| + GL_FLOAT, coords);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
|
| +
|
| + // PathParameter*: Using non-existing path object produces error.
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + glPathParameterfCHROMIUM(555, GL_PATH_STROKE_WIDTH_CHROMIUM, 5.0f);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
|
| +
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + glPathParameteriCHROMIUM(555, GL_PATH_JOIN_STYLE_CHROMIUM, GL_ROUND_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathCommandsErrors) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + static const GLenum kInvalidCoordType = GL_NONE;
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + GLubyte commands[] = {GL_MOVE_TO_CHROMIUM, GL_CLOSE_PATH_CHROMIUM};
|
| + GLfloat coords[] = {50.0f, 50.0f};
|
| +
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands, -4, GL_FLOAT,
|
| + coords);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + glPathCommandsCHROMIUM(path, -1, commands, arraysize(coords), GL_FLOAT,
|
| + coords);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands, arraysize(coords),
|
| + kInvalidCoordType, coords);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| +
|
| + // These can not distinquish between the check that should fail them.
|
| + // This should fail due to coord count * float size overflow.
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands,
|
| + std::numeric_limits<GLsizei>::max(), GL_FLOAT, coords);
|
| + // This should fail due to cmd count + coord count * short size.
|
| + glPathCommandsCHROMIUM(path, arraysize(commands), commands,
|
| + std::numeric_limits<GLsizei>::max(), GL_SHORT, coords);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_OPERATION), glGetError());
|
| +
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +}
|
| +
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathRenderingInvalidArgs) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + glPathCommandsCHROMIUM(path, 0, NULL, 0, GL_FLOAT, NULL);
|
| +
|
| + // Verify that normal calls work.
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F);
|
| EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_NO_ERROR), glGetError());
|
| +
|
| + // Using invalid fill mode causes INVALID_ENUM.
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM - 1, 0x7F);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM - 1, 0x7F,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| +
|
| + // Using invalid cover mode causes INVALID_ENUM.
|
| + glCoverFillPathCHROMIUM(path, GL_CONVEX_HULL_CHROMIUM - 1);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F,
|
| + GL_BOUNDING_BOX_CHROMIUM + 1);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_ENUM), glGetError());
|
| +
|
| + // Using mask+1 not being power of two causes INVALID_VALUE with up/down fill
|
| + // mode.
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x40);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_DOWN_CHROMIUM, 12,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| + EXPECT_EQ(static_cast<GLenum>(GL_INVALID_VALUE), glGetError());
|
| +
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +}
|
| +
|
| +// Tests that drawing with CHROMIUM_path_rendering functions work.
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathRendering) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + static const float kBlue[] = {0.0f, 0.0f, 1.0f, 1.0f};
|
| + static const float kGreen[] = {0.0f, 1.0f, 0.0f, 1.0f};
|
| +
|
| + SetupStateForTestPattern();
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + SetupPathStateForTestPattern(path);
|
| +
|
| + // Do the stencil fill, cover fill, stencil stroke, cover stroke
|
| + // in unconventional order:
|
| + // 1) stencil the stroke in stencil high bit
|
| + // 2) stencil the fill in low bits
|
| + // 3) cover the fill
|
| + // 4) cover the stroke
|
| + // This is done to check that glPathStencilFunc works, eg the mask
|
| + // goes through. Stencil func is not tested ATM, for simplicity.
|
| +
|
| + glPathStencilFuncCHROMIUM(GL_ALWAYS, 0, 0xFF);
|
| + glStencilStrokePathCHROMIUM(path, 0x80, 0x80);
|
| +
|
| + glPathStencilFuncCHROMIUM(GL_ALWAYS, 0, 0x7F);
|
| + glStencilFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F);
|
| +
|
| + glStencilFunc(GL_LESS, 0, 0x7F);
|
| + glStencilOp(GL_KEEP, GL_KEEP, GL_ZERO);
|
| + glUniform4fv(color_loc_, 1, kBlue);
|
| + glCoverFillPathCHROMIUM(path, GL_BOUNDING_BOX_CHROMIUM);
|
| +
|
| + glStencilFunc(GL_EQUAL, 0x80, 0x80);
|
| + glStencilOp(GL_KEEP, GL_KEEP, GL_ZERO);
|
| + glUniform4fv(color_loc_, 1, kGreen);
|
| + glCoverStrokePathCHROMIUM(path, GL_CONVEX_HULL_CHROMIUM);
|
| +
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +
|
| + // Verify the image.
|
| + VerifyTestPatternFill(0.0f, 0.0f);
|
| + VerifyTestPatternBg(0.0f, 0.0f);
|
| + VerifyTestPatternStroke(0.0f, 0.0f);
|
| +}
|
| +
|
| +// Tests that drawing with CHROMIUM_path_rendering
|
| +// StencilThenCover{Stroke,Fill}Path functions work.
|
| +TEST_F(CHROMIUMPathRenderingTest, TestPathRenderingThenFunctions) {
|
| + if (!GLTestHelper::HasExtension("GL_CHROMIUM_path_rendering"))
|
| + return;
|
| +
|
| + static float kBlue[] = {0.0f, 0.0f, 1.0f, 1.0f};
|
| + static float kGreen[] = {0.0f, 1.0f, 0.0f, 1.0f};
|
| +
|
| + SetupStateForTestPattern();
|
| +
|
| + GLuint path = glGenPathsCHROMIUM(1);
|
| + SetupPathStateForTestPattern(path);
|
| +
|
| + glPathStencilFuncCHROMIUM(GL_ALWAYS, 0, 0xFF);
|
| + glStencilFunc(GL_EQUAL, 0x80, 0x80);
|
| + glStencilOp(GL_KEEP, GL_KEEP, GL_ZERO);
|
| + glUniform4fv(color_loc_, 1, kGreen);
|
| + glStencilThenCoverStrokePathCHROMIUM(path, 0x80, 0x80,
|
| + GL_BOUNDING_BOX_CHROMIUM);
|
| +
|
| + glPathStencilFuncCHROMIUM(GL_ALWAYS, 0, 0x7F);
|
| + glStencilFunc(GL_LESS, 0, 0x7F);
|
| + glStencilOp(GL_KEEP, GL_KEEP, GL_ZERO);
|
| + glUniform4fv(color_loc_, 1, kBlue);
|
| + glStencilThenCoverFillPathCHROMIUM(path, GL_COUNT_UP_CHROMIUM, 0x7F,
|
| + GL_CONVEX_HULL_CHROMIUM);
|
| +
|
| + glDeletePathsCHROMIUM(path, 1);
|
| +
|
| + // Verify the image.
|
| + VerifyTestPatternFill(0.0f, 0.0f);
|
| + VerifyTestPatternBg(0.0f, 0.0f);
|
| + VerifyTestPatternStroke(0.0f, 0.0f);
|
| }
|
|
|
| } // namespace gpu
|
|
|