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 Vertex texture tests.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "es2fVertexTextureTests.hpp"
25 #include "glsTextureTestUtil.hpp"
26 #include "gluTexture.hpp"
27 #include "gluPixelTransfer.hpp"
28 #include "gluTextureUtil.hpp"
29 #include "tcuVector.hpp"
30 #include "tcuMatrix.hpp"
31 #include "tcuTextureUtil.hpp"
32 #include "tcuImageCompare.hpp"
33 #include "deRandom.hpp"
34 #include "deString.h"
35 
36 #include <string>
37 #include <vector>
38 
39 #include <limits>
40 
41 #include "glw.h"
42 
43 using tcu::TestLog;
44 using tcu::Vec2;
45 using tcu::Vec3;
46 using tcu::Vec4;
47 using tcu::IVec2;
48 using tcu::IVec3;
49 using tcu::IVec4;
50 using tcu::Mat3;
51 using std::string;
52 using std::vector;
53 
54 namespace deqp
55 {
56 
57 using namespace gls::TextureTestUtil;
58 
59 using gls::TextureTestUtil::TEXTURETYPE_2D;
60 using gls::TextureTestUtil::TEXTURETYPE_CUBE;
61 
62 namespace gles2
63 {
64 namespace Functional
65 {
66 
67 // The 2D case draws four images.
68 static const int MAX_2D_RENDER_WIDTH		= 128*2;
69 static const int MAX_2D_RENDER_HEIGHT		= 128*2;
70 
71 // The cube map case draws four 3-by-2 image groups.
72 static const int MAX_CUBE_RENDER_WIDTH		= 28*2*3;
73 static const int MAX_CUBE_RENDER_HEIGHT		= 28*2*2;
74 
75 static const int GRID_SIZE_2D				= 127;
76 static const int GRID_SIZE_CUBE				= 63;
77 
78 // Helpers for making texture coordinates "safe", i.e. move them further from coordinate bounary.
79 
80 // Moves x towards the closest K+targetFraction, where K is an integer.
81 // E.g. moveTowardsFraction(x, 0.5f) moves x away from integer boundaries.
moveTowardsFraction(float x,float targetFraction)82 static inline float moveTowardsFraction (float x, float targetFraction)
83 {
84 	const float strictness = 0.5f;
85 	DE_ASSERT(0.0f < strictness && strictness <= 1.0f);
86 	DE_ASSERT(de::inBounds(targetFraction, 0.0f, 1.0f));
87 	const float y = x + 0.5f - targetFraction;
88 	return deFloatFloor(y) + deFloatFrac(y)*(1.0f-strictness) + strictness*0.5f - 0.5f + targetFraction;
89 }
90 
safeCoord(float raw,int scale,float fraction)91 static inline float safeCoord (float raw, int scale, float fraction)
92 {
93 	const float scaleFloat = (float)scale;
94 	return moveTowardsFraction(raw*scaleFloat, fraction) / scaleFloat;
95 }
96 
97 template <int Size>
safeCoords(const tcu::Vector<float,Size> & raw,const tcu::Vector<int,Size> & scale,const tcu::Vector<float,Size> & fraction)98 static inline tcu::Vector<float, Size> safeCoords (const tcu::Vector<float, Size>& raw, const tcu::Vector<int, Size>& scale, const tcu::Vector<float, Size>& fraction)
99 {
100 	tcu::Vector<float, Size> result;
101 	for (int i = 0; i < Size; i++)
102 		result[i] = safeCoord(raw[i], scale[i], fraction[i]);
103 	return result;
104 }
105 
safe2DTexCoords(const Vec2 & raw,const IVec2 & textureSize)106 static inline Vec2 safe2DTexCoords (const Vec2& raw, const IVec2& textureSize)
107 {
108 	return safeCoords(raw, textureSize, Vec2(0.5f));
109 }
110 
111 namespace
112 {
113 
114 struct Rect
115 {
Rectdeqp::gles2::Functional::__anonc2e8d0630111::Rect116 			Rect	(int x_, int y_, int w_, int h_) : x(x_), y(y_), w(w_), h(h_) {}
posdeqp::gles2::Functional::__anonc2e8d0630111::Rect117 	IVec2	pos		(void) const { return IVec2(x, y); }
sizedeqp::gles2::Functional::__anonc2e8d0630111::Rect118 	IVec2	size	(void) const { return IVec2(w, h); }
119 
120 	int		x;
121 	int		y;
122 	int		w;
123 	int		h;
124 };
125 
126 template <TextureType> struct TexTypeTcuClass;
127 template <> struct TexTypeTcuClass<TEXTURETYPE_2D>			{ typedef tcu::Texture2D		t; };
128 template <> struct TexTypeTcuClass<TEXTURETYPE_CUBE>		{ typedef tcu::TextureCube		t; };
129 
130 template <TextureType> struct TexTypeSizeDims;
131 template <> struct TexTypeSizeDims<TEXTURETYPE_2D>			{ enum { V = 2 }; };
132 template <> struct TexTypeSizeDims<TEXTURETYPE_CUBE>		{ enum { V = 2 }; };
133 
134 template <TextureType> struct TexTypeCoordDims;
135 template <> struct TexTypeCoordDims<TEXTURETYPE_2D>			{ enum { V = 2 }; };
136 template <> struct TexTypeCoordDims<TEXTURETYPE_CUBE>		{ enum { V = 3 }; };
137 
138 template <TextureType TexType> struct TexTypeSizeIVec		{ typedef tcu::Vector<int,		TexTypeSizeDims<TexType>::V>	t; };
139 template <TextureType TexType> struct TexTypeCoordVec		{ typedef tcu::Vector<float,	TexTypeCoordDims<TexType>::V>	t; };
140 
141 template <TextureType> struct TexTypeCoordParams;
142 
143 template <> struct
144 TexTypeCoordParams<TEXTURETYPE_2D>
145 {
146 	Vec2 scale;
147 	Vec2 bias;
148 
TexTypeCoordParamsdeqp::gles2::Functional::__anonc2e8d0630111::TexTypeCoordParams149 	TexTypeCoordParams (const Vec2& scale_, const Vec2& bias_) : scale(scale_), bias(bias_) {}
150 };
151 
152 template <> struct
153 TexTypeCoordParams<TEXTURETYPE_CUBE>
154 {
155 	Vec2			scale;
156 	Vec2			bias;
157 	tcu::CubeFace	face;
158 
TexTypeCoordParamsdeqp::gles2::Functional::__anonc2e8d0630111::TexTypeCoordParams159 	TexTypeCoordParams (const Vec2& scale_, const Vec2& bias_, tcu::CubeFace face_) : scale(scale_), bias(bias_), face(face_) {}
160 };
161 
162 /*--------------------------------------------------------------------*//*!
163  * \brief Quad grid class containing position and texture coordinate data.
164  *
165  * A quad grid of size S means a grid consisting of S*S quads (S rows and
166  * S columns). The quads are rectangles with main axis aligned sides, and
167  * each consists of two triangles. Note that although there are only
168  * (S+1)*(S+1) distinct vertex positions, there are S*S*4 distinct vertices
169  * because we want texture coordinates to be constant across the vertices
170  * of a quad (to avoid interpolation issues), and thus each quad needs its
171  * own 4 vertices.
172  *
173  * Pointers returned by get*Ptr() are suitable for gl calls such as
174  * glVertexAttribPointer() (for position and tex coord) or glDrawElements()
175  * (for indices).
176  *//*--------------------------------------------------------------------*/
177 template <TextureType TexType>
178 class PosTexCoordQuadGrid
179 {
180 private:
181 	enum { TEX_COORD_DIMS = TexTypeCoordDims <TexType>::V };
182 	typedef typename TexTypeCoordVec<TexType>::t	TexCoordVec;
183 	typedef typename TexTypeSizeIVec<TexType>::t	TexSizeIVec;
184 	typedef TexTypeCoordParams<TexType>				TexCoordParams;
185 
186 public:
187 							PosTexCoordQuadGrid		(int gridSize, const IVec2& renderSize, const TexSizeIVec& textureSize, const TexCoordParams& texCoordParams, bool useSafeTexCoords);
188 
getSize(void) const189 	int						getSize					(void) const { return m_gridSize; }
190 	Vec4					getQuadLDRU				(int col, int row) const; //!< Vec4(leftX, downY, rightX, upY)
191 	const TexCoordVec&		getQuadTexCoord			(int col, int row) const;
192 
getNumIndices(void) const193 	int						getNumIndices			(void) const { return m_gridSize*m_gridSize*3*2; }
getPositionPtr(void) const194 	const float*			getPositionPtr			(void) const { DE_STATIC_ASSERT(sizeof(Vec2) == 2*sizeof(float)); return (float*)&m_positions[0]; }
getTexCoordPtr(void) const195 	const float*			getTexCoordPtr			(void) const { DE_STATIC_ASSERT(sizeof(TexCoordVec) == TEX_COORD_DIMS*(int)sizeof(float)); return (float*)&m_texCoords[0]; }
getIndexPtr(void) const196 	const deUint16*			getIndexPtr				(void) const { return &m_indices[0]; }
197 
198 private:
199 	void					initializeTexCoords		(const TexSizeIVec& textureSize, const TexCoordParams& texCoordParams, bool useSafeTexCoords);
200 
201 	const int				m_gridSize;
202 	vector<Vec2>			m_positions;
203 	vector<TexCoordVec>		m_texCoords;
204 	vector<deUint16>		m_indices;
205 };
206 
207 template <TextureType TexType>
getQuadLDRU(int col,int row) const208 Vec4 PosTexCoordQuadGrid<TexType>::getQuadLDRU (int col, int row) const
209 {
210 	int ndx00 = (row*m_gridSize + col) * 4;
211 	int ndx11 = ndx00 + 3;
212 
213 	return Vec4(m_positions[ndx00].x(),
214 				m_positions[ndx00].y(),
215 				m_positions[ndx11].x(),
216 				m_positions[ndx11].y());
217 }
218 
219 template <TextureType TexType>
getQuadTexCoord(int col,int row) const220 const typename TexTypeCoordVec<TexType>::t& PosTexCoordQuadGrid<TexType>::getQuadTexCoord (int col, int row) const
221 {
222 	return m_texCoords[(row*m_gridSize + col) * 4];
223 }
224 
225 template <TextureType TexType>
PosTexCoordQuadGrid(int gridSize,const IVec2 & renderSize,const TexSizeIVec & textureSize,const TexCoordParams & texCoordParams,bool useSafeTexCoords)226 PosTexCoordQuadGrid<TexType>::PosTexCoordQuadGrid (int gridSize, const IVec2& renderSize, const TexSizeIVec& textureSize, const TexCoordParams& texCoordParams, bool useSafeTexCoords)
227 	: m_gridSize(gridSize)
228 {
229 	DE_ASSERT(m_gridSize > 0 && m_gridSize*m_gridSize <= (int)std::numeric_limits<deUint16>::max() + 1);
230 
231 	const float gridSizeFloat = (float)m_gridSize;
232 
233 	m_positions.reserve(m_gridSize*m_gridSize*4);
234 	m_indices.reserve(m_gridSize*m_gridSize*3*2);
235 
236 	for (int y = 0; y < m_gridSize; y++)
237 	for (int x = 0; x < m_gridSize; x++)
238 	{
239 		float fx0 = (float)(x+0) / gridSizeFloat;
240 		float fx1 = (float)(x+1) / gridSizeFloat;
241 		float fy0 = (float)(y+0) / gridSizeFloat;
242 		float fy1 = (float)(y+1) / gridSizeFloat;
243 
244 		Vec2 quadVertices[4] = { Vec2(fx0, fy0), Vec2(fx1, fy0), Vec2(fx0, fy1), Vec2(fx1, fy1) };
245 
246 		int firstNdx = (int)m_positions.size();
247 
248 		for (int i = 0; i < DE_LENGTH_OF_ARRAY(quadVertices); i++)
249 			m_positions.push_back(safeCoords(quadVertices[i], renderSize, Vec2(0.0f)) * 2.0f - 1.0f);
250 
251 		m_indices.push_back(firstNdx + 0);
252 		m_indices.push_back(firstNdx + 1);
253 		m_indices.push_back(firstNdx + 2);
254 
255 		m_indices.push_back(firstNdx + 1);
256 		m_indices.push_back(firstNdx + 3);
257 		m_indices.push_back(firstNdx + 2);
258 	}
259 
260 	m_texCoords.reserve(m_gridSize*m_gridSize*4);
261 	initializeTexCoords(textureSize, texCoordParams, useSafeTexCoords);
262 
263 	DE_ASSERT((int)m_positions.size() == m_gridSize*m_gridSize*4);
264 	DE_ASSERT((int)m_indices.size() == m_gridSize*m_gridSize*3*2);
265 	DE_ASSERT((int)m_texCoords.size() == m_gridSize*m_gridSize*4);
266 }
267 
268 template <>
initializeTexCoords(const IVec2 & textureSize,const TexCoordParams & texCoordParams,bool useSafeTexCoords)269 void PosTexCoordQuadGrid<TEXTURETYPE_2D>::initializeTexCoords (const IVec2& textureSize, const TexCoordParams& texCoordParams, bool useSafeTexCoords)
270 {
271 	DE_ASSERT(m_texCoords.empty());
272 
273 	const float gridSizeFloat = (float)m_gridSize;
274 
275 	for (int y = 0; y < m_gridSize; y++)
276 	for (int x = 0; x < m_gridSize; x++)
277 	{
278 		Vec2 rawCoord = Vec2((float)x / gridSizeFloat, (float)y / gridSizeFloat) * texCoordParams.scale + texCoordParams.bias;
279 
280 		for (int i = 0; i < 4; i++)
281 			m_texCoords.push_back(useSafeTexCoords ? safe2DTexCoords(rawCoord, textureSize) : rawCoord);
282 	}
283 }
284 
285 template <>
initializeTexCoords(const IVec2 & textureSize,const TexCoordParams & texCoordParams,bool useSafeTexCoords)286 void PosTexCoordQuadGrid<TEXTURETYPE_CUBE>::initializeTexCoords (const IVec2& textureSize, const TexCoordParams& texCoordParams, bool useSafeTexCoords)
287 {
288 	DE_ASSERT(m_texCoords.empty());
289 
290 	const float		gridSizeFloat	= (float)m_gridSize;
291 	vector<float>	texBoundaries;
292 	computeQuadTexCoordCube(texBoundaries, texCoordParams.face);
293 	const Vec3		coordA			= Vec3(texBoundaries[0], texBoundaries[1], texBoundaries[2]);
294 	const Vec3		coordB			= Vec3(texBoundaries[3], texBoundaries[4], texBoundaries[5]);
295 	const Vec3		coordC			= Vec3(texBoundaries[6], texBoundaries[7], texBoundaries[8]);
296 	const Vec3		coordAB			= coordB - coordA;
297 	const Vec3		coordAC			= coordC - coordA;
298 
299 	for (int y = 0; y < m_gridSize; y++)
300 	for (int x = 0; x < m_gridSize; x++)
301 	{
302 		const Vec2 rawFaceCoord		= texCoordParams.scale * Vec2((float)x / gridSizeFloat, (float)y / gridSizeFloat) + texCoordParams.bias;
303 		const Vec2 safeFaceCoord	= useSafeTexCoords ? safe2DTexCoords(rawFaceCoord, textureSize) : rawFaceCoord;
304 		const Vec3 texCoord			= coordA + coordAC*safeFaceCoord.x() + coordAB*safeFaceCoord.y();
305 
306 		for (int i = 0; i < 4; i++)
307 			m_texCoords.push_back(texCoord);
308 	}
309 }
310 
311 } // anonymous
312 
isLevelNearest(deUint32 filter)313 static inline bool isLevelNearest (deUint32 filter)
314 {
315 	return filter == GL_NEAREST || filter == GL_NEAREST_MIPMAP_NEAREST || filter == GL_NEAREST_MIPMAP_LINEAR;
316 }
317 
getTextureSize(const glu::Texture2D & tex)318 static inline IVec2 getTextureSize (const glu::Texture2D& tex)
319 {
320 	const tcu::Texture2D& ref = tex.getRefTexture();
321 	return IVec2(ref.getWidth(), ref.getHeight());
322 }
323 
getTextureSize(const glu::TextureCube & tex)324 static inline IVec2 getTextureSize (const glu::TextureCube& tex)
325 {
326 	const tcu::TextureCube& ref = tex.getRefTexture();
327 	return IVec2(ref.getSize(), ref.getSize());
328 }
329 
330 template <TextureType TexType>
setPixelColors(const vector<Vec4> & quadColors,const Rect & region,const PosTexCoordQuadGrid<TexType> & grid,tcu::Surface & dst)331 static void setPixelColors (const vector<Vec4>& quadColors, const Rect& region, const PosTexCoordQuadGrid<TexType>& grid, tcu::Surface& dst)
332 {
333 	const int gridSize = grid.getSize();
334 
335 	for (int y = 0; y < gridSize; y++)
336 	for (int x = 0; x < gridSize; x++)
337 	{
338 		const Vec4	color	= quadColors[y*gridSize + x];
339 		const Vec4	ldru	= grid.getQuadLDRU(x, y) * 0.5f + 0.5f; // [-1, 1] -> [0, 1]
340 		const int	ix0		= deCeilFloatToInt32(ldru.x() * (float)region.w - 0.5f);
341 		const int	ix1		= deCeilFloatToInt32(ldru.z() * (float)region.w - 0.5f);
342 		const int	iy0		= deCeilFloatToInt32(ldru.y() * (float)region.h - 0.5f);
343 		const int	iy1		= deCeilFloatToInt32(ldru.w() * (float)region.h - 0.5f);
344 
345 		for (int iy = iy0; iy < iy1; iy++)
346 		for (int ix = ix0; ix < ix1; ix++)
347 		{
348 			DE_ASSERT(deInBounds32(ix + region.x, 0, dst.getWidth()));
349 			DE_ASSERT(deInBounds32(iy + region.y, 0, dst.getHeight()));
350 
351 			dst.setPixel(ix + region.x, iy + region.y, toRGBA(color));
352 		}
353 	}
354 }
355 
sample(const tcu::Texture2D & tex,const Vec2 & coord,float lod,const tcu::Sampler & sam)356 static inline Vec4 sample (const tcu::Texture2D&		tex, const Vec2& coord, float lod, const tcu::Sampler& sam) { return tex.sample(sam, coord.x(), coord.y(), lod); }
sample(const tcu::TextureCube & tex,const Vec3 & coord,float lod,const tcu::Sampler & sam)357 static inline Vec4 sample (const tcu::TextureCube&		tex, const Vec3& coord, float lod, const tcu::Sampler& sam) { return tex.sample(sam, coord.x(), coord.y(), coord.z(), lod); }
358 
359 template <TextureType TexType>
computeReference(const typename TexTypeTcuClass<TexType>::t & texture,float lod,const tcu::Sampler & sampler,const PosTexCoordQuadGrid<TexType> & grid,tcu::Surface & dst,const Rect & dstRegion)360 void computeReference (const typename TexTypeTcuClass<TexType>::t& texture, float lod, const tcu::Sampler& sampler, const PosTexCoordQuadGrid<TexType>& grid, tcu::Surface& dst, const Rect& dstRegion)
361 {
362 	const int		gridSize	= grid.getSize();
363 	vector<Vec4>	quadColors	(gridSize*gridSize);
364 
365 	for (int y = 0; y < gridSize; y++)
366 	for (int x = 0; x < gridSize; x++)
367 	{
368 		const int										ndx		= y*gridSize + x;
369 		const typename TexTypeCoordVec<TexType>::t&		coord	= grid.getQuadTexCoord(x, y);
370 
371 		quadColors[ndx] = sample(texture, coord, lod, sampler);
372 	}
373 
374 	setPixelColors(quadColors, dstRegion, grid, dst);
375 }
376 
compareImages(const glu::RenderContext & renderCtx,tcu::TestLog & log,const tcu::Surface & ref,const tcu::Surface & res)377 static bool compareImages (const glu::RenderContext& renderCtx, tcu::TestLog& log, const tcu::Surface& ref, const tcu::Surface& res)
378 {
379 	DE_ASSERT(renderCtx.getRenderTarget().getNumSamples() == 0);
380 
381 	const tcu::RGBA threshold = renderCtx.getRenderTarget().getPixelFormat().getColorThreshold() + tcu::RGBA(15,15,15,15);
382 	return tcu::pixelThresholdCompare(log, "Result", "Image compare result", ref, res, threshold, tcu::COMPARE_LOG_RESULT);
383 }
384 
385 class Vertex2DTextureCase : public TestCase
386 {
387 public:
388 								Vertex2DTextureCase		(Context& testCtx, const char* name, const char* desc, deUint32 minFilter, deUint32 magFilter, deUint32 wrapS, deUint32 wrapT);
389 								~Vertex2DTextureCase	(void);
390 
391 	void						init					(void);
392 	void						deinit					(void);
393 	IterateResult				iterate					(void);
394 
395 private:
396 	typedef PosTexCoordQuadGrid<TEXTURETYPE_2D> Grid;
397 
398 								Vertex2DTextureCase		(const Vertex2DTextureCase& other);
399 	Vertex2DTextureCase&		operator=				(const Vertex2DTextureCase& other);
400 
401 	float						calculateLod			(const Vec2& texScale, const Vec2& dstSize, int textureNdx) const;
402 	void						setupShaderInputs		(int textureNdx, float lod, const Grid& grid) const;
403 	void						renderCell				(int textureNdx, float lod, const Grid& grid) const;
404 	void						computeReferenceCell	(int textureNdx, float lod, const Grid& grid, tcu::Surface& dst, const Rect& dstRegion) const;
405 
406 	const deUint32				m_minFilter;
407 	const deUint32				m_magFilter;
408 	const deUint32				m_wrapS;
409 	const deUint32				m_wrapT;
410 
411 	const glu::ShaderProgram*	m_program;
412 	glu::Texture2D*				m_textures[2];	// 2 textures, a gradient texture and a grid texture.
413 };
414 
Vertex2DTextureCase(Context & testCtx,const char * name,const char * desc,deUint32 minFilter,deUint32 magFilter,deUint32 wrapS,deUint32 wrapT)415 Vertex2DTextureCase::Vertex2DTextureCase (Context& testCtx, const char* name, const char* desc, deUint32 minFilter, deUint32 magFilter, deUint32 wrapS, deUint32 wrapT)
416 	: TestCase				(testCtx, tcu::NODETYPE_SELF_VALIDATE, name, desc)
417 	, m_minFilter			(minFilter)
418 	, m_magFilter			(magFilter)
419 	, m_wrapS				(wrapS)
420 	, m_wrapT				(wrapT)
421 	, m_program				(DE_NULL)
422 {
423 	m_textures[0] = DE_NULL;
424 	m_textures[1] = DE_NULL;
425 }
426 
~Vertex2DTextureCase(void)427 Vertex2DTextureCase::~Vertex2DTextureCase(void)
428 {
429 	Vertex2DTextureCase::deinit();
430 }
431 
init(void)432 void Vertex2DTextureCase::init (void)
433 {
434 	const char* const vertexShader =
435 		"attribute highp vec2 a_position;\n"
436 		"attribute highp vec2 a_texCoord;\n"
437 		"uniform highp sampler2D u_texture;\n"
438 		"uniform highp float u_lod;\n"
439 		"varying mediump vec4 v_color;\n"
440 		"\n"
441 		"void main()\n"
442 		"{\n"
443 		"	gl_Position = vec4(a_position, 0.0, 1.0);\n"
444 		"	v_color = texture2DLod(u_texture, a_texCoord, u_lod);\n"
445 		"}\n";
446 
447 	const char* const fragmentShader =
448 		"varying mediump vec4 v_color;\n"
449 		"\n"
450 		"void main()\n"
451 		"{\n"
452 		"	gl_FragColor = v_color;\n"
453 		"}\n";
454 
455 	if (m_context.getRenderTarget().getNumSamples() != 0)
456 		throw tcu::NotSupportedError("MSAA config not supported by this test");
457 
458 	DE_ASSERT(!m_program);
459 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(vertexShader, fragmentShader));
460 
461 	if(!m_program->isOk())
462 	{
463 		m_testCtx.getLog() << *m_program;
464 
465 		GLint maxVertexTextures;
466 		glGetIntegerv(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, &maxVertexTextures);
467 
468 		if (maxVertexTextures < 1)
469 			throw tcu::NotSupportedError("Vertex texture image units not supported", "", __FILE__, __LINE__);
470 		else
471 			TCU_FAIL("Failed to compile shader");
472 	}
473 
474 	// Make the textures.
475 	try
476 	{
477 		// Compute suitable power-of-two sizes (for mipmaps).
478 		const int texWidth		= 1 << deLog2Ceil32(MAX_2D_RENDER_WIDTH / 2);
479 		const int texHeight		= 1 << deLog2Ceil32(MAX_2D_RENDER_HEIGHT / 2);
480 
481 		for (int i = 0; i < 2; i++)
482 		{
483 			DE_ASSERT(!m_textures[i]);
484 			m_textures[i] = new glu::Texture2D(m_context.getRenderContext(), GL_RGB, GL_UNSIGNED_BYTE, texWidth, texHeight);
485 		}
486 
487 		const bool						mipmaps		= (deIsPowerOfTwo32(texWidth) && deIsPowerOfTwo32(texHeight));
488 		const int						numLevels	= mipmaps ? deLog2Floor32(de::max(texWidth, texHeight))+1 : 1;
489 		const tcu::TextureFormatInfo	fmtInfo		= tcu::getTextureFormatInfo(m_textures[0]->getRefTexture().getFormat());
490 		const Vec4						cBias		= fmtInfo.valueMin;
491 		const Vec4						cScale		= fmtInfo.valueMax-fmtInfo.valueMin;
492 
493 		// Fill first with gradient texture.
494 		for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
495 		{
496 			const Vec4 gMin = Vec4(-0.5f, -0.5f, -0.5f, 2.0f)*cScale + cBias;
497 			const Vec4 gMax = Vec4( 1.0f,  1.0f,  1.0f, 0.0f)*cScale + cBias;
498 
499 			m_textures[0]->getRefTexture().allocLevel(levelNdx);
500 			tcu::fillWithComponentGradients(m_textures[0]->getRefTexture().getLevel(levelNdx), gMin, gMax);
501 		}
502 
503 		// Fill second with grid texture.
504 		for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
505 		{
506 			const deUint32 step		= 0x00ffffff / numLevels;
507 			const deUint32 rgb		= step*levelNdx;
508 			const deUint32 colorA	= 0xff000000 | rgb;
509 			const deUint32 colorB	= 0xff000000 | ~rgb;
510 
511 			m_textures[1]->getRefTexture().allocLevel(levelNdx);
512 			tcu::fillWithGrid(m_textures[1]->getRefTexture().getLevel(levelNdx), 4, toVec4(tcu::RGBA(colorA))*cScale + cBias, toVec4(tcu::RGBA(colorB))*cScale + cBias);
513 		}
514 
515 		// Upload.
516 		for (int i = 0; i < 2; i++)
517 			m_textures[i]->upload();
518 	}
519 	catch (const std::exception&)
520 	{
521 		// Clean up to save memory.
522 		Vertex2DTextureCase::deinit();
523 		throw;
524 	}
525 }
526 
deinit(void)527 void Vertex2DTextureCase::deinit (void)
528 {
529 	for (int i = 0; i < 2; i++)
530 	{
531 		delete m_textures[i];
532 		m_textures[i] = DE_NULL;
533 	}
534 
535 	delete m_program;
536 	m_program = DE_NULL;
537 }
538 
calculateLod(const Vec2 & texScale,const Vec2 & dstSize,int textureNdx) const539 float Vertex2DTextureCase::calculateLod (const Vec2& texScale, const Vec2& dstSize, int textureNdx) const
540 {
541 	const tcu::Texture2D&		refTexture	= m_textures[textureNdx]->getRefTexture();
542 	const Vec2					srcSize		= Vec2((float)refTexture.getWidth(), (float)refTexture.getHeight());
543 	const Vec2					sizeRatio	= texScale*srcSize / dstSize;
544 
545 	// \note In this particular case dv/dx and du/dy are zero, simplifying the expression.
546 	return deFloatLog2(de::max(sizeRatio.x(), sizeRatio.y()));
547 }
548 
iterate(void)549 Vertex2DTextureCase::IterateResult Vertex2DTextureCase::iterate (void)
550 {
551 	const int	viewportWidth		= deMin32(m_context.getRenderTarget().getWidth(), MAX_2D_RENDER_WIDTH);
552 	const int	viewportHeight		= deMin32(m_context.getRenderTarget().getHeight(), MAX_2D_RENDER_HEIGHT);
553 
554 	const int	viewportXOffsetMax	= m_context.getRenderTarget().getWidth() - viewportWidth;
555 	const int	viewportYOffsetMax	= m_context.getRenderTarget().getHeight() - viewportHeight;
556 
557 	de::Random	rnd					(deStringHash(getName()));
558 
559 	const int	viewportXOffset		= rnd.getInt(0, viewportXOffsetMax);
560 	const int	viewportYOffset		= rnd.getInt(0, viewportYOffsetMax);
561 
562 	glUseProgram(m_program->getProgram());
563 
564 	// Divide viewport into 4 cells.
565 	const int leftWidth		= viewportWidth / 2;
566 	const int rightWidth	= viewportWidth - leftWidth;
567 	const int bottomHeight	= viewportHeight / 2;
568 	const int topHeight		= viewportHeight - bottomHeight;
569 
570 	// Clear.
571 	glClearColor(0.125f, 0.25f, 0.5f, 1.0f);
572 	glClear(GL_COLOR_BUFFER_BIT);
573 
574 	// Texture scaling and offsetting vectors.
575 	const Vec2 texMinScale		(+1.8f, +1.8f);
576 	const Vec2 texMinOffset		(-0.3f, -0.2f);
577 	const Vec2 texMagScale		(+0.3f, +0.3f);
578 	const Vec2 texMagOffset		(+0.9f, +0.8f);
579 
580 	// Surface for the reference image.
581 	tcu::Surface refImage(viewportWidth, viewportHeight);
582 
583 	{
584 		const struct Render
585 		{
586 			const Rect	region;
587 			int			textureNdx;
588 			const Vec2	texCoordScale;
589 			const Vec2	texCoordOffset;
590 			Render (const Rect& r, int tN, const Vec2& tS, const Vec2& tO) : region(r), textureNdx(tN), texCoordScale(tS), texCoordOffset(tO) {}
591 		} renders[] =
592 		{
593 			Render(Rect(0,				0,				leftWidth,	bottomHeight),	0, texMinScale, texMinOffset),
594 			Render(Rect(leftWidth,		0,				rightWidth,	bottomHeight),	0, texMagScale, texMagOffset),
595 			Render(Rect(0,				bottomHeight,	leftWidth,	topHeight),		1, texMinScale, texMinOffset),
596 			Render(Rect(leftWidth,		bottomHeight,	rightWidth,	topHeight),		1, texMagScale, texMagOffset)
597 		};
598 
599 		for (int renderNdx = 0; renderNdx < DE_LENGTH_OF_ARRAY(renders); renderNdx++)
600 		{
601 			const Render&	rend				= renders[renderNdx];
602 			const float		lod					= calculateLod(rend.texCoordScale, rend.region.size().asFloat(), rend.textureNdx);
603 			const bool		useSafeTexCoords	= isLevelNearest(lod > 0.0f ? m_minFilter : m_magFilter);
604 			const Grid		grid				(GRID_SIZE_2D, rend.region.size(), getTextureSize(*m_textures[rend.textureNdx]),
605 												 TexTypeCoordParams<TEXTURETYPE_2D>(rend.texCoordScale, rend.texCoordOffset), useSafeTexCoords);
606 
607 			glViewport(viewportXOffset + rend.region.x, viewportYOffset + rend.region.y, rend.region.w, rend.region.h);
608 			renderCell				(rend.textureNdx, lod, grid);
609 			computeReferenceCell	(rend.textureNdx, lod, grid, refImage, rend.region);
610 		}
611 	}
612 
613 	// Read back rendered results.
614 	tcu::Surface resImage(viewportWidth, viewportHeight);
615 	glu::readPixels(m_context.getRenderContext(), viewportXOffset, viewportYOffset, resImage.getAccess());
616 
617 	glUseProgram(0);
618 
619 	// Compare and log.
620 	{
621 		const bool isOk = compareImages(m_context.getRenderContext(), m_testCtx.getLog(), refImage, resImage);
622 
623 		m_testCtx.setTestResult(isOk ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
624 								isOk ? "Pass"				: "Image comparison failed");
625 	}
626 
627 	return STOP;
628 }
629 
setupShaderInputs(int textureNdx,float lod,const Grid & grid) const630 void Vertex2DTextureCase::setupShaderInputs (int textureNdx, float lod, const Grid& grid) const
631 {
632 	const deUint32 programID = m_program->getProgram();
633 
634 	// SETUP ATTRIBUTES.
635 
636 	{
637 		const int positionLoc = glGetAttribLocation(programID, "a_position");
638 		if (positionLoc != -1)
639 		{
640 			glEnableVertexAttribArray(positionLoc);
641 			glVertexAttribPointer(positionLoc, 2, GL_FLOAT, GL_FALSE, 0, grid.getPositionPtr());
642 		}
643 	}
644 
645 	{
646 		const int texCoordLoc = glGetAttribLocation(programID, "a_texCoord");
647 		if (texCoordLoc != -1)
648 		{
649 			glEnableVertexAttribArray(texCoordLoc);
650 			glVertexAttribPointer(texCoordLoc, 2, GL_FLOAT, GL_FALSE, 0, grid.getTexCoordPtr());
651 		}
652 	}
653 
654 	// SETUP UNIFORMS.
655 
656 	{
657 		const int lodLoc = glGetUniformLocation(programID, "u_lod");
658 		if (lodLoc != -1)
659 			glUniform1f(lodLoc, lod);
660 	}
661 
662 	glActiveTexture(GL_TEXTURE0);
663 	glBindTexture(GL_TEXTURE_2D, m_textures[textureNdx]->getGLTexture());
664 	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S,		m_wrapS);
665 	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T,		m_wrapT);
666 	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER,	m_minFilter);
667 	glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER,	m_magFilter);
668 
669 	{
670 		const int texLoc = glGetUniformLocation(programID, "u_texture");
671 		if (texLoc != -1)
672 			glUniform1i(texLoc, 0);
673 	}
674 }
675 
676 // Renders one sub-image with given parameters.
renderCell(int textureNdx,float lod,const Grid & grid) const677 void Vertex2DTextureCase::renderCell (int textureNdx, float lod, const Grid& grid) const
678 {
679 	setupShaderInputs(textureNdx, lod, grid);
680 	glDrawElements(GL_TRIANGLES, grid.getNumIndices(), GL_UNSIGNED_SHORT, grid.getIndexPtr());
681 }
682 
computeReferenceCell(int textureNdx,float lod,const Grid & grid,tcu::Surface & dst,const Rect & dstRegion) const683 void Vertex2DTextureCase::computeReferenceCell (int textureNdx, float lod, const Grid& grid, tcu::Surface& dst, const Rect& dstRegion) const
684 {
685 	computeReference(m_textures[textureNdx]->getRefTexture(), lod, glu::mapGLSampler(m_wrapS, m_wrapT, m_minFilter, m_magFilter), grid, dst, dstRegion);
686 }
687 
688 class VertexCubeTextureCase : public TestCase
689 {
690 public:
691 								VertexCubeTextureCase	(Context& testCtx, const char* name, const char* desc, deUint32 minFilter, deUint32 magFilter, deUint32 wrapS, deUint32 wrapT);
692 								~VertexCubeTextureCase	(void);
693 
694 	void						init					(void);
695 	void						deinit					(void);
696 	IterateResult				iterate					(void);
697 
698 private:
699 	typedef PosTexCoordQuadGrid<TEXTURETYPE_CUBE> Grid;
700 
701 								VertexCubeTextureCase	(const VertexCubeTextureCase& other);
702 	VertexCubeTextureCase&		operator=				(const VertexCubeTextureCase& other);
703 
704 	float						calculateLod			(const Vec2& texScale, const Vec2& dstSize, int textureNdx) const;
705 	void						setupShaderInputs		(int textureNdx, float lod, const Grid& grid) const;
706 	void						renderCell				(int textureNdx, float lod, const Grid& grid) const;
707 	void						computeReferenceCell	(int textureNdx, float lod, const Grid& grid, tcu::Surface& dst, const Rect& dstRegion) const;
708 
709 	const deUint32				m_minFilter;
710 	const deUint32				m_magFilter;
711 	const deUint32				m_wrapS;
712 	const deUint32				m_wrapT;
713 
714 	const glu::ShaderProgram*	m_program;
715 	glu::TextureCube*			m_textures[2];	// 2 textures, a gradient texture and a grid texture.
716 };
717 
VertexCubeTextureCase(Context & testCtx,const char * name,const char * desc,deUint32 minFilter,deUint32 magFilter,deUint32 wrapS,deUint32 wrapT)718 VertexCubeTextureCase::VertexCubeTextureCase (Context& testCtx, const char* name, const char* desc, deUint32 minFilter, deUint32 magFilter, deUint32 wrapS, deUint32 wrapT)
719 	: TestCase				(testCtx, tcu::NODETYPE_SELF_VALIDATE, name, desc)
720 	, m_minFilter			(minFilter)
721 	, m_magFilter			(magFilter)
722 	, m_wrapS				(wrapS)
723 	, m_wrapT				(wrapT)
724 	, m_program				(DE_NULL)
725 {
726 	m_textures[0] = DE_NULL;
727 	m_textures[1] = DE_NULL;
728 }
729 
~VertexCubeTextureCase(void)730 VertexCubeTextureCase::~VertexCubeTextureCase(void)
731 {
732 	VertexCubeTextureCase::deinit();
733 }
734 
init(void)735 void VertexCubeTextureCase::init (void)
736 {
737 	const char* const vertexShader =
738 		"attribute highp vec2 a_position;\n"
739 		"attribute highp vec3 a_texCoord;\n"
740 		"uniform highp samplerCube u_texture;\n"
741 		"uniform highp float u_lod;\n"
742 		"varying mediump vec4 v_color;\n"
743 		"\n"
744 		"void main()\n"
745 		"{\n"
746 		"	gl_Position = vec4(a_position, 0.0, 1.0);\n"
747 		"	v_color = textureCubeLod(u_texture, a_texCoord, u_lod);\n"
748 		"}\n";
749 
750 	const char* const fragmentShader =
751 		"varying mediump vec4 v_color;\n"
752 		"\n"
753 		"void main()\n"
754 		"{\n"
755 		"	gl_FragColor = v_color;\n"
756 		"}\n";
757 
758 	if (m_context.getRenderTarget().getNumSamples() != 0)
759 		throw tcu::NotSupportedError("MSAA config not supported by this test");
760 
761 	DE_ASSERT(!m_program);
762 	m_program = new glu::ShaderProgram(m_context.getRenderContext(), glu::makeVtxFragSources(vertexShader, fragmentShader));
763 
764 	if(!m_program->isOk())
765 	{
766 		m_testCtx.getLog() << *m_program;
767 
768 		GLint maxVertexTextures;
769 		glGetIntegerv(GL_MAX_VERTEX_TEXTURE_IMAGE_UNITS, &maxVertexTextures);
770 
771 		if (maxVertexTextures < 1)
772 			throw tcu::NotSupportedError("Vertex texture image units not supported", "", __FILE__, __LINE__);
773 		else
774 			TCU_FAIL("Failed to compile shader");
775 	}
776 
777 	// Make the textures.
778 	try
779 	{
780 		// Compute suitable power-of-two sizes (for mipmaps).
781 		const int texWidth		= 1 << deLog2Ceil32(MAX_CUBE_RENDER_WIDTH / 3 / 2);
782 		const int texHeight		= 1 << deLog2Ceil32(MAX_CUBE_RENDER_HEIGHT / 2 / 2);
783 
784 		DE_ASSERT(texWidth == texHeight);
785 		DE_UNREF(texHeight);
786 
787 		for (int i = 0; i < 2; i++)
788 		{
789 			DE_ASSERT(!m_textures[i]);
790 			m_textures[i] = new glu::TextureCube(m_context.getRenderContext(), GL_RGB, GL_UNSIGNED_BYTE, texWidth);
791 		}
792 
793 		const bool						mipmaps		= deIsPowerOfTwo32(texWidth) != DE_FALSE;
794 		const int						numLevels	= mipmaps ? deLog2Floor32(texWidth)+1 : 1;
795 		const tcu::TextureFormatInfo	fmtInfo		= tcu::getTextureFormatInfo(m_textures[0]->getRefTexture().getFormat());
796 		const Vec4						cBias		= fmtInfo.valueMin;
797 		const Vec4						cScale		= fmtInfo.valueMax-fmtInfo.valueMin;
798 
799 		// Fill first with gradient texture.
800 		static const Vec4 gradients[tcu::CUBEFACE_LAST][2] =
801 		{
802 			{ Vec4(-1.0f, -1.0f, -1.0f, 2.0f), Vec4(1.0f, 1.0f, 1.0f, 0.0f) }, // negative x
803 			{ Vec4( 0.0f, -1.0f, -1.0f, 2.0f), Vec4(1.0f, 1.0f, 1.0f, 0.0f) }, // positive x
804 			{ Vec4(-1.0f,  0.0f, -1.0f, 2.0f), Vec4(1.0f, 1.0f, 1.0f, 0.0f) }, // negative y
805 			{ Vec4(-1.0f, -1.0f,  0.0f, 2.0f), Vec4(1.0f, 1.0f, 1.0f, 0.0f) }, // positive y
806 			{ Vec4(-1.0f, -1.0f, -1.0f, 0.0f), Vec4(1.0f, 1.0f, 1.0f, 1.0f) }, // negative z
807 			{ Vec4( 0.0f,  0.0f,  0.0f, 2.0f), Vec4(1.0f, 1.0f, 1.0f, 0.0f) }  // positive z
808 		};
809 		for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
810 		{
811 			for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
812 			{
813 				m_textures[0]->getRefTexture().allocLevel((tcu::CubeFace)face, levelNdx);
814 				tcu::fillWithComponentGradients(m_textures[0]->getRefTexture().getLevelFace(levelNdx, (tcu::CubeFace)face), gradients[face][0]*cScale + cBias, gradients[face][1]*cScale + cBias);
815 			}
816 		}
817 
818 		// Fill second with grid texture.
819 		for (int face = 0; face < tcu::CUBEFACE_LAST; face++)
820 		{
821 			for (int levelNdx = 0; levelNdx < numLevels; levelNdx++)
822 			{
823 				const deUint32 step		= 0x00ffffff / (numLevels*tcu::CUBEFACE_LAST);
824 				const deUint32 rgb		= step*levelNdx*face;
825 				const deUint32 colorA	= 0xff000000 | rgb;
826 				const deUint32 colorB	= 0xff000000 | ~rgb;
827 
828 				m_textures[1]->getRefTexture().allocLevel((tcu::CubeFace)face, levelNdx);
829 				tcu::fillWithGrid(m_textures[1]->getRefTexture().getLevelFace(levelNdx, (tcu::CubeFace)face), 4, toVec4(tcu::RGBA(colorA))*cScale + cBias, toVec4(tcu::RGBA(colorB))*cScale + cBias);
830 			}
831 		}
832 
833 		// Upload.
834 		for (int i = 0; i < 2; i++)
835 			m_textures[i]->upload();
836 	}
837 	catch (const std::exception&)
838 	{
839 		// Clean up to save memory.
840 		VertexCubeTextureCase::deinit();
841 		throw;
842 	}
843 }
844 
deinit(void)845 void VertexCubeTextureCase::deinit (void)
846 {
847 	for (int i = 0; i < 2; i++)
848 	{
849 		delete m_textures[i];
850 		m_textures[i] = DE_NULL;
851 	}
852 
853 	delete m_program;
854 	m_program = DE_NULL;
855 }
856 
calculateLod(const Vec2 & texScale,const Vec2 & dstSize,int textureNdx) const857 float VertexCubeTextureCase::calculateLod (const Vec2& texScale, const Vec2& dstSize, int textureNdx) const
858 {
859 	const tcu::TextureCube&		refTexture	= m_textures[textureNdx]->getRefTexture();
860 	const Vec2					srcSize		= Vec2((float)refTexture.getSize(), (float)refTexture.getSize());
861 	const Vec2					sizeRatio	= texScale*srcSize / dstSize;
862 
863 	// \note In this particular case, dv/dx and du/dy are zero, simplifying the expression.
864 	return deFloatLog2(de::max(sizeRatio.x(), sizeRatio.y()));
865 }
866 
iterate(void)867 VertexCubeTextureCase::IterateResult VertexCubeTextureCase::iterate (void)
868 {
869 	const int	viewportWidth		= deMin32(m_context.getRenderTarget().getWidth(), MAX_CUBE_RENDER_WIDTH);
870 	const int	viewportHeight		= deMin32(m_context.getRenderTarget().getHeight(), MAX_CUBE_RENDER_HEIGHT);
871 
872 	const int	viewportXOffsetMax	= m_context.getRenderTarget().getWidth() - viewportWidth;
873 	const int	viewportYOffsetMax	= m_context.getRenderTarget().getHeight() - viewportHeight;
874 
875 	de::Random	rnd					(deStringHash(getName()));
876 
877 	const int	viewportXOffset		= rnd.getInt(0, viewportXOffsetMax);
878 	const int	viewportYOffset		= rnd.getInt(0, viewportYOffsetMax);
879 
880 	glUseProgram(m_program->getProgram());
881 
882 	// Divide viewport into 4 areas.
883 	const int leftWidth		= viewportWidth / 2;
884 	const int rightWidth	= viewportWidth - leftWidth;
885 	const int bottomHeight	= viewportHeight / 2;
886 	const int topHeight		= viewportHeight - bottomHeight;
887 
888 	// Clear.
889 	glClearColor(0.125f, 0.25f, 0.5f, 1.0f);
890 	glClear(GL_COLOR_BUFFER_BIT);
891 
892 	// Texture scaling and offsetting vectors.
893 	const Vec2 texMinScale		(1.0f, 1.0f);
894 	const Vec2 texMinOffset		(0.0f, 0.0f);
895 	const Vec2 texMagScale		(0.3f, 0.3f);
896 	const Vec2 texMagOffset		(0.5f, 0.3f);
897 
898 	// Surface for the reference image.
899 	tcu::Surface refImage(viewportWidth, viewportHeight);
900 
901 	// Each of the four areas is divided into 6 cells.
902 	const int defCellWidth	= viewportWidth / 2 / 3;
903 	const int defCellHeight	= viewportHeight / 2 / 2;
904 
905 	for (int i = 0; i < tcu::CUBEFACE_LAST; i++)
906 	{
907 		const int	cellOffsetX			= defCellWidth * (i % 3);
908 		const int	cellOffsetY			= defCellHeight * (i / 3);
909 		const bool	isRightmostCell		= i == 2 || i == 5;
910 		const bool	isTopCell			= i >= 3;
911 		const int	leftCellWidth		= isRightmostCell	? leftWidth		- cellOffsetX : defCellWidth;
912 		const int	rightCellWidth		= isRightmostCell	? rightWidth	- cellOffsetX : defCellWidth;
913 		const int	bottomCellHeight	= isTopCell			? bottomHeight	- cellOffsetY : defCellHeight;
914 		const int	topCellHeight		= isTopCell			? topHeight		- cellOffsetY : defCellHeight;
915 
916 		const struct Render
917 		{
918 			const Rect	region;
919 			int			textureNdx;
920 			const Vec2	texCoordScale;
921 			const Vec2	texCoordOffset;
922 			Render (const Rect& r, int tN, const Vec2& tS, const Vec2& tO) : region(r), textureNdx(tN), texCoordScale(tS), texCoordOffset(tO) {}
923 		} renders[] =
924 		{
925 			Render(Rect(cellOffsetX + 0,			cellOffsetY + 0,				leftCellWidth,	bottomCellHeight),	0, texMinScale, texMinOffset),
926 			Render(Rect(cellOffsetX + leftWidth,	cellOffsetY + 0,				rightCellWidth,	bottomCellHeight),	0, texMagScale, texMagOffset),
927 			Render(Rect(cellOffsetX + 0,			cellOffsetY + bottomHeight,		leftCellWidth,	topCellHeight),		1, texMinScale, texMinOffset),
928 			Render(Rect(cellOffsetX + leftWidth,	cellOffsetY + bottomHeight,		rightCellWidth,	topCellHeight),		1, texMagScale, texMagOffset)
929 		};
930 
931 		for (int renderNdx = 0; renderNdx < DE_LENGTH_OF_ARRAY(renders); renderNdx++)
932 		{
933 			const Render&	rend				= renders[renderNdx];
934 			const float		lod					= calculateLod(rend.texCoordScale, rend.region.size().asFloat(), rend.textureNdx);
935 			const bool		useSafeTexCoords	= isLevelNearest(lod > 0.0f ? m_minFilter : m_magFilter);
936 			const Grid		grid				(GRID_SIZE_CUBE, rend.region.size(), getTextureSize(*m_textures[rend.textureNdx]),
937 												 TexTypeCoordParams<TEXTURETYPE_CUBE>(rend.texCoordScale, rend.texCoordOffset, (tcu::CubeFace)i), useSafeTexCoords);
938 
939 			glViewport(viewportXOffset + rend.region.x, viewportYOffset + rend.region.y, rend.region.w, rend.region.h);
940 			renderCell				(rend.textureNdx, lod, grid);
941 			computeReferenceCell	(rend.textureNdx, lod, grid, refImage, rend.region);
942 		}
943 	}
944 
945 	// Read back rendered results.
946 	tcu::Surface resImage(viewportWidth, viewportHeight);
947 	glu::readPixels(m_context.getRenderContext(), viewportXOffset, viewportYOffset, resImage.getAccess());
948 
949 	glUseProgram(0);
950 
951 	// Compare and log.
952 	{
953 		const bool isOk = compareImages(m_context.getRenderContext(), m_testCtx.getLog(), refImage, resImage);
954 
955 		m_testCtx.setTestResult(isOk ? QP_TEST_RESULT_PASS	: QP_TEST_RESULT_FAIL,
956 								isOk ? "Pass"				: "Image comparison failed");
957 	}
958 
959 	return STOP;
960 }
961 
setupShaderInputs(int textureNdx,float lod,const Grid & grid) const962 void VertexCubeTextureCase::setupShaderInputs (int textureNdx, float lod, const Grid& grid) const
963 {
964 	const deUint32 programID = m_program->getProgram();
965 
966 	// SETUP ATTRIBUTES.
967 
968 	{
969 		const int positionLoc = glGetAttribLocation(programID, "a_position");
970 		if (positionLoc != -1)
971 		{
972 			glEnableVertexAttribArray(positionLoc);
973 			glVertexAttribPointer(positionLoc, 2, GL_FLOAT, GL_FALSE, 0, grid.getPositionPtr());
974 		}
975 	}
976 
977 	{
978 		const int texCoordLoc = glGetAttribLocation(programID, "a_texCoord");
979 		if (texCoordLoc != -1)
980 		{
981 			glEnableVertexAttribArray(texCoordLoc);
982 			glVertexAttribPointer(texCoordLoc, 3, GL_FLOAT, GL_FALSE, 0, grid.getTexCoordPtr());
983 		}
984 	}
985 
986 	// SETUP UNIFORMS.
987 
988 	{
989 		const int lodLoc = glGetUniformLocation(programID, "u_lod");
990 		if (lodLoc != -1)
991 			glUniform1f(lodLoc, lod);
992 	}
993 
994 	glActiveTexture(GL_TEXTURE0);
995 	glBindTexture(GL_TEXTURE_CUBE_MAP, m_textures[textureNdx]->getGLTexture());
996 	glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_S,		m_wrapS);
997 	glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_WRAP_T,		m_wrapT);
998 	glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MIN_FILTER,	m_minFilter);
999 	glTexParameteri(GL_TEXTURE_CUBE_MAP, GL_TEXTURE_MAG_FILTER,	m_magFilter);
1000 
1001 	{
1002 		const int texLoc = glGetUniformLocation(programID, "u_texture");
1003 		if (texLoc != -1)
1004 			glUniform1i(texLoc, 0);
1005 	}
1006 }
1007 
1008 // Renders one cube face with given parameters.
renderCell(int textureNdx,float lod,const Grid & grid) const1009 void VertexCubeTextureCase::renderCell (int textureNdx, float lod, const Grid& grid) const
1010 {
1011 	setupShaderInputs(textureNdx, lod, grid);
1012 	glDrawElements(GL_TRIANGLES, grid.getNumIndices(), GL_UNSIGNED_SHORT, grid.getIndexPtr());
1013 }
1014 
1015 // Computes reference for one cube face with given parameters.
computeReferenceCell(int textureNdx,float lod,const Grid & grid,tcu::Surface & dst,const Rect & dstRegion) const1016 void VertexCubeTextureCase::computeReferenceCell (int textureNdx, float lod, const Grid& grid, tcu::Surface& dst, const Rect& dstRegion) const
1017 {
1018 	tcu::Sampler sampler = glu::mapGLSampler(m_wrapS, m_wrapT, m_minFilter, m_magFilter);
1019 	computeReference(m_textures[textureNdx]->getRefTexture(), lod, sampler, grid, dst, dstRegion);
1020 }
1021 
VertexTextureTests(Context & context)1022 VertexTextureTests::VertexTextureTests (Context& context)
1023 	: TestCaseGroup(context, "vertex", "Vertex Texture Tests")
1024 {
1025 }
1026 
~VertexTextureTests(void)1027 VertexTextureTests::~VertexTextureTests(void)
1028 {
1029 }
1030 
init(void)1031 void VertexTextureTests::init (void)
1032 {
1033 	// 2D and cube map groups, and their filtering and wrap sub-groups.
1034 	TestCaseGroup* const group2D				= new TestCaseGroup(m_context, "2d",			"2D Vertex Texture Tests");
1035 	TestCaseGroup* const groupCube				= new TestCaseGroup(m_context, "cube",			"Cube Map Vertex Texture Tests");
1036 	TestCaseGroup* const filteringGroup2D		= new TestCaseGroup(m_context, "filtering",		"2D Vertex Texture Filtering Tests");
1037 	TestCaseGroup* const wrapGroup2D			= new TestCaseGroup(m_context, "wrap",			"2D Vertex Texture Wrap Tests");
1038 	TestCaseGroup* const filteringGroupCube		= new TestCaseGroup(m_context, "filtering",		"Cube Map Vertex Texture Filtering Tests");
1039 	TestCaseGroup* const wrapGroupCube			= new TestCaseGroup(m_context, "wrap",			"Cube Map Vertex Texture Wrap Tests");
1040 
1041 	group2D->addChild(filteringGroup2D);
1042 	group2D->addChild(wrapGroup2D);
1043 	groupCube->addChild(filteringGroupCube);
1044 	groupCube->addChild(wrapGroupCube);
1045 
1046 	addChild(group2D);
1047 	addChild(groupCube);
1048 
1049 	static const struct
1050 	{
1051 		const char*		name;
1052 		GLenum			mode;
1053 	} wrapModes[] =
1054 	{
1055 		{ "clamp",		GL_CLAMP_TO_EDGE	},
1056 		{ "repeat",		GL_REPEAT			},
1057 		{ "mirror",		GL_MIRRORED_REPEAT	}
1058 	};
1059 
1060 	static const struct
1061 	{
1062 		const char*		name;
1063 		GLenum			mode;
1064 	} minFilterModes[] =
1065 	{
1066 		{ "nearest",				GL_NEAREST					},
1067 		{ "linear",					GL_LINEAR					},
1068 		{ "nearest_mipmap_nearest",	GL_NEAREST_MIPMAP_NEAREST	},
1069 		{ "linear_mipmap_nearest",	GL_LINEAR_MIPMAP_NEAREST	},
1070 		{ "nearest_mipmap_linear",	GL_NEAREST_MIPMAP_LINEAR	},
1071 		{ "linear_mipmap_linear",	GL_LINEAR_MIPMAP_LINEAR		}
1072 	};
1073 
1074 	static const struct
1075 	{
1076 		const char*		name;
1077 		GLenum			mode;
1078 	} magFilterModes[] =
1079 	{
1080 		{ "nearest",	GL_NEAREST	},
1081 		{ "linear",		GL_LINEAR	}
1082 	};
1083 
1084 #define FOR_EACH(ITERATOR, ARRAY, BODY)	\
1085 	for (int ITERATOR = 0; ITERATOR < DE_LENGTH_OF_ARRAY(ARRAY); ITERATOR++)	\
1086 		BODY
1087 
1088 	// 2D cases.
1089 
1090 	FOR_EACH(minFilter,		minFilterModes,
1091 	FOR_EACH(magFilter,		magFilterModes,
1092 	FOR_EACH(wrapMode,		wrapModes,
1093 		{
1094 			const string name = string("") + minFilterModes[minFilter].name + "_" + magFilterModes[magFilter].name + "_" + wrapModes[wrapMode].name;
1095 
1096 			filteringGroup2D->addChild(new Vertex2DTextureCase(m_context,
1097 															   name.c_str(), "",
1098 															   minFilterModes[minFilter].mode,
1099 															   magFilterModes[magFilter].mode,
1100 															   wrapModes[wrapMode].mode,
1101 															   wrapModes[wrapMode].mode));
1102 		})));
1103 
1104 	FOR_EACH(wrapSMode,		wrapModes,
1105 	FOR_EACH(wrapTMode,		wrapModes,
1106 		{
1107 			const string name = string("") + wrapModes[wrapSMode].name + "_" + wrapModes[wrapTMode].name;
1108 
1109 			wrapGroup2D->addChild(new Vertex2DTextureCase(m_context,
1110 														  name.c_str(), "",
1111 														  GL_LINEAR_MIPMAP_LINEAR,
1112 														  GL_LINEAR,
1113 														  wrapModes[wrapSMode].mode,
1114 														  wrapModes[wrapTMode].mode));
1115 		}));
1116 
1117 	// Cube map cases.
1118 
1119 	FOR_EACH(minFilter,		minFilterModes,
1120 	FOR_EACH(magFilter,		magFilterModes,
1121 	FOR_EACH(wrapMode,		wrapModes,
1122 		{
1123 			const string name = string("") + minFilterModes[minFilter].name + "_" + magFilterModes[magFilter].name + "_" + wrapModes[wrapMode].name;
1124 
1125 			filteringGroupCube->addChild(new VertexCubeTextureCase(m_context,
1126 																   name.c_str(), "",
1127 																   minFilterModes[minFilter].mode,
1128 																   magFilterModes[magFilter].mode,
1129 																   wrapModes[wrapMode].mode,
1130 																   wrapModes[wrapMode].mode));
1131 		})));
1132 
1133 	FOR_EACH(wrapSMode,		wrapModes,
1134 	FOR_EACH(wrapTMode,		wrapModes,
1135 		{
1136 			const string name = string("") + wrapModes[wrapSMode].name + "_" + wrapModes[wrapTMode].name;
1137 
1138 			wrapGroupCube->addChild(new VertexCubeTextureCase(m_context,
1139 															  name.c_str(), "",
1140 															  GL_LINEAR_MIPMAP_LINEAR,
1141 															  GL_LINEAR,
1142 															  wrapModes[wrapSMode].mode,
1143 															  wrapModes[wrapTMode].mode));
1144 		}));
1145 }
1146 
1147 } // Functional
1148 } // gles2
1149 } // deqp
1150