|  | // Copyright 2022 The Dawn & Tint Authors | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions are met: | 
|  | // | 
|  | // 1. Redistributions of source code must retain the above copyright notice, this | 
|  | //    list of conditions and the following disclaimer. | 
|  | // | 
|  | // 2. Redistributions in binary form must reproduce the above copyright notice, | 
|  | //    this list of conditions and the following disclaimer in the documentation | 
|  | //    and/or other materials provided with the distribution. | 
|  | // | 
|  | // 3. Neither the name of the copyright holder nor the names of its | 
|  | //    contributors may be used to endorse or promote products derived from | 
|  | //    this software without specific prior written permission. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" | 
|  | // AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
|  | // IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE | 
|  | // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE | 
|  | // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
|  | // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR | 
|  | // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER | 
|  | // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, | 
|  | // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  |  | 
|  | #include "src/tint/utils/containers/hashset.h" | 
|  |  | 
|  | #include <array> | 
|  | #include <random> | 
|  | #include <string> | 
|  | #include <tuple> | 
|  | #include <unordered_set> | 
|  |  | 
|  | #include "gmock/gmock.h" | 
|  | #include "src/tint/utils/containers/predicates.h" | 
|  |  | 
|  | namespace tint { | 
|  | namespace { | 
|  |  | 
|  | constexpr std::array kPrimes{ | 
|  | 2,   3,   5,   7,   11,  13,  17,  19,  23,  29,  31,  37,  41,  43,  47,  53, | 
|  | 59,  61,  67,  71,  73,  79,  83,  89,  97,  101, 103, 107, 109, 113, 127, 131, | 
|  | 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, | 
|  | 227, 229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, | 
|  | 313, 317, 331, 337, 347, 349, 353, 359, 367, 373, 379, 383, 389, 397, 401, 409, | 
|  | }; | 
|  |  | 
|  | TEST(Hashset, Empty) { | 
|  | Hashset<std::string, 8> set; | 
|  | EXPECT_EQ(set.Count(), 0u); | 
|  | } | 
|  |  | 
|  | TEST(Hashset, InitializerConstructor) { | 
|  | Hashset<int, 8> set{1, 5, 7}; | 
|  | EXPECT_EQ(set.Count(), 3u); | 
|  | EXPECT_TRUE(set.Contains(1)); | 
|  | EXPECT_FALSE(set.Contains(3)); | 
|  | EXPECT_TRUE(set.Contains(5)); | 
|  | EXPECT_TRUE(set.Contains(7)); | 
|  | EXPECT_FALSE(set.Contains(9)); | 
|  | } | 
|  |  | 
|  | TEST(Hashset, AddRemove) { | 
|  | Hashset<std::string, 8> set; | 
|  | EXPECT_TRUE(set.Add("hello")); | 
|  | EXPECT_EQ(set.Count(), 1u); | 
|  | EXPECT_TRUE(set.Contains("hello")); | 
|  | EXPECT_FALSE(set.Contains("world")); | 
|  | EXPECT_FALSE(set.Add("hello")); | 
|  | EXPECT_EQ(set.Count(), 1u); | 
|  | EXPECT_TRUE(set.Remove("hello")); | 
|  | EXPECT_EQ(set.Count(), 0u); | 
|  | EXPECT_FALSE(set.Contains("hello")); | 
|  | EXPECT_FALSE(set.Contains("world")); | 
|  | } | 
|  |  | 
|  | TEST(Hashset, AddMany) { | 
|  | Hashset<int, 8> set; | 
|  | for (size_t i = 0; i < kPrimes.size(); i++) { | 
|  | int prime = kPrimes[i]; | 
|  | ASSERT_TRUE(set.Add(prime)) << "i: " << i; | 
|  | ASSERT_FALSE(set.Add(prime)) << "i: " << i; | 
|  | ASSERT_EQ(set.Count(), i + 1); | 
|  | } | 
|  | ASSERT_EQ(set.Count(), kPrimes.size()); | 
|  | for (int prime : kPrimes) { | 
|  | ASSERT_TRUE(set.Contains(prime)) << prime; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(Hashset, Iterator) { | 
|  | Hashset<std::string, 8> set; | 
|  | set.Add("one"); | 
|  | set.Add("four"); | 
|  | set.Add("three"); | 
|  | set.Add("two"); | 
|  | EXPECT_THAT(set, testing::UnorderedElementsAre("one", "two", "three", "four")); | 
|  | } | 
|  |  | 
|  | TEST(Hashset, Vector) { | 
|  | Hashset<std::string, 8> set; | 
|  | set.Add("one"); | 
|  | set.Add("four"); | 
|  | set.Add("three"); | 
|  | set.Add("two"); | 
|  | auto vec = set.Vector(); | 
|  | EXPECT_THAT(vec, testing::UnorderedElementsAre("one", "two", "three", "four")); | 
|  | } | 
|  |  | 
|  | TEST(Hashset, Soak) { | 
|  | std::mt19937 rnd; | 
|  | std::unordered_set<std::string> reference; | 
|  | Hashset<std::string, 8> set; | 
|  | for (size_t i = 0; i < 1000000; i++) { | 
|  | std::string value = std::to_string(rnd() & 0x100); | 
|  | switch (rnd() % 5) { | 
|  | case 0: {  // Add | 
|  | auto expected = reference.emplace(value).second; | 
|  | ASSERT_EQ(set.Add(value), expected) << "i: " << i; | 
|  | ASSERT_TRUE(set.Contains(value)) << "i: " << i; | 
|  | break; | 
|  | } | 
|  | case 1: {  // Remove | 
|  | auto expected = reference.erase(value) != 0; | 
|  | ASSERT_EQ(set.Remove(value), expected) << "i: " << i; | 
|  | ASSERT_FALSE(set.Contains(value)) << "i: " << i; | 
|  | break; | 
|  | } | 
|  | case 2: {  // Contains | 
|  | auto expected = reference.count(value) != 0; | 
|  | ASSERT_EQ(set.Contains(value), expected) << "i: " << i; | 
|  | break; | 
|  | } | 
|  | case 3: {  // Copy / Move | 
|  | Hashset<std::string, 8> tmp(set); | 
|  | set = std::move(tmp); | 
|  | break; | 
|  | } | 
|  | case 4: {  // Clear | 
|  | reference.clear(); | 
|  | set.Clear(); | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(HashsetTest, Any) { | 
|  | Hashset<int, 8> set{1, 7, 5, 9}; | 
|  | EXPECT_TRUE(set.Any(Eq(1))); | 
|  | EXPECT_FALSE(set.Any(Eq(2))); | 
|  | EXPECT_FALSE(set.Any(Eq(3))); | 
|  | EXPECT_FALSE(set.Any(Eq(4))); | 
|  | EXPECT_TRUE(set.Any(Eq(5))); | 
|  | EXPECT_FALSE(set.Any(Eq(6))); | 
|  | EXPECT_TRUE(set.Any(Eq(7))); | 
|  | EXPECT_FALSE(set.Any(Eq(8))); | 
|  | EXPECT_TRUE(set.Any(Eq(9))); | 
|  | } | 
|  |  | 
|  | TEST(HashsetTest, All) { | 
|  | Hashset<int, 8> set{1, 7, 5, 9}; | 
|  | EXPECT_FALSE(set.All(Ne(1))); | 
|  | EXPECT_TRUE(set.All(Ne(2))); | 
|  | EXPECT_TRUE(set.All(Ne(3))); | 
|  | EXPECT_TRUE(set.All(Ne(4))); | 
|  | EXPECT_FALSE(set.All(Ne(5))); | 
|  | EXPECT_TRUE(set.All(Ne(6))); | 
|  | EXPECT_FALSE(set.All(Ne(7))); | 
|  | EXPECT_TRUE(set.All(Ne(8))); | 
|  | EXPECT_FALSE(set.All(Ne(9))); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace tint |