/* * Copyright (c) Contributors to the Open 3D Engine Project. * For complete copyright and license terms please see the LICENSE at the root of this distribution. * * SPDX-License-Identifier: Apache-2.0 OR MIT * */ #include #include namespace UnitTest { using namespace AZ::Utils; class ImageComparisonTests : public AllocatorsFixture { protected: static const AZ::RHI::Format DefaultFormat = AZ::RHI::Format::R8G8B8A8_UNORM; static constexpr size_t BytesPerPixel = 4; AZStd::vector CreateTestRGBAImageData(AZ::RHI::Size size) { AZStd::vector buffer; const size_t bufferSize = BytesPerPixel * size.m_width * size.m_height; buffer.reserve(bufferSize); for (uint32_t i = 0; i < bufferSize; ++i) { buffer.push_back(aznumeric_cast(i % 255)); } return buffer; } void SetPixel(AZStd::vector& image, size_t pixelIndex, uint8_t r, uint8_t g, uint8_t b) { image[pixelIndex * BytesPerPixel + 0] = r; image[pixelIndex * BytesPerPixel + 1] = g; image[pixelIndex * BytesPerPixel + 2] = b; }; }; TEST_F(ImageComparisonTests, Error_ImageSizesDontMatch) { AZ::RHI::Size sizeA{1, 1, 1}; AZ::RHI::Size sizeB{1, 2, 1}; float diffScore = -1.0f; auto result = CalcImageDiffRms( CreateTestRGBAImageData(sizeA), sizeA, DefaultFormat, CreateTestRGBAImageData(sizeB), sizeB, DefaultFormat, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::SizeMismatch); } TEST_F(ImageComparisonTests, Error_BufferSizeDoesntMatchImageSize) { AZ::RHI::Size size{1, 1, 1}; AZ::RHI::Size wrongSize{1, 2, 1}; float diffScore = -1.0f; auto result = CalcImageDiffRms( CreateTestRGBAImageData(size), wrongSize, DefaultFormat, CreateTestRGBAImageData(size), wrongSize, DefaultFormat, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::SizeMismatch); } TEST_F(ImageComparisonTests, Error_UnsupportedFormat) { AZ::RHI::Format format = AZ::RHI::Format::G8R8_G8B8_UNORM; AZ::RHI::Size size{1, 1, 1}; float diffScore = -1.0f; auto result = CalcImageDiffRms( CreateTestRGBAImageData(size), size, format, CreateTestRGBAImageData(size), size, format, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::UnsupportedFormat); } TEST_F(ImageComparisonTests, Error_FormatsDontMatch) { AZ::RHI::Size size{1, 1, 1}; float diffScore = -1.0f; auto result = CalcImageDiffRms( CreateTestRGBAImageData(size), size, AZ::RHI::Format::R8G8B8A8_SNORM, CreateTestRGBAImageData(size), size, AZ::RHI::Format::R8G8B8A8_UNORM, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::FormatMismatch); } TEST_F(ImageComparisonTests, CheckThreshold_SmallIdenticalImages) { AZ::RHI::Size size{16, 9, 1}; float diffScore = -1.0f; auto result = CalcImageDiffRms( CreateTestRGBAImageData(size), size, DefaultFormat, CreateTestRGBAImageData(size), size, DefaultFormat, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::Success); EXPECT_EQ(0.0f, diffScore); } TEST_F(ImageComparisonTests, CheckThreshold_LargeIdenticalImages) { AZ::RHI::Size size{1620, 1080, 1}; AZStd::vector imageA = CreateTestRGBAImageData(size); AZStd::vector imageB = imageA; float diffScore = -1.0f; auto result = CalcImageDiffRms( imageA, size, DefaultFormat, imageB, size, DefaultFormat, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::Success); EXPECT_EQ(0.0f, diffScore); } TEST_F(ImageComparisonTests, CheckMaxChannelDifference) { const AZStd::vector imageA = { 255, 255, 255 }; const AZStd::vector imageB = { 0, 125, 255 }; const int16_t maxChannelDiff = 255; const int16_t res = CalcMaxChannelDifference(imageA, imageB, 0); EXPECT_EQ(res, maxChannelDiff); } TEST_F(ImageComparisonTests, CheckThreshold_SmallImagesWithDifferences) { AZ::RHI::Size size{2, 2, 1}; AZStd::vector imageA = CreateTestRGBAImageData(size); AZStd::vector imageB = CreateTestRGBAImageData(size); // Difference of 1 (R) SetPixel(imageA, 0, 100, 200, 5); SetPixel(imageB, 0, 101, 200, 5); // Difference of 2 (G) SetPixel(imageA, 1, 255, 255, 255); SetPixel(imageB, 1, 255, 253, 255); // Difference of 5 (B) SetPixel(imageA, 2, 0, 0, 0); SetPixel(imageB, 2, 0, 0, 5); // Difference of 100 (RGB all different) SetPixel(imageA, 3, 100, 100, 100); SetPixel(imageB, 3, 101, 102, 0); float diffScore = -1.0f; auto result = CalcImageDiffRms( imageA, size, DefaultFormat, imageB, size, DefaultFormat, &diffScore); EXPECT_EQ(result, ImageDiffResultCode::Success); // Result should be: // sqrt( (1^2 + 2^2 + 5^2 + 100^2) / (255.0^2) / 4 ) EXPECT_FLOAT_EQ(0.19637232876f, diffScore); } TEST_F(ImageComparisonTests, CheckThreshold_IgnoreImperceptibleDifferences) { AZ::RHI::Size size{2, 2, 1}; AZStd::vector imageA = CreateTestRGBAImageData(size); AZStd::vector imageB = CreateTestRGBAImageData(size); // Difference of 1 (R) SetPixel(imageA, 0, 100, 200, 5); SetPixel(imageB, 0, 101, 200, 5); // Difference of 2 (G) SetPixel(imageA, 1, 255, 255, 255); SetPixel(imageB, 1, 255, 253, 255); // Difference of 5 (B) SetPixel(imageA, 2, 0, 0, 0); SetPixel(imageB, 2, 0, 0, 5); // Difference of 4 (RGB all different) SetPixel(imageA, 3, 100, 100, 100); SetPixel(imageB, 3, 101, 102, 96); const float minDiffFilter = 3.9f / 255.0f; float diffScore = -1.0f; float filteredDiffScore = -1.0f; auto result = CalcImageDiffRms( imageA, size, DefaultFormat, imageB, size, DefaultFormat, &diffScore, &filteredDiffScore, minDiffFilter); EXPECT_EQ(result, ImageDiffResultCode::Success); // Result should be: // sqrt( (1^2 + 2^2 + 5^2 + 4^2) / (255.0^2) / 4 ) EXPECT_FLOAT_EQ(0.01329868624, diffScore); // Result should be: // sqrt( (5^2 + 4^2) / (255.0^2) / 4 ) EXPECT_FLOAT_EQ(0.01255514556f, filteredDiffScore); } }