1 #ifndef _GLSLONGSTRESSCASE_HPP
2 #define _GLSLONGSTRESSCASE_HPP
3 /*-------------------------------------------------------------------------
4  * drawElements Quality Program OpenGL (ES) Module
5  * -----------------------------------------------
6  *
7  * Copyright 2014 The Android Open Source Project
8  *
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  *
13  *      http://www.apache.org/licenses/LICENSE-2.0
14  *
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  *
21  *//*!
22  * \file
23  * \brief Parametrized, long-running stress case.
24  *//*--------------------------------------------------------------------*/
25 
26 #include "tcuDefs.hpp"
27 #include "tcuTestCase.hpp"
28 #include "tcuTexture.hpp"
29 #include "tcuMatrix.hpp"
30 #include "gluRenderContext.hpp"
31 #include "gluShaderUtil.hpp"
32 #include "glsTextureTestUtil.hpp"
33 #include "deRandom.hpp"
34 #include "deSharedPtr.hpp"
35 
36 #include <string>
37 #include <vector>
38 #include <map>
39 
40 namespace deqp
41 {
42 namespace gls
43 {
44 
45 namespace LongStressCaseInternal
46 {
47 
48 template <typename T> class GLObjectManager;
49 class Program;
50 class Buffer;
51 class Texture;
52 class DebugInfoRenderer;
53 
54 }
55 
56 struct VarSpec
57 {
58 	union Value
59 	{
60 		float	f[4*4]; // \note Matrices are stored in column major order.
61 		int		i[4];
62 	};
63 
64 	std::string		name;
65 	glu::DataType	type;
66 	Value			minValue;
67 	Value			maxValue;
68 
69 	template <typename T>
VarSpecdeqp::gls::VarSpec70 	VarSpec (const std::string& name_, const T& minValue_, const T& maxValue_)	: name(name_) { set(minValue_, maxValue_); }
71 
72 	template <typename T>
VarSpecdeqp::gls::VarSpec73 	VarSpec (const std::string& name_, const T& value)							: name(name_) { set(value, value); }
74 
setdeqp::gls::VarSpec75 	void set (float minValue_, float maxValue_)
76 	{
77 		type			= glu::TYPE_FLOAT;
78 		minValue.f[0]	= minValue_;
79 		maxValue.f[0]	= maxValue_;
80 	}
81 
82 	template <int ValSize>
setdeqp::gls::VarSpec83 	void set (const tcu::Vector<float, ValSize>& minValue_, const tcu::Vector<float, ValSize>& maxValue_)
84 	{
85 		type = glu::getDataTypeFloatVec(ValSize);
86 		vecToArr(minValue_, minValue.f);
87 		vecToArr(maxValue_, maxValue.f);
88 	}
89 
90 	template <int ValRows, int ValCols>
setdeqp::gls::VarSpec91 	void set (const tcu::Matrix<float, ValRows, ValCols>& minValue_, const tcu::Matrix<float, ValRows, ValCols>& maxValue_)
92 	{
93 		type = glu::getDataTypeMatrix(ValCols, ValRows);
94 		matToArr(minValue_, minValue.f);
95 		matToArr(maxValue_, maxValue.f);
96 	}
97 
setdeqp::gls::VarSpec98 	void set (int minValue_, int maxValue_)
99 	{
100 		type			= glu::TYPE_INT;
101 		minValue.i[0]	= minValue_;
102 		maxValue.i[0]	= maxValue_;
103 	}
104 
105 	template <int ValSize>
setdeqp::gls::VarSpec106 	void set (const tcu::Vector<int, ValSize>& minValue_, const tcu::Vector<int, ValSize>& maxValue_)
107 	{
108 		type = glu::getDataTypeVector(glu::TYPE_INT, ValSize);
109 		vecToArr(minValue_, minValue.i);
110 		vecToArr(maxValue_, maxValue.i);
111 	}
112 
113 private:
114 	template <typename T, int SrcSize, int DstSize>
vecToArrdeqp::gls::VarSpec115 	static inline void vecToArr (const tcu::Vector<T, SrcSize>& src, T (&dst)[DstSize])
116 	{
117 		DE_STATIC_ASSERT(DstSize >= SrcSize);
118 		for (int i = 0; i < SrcSize; i++)
119 			dst[i] = src[i];
120 	}
121 
122 	template <int ValRows, int ValCols, int DstSize>
matToArrdeqp::gls::VarSpec123 	static inline void matToArr (const tcu::Matrix<float, ValRows, ValCols>& src, float (&dst)[DstSize])
124 	{
125 		DE_STATIC_ASSERT(DstSize >= ValRows*ValCols);
126 		tcu::Array<float, ValRows*ValCols> data = src.getColumnMajorData();
127 		for (int i = 0; i < ValRows*ValCols; i++)
128 			dst[i] = data[i];
129 	}
130 };
131 
132 struct TextureSpec
133 {
134 	glu::TextureTestUtil::TextureType	textureType;
135 	deUint32							textureUnit;
136 	int									width;
137 	int									height;
138 	deUint32							format;
139 	deUint32							dataType;
140 	deUint32							internalFormat;
141 	bool								useMipmap;
142 	deUint32							minFilter;
143 	deUint32							magFilter;
144 	deUint32							sWrap;
145 	deUint32							tWrap;
146 	tcu::Vec4							minValue;
147 	tcu::Vec4							maxValue;
148 
TextureSpecdeqp::gls::TextureSpec149 	TextureSpec (const glu::TextureTestUtil::TextureType	texType,
150 				 const deUint32								unit,
151 				 const int									width_,
152 				 const int									height_,
153 				 const deUint32								format_,
154 				 const deUint32								dataType_,
155 				 const deUint32								internalFormat_,
156 				 const bool									useMipmap_,
157 				 const deUint32								minFilter_,
158 				 const deUint32								magFilter_,
159 				 const deUint32								sWrap_,
160 				 const deUint32								tWrap_,
161 				 const tcu::Vec4&							minValue_,
162 				 const tcu::Vec4&							maxValue_)
163 		: textureType		(texType)
164 		, textureUnit		(unit)
165 		, width				(width_)
166 		, height			(height_)
167 		, format			(format_)
168 		, dataType			(dataType_)
169 		, internalFormat	(internalFormat_)
170 		, useMipmap			(useMipmap_)
171 		, minFilter			(minFilter_)
172 		, magFilter			(magFilter_)
173 		, sWrap				(sWrap_)
174 		, tWrap				(tWrap_)
175 		, minValue			(minValue_)
176 		, maxValue			(maxValue_)
177 	{
178 	}
179 };
180 
181 /*--------------------------------------------------------------------*//*!
182  * \brief Struct for a shader program sources and related data
183  *
184  * A ProgramContext holds a program's vertex and fragment shader sources
185  * as well as specifications of its attributes, uniforms, and textures.
186  * When given to a StressCase, the string ${NS} is replaced by a magic
187  * number that varies between different compilations of the same program;
188  * the same replacement is done in attributes' and uniforms' names. This
189  * can be used to avoid shader caching by the GL, by e.g. suffixing each
190  * attribute, uniform and varying name with ${NS} in the shader source.
191  *//*--------------------------------------------------------------------*/
192 struct ProgramContext
193 {
194 	std::string							vertexSource;
195 	std::string							fragmentSource;
196 	std::vector<VarSpec>				attributes;
197 	std::vector<VarSpec>				uniforms;
198 
199 	std::vector<TextureSpec>			textureSpecs;		//!< \note If multiple textures have same unit, one of them is picked randomly.
200 
201 	std::string							positionAttrName;	//!< \note Position attribute may get a bit more careful handling than just complete random.
202 
ProgramContextdeqp::gls::ProgramContext203 	ProgramContext (const char* const vtxShaderSource_,
204 					const char* const fragShaderSource_,
205 					const char* const positionAttrName_)
206 		: vertexSource		(vtxShaderSource_)
207 		, fragmentSource	(fragShaderSource_)
208 		, positionAttrName	(positionAttrName_)
209 	{
210 	}
211 };
212 
213 class LongStressCase : public tcu::TestCase
214 {
215 public:
216 	//! Probabilities for actions that may be taken on each iteration. \note The texture and buffer specific actions are randomized per texture or buffer.
217 	struct FeatureProbabilities
218 	{
219 		float rebuildProgram;				//!< Rebuild program, with variable name-mangling.
220 		float reuploadTexture;				//!< Reupload texture, even if it already exists and has been uploaded.
221 		float reuploadBuffer;				//!< Reupload buffer, even if it already exists and has been uploaded.
222 		float reuploadWithTexImage;			//!< Use glTexImage*() when re-uploading texture, not glTexSubImage*().
223 		float reuploadWithBufferData;		//!< Use glBufferData() when re-uploading buffer, not glBufferSubData().
224 		float deleteTexture;				//!< Delete texture at end of iteration, even if we could re-use it.
225 		float deleteBuffer;					//!< Delete buffer at end of iteration, even if we could re-use it.
226 		float wastefulTextureMemoryUsage;	//!< Don't re-use a texture, and don't delete it until given memory limit is hit.
227 		float wastefulBufferMemoryUsage;	//!< Don't re-use a buffer, and don't delete it until given memory limit is hit.
228 		float clientMemoryAttributeData;	//!< Use client memory for vertex attribute data when drawing (instead of GL buffers).
229 		float clientMemoryIndexData;		//!< Use client memory for vertex indices when drawing (instead of GL buffers).
230 		float randomBufferUploadTarget;		//!< Use a random target when setting buffer data (i.e. not necessarily the one it'll be ultimately bound to).
231 		float randomBufferUsage;			//!< Use a random buffer usage parameter with glBufferData(), instead of the ones specified as params for the case.
232 		float useDrawArrays;				//!< Use glDrawArrays() instead of glDrawElements().
233 		float separateAttributeBuffers;		//!< Give each vertex attribute its own buffer.
234 
235 		// Named parameter idiom: helpers that can be used when making temporaries, e.g. FeatureProbabilities().pReuploadTexture(1.0f).pReuploadWithTexImage(1.0f)
pRebuildProgramdeqp::gls::LongStressCase::FeatureProbabilities236 		FeatureProbabilities& pRebuildProgram				(const float prob) { rebuildProgram					= prob; return *this; }
pReuploadTexturedeqp::gls::LongStressCase::FeatureProbabilities237 		FeatureProbabilities& pReuploadTexture				(const float prob) { reuploadTexture				= prob; return *this; }
pReuploadBufferdeqp::gls::LongStressCase::FeatureProbabilities238 		FeatureProbabilities& pReuploadBuffer				(const float prob) { reuploadBuffer					= prob; return *this; }
pReuploadWithTexImagedeqp::gls::LongStressCase::FeatureProbabilities239 		FeatureProbabilities& pReuploadWithTexImage			(const float prob) { reuploadWithTexImage			= prob; return *this; }
pReuploadWithBufferDatadeqp::gls::LongStressCase::FeatureProbabilities240 		FeatureProbabilities& pReuploadWithBufferData		(const float prob) { reuploadWithBufferData			= prob; return *this; }
pDeleteTexturedeqp::gls::LongStressCase::FeatureProbabilities241 		FeatureProbabilities& pDeleteTexture				(const float prob) { deleteTexture					= prob; return *this; }
pDeleteBufferdeqp::gls::LongStressCase::FeatureProbabilities242 		FeatureProbabilities& pDeleteBuffer					(const float prob) { deleteBuffer					= prob; return *this; }
pWastefulTextureMemoryUsagedeqp::gls::LongStressCase::FeatureProbabilities243 		FeatureProbabilities& pWastefulTextureMemoryUsage	(const float prob) { wastefulTextureMemoryUsage		= prob; return *this; }
pWastefulBufferMemoryUsagedeqp::gls::LongStressCase::FeatureProbabilities244 		FeatureProbabilities& pWastefulBufferMemoryUsage	(const float prob) { wastefulBufferMemoryUsage		= prob; return *this; }
pClientMemoryAttributeDatadeqp::gls::LongStressCase::FeatureProbabilities245 		FeatureProbabilities& pClientMemoryAttributeData	(const float prob) { clientMemoryAttributeData		= prob; return *this; }
pClientMemoryIndexDatadeqp::gls::LongStressCase::FeatureProbabilities246 		FeatureProbabilities& pClientMemoryIndexData		(const float prob) { clientMemoryIndexData			= prob; return *this; }
pRandomBufferUploadTargetdeqp::gls::LongStressCase::FeatureProbabilities247 		FeatureProbabilities& pRandomBufferUploadTarget		(const float prob) { randomBufferUploadTarget		= prob; return *this; }
pRandomBufferUsagedeqp::gls::LongStressCase::FeatureProbabilities248 		FeatureProbabilities& pRandomBufferUsage			(const float prob) { randomBufferUsage				= prob; return *this; }
pUseDrawArraysdeqp::gls::LongStressCase::FeatureProbabilities249 		FeatureProbabilities& pUseDrawArrays				(const float prob) { useDrawArrays					= prob; return *this; }
pSeparateAttribBuffersdeqp::gls::LongStressCase::FeatureProbabilities250 		FeatureProbabilities& pSeparateAttribBuffers		(const float prob) { separateAttributeBuffers		= prob; return *this; }
251 
FeatureProbabilitiesdeqp::gls::LongStressCase::FeatureProbabilities252 		FeatureProbabilities (void)
253 			: rebuildProgram				(0.0f)
254 			, reuploadTexture				(0.0f)
255 			, reuploadBuffer				(0.0f)
256 			, reuploadWithTexImage			(0.0f)
257 			, reuploadWithBufferData		(0.0f)
258 			, deleteTexture					(0.0f)
259 			, deleteBuffer					(0.0f)
260 			, wastefulTextureMemoryUsage	(0.0f)
261 			, wastefulBufferMemoryUsage		(0.0f)
262 			, clientMemoryAttributeData		(0.0f)
263 			, clientMemoryIndexData			(0.0f)
264 			, randomBufferUploadTarget		(0.0f)
265 			, randomBufferUsage				(0.0f)
266 			, useDrawArrays					(0.0f)
267 			, separateAttributeBuffers		(0.0f)
268 		{
269 		}
270 	};
271 
272 															LongStressCase						(tcu::TestContext&						testCtx,
273 																								 const glu::RenderContext&				renderCtx,
274 																								 const char*							name,
275 																								 const char*							desc,
276 																								 int									maxTexMemoryUsageBytes, //!< Approximate upper bound on GL texture memory usage.
277 																								 int									maxBufMemoryUsageBytes, //!< Approximate upper bound on GL buffer memory usage.
278 																								 int									numDrawCallsPerIteration,
279 																								 int									numTrianglesPerDrawCall,
280 																								 const std::vector<ProgramContext>&		programContexts,
281 																								 const FeatureProbabilities&			probabilities,
282 																								 deUint32								indexBufferUsage,
283 																								 deUint32								attrBufferUsage,
284 																								 int									redundantBufferFactor = 1,
285 																								 bool									showDebugInfo = false);
286 
287 															~LongStressCase						(void);
288 
289 	void													init								(void);
290 	void													deinit								(void);
291 
292 	IterateResult											iterate								(void);
293 
294 private:
295 															LongStressCase						(const LongStressCase&);
296 	LongStressCase&											operator=							(const LongStressCase&);
297 
298 	const glu::RenderContext&								m_renderCtx;
299 	const int												m_maxTexMemoryUsageBytes;
300 	const int												m_maxBufMemoryUsageBytes;
301 	const int												m_numDrawCallsPerIteration;
302 	const int												m_numTrianglesPerDrawCall;
303 	const int												m_numVerticesPerDrawCall;
304 	const std::vector<ProgramContext>						m_programContexts;
305 	const FeatureProbabilities								m_probabilities;
306 	const deUint32											m_indexBufferUsage;
307 	const deUint32											m_attrBufferUsage;
308 	const int												m_redundantBufferFactor; //!< By what factor we allocate redundant buffers. Default is 1, i.e. no redundancy.
309 	const bool												m_showDebugInfo;
310 
311 	const int												m_numIterations;
312 	const bool												m_isGLES3;
313 
314 	int														m_currentIteration;
315 	deUint64												m_startTimeSeconds; //!< Set at beginning of first iteration.
316 	deUint64												m_lastLogTime;
317 	int														m_lastLogIteration;
318 	int														m_currentLogEntryNdx;
319 
320 	de::Random												m_rnd;
321 	LongStressCaseInternal::GLObjectManager<
322 		LongStressCaseInternal::Program>*					m_programs;
323 	LongStressCaseInternal::GLObjectManager<
324 		LongStressCaseInternal::Buffer>*					m_buffers;
325 	LongStressCaseInternal::GLObjectManager<
326 		LongStressCaseInternal::Texture>*					m_textures;
327 	std::vector<deUint16>									m_vertexIndices;
328 
329 	struct ProgramResources
330 	{
331 		std::vector<deUint8>							attrDataBuf;
332 		std::vector<int>								attrDataOffsets;
333 		std::vector<int>								attrDataSizes;
334 		std::vector<de::SharedPtr<tcu::TextureLevel> >	dummyTextures;
335 		std::string										shaderNameManglingSuffix;
336 	};
337 
338 	std::vector<ProgramResources>							m_programResources;
339 
340 	LongStressCaseInternal::DebugInfoRenderer*				m_debugInfoRenderer;
341 };
342 
343 
344 } // gls
345 } // deqp
346 
347 #endif // _GLSLONGSTRESSCASE_HPP
348