• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*-------------------------------------------------------------------------
2  * drawElements Quality Program Tester Core
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 Image comparison utilities.
22  *//*--------------------------------------------------------------------*/
23 
24 #include "tcuImageCompare.hpp"
25 #include "tcuSurface.hpp"
26 #include "tcuFuzzyImageCompare.hpp"
27 #include "tcuBilinearImageCompare.hpp"
28 #include "tcuTestLog.hpp"
29 #include "tcuVector.hpp"
30 #include "tcuVectorUtil.hpp"
31 #include "tcuRGBA.hpp"
32 #include "tcuTexture.hpp"
33 #include "tcuTextureUtil.hpp"
34 #include "tcuFloat.hpp"
35 
36 #include <string.h>
37 
38 namespace tcu
39 {
40 
41 namespace
42 {
43 
computeScaleAndBias(const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,tcu::Vec4 & scale,tcu::Vec4 & bias)44 void computeScaleAndBias (const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, tcu::Vec4& scale, tcu::Vec4& bias)
45 {
46 	Vec4 minVal;
47 	Vec4 maxVal;
48 	const float eps = 0.0001f;
49 
50 	{
51 		Vec4 refMin;
52 		Vec4 refMax;
53 		estimatePixelValueRange(reference, refMin, refMax);
54 
55 		minVal	= refMin;
56 		maxVal	= refMax;
57 	}
58 
59 	{
60 		Vec4 resMin;
61 		Vec4 resMax;
62 
63 		estimatePixelValueRange(result, resMin, resMax);
64 
65 		minVal[0] = de::min(minVal[0], resMin[0]);
66 		minVal[1] = de::min(minVal[1], resMin[1]);
67 		minVal[2] = de::min(minVal[2], resMin[2]);
68 		minVal[3] = de::min(minVal[3], resMin[3]);
69 
70 		maxVal[0] = de::max(maxVal[0], resMax[0]);
71 		maxVal[1] = de::max(maxVal[1], resMax[1]);
72 		maxVal[2] = de::max(maxVal[2], resMax[2]);
73 		maxVal[3] = de::max(maxVal[3], resMax[3]);
74 	}
75 
76 	for (int c = 0; c < 4; c++)
77 	{
78 		if (maxVal[c] - minVal[c] < eps)
79 		{
80 			scale[c]	= (maxVal[c] < eps) ? 1.0f : (1.0f / maxVal[c]);
81 			bias[c]		= (c == 3) ? (1.0f - maxVal[c]*scale[c]) : (0.0f - minVal[c]*scale[c]);
82 		}
83 		else
84 		{
85 			scale[c]	= 1.0f / (maxVal[c] - minVal[c]);
86 			bias[c]		= 0.0f - minVal[c]*scale[c];
87 		}
88 	}
89 }
90 
findNumPositionDeviationFailingPixels(const PixelBufferAccess & errorMask,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const UVec4 & threshold,const tcu::IVec3 & maxPositionDeviation,bool acceptOutOfBoundsAsAnyValue)91 static int findNumPositionDeviationFailingPixels (const PixelBufferAccess& errorMask, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const UVec4& threshold, const tcu::IVec3& maxPositionDeviation, bool acceptOutOfBoundsAsAnyValue)
92 {
93 	const tcu::IVec4	okColor				(0, 255, 0, 255);
94 	const tcu::IVec4	errorColor			(255, 0, 0, 255);
95 	const int			width				= reference.getWidth();
96 	const int			height				= reference.getHeight();
97 	const int			depth				= reference.getDepth();
98 	int					numFailingPixels	= 0;
99 
100 	// Accept pixels "sampling" over the image bounds pixels since "taps" could be anything
101 	const int			beginX				= (acceptOutOfBoundsAsAnyValue) ? (maxPositionDeviation.x()) : (0);
102 	const int			beginY				= (acceptOutOfBoundsAsAnyValue) ? (maxPositionDeviation.y()) : (0);
103 	const int			beginZ				= (acceptOutOfBoundsAsAnyValue) ? (maxPositionDeviation.z()) : (0);
104 	const int			endX				= (acceptOutOfBoundsAsAnyValue) ? (width  - maxPositionDeviation.x()) : (width);
105 	const int			endY				= (acceptOutOfBoundsAsAnyValue) ? (height - maxPositionDeviation.y()) : (height);
106 	const int			endZ				= (acceptOutOfBoundsAsAnyValue) ? (depth  - maxPositionDeviation.z()) : (depth);
107 
108 	TCU_CHECK_INTERNAL(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
109 	DE_ASSERT(endX > 0 && endY > 0 && endZ > 0);	// most likely a bug
110 
111 	tcu::clear(errorMask, okColor);
112 
113 	for (int z = beginZ; z < endZ; z++)
114 	{
115 		for (int y = beginY; y < endY; y++)
116 		{
117 			for (int x = beginX; x < endX; x++)
118 			{
119 				const IVec4	refPix = reference.getPixelInt(x, y, z);
120 				const IVec4	cmpPix = result.getPixelInt(x, y, z);
121 
122 				// Exact match
123 				{
124 					const UVec4	diff = abs(refPix - cmpPix).cast<deUint32>();
125 					const bool	isOk = boolAll(lessThanEqual(diff, threshold));
126 
127 					if (isOk)
128 						continue;
129 				}
130 
131 				// Find matching pixels for both result and reference pixel
132 
133 				{
134 					bool pixelFoundForReference = false;
135 
136 					// Find deviated result pixel for reference
137 
138 					for (int sz = de::max(0, z - maxPositionDeviation.z()); sz <= de::min(depth  - 1, z + maxPositionDeviation.z()) && !pixelFoundForReference; ++sz)
139 					for (int sy = de::max(0, y - maxPositionDeviation.y()); sy <= de::min(height - 1, y + maxPositionDeviation.y()) && !pixelFoundForReference; ++sy)
140 					for (int sx = de::max(0, x - maxPositionDeviation.x()); sx <= de::min(width  - 1, x + maxPositionDeviation.x()) && !pixelFoundForReference; ++sx)
141 					{
142 						const IVec4	deviatedCmpPix	= result.getPixelInt(sx, sy, sz);
143 						const UVec4	diff			= abs(refPix - deviatedCmpPix).cast<deUint32>();
144 						const bool	isOk			= boolAll(lessThanEqual(diff, threshold));
145 
146 						pixelFoundForReference		= isOk;
147 					}
148 
149 					if (!pixelFoundForReference)
150 					{
151 						errorMask.setPixel(errorColor, x, y, z);
152 						++numFailingPixels;
153 						continue;
154 					}
155 				}
156 				{
157 					bool pixelFoundForResult = false;
158 
159 					// Find deviated reference pixel for result
160 
161 					for (int sz = de::max(0, z - maxPositionDeviation.z()); sz <= de::min(depth  - 1, z + maxPositionDeviation.z()) && !pixelFoundForResult; ++sz)
162 					for (int sy = de::max(0, y - maxPositionDeviation.y()); sy <= de::min(height - 1, y + maxPositionDeviation.y()) && !pixelFoundForResult; ++sy)
163 					for (int sx = de::max(0, x - maxPositionDeviation.x()); sx <= de::min(width  - 1, x + maxPositionDeviation.x()) && !pixelFoundForResult; ++sx)
164 					{
165 						const IVec4	deviatedRefPix	= reference.getPixelInt(sx, sy, sz);
166 						const UVec4	diff			= abs(cmpPix - deviatedRefPix).cast<deUint32>();
167 						const bool	isOk			= boolAll(lessThanEqual(diff, threshold));
168 
169 						pixelFoundForResult			= isOk;
170 					}
171 
172 					if (!pixelFoundForResult)
173 					{
174 						errorMask.setPixel(errorColor, x, y, z);
175 						++numFailingPixels;
176 						continue;
177 					}
178 				}
179 			}
180 		}
181 	}
182 
183 	return numFailingPixels;
184 }
185 
186 } // anonymous
187 
188 /*--------------------------------------------------------------------*//*!
189  * \brief Fuzzy image comparison
190  *
191  * This image comparison is designed for comparing images rendered by 3D
192  * graphics APIs such as OpenGL. The comparison allows small local differences
193  * and compensates for aliasing.
194  *
195  * The algorithm first performs light blurring on both images and then
196  * does per-pixel analysis. Pixels are compared to 3x3 bilinear surface
197  * defined by adjecent pixels. This compensates for both 1-pixel deviations
198  * in geometry and aliasing in texture data.
199  *
200  * Error metric is computed based on the differences. On valid images the
201  * metric is usually <0.01. Thus good threshold values are in range 0.02 to
202  * 0.05.
203  *
204  * On failure error image is generated that shows where the failing pixels
205  * are.
206  *
207  * \note				Currently supports only UNORM_INT8 formats
208  * \param log			Test log for results
209  * \param imageSetName	Name for image set when logging results
210  * \param imageSetDesc	Description for image set
211  * \param reference		Reference image
212  * \param result		Result image
213  * \param threshold		Error metric threshold (good values are 0.02-0.05)
214  * \param logMode		Logging mode
215  * \return true if comparison passes, false otherwise
216  *//*--------------------------------------------------------------------*/
fuzzyCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,float threshold,CompareLogMode logMode)217 bool fuzzyCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, float threshold, CompareLogMode logMode)
218 {
219 	FuzzyCompareParams	params;		// Use defaults.
220 	TextureLevel		errorMask		(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), reference.getWidth(), reference.getHeight());
221 	float				difference		= fuzzyCompare(params, reference, result, errorMask.getAccess());
222 	bool				isOk			= difference <= threshold;
223 	Vec4				pixelBias		(0.0f, 0.0f, 0.0f, 0.0f);
224 	Vec4				pixelScale		(1.0f, 1.0f, 1.0f, 1.0f);
225 
226 	if (!isOk || logMode == COMPARE_LOG_EVERYTHING)
227 	{
228 		// Generate more accurate error mask.
229 		params.maxSampleSkip = 0;
230 		fuzzyCompare(params, reference, result, errorMask.getAccess());
231 
232 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8) && reference.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
233 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
234 
235 		if (!isOk)
236 			log << TestLog::Message << "Image comparison failed: difference = " << difference << ", threshold = " << threshold << TestLog::EndMessage;
237 
238 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
239 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
240 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
241 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
242 			<< TestLog::EndImageSet;
243 	}
244 	else if (logMode == COMPARE_LOG_RESULT)
245 	{
246 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
247 			computePixelScaleBias(result, pixelScale, pixelBias);
248 
249 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
250 			<< TestLog::Image("Result",		"Result",		result, pixelScale, pixelBias)
251 			<< TestLog::EndImageSet;
252 	}
253 
254 	return isOk;
255 }
256 
257 /*--------------------------------------------------------------------*//*!
258  * \brief Fuzzy image comparison
259  *
260  * This image comparison is designed for comparing images rendered by 3D
261  * graphics APIs such as OpenGL. The comparison allows small local differences
262  * and compensates for aliasing.
263  *
264  * The algorithm first performs light blurring on both images and then
265  * does per-pixel analysis. Pixels are compared to 3x3 bilinear surface
266  * defined by adjecent pixels. This compensates for both 1-pixel deviations
267  * in geometry and aliasing in texture data.
268  *
269  * Error metric is computed based on the differences. On valid images the
270  * metric is usually <0.01. Thus good threshold values are in range 0.02 to
271  * 0.05.
272  *
273  * On failure error image is generated that shows where the failing pixels
274  * are.
275  *
276  * \note				Currently supports only UNORM_INT8 formats
277  * \param log			Test log for results
278  * \param imageSetName	Name for image set when logging results
279  * \param imageSetDesc	Description for image set
280  * \param reference		Reference image
281  * \param result		Result image
282  * \param threshold		Error metric threshold (good values are 0.02-0.05)
283  * \param logMode		Logging mode
284  * \return true if comparison passes, false otherwise
285  *//*--------------------------------------------------------------------*/
fuzzyCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const Surface & reference,const Surface & result,float threshold,CompareLogMode logMode)286 bool fuzzyCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const Surface& reference, const Surface& result, float threshold, CompareLogMode logMode)
287 {
288 	return fuzzyCompare(log, imageSetName, imageSetDesc, reference.getAccess(), result.getAccess(), threshold, logMode);
289 }
290 
computeSquaredDiffSum(const ConstPixelBufferAccess & ref,const ConstPixelBufferAccess & cmp,const PixelBufferAccess & diffMask,int diffFactor)291 static deInt64 computeSquaredDiffSum (const ConstPixelBufferAccess& ref, const ConstPixelBufferAccess& cmp, const PixelBufferAccess& diffMask, int diffFactor)
292 {
293 	TCU_CHECK_INTERNAL(ref.getFormat().type == TextureFormat::UNORM_INT8 && cmp.getFormat().type == TextureFormat::UNORM_INT8);
294 	DE_ASSERT(ref.getWidth() == cmp.getWidth() && ref.getWidth() == diffMask.getWidth());
295 	DE_ASSERT(ref.getHeight() == cmp.getHeight() && ref.getHeight() == diffMask.getHeight());
296 
297 	deInt64 diffSum = 0;
298 
299 	for (int y = 0; y < cmp.getHeight(); y++)
300 	{
301 		for (int x = 0; x < cmp.getWidth(); x++)
302 		{
303 			IVec4	a		= ref.getPixelInt(x, y);
304 			IVec4	b		= cmp.getPixelInt(x, y);
305 			IVec4	diff	= abs(a - b);
306 			int		sum		= diff.x() + diff.y() + diff.z() + diff.w();
307 			int		sqSum	= diff.x()*diff.x() + diff.y()*diff.y() + diff.z()*diff.z() + diff.w()*diff.w();
308 
309 			diffMask.setPixel(tcu::RGBA(deClamp32(sum*diffFactor, 0, 255), deClamp32(255-sum*diffFactor, 0, 255), 0, 255).toVec(), x, y);
310 
311 			diffSum += (deInt64)sqSum;
312 		}
313 	}
314 
315 	return diffSum;
316 }
317 
318 /*--------------------------------------------------------------------*//*!
319  * \brief Per-pixel difference accuracy metric
320  *
321  * Computes accuracy metric using per-pixel differences between reference
322  * and result images.
323  *
324  * \note					Supports only integer- and fixed-point formats
325  * \param log				Test log for results
326  * \param imageSetName		Name for image set when logging results
327  * \param imageSetDesc		Description for image set
328  * \param reference			Reference image
329  * \param result			Result image
330  * \param bestScoreDiff		Scaling factor
331  * \param worstScoreDiff	Scaling factor
332  * \param logMode			Logging mode
333  * \return true if comparison passes, false otherwise
334  *//*--------------------------------------------------------------------*/
measurePixelDiffAccuracy(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,int bestScoreDiff,int worstScoreDiff,CompareLogMode logMode)335 int measurePixelDiffAccuracy (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, int bestScoreDiff, int worstScoreDiff, CompareLogMode logMode)
336 {
337 	TextureLevel	diffMask		(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), reference.getWidth(), reference.getHeight());
338 	int				diffFactor		= 8;
339 	deInt64			squaredSum		= computeSquaredDiffSum(reference, result, diffMask.getAccess(), diffFactor);
340 	float			sum				= deFloatSqrt((float)squaredSum);
341 	int				score			= deClamp32(deFloorFloatToInt32(100.0f - (de::max(sum-(float)bestScoreDiff, 0.0f) / (float)(worstScoreDiff-bestScoreDiff))*100.0f), 0, 100);
342 	const int		failThreshold	= 10;
343 	Vec4			pixelBias		(0.0f, 0.0f, 0.0f, 0.0f);
344 	Vec4			pixelScale		(1.0f, 1.0f, 1.0f, 1.0f);
345 
346 	if (logMode == COMPARE_LOG_EVERYTHING || score <= failThreshold)
347 	{
348 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8) && reference.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
349 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
350 
351 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
352 			<< TestLog::Image("Result",		"Result",			result,		pixelScale, pixelBias)
353 			<< TestLog::Image("Reference",	"Reference",		reference,	pixelScale, pixelBias)
354 			<< TestLog::Image("DiffMask",	"Difference",		diffMask)
355 			<< TestLog::EndImageSet;
356 	}
357 	else if (logMode == COMPARE_LOG_RESULT)
358 	{
359 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
360 			computePixelScaleBias(result, pixelScale, pixelBias);
361 
362 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
363 			<< TestLog::Image("Result",		"Result",			result,		pixelScale, pixelBias)
364 			<< TestLog::EndImageSet;
365 	}
366 
367 	if (logMode != COMPARE_LOG_ON_ERROR || score <= failThreshold)
368 		log << TestLog::Integer("DiffSum", "Squared difference sum", "", QP_KEY_TAG_NONE, squaredSum)
369 			<< TestLog::Integer("Score", "Score", "", QP_KEY_TAG_QUALITY, score);
370 
371 	return score;
372 }
373 
374 /*--------------------------------------------------------------------*//*!
375  * \brief Per-pixel difference accuracy metric
376  *
377  * Computes accuracy metric using per-pixel differences between reference
378  * and result images.
379  *
380  * \note					Supports only integer- and fixed-point formats
381  * \param log				Test log for results
382  * \param imageSetName		Name for image set when logging results
383  * \param imageSetDesc		Description for image set
384  * \param reference			Reference image
385  * \param result			Result image
386  * \param bestScoreDiff		Scaling factor
387  * \param worstScoreDiff	Scaling factor
388  * \param logMode			Logging mode
389  * \return true if comparison passes, false otherwise
390  *//*--------------------------------------------------------------------*/
measurePixelDiffAccuracy(TestLog & log,const char * imageSetName,const char * imageSetDesc,const Surface & reference,const Surface & result,int bestScoreDiff,int worstScoreDiff,CompareLogMode logMode)391 int measurePixelDiffAccuracy (TestLog& log, const char* imageSetName, const char* imageSetDesc, const Surface& reference, const Surface& result, int bestScoreDiff, int worstScoreDiff, CompareLogMode logMode)
392 {
393 	return measurePixelDiffAccuracy(log, imageSetName, imageSetDesc, reference.getAccess(), result.getAccess(), bestScoreDiff, worstScoreDiff, logMode);
394 }
395 
396 /*--------------------------------------------------------------------*//*!
397  * Returns the index of float in a float space without denormals
398  * so that:
399  * 1) f(0.0) = 0
400  * 2) f(-0.0) = 0
401  * 3) f(b) = f(a) + 1  <==>  b = nextAfter(a)
402  *
403  * See computeFloatFlushRelaxedULPDiff for details
404  *//*--------------------------------------------------------------------*/
getPositionOfIEEEFloatWithoutDenormals(float x)405 static deInt32 getPositionOfIEEEFloatWithoutDenormals (float x)
406 {
407 	DE_ASSERT(!deIsNaN(x)); // not sane
408 
409 	if (x == 0.0f)
410 		return 0;
411 	else if (x < 0.0f)
412 		return -getPositionOfIEEEFloatWithoutDenormals(-x);
413 	else
414 	{
415 		DE_ASSERT(x > 0.0f);
416 
417 		const tcu::Float32 f(x);
418 
419 		if (f.isDenorm())
420 		{
421 			// Denorms are flushed to zero
422 			return 0;
423 		}
424 		else
425 		{
426 			// sign is 0, and it's a normal number. Natural position is its bit
427 			// pattern but since we've collapsed the denorms, we must remove
428 			// the gap here too to keep the float enumeration continuous.
429 			//
430 			// Denormals occupy one exponent pattern. Removing one from
431 			// exponent should to the trick. Add one since the removed range
432 			// contained one representable value, 0.
433 			return (deInt32)(f.bits() - (1u << 23u) + 1u);
434 		}
435 	}
436 }
437 
computeFloatFlushRelaxedULPDiff(float a,float b)438 static deUint32 computeFloatFlushRelaxedULPDiff (float a, float b)
439 {
440 	if (deIsNaN(a) && deIsNaN(b))
441 		return 0;
442 	else if (deIsNaN(a) || deIsNaN(b))
443 	{
444 		return 0xFFFFFFFFu;
445 	}
446 	else
447 	{
448 		// Using the "definition 5" in Muller, Jean-Michel. "On the definition of ulp (x)" (2005)
449 		// assuming a floating point space is IEEE single precision floating point space without
450 		// denormals (and signed zeros).
451 		const deInt32 aIndex = getPositionOfIEEEFloatWithoutDenormals(a);
452 		const deInt32 bIndex = getPositionOfIEEEFloatWithoutDenormals(b);
453 		return (deUint32)de::abs(aIndex - bIndex);
454 	}
455 }
456 
computeFlushRelaxedULPDiff(const tcu::Vec4 & a,const tcu::Vec4 & b)457 static tcu::UVec4 computeFlushRelaxedULPDiff (const tcu::Vec4& a, const tcu::Vec4& b)
458 {
459 	return tcu::UVec4(computeFloatFlushRelaxedULPDiff(a.x(), b.x()),
460 					  computeFloatFlushRelaxedULPDiff(a.y(), b.y()),
461 					  computeFloatFlushRelaxedULPDiff(a.z(), b.z()),
462 					  computeFloatFlushRelaxedULPDiff(a.w(), b.w()));
463 }
464 
465 /*--------------------------------------------------------------------*//*!
466  * \brief Per-pixel threshold-based comparison
467  *
468  * This compare computes per-pixel differences between result and reference
469  * image. Comparison fails if any pixels exceed the given threshold value.
470  *
471  * This comparison uses ULP (units in last place) metric for computing the
472  * difference between floating-point values and thus this function can
473  * be used only for comparing floating-point texture data. In ULP calculation
474  * the denormal numbers are allowed to be flushed to zero.
475  *
476  * On failure error image is generated that shows where the failing pixels
477  * are.
478  *
479  * \param log			Test log for results
480  * \param imageSetName	Name for image set when logging results
481  * \param imageSetDesc	Description for image set
482  * \param reference		Reference image
483  * \param result		Result image
484  * \param threshold		Maximum allowed difference
485  * \param logMode		Logging mode
486  * \return true if comparison passes, false otherwise
487  *//*--------------------------------------------------------------------*/
floatUlpThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const UVec4 & threshold,CompareLogMode logMode)488 bool floatUlpThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const UVec4& threshold, CompareLogMode logMode)
489 {
490 	int					width				= reference.getWidth();
491 	int					height				= reference.getHeight();
492 	int					depth				= reference.getDepth();
493 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
494 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
495 	UVec4				maxDiff				(0, 0, 0, 0);
496 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
497 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
498 
499 	TCU_CHECK(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
500 
501 	for (int z = 0; z < depth; z++)
502 	{
503 		for (int y = 0; y < height; y++)
504 		{
505 			for (int x = 0; x < width; x++)
506 			{
507 				const Vec4	refPix	= reference.getPixel(x, y, z);
508 				const Vec4	cmpPix	= result.getPixel(x, y, z);
509 				const UVec4	diff	= computeFlushRelaxedULPDiff(refPix, cmpPix);
510 				const bool	isOk	= boolAll(lessThanEqual(diff, threshold));
511 
512 				maxDiff = max(maxDiff, diff);
513 
514 				errorMask.setPixel(isOk ? Vec4(0.0f, 1.0f, 0.0f, 1.0f) : Vec4(1.0f, 0.0f, 0.0f, 1.0f), x, y, z);
515 			}
516 		}
517 	}
518 
519 	bool compareOk = boolAll(lessThanEqual(maxDiff, threshold));
520 
521 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
522 	{
523 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
524 		if (tcu::getTextureChannelClass(reference.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
525 			tcu::getTextureChannelClass(result.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
526 		{
527 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
528 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
529 		}
530 
531 		if (!compareOk)
532 			log << TestLog::Message << "Image comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << TestLog::EndMessage;
533 
534 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
535 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
536 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
537 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
538 			<< TestLog::EndImageSet;
539 	}
540 	else if (logMode == COMPARE_LOG_RESULT)
541 	{
542 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
543 			computePixelScaleBias(result, pixelScale, pixelBias);
544 
545 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
546 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
547 			<< TestLog::EndImageSet;
548 	}
549 
550 	return compareOk;
551 }
552 
553 /*--------------------------------------------------------------------*//*!
554  * \brief Per-pixel threshold-based comparison
555  *
556  * This compare computes per-pixel differences between result and reference
557  * image. Comparison fails if any pixels exceed the given threshold value.
558  *
559  * This comparison can be used for floating-point and fixed-point formats.
560  * Difference is computed in floating-point space.
561  *
562  * On failure an error image is generated that shows where the failing
563  * pixels are.
564  *
565  * \param log			Test log for results
566  * \param imageSetName	Name for image set when logging results
567  * \param imageSetDesc	Description for image set
568  * \param reference		Reference image
569  * \param result		Result image
570  * \param threshold		Maximum allowed difference
571  * \param logMode		Logging mode
572  * \return true if comparison passes, false otherwise
573  *//*--------------------------------------------------------------------*/
floatThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const Vec4 & threshold,CompareLogMode logMode)574 bool floatThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const Vec4& threshold, CompareLogMode logMode)
575 {
576 	int					width				= reference.getWidth();
577 	int					height				= reference.getHeight();
578 	int					depth				= reference.getDepth();
579 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
580 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
581 	Vec4				maxDiff				(0.0f, 0.0f, 0.0f, 0.0f);
582 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
583 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
584 
585 	TCU_CHECK_INTERNAL(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
586 
587 	for (int z = 0; z < depth; z++)
588 	{
589 		for (int y = 0; y < height; y++)
590 		{
591 			for (int x = 0; x < width; x++)
592 			{
593 				Vec4	refPix		= reference.getPixel(x, y, z);
594 				Vec4	cmpPix		= result.getPixel(x, y, z);
595 
596 				Vec4	diff		= abs(refPix - cmpPix);
597 				bool	isOk		= boolAll(lessThanEqual(diff, threshold));
598 
599 				maxDiff = max(maxDiff, diff);
600 
601 				errorMask.setPixel(isOk ? Vec4(0.0f, 1.0f, 0.0f, 1.0f) : Vec4(1.0f, 0.0f, 0.0f, 1.0f), x, y, z);
602 			}
603 		}
604 	}
605 
606 	bool compareOk = boolAll(lessThanEqual(maxDiff, threshold));
607 
608 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
609 	{
610 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
611 		if (tcu::getTextureChannelClass(reference.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
612 			tcu::getTextureChannelClass(result.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
613 		{
614 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
615 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
616 		}
617 
618 		if (!compareOk)
619 			log << TestLog::Message << "Image comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << TestLog::EndMessage;
620 
621 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
622 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
623 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
624 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
625 			<< TestLog::EndImageSet;
626 	}
627 	else if (logMode == COMPARE_LOG_RESULT)
628 	{
629 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
630 			computePixelScaleBias(result, pixelScale, pixelBias);
631 
632 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
633 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
634 			<< TestLog::EndImageSet;
635 	}
636 
637 	return compareOk;
638 }
639 
640 /*--------------------------------------------------------------------*//*!
641  * \brief Per-pixel threshold-based comparison with ignore key
642  *
643  * This compare computes per-pixel differences between result and reference
644  * image. Comparison fails if any pixels exceed the given threshold value.
645  *
646  * Any pixels in reference that match the ignore key are ignored.
647  *
648  * This comparison can be used for floating-point and fixed-point formats.
649  * Difference is computed in floating-point space.
650  *
651  * On failure an error image is generated that shows where the failing
652  * pixels are.
653  *
654  * \param log			Test log for results
655  * \param imageSetName	Name for image set when logging results
656  * \param imageSetDesc	Description for image set
657  * \param reference		Reference image
658  * \param result		Result image
659  * \param ignorekey     Ignore key
660  * \param threshold		Maximum allowed difference
661  * \param logMode		Logging mode
662  * \return true if comparison passes, false otherwise
663  *//*--------------------------------------------------------------------*/
floatThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const Vec4 & ignorekey,const Vec4 & threshold,CompareLogMode logMode)664 bool floatThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const Vec4& ignorekey, const Vec4& threshold, CompareLogMode logMode)
665 {
666 	int					width = reference.getWidth();
667 	int					height = reference.getHeight();
668 	int					depth = reference.getDepth();
669 	TextureLevel		errorMaskStorage(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
670 	PixelBufferAccess	errorMask = errorMaskStorage.getAccess();
671 	Vec4				maxDiff(0.0f, 0.0f, 0.0f, 0.0f);
672 	Vec4				pixelBias(0.0f, 0.0f, 0.0f, 0.0f);
673 	Vec4				pixelScale(1.0f, 1.0f, 1.0f, 1.0f);
674 
675 	TCU_CHECK_INTERNAL(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
676 
677 	for (int z = 0; z < depth; z++)
678 	{
679 		for (int y = 0; y < height; y++)
680 		{
681 			for (int x = 0; x < width; x++)
682 			{
683 				Vec4	refPix = reference.getPixel(x, y, z);
684 				Vec4	cmpPix = result.getPixel(x, y, z);
685 
686 				if (refPix != ignorekey)
687 				{
688 
689 					Vec4	diff = abs(refPix - cmpPix);
690 					bool	isOk = boolAll(lessThanEqual(diff, threshold));
691 
692 					maxDiff = max(maxDiff, diff);
693 
694 					errorMask.setPixel(isOk ? Vec4(0.0f, 1.0f, 0.0f, 1.0f) : Vec4(1.0f, 0.0f, 0.0f, 1.0f), x, y, z);
695 				}
696 			}
697 		}
698 	}
699 
700 	bool compareOk = boolAll(lessThanEqual(maxDiff, threshold));
701 
702 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
703 	{
704 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
705 		if (tcu::getTextureChannelClass(reference.getFormat().type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
706 			tcu::getTextureChannelClass(result.getFormat().type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
707 		{
708 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
709 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
710 		}
711 
712 		if (!compareOk)
713 			log << TestLog::Message << "Image comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << TestLog::EndMessage;
714 
715 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
716 			<< TestLog::Image("Result", "Result", result, pixelScale, pixelBias)
717 			<< TestLog::Image("Reference", "Reference", reference, pixelScale, pixelBias)
718 			<< TestLog::Image("ErrorMask", "Error mask", errorMask)
719 			<< TestLog::EndImageSet;
720 	}
721 	else if (logMode == COMPARE_LOG_RESULT)
722 	{
723 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
724 			computePixelScaleBias(result, pixelScale, pixelBias);
725 
726 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
727 			<< TestLog::Image("Result", "Result", result, pixelScale, pixelBias)
728 			<< TestLog::EndImageSet;
729 	}
730 
731 	return compareOk;
732 }
733 
734 /*--------------------------------------------------------------------*//*!
735  * \brief Per-pixel threshold-based comparison
736  *
737  * This compare computes per-pixel differences between result and reference
738  * color. Comparison fails if any pixels exceed the given threshold value.
739  *
740  * This comparison can be used for floating-point and fixed-point formats.
741  * Difference is computed in floating-point space.
742  *
743  * On failure an error image is generated that shows where the failing
744  * pixels are.
745  *
746  * \param log			Test log for results
747  * \param imageSetName	Name for image set when logging results
748  * \param imageSetDesc	Description for image set
749  * \param reference		Reference color
750  * \param result		Result image
751  * \param threshold		Maximum allowed difference
752  * \param logMode		Logging mode
753  * \return true if comparison passes, false otherwise
754  *//*--------------------------------------------------------------------*/
floatThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const Vec4 & reference,const ConstPixelBufferAccess & result,const Vec4 & threshold,CompareLogMode logMode)755 bool floatThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const Vec4& reference, const ConstPixelBufferAccess& result, const Vec4& threshold, CompareLogMode logMode)
756 {
757 	const int			width				= result.getWidth();
758 	const int			height				= result.getHeight();
759 	const int			depth				= result.getDepth();
760 
761 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
762 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
763 	Vec4				maxDiff				(0.0f, 0.0f, 0.0f, 0.0f);
764 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
765 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
766 
767 	for (int z = 0; z < depth; z++)
768 	{
769 		for (int y = 0; y < height; y++)
770 		{
771 			for (int x = 0; x < width; x++)
772 			{
773 				const Vec4	cmpPix		= result.getPixel(x, y, z);
774 				const Vec4	diff		= abs(reference - cmpPix);
775 				const bool	isOk		= boolAll(lessThanEqual(diff, threshold));
776 
777 				maxDiff = max(maxDiff, diff);
778 
779 				errorMask.setPixel(isOk ? Vec4(0.0f, 1.0f, 0.0f, 1.0f) : Vec4(1.0f, 0.0f, 0.0f, 1.0f), x, y, z);
780 			}
781 		}
782 	}
783 
784 	bool compareOk = boolAll(lessThanEqual(maxDiff, threshold));
785 
786 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
787 	{
788 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
789 		if (tcu::getTextureChannelClass(result.getFormat().type) != tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
790 		{
791 			computeScaleAndBias(result, result, pixelScale, pixelBias);
792 			log << TestLog::Message << "Result image is normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
793 		}
794 
795 		if (!compareOk)
796 			log << TestLog::Message << "Image comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << ", reference = " << reference << TestLog::EndMessage;
797 
798 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
799 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
800 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
801 			<< TestLog::EndImageSet;
802 	}
803 	else if (logMode == COMPARE_LOG_RESULT)
804 	{
805 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
806 			computePixelScaleBias(result, pixelScale, pixelBias);
807 
808 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
809 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
810 			<< TestLog::EndImageSet;
811 	}
812 
813 	return compareOk;
814 }
815 
816 /*--------------------------------------------------------------------*//*!
817  * \brief Per-pixel threshold-based comparison
818  *
819  * This compare computes per-pixel differences between result and reference
820  * image. Comparison fails if any pixels exceed the given threshold value.
821  *
822  * This comparison can be used for integer- and fixed-point texture formats.
823  * Difference is computed in integer space.
824  *
825  * On failure error image is generated that shows where the failing pixels
826  * are.
827  *
828  * \param log			Test log for results
829  * \param imageSetName	Name for image set when logging results
830  * \param imageSetDesc	Description for image set
831  * \param reference		Reference image
832  * \param result		Result image
833  * \param threshold		Maximum allowed difference
834  * \param logMode		Logging mode
835  * \param use64Bits		Use 64-bit components when reading image data.
836  * \return true if comparison passes, false otherwise
837  *//*--------------------------------------------------------------------*/
intThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const UVec4 & threshold,CompareLogMode logMode,bool use64Bits)838 bool intThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const UVec4& threshold, CompareLogMode logMode, bool use64Bits)
839 {
840 	int					width				= reference.getWidth();
841 	int					height				= reference.getHeight();
842 	int					depth				= reference.getDepth();
843 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
844 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
845 	U64Vec4				maxDiff				(0u, 0u, 0u, 0u);
846 	U64Vec4				diff				(0u, 0u, 0u, 0u);
847 	const U64Vec4		threshold64			= threshold.cast<deUint64>();
848 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
849 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
850 
851 	TCU_CHECK_INTERNAL(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
852 
853 	for (int z = 0; z < depth; z++)
854 	{
855 		for (int y = 0; y < height; y++)
856 		{
857 			for (int x = 0; x < width; x++)
858 			{
859 				if (use64Bits)
860 				{
861 					I64Vec4	refPix	= reference.getPixelInt64(x, y, z);
862 					I64Vec4	cmpPix	= result.getPixelInt64(x, y, z);
863 					diff			= abs(refPix - cmpPix).cast<deUint64>();
864 				}
865 				else
866 				{
867 					IVec4	refPix	= reference.getPixelInt(x, y, z);
868 					IVec4	cmpPix	= result.getPixelInt(x, y, z);
869 					diff			= abs(refPix - cmpPix).cast<deUint64>();
870 				}
871 
872 				maxDiff = max(maxDiff, diff);
873 
874 				const bool isOk = boolAll(lessThanEqual(diff, threshold64));
875 				errorMask.setPixel(isOk ? IVec4(0, 0xff, 0, 0xff) : IVec4(0xff, 0, 0, 0xff), x, y, z);
876 			}
877 		}
878 	}
879 
880 	bool compareOk = boolAll(lessThanEqual(maxDiff, threshold64));
881 
882 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
883 	{
884 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
885 		if (tcu::getTextureChannelClass(reference.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
886 			tcu::getTextureChannelClass(result.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
887 		{
888 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
889 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
890 		}
891 
892 		if (!compareOk)
893 			log << TestLog::Message << "Image comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << TestLog::EndMessage;
894 
895 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
896 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
897 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
898 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
899 			<< TestLog::EndImageSet;
900 	}
901 	else if (logMode == COMPARE_LOG_RESULT)
902 	{
903 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
904 			computePixelScaleBias(result, pixelScale, pixelBias);
905 
906 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
907 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
908 			<< TestLog::EndImageSet;
909 	}
910 
911 	return compareOk;
912 }
913 
914 /*--------------------------------------------------------------------*//*!
915  * \brief Per-pixel depth/stencil threshold-based comparison
916  *
917  * This compare computes per-pixel differences between result and reference
918  * image. Comparison fails if any pixels exceed the given threshold value.
919  *
920  * This comparison can be used for depth and depth/stencil images.
921  * Difference is computed in integer space.
922  *
923  * On failure error image is generated that shows where the failing pixels
924  * are.
925  *
926  * \param log			Test log for results
927  * \param imageSetName	Name for image set when logging results
928  * \param imageSetDesc	Description for image set
929  * \param reference		Reference image
930  * \param result		Result image
931  * \param threshold		Maximum allowed depth difference (stencil must be exact)
932  * \param logMode		Logging mode
933  * \return true if comparison passes, false otherwise
934  *//*--------------------------------------------------------------------*/
dsThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const float threshold,CompareLogMode logMode)935 bool dsThresholdCompare(TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const float threshold, CompareLogMode logMode)
936 {
937 	int					width = reference.getWidth();
938 	int					height = reference.getHeight();
939 	int					depth = reference.getDepth();
940 	TextureLevel		errorMaskStorage(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
941 	PixelBufferAccess	errorMask = errorMaskStorage.getAccess();
942 	float				maxDiff = 0.0;
943 	bool				allStencilOk = true;
944 	bool				hasDepth = tcu::hasDepthComponent(result.getFormat().order);
945 	bool				hasStencil = tcu::hasStencilComponent(result.getFormat().order);
946 
947 	TCU_CHECK_INTERNAL(result.getWidth() == width && result.getHeight() == height && result.getDepth() == depth);
948 
949 	for (int z = 0; z < depth; z++)
950 	{
951 		for (int y = 0; y < height; y++)
952 		{
953 			for (int x = 0; x < width; x++)
954 			{
955 				bool	isOk = true;
956 
957 				if (hasDepth)
958 				{
959 					float refDepth	= reference.getPixDepth(x, y, z);
960 					float cmpDepth	= result.getPixDepth(x, y, z);
961 					float diff		= de::abs(refDepth - cmpDepth);
962 
963 					isOk = diff <= threshold;
964 					maxDiff = (float) deMax(maxDiff, diff);
965 				}
966 
967 				if (hasStencil)
968 				{
969 					deUint8 refStencil = (deUint8) reference.getPixStencil(x, y, z);
970 					deUint8 cmpStencil = (deUint8) result.getPixStencil(x, y, z);
971 
972 					bool isStencilOk = (refStencil == cmpStencil);
973 					allStencilOk = allStencilOk && isStencilOk;
974 					isOk = isOk && isStencilOk;
975 				}
976 
977 				errorMask.setPixel(isOk ? IVec4(0, 0xff, 0, 0xff) : IVec4(0xff, 0, 0, 0xff), x, y, z);
978 			}
979 		}
980 	}
981 
982 	bool compareOk = (maxDiff <= threshold) && allStencilOk;
983 
984 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
985 	{
986 		if (!compareOk)
987 		{
988 			if (maxDiff > threshold)
989 				log << TestLog::Message << "Depth comparison failed: max difference = " << maxDiff << ", threshold = " << threshold << TestLog::EndMessage;
990 			if (!allStencilOk)
991 				log << TestLog::Message << "Stencil comparison failed" << TestLog::EndMessage;
992 		}
993 
994 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
995 			// TODO: Convert depth/stencil buffers into separate depth & stencil for logging?
996 //			<< TestLog::Image("Result", "Result", result, pixelScale, pixelBias)
997 //			<< TestLog::Image("Reference", "Reference", reference, pixelScale, pixelBias)
998 			<< TestLog::Image("ErrorMask", "Error mask", errorMask)
999 			<< TestLog::EndImageSet;
1000 	}
1001 	else if (logMode == COMPARE_LOG_RESULT)
1002 	{
1003 #if 0
1004 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
1005 			computePixelScaleBias(result, pixelScale, pixelBias);
1006 
1007 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1008 			<< TestLog::Image("Result", "Result", result, pixelScale, pixelBias)
1009 			<< TestLog::EndImageSet;
1010 #endif
1011 	}
1012 
1013 	return compareOk;
1014 }
1015 
1016 /*--------------------------------------------------------------------*//*!
1017  * \brief Per-pixel threshold-based deviation-ignoring comparison
1018  *
1019  * This compare computes per-pixel differences between result and reference
1020  * image. Comparison fails if there is no pixel matching the given threshold
1021  * value in the search volume.
1022  *
1023  * If the search volume contains out-of-bounds pixels, comparison can be set
1024  * to either ignore these pixels in search or to accept any pixel that has
1025  * out-of-bounds pixels in its search volume.
1026  *
1027  * This comparison can be used for integer- and fixed-point texture formats.
1028  * Difference is computed in integer space.
1029  *
1030  * On failure error image is generated that shows where the failing pixels
1031  * are.
1032  *
1033  * \param log							Test log for results
1034  * \param imageSetName					Name for image set when logging results
1035  * \param imageSetDesc					Description for image set
1036  * \param reference						Reference image
1037  * \param result						Result image
1038  * \param threshold						Maximum allowed difference
1039  * \param maxPositionDeviation			Maximum allowed distance in the search
1040  *										volume.
1041  * \param acceptOutOfBoundsAsAnyValue	Accept any pixel in the boundary region
1042  * \param logMode						Logging mode
1043  * \return true if comparison passes, false otherwise
1044  *//*--------------------------------------------------------------------*/
intThresholdPositionDeviationCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const UVec4 & threshold,const tcu::IVec3 & maxPositionDeviation,bool acceptOutOfBoundsAsAnyValue,CompareLogMode logMode)1045 bool intThresholdPositionDeviationCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const UVec4& threshold, const tcu::IVec3& maxPositionDeviation, bool acceptOutOfBoundsAsAnyValue, CompareLogMode logMode)
1046 {
1047 	const int			width				= reference.getWidth();
1048 	const int			height				= reference.getHeight();
1049 	const int			depth				= reference.getDepth();
1050 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
1051 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
1052 	const int			numFailingPixels	= findNumPositionDeviationFailingPixels(errorMask, reference, result, threshold, maxPositionDeviation, acceptOutOfBoundsAsAnyValue);
1053 	const bool			compareOk			= numFailingPixels == 0;
1054 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
1055 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
1056 
1057 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
1058 	{
1059 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
1060 		if (tcu::getTextureChannelClass(reference.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
1061 			tcu::getTextureChannelClass(result.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
1062 		{
1063 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
1064 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
1065 		}
1066 
1067 		if (!compareOk)
1068 			log	<< TestLog::Message
1069 				<< "Image comparison failed:\n"
1070 				<< "\tallowed position deviation = " << maxPositionDeviation << "\n"
1071 				<< "\tcolor threshold = " << threshold
1072 				<< TestLog::EndMessage;
1073 
1074 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1075 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
1076 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
1077 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
1078 			<< TestLog::EndImageSet;
1079 	}
1080 	else if (logMode == COMPARE_LOG_RESULT)
1081 	{
1082 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
1083 			computePixelScaleBias(result, pixelScale, pixelBias);
1084 
1085 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1086 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
1087 			<< TestLog::EndImageSet;
1088 	}
1089 
1090 	return compareOk;
1091 }
1092 
1093 /*--------------------------------------------------------------------*//*!
1094  * \brief Per-pixel threshold-based deviation-ignoring comparison
1095  *
1096  * This compare computes per-pixel differences between result and reference
1097  * image. Pixel fails the test if there is no pixel matching the given
1098  * threshold value in the search volume. Comparison fails if the number of
1099  * failing pixels exceeds the given limit.
1100  *
1101  * If the search volume contains out-of-bounds pixels, comparison can be set
1102  * to either ignore these pixels in search or to accept any pixel that has
1103  * out-of-bounds pixels in its search volume.
1104  *
1105  * This comparison can be used for integer- and fixed-point texture formats.
1106  * Difference is computed in integer space.
1107  *
1108  * On failure error image is generated that shows where the failing pixels
1109  * are.
1110  *
1111  * \param log							Test log for results
1112  * \param imageSetName					Name for image set when logging results
1113  * \param imageSetDesc					Description for image set
1114  * \param reference						Reference image
1115  * \param result						Result image
1116  * \param threshold						Maximum allowed difference
1117  * \param maxPositionDeviation			Maximum allowed distance in the search
1118  *										volume.
1119  * \param acceptOutOfBoundsAsAnyValue	Accept any pixel in the boundary region
1120  * \param maxAllowedFailingPixels		Maximum number of failing pixels
1121  * \param logMode						Logging mode
1122  * \return true if comparison passes, false otherwise
1123  *//*--------------------------------------------------------------------*/
intThresholdPositionDeviationErrorThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const UVec4 & threshold,const tcu::IVec3 & maxPositionDeviation,bool acceptOutOfBoundsAsAnyValue,int maxAllowedFailingPixels,CompareLogMode logMode)1124 bool intThresholdPositionDeviationErrorThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const UVec4& threshold, const tcu::IVec3& maxPositionDeviation, bool acceptOutOfBoundsAsAnyValue, int maxAllowedFailingPixels, CompareLogMode logMode)
1125 {
1126 	const int			width				= reference.getWidth();
1127 	const int			height				= reference.getHeight();
1128 	const int			depth				= reference.getDepth();
1129 	TextureLevel		errorMaskStorage	(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), width, height, depth);
1130 	PixelBufferAccess	errorMask			= errorMaskStorage.getAccess();
1131 	const int			numFailingPixels	= findNumPositionDeviationFailingPixels(errorMask, reference, result, threshold, maxPositionDeviation, acceptOutOfBoundsAsAnyValue);
1132 	const bool			compareOk			= numFailingPixels <= maxAllowedFailingPixels;
1133 	Vec4				pixelBias			(0.0f, 0.0f, 0.0f, 0.0f);
1134 	Vec4				pixelScale			(1.0f, 1.0f, 1.0f, 1.0f);
1135 
1136 	if (!compareOk || logMode == COMPARE_LOG_EVERYTHING)
1137 	{
1138 		// All formats except normalized unsigned fixed point ones need remapping in order to fit into unorm channels in logged images.
1139 		if (tcu::getTextureChannelClass(reference.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT ||
1140 			tcu::getTextureChannelClass(result.getFormat().type)	!= tcu::TEXTURECHANNELCLASS_UNSIGNED_FIXED_POINT)
1141 		{
1142 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
1143 			log << TestLog::Message << "Result and reference images are normalized with formula p * " << pixelScale << " + " << pixelBias << TestLog::EndMessage;
1144 		}
1145 
1146 		if (!compareOk)
1147 			log	<< TestLog::Message
1148 				<< "Image comparison failed:\n"
1149 				<< "\tallowed position deviation = " << maxPositionDeviation << "\n"
1150 				<< "\tcolor threshold = " << threshold
1151 				<< TestLog::EndMessage;
1152 		log << TestLog::Message << "Number of failing pixels = " << numFailingPixels << ", max allowed = " << maxAllowedFailingPixels << TestLog::EndMessage;
1153 
1154 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1155 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
1156 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
1157 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
1158 			<< TestLog::EndImageSet;
1159 	}
1160 	else if (logMode == COMPARE_LOG_RESULT)
1161 	{
1162 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
1163 			computePixelScaleBias(result, pixelScale, pixelBias);
1164 
1165 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1166 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
1167 			<< TestLog::EndImageSet;
1168 	}
1169 
1170 	return compareOk;
1171 }
1172 
1173 /*--------------------------------------------------------------------*//*!
1174  * \brief Per-pixel threshold-based comparison
1175  *
1176  * This compare computes per-pixel differences between result and reference
1177  * image. Comparison fails if any pixels exceed the given threshold value.
1178  *
1179  * On failure error image is generated that shows where the failing pixels
1180  * are.
1181  *
1182  * \param log			Test log for results
1183  * \param imageSetName	Name for image set when logging results
1184  * \param imageSetDesc	Description for image set
1185  * \param reference		Reference image
1186  * \param result		Result image
1187  * \param threshold		Maximum allowed difference
1188  * \param logMode		Logging mode
1189  * \return true if comparison passes, false otherwise
1190  *//*--------------------------------------------------------------------*/
pixelThresholdCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const Surface & reference,const Surface & result,const RGBA & threshold,CompareLogMode logMode)1191 bool pixelThresholdCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const Surface& reference, const Surface& result, const RGBA& threshold, CompareLogMode logMode)
1192 {
1193 	return intThresholdCompare(log, imageSetName, imageSetDesc, reference.getAccess(), result.getAccess(), threshold.toIVec().cast<deUint32>(), logMode);
1194 }
1195 
1196 /*--------------------------------------------------------------------*//*!
1197  * \brief Bilinear image comparison
1198  *
1199  * \todo [pyry] Describe
1200  *
1201  * On failure error image is generated that shows where the failing pixels
1202  * are.
1203  *
1204  * \note				Currently supports only RGBA, UNORM_INT8 formats
1205  * \param log			Test log for results
1206  * \param imageSetName	Name for image set when logging results
1207  * \param imageSetDesc	Description for image set
1208  * \param reference		Reference image
1209  * \param result		Result image
1210  * \param threshold		Maximum local difference
1211  * \param logMode		Logging mode
1212  * \return true if comparison passes, false otherwise
1213  *//*--------------------------------------------------------------------*/
bilinearCompare(TestLog & log,const char * imageSetName,const char * imageSetDesc,const ConstPixelBufferAccess & reference,const ConstPixelBufferAccess & result,const RGBA threshold,CompareLogMode logMode)1214 bool bilinearCompare (TestLog& log, const char* imageSetName, const char* imageSetDesc, const ConstPixelBufferAccess& reference, const ConstPixelBufferAccess& result, const RGBA threshold, CompareLogMode logMode)
1215 {
1216 	TextureLevel		errorMask		(TextureFormat(TextureFormat::RGB, TextureFormat::UNORM_INT8), reference.getWidth(), reference.getHeight());
1217 	bool				isOk			= bilinearCompare(reference, result, errorMask, threshold);
1218 	Vec4				pixelBias		(0.0f, 0.0f, 0.0f, 0.0f);
1219 	Vec4				pixelScale		(1.0f, 1.0f, 1.0f, 1.0f);
1220 
1221 	if (!isOk || logMode == COMPARE_LOG_EVERYTHING)
1222 	{
1223 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8) && reference.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
1224 			computeScaleAndBias(reference, result, pixelScale, pixelBias);
1225 
1226 		if (!isOk)
1227 			log << TestLog::Message << "Image comparison failed, threshold = " << threshold << TestLog::EndMessage;
1228 
1229 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1230 			<< TestLog::Image("Result",		"Result",		result,		pixelScale, pixelBias)
1231 			<< TestLog::Image("Reference",	"Reference",	reference,	pixelScale, pixelBias)
1232 			<< TestLog::Image("ErrorMask",	"Error mask",	errorMask)
1233 			<< TestLog::EndImageSet;
1234 	}
1235 	else if (logMode == COMPARE_LOG_RESULT)
1236 	{
1237 		if (result.getFormat() != TextureFormat(TextureFormat::RGBA, TextureFormat::UNORM_INT8))
1238 			computePixelScaleBias(result, pixelScale, pixelBias);
1239 
1240 		log << TestLog::ImageSet(imageSetName, imageSetDesc)
1241 			<< TestLog::Image("Result",		"Result",		result, pixelScale, pixelBias)
1242 			<< TestLog::EndImageSet;
1243 	}
1244 
1245 	return isOk;
1246 }
1247 
1248 } // tcu
1249