| // 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 |