xref: /aosp_15_r20/external/ComputeLibrary/tests/validation/reference/UtilsQuantizedAsymm.h (revision c217d954acce2dbc11938adb493fc0abd69584f3)
1*c217d954SCole Faust /*
2*c217d954SCole Faust  * Copyright (c) 2017-2021 Arm Limited.
3*c217d954SCole Faust  *
4*c217d954SCole Faust  * SPDX-License-Identifier: MIT
5*c217d954SCole Faust  *
6*c217d954SCole Faust  * Permission is hereby granted, free of charge, to any person obtaining a copy
7*c217d954SCole Faust  * of this software and associated documentation files (the "Software"), to
8*c217d954SCole Faust  * deal in the Software without restriction, including without limitation the
9*c217d954SCole Faust  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*c217d954SCole Faust  * sell copies of the Software, and to permit persons to whom the Software is
11*c217d954SCole Faust  * furnished to do so, subject to the following conditions:
12*c217d954SCole Faust  *
13*c217d954SCole Faust  * The above copyright notice and this permission notice shall be included in all
14*c217d954SCole Faust  * copies or substantial portions of the Software.
15*c217d954SCole Faust  *
16*c217d954SCole Faust  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*c217d954SCole Faust  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*c217d954SCole Faust  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*c217d954SCole Faust  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*c217d954SCole Faust  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21*c217d954SCole Faust  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
22*c217d954SCole Faust  * SOFTWARE.
23*c217d954SCole Faust  */
24*c217d954SCole Faust #ifndef ARM_COMPUTE_TEST_VALIDATION_UTILS_QUANTIZED_ASYMM_H
25*c217d954SCole Faust #define ARM_COMPUTE_TEST_VALIDATION_UTILS_QUANTIZED_ASYMM_H
26*c217d954SCole Faust 
27*c217d954SCole Faust #include <cstdint>
28*c217d954SCole Faust 
29*c217d954SCole Faust namespace arm_compute
30*c217d954SCole Faust {
31*c217d954SCole Faust namespace test
32*c217d954SCole Faust {
33*c217d954SCole Faust namespace validation
34*c217d954SCole Faust {
35*c217d954SCole Faust namespace
36*c217d954SCole Faust {
37*c217d954SCole Faust #if __clang__
38*c217d954SCole Faust // This has been tested on clang 7.0.2 (__clang_major__ == 7 && __clang_minor__ == 0 && __clang_patchlevel__ == 2)
to_int64(int32_t val)39*c217d954SCole Faust inline int64_t to_int64(int32_t val)
40*c217d954SCole Faust {
41*c217d954SCole Faust     return static_cast<int64_t>(val) | ((val < 0) ? (((1ll << 32) - 1) << 32) : 0);
42*c217d954SCole Faust }
43*c217d954SCole Faust #else  // __clang__
44*c217d954SCole Faust inline int64_t to_int64(int32_t val)
45*c217d954SCole Faust {
46*c217d954SCole Faust     return static_cast<int64_t>(val);
47*c217d954SCole Faust }
48*c217d954SCole Faust #endif // __clang__
49*c217d954SCole Faust } // namespace
50*c217d954SCole Faust 
51*c217d954SCole Faust /** Rounded to nearest division by a power-of-two. */
asymm_rounding_divide_by_pow2(int32_t x,int exponent)52*c217d954SCole Faust inline int32_t asymm_rounding_divide_by_pow2(int32_t x, int exponent)
53*c217d954SCole Faust {
54*c217d954SCole Faust     const int32_t mask      = (1 << exponent) - 1;
55*c217d954SCole Faust     const int32_t threshold = (mask >> 1) + (x < 0 ? 1 : 0);
56*c217d954SCole Faust     return (x >> exponent) + ((x & mask) > threshold ? 1 : 0);
57*c217d954SCole Faust }
58*c217d954SCole Faust 
59*c217d954SCole Faust /** Multiplication of two integers. The same as ARMv7 Arm® Neon™ VQRDMULH instruction. */
asymm_int_mult(int32_t a,int32_t b)60*c217d954SCole Faust inline int32_t asymm_int_mult(int32_t a, int32_t b)
61*c217d954SCole Faust {
62*c217d954SCole Faust     const bool    overflow     = a == b && a == std::numeric_limits<int32_t>::min();
63*c217d954SCole Faust     const int64_t a_64         = to_int64(a);
64*c217d954SCole Faust     const int64_t b_64         = to_int64(b);
65*c217d954SCole Faust     const int64_t ab_64        = a_64 * b_64;
66*c217d954SCole Faust     const int32_t nudge        = ab_64 >= 0 ? (1 << 30) : (1 - (1 << 30));
67*c217d954SCole Faust     const int32_t ab_x2_high32 = static_cast<int32_t>((ab_64 + nudge) / (1ll << 31));
68*c217d954SCole Faust     return overflow ? std::numeric_limits<int32_t>::max() : ab_x2_high32;
69*c217d954SCole Faust }
70*c217d954SCole Faust 
71*c217d954SCole Faust /** Quantize down the input value in range [min, max]. */
quantize_down_scale_by_fixedpoint(int32_t val,int32_t result_mult_int,int32_t result_shift,int32_t result_offset_after_shift,int32_t min,int32_t max)72*c217d954SCole Faust inline int32_t quantize_down_scale_by_fixedpoint(int32_t val, int32_t result_mult_int, int32_t result_shift,
73*c217d954SCole Faust                                                  int32_t result_offset_after_shift, int32_t min, int32_t max)
74*c217d954SCole Faust {
75*c217d954SCole Faust     int32_t res = 0;
76*c217d954SCole Faust     if(result_shift < 0)
77*c217d954SCole Faust     {
78*c217d954SCole Faust         res = asymm_int_mult(val * (1 << (-result_shift)), result_mult_int);
79*c217d954SCole Faust     }
80*c217d954SCole Faust     else
81*c217d954SCole Faust     {
82*c217d954SCole Faust         res = asymm_rounding_divide_by_pow2(asymm_int_mult(val, result_mult_int), result_shift);
83*c217d954SCole Faust     }
84*c217d954SCole Faust     res += result_offset_after_shift;
85*c217d954SCole Faust     res = utility::clamp<int32_t>(res, min, max);
86*c217d954SCole Faust     return res;
87*c217d954SCole Faust }
88*c217d954SCole Faust } // namespace validation
89*c217d954SCole Faust } // namespace test
90*c217d954SCole Faust } // namespace arm_compute
91*c217d954SCole Faust #endif /* ARM_COMPUTE_TEST_VALIDATION_UTILS_QUANTIZED_ASYMM_H */
92