1*4bdc9457SAndroid Build Coastguard Worker // Copyright 2021 Google LLC 2*4bdc9457SAndroid Build Coastguard Worker // 3*4bdc9457SAndroid Build Coastguard Worker // This source code is licensed under the BSD-style license found in the 4*4bdc9457SAndroid Build Coastguard Worker // LICENSE file in the root directory of this source tree. 5*4bdc9457SAndroid Build Coastguard Worker 6*4bdc9457SAndroid Build Coastguard Worker #include <algorithm> 7*4bdc9457SAndroid Build Coastguard Worker #include <cmath> 8*4bdc9457SAndroid Build Coastguard Worker #include <cstddef> 9*4bdc9457SAndroid Build Coastguard Worker #include <cstdint> 10*4bdc9457SAndroid Build Coastguard Worker #include <cstdlib> 11*4bdc9457SAndroid Build Coastguard Worker #include <iomanip> 12*4bdc9457SAndroid Build Coastguard Worker #include <ios> 13*4bdc9457SAndroid Build Coastguard Worker #include <vector> 14*4bdc9457SAndroid Build Coastguard Worker 15*4bdc9457SAndroid Build Coastguard Worker #include <gtest/gtest.h> 16*4bdc9457SAndroid Build Coastguard Worker 17*4bdc9457SAndroid Build Coastguard Worker #include <fp16.h> 18*4bdc9457SAndroid Build Coastguard Worker 19*4bdc9457SAndroid Build Coastguard Worker #include <xnnpack/aligned-allocator.h> 20*4bdc9457SAndroid Build Coastguard Worker #include <xnnpack/common.h> 21*4bdc9457SAndroid Build Coastguard Worker #include <xnnpack/isa-checks.h> 22*4bdc9457SAndroid Build Coastguard Worker #include <xnnpack/math.h> 23*4bdc9457SAndroid Build Coastguard Worker #include <xnnpack/math-stubs.h> 24*4bdc9457SAndroid Build Coastguard Worker 25*4bdc9457SAndroid Build Coastguard Worker 26*4bdc9457SAndroid Build Coastguard Worker constexpr int kBlockSize = 1024; 27*4bdc9457SAndroid Build Coastguard Worker 28*4bdc9457SAndroid Build Coastguard Worker #if XNN_ARCH_ARM || XNN_ARCH_ARM64 TEST(CVT__NEON,positive_normal)29*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON, positive_normal) { 30*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON; 31*4bdc9457SAndroid Build Coastguard Worker 32*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 33*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 34*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 35*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 36*4bdc9457SAndroid Build Coastguard Worker zero_point++) 37*4bdc9457SAndroid Build Coastguard Worker { 38*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 39*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 40*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 41*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 42*4bdc9457SAndroid Build Coastguard Worker } 43*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neon(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 44*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 45*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 46*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 47*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 48*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 49*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 50*4bdc9457SAndroid Build Coastguard Worker } 51*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 52*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 53*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 54*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 55*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 56*4bdc9457SAndroid Build Coastguard Worker } 57*4bdc9457SAndroid Build Coastguard Worker } 58*4bdc9457SAndroid Build Coastguard Worker } 59*4bdc9457SAndroid Build Coastguard Worker } 60*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON,negative_normal)61*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON, negative_normal) { 62*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON; 63*4bdc9457SAndroid Build Coastguard Worker 64*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 65*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 66*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 67*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 68*4bdc9457SAndroid Build Coastguard Worker zero_point++) 69*4bdc9457SAndroid Build Coastguard Worker { 70*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) zero_point); 71*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 72*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 73*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 74*4bdc9457SAndroid Build Coastguard Worker } 75*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neon(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 76*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 77*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 78*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 79*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 80*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 81*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 82*4bdc9457SAndroid Build Coastguard Worker } 83*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 84*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 85*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 86*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 87*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 88*4bdc9457SAndroid Build Coastguard Worker } 89*4bdc9457SAndroid Build Coastguard Worker } 90*4bdc9457SAndroid Build Coastguard Worker } 91*4bdc9457SAndroid Build Coastguard Worker } 92*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON,positive_saturation)93*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON, positive_saturation) { 94*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON; 95*4bdc9457SAndroid Build Coastguard Worker 96*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 97*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 98*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 99*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 100*4bdc9457SAndroid Build Coastguard Worker zero_point++) 101*4bdc9457SAndroid Build Coastguard Worker { 102*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 103*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 104*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 105*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 106*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 107*4bdc9457SAndroid Build Coastguard Worker } 108*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neon(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 109*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 110*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::max(); 111*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 112*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 113*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 114*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 115*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 116*4bdc9457SAndroid Build Coastguard Worker } 117*4bdc9457SAndroid Build Coastguard Worker } 118*4bdc9457SAndroid Build Coastguard Worker } 119*4bdc9457SAndroid Build Coastguard Worker } 120*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON,negative_saturation)121*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEON, negative_saturation) { 122*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON; 123*4bdc9457SAndroid Build Coastguard Worker 124*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 125*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 126*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 127*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 128*4bdc9457SAndroid Build Coastguard Worker zero_point++) 129*4bdc9457SAndroid Build Coastguard Worker { 130*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) zero_point); 131*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 132*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 133*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 134*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 135*4bdc9457SAndroid Build Coastguard Worker } 136*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neon(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 137*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 138*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::min(); 139*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 140*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 141*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 142*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 143*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 144*4bdc9457SAndroid Build Coastguard Worker } 145*4bdc9457SAndroid Build Coastguard Worker } 146*4bdc9457SAndroid Build Coastguard Worker } 147*4bdc9457SAndroid Build Coastguard Worker } 148*4bdc9457SAndroid Build Coastguard Worker #endif // XNN_ARCH_ARM || XNN_ARCH_ARM64 149*4bdc9457SAndroid Build Coastguard Worker 150*4bdc9457SAndroid Build Coastguard Worker #if XNN_ARCH_ARM || XNN_ARCH_ARM64 TEST(CVT__NEONV8,positive_normal)151*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8, positive_normal) { 152*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON_V8; 153*4bdc9457SAndroid Build Coastguard Worker 154*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 155*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 156*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 157*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 158*4bdc9457SAndroid Build Coastguard Worker zero_point++) 159*4bdc9457SAndroid Build Coastguard Worker { 160*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 161*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 162*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 163*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 164*4bdc9457SAndroid Build Coastguard Worker } 165*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neonv8(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 166*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 167*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 168*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 169*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 170*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 171*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 172*4bdc9457SAndroid Build Coastguard Worker } 173*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 174*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 175*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 176*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 177*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 178*4bdc9457SAndroid Build Coastguard Worker } 179*4bdc9457SAndroid Build Coastguard Worker } 180*4bdc9457SAndroid Build Coastguard Worker } 181*4bdc9457SAndroid Build Coastguard Worker } 182*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8,negative_normal)183*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8, negative_normal) { 184*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON_V8; 185*4bdc9457SAndroid Build Coastguard Worker 186*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 187*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 188*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 189*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 190*4bdc9457SAndroid Build Coastguard Worker zero_point++) 191*4bdc9457SAndroid Build Coastguard Worker { 192*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) zero_point); 193*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 194*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 195*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 196*4bdc9457SAndroid Build Coastguard Worker } 197*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neonv8(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 198*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 199*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 200*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 201*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 202*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 203*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 204*4bdc9457SAndroid Build Coastguard Worker } 205*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 206*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 207*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 208*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 209*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 210*4bdc9457SAndroid Build Coastguard Worker } 211*4bdc9457SAndroid Build Coastguard Worker } 212*4bdc9457SAndroid Build Coastguard Worker } 213*4bdc9457SAndroid Build Coastguard Worker } 214*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8,positive_saturation)215*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8, positive_saturation) { 216*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON_V8; 217*4bdc9457SAndroid Build Coastguard Worker 218*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 219*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 220*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 221*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 222*4bdc9457SAndroid Build Coastguard Worker zero_point++) 223*4bdc9457SAndroid Build Coastguard Worker { 224*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 225*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 226*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 227*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 228*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 229*4bdc9457SAndroid Build Coastguard Worker } 230*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neonv8(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 231*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 232*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::max(); 233*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 234*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 235*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 236*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 237*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 238*4bdc9457SAndroid Build Coastguard Worker } 239*4bdc9457SAndroid Build Coastguard Worker } 240*4bdc9457SAndroid Build Coastguard Worker } 241*4bdc9457SAndroid Build Coastguard Worker } 242*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8,negative_saturation)243*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__NEONV8, negative_saturation) { 244*4bdc9457SAndroid Build Coastguard Worker TEST_REQUIRES_ARM_NEON_V8; 245*4bdc9457SAndroid Build Coastguard Worker 246*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 247*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 248*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 249*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 250*4bdc9457SAndroid Build Coastguard Worker zero_point++) 251*4bdc9457SAndroid Build Coastguard Worker { 252*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) zero_point); 253*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 254*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 255*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 256*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 257*4bdc9457SAndroid Build Coastguard Worker } 258*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__neonv8(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 259*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 260*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::min(); 261*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 262*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 263*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 264*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 265*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 266*4bdc9457SAndroid Build Coastguard Worker } 267*4bdc9457SAndroid Build Coastguard Worker } 268*4bdc9457SAndroid Build Coastguard Worker } 269*4bdc9457SAndroid Build Coastguard Worker } 270*4bdc9457SAndroid Build Coastguard Worker #endif // XNN_ARCH_ARM || XNN_ARCH_ARM64 271*4bdc9457SAndroid Build Coastguard Worker 272*4bdc9457SAndroid Build Coastguard Worker #if XNN_ARCH_WASMSIMD || XNN_ARCH_WASMRELAXEDSIMD TEST(CVT__WASMSIMD,positive_normal)273*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD, positive_normal) { 274*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 275*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 276*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 277*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 278*4bdc9457SAndroid Build Coastguard Worker zero_point++) 279*4bdc9457SAndroid Build Coastguard Worker { 280*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 281*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 282*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 283*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 284*4bdc9457SAndroid Build Coastguard Worker } 285*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__wasmsimd(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 286*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 287*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 288*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 289*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 290*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 291*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 292*4bdc9457SAndroid Build Coastguard Worker } 293*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 294*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 295*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 296*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 297*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 298*4bdc9457SAndroid Build Coastguard Worker } 299*4bdc9457SAndroid Build Coastguard Worker } 300*4bdc9457SAndroid Build Coastguard Worker } 301*4bdc9457SAndroid Build Coastguard Worker } 302*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD,negative_normal)303*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD, negative_normal) { 304*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 305*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 306*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 307*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 308*4bdc9457SAndroid Build Coastguard Worker zero_point++) 309*4bdc9457SAndroid Build Coastguard Worker { 310*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = float_as_uint32((float) zero_point); 311*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = 0; n < max_input; n += kBlockSize) { 312*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 313*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 314*4bdc9457SAndroid Build Coastguard Worker } 315*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__wasmsimd(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 316*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 317*4bdc9457SAndroid Build Coastguard Worker long reference_output = std::lrintf(inputs[i]) + long(zero_point); 318*4bdc9457SAndroid Build Coastguard Worker if (inputs[i] >= float(std::numeric_limits<long>::max())) { 319*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::max(); 320*4bdc9457SAndroid Build Coastguard Worker } else if (inputs[i] <= float(std::numeric_limits<long>::min())) { 321*4bdc9457SAndroid Build Coastguard Worker reference_output = std::numeric_limits<uint8_t>::min(); 322*4bdc9457SAndroid Build Coastguard Worker } 323*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, long(outputs[i])) 324*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 325*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 326*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 327*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 328*4bdc9457SAndroid Build Coastguard Worker } 329*4bdc9457SAndroid Build Coastguard Worker } 330*4bdc9457SAndroid Build Coastguard Worker } 331*4bdc9457SAndroid Build Coastguard Worker } 332*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD,positive_saturation)333*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD, positive_saturation) { 334*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 335*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 336*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 337*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 338*4bdc9457SAndroid Build Coastguard Worker zero_point++) 339*4bdc9457SAndroid Build Coastguard Worker { 340*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) (std::numeric_limits<uint8_t>::max() - zero_point)); 341*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 342*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 343*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 344*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(std::min<uint32_t>(n + i, max_input)); 345*4bdc9457SAndroid Build Coastguard Worker } 346*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__wasmsimd(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 347*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 348*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::max(); 349*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 350*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 351*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 352*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 353*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 354*4bdc9457SAndroid Build Coastguard Worker } 355*4bdc9457SAndroid Build Coastguard Worker } 356*4bdc9457SAndroid Build Coastguard Worker } 357*4bdc9457SAndroid Build Coastguard Worker } 358*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD,negative_saturation)359*4bdc9457SAndroid Build Coastguard Worker TEST(CVT__WASMSIMD, negative_saturation) { 360*4bdc9457SAndroid Build Coastguard Worker std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); 361*4bdc9457SAndroid Build Coastguard Worker std::vector<uint8_t, AlignedAllocator<uint8_t, 64>> outputs(kBlockSize); 362*4bdc9457SAndroid Build Coastguard Worker for (int32_t zero_point = std::numeric_limits<uint8_t>::min(); 363*4bdc9457SAndroid Build Coastguard Worker zero_point <= std::numeric_limits<uint8_t>::max(); 364*4bdc9457SAndroid Build Coastguard Worker zero_point++) 365*4bdc9457SAndroid Build Coastguard Worker { 366*4bdc9457SAndroid Build Coastguard Worker const uint32_t min_input = float_as_uint32((float) zero_point); 367*4bdc9457SAndroid Build Coastguard Worker const uint32_t max_input = UINT32_C(0x7F800000); 368*4bdc9457SAndroid Build Coastguard Worker for (uint32_t n = min_input; n < max_input; n += kBlockSize) { 369*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 370*4bdc9457SAndroid Build Coastguard Worker inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::min<uint32_t>(n + i, max_input)); 371*4bdc9457SAndroid Build Coastguard Worker } 372*4bdc9457SAndroid Build Coastguard Worker xnn_math_f32_qu8_cvt__wasmsimd(kBlockSize * sizeof(uint8_t), inputs.data(), outputs.data(), uint8_t(zero_point)); 373*4bdc9457SAndroid Build Coastguard Worker for (uint32_t i = 0; i < kBlockSize; i++) { 374*4bdc9457SAndroid Build Coastguard Worker const int32_t reference_output = std::numeric_limits<uint8_t>::min(); 375*4bdc9457SAndroid Build Coastguard Worker ASSERT_EQ(reference_output, uint32_t(outputs[i])) 376*4bdc9457SAndroid Build Coastguard Worker << "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) 377*4bdc9457SAndroid Build Coastguard Worker << ", reference = " << std::dec << reference_output 378*4bdc9457SAndroid Build Coastguard Worker << ", optimized = " << std::dec << uint32_t(outputs[i]) 379*4bdc9457SAndroid Build Coastguard Worker << ", zero point = " << std::dec << zero_point; 380*4bdc9457SAndroid Build Coastguard Worker } 381*4bdc9457SAndroid Build Coastguard Worker } 382*4bdc9457SAndroid Build Coastguard Worker } 383*4bdc9457SAndroid Build Coastguard Worker } 384*4bdc9457SAndroid Build Coastguard Worker #endif // XNN_ARCH_WASMSIMD || XNN_ARCH_WASMRELAXEDSIMD 385