|
|
|
|
|
|
|
|
|
|
|
#include <algorithm> |
|
#include <cmath> |
|
#include <cstddef> |
|
#include <cstdint> |
|
#include <cstdlib> |
|
#include <iomanip> |
|
#include <ios> |
|
#include <limits> |
|
#include <vector> |
|
|
|
#include <gtest/gtest.h> |
|
|
|
#include <xnnpack/aligned-allocator.h> |
|
#include <xnnpack/common.h> |
|
#include <xnnpack/math.h> |
|
#include <xnnpack/math-stubs.h> |
|
|
|
|
|
constexpr int kBlockSize = 1024; |
|
|
|
#if XNN_ARCH_X86 || XNN_ARCH_X86_64 |
|
TEST(ROUNDNE__SSE_ADDSUB, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE_ADDSUB, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_X86 || XNN_ARCH_X86_64 |
|
TEST(ROUNDNE__SSE2_CVT, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, DISABLED_positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE2_CVT, DISABLED_negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse2_cvt(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_X86 || XNN_ARCH_X86_64 |
|
TEST(ROUNDNE__SSE41, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SSE41, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__sse41(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_ARM || XNN_ARCH_ARM64 |
|
TEST(ROUNDNE__NEON_ADDSUB, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEON_ADDSUB, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neon_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_ARM || XNN_ARCH_ARM64 |
|
TEST(ROUNDNE__NEONV8, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__NEONV8, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__neonv8(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_WASMSIMD || XNN_ARCH_WASMRELAXEDSIMD |
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_ADDSUB, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
#if XNN_ARCH_WASMSIMD || XNN_ARCH_WASMRELAXEDSIMD |
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__WASMSIMD_NATIVE, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__wasmsimd_native(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
#endif |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x00000000); n < UINT32_C(0x4B800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_normal) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x80000000); n < UINT32_C(0xCB800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x4B800000); n < UINT32_C(0x7F800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_integral) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0xCB800000); n < UINT32_C(0xFF800000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), +std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_infinity) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
std::fill(inputs.begin(), inputs.end(), -std::numeric_limits<float>::infinity()); |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[0])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[0])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[0]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[0]); |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(n + i); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7FC00000); n < UINT32_C(0x80000000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | (n + i)); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_snan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output & UINT32_C(0xFFBFFFFF), float_as_uint32(outputs[i]) & UINT32_C(0xFFBFFFFF)) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, positive_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|
|
TEST(ROUNDNE__SCALAR_ADDSUB, negative_snan_to_qnan) { |
|
std::vector<float, AlignedAllocator<float, 64>> inputs(kBlockSize); |
|
std::vector<float, AlignedAllocator<float, 64>> outputs(kBlockSize); |
|
for (uint32_t n = UINT32_C(0x7F800000); n < UINT32_C(0x7FC00000); n += kBlockSize) { |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
inputs[i] = uint32_as_float(UINT32_C(0x80000000) | std::max<uint32_t>(n + i, UINT32_C(0x7F800001))); |
|
} |
|
xnn_math_f32_roundne__scalar_addsub(kBlockSize * sizeof(float), inputs.data(), outputs.data()); |
|
for (uint32_t i = 0; i < kBlockSize; i++) { |
|
const uint32_t reference_output = float_as_uint32(std::nearbyint(inputs[i])); |
|
ASSERT_EQ(reference_output, float_as_uint32(outputs[i])) |
|
<< "input = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(inputs[i]) |
|
<< ", reference = 0x" << std::hex << std::setw(8) << std::setfill('0') << reference_output |
|
<< ", optimized = 0x" << std::hex << std::setw(8) << std::setfill('0') << float_as_uint32(outputs[i]); |
|
} |
|
} |
|
} |
|
|