// Copyright (c) 2019 Google Inc. // // 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 "source/util/bitutils.h" #include "gmock/gmock.h" namespace spvtools { namespace utils { namespace { using BitUtilsTest = ::testing::Test; TEST(BitUtilsTest, MutateBitsWholeWord) { const uint32_t zero_u32 = 0; const uint32_t max_u32 = ~0; EXPECT_EQ(MutateBits(zero_u32, 0, 0, false), zero_u32); EXPECT_EQ(MutateBits(max_u32, 0, 0, false), max_u32); EXPECT_EQ(MutateBits(zero_u32, 0, 32, false), zero_u32); EXPECT_EQ(MutateBits(zero_u32, 0, 32, true), max_u32); EXPECT_EQ(MutateBits(max_u32, 0, 32, true), max_u32); EXPECT_EQ(MutateBits(max_u32, 0, 32, false), zero_u32); } TEST(BitUtilsTest, MutateBitsLow) { const uint32_t zero_u32 = 0; const uint32_t one_u32 = 1; const uint32_t max_u32 = ~0; EXPECT_EQ(MutateBits(zero_u32, 0, 1, false), zero_u32); EXPECT_EQ(MutateBits(zero_u32, 0, 1, true), one_u32); EXPECT_EQ(MutateBits(max_u32, 0, 1, true), max_u32); EXPECT_EQ(MutateBits(one_u32, 0, 32, false), zero_u32); EXPECT_EQ(MutateBits(one_u32, 0, 1, true), one_u32); EXPECT_EQ(MutateBits(one_u32, 0, 1, false), zero_u32); EXPECT_EQ(MutateBits(zero_u32, 0, 3, true), uint32_t(7)); EXPECT_EQ(MutateBits(uint32_t(7), 0, 2, false), uint32_t(4)); } TEST(BitUtilsTest, MutateBitsHigh) { const uint8_t zero_u8 = 0; const uint8_t one_u8 = 1; const uint8_t max_u8 = 255; EXPECT_EQ(MutateBits(zero_u8, 7, 0, true), zero_u8); EXPECT_EQ(MutateBits(zero_u8, 7, 1, true), uint8_t(128)); EXPECT_EQ(MutateBits(one_u8, 7, 1, true), uint8_t(129)); EXPECT_EQ(MutateBits(max_u8, 7, 1, true), max_u8); EXPECT_EQ(MutateBits(max_u8, 7, 1, false), uint8_t(127)); EXPECT_EQ(MutateBits(max_u8, 6, 2, true), max_u8); EXPECT_EQ(MutateBits(max_u8, 6, 2, false), uint8_t(63)); } TEST(BitUtilsTest, MutateBitsUint8Mid) { const uint8_t zero_u8 = 0; const uint8_t max_u8 = 255; EXPECT_EQ(MutateBits(zero_u8, 1, 2, true), uint8_t(6)); EXPECT_EQ(MutateBits(max_u8, 1, 2, true), max_u8); EXPECT_EQ(MutateBits(max_u8, 1, 2, false), uint8_t(0xF9)); EXPECT_EQ(MutateBits(zero_u8, 2, 3, true), uint8_t(0x1C)); } TEST(BitUtilsTest, MutateBitsUint64Mid) { const uint64_t zero_u64 = 0; const uint64_t max_u64 = ~zero_u64; EXPECT_EQ(MutateBits(zero_u64, 1, 2, true), uint64_t(6)); EXPECT_EQ(MutateBits(max_u64, 1, 2, true), max_u64); EXPECT_EQ(MutateBits(max_u64, 1, 2, false), uint64_t(0xFFFFFFFFFFFFFFF9)); EXPECT_EQ(MutateBits(zero_u64, 2, 3, true), uint64_t(0x000000000000001C)); EXPECT_EQ(MutateBits(zero_u64, 2, 35, true), uint64_t(0x0000001FFFFFFFFC)); EXPECT_EQ(MutateBits(zero_u64, 36, 4, true), uint64_t(0x000000F000000000)); EXPECT_EQ(MutateBits(max_u64, 36, 4, false), uint64_t(0xFFFFFF0FFFFFFFFF)); } TEST(BitUtilsTest, SetHighBitsUint32) { const uint32_t zero_u32 = 0; const uint32_t one_u32 = 1; const uint32_t max_u32 = ~zero_u32; EXPECT_EQ(SetHighBits(zero_u32, 0), zero_u32); EXPECT_EQ(SetHighBits(zero_u32, 1), 0x80000000); EXPECT_EQ(SetHighBits(one_u32, 1), 0x80000001); EXPECT_EQ(SetHighBits(one_u32, 2), 0xC0000001); EXPECT_EQ(SetHighBits(zero_u32, 31), 0xFFFFFFFE); EXPECT_EQ(SetHighBits(zero_u32, 32), max_u32); EXPECT_EQ(SetHighBits(max_u32, 32), max_u32); } TEST(BitUtilsTest, ClearHighBitsUint32) { const uint32_t zero_u32 = 0; const uint32_t one_u32 = 1; const uint32_t max_u32 = ~zero_u32; EXPECT_EQ(ClearHighBits(zero_u32, 0), zero_u32); EXPECT_EQ(ClearHighBits(zero_u32, 1), zero_u32); EXPECT_EQ(ClearHighBits(one_u32, 1), one_u32); EXPECT_EQ(ClearHighBits(one_u32, 31), one_u32); EXPECT_EQ(ClearHighBits(one_u32, 32), zero_u32); EXPECT_EQ(ClearHighBits(max_u32, 0), max_u32); EXPECT_EQ(ClearHighBits(max_u32, 1), 0x7FFFFFFF); EXPECT_EQ(ClearHighBits(max_u32, 2), 0x3FFFFFFF); EXPECT_EQ(ClearHighBits(max_u32, 31), one_u32); EXPECT_EQ(ClearHighBits(max_u32, 32), zero_u32); } TEST(BitUtilsTest, IsBitSetAtPositionZero) { const uint32_t zero_u32 = 0; for (size_t i = 0; i != 32; ++i) { EXPECT_FALSE(IsBitAtPositionSet(zero_u32, i)); } const uint8_t zero_u8 = 0; for (size_t i = 0; i != 8; ++i) { EXPECT_FALSE(IsBitAtPositionSet(zero_u8, i)); } const uint64_t zero_u64 = 0; for (size_t i = 0; i != 64; ++i) { EXPECT_FALSE(IsBitAtPositionSet(zero_u64, i)); } } TEST(BitUtilsTest, IsBitSetAtPositionOne) { const uint32_t one_u32 = 1; for (size_t i = 0; i != 32; ++i) { if (i == 0) { EXPECT_TRUE(IsBitAtPositionSet(one_u32, i)); } else { EXPECT_FALSE(IsBitAtPositionSet(one_u32, i)); } } const uint32_t two_to_17_u32 = 1 << 17; for (size_t i = 0; i != 32; ++i) { if (i == 17) { EXPECT_TRUE(IsBitAtPositionSet(two_to_17_u32, i)); } else { EXPECT_FALSE(IsBitAtPositionSet(two_to_17_u32, i)); } } const uint8_t two_to_4_u8 = 1 << 4; for (size_t i = 0; i != 8; ++i) { if (i == 4) { EXPECT_TRUE(IsBitAtPositionSet(two_to_4_u8, i)); } else { EXPECT_FALSE(IsBitAtPositionSet(two_to_4_u8, i)); } } const uint64_t two_to_55_u64 = uint64_t(1) << 55; for (size_t i = 0; i != 64; ++i) { if (i == 55) { EXPECT_TRUE(IsBitAtPositionSet(two_to_55_u64, i)); } else { EXPECT_FALSE(IsBitAtPositionSet(two_to_55_u64, i)); } } } TEST(BitUtilsTest, IsBitSetAtPositionAll) { const uint32_t max_u32 = ~0; for (size_t i = 0; i != 32; ++i) { EXPECT_TRUE(IsBitAtPositionSet(max_u32, i)); } const uint32_t max_u8 = ~uint8_t(0); for (size_t i = 0; i != 8; ++i) { EXPECT_TRUE(IsBitAtPositionSet(max_u8, i)); } const uint64_t max_u64 = ~uint64_t(0); for (size_t i = 0; i != 64; ++i) { EXPECT_TRUE(IsBitAtPositionSet(max_u64, i)); } } } // namespace } // namespace utils } // namespace spvtools