1 /*-------------------------------------------------------------------------
2 * drawElements Quality Program OpenGL ES 2.0 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 Texture access function tests.
22 *//*--------------------------------------------------------------------*/
23
24 #include "es2fShaderTextureFunctionTests.hpp"
25 #include "glsShaderRenderCase.hpp"
26 #include "glsShaderLibrary.hpp"
27 #include "glsTextureTestUtil.hpp"
28 #include "gluTexture.hpp"
29 #include "gluTextureUtil.hpp"
30 #include "tcuTextureUtil.hpp"
31 #include "tcuMatrix.hpp"
32 #include "tcuMatrixUtil.hpp"
33
34 #include <sstream>
35
36 #include "glwEnums.hpp"
37 #include "glwFunctions.hpp"
38
39 namespace deqp
40 {
41 namespace gles2
42 {
43 namespace Functional
44 {
45
46 namespace
47 {
48
49 enum Function
50 {
51 FUNCTION_TEXTURE = 0, //!< texture(), textureOffset()
52 FUNCTION_TEXTUREPROJ, //!< textureProj(), textureProjOffset()
53 FUNCTION_TEXTUREPROJ3, //!< textureProj(sampler2D, vec3)
54 FUNCTION_TEXTURELOD, // ...
55 FUNCTION_TEXTUREPROJLOD,
56 FUNCTION_TEXTUREPROJLOD3, //!< textureProjLod(sampler2D, vec3)
57
58 FUNCTION_LAST
59 };
60
functionHasProj(Function function)61 inline bool functionHasProj (Function function)
62 {
63 return function == FUNCTION_TEXTUREPROJ ||
64 function == FUNCTION_TEXTUREPROJ3 ||
65 function == FUNCTION_TEXTUREPROJLOD ||
66 function == FUNCTION_TEXTUREPROJLOD3;
67 }
68
functionHasLod(Function function)69 inline bool functionHasLod (Function function)
70 {
71 return function == FUNCTION_TEXTURELOD ||
72 function == FUNCTION_TEXTUREPROJLOD ||
73 function == FUNCTION_TEXTUREPROJLOD3;
74 }
75
76 struct TextureLookupSpec
77 {
78 Function function;
79
80 tcu::Vec4 minCoord;
81 tcu::Vec4 maxCoord;
82
83 // Bias
84 bool useBias;
85
86 // Bias or Lod for *Lod* functions
87 float minLodBias;
88 float maxLodBias;
89
TextureLookupSpecdeqp::gles2::Functional::__anonaaf110620111::TextureLookupSpec90 TextureLookupSpec (void)
91 : function (FUNCTION_LAST)
92 , minCoord (0.0f)
93 , maxCoord (1.0f)
94 , useBias (false)
95 , minLodBias (0.0f)
96 , maxLodBias (0.0f)
97 {
98 }
99
TextureLookupSpecdeqp::gles2::Functional::__anonaaf110620111::TextureLookupSpec100 TextureLookupSpec (Function function_,
101 const tcu::Vec4& minCoord_,
102 const tcu::Vec4& maxCoord_,
103 bool useBias_,
104 float minLodBias_,
105 float maxLodBias_)
106 : function (function_)
107 , minCoord (minCoord_)
108 , maxCoord (maxCoord_)
109 , useBias (useBias_)
110 , minLodBias (minLodBias_)
111 , maxLodBias (maxLodBias_)
112 {
113 }
114 };
115
116 enum TextureType
117 {
118 TEXTURETYPE_2D,
119 TEXTURETYPE_CUBE_MAP,
120
121 TEXTURETYPE_LAST
122 };
123
124 struct TextureSpec
125 {
126 TextureType type; //!< Texture type (2D, cubemap, ...)
127 deUint32 format;
128 deUint32 dataType;
129 int width;
130 int height;
131 int numLevels;
132 tcu::Sampler sampler;
133
TextureSpecdeqp::gles2::Functional::__anonaaf110620111::TextureSpec134 TextureSpec (void)
135 : type (TEXTURETYPE_LAST)
136 , format (GL_NONE)
137 , dataType (GL_NONE)
138 , width (0)
139 , height (0)
140 , numLevels (0)
141 {
142 }
143
TextureSpecdeqp::gles2::Functional::__anonaaf110620111::TextureSpec144 TextureSpec (TextureType type_,
145 deUint32 format_,
146 deUint32 dataType_,
147 int width_,
148 int height_,
149 int numLevels_,
150 const tcu::Sampler& sampler_)
151 : type (type_)
152 , format (format_)
153 , dataType (dataType_)
154 , width (width_)
155 , height (height_)
156 , numLevels (numLevels_)
157 , sampler (sampler_)
158 {
159 }
160 };
161
162 struct TexLookupParams
163 {
164 float lod;
165 tcu::Vec4 scale;
166 tcu::Vec4 bias;
167
TexLookupParamsdeqp::gles2::Functional::__anonaaf110620111::TexLookupParams168 TexLookupParams (void)
169 : lod (0.0f)
170 , scale (1.0f)
171 , bias (0.0f)
172 {
173 }
174 };
175
176 } // anonymous
177
178 using tcu::Vec2;
179 using tcu::Vec3;
180 using tcu::Vec4;
181 using tcu::IVec2;
182 using tcu::IVec3;
183 using tcu::IVec4;
184
185 typedef void (*TexEvalFunc) (gls::ShaderEvalContext& c, const TexLookupParams& lookupParams);
186
texture2D(const gls::ShaderEvalContext & c,float s,float t,float lod)187 inline Vec4 texture2D (const gls::ShaderEvalContext& c, float s, float t, float lod) { return c.textures[0].tex2D->sample(c.textures[0].sampler, s, t, lod); }
textureCube(const gls::ShaderEvalContext & c,float s,float t,float r,float lod)188 inline Vec4 textureCube (const gls::ShaderEvalContext& c, float s, float t, float r, float lod) { return c.textures[0].texCube->sample(c.textures[0].sampler, s, t, r, lod); }
189
190 // Eval functions.
evalTexture2D(gls::ShaderEvalContext & c,const TexLookupParams & p)191 static void evalTexture2D (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x(), c.in[0].y(), p.lod)*p.scale + p.bias; }
evalTextureCube(gls::ShaderEvalContext & c,const TexLookupParams & p)192 static void evalTextureCube (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = textureCube(c, c.in[0].x(), c.in[0].y(), c.in[0].z(), p.lod)*p.scale + p.bias; }
193
evalTexture2DBias(gls::ShaderEvalContext & c,const TexLookupParams & p)194 static void evalTexture2DBias (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x(), c.in[0].y(), p.lod+c.in[1].x())*p.scale + p.bias; }
evalTextureCubeBias(gls::ShaderEvalContext & c,const TexLookupParams & p)195 static void evalTextureCubeBias (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = textureCube(c, c.in[0].x(), c.in[0].y(), c.in[0].z(), p.lod+c.in[1].x())*p.scale + p.bias; }
196
evalTexture2DProj3(gls::ShaderEvalContext & c,const TexLookupParams & p)197 static void evalTexture2DProj3 (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].z(), c.in[0].y()/c.in[0].z(), p.lod)*p.scale + p.bias; }
evalTexture2DProj3Bias(gls::ShaderEvalContext & c,const TexLookupParams & p)198 static void evalTexture2DProj3Bias (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].z(), c.in[0].y()/c.in[0].z(), p.lod+c.in[1].x())*p.scale + p.bias; }
evalTexture2DProj(gls::ShaderEvalContext & c,const TexLookupParams & p)199 static void evalTexture2DProj (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].w(), c.in[0].y()/c.in[0].w(), p.lod)*p.scale + p.bias; }
evalTexture2DProjBias(gls::ShaderEvalContext & c,const TexLookupParams & p)200 static void evalTexture2DProjBias (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].w(), c.in[0].y()/c.in[0].w(), p.lod+c.in[1].x())*p.scale + p.bias; }
201
evalTexture2DLod(gls::ShaderEvalContext & c,const TexLookupParams & p)202 static void evalTexture2DLod (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x(), c.in[0].y(), c.in[1].x())*p.scale + p.bias; }
evalTextureCubeLod(gls::ShaderEvalContext & c,const TexLookupParams & p)203 static void evalTextureCubeLod (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = textureCube(c, c.in[0].x(), c.in[0].y(), c.in[0].z(), c.in[1].x())*p.scale + p.bias; }
204
evalTexture2DProjLod3(gls::ShaderEvalContext & c,const TexLookupParams & p)205 static void evalTexture2DProjLod3 (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].z(), c.in[0].y()/c.in[0].z(), c.in[1].x())*p.scale + p.bias; }
evalTexture2DProjLod(gls::ShaderEvalContext & c,const TexLookupParams & p)206 static void evalTexture2DProjLod (gls::ShaderEvalContext& c, const TexLookupParams& p) { c.color = texture2D(c, c.in[0].x()/c.in[0].w(), c.in[0].y()/c.in[0].w(), c.in[1].x())*p.scale + p.bias; }
207
208 class TexLookupEvaluator : public gls::ShaderEvaluator
209 {
210 public:
TexLookupEvaluator(TexEvalFunc evalFunc,const TexLookupParams & lookupParams)211 TexLookupEvaluator (TexEvalFunc evalFunc, const TexLookupParams& lookupParams) : m_evalFunc(evalFunc), m_lookupParams(lookupParams) {}
212
evaluate(gls::ShaderEvalContext & ctx)213 virtual void evaluate (gls::ShaderEvalContext& ctx) { m_evalFunc(ctx, m_lookupParams); }
214
215 private:
216 TexEvalFunc m_evalFunc;
217 const TexLookupParams& m_lookupParams;
218 };
219
220 class ShaderTextureFunctionCase : public gls::ShaderRenderCase
221 {
222 public:
223 ShaderTextureFunctionCase (Context& context, const char* name, const char* desc, const TextureLookupSpec& lookup, const TextureSpec& texture, TexEvalFunc evalFunc, bool isVertexCase);
224 ~ShaderTextureFunctionCase (void);
225
226 void init (void);
227 void deinit (void);
228
229 protected:
230 void setupUniforms (int programID, const tcu::Vec4& constCoords);
231
232 private:
233 void initTexture (void);
234 void initShaderSources (void);
235
236 TextureLookupSpec m_lookupSpec;
237 TextureSpec m_textureSpec;
238
239 TexLookupParams m_lookupParams;
240 TexLookupEvaluator m_evaluator;
241
242 glu::Texture2D* m_texture2D;
243 glu::TextureCube* m_textureCube;
244 };
245
ShaderTextureFunctionCase(Context & context,const char * name,const char * desc,const TextureLookupSpec & lookup,const TextureSpec & texture,TexEvalFunc evalFunc,bool isVertexCase)246 ShaderTextureFunctionCase::ShaderTextureFunctionCase (Context& context, const char* name, const char* desc, const TextureLookupSpec& lookup, const TextureSpec& texture, TexEvalFunc evalFunc, bool isVertexCase)
247 : gls::ShaderRenderCase(context.getTestContext(), context.getRenderContext(), context.getContextInfo(), name, desc, isVertexCase, m_evaluator)
248 , m_lookupSpec (lookup)
249 , m_textureSpec (texture)
250 , m_evaluator (evalFunc, m_lookupParams)
251 , m_texture2D (DE_NULL)
252 , m_textureCube (DE_NULL)
253 {
254 }
255
~ShaderTextureFunctionCase(void)256 ShaderTextureFunctionCase::~ShaderTextureFunctionCase (void)
257 {
258 delete m_texture2D;
259 delete m_textureCube;
260 }
261
init(void)262 void ShaderTextureFunctionCase::init (void)
263 {
264 if (m_isVertexCase)
265 {
266 const glw::Functions& gl = m_renderCtx.getFunctions();
267 int numVertexUnits = 0;
268 gl.getIntegerv(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, &numVertexUnits);
269 if (numVertexUnits < 1)
270 throw tcu::NotSupportedError("Vertex shader texture access is not supported");
271 }
272
273 {
274 // Base coord scale & bias
275 Vec4 s = m_lookupSpec.maxCoord-m_lookupSpec.minCoord;
276 Vec4 b = m_lookupSpec.minCoord;
277
278 float baseCoordTrans[] =
279 {
280 s.x(), 0.0f, 0.f, b.x(),
281 0.f, s.y(), 0.f, b.y(),
282 s.z()/2.f, -s.z()/2.f, 0.f, s.z()/2.f + b.z(),
283 -s.w()/2.f, s.w()/2.f, 0.f, s.w()/2.f + b.w()
284 };
285
286 m_userAttribTransforms.push_back(tcu::Mat4(baseCoordTrans));
287 }
288
289 if (functionHasLod(m_lookupSpec.function) || m_lookupSpec.useBias)
290 {
291 float s = m_lookupSpec.maxLodBias-m_lookupSpec.minLodBias;
292 float b = m_lookupSpec.minLodBias;
293 float lodCoordTrans[] =
294 {
295 s/2.0f, s/2.0f, 0.f, b,
296 0.0f, 0.0f, 0.0f, 0.0f,
297 0.0f, 0.0f, 0.0f, 0.0f,
298 0.0f, 0.0f, 0.0f, 0.0f
299 };
300
301 m_userAttribTransforms.push_back(tcu::Mat4(lodCoordTrans));
302 }
303
304 initShaderSources();
305 initTexture();
306
307 gls::ShaderRenderCase::init();
308 }
309
initTexture(void)310 void ShaderTextureFunctionCase::initTexture (void)
311 {
312 static const IVec4 texCubeSwz[] =
313 {
314 IVec4(0,0,1,1),
315 IVec4(1,1,0,0),
316 IVec4(0,1,0,1),
317 IVec4(1,0,1,0),
318 IVec4(0,1,1,0),
319 IVec4(1,0,0,1)
320 };
321 DE_STATIC_ASSERT(DE_LENGTH_OF_ARRAY(texCubeSwz) == tcu::CUBEFACE_LAST);
322
323 tcu::TextureFormat texFmt = glu::mapGLTransferFormat(m_textureSpec.format, m_textureSpec.dataType);
324 tcu::TextureFormatInfo fmtInfo = tcu::getTextureFormatInfo(texFmt);
325 tcu::IVec2 viewportSize = getViewportSize();
326 bool isProj = functionHasProj(m_lookupSpec.function);
327 float proj = isProj ? 1.0f/m_lookupSpec.minCoord[m_lookupSpec.function == FUNCTION_TEXTUREPROJ3 ? 2 : 3] : 1.0f;
328
329 switch (m_textureSpec.type)
330 {
331 case TEXTURETYPE_2D:
332 {
333 float cStep = 1.0f / (float)de::max(1, m_textureSpec.numLevels-1);
334 Vec4 cScale = fmtInfo.valueMax-fmtInfo.valueMin;
335 Vec4 cBias = fmtInfo.valueMin;
336 int baseCellSize = de::min(m_textureSpec.width/4, m_textureSpec.height/4);
337
338 m_texture2D = new glu::Texture2D(m_renderCtx, m_textureSpec.format, m_textureSpec.dataType, m_textureSpec.width, m_textureSpec.height);
339 for (int level = 0; level < m_textureSpec.numLevels; level++)
340 {
341 float fA = float(level)*cStep;
342 float fB = 1.0f-fA;
343 Vec4 colorA = cBias + cScale*Vec4(fA, fB, fA, fB);
344 Vec4 colorB = cBias + cScale*Vec4(fB, fA, fB, fA);
345
346 m_texture2D->getRefTexture().allocLevel(level);
347 tcu::fillWithGrid(m_texture2D->getRefTexture().getLevel(level), de::max(1, baseCellSize>>level), colorA, colorB);
348 }
349 m_texture2D->upload();
350
351 // Compute LOD.
352 float dudx = (m_lookupSpec.maxCoord[0]-m_lookupSpec.minCoord[0])*proj*(float)m_textureSpec.width / (float)viewportSize[0];
353 float dvdy = (m_lookupSpec.maxCoord[1]-m_lookupSpec.minCoord[1])*proj*(float)m_textureSpec.height / (float)viewportSize[1];
354 m_lookupParams.lod = glu::TextureTestUtil::computeLodFromDerivates(glu::TextureTestUtil::LODMODE_EXACT, dudx, 0.0f, 0.0f, dvdy);
355
356 // Append to texture list.
357 m_textures.push_back(gls::TextureBinding(m_texture2D, m_textureSpec.sampler));
358 break;
359 }
360
361 case TEXTURETYPE_CUBE_MAP:
362 {
363 float cStep = 1.0f / (float)de::max(1, m_textureSpec.numLevels-1);
364 Vec4 cScale = fmtInfo.valueMax-fmtInfo.valueMin;
365 Vec4 cBias = fmtInfo.valueMin;
366 int baseCellSize = de::min(m_textureSpec.width/4, m_textureSpec.height/4);
367
368 DE_ASSERT(m_textureSpec.width == m_textureSpec.height);
369 m_textureCube = new glu::TextureCube(m_renderCtx, m_textureSpec.format, m_textureSpec.dataType, m_textureSpec.width);
370 for (int level = 0; level < m_textureSpec.numLevels; level++)
371 {
372 float fA = float(level)*cStep;
373 float fB = 1.0f-fA;
374 Vec2 f (fA, fB);
375
376 for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
377 {
378 const IVec4& swzA = texCubeSwz[face];
379 IVec4 swzB = 1-swzA;
380 Vec4 colorA = cBias + cScale*f.swizzle(swzA[0], swzA[1], swzA[2], swzA[3]);
381 Vec4 colorB = cBias + cScale*f.swizzle(swzB[0], swzB[1], swzB[2], swzB[3]);
382
383 m_textureCube->getRefTexture().allocLevel((tcu::CubeFace)face, level);
384 tcu::fillWithGrid(m_textureCube->getRefTexture().getLevelFace(level, (tcu::CubeFace)face), de::max(1, baseCellSize>>level), colorA, colorB);
385 }
386 }
387 m_textureCube->upload();
388
389 // Compute LOD \note Assumes that only single side is accessed and R is constant major axis.
390 DE_ASSERT(de::abs(m_lookupSpec.minCoord[2] - m_lookupSpec.maxCoord[2]) < 0.005);
391 DE_ASSERT(de::abs(m_lookupSpec.minCoord[0]) < de::abs(m_lookupSpec.minCoord[2]) && de::abs(m_lookupSpec.maxCoord[0]) < de::abs(m_lookupSpec.minCoord[2]));
392 DE_ASSERT(de::abs(m_lookupSpec.minCoord[1]) < de::abs(m_lookupSpec.minCoord[2]) && de::abs(m_lookupSpec.maxCoord[1]) < de::abs(m_lookupSpec.minCoord[2]));
393
394 tcu::CubeFaceFloatCoords c00 = tcu::getCubeFaceCoords(Vec3(m_lookupSpec.minCoord[0]*proj, m_lookupSpec.minCoord[1]*proj, m_lookupSpec.minCoord[2]*proj));
395 tcu::CubeFaceFloatCoords c10 = tcu::getCubeFaceCoords(Vec3(m_lookupSpec.maxCoord[0]*proj, m_lookupSpec.minCoord[1]*proj, m_lookupSpec.minCoord[2]*proj));
396 tcu::CubeFaceFloatCoords c01 = tcu::getCubeFaceCoords(Vec3(m_lookupSpec.minCoord[0]*proj, m_lookupSpec.maxCoord[1]*proj, m_lookupSpec.minCoord[2]*proj));
397 float dudx = (c10.s - c00.s)*(float)m_textureSpec.width / (float)viewportSize[0];
398 float dvdy = (c01.t - c00.t)*(float)m_textureSpec.height / (float)viewportSize[1];
399
400 m_lookupParams.lod = glu::TextureTestUtil::computeLodFromDerivates(glu::TextureTestUtil::LODMODE_EXACT, dudx, 0.0f, 0.0f, dvdy);
401
402 m_textures.push_back(gls::TextureBinding(m_textureCube, m_textureSpec.sampler));
403 break;
404 }
405
406 default:
407 DE_ASSERT(DE_FALSE);
408 }
409
410 // Set lookup scale & bias
411 m_lookupParams.scale = fmtInfo.lookupScale;
412 m_lookupParams.bias = fmtInfo.lookupBias;
413 }
414
initShaderSources(void)415 void ShaderTextureFunctionCase::initShaderSources (void)
416 {
417 Function function = m_lookupSpec.function;
418 bool isVtxCase = m_isVertexCase;
419 bool isProj = functionHasProj(function);
420 bool is2DProj4 = m_textureSpec.type == TEXTURETYPE_2D && (function == FUNCTION_TEXTUREPROJ || function == FUNCTION_TEXTUREPROJLOD);
421 bool hasLodBias = functionHasLod(m_lookupSpec.function) || m_lookupSpec.useBias;
422 int texCoordComps = m_textureSpec.type == TEXTURETYPE_2D ? 2 : 3;
423 int extraCoordComps = isProj ? (is2DProj4 ? 2 : 1) : 0;
424 glu::DataType coordType = glu::getDataTypeFloatVec(texCoordComps+extraCoordComps);
425 glu::Precision coordPrec = glu::PRECISION_MEDIUMP;
426 const char* coordTypeName = glu::getDataTypeName(coordType);
427 const char* coordPrecName = glu::getPrecisionName(coordPrec);
428 tcu::TextureFormat texFmt = glu::mapGLTransferFormat(m_textureSpec.format, m_textureSpec.dataType);
429 glu::DataType samplerType = glu::TYPE_LAST;
430 const char* baseFuncName = m_textureSpec.type == TEXTURETYPE_2D ? "texture2D" : "textureCube";
431 const char* funcExt = DE_NULL;
432
433 switch (m_textureSpec.type)
434 {
435 case TEXTURETYPE_2D: samplerType = glu::getSampler2DType(texFmt); break;
436 case TEXTURETYPE_CUBE_MAP: samplerType = glu::getSamplerCubeType(texFmt); break;
437 default:
438 DE_ASSERT(DE_FALSE);
439 }
440
441 switch (m_lookupSpec.function)
442 {
443 case FUNCTION_TEXTURE: funcExt = ""; break;
444 case FUNCTION_TEXTUREPROJ: funcExt = "Proj"; break;
445 case FUNCTION_TEXTUREPROJ3: funcExt = "Proj"; break;
446 case FUNCTION_TEXTURELOD: funcExt = "Lod"; break;
447 case FUNCTION_TEXTUREPROJLOD: funcExt = "ProjLod"; break;
448 case FUNCTION_TEXTUREPROJLOD3: funcExt = "ProjLod"; break;
449 default:
450 DE_ASSERT(DE_FALSE);
451 }
452
453 std::ostringstream vert;
454 std::ostringstream frag;
455 std::ostringstream& op = isVtxCase ? vert : frag;
456
457 vert << "attribute highp vec4 a_position;\n"
458 << "attribute " << coordPrecName << " " << coordTypeName << " a_in0;\n";
459
460 if (hasLodBias)
461 vert << "attribute " << coordPrecName << " float a_in1;\n";
462
463 if (isVtxCase)
464 {
465 vert << "varying mediump vec4 v_color;\n";
466 frag << "varying mediump vec4 v_color;\n";
467 }
468 else
469 {
470 vert << "varying " << coordPrecName << " " << coordTypeName << " v_texCoord;\n";
471 frag << "varying " << coordPrecName << " " << coordTypeName << " v_texCoord;\n";
472
473 if (hasLodBias)
474 {
475 vert << "varying " << coordPrecName << " float v_lodBias;\n";
476 frag << "varying " << coordPrecName << " float v_lodBias;\n";
477 }
478 }
479
480 // Uniforms
481 op << "uniform lowp " << glu::getDataTypeName(samplerType) << " u_sampler;\n";
482
483 vert << "\nvoid main()\n{\n"
484 << "\tgl_Position = a_position;\n";
485 frag << "\nvoid main()\n{\n";
486
487 if (isVtxCase)
488 vert << "\tv_color = ";
489 else
490 frag << "\tgl_FragColor = ";
491
492 // Op.
493 {
494 const char* texCoord = isVtxCase ? "a_in0" : "v_texCoord";
495 const char* lodBias = isVtxCase ? "a_in1" : "v_lodBias";
496
497 op << baseFuncName << funcExt;
498 op << "(u_sampler, " << texCoord;
499
500 if (functionHasLod(function) || m_lookupSpec.useBias)
501 op << ", " << lodBias;
502
503 op << ");\n";
504 }
505
506 if (isVtxCase)
507 frag << "\tgl_FragColor = v_color;\n";
508 else
509 {
510 vert << "\tv_texCoord = a_in0;\n";
511
512 if (hasLodBias)
513 vert << "\tv_lodBias = a_in1;\n";
514 }
515
516 vert << "}\n";
517 frag << "}\n";
518
519 m_vertShaderSource = vert.str();
520 m_fragShaderSource = frag.str();
521 }
522
deinit(void)523 void ShaderTextureFunctionCase::deinit (void)
524 {
525 gls::ShaderRenderCase::deinit();
526
527 delete m_texture2D;
528 delete m_textureCube;
529
530 m_texture2D = DE_NULL;
531 m_textureCube = DE_NULL;
532 }
533
setupUniforms(int programID,const tcu::Vec4 &)534 void ShaderTextureFunctionCase::setupUniforms (int programID, const tcu::Vec4&)
535 {
536 const glw::Functions& gl = m_renderCtx.getFunctions();
537 gl.uniform1i(gl.getUniformLocation(programID, "u_sampler"), 0);
538 }
539
ShaderTextureFunctionTests(Context & context)540 ShaderTextureFunctionTests::ShaderTextureFunctionTests (Context& context)
541 : TestCaseGroup(context, "texture_functions", "Texture Access Function Tests")
542 {
543 }
544
~ShaderTextureFunctionTests(void)545 ShaderTextureFunctionTests::~ShaderTextureFunctionTests (void)
546 {
547 }
548
549 struct TexFuncCaseSpec
550 {
551 const char* name;
552 TextureLookupSpec lookupSpec;
553 TextureSpec texSpec;
554 TexEvalFunc evalFunc;
555 };
556
557 #define CASE_SPEC(NAME, FUNC, MINCOORD, MAXCOORD, USEBIAS, MINLOD, MAXLOD, TEXSPEC, EVALFUNC) \
558 { #NAME, TextureLookupSpec(FUNC, MINCOORD, MAXCOORD, USEBIAS, MINLOD, MAXLOD), TEXSPEC, EVALFUNC }
559
createCaseGroup(TestCaseGroup * parent,const char * groupName,const char * groupDesc,const TexFuncCaseSpec * cases,int numCases,bool isVertex)560 static void createCaseGroup (TestCaseGroup* parent, const char* groupName, const char* groupDesc, const TexFuncCaseSpec* cases, int numCases, bool isVertex)
561 {
562 tcu::TestCaseGroup* group = new tcu::TestCaseGroup(parent->getTestContext(), groupName, groupDesc);
563 parent->addChild(group);
564
565 for (int ndx = 0; ndx < numCases; ndx++)
566 group->addChild(new ShaderTextureFunctionCase(parent->getContext(), cases[ndx].name, "", cases[ndx].lookupSpec, cases[ndx].texSpec, cases[ndx].evalFunc, isVertex));
567 }
568
init(void)569 void ShaderTextureFunctionTests::init (void)
570 {
571 const glw::Functions& gl = m_context.getRenderContext().getFunctions();
572
573 // Samplers
574 static const tcu::Sampler samplerLinearNoMipmap (tcu::Sampler::REPEAT_GL, tcu::Sampler::REPEAT_GL, tcu::Sampler::REPEAT_GL,
575 tcu::Sampler::LINEAR, tcu::Sampler::LINEAR);
576 static tcu::Sampler samplerLinearMipmap (tcu::Sampler::REPEAT_GL, tcu::Sampler::REPEAT_GL, tcu::Sampler::REPEAT_GL,
577 tcu::Sampler::LINEAR_MIPMAP_NEAREST, tcu::Sampler::LINEAR);
578
579 // GL_MAJOR_VERSION query does not exist on GLES2
580 // so succeeding query implies GLES3+ hardware.
581 glw::GLint majorVersion = 0;
582 gl.getIntegerv(GL_MAJOR_VERSION, &majorVersion);
583 samplerLinearMipmap.seamlessCubeMap = (gl.getError() == GL_NO_ERROR);
584
585 // Default textures.
586 // Type Format DataType W H L Sampler
587 static const TextureSpec tex2D (TEXTURETYPE_2D, GL_RGBA, GL_UNSIGNED_BYTE, 256, 256, 1, samplerLinearNoMipmap);
588 static const TextureSpec tex2DMipmap (TEXTURETYPE_2D, GL_RGBA, GL_UNSIGNED_BYTE, 256, 256, 9, samplerLinearMipmap);
589
590 static const TextureSpec texCube (TEXTURETYPE_CUBE_MAP, GL_RGBA, GL_UNSIGNED_BYTE, 256, 256, 1, samplerLinearNoMipmap);
591 static const TextureSpec texCubeMipmap (TEXTURETYPE_CUBE_MAP, GL_RGBA, GL_UNSIGNED_BYTE, 256, 256, 9, samplerLinearMipmap);
592
593 // Vertex cases
594 static const TexFuncCaseSpec vertexCases[] =
595 {
596 // Name Function MinCoord MaxCoord Bias? MinLod MaxLod Texture EvalFunc
597 CASE_SPEC(texture2d, FUNCTION_TEXTURE, Vec4(-0.2f, -0.4f, 0.0f, 0.0f), Vec4( 1.5f, 2.3f, 0.0f, 0.0f), false, 0.0f, 0.0f, tex2D, evalTexture2D),
598 // CASE_SPEC(texture2d_bias, FUNCTION_TEXTURE, Vec4(-0.2f, -0.4f, 0.0f, 0.0f), Vec4( 1.5f, 2.3f, 0.0f, 0.0f), true, -2.0f, 2.0f, tex2D, evalTexture2DBias),
599 CASE_SPEC(texture2dproj_vec3, FUNCTION_TEXTUREPROJ3, Vec4(-0.3f, -0.6f, 1.5f, 0.0f), Vec4(2.25f, 3.45f, 1.5f, 0.0f), false, 0.0f, 0.0f, tex2D, evalTexture2DProj3),
600 CASE_SPEC(texture2dproj_vec4, FUNCTION_TEXTUREPROJ, Vec4(-0.3f, -0.6f, 0.0f, 1.5f), Vec4(2.25f, 3.45f, 0.0f, 1.5f), false, 0.0f, 0.0f, tex2D, evalTexture2DProj),
601 CASE_SPEC(texture2dlod, FUNCTION_TEXTURELOD, Vec4(-0.2f, -0.4f, 0.0f, 0.0f), Vec4( 1.5f, 2.3f, 0.0f, 0.0f), false, -1.0f, 9.0f, tex2DMipmap, evalTexture2DLod),
602 // CASE_SPEC(texture2dproj_vec3_bias, FUNCTION_TEXTUREPROJ3, Vec4(-0.3f, -0.6f, 1.5f, 0.0f), Vec4(2.25f, 3.45f, 1.5f, 0.0f), true, -2.0f, 2.0f, tex2D, evalTexture2DProj3Bias),
603 // CASE_SPEC(texture2dproj_vec4_bias, FUNCTION_TEXTUREPROJ, Vec4(-0.3f, -0.6f, 0.0f, 1.5f), Vec4(2.25f, 3.45f, 0.0f, 1.5f), true, -2.0f, 2.0f, tex2D, evalTexture2DProjBias),
604 CASE_SPEC(texture2dprojlod_vec3, FUNCTION_TEXTUREPROJLOD3, Vec4(-0.3f, -0.6f, 1.5f, 0.0f), Vec4(2.25f, 3.45f, 1.5f, 0.0f), false, -1.0f, 9.0f, tex2D, evalTexture2DProjLod3),
605 CASE_SPEC(texture2dprojlod_vec4, FUNCTION_TEXTUREPROJLOD, Vec4(-0.3f, -0.6f, 0.0f, 1.5f), Vec4(2.25f, 3.45f, 0.0f, 1.5f), false, -1.0f, 9.0f, tex2D, evalTexture2DProjLod),
606 CASE_SPEC(texturecube, FUNCTION_TEXTURE, Vec4(-1.0f, -1.0f, 1.01f, 0.0f), Vec4( 1.0f, 1.0f, 1.01f, 0.0f), false, 0.0f, 0.0f, texCube, evalTextureCube),
607 // CASE_SPEC(texturecube_bias, FUNCTION_TEXTURE, Vec4(-1.0f, -1.0f, -1.01f, 0.0f), Vec4( 1.0f, 1.0f, -1.01f, 0.0f), true, -2.0f, 2.0f, texCube, evalTextureCubeBias),
608 CASE_SPEC(texturecubelod, FUNCTION_TEXTURELOD, Vec4(-1.0f, -1.0f, 1.01f, 0.0f), Vec4( 1.0f, 1.0f, 1.01f, 0.0f), false, -1.0f, 9.0f, texCubeMipmap, evalTextureCubeLod),
609 };
610 createCaseGroup(this, "vertex", "Vertex Shader Texture Lookups", &vertexCases[0], DE_LENGTH_OF_ARRAY(vertexCases), true);
611
612 // Fragment cases
613 static const TexFuncCaseSpec fragmentCases[] =
614 {
615 // Name Function MinCoord MaxCoord Bias? MinLod MaxLod Texture EvalFunc
616 CASE_SPEC(texture2d, FUNCTION_TEXTURE, Vec4(-0.2f, -0.4f, 0.0f, 0.0f), Vec4( 1.5f, 2.3f, 0.0f, 0.0f), false, 0.0f, 0.0f, tex2DMipmap, evalTexture2D),
617 CASE_SPEC(texture2d_bias, FUNCTION_TEXTURE, Vec4(-0.2f, -0.4f, 0.0f, 0.0f), Vec4( 1.5f, 2.3f, 0.0f, 0.0f), true, -2.0f, 2.0f, tex2DMipmap, evalTexture2DBias),
618 CASE_SPEC(texture2dproj_vec3, FUNCTION_TEXTUREPROJ3, Vec4(-0.3f, -0.6f, 1.5f, 0.0f), Vec4(2.25f, 3.45f, 1.5f, 0.0f), false, 0.0f, 0.0f, tex2DMipmap, evalTexture2DProj3),
619 CASE_SPEC(texture2dproj_vec4, FUNCTION_TEXTUREPROJ, Vec4(-0.3f, -0.6f, 0.0f, 1.5f), Vec4(2.25f, 3.45f, 0.0f, 1.5f), false, 0.0f, 0.0f, tex2DMipmap, evalTexture2DProj),
620 CASE_SPEC(texture2dproj_vec3_bias, FUNCTION_TEXTUREPROJ3, Vec4(-0.3f, -0.6f, 1.5f, 0.0f), Vec4(2.25f, 3.45f, 1.5f, 0.0f), true, -2.0f, 2.0f, tex2DMipmap, evalTexture2DProj3Bias),
621 CASE_SPEC(texture2dproj_vec4_bias, FUNCTION_TEXTUREPROJ, Vec4(-0.3f, -0.6f, 0.0f, 1.5f), Vec4(2.25f, 3.45f, 0.0f, 1.5f), true, -2.0f, 2.0f, tex2DMipmap, evalTexture2DProjBias),
622 CASE_SPEC(texturecube, FUNCTION_TEXTURE, Vec4(-1.0f, -1.0f, 1.01f, 0.0f), Vec4( 1.0f, 1.0f, 1.01f, 0.0f), false, 0.0f, 0.0f, texCubeMipmap, evalTextureCube),
623 CASE_SPEC(texturecube_bias, FUNCTION_TEXTURE, Vec4(-1.0f, -1.0f, -1.01f, 0.0f), Vec4( 1.0f, 1.0f, -1.01f, 0.0f), true, -2.0f, 2.0f, texCubeMipmap, evalTextureCubeBias)
624 };
625 createCaseGroup(this, "fragment", "Fragment Shader Texture Lookups", &fragmentCases[0], DE_LENGTH_OF_ARRAY(fragmentCases), false);
626
627 // Negative cases.
628 {
629 gls::ShaderLibrary library(m_testCtx, m_context.getRenderContext(), m_context.getContextInfo());
630 std::vector<tcu::TestNode*> negativeCases = library.loadShaderFile("shaders/invalid_texture_functions.test");
631
632 tcu::TestCaseGroup* group = new tcu::TestCaseGroup(m_testCtx, "invalid", "Invalid texture function usage", negativeCases);
633 addChild(group);
634 }
635 }
636
637 } // Functional
638 } // gles3
639 } // deqp
640