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