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 Sample shading tests
22 *//*--------------------------------------------------------------------*/
23
24 #include "es31fSampleShadingTests.hpp"
25 #include "es31fMultisampleShaderRenderCase.hpp"
26 #include "tcuRenderTarget.hpp"
27 #include "tcuSurface.hpp"
28 #include "glsStateQueryUtil.hpp"
29 #include "gluCallLogWrapper.hpp"
30 #include "gluContextInfo.hpp"
31 #include "gluShaderProgram.hpp"
32 #include "gluRenderContext.hpp"
33 #include "gluPixelTransfer.hpp"
34 #include "glwFunctions.hpp"
35 #include "glwEnums.hpp"
36 #include "deStringUtil.hpp"
37 #include "deRandom.hpp"
38
39 #include <map>
40
41 namespace deqp
42 {
43 namespace gles31
44 {
45 namespace Functional
46 {
47 namespace
48 {
49
50 using namespace gls::StateQueryUtil;
51
52 class SampleShadingStateCase : public TestCase
53 {
54 public:
55 SampleShadingStateCase (Context& ctx, const char* name, const char* desc, QueryType);
56
57 void init (void);
58 IterateResult iterate (void);
59
60 private:
61 const QueryType m_verifier;
62 };
63
SampleShadingStateCase(Context & ctx,const char * name,const char * desc,QueryType type)64 SampleShadingStateCase::SampleShadingStateCase (Context& ctx, const char* name, const char* desc, QueryType type)
65 : TestCase (ctx, name, desc)
66 , m_verifier (type)
67 {
68 }
69
init(void)70 void SampleShadingStateCase::init (void)
71 {
72 if (!m_context.getContextInfo().isExtensionSupported("GL_OES_sample_shading"))
73 throw tcu::NotSupportedError("Test requires GL_OES_sample_shading extension");
74 }
75
iterate(void)76 SampleShadingStateCase::IterateResult SampleShadingStateCase::iterate (void)
77 {
78 glu::CallLogWrapper gl (m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
79 tcu::ResultCollector result (m_testCtx.getLog(), " // ERROR: ");
80 gl.enableLogging(true);
81
82 // initial
83 {
84 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying initial value" << tcu::TestLog::EndMessage;
85 verifyStateBoolean(result, gl, GL_SAMPLE_SHADING, false, m_verifier);
86 }
87
88 // true and false too
89 {
90 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying random values" << tcu::TestLog::EndMessage;
91
92 gl.glEnable(GL_SAMPLE_SHADING);
93 verifyStateBoolean(result, gl, GL_SAMPLE_SHADING, true, m_verifier);
94
95 gl.glDisable(GL_SAMPLE_SHADING);
96 verifyStateBoolean(result, gl, GL_SAMPLE_SHADING, false, m_verifier);
97 }
98
99 result.setTestContextResult(m_testCtx);
100 return STOP;
101 }
102
103 class MinSampleShadingValueCase : public TestCase
104 {
105 public:
106 MinSampleShadingValueCase (Context& ctx, const char* name, const char* desc, QueryType);
107
108 void init (void);
109 IterateResult iterate (void);
110
111 private:
112 const QueryType m_verifier;
113 };
114
MinSampleShadingValueCase(Context & ctx,const char * name,const char * desc,QueryType type)115 MinSampleShadingValueCase::MinSampleShadingValueCase (Context& ctx, const char* name, const char* desc, QueryType type)
116 : TestCase (ctx, name, desc)
117 , m_verifier (type)
118 {
119 }
120
init(void)121 void MinSampleShadingValueCase::init (void)
122 {
123 if (!m_context.getContextInfo().isExtensionSupported("GL_OES_sample_shading"))
124 throw tcu::NotSupportedError("Test requires GL_OES_sample_shading extension");
125 }
126
iterate(void)127 MinSampleShadingValueCase::IterateResult MinSampleShadingValueCase::iterate (void)
128 {
129 glu::CallLogWrapper gl (m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
130 tcu::ResultCollector result (m_testCtx.getLog(), " // ERROR: ");
131
132 gl.enableLogging(true);
133
134 // initial
135 {
136 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying initial value" << tcu::TestLog::EndMessage;
137 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.0, m_verifier);
138 }
139
140 // special values
141 {
142 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying special values" << tcu::TestLog::EndMessage;
143
144 gl.glMinSampleShading(0.0f);
145 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.0, m_verifier);
146
147 gl.glMinSampleShading(1.0f);
148 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 1.0, m_verifier);
149
150 gl.glMinSampleShading(0.5f);
151 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.5, m_verifier);
152 }
153
154 // random values
155 {
156 const int numRandomTests = 10;
157 de::Random rnd (0xde123);
158
159 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying random values" << tcu::TestLog::EndMessage;
160
161 for (int randNdx = 0; randNdx < numRandomTests; ++randNdx)
162 {
163 const float value = rnd.getFloat();
164
165 gl.glMinSampleShading(value);
166 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, value, m_verifier);
167 }
168 }
169
170 result.setTestContextResult(m_testCtx);
171 return STOP;
172 }
173
174 class MinSampleShadingValueClampingCase : public TestCase
175 {
176 public:
177 MinSampleShadingValueClampingCase (Context& ctx, const char* name, const char* desc);
178
179 void init (void);
180 IterateResult iterate (void);
181 };
182
MinSampleShadingValueClampingCase(Context & ctx,const char * name,const char * desc)183 MinSampleShadingValueClampingCase::MinSampleShadingValueClampingCase (Context& ctx, const char* name, const char* desc)
184 : TestCase(ctx, name, desc)
185 {
186 }
187
init(void)188 void MinSampleShadingValueClampingCase::init (void)
189 {
190 if (!m_context.getContextInfo().isExtensionSupported("GL_OES_sample_shading"))
191 throw tcu::NotSupportedError("Test requires GL_OES_sample_shading extension");
192 }
193
iterate(void)194 MinSampleShadingValueClampingCase::IterateResult MinSampleShadingValueClampingCase::iterate (void)
195 {
196 glu::CallLogWrapper gl (m_context.getRenderContext().getFunctions(), m_testCtx.getLog());
197 tcu::ResultCollector result (m_testCtx.getLog(), " // ERROR: ");
198 gl.enableLogging(true);
199
200 m_testCtx.setTestResult(QP_TEST_RESULT_PASS, "Pass");
201
202 // special values
203 {
204 m_testCtx.getLog() << tcu::TestLog::Message << "Verifying clamped values. Value is clamped when specified." << tcu::TestLog::EndMessage;
205
206 gl.glMinSampleShading(-0.5f);
207 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.0, QUERY_FLOAT);
208
209 gl.glMinSampleShading(-1.0f);
210 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.0, QUERY_FLOAT);
211
212 gl.glMinSampleShading(-1.5f);
213 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 0.0, QUERY_FLOAT);
214
215 gl.glMinSampleShading(1.5f);
216 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 1.0, QUERY_FLOAT);
217
218 gl.glMinSampleShading(2.0f);
219 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 1.0, QUERY_FLOAT);
220
221 gl.glMinSampleShading(2.5f);
222 verifyStateFloat(result, gl, GL_MIN_SAMPLE_SHADING_VALUE, 1.0, QUERY_FLOAT);
223 }
224
225 result.setTestContextResult(m_testCtx);
226 return STOP;
227 }
228
229 class SampleShadingRenderingCase : public MultisampleShaderRenderUtil::MultisampleRenderCase
230 {
231 public:
232 enum TestType
233 {
234 TEST_DISCARD = 0,
235 TEST_COLOR,
236
237 TEST_LAST
238 };
239 SampleShadingRenderingCase (Context& ctx, const char* name, const char* desc, RenderTarget target, int numSamples, TestType type);
240 ~SampleShadingRenderingCase (void);
241
242 void init (void);
243 private:
244 void setShadingValue (int sampleCount);
245
246 void preDraw (void);
247 void postDraw (void);
248 std::string getIterationDescription (int iteration) const;
249
250 bool verifyImage (const tcu::Surface& resultImage);
251
252 std::string genFragmentSource (int numSamples) const;
253
254 enum
255 {
256 RENDER_SIZE = 128
257 };
258
259 const TestType m_type;
260 };
261
SampleShadingRenderingCase(Context & ctx,const char * name,const char * desc,RenderTarget target,int numSamples,TestType type)262 SampleShadingRenderingCase::SampleShadingRenderingCase (Context& ctx, const char* name, const char* desc, RenderTarget target, int numSamples, TestType type)
263 : MultisampleShaderRenderUtil::MultisampleRenderCase (ctx, name, desc, numSamples, target, RENDER_SIZE)
264 , m_type (type)
265 {
266 DE_ASSERT(type < TEST_LAST);
267 }
268
~SampleShadingRenderingCase(void)269 SampleShadingRenderingCase::~SampleShadingRenderingCase (void)
270 {
271 deinit();
272 }
273
init(void)274 void SampleShadingRenderingCase::init (void)
275 {
276 // requirements
277
278 if (!m_context.getContextInfo().isExtensionSupported("GL_OES_sample_shading"))
279 throw tcu::NotSupportedError("Test requires GL_OES_sample_shading extension");
280 if (m_renderTarget == TARGET_DEFAULT && m_context.getRenderTarget().getNumSamples() <= 1)
281 throw tcu::NotSupportedError("Multisampled default framebuffer required");
282
283 // test purpose and expectations
284 m_testCtx.getLog()
285 << tcu::TestLog::Message
286 << "Verifying that a varying is given at least N different values for different samples within a single pixel.\n"
287 << " Render high-frequency function, map result to black/white. Modify N with glMinSampleShading().\n"
288 << " => Resulting image should contain N+1 shades of gray.\n"
289 << tcu::TestLog::EndMessage;
290
291 // setup resources
292
293 MultisampleShaderRenderUtil::MultisampleRenderCase::init();
294
295 // set iterations
296
297 m_numIterations = m_numTargetSamples + 1;
298 }
299
setShadingValue(int sampleCount)300 void SampleShadingRenderingCase::setShadingValue (int sampleCount)
301 {
302 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
303
304 if (sampleCount == 0)
305 {
306 gl.disable(GL_SAMPLE_SHADING);
307 gl.minSampleShading(1.0f);
308 GLU_EXPECT_NO_ERROR(gl.getError(), "set ratio");
309 }
310 else
311 {
312 // Minimum number of samples is max(ceil(<mss> * <samples>),1). Decrease mss with epsilon to prevent
313 // ceiling to a too large sample count.
314 const float epsilon = 0.25f / (float)m_numTargetSamples;
315 const float ratio = (sampleCount / (float)m_numTargetSamples) - epsilon;
316
317 gl.enable(GL_SAMPLE_SHADING);
318 gl.minSampleShading(ratio);
319 GLU_EXPECT_NO_ERROR(gl.getError(), "set ratio");
320
321 m_testCtx.getLog()
322 << tcu::TestLog::Message
323 << "Setting MIN_SAMPLE_SHADING_VALUE = " << ratio << "\n"
324 << "Requested sample count: shadingValue * numSamples = " << ratio << " * " << m_numTargetSamples << " = " << (ratio * m_numTargetSamples) << "\n"
325 << "Minimum sample count: ceil(shadingValue * numSamples) = ceil(" << (ratio * m_numTargetSamples) << ") = " << sampleCount
326 << tcu::TestLog::EndMessage;
327
328 // can't fail with reasonable values of numSamples
329 DE_ASSERT(deFloatCeil(ratio * m_numTargetSamples) == float(sampleCount));
330 }
331 }
332
preDraw(void)333 void SampleShadingRenderingCase::preDraw (void)
334 {
335 setShadingValue(m_iteration);
336 }
337
postDraw(void)338 void SampleShadingRenderingCase::postDraw (void)
339 {
340 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
341
342 gl.disable(GL_SAMPLE_SHADING);
343 gl.minSampleShading(1.0f);
344 }
345
getIterationDescription(int iteration) const346 std::string SampleShadingRenderingCase::getIterationDescription (int iteration) const
347 {
348 if (iteration == 0)
349 return "Disabled SAMPLE_SHADING";
350 else
351 return "Samples per pixel: " + de::toString(iteration);
352 }
353
verifyImage(const tcu::Surface & resultImage)354 bool SampleShadingRenderingCase::verifyImage (const tcu::Surface& resultImage)
355 {
356 const int numShadesRequired = (m_iteration == 0) ? (2) : (m_iteration + 1);
357 const int rareThreshold = 100;
358 int rareCount = 0;
359 std::map<deUint32, int> shadeFrequency;
360
361 // we should now have n+1 different shades of white, n = num samples
362
363 m_testCtx.getLog()
364 << tcu::TestLog::Image("ResultImage", "Result Image", resultImage.getAccess())
365 << tcu::TestLog::Message
366 << "Verifying image has (at least) " << numShadesRequired << " different shades.\n"
367 << "Excluding pixels with no full coverage (pixels on the shared edge of the triangle pair)."
368 << tcu::TestLog::EndMessage;
369
370 for (int y = 0; y < RENDER_SIZE; ++y)
371 for (int x = 0; x < RENDER_SIZE; ++x)
372 {
373 const tcu::RGBA color = resultImage.getPixel(x, y);
374 const deUint32 packed = ((deUint32)color.getRed()) + ((deUint32)color.getGreen() << 8) + ((deUint32)color.getGreen() << 16);
375
376 // on the triangle edge, skip
377 if (x == y)
378 continue;
379
380 if (shadeFrequency.find(packed) == shadeFrequency.end())
381 shadeFrequency[packed] = 1;
382 else
383 shadeFrequency[packed] = shadeFrequency[packed] + 1;
384 }
385
386 for (std::map<deUint32, int>::const_iterator it = shadeFrequency.begin(); it != shadeFrequency.end(); ++it)
387 if (it->second < rareThreshold)
388 rareCount++;
389
390 m_testCtx.getLog()
391 << tcu::TestLog::Message
392 << "Found " << (int)shadeFrequency.size() << " different shades.\n"
393 << "\tRare (less than " << rareThreshold << " pixels): " << rareCount << "\n"
394 << "\tCommon: " << (int)shadeFrequency.size() - rareCount << "\n"
395 << tcu::TestLog::EndMessage;
396
397 if ((int)shadeFrequency.size() < numShadesRequired)
398 {
399 m_testCtx.getLog() << tcu::TestLog::Message << "Image verification failed." << tcu::TestLog::EndMessage;
400 return false;
401 }
402 return true;
403 }
404
genFragmentSource(int numSamples) const405 std::string SampleShadingRenderingCase::genFragmentSource (int numSamples) const
406 {
407 DE_UNREF(numSamples);
408
409 std::ostringstream buf;
410
411 buf << "#version 310 es\n"
412 "in highp vec4 v_position;\n"
413 "layout(location = 0) out mediump vec4 fragColor;\n"
414 "void main (void)\n"
415 "{\n"
416 " highp float field = dot(v_position.xy, v_position.xy) + dot(21.0 * v_position.xx, sin(3.1 * v_position.xy));\n"
417 " fragColor = vec4(1.0, 1.0, 1.0, 1.0);\n"
418 "\n"
419 " if (fract(field) > 0.5)\n";
420
421 if (m_type == TEST_DISCARD)
422 buf << " discard;\n";
423 else if (m_type == TEST_COLOR)
424 buf << " fragColor = vec4(0.0, 0.0, 0.0, 1.0);\n";
425 else
426 DE_ASSERT(false);
427
428 buf << "}";
429
430 return buf.str();
431 }
432
433 } // anonymous
434
SampleShadingTests(Context & context)435 SampleShadingTests::SampleShadingTests (Context& context)
436 : TestCaseGroup(context, "sample_shading", "Test sample shading")
437 {
438 }
439
~SampleShadingTests(void)440 SampleShadingTests::~SampleShadingTests (void)
441 {
442 }
443
init(void)444 void SampleShadingTests::init (void)
445 {
446 tcu::TestCaseGroup* const stateQueryGroup = new tcu::TestCaseGroup(m_testCtx, "state_query", "State query tests.");
447 tcu::TestCaseGroup* const minSamplesGroup = new tcu::TestCaseGroup(m_testCtx, "min_sample_shading", "Min sample shading tests.");
448
449 addChild(stateQueryGroup);
450 addChild(minSamplesGroup);
451
452 // .state query
453 {
454 stateQueryGroup->addChild(new SampleShadingStateCase (m_context, "sample_shading_is_enabled", "test SAMPLE_SHADING", QUERY_ISENABLED));
455 stateQueryGroup->addChild(new SampleShadingStateCase (m_context, "sample_shading_get_boolean", "test SAMPLE_SHADING", QUERY_BOOLEAN));
456 stateQueryGroup->addChild(new SampleShadingStateCase (m_context, "sample_shading_get_integer", "test SAMPLE_SHADING", QUERY_INTEGER));
457 stateQueryGroup->addChild(new SampleShadingStateCase (m_context, "sample_shading_get_float", "test SAMPLE_SHADING", QUERY_FLOAT));
458 stateQueryGroup->addChild(new SampleShadingStateCase (m_context, "sample_shading_get_integer64", "test SAMPLE_SHADING", QUERY_INTEGER64));
459 stateQueryGroup->addChild(new MinSampleShadingValueCase (m_context, "min_sample_shading_value_get_boolean", "test MIN_SAMPLE_SHADING_VALUE", QUERY_BOOLEAN));
460 stateQueryGroup->addChild(new MinSampleShadingValueCase (m_context, "min_sample_shading_value_get_integer", "test MIN_SAMPLE_SHADING_VALUE", QUERY_INTEGER));
461 stateQueryGroup->addChild(new MinSampleShadingValueCase (m_context, "min_sample_shading_value_get_float", "test MIN_SAMPLE_SHADING_VALUE", QUERY_FLOAT));
462 stateQueryGroup->addChild(new MinSampleShadingValueCase (m_context, "min_sample_shading_value_get_integer64", "test MIN_SAMPLE_SHADING_VALUE", QUERY_INTEGER64));
463 stateQueryGroup->addChild(new MinSampleShadingValueClampingCase (m_context, "min_sample_shading_value_clamping", "test MIN_SAMPLE_SHADING_VALUE clamping"));
464 }
465
466 // .min_sample_count
467 {
468 static const struct Target
469 {
470 SampleShadingRenderingCase::RenderTarget target;
471 int numSamples;
472 const char* name;
473 } targets[] =
474 {
475 { SampleShadingRenderingCase::TARGET_DEFAULT, 0, "default_framebuffer" },
476 { SampleShadingRenderingCase::TARGET_TEXTURE, 2, "multisample_texture_samples_2" },
477 { SampleShadingRenderingCase::TARGET_TEXTURE, 4, "multisample_texture_samples_4" },
478 { SampleShadingRenderingCase::TARGET_TEXTURE, 8, "multisample_texture_samples_8" },
479 { SampleShadingRenderingCase::TARGET_TEXTURE, 16, "multisample_texture_samples_16" },
480 { SampleShadingRenderingCase::TARGET_RENDERBUFFER, 2, "multisample_renderbuffer_samples_2" },
481 { SampleShadingRenderingCase::TARGET_RENDERBUFFER, 4, "multisample_renderbuffer_samples_4" },
482 { SampleShadingRenderingCase::TARGET_RENDERBUFFER, 8, "multisample_renderbuffer_samples_8" },
483 { SampleShadingRenderingCase::TARGET_RENDERBUFFER, 16, "multisample_renderbuffer_samples_16" },
484 };
485
486 for (int ndx = 0; ndx < DE_LENGTH_OF_ARRAY(targets); ++ndx)
487 {
488 minSamplesGroup->addChild(new SampleShadingRenderingCase(m_context, (std::string(targets[ndx].name) + "_color").c_str(), "Test multiple samples per pixel with color", targets[ndx].target, targets[ndx].numSamples, SampleShadingRenderingCase::TEST_COLOR));
489 minSamplesGroup->addChild(new SampleShadingRenderingCase(m_context, (std::string(targets[ndx].name) + "_discard").c_str(), "Test multiple samples per pixel with", targets[ndx].target, targets[ndx].numSamples, SampleShadingRenderingCase::TEST_DISCARD));
490 }
491 }
492 }
493
494 } // Functional
495 } // gles31
496 } // deqp
497