[Piglit] [PATCH v2] arb_program_interface_query: GetProgramResourceLocation test

Tapani Pälli tapani.palli at intel.com
Thu Feb 19 04:54:47 PST 2015


All tests pass on Nvidia GTX 660 (binary driver version 331.38).
GetProgramResourceLocationIndex will be covered by a separate test.

v2:
   - separate subroutine and compute enums and test only when
     extensions are available.
   - add some more inputs and outputs
   - add validation against old matching API functions

Signed-off-by: Tapani Pälli <tapani.palli at intel.com>
---
 tests/all.py                                       |   4 +
 tests/spec/CMakeLists.txt                          |   1 +
 .../arb_program_interface_query/CMakeLists.gl.txt  |  12 +
 .../arb_program_interface_query/CMakeLists.txt     |   1 +
 .../resource-location.c                            | 436 +++++++++++++++++++++
 5 files changed, 454 insertions(+)
 create mode 100644 tests/spec/arb_program_interface_query/CMakeLists.gl.txt
 create mode 100644 tests/spec/arb_program_interface_query/CMakeLists.txt
 create mode 100644 tests/spec/arb_program_interface_query/resource-location.c

diff --git a/tests/all.py b/tests/all.py
index 6192a15..ebe8977 100644
--- a/tests/all.py
+++ b/tests/all.py
@@ -2192,6 +2192,10 @@ for test_type in ('shader', 'api'):
     test_name = 'overlapping-locations-input-attribs {0}'.format(test_type)
     arb_explicit_attrib_location[test_name] = PiglitGLTest(test_name)
 
+arb_program_interface_query = {}
+spec['ARB_program_interface_query'] = arb_program_interface_query
+add_plain_test(arb_program_interface_query, 'arb_program_interface_query-resource-location')
+
 # Group ARB_explicit_uniform_location
 arb_explicit_uniform_location = {}
 spec['ARB_explicit_uniform_location'] = arb_explicit_uniform_location
diff --git a/tests/spec/CMakeLists.txt b/tests/spec/CMakeLists.txt
index dfd822b..dcb8857 100644
--- a/tests/spec/CMakeLists.txt
+++ b/tests/spec/CMakeLists.txt
@@ -114,3 +114,4 @@ add_subdirectory (arb_blend_func_extended)
 add_subdirectory (ext_unpack_subimage)
 add_subdirectory (arb_vertex_array_object)
 add_subdirectory (oes_texture_float)
+add_subdirectory (arb_program_interface_query)
diff --git a/tests/spec/arb_program_interface_query/CMakeLists.gl.txt b/tests/spec/arb_program_interface_query/CMakeLists.gl.txt
new file mode 100644
index 0000000..6fb7eaf
--- /dev/null
+++ b/tests/spec/arb_program_interface_query/CMakeLists.gl.txt
@@ -0,0 +1,12 @@
+include_directories(
+       ${GLEXT_INCLUDE_DIR}
+       ${OPENGL_INCLUDE_PATH}
+)
+
+link_libraries (
+       piglitutil_${piglit_target_api}
+       ${OPENGL_gl_LIBRARY}
+       ${OPENGL_glu_LIBRARY}
+)
+
+piglit_add_executable (arb_program_interface_query-resource-location resource-location.c)
diff --git a/tests/spec/arb_program_interface_query/CMakeLists.txt b/tests/spec/arb_program_interface_query/CMakeLists.txt
new file mode 100644
index 0000000..144a306
--- /dev/null
+++ b/tests/spec/arb_program_interface_query/CMakeLists.txt
@@ -0,0 +1 @@
+piglit_include_target_api()
diff --git a/tests/spec/arb_program_interface_query/resource-location.c b/tests/spec/arb_program_interface_query/resource-location.c
new file mode 100644
index 0000000..baec18f
--- /dev/null
+++ b/tests/spec/arb_program_interface_query/resource-location.c
@@ -0,0 +1,436 @@
+/*
+ * Copyright © 2014 Intel Corporation
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a
+ * copy of this software and associated documentation files (the "Software"),
+ * to deal in the Software without restriction, including without limitation
+ * the rights to use, copy, modify, merge, publish, distribute, sublicense,
+ * and/or sell copies of the Software, and to permit persons to whom the
+ * Software is furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice (including the next
+ * paragraph) shall be included in all copies or substantial portions of the
+ * Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
+ * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
+ * IN THE SOFTWARE.
+ */
+
+/**
+ * \file resource-location.c
+ *
+ * Tests GetProgramResourceLocation interface. Iterates over valid enums and
+ * checks for an invalid one. Then the test compiles shader programs to query
+ * locations of all valid enums and validates the result. Tests verify location
+ * and index values against old matching API functions.
+ *
+ * From the GL_ARB_program_interface_query spec:
+ *     "The commands
+ *
+ *     int GetProgramResourceLocation(uint program, enum programInterface,
+ *                                    const char *name);
+ *     int GetProgramResourceLocationIndex(uint program, enum programInterface,
+ *                                         const char *name);
+ *
+ *     returns the location or the fragment color index, respectively, assigned
+ *     to the variable named <name> in interface <programInterface> of program
+ *     object <program>.  For both commands, the error INVALID_OPERATION is
+ *     generated if <program> has not been linked or was last linked
+ *     unsuccessfully.  For GetProgramResourceLocation, <programInterface> must
+ *     be one of UNIFORM, PROGRAM_INPUT, PROGRAM_OUTPUT,
+ *     VERTEX_SUBROUTINE_UNIFORM, TESS_CONTROL_SUBROUTINE_UNIFORM,
+ *     TESS_EVALUATION_SUBROUTINE_UNIFORM, GEOMETRY_SUBROUTINE_UNIFORM,
+ *     FRAGMENT_SUBROUTINE_UNIFORM, or COMPUTE_SUBROUTINE_UNIFORM.  For
+ *     GetProgramResourceLocationIndex, <programInterface> must be
+ *     PROGRAM_OUTPUT.  The value -1 will be returned by either command if an
+ *     error occurs, if <name> does not identify an active variable on
+ *     <programInterface>, or if <name> identifies an active variable that does
+ *     not have a valid location assigned, as described above.  The locations
+ *     returned by these commands are the same locations returned when querying
+ *     the LOCATION and LOCATION_INDEX resource properties.
+ */
+
+#include "piglit-util-gl.h"
+
+PIGLIT_GL_TEST_CONFIG_BEGIN
+
+	config.supports_gl_compat_version = 32;
+        config.supports_gl_core_version = 32;
+        config.window_visual = PIGLIT_GL_VISUAL_RGB;
+
+PIGLIT_GL_TEST_CONFIG_END
+
+static const GLenum valid_enums[] = {
+	GL_UNIFORM,
+	GL_PROGRAM_INPUT,
+	GL_PROGRAM_OUTPUT,
+};
+
+static const GLenum valid_enums_sub[] = {
+	GL_VERTEX_SUBROUTINE_UNIFORM,
+	GL_GEOMETRY_SUBROUTINE_UNIFORM,
+	GL_FRAGMENT_SUBROUTINE_UNIFORM,
+};
+
+static const GLenum valid_enums_sub_tes[] = {
+	GL_TESS_CONTROL_SUBROUTINE_UNIFORM,
+	GL_TESS_EVALUATION_SUBROUTINE_UNIFORM,
+	GL_COMPUTE_SUBROUTINE_UNIFORM
+};
+
+static const GLenum valid_enums_sub_com[] = {
+	GL_COMPUTE_SUBROUTINE_UNIFORM
+};
+
+
+static const char vs_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+        "layout (location = 3) in vec4 input0;\n"
+        "layout (location = 6) in vec4 input1;\n"
+        "void main() {\n"
+                "gl_Position = input0 * input1;\n"
+        "}";
+
+static const char fs_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location  : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"layout (location = 9) uniform vec4 color;\n"
+	"layout (location = 0) out vec4 output0;\n"
+        "void main() {\n"
+                "output0 = color;\n"
+        "}";
+
+static const char vs_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+        "in vec4 vertex;\n"
+	"subroutine vec4 vs_offset();\n"
+	"layout (location = 3) subroutine uniform vs_offset VERTEX;\n"
+	"subroutine (vs_offset) vec4 x() { return vec4(1.0, 0.0, 0.0, 0.0); }\n"
+        "void main() {\n"
+                "gl_Position = vertex + VERTEX();\n"
+        "}";
+
+static const char fs_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+	"subroutine vec4 fs_offset();\n"
+	"layout (location = 3) subroutine uniform fs_offset FRAGMENT;\n"
+	"subroutine (fs_offset) vec4 red() { return vec4(1.0, 0.0, 0.0, 1.0); }\n"
+	"out vec4 result;\n"
+        "void main() {\n"
+                "result = FRAGMENT();\n"
+        "}";
+
+static const char gs_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+	"#extension GL_EXT_geometry_shader4 : require\n"
+	"layout(triangles) in;\n"
+	"layout(triangle_strip, max_vertices = 3) out;\n"
+	"subroutine vec4 gs_offset();\n"
+	"layout (location = 3) subroutine uniform gs_offset GEOMETRY;\n"
+	"subroutine (gs_offset) vec4 x() { return vec4(1.0, 0.0, 0.0, 0.0); }\n"
+        "void main() {\n"
+		"for(int i = 0; i < 3; i++) {\n"
+			"gl_Position = gl_in[i].gl_Position + GEOMETRY();\n"
+			"EmitVertex();\n"
+		"}\n"
+		"EndPrimitive();\n"
+        "}";
+
+static const char tcs_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+	"#extension GL_ARB_tessellation_shader : require\n"
+	"layout(vertices = 3) out;\n"
+	"subroutine vec4 tcs_offset();\n"
+	"layout (location = 3) subroutine uniform tcs_offset TESS_CONTROL;\n"
+	"subroutine (tcs_offset) vec4 x() { return vec4(1.0, 0.0, 0.0, 0.0); }\n"
+        "void main() {\n"
+		"gl_out[gl_InvocationID].gl_Position = vec4(0.0); + TESS_CONTROL();\n"
+        "}";
+
+static const char tes_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+	"#extension GL_ARB_tessellation_shader : require\n"
+	"layout(triangles) in;\n"
+	"subroutine vec4 tes_offset();\n"
+	"layout (location = 3) subroutine uniform tes_offset TESS_EVALUATION;\n"
+	"subroutine (tes_offset) vec4 x() { return vec4(1.0, 0.0, 0.0, 0.0); }\n"
+        "void main() {\n"
+		"gl_Position = vec4(0.0) + TESS_EVALUATION();\n"
+        "}";
+
+static const char compute_subroutine_text[] =
+	"#version 150\n"
+	"#extension GL_ARB_explicit_attrib_location : require\n"
+	"#extension GL_ARB_explicit_uniform_location : require\n"
+	"#extension GL_ARB_shader_subroutine : require\n"
+	"#extension GL_ARB_shader_image_load_store : require\n"
+	"#extension GL_ARB_compute_shader : require\n"
+	"layout(local_size_x = 4) in;\n"
+	"layout(size4x32) uniform image2D tex;\n"
+	"subroutine vec4 com_offset();\n"
+	"layout (location = 3) subroutine uniform com_offset COMPUTE;\n"
+	"subroutine (com_offset) vec4 x() { return vec4(1.0, 0.0, 0.0, 0.0); }\n"
+        "void main() {\n"
+		"imageStore(tex, ivec2(0.0), COMPUTE());\n"
+        "}";
+
+enum piglit_result
+piglit_display(void)
+{
+	return PIGLIT_FAIL;
+}
+
+#define str(s) #s
+#define CHECK_SUB(s) \
+	if (glGetProgramResourceLocation(prog, \
+		GL_##s ## _SUBROUTINE_UNIFORM, str(s)) != 3) { \
+		piglit_report_subtest_result(PIGLIT_FAIL, __func__); \
+		return PIGLIT_FAIL; \
+	}
+
+/* Test subroutine uniform location query with compute. */
+static bool
+test_subroutine_stages_compute()
+{
+	GLuint prog, i;
+
+	if (!piglit_is_extension_supported("GL_ARB_shader_subroutine")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+	if (!piglit_is_extension_supported("GL_ARB_compute_shader")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+        prog = piglit_build_simple_program_multiple_shaders(
+                     GL_COMPUTE_SHADER, compute_subroutine_text,
+                     0);
+
+	glUseProgram(prog);
+
+	/* Iterate through all valid subroutine enums passing invalid name. */
+	for (i = 0; i < (sizeof(valid_enums_sub_com)/sizeof(GLenum)); i++) {
+		glGetProgramResourceLocation(prog, valid_enums_sub_com[i],
+                                             "name");
+			if (!piglit_check_gl_error(GL_NO_ERROR))
+				piglit_report_result(PIGLIT_FAIL);
+	}
+
+	CHECK_SUB(COMPUTE);
+
+	piglit_report_subtest_result(PIGLIT_PASS, __func__);
+	return true;
+}
+
+/* Test subroutine uniform location query with tessellation. */
+static bool
+test_subroutine_stages_tcs_tes()
+{
+	GLuint prog, i;
+
+	if (!piglit_is_extension_supported("GL_ARB_shader_subroutine")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+	if (!piglit_is_extension_supported("GL_ARB_tessellation_shader")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+        prog = piglit_build_simple_program_multiple_shaders(
+                     GL_VERTEX_SHADER, vs_text,
+                     GL_TESS_CONTROL_SHADER, tcs_subroutine_text,
+                     GL_TESS_EVALUATION_SHADER, tes_subroutine_text,
+                     GL_FRAGMENT_SHADER, fs_text,
+                     0);
+
+	glUseProgram(prog);
+
+	/* Iterate through all valid subroutine enums passing invalid name. */
+	for (i = 0; i < (sizeof(valid_enums_sub_tes)/sizeof(GLenum)); i++) {
+		glGetProgramResourceLocation(prog, valid_enums_sub_tes[i],
+                                             "name");
+			if (!piglit_check_gl_error(GL_NO_ERROR))
+				piglit_report_result(PIGLIT_FAIL);
+	}
+
+	CHECK_SUB(TESS_CONTROL);
+	CHECK_SUB(TESS_EVALUATION);
+
+	piglit_report_subtest_result(PIGLIT_PASS, __func__);
+	return true;
+}
+
+/* Test subroutine uniform location query with vs, fs and gs. */
+static bool
+test_subroutine_stages_vs_fs_gs()
+{
+	GLuint prog, i;
+
+	if (!piglit_is_extension_supported("GL_ARB_shader_subroutine")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+	if (!piglit_is_extension_supported("GL_ARB_geometry_shader4")) {
+		piglit_report_subtest_result(PIGLIT_SKIP, __func__);
+		return true;
+	}
+
+        prog = piglit_build_simple_program_multiple_shaders(
+                     GL_VERTEX_SHADER, vs_subroutine_text,
+                     GL_GEOMETRY_SHADER, gs_subroutine_text,
+                     GL_FRAGMENT_SHADER, fs_subroutine_text,
+                     0);
+
+	glUseProgram(prog);
+
+	/* Iterate through all valid subroutine enums passing invalid name. */
+	for (i = 0; i < (sizeof(valid_enums_sub)/sizeof(GLenum)); i++) {
+		glGetProgramResourceLocation(prog, valid_enums_sub[i], "name");
+			if (!piglit_check_gl_error(GL_NO_ERROR))
+				piglit_report_result(PIGLIT_FAIL);
+	}
+
+	CHECK_SUB(VERTEX);
+	CHECK_SUB(FRAGMENT);
+	CHECK_SUB(GEOMETRY);
+
+	piglit_report_subtest_result(PIGLIT_PASS, __func__);
+	return true;
+}
+
+
+static void
+validate_location(GLuint prog, GLenum iface, const char *name, GLuint value)
+{
+	GLuint loc = glGetProgramResourceLocation(prog, iface, name);
+
+	/* Validate result against old API. */
+	switch (iface) {
+	case GL_UNIFORM:
+		if (glGetUniformLocation(prog, name) != loc)
+			piglit_report_result(PIGLIT_FAIL);
+		break;
+	case GL_PROGRAM_INPUT:
+		if (glGetAttribLocation(prog, name) != loc)
+			piglit_report_result(PIGLIT_FAIL);
+		break;
+	case GL_PROGRAM_OUTPUT:
+		if (glGetFragDataLocation(prog, name) != loc)
+			piglit_report_result(PIGLIT_FAIL);
+		break;
+	}
+
+	/* Expected value. */
+	if (loc != value) {
+		fprintf(stderr, "got value %d for %s, expected %d\n",
+			loc, name, value);
+		piglit_report_result(PIGLIT_FAIL);
+	}
+
+	/* No errors should have happened. */
+	if (!piglit_check_gl_error(GL_NO_ERROR))
+		piglit_report_result(PIGLIT_FAIL);
+}
+
+void
+piglit_init(int argc, char **argv)
+{
+	GLuint prog, shader;
+	unsigned i;
+	bool pass = true;
+
+	piglit_require_extension("GL_ARB_program_interface_query");
+	piglit_require_extension("GL_ARB_explicit_attrib_location");
+	piglit_require_extension("GL_ARB_explicit_uniform_location");
+
+	/* Test invalid program. */
+	glGetProgramResourceLocation(42, GL_UNIFORM, "name");
+	if (!piglit_check_gl_error(GL_INVALID_VALUE)) {
+		piglit_report_subtest_result(PIGLIT_FAIL, "invalid program test 1");
+		pass = false;
+	}
+
+	/* Test passing a shader, not program. */
+	shader = piglit_compile_shader_text(GL_VERTEX_SHADER, vs_text);
+	glGetProgramResourceLocation(shader, GL_UNIFORM, "name");
+	if (!piglit_check_gl_error(GL_INVALID_OPERATION)) {
+		piglit_report_subtest_result(PIGLIT_FAIL, "invalid program test 2");
+		pass = false;
+	}
+
+	prog = piglit_build_simple_program_unlinked(vs_text, fs_text);
+
+	if (!piglit_check_gl_error(GL_NO_ERROR))
+		piglit_report_result(PIGLIT_FAIL);
+
+	/* Test unlinked program. */
+	glGetProgramResourceLocation(prog, GL_UNIFORM, "name");
+	if (!piglit_check_gl_error(GL_INVALID_OPERATION)) {
+		piglit_report_subtest_result(PIGLIT_FAIL, "invalid program test 3");
+		pass = false;
+	}
+
+	if (!piglit_check_gl_error(GL_NO_ERROR))
+		piglit_report_result(PIGLIT_FAIL);
+
+	/* Test a linked program. */
+	glLinkProgram(prog);
+	glUseProgram(prog);
+
+	/* Iterate through all valid enums passing invalid name. */
+	for (i = 0; i < (sizeof(valid_enums)/sizeof(GLenum)); i++) {
+		glGetProgramResourceLocation(prog, valid_enums[i], "name");
+			if (!piglit_check_gl_error(GL_NO_ERROR))
+				piglit_report_result(PIGLIT_FAIL);
+	}
+
+	/* Test invalid enum, there is no defined error by the spec. */
+	glGetProgramResourceLocation(prog, GL_ATOMIC_COUNTER_BUFFER, "name");
+	if (glGetError() == GL_NO_ERROR) {
+		piglit_report_subtest_result(PIGLIT_FAIL, "invalid enum test");
+		pass = false;
+	}
+
+	/* Valid inputs. */
+	validate_location(prog, GL_UNIFORM,        "color",   9);
+	validate_location(prog, GL_PROGRAM_INPUT,  "input0",  3);
+	validate_location(prog, GL_PROGRAM_INPUT,  "input1",  6);
+	validate_location(prog, GL_PROGRAM_OUTPUT, "output0", 0);
+
+	if (!piglit_check_gl_error(GL_NO_ERROR))
+		piglit_report_result(PIGLIT_FAIL);
+
+	/* Tests that require GL_ARB_shader_subroutine. */
+	pass = test_subroutine_stages_vs_fs_gs() && pass;
+	pass = test_subroutine_stages_tcs_tes() && pass;
+	pass = test_subroutine_stages_compute() && pass;
+
+	piglit_report_result(pass ? PIGLIT_PASS : PIGLIT_FAIL);
+}
-- 
2.1.0



More information about the Piglit mailing list