// Copyright 2019 Google LLC // SPDX-License-Identifier: Apache-2.0 // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. #include #include #undef HWY_TARGET_INCLUDE #define HWY_TARGET_INCLUDE "tests/logical_test.cc" #include "hwy/foreach_target.h" // IWYU pragma: keep #include "hwy/highway.h" #include "hwy/tests/test_util-inl.h" HWY_BEFORE_NAMESPACE(); namespace hwy { namespace HWY_NAMESPACE { struct TestNot { template HWY_NOINLINE void operator()(T /*unused*/, D d) { const Vec v0 = Zero(d); const Vec ones = VecFromMask(d, Eq(v0, v0)); const Vec v1 = Set(d, 1); const Vec vnot1 = Set(d, static_cast(~static_cast(1))); HWY_ASSERT_VEC_EQ(d, v0, Not(ones)); HWY_ASSERT_VEC_EQ(d, ones, Not(v0)); HWY_ASSERT_VEC_EQ(d, v1, Not(vnot1)); HWY_ASSERT_VEC_EQ(d, vnot1, Not(v1)); } }; HWY_NOINLINE void TestAllNot() { ForIntegerTypes(ForPartialVectors()); } struct TestLogical { template HWY_NOINLINE void operator()(T /*unused*/, D d) { const auto v0 = Zero(d); const auto vi = Iota(d, 0); auto v = vi; v = And(v, vi); HWY_ASSERT_VEC_EQ(d, vi, v); v = And(v, v0); HWY_ASSERT_VEC_EQ(d, v0, v); v = Or(v, vi); HWY_ASSERT_VEC_EQ(d, vi, v); v = Or(v, v0); HWY_ASSERT_VEC_EQ(d, vi, v); v = Xor(v, vi); HWY_ASSERT_VEC_EQ(d, v0, v); v = Xor(v, v0); HWY_ASSERT_VEC_EQ(d, v0, v); HWY_ASSERT_VEC_EQ(d, v0, And(v0, vi)); HWY_ASSERT_VEC_EQ(d, v0, And(vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, And(vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, Or(v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, Or(vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, Or(vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, Xor(v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, Xor(vi, v0)); HWY_ASSERT_VEC_EQ(d, v0, Xor(vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, AndNot(v0, vi)); HWY_ASSERT_VEC_EQ(d, v0, AndNot(vi, v0)); HWY_ASSERT_VEC_EQ(d, v0, AndNot(vi, vi)); HWY_ASSERT_VEC_EQ(d, v0, Or3(v0, v0, v0)); HWY_ASSERT_VEC_EQ(d, vi, Or3(v0, vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, Or3(v0, v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, Or3(v0, vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, Or3(vi, v0, v0)); HWY_ASSERT_VEC_EQ(d, vi, Or3(vi, vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, Or3(vi, v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, Or3(vi, vi, vi)); HWY_ASSERT_VEC_EQ(d, v0, Xor3(v0, v0, v0)); HWY_ASSERT_VEC_EQ(d, vi, Xor3(v0, vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, Xor3(v0, v0, vi)); HWY_ASSERT_VEC_EQ(d, v0, Xor3(v0, vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, Xor3(vi, v0, v0)); HWY_ASSERT_VEC_EQ(d, v0, Xor3(vi, vi, v0)); HWY_ASSERT_VEC_EQ(d, v0, Xor3(vi, v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, Xor3(vi, vi, vi)); HWY_ASSERT_VEC_EQ(d, v0, OrAnd(v0, v0, v0)); HWY_ASSERT_VEC_EQ(d, v0, OrAnd(v0, vi, v0)); HWY_ASSERT_VEC_EQ(d, v0, OrAnd(v0, v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, OrAnd(v0, vi, vi)); HWY_ASSERT_VEC_EQ(d, vi, OrAnd(vi, v0, v0)); HWY_ASSERT_VEC_EQ(d, vi, OrAnd(vi, vi, v0)); HWY_ASSERT_VEC_EQ(d, vi, OrAnd(vi, v0, vi)); HWY_ASSERT_VEC_EQ(d, vi, OrAnd(vi, vi, vi)); } }; HWY_NOINLINE void TestAllLogical() { ForAllTypes(ForPartialVectors()); } struct TestCopySign { template HWY_NOINLINE void operator()(T /*unused*/, D d) { const auto v0 = Zero(d); const auto vp = Iota(d, 1); const auto vn = Iota(d, -1E5); // assumes N < 10^5 // Zero remains zero regardless of sign HWY_ASSERT_VEC_EQ(d, v0, CopySign(v0, v0)); HWY_ASSERT_VEC_EQ(d, v0, CopySign(v0, vp)); HWY_ASSERT_VEC_EQ(d, v0, CopySign(v0, vn)); HWY_ASSERT_VEC_EQ(d, v0, CopySignToAbs(v0, v0)); HWY_ASSERT_VEC_EQ(d, v0, CopySignToAbs(v0, vp)); HWY_ASSERT_VEC_EQ(d, v0, CopySignToAbs(v0, vn)); // Positive input, positive sign => unchanged HWY_ASSERT_VEC_EQ(d, vp, CopySign(vp, vp)); HWY_ASSERT_VEC_EQ(d, vp, CopySignToAbs(vp, vp)); // Positive input, negative sign => negated HWY_ASSERT_VEC_EQ(d, Neg(vp), CopySign(vp, vn)); HWY_ASSERT_VEC_EQ(d, Neg(vp), CopySignToAbs(vp, vn)); // Negative input, negative sign => unchanged HWY_ASSERT_VEC_EQ(d, vn, CopySign(vn, vn)); // Negative input, positive sign => negated HWY_ASSERT_VEC_EQ(d, Neg(vn), CopySign(vn, vp)); } }; HWY_NOINLINE void TestAllCopySign() { ForFloatTypes(ForPartialVectors()); } struct TestBroadcastSignBit { template HWY_NOINLINE void operator()(T /*unused*/, D d) { const auto s0 = Zero(d); const auto s1 = Set(d, -1); // all bit set const auto vpos = And(Iota(d, 0), Set(d, LimitsMax())); const auto vneg = Sub(s1, vpos); HWY_ASSERT_VEC_EQ(d, s0, BroadcastSignBit(vpos)); HWY_ASSERT_VEC_EQ(d, s0, BroadcastSignBit(Set(d, LimitsMax()))); HWY_ASSERT_VEC_EQ(d, s1, BroadcastSignBit(vneg)); HWY_ASSERT_VEC_EQ(d, s1, BroadcastSignBit(Set(d, LimitsMin()))); HWY_ASSERT_VEC_EQ(d, s1, BroadcastSignBit(Set(d, LimitsMin() / 2))); } }; HWY_NOINLINE void TestAllBroadcastSignBit() { ForSignedTypes(ForPartialVectors()); } struct TestTestBit { template HWY_NOINLINE void operator()(T /*unused*/, D d) { const size_t kNumBits = sizeof(T) * 8; for (size_t i = 0; i < kNumBits; ++i) { const Vec bit1 = Set(d, static_cast(1ull << i)); const Vec bit2 = Set(d, static_cast(1ull << ((i + 1) % kNumBits))); const Vec bit3 = Set(d, static_cast(1ull << ((i + 2) % kNumBits))); const Vec bits12 = Or(bit1, bit2); const Vec bits23 = Or(bit2, bit3); HWY_ASSERT(AllTrue(d, TestBit(bit1, bit1))); HWY_ASSERT(AllTrue(d, TestBit(bits12, bit1))); HWY_ASSERT(AllTrue(d, TestBit(bits12, bit2))); HWY_ASSERT(AllFalse(d, TestBit(bits12, bit3))); HWY_ASSERT(AllFalse(d, TestBit(bits23, bit1))); HWY_ASSERT(AllFalse(d, TestBit(bit1, bit2))); HWY_ASSERT(AllFalse(d, TestBit(bit2, bit1))); HWY_ASSERT(AllFalse(d, TestBit(bit1, bit3))); HWY_ASSERT(AllFalse(d, TestBit(bit3, bit1))); HWY_ASSERT(AllFalse(d, TestBit(bit2, bit3))); HWY_ASSERT(AllFalse(d, TestBit(bit3, bit2))); } } }; HWY_NOINLINE void TestAllTestBit() { ForIntegerTypes(ForPartialVectors()); } class TestBitwiseIfThenElse { private: template static T ValueFromBitPattern(hwy::FloatTag /* type_tag */, T /* unused */, uint64_t bits) { using TI = MakeSigned; return ConvertScalarTo( ConvertScalarTo(static_cast(bits & MantissaMask())) + MantissaEnd()); } template static MakeUnsigned ValueFromBitPattern(hwy::NonFloatTag /* type_tag */, T /* unused */, uint64_t bits) { return static_cast>(bits); } public: template HWY_NOINLINE void operator()(T /*unused*/, D d) { using TU = MakeUnsigned; using TVal = RemoveConst(), T(), uint64_t{0}))>; static_assert(!IsFloat() || IsSame(), "TVal should be the same as T if T is a floating-point type"); static_assert(IsFloat() || IsSame(), "TVal should be the same as TU if T is a integer type"); static TVal a0 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0x0FF00FF00FF00FF0u}); static TVal b0 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0x33CC33CC33CC33CCu}); static TVal c0 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0x55AA55AA55AA55AAu}); static TVal a1 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0xF00FF00FF00FF00Fu}); static TVal b1 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0xCC33CC33CC33CC33u}); static TVal c1 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0xAA55AA55AA55AA55u}); const RebindToUnsigned du; const Rebind d_val; const auto v_a0 = BitCast(d, Set(d_val, a0)); const auto v_b0 = BitCast(d, Set(d_val, b0)); const auto v_c0 = BitCast(d, Set(d_val, c0)); const auto v_a1 = BitCast(d, Set(d_val, a1)); const auto v_b1 = BitCast(d, Set(d_val, b1)); const auto v_c1 = BitCast(d, Set(d_val, c1)); static TVal expected_1 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0x53CA53CA53CA53CAu}); HWY_ASSERT_VEC_EQ(d, BitCast(d, Set(d_val, expected_1)), BitwiseIfThenElse(v_a0, v_b0, v_c0)); static TVal expected_2 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0xCA53CA53CA53CA53u}); HWY_ASSERT_VEC_EQ(d, BitCast(d, Set(d_val, expected_2)), BitwiseIfThenElse(v_a1, v_b1, v_c1)); static TVal expected_3 = ValueFromBitPattern(IsFloatTag(), T(), uint64_t{0x1DB81DB81DB81DB8u}); HWY_ASSERT_VEC_EQ(d, BitCast(d, Set(d_val, expected_3)), BitwiseIfThenElse(v_b1, v_a0, v_c0)); const auto v_all_ones = BitCast(d, Set(du, static_cast(-1))); HWY_ASSERT_VEC_EQ(d, v_a0, BitwiseIfThenElse(v_all_ones, v_a0, v_b0)); HWY_ASSERT_VEC_EQ(d, v_b0, BitwiseIfThenElse(Zero(d), v_a0, v_b0)); } }; HWY_NOINLINE void TestAllBitwiseIfThenElse() { ForAllTypes(ForPartialVectors()); } // NOLINTNEXTLINE(google-readability-namespace-comments) } // namespace HWY_NAMESPACE } // namespace hwy HWY_AFTER_NAMESPACE(); #if HWY_ONCE namespace hwy { HWY_BEFORE_TEST(HwyLogicalTest); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllNot); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllLogical); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllCopySign); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllBroadcastSignBit); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllTestBit); HWY_EXPORT_AND_TEST_P(HwyLogicalTest, TestAllBitwiseIfThenElse); } // namespace hwy #endif