blob: fcda4b6a2c613a9d9aa4960c07e2c8f2b54bb4c1 [file] [log] [blame]
/*-------------------------------------------------------------------------
* drawElements Quality Program OpenGL ES 3.0 Module
* -------------------------------------------------
*
* Copyright 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*//*!
* \file
* \brief Negative Fragment Pipe API tests.
*//*--------------------------------------------------------------------*/
#include "es3fNegativeFragmentApiTests.hpp"
#include "es3fApiCase.hpp"
#include "gluContextInfo.hpp"
#include "glwDefs.hpp"
#include "glwEnums.hpp"
using namespace glw; // GL types
namespace deqp
{
namespace gles3
{
namespace Functional
{
using tcu::TestLog;
static bool checkDrawBuffersIndexedSupport(Context &ctx)
{
return contextSupports(ctx.getRenderContext().getType(), glu::ApiType::es(3, 2)) ||
contextSupports(ctx.getRenderContext().getType(), glu::ApiType::core(4, 5)) ||
ctx.getContextInfo().isExtensionSupported("GL_EXT_draw_buffers_indexed");
}
NegativeFragmentApiTests::NegativeFragmentApiTests(Context &context)
: TestCaseGroup(context, "fragment", "Negative Fragment API Cases")
{
}
NegativeFragmentApiTests::~NegativeFragmentApiTests(void)
{
}
void NegativeFragmentApiTests::init(void)
{
ES3F_ADD_API_CASE(scissor, "Invalid glScissor() usage", {
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if either width or height is negative.");
glScissor(0, 0, -1, 0);
expectError(GL_INVALID_VALUE);
glScissor(0, 0, 0, -1);
expectError(GL_INVALID_VALUE);
glScissor(0, 0, -1, -1);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(depth_func, "Invalid glDepthFunc() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if func is not an accepted value.");
glDepthFunc(-1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(viewport, "Invalid glViewport() usage", {
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if either width or height is negative.");
glViewport(0, 0, -1, 1);
expectError(GL_INVALID_VALUE);
glViewport(0, 0, 1, -1);
expectError(GL_INVALID_VALUE);
glViewport(0, 0, -1, -1);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
// Stencil functions
ES3F_ADD_API_CASE(stencil_func, "Invalid glStencilFunc() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if func is not one of the eight accepted values.");
glStencilFunc(-1, 0, 1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(stencil_func_separate, "Invalid glStencilFuncSeparate() usage", {
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if face is not GL_FRONT, GL_BACK, or GL_FRONT_AND_BACK.");
glStencilFuncSeparate(-1, GL_NEVER, 0, 1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if func is not one of the eight accepted values.");
glStencilFuncSeparate(GL_FRONT, -1, 0, 1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(stencil_op, "Invalid glStencilOp() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if sfail, dpfail, or dppass is any value other "
"than the defined symbolic constant values.");
glStencilOp(-1, GL_ZERO, GL_REPLACE);
expectError(GL_INVALID_ENUM);
glStencilOp(GL_KEEP, -1, GL_REPLACE);
expectError(GL_INVALID_ENUM);
glStencilOp(GL_KEEP, GL_ZERO, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(stencil_op_separate, "Invalid glStencilOpSeparate() usage", {
m_log << TestLog::Section(
"",
"GL_INVALID_ENUM is generated if face is any value other than GL_FRONT, GL_BACK, or GL_FRONT_AND_BACK.");
glStencilOpSeparate(-1, GL_KEEP, GL_ZERO, GL_REPLACE);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if sfail, dpfail, or dppass is any value other "
"than the eight defined symbolic constant values.");
glStencilOpSeparate(GL_FRONT, -1, GL_ZERO, GL_REPLACE);
expectError(GL_INVALID_ENUM);
glStencilOpSeparate(GL_FRONT, GL_KEEP, -1, GL_REPLACE);
expectError(GL_INVALID_ENUM);
glStencilOpSeparate(GL_FRONT, GL_KEEP, GL_ZERO, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(stencil_mask_separate, "Invalid glStencilMaskSeparate() usage", {
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if face is not GL_FRONT, GL_BACK, or GL_FRONT_AND_BACK.");
glStencilMaskSeparate(-1, 0);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
// Blend functions
ES3F_ADD_API_CASE(blend_equation, "Invalid glBlendEquation() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if mode is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquation(-1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_equation_separate, "Invalid glBlendEquationSeparate() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if modeRGB is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquationSeparate(-1, GL_FUNC_ADD);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if modeAlpha is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquationSeparate(GL_FUNC_ADD, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_func, "Invalid glBlendFunc() usage", {
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if either sfactor or dfactor is not an accepted value.");
glBlendFunc(-1, GL_ONE);
expectError(GL_INVALID_ENUM);
glBlendFunc(GL_ONE, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_func_separate, "Invalid glBlendFuncSeparate() usage", {
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if srcRGB, dstRGB, srcAlpha, or dstAlpha is not an accepted value.");
glBlendFuncSeparate(-1, GL_ONE, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparate(GL_ZERO, -1, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparate(GL_ZERO, GL_ONE, -1, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparate(GL_ZERO, GL_ONE, GL_SRC_COLOR, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_equationi, "Invalid glBlendEquationi() usage", {
glw::GLint maxDrawBuffers = -1;
glGetIntegerv(GL_MAX_DRAW_BUFFERS, &maxDrawBuffers);
if (!checkDrawBuffersIndexedSupport(m_context))
throw tcu::NotSupportedError("GL_EXT_draw_buffers_indexed is not supported", DE_NULL, __FILE__, __LINE__);
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if mode is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquationi(0, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if buf is not in the range zero to the value of "
"MAX_DRAW_BUFFERS minus one.");
glBlendEquationi(-1, GL_FUNC_ADD);
expectError(GL_INVALID_VALUE);
glBlendEquationi(maxDrawBuffers, GL_FUNC_ADD);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_equation_separatei, "Invalid glBlendEquationSeparatei() usage", {
glw::GLint maxDrawBuffers = -1;
glGetIntegerv(GL_MAX_DRAW_BUFFERS, &maxDrawBuffers);
if (!checkDrawBuffersIndexedSupport(m_context))
throw tcu::NotSupportedError("GL_EXT_draw_buffers_indexed is not supported", DE_NULL, __FILE__, __LINE__);
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if modeRGB is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquationSeparatei(0, -1, GL_FUNC_ADD);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if modeAlpha is not GL_FUNC_ADD, GL_FUNC_SUBTRACT, "
"GL_FUNC_REVERSE_SUBTRACT, GL_MAX or GL_MIN.");
glBlendEquationSeparatei(0, GL_FUNC_ADD, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if buf is not in the range zero to the value of "
"MAX_DRAW_BUFFERS minus one.");
glBlendEquationSeparatei(-1, GL_FUNC_ADD, GL_FUNC_ADD);
expectError(GL_INVALID_VALUE);
glBlendEquationSeparatei(maxDrawBuffers, GL_FUNC_ADD, GL_FUNC_ADD);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_funci, "Invalid glBlendFunci() usage", {
glw::GLint maxDrawBuffers = -1;
glGetIntegerv(GL_MAX_DRAW_BUFFERS, &maxDrawBuffers);
if (!checkDrawBuffersIndexedSupport(m_context))
throw tcu::NotSupportedError("GL_EXT_draw_buffers_indexed is not supported", DE_NULL, __FILE__, __LINE__);
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if either sfactor or dfactor is not an accepted value.");
glBlendFunci(0, -1, GL_ONE);
expectError(GL_INVALID_ENUM);
glBlendFunci(0, GL_ONE, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if buf is not in the range zero to the value of "
"MAX_DRAW_BUFFERS minus one.");
glBlendFunci(-1, GL_ONE, GL_ONE);
expectError(GL_INVALID_VALUE);
glBlendFunci(maxDrawBuffers, GL_ONE, GL_ONE);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(blend_func_separatei, "Invalid glBlendFuncSeparatei() usage", {
glw::GLint maxDrawBuffers = -1;
glGetIntegerv(GL_MAX_DRAW_BUFFERS, &maxDrawBuffers);
if (!checkDrawBuffersIndexedSupport(m_context))
throw tcu::NotSupportedError("GL_EXT_draw_buffers_indexed is not supported", DE_NULL, __FILE__, __LINE__);
m_log << TestLog::Section(
"", "GL_INVALID_ENUM is generated if srcRGB, dstRGB, srcAlpha, or dstAlpha is not an accepted value.");
glBlendFuncSeparatei(0, -1, GL_ONE, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparatei(0, GL_ZERO, -1, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparatei(0, GL_ZERO, GL_ONE, -1, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_ENUM);
glBlendFuncSeparatei(0, GL_ZERO, GL_ONE, GL_SRC_COLOR, -1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if buf is not in the range zero to the value of "
"MAX_DRAW_BUFFERS minus one.");
glBlendFuncSeparatei(-1, GL_ONE, GL_ONE, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_VALUE);
glBlendFuncSeparatei(maxDrawBuffers, GL_ONE, GL_ONE, GL_SRC_COLOR, GL_ONE_MINUS_SRC_COLOR);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
// Rasterization API functions
ES3F_ADD_API_CASE(cull_face, "Invalid glCullFace() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if mode is not an accepted value.");
glCullFace(-1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(front_face, "Invalid glFrontFace() usage", {
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if mode is not an accepted value.");
glFrontFace(-1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(line_width, "Invalid glLineWidth() usage", {
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if width is less than or equal to 0.");
glLineWidth(0);
expectError(GL_INVALID_VALUE);
glLineWidth(-1);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
// Asynchronous queries
ES3F_ADD_API_CASE(gen_queries, "Invalid glGenQueries() usage", {
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if n is negative.");
GLuint ids;
glGenQueries(-1, &ids);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(begin_query, "Invalid glBeginQuery() usage", {
GLuint ids[3];
glGenQueries(3, ids);
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if target is not one of the accepted tokens.");
glBeginQuery(-1, ids[0]);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_OPERATION is generated if glBeginQuery is executed while a query "
"object of the same target is already active.");
glBeginQuery(GL_ANY_SAMPLES_PASSED, ids[0]);
expectError(GL_NO_ERROR);
glBeginQuery(GL_ANY_SAMPLES_PASSED, ids[1]);
expectError(GL_INVALID_OPERATION);
// \note GL_ANY_SAMPLES_PASSED and GL_ANY_SAMPLES_PASSED_CONSERVATIVE alias to the same target for the purposes of this error.
glBeginQuery(GL_ANY_SAMPLES_PASSED_CONSERVATIVE, ids[1]);
expectError(GL_INVALID_OPERATION);
glBeginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN, ids[1]);
expectError(GL_NO_ERROR);
glBeginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN, ids[2]);
expectError(GL_INVALID_OPERATION);
glEndQuery(GL_ANY_SAMPLES_PASSED);
glEndQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN);
expectError(GL_NO_ERROR);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_OPERATION is generated if id is 0.");
glBeginQuery(GL_ANY_SAMPLES_PASSED, 0);
expectError(GL_INVALID_OPERATION);
m_log << TestLog::EndSection;
m_log << TestLog::Section("",
"GL_INVALID_OPERATION is generated if id not a name returned from a previous call to "
"glGenQueries, or if such a name has since been deleted with glDeleteQueries.");
glBeginQuery(GL_ANY_SAMPLES_PASSED, -1);
expectError(GL_INVALID_OPERATION);
glDeleteQueries(1, &ids[2]);
expectError(GL_NO_ERROR);
glBeginQuery(GL_ANY_SAMPLES_PASSED, ids[2]);
expectError(GL_INVALID_OPERATION);
m_log << TestLog::EndSection;
m_log << TestLog::Section(
"", "GL_INVALID_OPERATION is generated if id is the name of an already active query object.");
glBeginQuery(GL_ANY_SAMPLES_PASSED, ids[0]);
expectError(GL_NO_ERROR);
glBeginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN, ids[0]);
expectError(GL_INVALID_OPERATION);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_OPERATION is generated if id refers to an existing query object "
"whose type does not does not match target.");
glEndQuery(GL_ANY_SAMPLES_PASSED);
expectError(GL_NO_ERROR);
glBeginQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN, ids[0]);
expectError(GL_INVALID_OPERATION);
m_log << TestLog::EndSection;
glDeleteQueries(2, &ids[0]);
expectError(GL_NO_ERROR);
});
ES3F_ADD_API_CASE(end_query, "Invalid glEndQuery() usage", {
GLuint id;
glGenQueries(1, &id);
m_log << TestLog::Section("", "GL_INVALID_ENUM is generated if target is not one of the accepted tokens.");
glEndQuery(-1);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_OPERATION is generated if glEndQuery is executed when a query object "
"of the same target is not active.");
glEndQuery(GL_ANY_SAMPLES_PASSED);
expectError(GL_INVALID_OPERATION);
glBeginQuery(GL_ANY_SAMPLES_PASSED, id);
expectError(GL_NO_ERROR);
glEndQuery(GL_TRANSFORM_FEEDBACK_PRIMITIVES_WRITTEN);
expectError(GL_INVALID_OPERATION);
glEndQuery(GL_ANY_SAMPLES_PASSED);
expectError(GL_NO_ERROR);
m_log << TestLog::EndSection;
glDeleteQueries(1, &id);
expectError(GL_NO_ERROR);
});
ES3F_ADD_API_CASE(delete_queries, "Invalid glDeleteQueries() usage", {
GLuint id;
glGenQueries(1, &id);
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if n is negative.");
glDeleteQueries(-1, &id);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
glDeleteQueries(1, &id);
});
// Sync objects
ES3F_ADD_API_CASE(fence_sync, "Invalid glFenceSync() usage", {
m_log << TestLog::Section("",
"GL_INVALID_ENUM is generated if condition is not GL_SYNC_GPU_COMMANDS_COMPLETE.");
glFenceSync(-1, 0);
expectError(GL_INVALID_ENUM);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if flags is not zero.");
glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0x0010);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
});
ES3F_ADD_API_CASE(wait_sync, "Invalid glWaitSync() usage", {
GLsync sync = glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if sync is not the name of a sync object.");
glWaitSync(0, 0, GL_TIMEOUT_IGNORED);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if flags is not zero.");
glWaitSync(sync, 0x0010, GL_TIMEOUT_IGNORED);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if timeout is not GL_TIMEOUT_IGNORED.");
glWaitSync(sync, 0, 0);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
glDeleteSync(sync);
});
ES3F_ADD_API_CASE(client_wait_sync, "Invalid glClientWaitSync() usage", {
GLsync sync = glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
m_log << TestLog::Section("",
"GL_INVALID_VALUE is generated if sync is not the name of an existing sync object.");
glClientWaitSync(0, 0, 10000);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
m_log << TestLog::Section("", "GL_INVALID_VALUE is generated if flags contains any unsupported flag.");
glClientWaitSync(sync, 0x00000004, 10000);
expectError(GL_INVALID_VALUE);
m_log << TestLog::EndSection;
glDeleteSync(sync);
});
ES3F_ADD_API_CASE(delete_sync, "Invalid glDeleteSync() usage", {
m_log << TestLog::Section(
"", "GL_INVALID_VALUE is generated if sync is neither zero or the name of a sync object.");
glDeleteSync((GLsync)1);
expectError(GL_INVALID_VALUE);
glDeleteSync(0);
expectError(GL_NO_ERROR);
m_log << TestLog::EndSection;
});
}
} // namespace Functional
} // namespace gles3
} // namespace deqp