1b0386a51SErik Pilkington //===----------------------------------------------------------------------===//
2b0386a51SErik Pilkington //
357b08b09SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
457b08b09SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
557b08b09SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6b0386a51SErik Pilkington //
7b0386a51SErik Pilkington //===----------------------------------------------------------------------===//
8b0386a51SErik Pilkington 
931cbe0f2SLouis Dionne // UNSUPPORTED: c++03, c++11, c++14
10b0386a51SErik Pilkington 
11b0386a51SErik Pilkington #include <unordered_set>
12b0386a51SErik Pilkington #include <unordered_map>
13b0386a51SErik Pilkington #include <set>
14b0386a51SErik Pilkington #include <map>
157fc6a556SMarshall Clow #include "test_macros.h"
16b0386a51SErik Pilkington #include "min_allocator.h"
17b0386a51SErik Pilkington 
18b0386a51SErik Pilkington // [container.node.overview] Table 83.
19b0386a51SErik Pilkington template <class K, class T, class C1, class C2, class H1, class H2, class E1, class E2, class A_set, class A_map>
20b0386a51SErik Pilkington struct node_compatibility_table
21b0386a51SErik Pilkington {
22b0386a51SErik Pilkington     static constexpr bool value =
23*9d7ae0acSLouis Dionne         std::is_same_v<typename std::map<K, T, C1, A_map>::node_type,               typename std::map<K, T, C2, A_map>::node_type> &&
24*9d7ae0acSLouis Dionne         std::is_same_v<typename std::map<K, T, C1, A_map>::node_type,               typename std::multimap<K, T, C2, A_map>::node_type> &&
25*9d7ae0acSLouis Dionne         std::is_same_v<typename std::set<K, C1, A_set>::node_type,                  typename std::set<K, C2, A_set>::node_type> &&
26*9d7ae0acSLouis Dionne         std::is_same_v<typename std::set<K, C1, A_set>::node_type,                  typename std::multiset<K, C2, A_set>::node_type> &&
27*9d7ae0acSLouis Dionne         std::is_same_v<typename std::unordered_map<K, T, H1, E1, A_map>::node_type, typename std::unordered_map<K, T, H2, E2, A_map>::node_type> &&
28*9d7ae0acSLouis Dionne         std::is_same_v<typename std::unordered_map<K, T, H1, E1, A_map>::node_type, typename std::unordered_multimap<K, T, H2, E2, A_map>::node_type> &&
29*9d7ae0acSLouis Dionne         std::is_same_v<typename std::unordered_set<K, H1, E1, A_set>::node_type,    typename std::unordered_set<K, H2, E2, A_set>::node_type> &&
30*9d7ae0acSLouis Dionne         std::is_same_v<typename std::unordered_set<K, H1, E1, A_set>::node_type,    typename std::unordered_multiset<K, H2, E2, A_set>::node_type>;
31b0386a51SErik Pilkington };
32b0386a51SErik Pilkington 
33b0386a51SErik Pilkington template <class T> struct my_hash
34b0386a51SErik Pilkington {
35b0386a51SErik Pilkington     using argument_type = T;
36*9d7ae0acSLouis Dionne     using result_type = std::size_t;
37b0386a51SErik Pilkington     my_hash() = default;
operator ()my_hash38*9d7ae0acSLouis Dionne     std::size_t operator()(const T&) const {return 0;}
39b0386a51SErik Pilkington };
40b0386a51SErik Pilkington 
41b0386a51SErik Pilkington template <class T> struct my_compare
42b0386a51SErik Pilkington {
43b0386a51SErik Pilkington     my_compare() = default;
operator ()my_compare44b0386a51SErik Pilkington     bool operator()(const T&, const T&) const {return true;}
45b0386a51SErik Pilkington };
46b0386a51SErik Pilkington 
47b0386a51SErik Pilkington template <class T> struct my_equal
48b0386a51SErik Pilkington {
49b0386a51SErik Pilkington     my_equal() = default;
operator ()my_equal50b0386a51SErik Pilkington     bool operator()(const T&, const T&) const {return true;}
51b0386a51SErik Pilkington };
52b0386a51SErik Pilkington 
53b0386a51SErik Pilkington struct Static
54b0386a51SErik Pilkington {
55b0386a51SErik Pilkington     Static() = default;
56b0386a51SErik Pilkington     Static(const Static&) = delete;
57b0386a51SErik Pilkington     Static(Static&&) = delete;
58b0386a51SErik Pilkington     Static& operator=(const Static&) = delete;
59b0386a51SErik Pilkington     Static& operator=(Static&&) = delete;
60b0386a51SErik Pilkington };
61b0386a51SErik Pilkington 
62b0386a51SErik Pilkington namespace std
63b0386a51SErik Pilkington {
64b0386a51SErik Pilkington template <> struct hash<Static>
65b0386a51SErik Pilkington {
66b0386a51SErik Pilkington     using argument_type = Static;
67*9d7ae0acSLouis Dionne     using result_type = std::size_t;
68b0386a51SErik Pilkington     hash() = default;
69*9d7ae0acSLouis Dionne     std::size_t operator()(const Static&) const;
70b0386a51SErik Pilkington };
71b0386a51SErik Pilkington }
72b0386a51SErik Pilkington 
73b0386a51SErik Pilkington static_assert(node_compatibility_table<
74b0386a51SErik Pilkington                   int, int, std::less<int>, std::less<int>, std::hash<int>,
75b0386a51SErik Pilkington                   std::hash<int>, std::equal_to<int>, std::equal_to<int>,
76b0386a51SErik Pilkington                   std::allocator<int>,
77b0386a51SErik Pilkington                   std::allocator<std::pair<const int, int>>>::value,
78b0386a51SErik Pilkington               "");
79b0386a51SErik Pilkington 
80b0386a51SErik Pilkington static_assert(
81b0386a51SErik Pilkington     node_compatibility_table<int, int, std::less<int>, my_compare<int>,
82b0386a51SErik Pilkington                              std::hash<int>, my_hash<int>, std::equal_to<int>,
83*9d7ae0acSLouis Dionne                              my_equal<int>, std::allocator<int>,
84*9d7ae0acSLouis Dionne                              std::allocator<std::pair<const int, int>>>::value,
85b0386a51SErik Pilkington     "");
86b0386a51SErik Pilkington 
87b0386a51SErik Pilkington static_assert(node_compatibility_table<
88b0386a51SErik Pilkington                   Static, int, my_compare<Static>, std::less<Static>,
89b0386a51SErik Pilkington                   my_hash<Static>, std::hash<Static>, my_equal<Static>,
90b0386a51SErik Pilkington                   std::equal_to<Static>, min_allocator<Static>,
91b0386a51SErik Pilkington                   min_allocator<std::pair<const Static, int>>>::value,
92b0386a51SErik Pilkington               "");
93b0386a51SErik Pilkington 
94b0386a51SErik Pilkington template <class Container>
test_node_handle_operations()95b0386a51SErik Pilkington void test_node_handle_operations()
96b0386a51SErik Pilkington {
97b0386a51SErik Pilkington     Container c;
98b0386a51SErik Pilkington 
99b0386a51SErik Pilkington     typename Container::node_type nt1, nt2 = c.extract(c.emplace().first);
100b0386a51SErik Pilkington     assert(nt2.get_allocator() == c.get_allocator());
101b0386a51SErik Pilkington     assert(!nt2.empty());
102b0386a51SErik Pilkington     assert(nt1.empty());
103b0386a51SErik Pilkington     std::swap(nt1, nt2);
104b0386a51SErik Pilkington     assert(nt1.get_allocator() == c.get_allocator());
105b0386a51SErik Pilkington     assert(nt2.empty());
106b0386a51SErik Pilkington }
107b0386a51SErik Pilkington 
108b0386a51SErik Pilkington template <class Container>
test_node_handle_operations_multi()109b0386a51SErik Pilkington void test_node_handle_operations_multi()
110b0386a51SErik Pilkington {
111b0386a51SErik Pilkington     Container c;
112b0386a51SErik Pilkington 
113b0386a51SErik Pilkington     typename Container::node_type nt1, nt2 = c.extract(c.emplace());
114b0386a51SErik Pilkington     assert(nt2.get_allocator() == c.get_allocator());
115b0386a51SErik Pilkington     assert(!nt2.empty());
116b0386a51SErik Pilkington     assert(nt1.empty());
117b0386a51SErik Pilkington     std::swap(nt1, nt2);
118b0386a51SErik Pilkington     assert(nt1.get_allocator() == c.get_allocator());
119b0386a51SErik Pilkington     assert(nt2.empty());
120b0386a51SErik Pilkington }
121b0386a51SErik Pilkington 
test_typedef()12277e898cbSBilly Robert O'Neal III template <class> void test_typedef() {}
12377e898cbSBilly Robert O'Neal III 
124b0386a51SErik Pilkington template <class Container>
test_insert_return_type()125b0386a51SErik Pilkington void test_insert_return_type()
126b0386a51SErik Pilkington {
12777e898cbSBilly Robert O'Neal III     test_typedef<typename Container::insert_return_type>();
128b0386a51SErik Pilkington }
129b0386a51SErik Pilkington 
main(int,char **)1302df59c50SJF Bastien int main(int, char**)
131b0386a51SErik Pilkington {
132b0386a51SErik Pilkington     test_node_handle_operations<std::map<int, int>>();
133b0386a51SErik Pilkington     test_node_handle_operations_multi<std::multimap<int, int>>();
134b0386a51SErik Pilkington     test_node_handle_operations<std::set<int>>();
135b0386a51SErik Pilkington     test_node_handle_operations_multi<std::multiset<int>>();
136b0386a51SErik Pilkington     test_node_handle_operations<std::unordered_map<int, int>>();
137b0386a51SErik Pilkington     test_node_handle_operations_multi<std::unordered_multimap<int, int>>();
138b0386a51SErik Pilkington     test_node_handle_operations<std::unordered_set<int>>();
139b0386a51SErik Pilkington     test_node_handle_operations_multi<std::unordered_multiset<int>>();
140b0386a51SErik Pilkington 
141b0386a51SErik Pilkington     test_insert_return_type<std::map<int, int>>();
142b0386a51SErik Pilkington     test_insert_return_type<std::set<int>>();
143b0386a51SErik Pilkington     test_insert_return_type<std::unordered_map<int, int>>();
144b0386a51SErik Pilkington     test_insert_return_type<std::unordered_set<int>>();
1452df59c50SJF Bastien 
1462df59c50SJF Bastien   return 0;
147b0386a51SErik Pilkington }
148