2 // Copyright 2005-2009 Daniel James.
3 // Distributed under the Boost Software License, Version 1.0. (See accompanying
4 // file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
6 #if defined(BOOST_MSVC)
8 #pragma warning(disable:4100) // unreferenced formal parameter
9 #pragma warning(disable:4610) // class can never be instantiated
10 #pragma warning(disable:4510) // default constructor could not be generated
13 #include <boost/concept_check.hpp>
15 #if defined(BOOST_MSVC)
19 #include <boost/static_assert.hpp>
20 #include <boost/type_traits/is_same.hpp>
21 #include <boost/type_traits/is_convertible.hpp>
22 #include <boost/iterator/iterator_traits.hpp>
23 #include <boost/limits.hpp>
24 #include <boost/utility/swap.hpp>
25 #include "../helpers/check_return_type.hpp"
27 typedef long double comparison_type;
29 template <class T> void sink(T const&) {}
30 template <class T> T rvalue(T const& v) { return v; }
31 template <class T> T rvalue_default() { return T(); }
33 template <class X, class T>
34 void container_test(X& r, T const&)
36 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
37 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
38 typedef BOOST_DEDUCED_TYPENAME X::difference_type difference_type;
39 typedef BOOST_DEDUCED_TYPENAME X::size_type size_type;
41 typedef BOOST_DEDUCED_TYPENAME
42 boost::iterator_value<iterator>::type iterator_value_type;
43 typedef BOOST_DEDUCED_TYPENAME
44 boost::iterator_value<const_iterator>::type const_iterator_value_type;
45 typedef BOOST_DEDUCED_TYPENAME
46 boost::iterator_difference<iterator>::type iterator_difference_type;
47 typedef BOOST_DEDUCED_TYPENAME
48 boost::iterator_difference<const_iterator>::type
49 const_iterator_difference_type;
51 typedef BOOST_DEDUCED_TYPENAME X::value_type value_type;
52 typedef BOOST_DEDUCED_TYPENAME X::reference reference;
53 typedef BOOST_DEDUCED_TYPENAME X::const_reference const_reference;
57 BOOST_STATIC_ASSERT((boost::is_same<T, value_type>::value));
58 boost::function_requires<boost::CopyConstructibleConcept<X> >();
60 // reference_type / const_reference_type
62 BOOST_STATIC_ASSERT((boost::is_same<T&, reference>::value));
63 BOOST_STATIC_ASSERT((boost::is_same<T const&, const_reference>::value));
67 boost::function_requires<boost::InputIteratorConcept<iterator> >();
68 BOOST_STATIC_ASSERT((boost::is_same<T, iterator_value_type>::value));
69 BOOST_STATIC_ASSERT((boost::is_convertible<iterator, const_iterator>::value));
73 boost::function_requires<boost::InputIteratorConcept<const_iterator> >();
74 BOOST_STATIC_ASSERT((boost::is_same<T, const_iterator_value_type>::value));
78 BOOST_STATIC_ASSERT(std::numeric_limits<difference_type>::is_signed);
79 BOOST_STATIC_ASSERT(std::numeric_limits<difference_type>::is_integer);
80 BOOST_STATIC_ASSERT((boost::is_same<difference_type,
81 iterator_difference_type>::value));
82 BOOST_STATIC_ASSERT((boost::is_same<difference_type,
83 const_iterator_difference_type>::value));
87 BOOST_STATIC_ASSERT(!std::numeric_limits<size_type>::is_signed);
88 BOOST_STATIC_ASSERT(std::numeric_limits<size_type>::is_integer);
90 // size_type can represent any non-negative value type of difference_type
91 // I'm not sure about either of these tests...
92 size_type max_diff = static_cast<size_type>(
93 (std::numeric_limits<difference_type>::max)());
94 difference_type converted_diff(static_cast<difference_type>(max_diff));
95 BOOST_TEST((std::numeric_limits<difference_type>::max)()
99 static_cast<comparison_type>(
100 (std::numeric_limits<size_type>::max)()) >
101 static_cast<comparison_type>(
102 (std::numeric_limits<difference_type>::max)()));
106 // I don't test the runtime post-conditions here.
108 #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
109 // It isn't specified in the container requirements that the no argument
110 // constructor is implicit, but it is defined that way in the concrete
111 // container specification.
117 BOOST_TEST(u.size() == 0);
118 BOOST_TEST(X().size() == 0);
126 X u4(rvalue(a_const));
127 X u5 = rvalue(a_const);
131 test::check_return_type<X>::equals_ref(r = a);
135 typedef BOOST_DEDUCED_TYPENAME X::allocator_type allocator_type;
136 test::check_return_type<allocator_type>::equals(a_const.get_allocator());
138 allocator_type m = a.get_allocator();
143 sink(X(rvalue(a_const), m));
144 X c3(rvalue(a_const), m);
146 // Avoid unused variable warnings:
159 void unordered_destructible_test(X&)
161 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
162 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
163 typedef BOOST_DEDUCED_TYPENAME X::size_type size_type;
167 #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
168 X x2(rvalue_default<X>());
169 X x3 = rvalue_default<X>();
170 // This can only be done if propagate_on_container_move_assignment::value
172 // x2 = rvalue_default<X>();
178 ::operator delete((void*)(&a1));
182 test::check_return_type<iterator>::equals(a.begin());
183 test::check_return_type<const_iterator>::equals(a_const.begin());
184 test::check_return_type<const_iterator>::equals(a.cbegin());
185 test::check_return_type<const_iterator>::equals(a_const.cbegin());
186 test::check_return_type<iterator>::equals(a.end());
187 test::check_return_type<const_iterator>::equals(a_const.end());
188 test::check_return_type<const_iterator>::equals(a.cend());
189 test::check_return_type<const_iterator>::equals(a_const.cend());
194 test::check_return_type<size_type>::equals(a.size());
195 test::check_return_type<size_type>::equals(a.max_size());
196 test::check_return_type<bool>::convertible(a.empty());
200 typedef BOOST_DEDUCED_TYPENAME X::allocator_type allocator_type;
201 test::check_return_type<allocator_type>::equals(a_const.get_allocator());
204 template <class X, class Key>
205 void unordered_set_test(X&, Key const&)
207 typedef BOOST_DEDUCED_TYPENAME X::value_type value_type;
208 typedef BOOST_DEDUCED_TYPENAME X::key_type key_type;
210 BOOST_STATIC_ASSERT((boost::is_same<value_type, key_type>::value));
212 // iterator pointer / const_pointer_type
214 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
215 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
216 typedef BOOST_DEDUCED_TYPENAME X::local_iterator local_iterator;
217 typedef BOOST_DEDUCED_TYPENAME X::const_local_iterator const_local_iterator;
218 typedef BOOST_DEDUCED_TYPENAME
219 boost::iterator_pointer<iterator>::type iterator_pointer;
220 typedef BOOST_DEDUCED_TYPENAME
221 boost::iterator_pointer<const_iterator>::type
222 const_iterator_pointer;
223 typedef BOOST_DEDUCED_TYPENAME
224 boost::iterator_pointer<local_iterator>::type local_iterator_pointer;
225 typedef BOOST_DEDUCED_TYPENAME
226 boost::iterator_pointer<const_local_iterator>::type
227 const_local_iterator_pointer;
229 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, iterator_pointer>::value));
230 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, const_iterator_pointer>::value));
231 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, local_iterator_pointer>::value));
232 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, const_local_iterator_pointer>::value));
235 template <class X, class Key, class T>
236 void unordered_map_test(X& r, Key const& k, T const& v)
238 typedef BOOST_DEDUCED_TYPENAME X::value_type value_type;
239 typedef BOOST_DEDUCED_TYPENAME X::key_type key_type;
241 BOOST_STATIC_ASSERT((
242 boost::is_same<value_type, std::pair<key_type const, T> >::value));
244 // iterator pointer / const_pointer_type
246 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
247 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
248 typedef BOOST_DEDUCED_TYPENAME X::local_iterator local_iterator;
249 typedef BOOST_DEDUCED_TYPENAME X::const_local_iterator const_local_iterator;
250 typedef BOOST_DEDUCED_TYPENAME
251 boost::iterator_pointer<iterator>::type iterator_pointer;
252 typedef BOOST_DEDUCED_TYPENAME
253 boost::iterator_pointer<const_iterator>::type
254 const_iterator_pointer;
255 typedef BOOST_DEDUCED_TYPENAME
256 boost::iterator_pointer<local_iterator>::type local_iterator_pointer;
257 typedef BOOST_DEDUCED_TYPENAME
258 boost::iterator_pointer<const_local_iterator>::type
259 const_local_iterator_pointer;
261 BOOST_STATIC_ASSERT((boost::is_same<value_type*, iterator_pointer>::value));
262 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, const_iterator_pointer>::value));
263 BOOST_STATIC_ASSERT((boost::is_same<value_type*, local_iterator_pointer>::value));
264 BOOST_STATIC_ASSERT((boost::is_same<value_type const*, const_local_iterator_pointer>::value));
268 r.insert(std::pair<Key const, T>(k, v));
274 r.emplace(k_lvalue, v_lvalue);
275 r.emplace(rvalue(k), rvalue(v));
277 r.emplace(boost::unordered::piecewise_construct,
278 boost::make_tuple(k), boost::make_tuple(v));
282 void equality_test(X& r)
284 X const a = r, b = r;
286 test::check_return_type<bool>::equals(a == b);
287 test::check_return_type<bool>::equals(a != b);
288 test::check_return_type<bool>::equals(boost::operator==(a, b));
289 test::check_return_type<bool>::equals(boost::operator!=(a, b));
292 template <class X, class T>
293 void unordered_unique_test(X& r, T const& t)
295 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
296 test::check_return_type<std::pair<iterator, bool> >::equals(r.insert(t));
297 test::check_return_type<std::pair<iterator, bool> >::equals(r.emplace(t));
300 template <class X, class T>
301 void unordered_equivalent_test(X& r, T const& t)
303 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
304 test::check_return_type<iterator>::equals(r.insert(t));
305 test::check_return_type<iterator>::equals(r.emplace(t));
308 template <class X, class Key, class T>
309 void unordered_map_functions(X&, Key const& k, T const&)
311 typedef BOOST_DEDUCED_TYPENAME X::mapped_type mapped_type;
314 test::check_return_type<mapped_type>::equals_ref(a[k]);
315 test::check_return_type<mapped_type>::equals_ref(a.at(k));
318 test::check_return_type<mapped_type const>::equals_ref(b.at(k));
321 template <class X, class Key, class Hash, class Pred>
322 void unordered_test(X& x, Key& k, Hash& hf, Pred& eq)
324 unordered_destructible_test(x);
326 typedef BOOST_DEDUCED_TYPENAME X::key_type key_type;
327 typedef BOOST_DEDUCED_TYPENAME X::hasher hasher;
328 typedef BOOST_DEDUCED_TYPENAME X::key_equal key_equal;
329 typedef BOOST_DEDUCED_TYPENAME X::size_type size_type;
331 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
332 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
333 typedef BOOST_DEDUCED_TYPENAME X::local_iterator local_iterator;
334 typedef BOOST_DEDUCED_TYPENAME X::const_local_iterator const_local_iterator;
336 typedef BOOST_DEDUCED_TYPENAME
337 boost::BOOST_ITERATOR_CATEGORY<iterator>::type
339 typedef BOOST_DEDUCED_TYPENAME
340 boost::iterator_difference<iterator>::type
342 typedef BOOST_DEDUCED_TYPENAME
343 boost::iterator_pointer<iterator>::type
345 typedef BOOST_DEDUCED_TYPENAME
346 boost::iterator_reference<iterator>::type
349 typedef BOOST_DEDUCED_TYPENAME
350 boost::BOOST_ITERATOR_CATEGORY<local_iterator>::type
351 local_iterator_category;
352 typedef BOOST_DEDUCED_TYPENAME
353 boost::iterator_difference<local_iterator>::type
354 local_iterator_difference;
355 typedef BOOST_DEDUCED_TYPENAME
356 boost::iterator_pointer<local_iterator>::type
357 local_iterator_pointer;
358 typedef BOOST_DEDUCED_TYPENAME
359 boost::iterator_reference<local_iterator>::type
360 local_iterator_reference;
362 typedef BOOST_DEDUCED_TYPENAME
363 boost::BOOST_ITERATOR_CATEGORY<const_iterator>::type
364 const_iterator_category;
365 typedef BOOST_DEDUCED_TYPENAME
366 boost::iterator_difference<const_iterator>::type
367 const_iterator_difference;
368 typedef BOOST_DEDUCED_TYPENAME
369 boost::iterator_pointer<const_iterator>::type
370 const_iterator_pointer;
371 typedef BOOST_DEDUCED_TYPENAME
372 boost::iterator_reference<const_iterator>::type
373 const_iterator_reference;
375 typedef BOOST_DEDUCED_TYPENAME
376 boost::BOOST_ITERATOR_CATEGORY<const_local_iterator>::type
377 const_local_iterator_category;
378 typedef BOOST_DEDUCED_TYPENAME
379 boost::iterator_difference<const_local_iterator>::type
380 const_local_iterator_difference;
381 typedef BOOST_DEDUCED_TYPENAME
382 boost::iterator_pointer<const_local_iterator>::type
383 const_local_iterator_pointer;
384 typedef BOOST_DEDUCED_TYPENAME
385 boost::iterator_reference<const_local_iterator>::type
386 const_local_iterator_reference;
387 typedef BOOST_DEDUCED_TYPENAME X::allocator_type allocator_type;
389 BOOST_STATIC_ASSERT((boost::is_same<Key, key_type>::value));
390 //boost::function_requires<boost::CopyConstructibleConcept<key_type> >();
391 //boost::function_requires<boost::AssignableConcept<key_type> >();
393 BOOST_STATIC_ASSERT((boost::is_same<Hash, hasher>::value));
394 test::check_return_type<std::size_t>::equals(hf(k));
396 BOOST_STATIC_ASSERT((boost::is_same<Pred, key_equal>::value));
397 test::check_return_type<bool>::convertible(eq(k, k));
399 boost::function_requires<boost::InputIteratorConcept<local_iterator> >();
400 BOOST_STATIC_ASSERT((boost::is_same<local_iterator_category,
401 iterator_category>::value));
402 BOOST_STATIC_ASSERT((boost::is_same<local_iterator_difference,
403 iterator_difference>::value));
404 BOOST_STATIC_ASSERT((boost::is_same<local_iterator_pointer,
405 iterator_pointer>::value));
406 BOOST_STATIC_ASSERT((boost::is_same<local_iterator_reference,
407 iterator_reference>::value));
409 boost::function_requires<
410 boost::InputIteratorConcept<const_local_iterator> >();
411 BOOST_STATIC_ASSERT((boost::is_same<const_local_iterator_category,
412 const_iterator_category>::value));
413 BOOST_STATIC_ASSERT((boost::is_same<const_local_iterator_difference,
414 const_iterator_difference>::value));
415 BOOST_STATIC_ASSERT((boost::is_same<const_local_iterator_pointer,
416 const_iterator_pointer>::value));
417 BOOST_STATIC_ASSERT((boost::is_same<const_local_iterator_reference,
418 const_iterator_reference>::value));
421 allocator_type m = a.get_allocator();
435 X a1a(10, hf, eq, m);
443 test::check_return_type<size_type>::equals(a.erase(k));
445 const_iterator q1 = a.cbegin(), q2 = a.cend();
446 test::check_return_type<iterator>::equals(a.erase(q1, q2));
452 test::check_return_type<hasher>::equals(b.hash_function());
453 test::check_return_type<key_equal>::equals(b.key_eq());
455 test::check_return_type<iterator>::equals(a.find(k));
456 test::check_return_type<const_iterator>::equals(b.find(k));
457 test::check_return_type<size_type>::equals(b.count(k));
458 test::check_return_type<std::pair<iterator, iterator> >::equals(
460 test::check_return_type<std::pair<const_iterator, const_iterator> >::equals(
462 test::check_return_type<size_type>::equals(b.bucket_count());
463 test::check_return_type<size_type>::equals(b.max_bucket_count());
464 test::check_return_type<size_type>::equals(b.bucket(k));
465 test::check_return_type<size_type>::equals(b.bucket_size(0));
467 test::check_return_type<local_iterator>::equals(a.begin(0));
468 test::check_return_type<const_local_iterator>::equals(b.begin(0));
469 test::check_return_type<local_iterator>::equals(a.end(0));
470 test::check_return_type<const_local_iterator>::equals(b.end(0));
472 test::check_return_type<const_local_iterator>::equals(a.cbegin(0));
473 test::check_return_type<const_local_iterator>::equals(b.cbegin(0));
474 test::check_return_type<const_local_iterator>::equals(a.cend(0));
475 test::check_return_type<const_local_iterator>::equals(b.cend(0));
477 test::check_return_type<float>::equals(b.load_factor());
478 test::check_return_type<float>::equals(b.max_load_factor());
479 a.max_load_factor((float) 2.0);
482 // Avoid unused variable warnings:
495 template <class X, class Key, class T, class Hash, class Pred>
496 void unordered_copyable_test(X& x, Key& k, T& t, Hash& hf, Pred& eq)
498 unordered_test(x, k, hf, eq);
500 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
501 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
502 typedef BOOST_DEDUCED_TYPENAME X::allocator_type allocator_type;
505 allocator_type m = a.get_allocator();
507 BOOST_DEDUCED_TYPENAME X::value_type* i = 0;
508 BOOST_DEDUCED_TYPENAME X::value_type* j = 0;
513 X a5(i, j, 10, hf, eq);
521 X(i, j, 10, hf, eq, m);
522 X a5a(i, j, 10, hf, eq, m);
524 X a6a(i, j, 10, hf, m);
528 // Not specified for some reason (maybe ambiguity with another constructor?)
533 #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
534 std::size_t min_buckets = 10;
537 X({t}, min_buckets, hf);
538 X({t}, min_buckets, hf, eq);
540 X({t}, min_buckets, m);
541 X({t}, min_buckets, hf, m);
542 X({t}, min_buckets, hf, eq, m);
553 const_iterator q = a.cbegin();
555 test::check_return_type<iterator>::equals(a.insert(q, t));
556 test::check_return_type<iterator>::equals(a.emplace_hint(q, t));
559 #if !defined(BOOST_NO_CXX11_HDR_INITIALIZER_LIST)
560 std::initializer_list<T> list = {t};
564 #if !BOOST_WORKAROUND(BOOST_MSVC, < 1900)
574 test::check_return_type<iterator>::equals(a10.erase(q));
576 // Avoid unused variable warnings:
590 template <class X, class Key, class T, class Hash, class Pred>
591 void unordered_movable_test(X& x, Key& k, T& /* t */, Hash& hf, Pred& eq)
593 unordered_test(x, k, hf, eq);
595 typedef BOOST_DEDUCED_TYPENAME X::iterator iterator;
596 typedef BOOST_DEDUCED_TYPENAME X::const_iterator const_iterator;
597 typedef BOOST_DEDUCED_TYPENAME X::allocator_type allocator_type;
599 #if !defined(BOOST_NO_CXX11_RVALUE_REFERENCES)
600 X x1(rvalue_default<X>());
601 X x2(boost::move(x1));
602 x1 = rvalue_default<X>();
603 x2 = boost::move(x1);
607 allocator_type m = a.get_allocator();
609 test::minimal::constructor_param* i = 0;
610 test::minimal::constructor_param* j = 0;
615 X a5(i, j, 10, hf, eq);
623 X(i, j, 10, hf, eq, m);
624 X a5a(i, j, 10, hf, eq, m);
626 X a6a(i, j, 10, hf, m);
630 // Not specified for some reason (maybe ambiguity with another constructor?)
635 const_iterator q = a.cbegin();
637 test::minimal::constructor_param v;
639 test::check_return_type<iterator>::equals(a.emplace_hint(q, v));
642 a.emplace(boost::move(v1));
644 a.insert(boost::move(v2));
646 test::check_return_type<iterator>::equals(
647 a.emplace_hint(q, boost::move(v3)));
649 test::check_return_type<iterator>::equals(
650 a.insert(q, boost::move(v4)));
656 a10.insert(boost::move(v5));
658 test::check_return_type<iterator>::equals(a10.erase(q));
660 // Avoid unused variable warnings:
673 template <class X, class T>
674 void unordered_set_member_test(X& x, T& t)
678 x1.begin()->dummy_member();
679 x1.cbegin()->dummy_member();
682 template <class X, class T>
683 void unordered_map_member_test(X& x, T& t)
687 x1.begin()->first.dummy_member();
688 x1.cbegin()->first.dummy_member();
689 x1.begin()->second.dummy_member();
690 x1.cbegin()->second.dummy_member();