diff options
Diffstat (limited to 'src/dsp/warp_test.cc')
-rw-r--r-- | src/dsp/warp_test.cc | 649 |
1 files changed, 649 insertions, 0 deletions
diff --git a/src/dsp/warp_test.cc b/src/dsp/warp_test.cc new file mode 100644 index 0000000..e7384f4 --- /dev/null +++ b/src/dsp/warp_test.cc @@ -0,0 +1,649 @@ +// Copyright 2021 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/warp.h" + +#include <algorithm> +#include <cassert> +#include <cmath> +#include <cstddef> +#include <cstdint> +#include <cstdio> +#include <cstdlib> +#include <ostream> +#include <string> +#include <type_traits> + +#include "absl/base/macros.h" +#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/post_filter.h" +#include "src/utils/common.h" +#include "src/utils/constants.h" +#include "src/utils/cpu.h" +#include "src/utils/memory.h" +#include "tests/block_utils.h" +#include "tests/third_party/libvpx/acm_random.h" +#include "tests/utils.h" + +namespace libgav1 { +namespace dsp { +namespace { + +constexpr int kSourceBorderHorizontal = 16; +constexpr int kSourceBorderVertical = 13; + +constexpr int kMaxSourceBlockWidth = + kMaxSuperBlockSizeInPixels + kSourceBorderHorizontal * 2; +constexpr int kMaxSourceBlockHeight = + kMaxSuperBlockSizeInPixels + kSourceBorderVertical * 2; +constexpr int kMaxDestBlockWidth = + kMaxSuperBlockSizeInPixels + kConvolveBorderLeftTop * 2; +constexpr int kMaxDestBlockHeight = + kMaxSuperBlockSizeInPixels + kConvolveBorderLeftTop * 2; + +constexpr uint16_t kDivisorLookup[257] = { + 16384, 16320, 16257, 16194, 16132, 16070, 16009, 15948, 15888, 15828, 15768, + 15709, 15650, 15592, 15534, 15477, 15420, 15364, 15308, 15252, 15197, 15142, + 15087, 15033, 14980, 14926, 14873, 14821, 14769, 14717, 14665, 14614, 14564, + 14513, 14463, 14413, 14364, 14315, 14266, 14218, 14170, 14122, 14075, 14028, + 13981, 13935, 13888, 13843, 13797, 13752, 13707, 13662, 13618, 13574, 13530, + 13487, 13443, 13400, 13358, 13315, 13273, 13231, 13190, 13148, 13107, 13066, + 13026, 12985, 12945, 12906, 12866, 12827, 12788, 12749, 12710, 12672, 12633, + 12596, 12558, 12520, 12483, 12446, 12409, 12373, 12336, 12300, 12264, 12228, + 12193, 12157, 12122, 12087, 12053, 12018, 11984, 11950, 11916, 11882, 11848, + 11815, 11782, 11749, 11716, 11683, 11651, 11619, 11586, 11555, 11523, 11491, + 11460, 11429, 11398, 11367, 11336, 11305, 11275, 11245, 11215, 11185, 11155, + 11125, 11096, 11067, 11038, 11009, 10980, 10951, 10923, 10894, 10866, 10838, + 10810, 10782, 10755, 10727, 10700, 10673, 10645, 10618, 10592, 10565, 10538, + 10512, 10486, 10460, 10434, 10408, 10382, 10356, 10331, 10305, 10280, 10255, + 10230, 10205, 10180, 10156, 10131, 10107, 10082, 10058, 10034, 10010, 9986, + 9963, 9939, 9916, 9892, 9869, 9846, 9823, 9800, 9777, 9754, 9732, + 9709, 9687, 9664, 9642, 9620, 9598, 9576, 9554, 9533, 9511, 9489, + 9468, 9447, 9425, 9404, 9383, 9362, 9341, 9321, 9300, 9279, 9259, + 9239, 9218, 9198, 9178, 9158, 9138, 9118, 9098, 9079, 9059, 9039, + 9020, 9001, 8981, 8962, 8943, 8924, 8905, 8886, 8867, 8849, 8830, + 8812, 8793, 8775, 8756, 8738, 8720, 8702, 8684, 8666, 8648, 8630, + 8613, 8595, 8577, 8560, 8542, 8525, 8508, 8490, 8473, 8456, 8439, + 8422, 8405, 8389, 8372, 8355, 8339, 8322, 8306, 8289, 8273, 8257, + 8240, 8224, 8208, 8192}; + +template <bool is_compound> +const char* GetDigest8bpp(int id) { + static const char* const kDigest[] = { + "77ba358a0f5e19a8e69fa0a95712578e", "141b23d13a04e0b84d26d514de76d6b0", + "b0265858454b979852ffadae323f0fb7", "9cf38e3579265b656f1f2100ba15b0e9", + "ab51d05cc255ef8e37921182df1d89b1", "e3e96f90a4b07ca733e40f057dc01c41", + "4eee8c1a52a62a266db9b1c9338e124c", "901a87d8f88f6324dbc0960a6de861ac", + "da9cb6faf6adaeeae12b6784f39186c5", "14450ab05536cdb0d2f499716ccb559d", + "566b396cbf008bbb869b364fdc81860d", "681a872baf2de4e58d73ea9ab8643a72", + "7f17d290d513a7416761b3a01f10fd2f", + }; + static const char* const kCompoundDigest[] = { + "7e9339d265b7beac7bbe32fe7bb0fccb", "f747d663b427bb38a3ff36b0815a394c", + "858cf54d2253281a919fbdb48fe91c53", "4721dd97a212c6068bd488f400259afc", + "36878c7906492bc740112abdea77616f", "89deb68aa35764bbf3024b501a6bed50", + "8ac5b08f9b2afd38143c357646af0f82", "bf6e2a64835ea0c9d7467394253d0eb2", + "7b0a539acd2a27eff398dd084abad933", "61c8d81b397c1cf727ff8a9fabab90af", + "4d412349a25a832c1fb3fb29e3f0e2b3", "2c6dd2a9a4ede9fa00adb567ba646f30", + "b2a0ce68db3cadd207299f73112bed74", + }; + return is_compound ? kCompoundDigest[id] : kDigest[id]; +} + +#if LIBGAV1_MAX_BITDEPTH >= 10 +template <bool is_compound> +const char* GetDigest10bpp(int id) { + static const char* const kDigest[] = { + "1fef54f56a0bafccf7f8da1ac3b18b76", "8a65c72f171feafa2f393d31d6b7fe1b", + "808019346f2f1f45f8cf2e9fc9a49320", "c28e2f2c6c830a29bcc2452166cba521", + "f040674d6f54e8910d655f0d11fd8cdd", "473af9bb1c6023965c2284b716feef97", + "e4f6d7babd0813d5afb0f575ebfa8166", "58f96ef8a880963a213624bb0d06d47c", + "1ec0995fa4490628b679d03683233388", "9526fb102fde7dc1a7e160e65af6da33", + "f0457427d0c0e31d82ea4f612f7f86f1", "ddc82ae298cccebad493ba9de0f69fbd", + "5ed615091e2f62df26de7e91a985cb81", + }; + static const char* const kCompoundDigest[] = { + "8e6986ae143260e0b8b4887f15a141a1", "0a7f0db8316b8c3569f08834dd0c6f50", + "90705b2e7dbe083e8a1f70f29d6f257e", "e428a75bea77d769d21f3f7a1d2b0b38", + "a570b13d790c085c4ab50d71dd085d56", "e5d043c6cd6ff6dbab6e38a8877e93bd", + "12ea96991e46e3e9aa78ab812ffa0525", "84293a94a53f1cf814fa25e793c3fe27", + "b98a7502c84ac8437266f702dcc0a92e", "d8db5d52e9b0a5be0ad2d517d5bd16e9", + "f3be504bbb609ce4cc71c5539252638a", "fcde83b54e14e9de23460644f244b047", + "42eb66e752e9ef289b47053b5c73fdd6", + }; + return is_compound ? kCompoundDigest[id] : kDigest[id]; +} +#endif + +int RandomWarpedParam(int seed_offset, int bits) { + libvpx_test::ACMRandom rnd(seed_offset + + libvpx_test::ACMRandom::DeterministicSeed()); + // 1 in 8 chance of generating zero (arbitrary). + const bool zero = (rnd.Rand16() & 7) == 0; + if (zero) return 0; + // Generate uniform values in the range [-(1 << bits), 1] U [1, 1 << + // bits]. + const int mask = (1 << bits) - 1; + const int value = 1 + (rnd.RandRange(1u << 31) & mask); + const bool sign = (rnd.Rand16() & 1) != 0; + return sign ? value : -value; +} + +// This function is a copy from warp_prediction.cc. +template <typename T> +void GenerateApproximateDivisor(T value, int16_t* division_factor, + int16_t* division_shift) { + const int n = FloorLog2(std::abs(value)); + const T e = std::abs(value) - (static_cast<T>(1) << n); + const int entry = (n > kDivisorLookupBits) + ? RightShiftWithRounding(e, n - kDivisorLookupBits) + : static_cast<int>(e << (kDivisorLookupBits - n)); + *division_shift = n + kDivisorLookupPrecisionBits; + *division_factor = + (value < 0) ? -kDivisorLookup[entry] : kDivisorLookup[entry]; +} + +// This function is a copy from warp_prediction.cc. +int16_t GetShearParameter(int value) { + return static_cast<int16_t>( + LeftShift(RightShiftWithRoundingSigned(value, kWarpParamRoundingBits), + kWarpParamRoundingBits)); +} + +// This function is a copy from warp_prediction.cc. +// This function is used here to help generate valid warp parameters. +bool SetupShear(const int* params, int16_t* alpha, int16_t* beta, + int16_t* gamma, int16_t* delta) { + int16_t division_shift; + int16_t division_factor; + GenerateApproximateDivisor<int32_t>(params[2], &division_factor, + &division_shift); + const int alpha0 = + Clip3(params[2] - (1 << kWarpedModelPrecisionBits), INT16_MIN, INT16_MAX); + const int beta0 = Clip3(params[3], INT16_MIN, INT16_MAX); + const int64_t v = LeftShift(params[4], kWarpedModelPrecisionBits); + const int gamma0 = + Clip3(RightShiftWithRoundingSigned(v * division_factor, division_shift), + INT16_MIN, INT16_MAX); + const int64_t w = static_cast<int64_t>(params[3]) * params[4]; + const int delta0 = Clip3( + params[5] - + RightShiftWithRoundingSigned(w * division_factor, division_shift) - + (1 << kWarpedModelPrecisionBits), + INT16_MIN, INT16_MAX); + + *alpha = GetShearParameter(alpha0); + *beta = GetShearParameter(beta0); + *gamma = GetShearParameter(gamma0); + *delta = GetShearParameter(delta0); + if ((4 * std::abs(*alpha) + 7 * std::abs(*beta) >= + (1 << kWarpedModelPrecisionBits)) || + (4 * std::abs(*gamma) + 4 * std::abs(*delta) >= + (1 << kWarpedModelPrecisionBits))) { + return false; // NOLINT (easier condition to understand). + } + + return true; +} + +void GenerateWarpedModel(int* params, int16_t* alpha, int16_t* beta, + int16_t* gamma, int16_t* delta, int seed) { + do { + params[0] = RandomWarpedParam(seed, kWarpedModelPrecisionBits + 6); + params[1] = RandomWarpedParam(seed, kWarpedModelPrecisionBits + 6); + params[2] = RandomWarpedParam(seed, kWarpedModelPrecisionBits - 3) + + (1 << kWarpedModelPrecisionBits); + params[3] = RandomWarpedParam(seed, kWarpedModelPrecisionBits - 3); + params[4] = RandomWarpedParam(seed, kWarpedModelPrecisionBits - 3); + params[5] = RandomWarpedParam(seed, kWarpedModelPrecisionBits - 3) + + (1 << kWarpedModelPrecisionBits); + ++seed; + } while (params[2] == 0 || !SetupShear(params, alpha, beta, gamma, delta)); +} + +struct WarpTestParam { + WarpTestParam(int width, int height) : width(width), height(height) {} + int width; + int height; +}; + +template <bool is_compound, int bitdepth, typename Pixel> +class WarpTest : public testing::TestWithParam<WarpTestParam> { + public: + WarpTest() = default; + ~WarpTest() override = default; + + void SetUp() override { + test_utils::ResetDspTable(bitdepth); + WarpInit_C(); + const dsp::Dsp* const dsp = dsp::GetDspTable(bitdepth); + ASSERT_NE(dsp, nullptr); + 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/")) { + } else if (absl::StartsWith(test_case, "NEON/")) { + WarpInit_NEON(); + } else if (absl::StartsWith(test_case, "SSE41/")) { + WarpInit_SSE4_1(); + } else { + FAIL() << "Unrecognized architecture prefix in test case name: " + << test_case; + } + func_ = is_compound ? dsp->warp_compound : dsp->warp; + } + + protected: + using DestType = + typename std::conditional<is_compound, uint16_t, Pixel>::type; + + void SetInputData(bool use_fixed_values, int value); + void Test(bool use_fixed_values, int value, int num_runs = 1); + void TestFixedValues(); + void TestRandomValues(); + void TestSpeed(); + + const WarpTestParam param_ = GetParam(); + + private: + int warp_params_[8]; + dsp::WarpFunc func_; + // Warp filters are 7-tap, which needs 3 pixels (kConvolveBorderLeftTop) + // padding. Destination buffer indices are based on subsampling values (x+y): + // 0: (4:4:4), 1:(4:2:2), 2: (4:2:0). + Pixel source_[kMaxSourceBlockHeight * kMaxSourceBlockWidth] = {}; + DestType dest_[3][kMaxDestBlockHeight * kMaxDestBlockWidth] = {}; +}; + +template <bool is_compound, int bitdepth, typename Pixel> +void WarpTest<is_compound, bitdepth, Pixel>::SetInputData(bool use_fixed_values, + int value) { + if (use_fixed_values) { + for (int y = 0; y < param_.height; ++y) { + const int row = kSourceBorderVertical + y; + Memset(source_ + row * kMaxSourceBlockWidth + kSourceBorderHorizontal, + value, param_.width); + } + } else { + const int mask = (1 << bitdepth) - 1; + libvpx_test::ACMRandom rnd(libvpx_test::ACMRandom::DeterministicSeed()); + for (int y = 0; y < param_.height; ++y) { + const int row = kSourceBorderVertical + y; + for (int x = 0; x < param_.width; ++x) { + const int column = kSourceBorderHorizontal + x; + source_[row * kMaxSourceBlockWidth + column] = rnd.Rand16() & mask; + } + } + } + PostFilter::ExtendFrame<Pixel>( + &source_[kSourceBorderVertical * kMaxSourceBlockWidth + + kSourceBorderHorizontal], + param_.width, param_.height, kMaxSourceBlockWidth, + kSourceBorderHorizontal, kSourceBorderHorizontal, kSourceBorderVertical, + kSourceBorderVertical); +} + +template <bool is_compound, int bitdepth, typename Pixel> +void WarpTest<is_compound, bitdepth, Pixel>::Test(bool use_fixed_values, + int value, + int num_runs /*= 1*/) { + if (func_ == nullptr) return; + SetInputData(use_fixed_values, value); + libvpx_test::ACMRandom rnd(libvpx_test::ACMRandom::DeterministicSeed()); + const int source_offset = + kSourceBorderVertical * kMaxSourceBlockWidth + kSourceBorderHorizontal; + const int dest_offset = + kConvolveBorderLeftTop * kMaxDestBlockWidth + kConvolveBorderLeftTop; + const Pixel* const src = source_ + source_offset; + const ptrdiff_t src_stride = kMaxSourceBlockWidth * sizeof(Pixel); + const ptrdiff_t dst_stride = + is_compound ? kMaxDestBlockWidth : kMaxDestBlockWidth * sizeof(Pixel); + + absl::Duration elapsed_time; + for (int subsampling_x = 0; subsampling_x <= 1; ++subsampling_x) { + for (int subsampling_y = 0; subsampling_y <= 1; ++subsampling_y) { + if (subsampling_x == 0 && subsampling_y == 1) { + // When both are 0: 4:4:4 + // When both are 1: 4:2:0 + // When only |subsampling_x| is 1: 4:2:2 + // Having only |subsampling_y| == 1 is unsupported. + continue; + } + int params[8]; + int16_t alpha; + int16_t beta; + int16_t gamma; + int16_t delta; + GenerateWarpedModel(params, &alpha, &beta, &gamma, &delta, rnd.Rand8()); + + const int dest_id = subsampling_x + subsampling_y; + DestType* const dst = dest_[dest_id] + dest_offset; + const absl::Time start = absl::Now(); + for (int n = 0; n < num_runs; ++n) { + func_(src, src_stride, param_.width, param_.height, params, + subsampling_x, subsampling_y, 0, 0, param_.width, param_.height, + alpha, beta, gamma, delta, dst, dst_stride); + } + elapsed_time += absl::Now() - start; + } + } + + if (use_fixed_values) { + // For fixed values, input and output are identical. + for (size_t i = 0; i < ABSL_ARRAYSIZE(dest_); ++i) { + // |is_compound| holds a few more bits of precision and an offset value. + Pixel compensated_dest[kMaxDestBlockWidth * kMaxDestBlockHeight]; + const int compound_offset = (bitdepth == 8) ? 0 : kCompoundOffset; + if (is_compound) { + for (int y = 0; y < param_.height; ++y) { + for (int x = 0; x < param_.width; ++x) { + const int compound_value = + dest_[i][dest_offset + y * kMaxDestBlockWidth + x]; + const int remove_offset = compound_value - compound_offset; + const int full_shift = + remove_offset >> + (kInterRoundBitsVertical - kInterRoundBitsCompoundVertical); + compensated_dest[y * kMaxDestBlockWidth + x] = + Clip3(full_shift, 0, (1 << bitdepth) - 1); + } + } + } + Pixel* pixel_dest = + is_compound ? compensated_dest + : reinterpret_cast<Pixel*>(dest_[i] + dest_offset); + const bool success = test_utils::CompareBlocks( + src, pixel_dest, param_.width, param_.height, kMaxSourceBlockWidth, + kMaxDestBlockWidth, false); + EXPECT_TRUE(success) << "subsampling_x + subsampling_y: " << i; + } + } else { + // (width, height): + // (8, 8), id = 0. (8, 16), id = 1. (16, 8), id = 2. + // (16, 16), id = 3. (16, 32), id = 4. (32, 16), id = 5. + // ... + // (128, 128), id = 12. + int id; + if (param_.width == param_.height) { + id = 3 * static_cast<int>(FloorLog2(param_.width) - 3); + } else if (param_.width < param_.height) { + id = 1 + 3 * static_cast<int>(FloorLog2(param_.width) - 3); + } else { + id = 2 + 3 * static_cast<int>(FloorLog2(param_.height) - 3); + } + + const char* expected_digest; + if (bitdepth == 8) { + expected_digest = GetDigest8bpp<is_compound>(id); + } else { +#if LIBGAV1_MAX_BITDEPTH >= 10 + expected_digest = GetDigest10bpp<is_compound>(id); +#endif + } + test_utils::CheckMd5Digest( + "Warp", absl::StrFormat("%dx%d", param_.width, param_.height).c_str(), + expected_digest, dest_, sizeof(dest_), elapsed_time); + } +} + +template <bool is_compound, int bitdepth, typename Pixel> +void WarpTest<is_compound, bitdepth, Pixel>::TestFixedValues() { + Test(true, 0); + Test(true, 1); + Test(true, 128); + Test(true, (1 << bitdepth) - 1); +} + +template <bool is_compound, int bitdepth, typename Pixel> +void WarpTest<is_compound, bitdepth, Pixel>::TestRandomValues() { + Test(false, 0); +} + +template <bool is_compound, int bitdepth, typename Pixel> +void WarpTest<is_compound, bitdepth, Pixel>::TestSpeed() { + const int num_runs = static_cast<int>(1.0e7 / (param_.width * param_.height)); + Test(false, 0, num_runs); +} + +void ApplyFilterToSignedInput(const int min_input, const int max_input, + const int8_t filter[kSubPixelTaps], + int* min_output, int* max_output) { + int min = 0, max = 0; + for (int i = 0; i < kSubPixelTaps; ++i) { + const int tap = filter[i]; + if (tap > 0) { + max += max_input * tap; + min += min_input * tap; + } else { + min += max_input * tap; + max += min_input * tap; + } + } + *min_output = min; + *max_output = max; +} + +void ApplyFilterToUnsignedInput(const int max_input, + const int8_t filter[kSubPixelTaps], + int* min_output, int* max_output) { + ApplyFilterToSignedInput(0, max_input, filter, min_output, max_output); +} + +// Validate the maximum ranges for different parts of the Warp process. +template <int bitdepth> +void ShowRange() { + constexpr int horizontal_bits = (bitdepth == kBitdepth12) + ? kInterRoundBitsHorizontal12bpp + : kInterRoundBitsHorizontal; + constexpr int vertical_bits = (bitdepth == kBitdepth12) + ? kInterRoundBitsVertical12bpp + : kInterRoundBitsVertical; + constexpr int compound_vertical_bits = kInterRoundBitsCompoundVertical; + + constexpr int compound_offset = (bitdepth == 8) ? 0 : kCompoundOffset; + + constexpr int max_input = (1 << bitdepth) - 1; + + const int8_t* worst_warp_filter = kWarpedFilters8[93]; + + // First pass. + printf("Bitdepth: %2d Input range: [%8d, %8d]\n", bitdepth, 0, + max_input); + + int min = 0, max = 0; + ApplyFilterToUnsignedInput(max_input, worst_warp_filter, &min, &max); + + int first_pass_offset; + if (bitdepth == 8) { + // Derive an offset for 8 bit. + for (first_pass_offset = 1; - first_pass_offset > min; + first_pass_offset <<= 1) { + } + printf(" 8bpp intermediate offset: %d.\n", first_pass_offset); + min += first_pass_offset; + max += first_pass_offset; + assert(min > 0); + assert(max < UINT16_MAX); + } else { + // 10bpp and 12bpp require int32_t for the intermediate values. Adding an + // offset is not required. + assert(min > INT32_MIN); + assert(max > INT16_MAX && max < INT32_MAX); + } + + printf(" intermediate range: [%8d, %8d]\n", min, max); + + const int first_pass_min = RightShiftWithRounding(min, horizontal_bits); + const int first_pass_max = RightShiftWithRounding(max, horizontal_bits); + + printf(" first pass output range: [%8d, %8d]\n", first_pass_min, + first_pass_max); + + // Second pass. + if (bitdepth == 8) { + ApplyFilterToUnsignedInput(first_pass_max, worst_warp_filter, &min, &max); + } else { + ApplyFilterToSignedInput(first_pass_min, first_pass_max, worst_warp_filter, + &min, &max); + } + + if (bitdepth == 8) { + // Remove the offset that was applied in the first pass since we must use + // int32_t for this phase anyway. 128 is the sum of the filter taps. + const int offset_removal = (first_pass_offset >> horizontal_bits) * 128; + printf(" 8bpp intermediate offset removal: %d.\n", offset_removal); + max -= offset_removal; + min -= offset_removal; + assert(min < INT16_MIN && min > INT32_MIN); + assert(max > INT16_MAX && max < INT32_MAX); + } else { + // 10bpp and 12bpp require int32_t for the intermediate values. Adding an + // offset is not required. + assert(min > INT32_MIN); + assert(max > INT16_MAX && max < INT32_MAX); + } + + printf(" intermediate range: [%8d, %8d]\n", min, max); + + // Second pass non-compound output is clipped to Pixel values. + const int second_pass_min = + Clip3(RightShiftWithRounding(min, vertical_bits), 0, max_input); + const int second_pass_max = + Clip3(RightShiftWithRounding(max, vertical_bits), 0, max_input); + printf(" second pass output range: [%8d, %8d]\n", second_pass_min, + second_pass_max); + + // Output is Pixel so matches Pixel values. + assert(second_pass_min == 0); + assert(second_pass_max == max_input); + + const int compound_second_pass_min = + RightShiftWithRounding(min, compound_vertical_bits) + compound_offset; + const int compound_second_pass_max = + RightShiftWithRounding(max, compound_vertical_bits) + compound_offset; + + printf(" compound second pass output range: [%8d, %8d]\n", + compound_second_pass_min, compound_second_pass_max); + + if (bitdepth == 8) { + // 8bpp output is int16_t without an offset. + assert(compound_second_pass_min > INT16_MIN); + assert(compound_second_pass_max < INT16_MAX); + } else { + // 10bpp and 12bpp use the offset to fit inside uint16_t. + assert(compound_second_pass_min > 0); + assert(compound_second_pass_max < UINT16_MAX); + } + + printf("\n"); +} + +TEST(WarpTest, ShowRange) { + ShowRange<kBitdepth8>(); + ShowRange<kBitdepth10>(); + ShowRange<kBitdepth12>(); +} + +using WarpTest8bpp = WarpTest</*is_compound=*/false, 8, uint8_t>; +// TODO(jzern): Coverage could be added for kInterRoundBitsCompoundVertical via +// WarpCompoundTest. +// using WarpCompoundTest8bpp = WarpTest</*is_compound=*/true, 8, uint8_t>; + +// Verifies the sum of the warped filter coefficients is 128 for every filter. +// +// Verifies the properties used in the calculation of ranges of variables in +// the block warp process: +// * The maximum sum of the positive warped filter coefficients is 175. +// * The minimum (i.e., most negative) sum of the negative warped filter +// coefficients is -47. +// +// NOTE: This test is independent of the bitdepth and the implementation of the +// block warp function, so it just needs to be a test in the WarpTest8bpp class +// and does not need to be defined with TEST_P. +TEST(WarpTest8bpp, WarpedFilterCoefficientSums) { + int max_positive_sum = 0; + int min_negative_sum = 0; + for (const auto& filter : kWarpedFilters) { + int sum = 0; + int positive_sum = 0; + int negative_sum = 0; + for (const auto coefficient : filter) { + sum += coefficient; + if (coefficient > 0) { + positive_sum += coefficient; + } else { + negative_sum += coefficient; + } + } + EXPECT_EQ(sum, 128); + max_positive_sum = std::max(positive_sum, max_positive_sum); + min_negative_sum = std::min(negative_sum, min_negative_sum); + } + EXPECT_EQ(max_positive_sum, 175); + EXPECT_EQ(min_negative_sum, -47); +} + +TEST_P(WarpTest8bpp, FixedValues) { TestFixedValues(); } + +TEST_P(WarpTest8bpp, RandomValues) { TestRandomValues(); } + +TEST_P(WarpTest8bpp, DISABLED_Speed) { TestSpeed(); } +const WarpTestParam warp_test_param[] = { + WarpTestParam(8, 8), WarpTestParam(8, 16), WarpTestParam(16, 8), + WarpTestParam(16, 16), WarpTestParam(16, 32), WarpTestParam(32, 16), + WarpTestParam(32, 32), WarpTestParam(32, 64), WarpTestParam(64, 32), + WarpTestParam(64, 64), WarpTestParam(64, 128), WarpTestParam(128, 64), + WarpTestParam(128, 128), +}; + +INSTANTIATE_TEST_SUITE_P(C, WarpTest8bpp, testing::ValuesIn(warp_test_param)); + +#if LIBGAV1_ENABLE_NEON +INSTANTIATE_TEST_SUITE_P(NEON, WarpTest8bpp, + testing::ValuesIn(warp_test_param)); +#endif + +#if LIBGAV1_ENABLE_SSE4_1 +INSTANTIATE_TEST_SUITE_P(SSE41, WarpTest8bpp, + testing::ValuesIn(warp_test_param)); +#endif + +#if LIBGAV1_MAX_BITDEPTH >= 10 +using WarpTest10bpp = WarpTest</*is_compound=*/false, 10, uint16_t>; +// TODO(jzern): Coverage could be added for kInterRoundBitsCompoundVertical via +// WarpCompoundTest. +// using WarpCompoundTest10bpp = WarpTest</*is_compound=*/true, 10, uint16_t>; + +TEST_P(WarpTest10bpp, FixedValues) { TestFixedValues(); } + +TEST_P(WarpTest10bpp, RandomValues) { TestRandomValues(); } + +TEST_P(WarpTest10bpp, DISABLED_Speed) { TestSpeed(); } + +INSTANTIATE_TEST_SUITE_P(C, WarpTest10bpp, testing::ValuesIn(warp_test_param)); +#endif + +std::ostream& operator<<(std::ostream& os, const WarpTestParam& warp_param) { + return os << "BlockSize" << warp_param.width << "x" << warp_param.height; +} + +} // namespace +} // namespace dsp +} // namespace libgav1 |