aboutsummaryrefslogtreecommitdiff
path: root/src/dsp/distance_weighted_blend_test.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/dsp/distance_weighted_blend_test.cc')
-rw-r--r--src/dsp/distance_weighted_blend_test.cc324
1 files changed, 324 insertions, 0 deletions
diff --git a/src/dsp/distance_weighted_blend_test.cc b/src/dsp/distance_weighted_blend_test.cc
new file mode 100644
index 0000000..b3f3a2e
--- /dev/null
+++ b/src/dsp/distance_weighted_blend_test.cc
@@ -0,0 +1,324 @@
+// Copyright 2020 The libgav1 Authors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "src/dsp/distance_weighted_blend.h"
+
+#include <cstdint>
+#include <ostream>
+#include <string>
+#include <type_traits>
+
+#include "absl/strings/match.h"
+#include "absl/strings/str_format.h"
+#include "absl/strings/string_view.h"
+#include "absl/time/clock.h"
+#include "absl/time/time.h"
+#include "gtest/gtest.h"
+#include "src/dsp/constants.h"
+#include "src/dsp/dsp.h"
+#include "src/utils/common.h"
+#include "src/utils/constants.h"
+#include "src/utils/cpu.h"
+#include "src/utils/memory.h"
+#include "tests/third_party/libvpx/acm_random.h"
+#include "tests/utils.h"
+
+namespace libgav1 {
+namespace dsp {
+namespace {
+
+constexpr int kNumSpeedTests = 500000;
+
+constexpr int kQuantizedDistanceLookup[4][2] = {
+ {9, 7}, {11, 5}, {12, 4}, {13, 3}};
+
+struct TestParam {
+ TestParam(int width, int height) : width(width), height(height) {}
+ int width;
+ int height;
+};
+
+std::ostream& operator<<(std::ostream& os, const TestParam& param) {
+ return os << "BlockSize" << param.width << "x" << param.height;
+}
+
+template <int bitdepth, typename Pixel>
+class DistanceWeightedBlendTest : public testing::TestWithParam<TestParam>,
+ public test_utils::MaxAlignedAllocable {
+ public:
+ DistanceWeightedBlendTest() = default;
+ ~DistanceWeightedBlendTest() override = default;
+
+ void SetUp() override {
+ test_utils::ResetDspTable(bitdepth);
+ DistanceWeightedBlendInit_C();
+ const dsp::Dsp* const dsp = dsp::GetDspTable(bitdepth);
+ ASSERT_NE(dsp, nullptr);
+ base_func_ = dsp->distance_weighted_blend;
+ const testing::TestInfo* const test_info =
+ testing::UnitTest::GetInstance()->current_test_info();
+ const absl::string_view test_case = test_info->test_suite_name();
+ if (absl::StartsWith(test_case, "C/")) {
+ base_func_ = nullptr;
+ } else if (absl::StartsWith(test_case, "SSE41/")) {
+ if ((GetCpuInfo() & kSSE4_1) != 0) {
+ DistanceWeightedBlendInit_SSE4_1();
+ }
+ } else if (absl::StartsWith(test_case, "NEON/")) {
+ DistanceWeightedBlendInit_NEON();
+ } else {
+ FAIL() << "Unrecognized architecture prefix in test case name: "
+ << test_case;
+ }
+ func_ = dsp->distance_weighted_blend;
+ }
+
+ protected:
+ void Test(const char* digest, int num_tests);
+
+ private:
+ using PredType =
+ typename std::conditional<bitdepth == 8, int16_t, uint16_t>::type;
+ static constexpr int kDestStride = kMaxSuperBlockSizeInPixels;
+ const int width_ = GetParam().width;
+ const int height_ = GetParam().height;
+ alignas(kMaxAlignment) PredType
+ source1_[kMaxSuperBlockSizeInPixels * kMaxSuperBlockSizeInPixels];
+ alignas(kMaxAlignment) PredType
+ source2_[kMaxSuperBlockSizeInPixels * kMaxSuperBlockSizeInPixels];
+ Pixel dest_[kMaxSuperBlockSizeInPixels * kMaxSuperBlockSizeInPixels] = {};
+ Pixel reference_[kMaxSuperBlockSizeInPixels * kMaxSuperBlockSizeInPixels] =
+ {};
+ dsp::DistanceWeightedBlendFunc base_func_;
+ dsp::DistanceWeightedBlendFunc func_;
+};
+
+template <int bitdepth, typename Pixel>
+void DistanceWeightedBlendTest<bitdepth, Pixel>::Test(const char* digest,
+ int num_tests) {
+ if (func_ == nullptr) return;
+ libvpx_test::ACMRandom rnd(libvpx_test::ACMRandom::DeterministicSeed());
+ PredType* src_1 = source1_;
+ PredType* src_2 = source2_;
+
+ const int index = rnd.Rand8() & 3;
+ const uint8_t weight_0 = kQuantizedDistanceLookup[index][0];
+ const uint8_t weight_1 = kQuantizedDistanceLookup[index][1];
+ // In libgav1, predictors have an offset which are later subtracted and
+ // clipped in distance weighted blending. Therefore we add the offset
+ // here to match libaom's implementation.
+ for (int y = 0; y < height_; ++y) {
+ for (int x = 0; x < width_; ++x) {
+ // distance_weighted_blend is applied to compound prediction values. This
+ // implies a range far exceeding that of pixel values.
+ // The ranges include kCompoundOffset in 10bpp and 12bpp.
+ // see: src/dsp/convolve.cc & src/dsp/warp.cc.
+ static constexpr int kCompoundPredictionRange[3][2] = {
+ // 8bpp
+ {-5132, 9212},
+ // 10bpp
+ {3988, 61532},
+ // 12bpp
+ {3974, 61559},
+ };
+ constexpr int bitdepth_index = (bitdepth - 8) >> 1;
+ const int min_val = kCompoundPredictionRange[bitdepth_index][0];
+ const int max_val = kCompoundPredictionRange[bitdepth_index][1];
+ src_1[x] = static_cast<PredType>(rnd(max_val - min_val) + min_val);
+ src_2[x] = static_cast<PredType>(rnd(max_val - min_val) + min_val);
+ }
+ src_1 += width_;
+ src_2 += width_;
+ }
+ absl::Duration elapsed_time;
+ for (int i = 0; i < num_tests; ++i) {
+ const absl::Time start = absl::Now();
+ func_(source1_, source2_, weight_0, weight_1, width_, height_, dest_,
+ sizeof(Pixel) * kDestStride);
+ elapsed_time += absl::Now() - start;
+ }
+
+ test_utils::CheckMd5Digest(
+ "DistanceWeightedBlend",
+ absl::StrFormat("BlockSize%dx%d", width_, height_).c_str(), digest, dest_,
+ sizeof(dest_), elapsed_time);
+}
+
+const TestParam kTestParam[] = {
+ TestParam(4, 4), TestParam(4, 8), TestParam(4, 16),
+ TestParam(8, 4), TestParam(8, 8), TestParam(8, 16),
+ TestParam(8, 32), TestParam(16, 4), TestParam(16, 8),
+ TestParam(16, 16), TestParam(16, 32), TestParam(16, 64),
+ TestParam(32, 8), TestParam(32, 16), TestParam(32, 32),
+ TestParam(32, 64), TestParam(32, 128), TestParam(64, 16),
+ TestParam(64, 32), TestParam(64, 64), TestParam(64, 128),
+ TestParam(128, 32), TestParam(128, 64), TestParam(128, 128),
+};
+
+const char* GetDistanceWeightedBlendDigest8bpp(const TestParam block_size) {
+ static const char* const kDigestsWidth4[] = {
+ "ebf389f724f8ab46a2cac895e4e073ca",
+ "09acd567b6b12c8cf8eb51d8b86eb4bf",
+ "57bb4d65695d8ec6752f2bd8686b64fd",
+ };
+ static const char* const kDigestsWidth8[] = {
+ "270905ac76f9a2cba8a552eb0bf7c8c1",
+ "f0801c8574d2c271ef2bbea77a1d7352",
+ "e761b580e3312be33a227492a233ce72",
+ "ff214dab1a7e98e2285961d6421720c6",
+ };
+ static const char* const kDigestsWidth16[] = {
+ "4f712609a36e817f9752326d58562ff8", "14243f5c5f7c7104160c1f2cef0a0fbc",
+ "3ac3f3161b7c8dd8436b02abfdde104a", "81a00b704e0e41a5dbe6436ac70c098d",
+ "af8fd02017c7acdff788be742d700baa",
+ };
+ static const char* const kDigestsWidth32[] = {
+ "ee34332c66a6d6ed8ce64031aafe776c", "b5e3d22bd2dbdb624c8b86a1afb5ce6d",
+ "607ffc22098d81b7e37a7bf62f4af5d3", "3823dbf043b4682f56d5ca698e755ea5",
+ "57f7e8d1e67645269ce760a2c8da4afc",
+ };
+ static const char* const kDigestsWidth64[] = {
+ "4acf556b921956c2bc24659cd5128401",
+ "a298c544c9c3b27924b4c23cc687ea5a",
+ "539e2df267782ce61c70103b23b7d922",
+ "3b0cb2a0b5d384efee4d81401025bec1",
+ };
+ static const char* const kDigestsWidth128[] = {
+ "d71ee689a40ff5f390d07717df4b7233",
+ "8b56b636dd712c2f8d138badb7219991",
+ "8cfc8836908902b8f915639b7bff45b3",
+ };
+ const int height_index =
+ FloorLog2(block_size.height) - FloorLog2(block_size.width) + 2;
+ switch (block_size.width) {
+ case 4:
+ return kDigestsWidth4[height_index - 2];
+ case 8:
+ return kDigestsWidth8[height_index - 1];
+ case 16:
+ return kDigestsWidth16[height_index];
+ case 32:
+ return kDigestsWidth32[height_index];
+ case 64:
+ return kDigestsWidth64[height_index];
+ default:
+ EXPECT_EQ(block_size.width, 128)
+ << "Unknown width parameter: " << block_size.width;
+ return kDigestsWidth128[height_index];
+ }
+}
+
+using DistanceWeightedBlendTest8bpp = DistanceWeightedBlendTest<8, uint8_t>;
+
+TEST_P(DistanceWeightedBlendTest8bpp, Blending) {
+ Test(GetDistanceWeightedBlendDigest8bpp(GetParam()), 1);
+}
+
+TEST_P(DistanceWeightedBlendTest8bpp, DISABLED_Speed) {
+ Test(GetDistanceWeightedBlendDigest8bpp(GetParam()), kNumSpeedTests);
+}
+
+INSTANTIATE_TEST_SUITE_P(C, DistanceWeightedBlendTest8bpp,
+ testing::ValuesIn(kTestParam));
+
+#if LIBGAV1_ENABLE_NEON
+INSTANTIATE_TEST_SUITE_P(NEON, DistanceWeightedBlendTest8bpp,
+ testing::ValuesIn(kTestParam));
+#endif
+
+#if LIBGAV1_ENABLE_SSE4_1
+INSTANTIATE_TEST_SUITE_P(SSE41, DistanceWeightedBlendTest8bpp,
+ testing::ValuesIn(kTestParam));
+#endif
+
+#if LIBGAV1_MAX_BITDEPTH >= 10
+const char* GetDistanceWeightedBlendDigest10bpp(const TestParam block_size) {
+ static const char* const kDigestsWidth4[] = {
+ "55f594b56e16d5c401274affebbcc3d3",
+ "69df14da4bb33a8f7d7087921008e919",
+ "1b61f33604c54015794198a13bfebf46",
+ };
+ static const char* const kDigestsWidth8[] = {
+ "825a938185b152f7cf09bf1c0723ce2b",
+ "85ea315c51d979bc9b45834d6b40ec6f",
+ "92ebde208e8c39f7ec6de2de82182dbb",
+ "520f84716db5b43684dbb703806383fe",
+ };
+ static const char* const kDigestsWidth16[] = {
+ "12ca23e3e2930005a0511646e8c83da4", "6208694a6744f4a3906f58c1add670e3",
+ "a33d63889df989a3bbf84ff236614267", "34830846ecb0572a98bbd192fed02b16",
+ "34bb2f79c0bd7f9a80691b8af597f2a8",
+ };
+ static const char* const kDigestsWidth32[] = {
+ "fa97f2d0e3143f1f44d3ac018b0d696d", "3df4a22456c9ab6ed346ab1b9750ae7d",
+ "6276a058b35c6131bc0c94a4b4a37ebc", "9ca42da5d2d5eb339df03ae2c7a26914",
+ "2ff0dc010a7b40830fb47423a9beb894",
+ };
+ static const char* const kDigestsWidth64[] = {
+ "800e692c520f99223bc24c1ac95a0166",
+ "818b6d20426585ef7fe844015a03aaf5",
+ "fb48691ccfff083e01d74826e88e613f",
+ "0bd350bc5bc604a224d77a5f5a422698",
+ };
+ static const char* const kDigestsWidth128[] = {
+ "02aac5d5669c1245da876c5440c4d829",
+ "a130840813cd6bd69d09bcf5f8d0180f",
+ "6ece1846bea55e8f8f2ed7fbf73718de",
+ };
+ const int height_index =
+ FloorLog2(block_size.height) - FloorLog2(block_size.width) + 2;
+ switch (block_size.width) {
+ case 4:
+ return kDigestsWidth4[height_index - 2];
+ case 8:
+ return kDigestsWidth8[height_index - 1];
+ case 16:
+ return kDigestsWidth16[height_index];
+ case 32:
+ return kDigestsWidth32[height_index];
+ case 64:
+ return kDigestsWidth64[height_index];
+ default:
+ EXPECT_EQ(block_size.width, 128)
+ << "Unknown width parameter: " << block_size.width;
+ return kDigestsWidth128[height_index];
+ }
+}
+
+using DistanceWeightedBlendTest10bpp = DistanceWeightedBlendTest<10, uint16_t>;
+
+TEST_P(DistanceWeightedBlendTest10bpp, Blending) {
+ Test(GetDistanceWeightedBlendDigest10bpp(GetParam()), 1);
+}
+
+TEST_P(DistanceWeightedBlendTest10bpp, DISABLED_Speed) {
+ Test(GetDistanceWeightedBlendDigest10bpp(GetParam()), kNumSpeedTests);
+}
+
+INSTANTIATE_TEST_SUITE_P(C, DistanceWeightedBlendTest10bpp,
+ testing::ValuesIn(kTestParam));
+
+#if LIBGAV1_ENABLE_SSE4_1
+INSTANTIATE_TEST_SUITE_P(SSE41, DistanceWeightedBlendTest10bpp,
+ testing::ValuesIn(kTestParam));
+#endif
+#if LIBGAV1_ENABLE_NEON
+INSTANTIATE_TEST_SUITE_P(NEON, DistanceWeightedBlendTest10bpp,
+ testing::ValuesIn(kTestParam));
+#endif
+#endif // LIBGAV1_MAX_BITDEPTH >= 10
+
+} // namespace
+} // namespace dsp
+} // namespace libgav1