b6ecd9881e506a10e672114b00288eb5a094f260
[WebKit-https.git] / Tools / TestWebKitAPI / Tests / WTF / HashMap.cpp
1 /*
2  * Copyright (C) 2011 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27
28 #include "CopyMoveCounter.h"
29 #include "MoveOnly.h"
30 #include <string>
31 #include <wtf/HashMap.h>
32 #include <wtf/text/StringHash.h>
33
34 namespace TestWebKitAPI {
35
36 typedef WTF::HashMap<int, int> IntHashMap;
37
38 TEST(WTF_HashMap, HashTableIteratorComparison)
39 {
40     IntHashMap map;
41     map.add(1, 2);
42     ASSERT_TRUE(map.begin() != map.end());
43     ASSERT_FALSE(map.begin() == map.end());
44
45     IntHashMap::const_iterator begin = map.begin();
46     ASSERT_TRUE(begin == map.begin());
47     ASSERT_TRUE(map.begin() == begin);
48     ASSERT_TRUE(begin != map.end());
49     ASSERT_TRUE(map.end() != begin);
50     ASSERT_FALSE(begin != map.begin());
51     ASSERT_FALSE(map.begin() != begin);
52     ASSERT_FALSE(begin == map.end());
53     ASSERT_FALSE(map.end() == begin);
54 }
55
56 struct TestDoubleHashTraits : HashTraits<double> {
57     static const int minimumTableSize = 8;
58 };
59
60 typedef HashMap<double, int64_t, DefaultHash<double>::Hash, TestDoubleHashTraits> DoubleHashMap;
61
62 static int bucketForKey(double key)
63 {
64     return DefaultHash<double>::Hash::hash(key) & (TestDoubleHashTraits::minimumTableSize - 1);
65 }
66
67 TEST(WTF_HashMap, DoubleHashCollisions)
68 {
69     // The "clobber" key here is one that ends up stealing the bucket that the -0 key
70     // originally wants to be in. This makes the 0 and -0 keys collide and the test then
71     // fails unless the FloatHash::equals() implementation can distinguish them.
72     const double clobberKey = 6;
73     const double zeroKey = 0;
74     const double negativeZeroKey = -zeroKey;
75
76     DoubleHashMap map;
77
78     map.add(clobberKey, 1);
79     map.add(zeroKey, 2);
80     map.add(negativeZeroKey, 3);
81
82     ASSERT_EQ(bucketForKey(clobberKey), bucketForKey(negativeZeroKey));
83     ASSERT_EQ(map.get(clobberKey), 1);
84     ASSERT_EQ(map.get(zeroKey), 2);
85     ASSERT_EQ(map.get(negativeZeroKey), 3);
86 }
87
88 TEST(WTF_HashMap, MoveOnlyValues)
89 {
90     HashMap<unsigned, MoveOnly> moveOnlyValues;
91
92     for (size_t i = 0; i < 100; ++i) {
93         MoveOnly moveOnly(i + 1);
94         moveOnlyValues.set(i + 1, WTF::move(moveOnly));
95     }
96
97     for (size_t i = 0; i < 100; ++i) {
98         auto it = moveOnlyValues.find(i + 1);
99         ASSERT_FALSE(it == moveOnlyValues.end());
100     }
101
102     for (size_t i = 0; i < 50; ++i)
103         ASSERT_EQ(moveOnlyValues.take(i + 1).value(), i + 1);
104
105     for (size_t i = 50; i < 100; ++i)
106         ASSERT_TRUE(moveOnlyValues.remove(i + 1));
107
108     ASSERT_TRUE(moveOnlyValues.isEmpty());
109 }
110
111 TEST(WTF_HashMap, MoveOnlyKeys)
112 {
113     HashMap<MoveOnly, unsigned> moveOnlyKeys;
114
115     for (size_t i = 0; i < 100; ++i) {
116         MoveOnly moveOnly(i + 1);
117         moveOnlyKeys.set(WTF::move(moveOnly), i + 1);
118     }
119
120     for (size_t i = 0; i < 100; ++i) {
121         auto it = moveOnlyKeys.find(MoveOnly(i + 1));
122         ASSERT_FALSE(it == moveOnlyKeys.end());
123     }
124
125     for (size_t i = 0; i < 100; ++i)
126         ASSERT_FALSE(moveOnlyKeys.add(MoveOnly(i + 1), i + 1).isNewEntry);
127
128     for (size_t i = 0; i < 100; ++i)
129         ASSERT_TRUE(moveOnlyKeys.remove(MoveOnly(i + 1)));
130
131     ASSERT_TRUE(moveOnlyKeys.isEmpty());
132 }
133
134 TEST(WTF_HashMap, InitializerList)
135 {
136     HashMap<unsigned, std::string> map = {
137         { 1, "one" },
138         { 2, "two" },
139         { 3, "three" },
140         { 4, "four" },
141     };
142
143     EXPECT_EQ(4, map.size());
144
145     EXPECT_EQ("one", map.get(1));
146     EXPECT_EQ("two", map.get(2));
147     EXPECT_EQ("three", map.get(3));
148     EXPECT_EQ("four", map.get(4));
149     EXPECT_EQ(std::string(), map.get(5));
150 }
151
152 TEST(WTF_HashMap, EfficientGetter)
153 {
154     HashMap<unsigned, CopyMoveCounter> map;
155     map.set(1, CopyMoveCounter());
156
157     {
158         CopyMoveCounter::TestingScope scope;
159         map.get(1);
160         EXPECT_EQ(0U, CopyMoveCounter::constructionCount);
161         EXPECT_EQ(1U, CopyMoveCounter::copyCount);
162         EXPECT_EQ(0U, CopyMoveCounter::moveCount);
163     }
164
165     {
166         CopyMoveCounter::TestingScope scope;
167         map.get(2);
168         EXPECT_EQ(1U, CopyMoveCounter::constructionCount);
169         EXPECT_EQ(0U, CopyMoveCounter::copyCount);
170         EXPECT_EQ(1U, CopyMoveCounter::moveCount);
171     }
172 }
173
174 } // namespace TestWebKitAPI