1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program OpenGL ES 3.1 Module
3 * -------------------------------------------------
4 *
5 * Copyright 2014 The Android Open Source Project
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 *//*!
20 * \file
21 * \brief SSBO array length tests.
22 *//*--------------------------------------------------------------------*/
23
24 #include "es31fSSBOArrayLengthTests.hpp"
25 #include "gluShaderProgram.hpp"
26 #include "gluRenderContext.hpp"
27 #include "tcuTestLog.hpp"
28 #include "glwFunctions.hpp"
29 #include "glwEnums.hpp"
30 #include "deStringUtil.hpp"
31
32 #include <sstream>
33
34 namespace deqp
35 {
36 namespace gles31
37 {
38 namespace Functional
39 {
40 namespace
41 {
42
43 class SSBOArrayLengthCase : public TestCase
44 {
45 public:
46 enum ArrayAccess
47 {
48 ACCESS_DEFAULT = 0,
49 ACCESS_WRITEONLY,
50 ACCESS_READONLY,
51
52 ACCESS_LAST
53 };
54
55 SSBOArrayLengthCase (Context& context, const char* name, const char* desc, ArrayAccess access, bool sized);
56 ~SSBOArrayLengthCase (void);
57
58 void init (void);
59 void deinit (void);
60 IterateResult iterate (void);
61
62 private:
63 std::string genComputeSource (void) const;
64
65 const ArrayAccess m_access;
66 const bool m_sized;
67
68 glu::ShaderProgram* m_shader;
69 deUint32 m_targetBufferID;
70 deUint32 m_outputBufferID;
71
72 static const int s_fixedBufferSize = 16;
73 };
74
SSBOArrayLengthCase(Context & context,const char * name,const char * desc,ArrayAccess access,bool sized)75 SSBOArrayLengthCase::SSBOArrayLengthCase (Context& context, const char* name, const char* desc, ArrayAccess access, bool sized)
76 : TestCase (context, name, desc)
77 , m_access (access)
78 , m_sized (sized)
79 , m_shader (DE_NULL)
80 , m_targetBufferID (0)
81 , m_outputBufferID (0)
82 {
83 }
84
~SSBOArrayLengthCase(void)85 SSBOArrayLengthCase::~SSBOArrayLengthCase (void)
86 {
87 deinit();
88 }
89
init(void)90 void SSBOArrayLengthCase::init (void)
91 {
92 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
93 const deUint32 invalidValues[] = { 0xFFFFFFFFUL, 0xAAAAAAAAUL };
94
95 // program
96 m_shader = new glu::ShaderProgram(m_context.getRenderContext(), glu::ProgramSources() << glu::ComputeSource(genComputeSource()));
97 m_testCtx.getLog() << *m_shader;
98
99 if (!m_shader->isOk())
100 throw tcu::TestError("Failed to build shader");
101
102 // gen and attach buffers
103 gl.genBuffers(1, &m_outputBufferID);
104 gl.bindBuffer(GL_SHADER_STORAGE_BUFFER, m_outputBufferID);
105 gl.bufferData(GL_SHADER_STORAGE_BUFFER, 2 * (int)sizeof(deUint32), invalidValues, GL_DYNAMIC_COPY);
106
107 gl.genBuffers(1, &m_targetBufferID);
108 gl.bindBuffer(GL_SHADER_STORAGE_BUFFER, m_targetBufferID);
109
110 GLU_EXPECT_NO_ERROR(gl.getError(), "create buffers");
111
112 gl.bindBufferBase(GL_SHADER_STORAGE_BUFFER, 0, m_outputBufferID);
113 gl.bindBufferBase(GL_SHADER_STORAGE_BUFFER, 1, m_targetBufferID);
114
115 GLU_EXPECT_NO_ERROR(gl.getError(), "bind buffers");
116
117 // check the ssbo has expected layout
118 {
119 const deUint32 index = gl.getProgramResourceIndex(m_shader->getProgram(), GL_BUFFER_VARIABLE, "Out.outLength");
120 const glw::GLenum prop = GL_OFFSET;
121 glw::GLint result = 0;
122
123 if (index == GL_INVALID_INDEX)
124 throw tcu::TestError("Failed to find outLength variable");
125
126 gl.getProgramResourceiv(m_shader->getProgram(), GL_BUFFER_VARIABLE, index, 1, &prop, 1, DE_NULL, &result);
127
128 if (result != 0)
129 throw tcu::TestError("Unexpected outLength location");
130 }
131 {
132 const deUint32 index = gl.getProgramResourceIndex(m_shader->getProgram(), GL_BUFFER_VARIABLE, "Out.unused");
133 const glw::GLenum prop = GL_OFFSET;
134 glw::GLint result = 0;
135
136 if (index == GL_INVALID_INDEX)
137 throw tcu::TestError("Failed to find unused variable");
138
139 gl.getProgramResourceiv(m_shader->getProgram(), GL_BUFFER_VARIABLE, index, 1, &prop, 1, DE_NULL, &result);
140
141 if (result != 4)
142 throw tcu::TestError("Unexpected unused location");
143 }
144 {
145 const deUint32 index = gl.getProgramResourceIndex(m_shader->getProgram(), GL_BUFFER_VARIABLE, "Target.array");
146 const glw::GLenum prop = GL_ARRAY_STRIDE;
147 glw::GLint result = 0;
148
149 if (index == GL_INVALID_INDEX)
150 throw tcu::TestError("Failed to find array variable");
151
152 gl.getProgramResourceiv(m_shader->getProgram(), GL_BUFFER_VARIABLE, index, 1, &prop, 1, DE_NULL, &result);
153
154 if (result != 4)
155 throw tcu::TestError("Unexpected array stride");
156 }
157 }
158
deinit(void)159 void SSBOArrayLengthCase::deinit (void)
160 {
161 if (m_shader)
162 {
163 delete m_shader;
164 m_shader = DE_NULL;
165 }
166
167 if (m_targetBufferID)
168 {
169 m_context.getRenderContext().getFunctions().deleteBuffers(1, &m_targetBufferID);
170 m_targetBufferID = 0;
171 }
172
173 if (m_outputBufferID)
174 {
175 m_context.getRenderContext().getFunctions().deleteBuffers(1, &m_outputBufferID);
176 m_outputBufferID = 0;
177 }
178 }
179
iterate(void)180 SSBOArrayLengthCase::IterateResult SSBOArrayLengthCase::iterate (void)
181 {
182 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
183 bool error = false;
184
185 // Update buffer size
186
187 m_testCtx.getLog() << tcu::TestLog::Message << "Allocating float memory buffer with " << static_cast<int>(s_fixedBufferSize) << " elements." << tcu::TestLog::EndMessage;
188
189 gl.bindBuffer(GL_SHADER_STORAGE_BUFFER, m_targetBufferID);
190 gl.bufferData(GL_SHADER_STORAGE_BUFFER, s_fixedBufferSize * (int)sizeof(float), DE_NULL, GL_DYNAMIC_COPY);
191
192 GLU_EXPECT_NO_ERROR(gl.getError(), "update buffer");
193
194 // Run compute
195
196 m_testCtx.getLog() << tcu::TestLog::Message << "Running compute shader." << tcu::TestLog::EndMessage;
197
198 gl.useProgram(m_shader->getProgram());
199 gl.dispatchCompute(1, 1, 1);
200
201 GLU_EXPECT_NO_ERROR(gl.getError(), "dispatch");
202
203 // Verify
204 {
205 const void* ptr;
206
207 gl.bindBuffer(GL_SHADER_STORAGE_BUFFER, m_outputBufferID);
208 ptr = gl.mapBufferRange(GL_SHADER_STORAGE_BUFFER, 0, (int)sizeof(deUint32), GL_MAP_READ_BIT);
209 GLU_EXPECT_NO_ERROR(gl.getError(), "map");
210
211 if (!ptr)
212 throw tcu::TestError("mapBufferRange returned NULL");
213
214 if (*(const deUint32*)ptr != (deUint32)s_fixedBufferSize)
215 {
216 m_testCtx.getLog() << tcu::TestLog::Message << "ERROR: Length returned was " << *(const deUint32*)ptr << ", expected " << static_cast<int>(s_fixedBufferSize) << tcu::TestLog::EndMessage;
217 error = true;
218 }
219 else
220 m_testCtx.getLog() << tcu::TestLog::Message << "Length returned was correct." << tcu::TestLog::EndMessage;
221
222 if (gl.unmapBuffer(GL_SHADER_STORAGE_BUFFER) == GL_FALSE)
223 throw tcu::TestError("unmapBuffer returned false");
224
225 GLU_EXPECT_NO_ERROR(gl.getError(), "unmap");
226 }
227
228 if (!error)
229 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
230 else
231 m_testCtx.setTestResult(QP_TEST_RESULT_FAIL, "Fail");
232 return STOP;
233 }
234
genComputeSource(void) const235 std::string SSBOArrayLengthCase::genComputeSource (void) const
236 {
237 const std::string qualifierStr = (m_access == ACCESS_READONLY) ? ("readonly ") : (m_access == ACCESS_WRITEONLY) ? ("writeonly ") : ("");
238 const std::string sizeStr = (m_sized) ? (de::toString(static_cast<int>(s_fixedBufferSize))) : ("");
239
240 std::ostringstream buf;
241 buf << "#version 310 es\n"
242 << "layout(local_size_x = 1, local_size_y = 1) in;\n"
243 << "layout(std430) buffer;\n"
244 << "\n"
245 << "layout(binding = 0) buffer Out\n"
246 << "{\n"
247 << " int outLength;\n"
248 << " uint unused;\n"
249 << "} sb_out;\n"
250 << "layout(binding = 1) " << qualifierStr << "buffer Target\n"
251 << "{\n"
252 << " float array[" << sizeStr << "];\n"
253 << "} sb_target;\n\n"
254 << "void main (void)\n"
255 << "{\n";
256
257 // read
258 if (m_access == ACCESS_READONLY || m_access == ACCESS_DEFAULT)
259 buf << " sb_out.unused = uint(sb_target.array[1]);\n";
260
261 // write
262 if (m_access == ACCESS_WRITEONLY || m_access == ACCESS_DEFAULT)
263 buf << " sb_target.array[2] = float(sb_out.unused);\n";
264
265 // actual test
266 buf << "\n"
267 << " sb_out.outLength = sb_target.array.length();\n"
268 << "}\n";
269
270 return buf.str();
271 }
272
273 } // anonymous
274
SSBOArrayLengthTests(Context & context)275 SSBOArrayLengthTests::SSBOArrayLengthTests (Context& context)
276 : TestCaseGroup(context, "array_length", "Test array.length()")
277 {
278 }
279
~SSBOArrayLengthTests(void)280 SSBOArrayLengthTests::~SSBOArrayLengthTests (void)
281 {
282 }
283
init(void)284 void SSBOArrayLengthTests::init (void)
285 {
286 static const struct Qualifier
287 {
288 SSBOArrayLengthCase::ArrayAccess access;
289 const char* name;
290 const char* desc;
291 } qualifiers[] =
292 {
293 { SSBOArrayLengthCase::ACCESS_DEFAULT, "", "" },
294 { SSBOArrayLengthCase::ACCESS_WRITEONLY, "writeonly_", "writeonly" },
295 { SSBOArrayLengthCase::ACCESS_READONLY, "readonly_", "readonly" },
296 };
297
298 static const bool arraysSized[] = { true, false };
299
300 for (int sizeNdx = 0; sizeNdx < DE_LENGTH_OF_ARRAY(arraysSized); ++sizeNdx)
301 for (int qualifierNdx = 0; qualifierNdx < DE_LENGTH_OF_ARRAY(qualifiers); ++qualifierNdx)
302 {
303 const std::string name = std::string() + ((arraysSized[sizeNdx]) ? ("sized_") : ("unsized_")) + qualifiers[qualifierNdx].name + "array";
304 const std::string desc = std::string("Test length() of ") + ((arraysSized[sizeNdx]) ? ("sized ") : ("unsized ")) + qualifiers[qualifierNdx].name + " array";
305
306 this->addChild(new SSBOArrayLengthCase(m_context, name.c_str(), desc.c_str(), qualifiers[qualifierNdx].access, arraysSized[sizeNdx]));
307 }
308 }
309
310 } // Functional
311 } // gles31
312 } // deqp
313