1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include <algorithm>
18 #include <forward_list>
19 #include <vector>
20 
21 #include "gtest/gtest.h"
22 #include "intrusive_forward_list.h"
23 
24 namespace art {
25 
26 struct IFLTestValue : public IntrusiveForwardListNode<IFLTestValue> {
27   // Deliberately not explicit.
IFLTestValueart::IFLTestValue28   IFLTestValue(int v) : value(v) { }  // NOLINT(runtime/explicit)
29 
30   int value;
31 };
32 using IFLTestValueList = IntrusiveForwardList<IFLTestValue>;
33 using ConstIFLTestValueList = IntrusiveForwardList<const IFLTestValue>;
34 
operator ==(const IFLTestValue & lhs,const IFLTestValue & rhs)35 bool operator==(const IFLTestValue& lhs, const IFLTestValue& rhs) {
36   return lhs.value == rhs.value;
37 }
38 
operator <(const IFLTestValue & lhs,const IFLTestValue & rhs)39 bool operator<(const IFLTestValue& lhs, const IFLTestValue& rhs) {
40   return lhs.value < rhs.value;
41 }
42 
43 struct IFLTestValue2 {
44   // Deliberately not explicit.
IFLTestValue2art::IFLTestValue245   IFLTestValue2(int v) : hook(), value(v) { }  // NOLINT(runtime/explicit)
46 
47   IntrusiveForwardListHook hook;
48   int value;
49 };
50 using IFLTestValue2List =
51     IntrusiveForwardList<IFLTestValue2, IntrusiveForwardListMemberHookTraits<IFLTestValue2>>;
52 
operator ==(const IFLTestValue2 & lhs,const IFLTestValue2 & rhs)53 bool operator==(const IFLTestValue2& lhs, const IFLTestValue2& rhs) {
54   return lhs.value == rhs.value;
55 }
56 
operator <(const IFLTestValue2 & lhs,const IFLTestValue2 & rhs)57 bool operator<(const IFLTestValue2& lhs, const IFLTestValue2& rhs) {
58   return lhs.value < rhs.value;
59 }
60 
61 #define ASSERT_LISTS_EQUAL(expected, value)                                         \
62   do {                                                                              \
63     ASSERT_EQ((expected).empty(), (value).empty());                                 \
64     ASSERT_EQ(std::distance((expected).begin(), (expected).end()),                  \
65               std::distance((value).begin(), (value).end()));                       \
66     ASSERT_TRUE(std::equal((expected).begin(), (expected).end(), (value).begin())); \
67   } while (false)
68 
69 class IntrusiveForwardListTest : public testing::Test {
70  public:
71   template <typename ListType>
72   void IteratorToConstIterator();
73 
74   template <typename ListType>
75   void IteratorOperators();
76 
77   template <typename ListType>
78   void ConstructRange();
79 
80   template <typename ListType>
81   void Assign();
82 
83   template <typename ListType>
84   void PushPop();
85 
86   template <typename ListType>
87   void InsertAfter1();
88 
89   template <typename ListType>
90   void InsertAfter2();
91 
92   template <typename ListType>
93   void EraseAfter1();
94 
95   template <typename ListType>
96   void EraseAfter2();
97 
98   template <typename ListType>
99   void SwapClear();
100 
101   template <typename ListType>
102   void SpliceAfter();
103 
104   template <typename ListType>
105   void Remove();
106 
107   template <typename ListType>
108   void Unique();
109 
110   template <typename ListType>
111   void Merge();
112 
113   template <typename ListType>
114   void Sort1();
115 
116   template <typename ListType>
117   void Sort2();
118 
119   template <typename ListType>
120   void Reverse();
121 
122   template <typename ListType>
123   void ModifyValue();
124 };
125 
126 template <typename ListType>
IteratorToConstIterator()127 void IntrusiveForwardListTest::IteratorToConstIterator() {
128   ListType ifl;
129   typename ListType::iterator begin = ifl.begin();
130   typename ListType::const_iterator cbegin = ifl.cbegin();
131   typename ListType::const_iterator converted_begin = begin;
132   ASSERT_TRUE(converted_begin == cbegin);
133 }
134 
TEST_F(IntrusiveForwardListTest,IteratorToConstIterator)135 TEST_F(IntrusiveForwardListTest, IteratorToConstIterator) {
136   IteratorToConstIterator<IFLTestValueList>();
137   IteratorToConstIterator<ConstIFLTestValueList>();
138   IteratorToConstIterator<IFLTestValue2List>();
139 }
140 
141 template <typename ListType>
IteratorOperators()142 void IntrusiveForwardListTest::IteratorOperators() {
143   using ValueType = typename ListType::value_type;
144   ListType ifl;
145   ASSERT_TRUE(ifl.begin() == ifl.cbegin());
146   ASSERT_FALSE(ifl.begin() != ifl.cbegin());
147   ASSERT_TRUE(ifl.end() == ifl.cend());
148   ASSERT_FALSE(ifl.end() != ifl.cend());
149 
150   ASSERT_TRUE(ifl.begin() == ifl.end());  // Empty.
151   ASSERT_FALSE(ifl.begin() != ifl.end());  // Empty.
152 
153   ValueType value(1);
154   ifl.insert_after(ifl.cbefore_begin(), value);
155 
156   ASSERT_FALSE(ifl.begin() == ifl.end());  // Not empty.
157   ASSERT_TRUE(ifl.begin() != ifl.end());  // Not empty.
158 }
159 
TEST_F(IntrusiveForwardListTest,IteratorOperators)160 TEST_F(IntrusiveForwardListTest, IteratorOperators) {
161   IteratorOperators<IFLTestValueList>();
162   IteratorOperators<ConstIFLTestValueList>();
163   IteratorOperators<IFLTestValue2List>();
164 }
165 
166 template <typename ListType>
ConstructRange()167 void IntrusiveForwardListTest::ConstructRange() {
168   using ValueType = typename ListType::value_type;
169   std::forward_list<int> ref({ 1, 2, 7 });
170   std::vector<ValueType> storage(ref.begin(), ref.end());
171   ListType ifl(storage.begin(), storage.end());
172   ASSERT_LISTS_EQUAL(ref, ifl);
173 }
174 
TEST_F(IntrusiveForwardListTest,ConstructRange)175 TEST_F(IntrusiveForwardListTest, ConstructRange) {
176   ConstructRange<IFLTestValueList>();
177   ConstructRange<ConstIFLTestValueList>();
178   ConstructRange<IFLTestValue2List>();
179 }
180 
181 template <typename ListType>
Assign()182 void IntrusiveForwardListTest::Assign() {
183   using ValueType = typename ListType::value_type;
184   std::forward_list<int> ref1({ 2, 8, 5 });
185   std::vector<ValueType> storage1(ref1.begin(), ref1.end());
186   ListType ifl;
187   ifl.assign(storage1.begin(), storage1.end());
188   ASSERT_LISTS_EQUAL(ref1, ifl);
189   std::forward_list<int> ref2({ 7, 1, 3 });
190   std::vector<ValueType> storage2(ref2.begin(), ref2.end());
191   ifl.assign(storage2.begin(), storage2.end());
192   ASSERT_LISTS_EQUAL(ref2, ifl);
193 }
194 
TEST_F(IntrusiveForwardListTest,Assign)195 TEST_F(IntrusiveForwardListTest, Assign) {
196   Assign<IFLTestValueList>();
197   Assign<ConstIFLTestValueList>();
198   Assign<IFLTestValue2List>();
199 }
200 
201 template <typename ListType>
PushPop()202 void IntrusiveForwardListTest::PushPop() {
203   using ValueType = typename ListType::value_type;
204   ValueType value3(3);
205   ValueType value7(7);
206   std::forward_list<int> ref;
207   ListType ifl;
208   ASSERT_LISTS_EQUAL(ref, ifl);
209   ref.push_front(3);
210   ifl.push_front(value3);
211   ASSERT_LISTS_EQUAL(ref, ifl);
212   ASSERT_EQ(3, ifl.front());
213   ref.push_front(7);
214   ifl.push_front(value7);
215   ASSERT_LISTS_EQUAL(ref, ifl);
216   ASSERT_EQ(7, ifl.front());
217   ref.pop_front();
218   ifl.pop_front();
219   ASSERT_LISTS_EQUAL(ref, ifl);
220   ASSERT_EQ(3, ifl.front());
221   ref.pop_front();
222   ifl.pop_front();
223   ASSERT_LISTS_EQUAL(ref, ifl);
224 }
225 
TEST_F(IntrusiveForwardListTest,PushPop)226 TEST_F(IntrusiveForwardListTest, PushPop) {
227   PushPop<IFLTestValueList>();
228   PushPop<ConstIFLTestValueList>();
229   PushPop<IFLTestValue2List>();
230 }
231 
232 template <typename ListType>
InsertAfter1()233 void IntrusiveForwardListTest::InsertAfter1() {
234   using ValueType = typename ListType::value_type;
235   ValueType value4(4);
236   ValueType value8(8);
237   ValueType value5(5);
238   ValueType value3(3);
239   std::forward_list<int> ref;
240   ListType ifl;
241 
242   auto ref_it = ref.insert_after(ref.before_begin(), 4);
243   auto ifl_it = ifl.insert_after(ifl.before_begin(), value4);
244   ASSERT_LISTS_EQUAL(ref, ifl);
245   ASSERT_EQ(*ref_it, *ifl_it);
246   CHECK(ref_it == ref.begin());
247   ASSERT_TRUE(ifl_it == ifl.begin());
248 
249   ref_it = ref.insert_after(ref.begin(), 8);
250   ifl_it = ifl.insert_after(ifl.begin(), value8);
251   ASSERT_LISTS_EQUAL(ref, ifl);
252   ASSERT_EQ(*ref_it, *ifl_it);
253   CHECK(ref_it != ref.end());
254   ASSERT_TRUE(ifl_it != ifl.end());
255   CHECK(++ref_it == ref.end());
256   ASSERT_TRUE(++ifl_it == ifl.end());
257 
258   ref_it = ref.insert_after(ref.begin(), 5);
259   ifl_it = ifl.insert_after(ifl.begin(), value5);
260   ASSERT_LISTS_EQUAL(ref, ifl);
261   ASSERT_EQ(*ref_it, *ifl_it);
262 
263   ref_it = ref.insert_after(ref_it, 3);
264   ifl_it = ifl.insert_after(ifl_it, value3);
265   ASSERT_LISTS_EQUAL(ref, ifl);
266   ASSERT_EQ(*ref_it, *ifl_it);
267 }
268 
TEST_F(IntrusiveForwardListTest,InsertAfter1)269 TEST_F(IntrusiveForwardListTest, InsertAfter1) {
270   InsertAfter1<IFLTestValueList>();
271   InsertAfter1<ConstIFLTestValueList>();
272   InsertAfter1<IFLTestValue2List>();
273 }
274 
275 template <typename ListType>
InsertAfter2()276 void IntrusiveForwardListTest::InsertAfter2() {
277   using ValueType = typename ListType::value_type;
278   std::forward_list<int> ref;
279   ListType ifl;
280 
281   auto ref_it = ref.insert_after(ref.before_begin(), { 2, 8, 5 });
282   std::vector<ValueType> storage1({ { 2 }, { 8 }, { 5 } });
283   auto ifl_it = ifl.insert_after(ifl.before_begin(), storage1.begin(), storage1.end());
284   ASSERT_LISTS_EQUAL(ref, ifl);
285   ASSERT_EQ(*ref_it, *ifl_it);
286 
287   std::vector<ValueType> storage2({ { 7 }, { 2 } });
288   ref_it = ref.insert_after(ref.begin(), { 7, 2 });
289   ifl_it = ifl.insert_after(ifl.begin(), storage2.begin(), storage2.end());
290   ASSERT_LISTS_EQUAL(ref, ifl);
291   ASSERT_EQ(*ref_it, *ifl_it);
292 
293   std::vector<ValueType> storage3({ { 1 }, { 3 }, { 4 }, { 9 } });
294   ref_it = ref.begin();
295   ifl_it = ifl.begin();
296   std::advance(ref_it, std::distance(ref.begin(), ref.end()) - 1);
297   std::advance(ifl_it, std::distance(ifl.begin(), ifl.end()) - 1);
298   ref_it = ref.insert_after(ref_it, { 1, 3, 4, 9 });
299   ifl_it = ifl.insert_after(ifl_it, storage3.begin(), storage3.end());
300   ASSERT_LISTS_EQUAL(ref, ifl);
301 }
302 
TEST_F(IntrusiveForwardListTest,InsertAfter2)303 TEST_F(IntrusiveForwardListTest, InsertAfter2) {
304   InsertAfter2<IFLTestValueList>();
305   InsertAfter2<ConstIFLTestValueList>();
306   InsertAfter2<IFLTestValue2List>();
307 }
308 
309 template <typename ListType>
EraseAfter1()310 void IntrusiveForwardListTest::EraseAfter1() {
311   using ValueType = typename ListType::value_type;
312   std::forward_list<int> ref({ 1, 2, 7, 4, 5 });
313   std::vector<ValueType> storage(ref.begin(), ref.end());
314   ListType ifl(storage.begin(), storage.end());
315   ASSERT_LISTS_EQUAL(ref, ifl);
316   CHECK_EQ(std::distance(ref.begin(), ref.end()), 5);
317 
318   auto ref_it = ref.begin();
319   auto ifl_it = ifl.begin();
320   std::advance(ref_it, 2);
321   std::advance(ifl_it, 2);
322   ref_it = ref.erase_after(ref_it);
323   ifl_it = ifl.erase_after(ifl_it);
324   ASSERT_LISTS_EQUAL(ref, ifl);
325   CHECK_EQ(std::distance(ref.begin(), ref.end()), 4);
326   CHECK(ref_it != ref.end());
327   ASSERT_TRUE(ifl_it != ifl.end());
328   CHECK(++ref_it == ref.end());
329   ASSERT_TRUE(++ifl_it == ifl.end());
330 
331   ref_it = ref.begin();
332   ifl_it = ifl.begin();
333   std::advance(ref_it, 2);
334   std::advance(ifl_it, 2);
335   ref_it = ref.erase_after(ref_it);
336   ifl_it = ifl.erase_after(ifl_it);
337   ASSERT_LISTS_EQUAL(ref, ifl);
338   CHECK_EQ(std::distance(ref.begin(), ref.end()), 3);
339   CHECK(ref_it == ref.end());
340   ASSERT_TRUE(ifl_it == ifl.end());
341 
342   ref_it = ref.erase_after(ref.begin());
343   ifl_it = ifl.erase_after(ifl.begin());
344   ASSERT_LISTS_EQUAL(ref, ifl);
345   CHECK_EQ(std::distance(ref.begin(), ref.end()), 2);
346   CHECK(ref_it != ref.end());
347   ASSERT_TRUE(ifl_it != ifl.end());
348   CHECK(++ref_it == ref.end());
349   ASSERT_TRUE(++ifl_it == ifl.end());
350 
351   ref_it = ref.erase_after(ref.before_begin());
352   ifl_it = ifl.erase_after(ifl.before_begin());
353   ASSERT_LISTS_EQUAL(ref, ifl);
354   CHECK_EQ(std::distance(ref.begin(), ref.end()), 1);
355   CHECK(ref_it == ref.begin());
356   ASSERT_TRUE(ifl_it == ifl.begin());
357 
358   ref_it = ref.erase_after(ref.before_begin());
359   ifl_it = ifl.erase_after(ifl.before_begin());
360   ASSERT_LISTS_EQUAL(ref, ifl);
361   CHECK_EQ(std::distance(ref.begin(), ref.end()), 0);
362   CHECK(ref_it == ref.begin());
363   ASSERT_TRUE(ifl_it == ifl.begin());
364 }
365 
TEST_F(IntrusiveForwardListTest,EraseAfter1)366 TEST_F(IntrusiveForwardListTest, EraseAfter1) {
367   EraseAfter1<IFLTestValueList>();
368   EraseAfter1<ConstIFLTestValueList>();
369   EraseAfter1<IFLTestValue2List>();
370 }
371 
372 template <typename ListType>
EraseAfter2()373 void IntrusiveForwardListTest::EraseAfter2() {
374   using ValueType = typename ListType::value_type;
375   std::forward_list<int> ref({ 1, 2, 7, 4, 5, 3, 2, 8, 9 });
376   std::vector<ValueType> storage(ref.begin(), ref.end());
377   ListType ifl(storage.begin(), storage.end());
378   ASSERT_LISTS_EQUAL(ref, ifl);
379   CHECK_EQ(std::distance(ref.begin(), ref.end()), 9);
380 
381   auto ref_it = ref.begin();
382   auto ifl_it = ifl.begin();
383   std::advance(ref_it, 3);
384   std::advance(ifl_it, 3);
385   ref_it = ref.erase_after(ref.begin(), ref_it);
386   ifl_it = ifl.erase_after(ifl.begin(), ifl_it);
387   ASSERT_LISTS_EQUAL(ref, ifl);
388   ASSERT_EQ(std::distance(ref.begin(), ref_it), std::distance(ifl.begin(), ifl_it));
389   CHECK_EQ(std::distance(ref.begin(), ref.end()), 7);
390 
391   ref_it = ref.erase_after(ref_it, ref.end());
392   ifl_it = ifl.erase_after(ifl_it, ifl.end());
393   ASSERT_LISTS_EQUAL(ref, ifl);
394   CHECK(ref_it == ref.end());
395   ASSERT_TRUE(ifl_it == ifl.end());
396   CHECK_EQ(std::distance(ref.begin(), ref.end()), 2);
397 
398   ref_it = ref.erase_after(ref.before_begin(), ref.end());
399   ifl_it = ifl.erase_after(ifl.before_begin(), ifl.end());
400   ASSERT_LISTS_EQUAL(ref, ifl);
401   CHECK(ref_it == ref.end());
402   ASSERT_TRUE(ifl_it == ifl.end());
403   CHECK_EQ(std::distance(ref.begin(), ref.end()), 0);
404 }
405 
TEST_F(IntrusiveForwardListTest,EraseAfter2)406 TEST_F(IntrusiveForwardListTest, EraseAfter2) {
407   EraseAfter2<IFLTestValueList>();
408   EraseAfter2<ConstIFLTestValueList>();
409   EraseAfter2<IFLTestValue2List>();
410 }
411 
412 template <typename ListType>
SwapClear()413 void IntrusiveForwardListTest::SwapClear() {
414   using ValueType = typename ListType::value_type;
415   std::forward_list<int> ref1({ 1, 2, 7 });
416   std::vector<ValueType> storage1(ref1.begin(), ref1.end());
417   ListType ifl1(storage1.begin(), storage1.end());
418   std::forward_list<int> ref2({ 3, 8, 6 });
419   std::vector<ValueType> storage2(ref2.begin(), ref2.end());
420   ListType ifl2(storage2.begin(), storage2.end());
421   ASSERT_LISTS_EQUAL(ref1, ifl1);
422   ASSERT_LISTS_EQUAL(ref2, ifl2);
423   ref1.swap(ref2);
424   ifl1.swap(ifl2);
425   ASSERT_LISTS_EQUAL(ref1, ifl1);
426   ASSERT_LISTS_EQUAL(ref2, ifl2);
427   ref1.clear();
428   ifl1.clear();
429   ASSERT_LISTS_EQUAL(ref1, ifl1);
430   ASSERT_LISTS_EQUAL(ref2, ifl2);
431   swap(ref1, ref2);
432   swap(ifl1, ifl2);
433   ASSERT_LISTS_EQUAL(ref1, ifl1);
434   ASSERT_LISTS_EQUAL(ref2, ifl2);
435   ref1.clear();
436   ifl1.clear();
437   ASSERT_LISTS_EQUAL(ref1, ifl1);
438   ASSERT_LISTS_EQUAL(ref2, ifl2);
439 }
440 
TEST_F(IntrusiveForwardListTest,SwapClear)441 TEST_F(IntrusiveForwardListTest, SwapClear) {
442   SwapClear<IFLTestValueList>();
443   SwapClear<ConstIFLTestValueList>();
444   SwapClear<IFLTestValue2List>();
445 }
446 
447 template <typename ListType>
SpliceAfter()448 void IntrusiveForwardListTest::SpliceAfter() {
449   using ValueType = typename ListType::value_type;
450   std::forward_list<int> ref1({ 3, 1, 2, 7, 4, 5, 4, 8, 7 });
451   std::forward_list<int> ref2;
452   std::vector<ValueType> storage(ref1.begin(), ref1.end());
453   ListType ifl1(storage.begin(), storage.end());
454   ListType ifl2;
455   ASSERT_LISTS_EQUAL(ref1, ifl1);
456   ASSERT_LISTS_EQUAL(ref2, ifl2);
457 
458   // Move everything to ref2/ifl2.
459   ref2.splice_after(ref2.before_begin(), ref1);
460   ifl2.splice_after(ifl2.before_begin(), ifl1);
461   ASSERT_LISTS_EQUAL(ref1, ifl1);
462   ASSERT_LISTS_EQUAL(ref2, ifl2);
463 
464   // Move first element (3) to ref1/ifl1.
465   ref1.splice_after(ref1.before_begin(), ref2, ref2.before_begin());
466   ifl1.splice_after(ifl1.before_begin(), ifl2, ifl2.before_begin());
467   ASSERT_LISTS_EQUAL(ref1, ifl1);
468   ASSERT_LISTS_EQUAL(ref2, ifl2);
469 
470   // Move second element (2) to ref1/ifl1 after the first element (3).
471   ref1.splice_after(ref1.begin(), ref2, ref2.begin());
472   ifl1.splice_after(ifl1.begin(), ifl2, ifl2.begin());
473   ASSERT_LISTS_EQUAL(ref1, ifl1);
474   ASSERT_LISTS_EQUAL(ref2, ifl2);
475 
476   // Move everything from ref2/ifl2 between the 2 elements now in ref1/ifl1.
477   ref1.splice_after(ref1.begin(), ref2);
478   ifl1.splice_after(ifl1.begin(), ifl2);
479   ASSERT_LISTS_EQUAL(ref1, ifl1);
480   ASSERT_LISTS_EQUAL(ref2, ifl2);
481 
482   std::forward_list<int> check({ 3, 1, 7, 4, 5, 4, 8, 7, 2 });
483   ASSERT_LISTS_EQUAL(check, ifl1);
484   ASSERT_TRUE(ifl2.empty());
485 
486   // Empty splice_after().
487   ref2.splice_after(
488       ref2.before_begin(), ref1, ref1.before_begin(), ref1.begin());
489   ifl2.splice_after(ifl2.before_begin(), ifl1, ifl1.before_begin(), ifl1.begin());
490   ASSERT_LISTS_EQUAL(ref1, ifl1);
491   ASSERT_LISTS_EQUAL(ref2, ifl2);
492 
493   // Move { 1, 7 } to ref2/ifl2.
494   auto ref_it = ref1.begin();
495   auto ifl_it = ifl1.begin();
496   std::advance(ref_it, 3);
497   std::advance(ifl_it, 3);
498   ref2.splice_after(ref2.before_begin(), ref1, ref1.begin(), ref_it);
499   ifl2.splice_after(ifl2.before_begin(), ifl1, ifl1.begin(), ifl_it);
500   ASSERT_LISTS_EQUAL(ref1, ifl1);
501   ASSERT_LISTS_EQUAL(ref2, ifl2);
502 
503   // Move { 8, 7, 2 } to the beginning of ref1/ifl1.
504   ref_it = ref1.begin();
505   ifl_it = ifl1.begin();
506   std::advance(ref_it, 3);
507   std::advance(ifl_it, 3);
508   ref1.splice_after(ref1.before_begin(), ref1, ref_it, ref1.end());
509   ifl1.splice_after(ifl1.before_begin(), ifl1, ifl_it, ifl1.end());
510   ASSERT_LISTS_EQUAL(ref1, ifl1);
511 
512   check.assign({ 8, 7, 2, 3, 4, 5, 4 });
513   ASSERT_LISTS_EQUAL(check, ifl1);
514   check.assign({ 1, 7 });
515   ASSERT_LISTS_EQUAL(check, ifl2);
516 
517   // Move all but the first element to ref2/ifl2.
518   ref_it = ref2.begin();
519   ifl_it = ifl2.begin();
520   std::advance(ref_it, 1);
521   std::advance(ifl_it, 1);
522   ref2.splice_after(ref_it, ref1, ref1.begin(), ref1.end());
523   ifl2.splice_after(ifl_it, ifl1, ifl1.begin(), ifl1.end());
524   ASSERT_LISTS_EQUAL(ref1, ifl1);
525   ASSERT_LISTS_EQUAL(ref2, ifl2);
526 
527   check.assign({8});
528   ASSERT_LISTS_EQUAL(check, ifl1);
529 
530   // Move the first element of ref1/ifl1 to the beginning of ref1/ifl1 (do nothing).
531   ref1.splice_after(ref1.before_begin(), ref1, ref1.before_begin());
532   ifl1.splice_after(ifl1.before_begin(), ifl1, ifl1.before_begin());
533   ASSERT_LISTS_EQUAL(ref1, ifl1);
534   ASSERT_LISTS_EQUAL(check, ifl1);
535 
536   // Move the first element of ref2/ifl2 after itself (do nothing).
537   ref1.splice_after(ref1.begin(), ref1, ref1.before_begin());
538   ifl1.splice_after(ifl1.begin(), ifl1, ifl1.before_begin());
539   ASSERT_LISTS_EQUAL(ref1, ifl1);
540   ASSERT_LISTS_EQUAL(check, ifl1);
541 
542   check.assign({ 1, 7, 7, 2, 3, 4, 5, 4 });
543   ASSERT_LISTS_EQUAL(check, ifl2);
544 
545   // Move the first element of ref2/ifl2 to the beginning of ref2/ifl2 (do nothing).
546   ref2.splice_after(ref2.before_begin(), ref2, ref2.before_begin());
547   ifl2.splice_after(ifl2.before_begin(), ifl2, ifl2.before_begin());
548   ASSERT_LISTS_EQUAL(ref2, ifl2);
549   ASSERT_LISTS_EQUAL(check, ifl2);
550 
551   // Move the first element of ref2/ifl2 after itself (do nothing).
552   ref2.splice_after(ref2.begin(), ref2, ref2.before_begin());
553   ifl2.splice_after(ifl2.begin(), ifl2, ifl2.before_begin());
554   ASSERT_LISTS_EQUAL(ref2, ifl2);
555   ASSERT_LISTS_EQUAL(check, ifl2);
556 }
557 
TEST_F(IntrusiveForwardListTest,SpliceAfter)558 TEST_F(IntrusiveForwardListTest, SpliceAfter) {
559   SpliceAfter<IFLTestValueList>();
560   SpliceAfter<ConstIFLTestValueList>();
561   SpliceAfter<IFLTestValue2List>();
562 }
563 
564 template <typename ListType>
Remove()565 void IntrusiveForwardListTest::Remove() {
566   using ValueType = typename ListType::value_type;
567   std::forward_list<int> ref({ 3, 1, 2, 7, 4, 5, 4, 8, 7 });
568   std::vector<ValueType> storage(ref.begin(), ref.end());
569   ListType ifl(storage.begin(), storage.end());
570   ASSERT_LISTS_EQUAL(ref, ifl);
571   ref.remove(1);
572   ifl.remove(1);
573   ASSERT_LISTS_EQUAL(ref, ifl);
574   ref.remove(4);
575   ifl.remove(4);
576   ASSERT_LISTS_EQUAL(ref, ifl);
577   auto odd = [](ValueType value) { return (value.value & 1) != 0; };
578   ref.remove_if(odd);
579   ifl.remove_if(odd);
580   ASSERT_LISTS_EQUAL(ref, ifl);
581   auto all = []([[maybe_unused]] ValueType value) { return true; };
582   ref.remove_if(all);
583   ifl.remove_if(all);
584   ASSERT_LISTS_EQUAL(ref, ifl);
585 }
586 
TEST_F(IntrusiveForwardListTest,Remove)587 TEST_F(IntrusiveForwardListTest, Remove) {
588   Remove<IFLTestValueList>();
589   Remove<ConstIFLTestValueList>();
590   Remove<IFLTestValue2List>();
591 }
592 
593 template <typename ListType>
Unique()594 void IntrusiveForwardListTest::Unique() {
595   using ValueType = typename ListType::value_type;
596   std::forward_list<int> ref({ 3, 1, 1, 2, 3, 3, 7, 7, 4, 4, 5, 7 });
597   std::vector<ValueType> storage(ref.begin(), ref.end());
598   ListType ifl(storage.begin(), storage.end());
599   ASSERT_LISTS_EQUAL(ref, ifl);
600   ref.unique();
601   ifl.unique();
602   ASSERT_LISTS_EQUAL(ref, ifl);
603   std::forward_list<int> check({ 3, 1, 2, 3, 7, 4, 5, 7 });
604   ASSERT_LISTS_EQUAL(check, ifl);
605 
606   auto bin_pred = [](const ValueType& lhs, const ValueType& rhs) {
607     return (lhs.value & ~1) == (rhs.value & ~1);
608   };
609   ref.unique(bin_pred);
610   ifl.unique(bin_pred);
611   ASSERT_LISTS_EQUAL(ref, ifl);
612   check.assign({ 3, 1, 2, 7, 4, 7 });
613   ASSERT_LISTS_EQUAL(check, ifl);
614 }
615 
TEST_F(IntrusiveForwardListTest,Unique)616 TEST_F(IntrusiveForwardListTest, Unique) {
617   Unique<IFLTestValueList>();
618   Unique<ConstIFLTestValueList>();
619   Unique<IFLTestValue2List>();
620 }
621 
622 template <typename ListType>
Merge()623 void IntrusiveForwardListTest::Merge() {
624   using ValueType = typename ListType::value_type;
625   std::forward_list<int> ref1({ 1, 4, 8, 8, 12 });
626   std::vector<ValueType> storage1(ref1.begin(), ref1.end());
627   ListType ifl1(storage1.begin(), storage1.end());
628   std::forward_list<int> ref2({ 3, 5, 6, 7, 9 });
629   std::vector<ValueType> storage2(ref2.begin(), ref2.end());
630   ListType ifl2(storage2.begin(), storage2.end());
631   ASSERT_LISTS_EQUAL(ref1, ifl1);
632   ASSERT_LISTS_EQUAL(ref2, ifl2);
633   CHECK(std::is_sorted(ref1.begin(), ref1.end()));
634   CHECK(std::is_sorted(ref2.begin(), ref2.end()));
635   ref1.merge(ref2);
636   ifl1.merge(ifl2);
637   ASSERT_LISTS_EQUAL(ref1, ifl1);
638   ASSERT_LISTS_EQUAL(ref2, ifl2);
639   CHECK(ref2.empty());
640   std::forward_list<int> check({ 1, 3, 4, 5, 6, 7, 8, 8, 9, 12 });
641   ASSERT_LISTS_EQUAL(check, ifl1);
642 }
643 
TEST_F(IntrusiveForwardListTest,Merge)644 TEST_F(IntrusiveForwardListTest, Merge) {
645   Merge<IFLTestValueList>();
646   Merge<ConstIFLTestValueList>();
647   Merge<IFLTestValue2List>();
648 }
649 
650 template <typename ListType>
Sort1()651 void IntrusiveForwardListTest::Sort1() {
652   using ValueType = typename ListType::value_type;
653   std::forward_list<int> ref({ 2, 9, 8, 3, 7, 4, 1, 5, 3, 0 });
654   std::vector<ValueType> storage(ref.begin(), ref.end());
655   ListType ifl(storage.begin(), storage.end());
656   ASSERT_LISTS_EQUAL(ref, ifl);
657   CHECK(!std::is_sorted(ref.begin(), ref.end()));
658   ref.sort();
659   ifl.sort();
660   ASSERT_LISTS_EQUAL(ref, ifl);
661   std::forward_list<int> check({ 0, 1, 2, 3, 3, 4, 5, 7, 8, 9 });
662   ASSERT_LISTS_EQUAL(check, ifl);
663 }
664 
TEST_F(IntrusiveForwardListTest,Sort1)665 TEST_F(IntrusiveForwardListTest, Sort1) {
666   Sort1<IFLTestValueList>();
667   Sort1<ConstIFLTestValueList>();
668   Sort1<IFLTestValue2List>();
669 }
670 
671 template <typename ListType>
Sort2()672 void IntrusiveForwardListTest::Sort2() {
673   using ValueType = typename ListType::value_type;
674   std::forward_list<int> ref({ 2, 9, 8, 3, 7, 4, 1, 5, 3, 0 });
675   std::vector<ValueType> storage(ref.begin(), ref.end());
676   ListType ifl(storage.begin(), storage.end());
677   ASSERT_LISTS_EQUAL(ref, ifl);
678   auto cmp = [](const ValueType& lhs, const ValueType& rhs) {
679     return (lhs.value & ~1) < (rhs.value & ~1);
680   };
681   CHECK(!std::is_sorted(ref.begin(), ref.end(), cmp));
682   ref.sort(cmp);
683   ifl.sort(cmp);
684   ASSERT_LISTS_EQUAL(ref, ifl);
685   std::forward_list<int> check({ 1, 0, 2, 3, 3, 4, 5, 7, 9, 8 });
686   ASSERT_LISTS_EQUAL(check, ifl);
687 }
688 
TEST_F(IntrusiveForwardListTest,Sort2)689 TEST_F(IntrusiveForwardListTest, Sort2) {
690   Sort2<IFLTestValueList>();
691   Sort2<ConstIFLTestValueList>();
692   Sort2<IFLTestValue2List>();
693 }
694 
695 template <typename ListType>
Reverse()696 void IntrusiveForwardListTest::Reverse() {
697   using ValueType = typename ListType::value_type;
698   std::forward_list<int> ref({ 8, 3, 5, 4, 1, 3 });
699   std::vector<ValueType> storage(ref.begin(), ref.end());
700   ListType ifl(storage.begin(), storage.end());
701   ASSERT_LISTS_EQUAL(ref, ifl);
702   CHECK(!std::is_sorted(ref.begin(), ref.end()));
703   ref.reverse();
704   ifl.reverse();
705   ASSERT_LISTS_EQUAL(ref, ifl);
706   std::forward_list<int> check({ 3, 1, 4, 5, 3, 8 });
707   ASSERT_LISTS_EQUAL(check, ifl);
708 }
709 
TEST_F(IntrusiveForwardListTest,Reverse)710 TEST_F(IntrusiveForwardListTest, Reverse) {
711   Reverse<IFLTestValueList>();
712   Reverse<ConstIFLTestValueList>();
713   Reverse<IFLTestValue2List>();
714 }
715 
716 template <typename ListType>
ModifyValue()717 void IntrusiveForwardListTest::ModifyValue() {
718   using ValueType = typename ListType::value_type;
719   std::forward_list<int> ref({ 3, 7, 42 });
720   std::vector<ValueType> storage(ref.begin(), ref.end());
721   ListType ifl(storage.begin(), storage.end());
722   ASSERT_LISTS_EQUAL(ref, ifl);
723 
724   auto add1 = [](const ValueType& value) { return value.value + 1; };
725   std::transform(ref.begin(), ref.end(), ref.begin(), add1);
726   std::transform(ifl.begin(), ifl.end(), ifl.begin(), add1);
727   ASSERT_LISTS_EQUAL(ref, ifl);
728 }
729 
TEST_F(IntrusiveForwardListTest,ModifyValue)730 TEST_F(IntrusiveForwardListTest, ModifyValue) {
731   ModifyValue<IFLTestValueList>();
732   // Does not compile with ConstIFLTestValueList because LHS of the assignment is const.
733   // ModifyValue<ConstIFLTestValueList>();
734   static_assert(std::is_const_v<ConstIFLTestValueList::iterator::value_type>);
735   ModifyValue<IFLTestValue2List>();
736 }
737 
738 struct Tag1;
739 struct Tag2;
740 struct TwoListsValue : public IntrusiveForwardListNode<TwoListsValue, Tag1>,
741                        public IntrusiveForwardListNode<TwoListsValue, Tag2> {
742   // Deliberately not explicit.
TwoListsValueart::TwoListsValue743   TwoListsValue(int v) : value(v) { }  // NOLINT(runtime/explicit)
744 
745   int value;
746 };
747 using FirstList =
748     IntrusiveForwardList<TwoListsValue, IntrusiveForwardListBaseHookTraits<TwoListsValue, Tag1>>;
749 using SecondList =
750     IntrusiveForwardList<TwoListsValue, IntrusiveForwardListBaseHookTraits<TwoListsValue, Tag2>>;
751 
operator ==(const TwoListsValue & lhs,const TwoListsValue & rhs)752 bool operator==(const TwoListsValue& lhs, const TwoListsValue& rhs) {
753   return lhs.value == rhs.value;
754 }
755 
TEST_F(IntrusiveForwardListTest,TwoLists)756 TEST_F(IntrusiveForwardListTest, TwoLists) {
757   // Test that a value can be in two lists at the same time and the hooks do not interfere.
758   std::vector<TwoListsValue> storage({ 0, 1, 2, 3, 4, 5, 6, 7, 8, 9 });  // storage[i] = i
759 
760   std::vector<int> order1({ 3, 1, 7, 2, 8, 9, 4, 0, 6, 5 });
761   FirstList list1;
762   auto pos1 = list1.before_begin();
763   for (size_t idx : order1) {
764     pos1 = list1.insert_after(pos1, storage[idx]);
765   }
766 
767   std::vector<int> order2({ 8, 5, 1, 6, 7, 2, 9, 3, 0, 4 });
768   SecondList list2;
769   auto pos2 = list2.before_begin();
770   for (size_t idx : order2) {
771     pos2 = list2.insert_after(pos2, storage[idx]);
772   }
773 
774   // Using `storage[i] = i`, we can easily compare that nodes of each list are in the right order.
775   ASSERT_LISTS_EQUAL(order1, list1);
776   ASSERT_LISTS_EQUAL(order2, list2);
777 }
778 
779 }  // namespace art
780