////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2004-2013. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // // See http://www.boost.org/libs/container for documentation. // ////////////////////////////////////////////////////////////////////////////// #include #include #include #include #include "print_container.hpp" #include "dummy_test_allocator.hpp" #include "movable_int.hpp" #include "flat_map_test.hpp" #include "propagate_allocator_test.hpp" #include "container_common_tests.hpp" #include "emplace_test.hpp" #include "../../intrusive/test/iterator_test.hpp" #include "flat_map_test.hpp" #include #include using namespace boost::container; class recursive_flat_map { public: recursive_flat_map(const recursive_flat_map &c) : map_(c.map_) {} recursive_flat_map & operator =(const recursive_flat_map &c) { map_= c.map_; return *this; } flat_map map_; flat_map::iterator it_; flat_map::const_iterator cit_; flat_map::reverse_iterator rit_; flat_map::const_reverse_iterator crit_; friend bool operator< (const recursive_flat_map &a, const recursive_flat_map &b) { return a.map_ < b.map_; } }; class recursive_flat_multimap { public: recursive_flat_multimap(const recursive_flat_multimap &c) : map_(c.map_) {} recursive_flat_multimap & operator =(const recursive_flat_multimap &c) { map_= c.map_; return *this; } flat_multimap map_; flat_multimap::iterator it_; flat_multimap::const_iterator cit_; flat_multimap::reverse_iterator rit_; flat_multimap::const_reverse_iterator crit_; friend bool operator< (const recursive_flat_multimap &a, const recursive_flat_multimap &b) { return a.map_ < b.map_; } }; template void test_move() { //Now test move semantics C original; C move_ctor(boost::move(original)); C move_assign; move_assign = boost::move(move_ctor); move_assign.swap(original); } namespace boost{ namespace container { namespace test{ bool constructor_template_auto_deduction_test() { #ifndef BOOST_CONTAINER_NO_CXX17_CTAD using namespace boost::container; const std::size_t NumElements = 100; { std::map int_map; for(std::size_t i = 0; i != NumElements; ++i){ int_map.insert(std::map::value_type(static_cast(i), static_cast(i))); } std::multimap int_mmap; for (std::size_t i = 0; i != NumElements; ++i) { int_mmap.insert(std::multimap::value_type(static_cast(i), static_cast(i))); } typedef std::less comp_int_t; typedef std::allocator > alloc_pair_int_t; //range { auto fmap = flat_map(int_map.begin(), int_map.end()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(int_mmap.begin(), int_mmap.end()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+comp { auto fmap = flat_map(int_map.begin(), int_map.end(), comp_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(int_mmap.begin(), int_mmap.end(), comp_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+comp+alloc { auto fmap = flat_map(int_map.begin(), int_map.end(), comp_int_t(), alloc_pair_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(int_mmap.begin(), int_mmap.end(), comp_int_t(), alloc_pair_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+alloc { auto fmap = flat_map(int_map.begin(), int_map.end(), alloc_pair_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(int_mmap.begin(), int_mmap.end(), alloc_pair_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //ordered_unique_range / ordered_range //range { auto fmap = flat_map(ordered_unique_range, int_map.begin(), int_map.end()); if(!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(ordered_range, int_mmap.begin(), int_mmap.end()); if(!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+comp { auto fmap = flat_map(ordered_unique_range, int_map.begin(), int_map.end(), comp_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(ordered_range, int_mmap.begin(), int_mmap.end(), comp_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+comp+alloc { auto fmap = flat_map(ordered_unique_range, int_map.begin(), int_map.end(), comp_int_t(), alloc_pair_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(ordered_range, int_mmap.begin(), int_mmap.end(), comp_int_t(), alloc_pair_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } //range+alloc { auto fmap = flat_map(ordered_unique_range, int_map.begin(), int_map.end(),alloc_pair_int_t()); if (!CheckEqualContainers(int_map, fmap)) return false; auto fmmap = flat_multimap(ordered_range, int_mmap.begin(), int_mmap.end(),alloc_pair_int_t()); if (!CheckEqualContainers(int_mmap, fmmap)) return false; } } #endif return true; } }}} template struct GetMapContainer { template struct apply { typedef std::pair type_t; typedef flat_map< ValueType , ValueType , std::less , typename boost::container::dtl::container_or_allocator_rebind::type > map_type; typedef flat_multimap< ValueType , ValueType , std::less , typename boost::container::dtl::container_or_allocator_rebind::type > multimap_type; }; }; //To test default parameters template<> struct GetMapContainer { template struct apply { typedef std::pair type_t; typedef flat_map< ValueType , ValueType > map_type; typedef flat_multimap< ValueType , ValueType > multimap_type; }; }; struct boost_container_flat_map; struct boost_container_flat_multimap; namespace boost { namespace container { namespace test { template<> struct alloc_propagate_base { template struct apply { typedef typename boost::container::allocator_traits:: template portable_rebind_alloc >::type TypeAllocator; typedef boost::container::flat_map, TypeAllocator> type; }; }; template<> struct alloc_propagate_base { template struct apply { typedef typename boost::container::allocator_traits:: template portable_rebind_alloc >::type TypeAllocator; typedef boost::container::flat_multimap, TypeAllocator> type; }; }; template struct get_real_stored_allocator > { typedef typename flat_map::impl_stored_allocator_type type; }; template struct get_real_stored_allocator > { typedef typename flat_multimap::impl_stored_allocator_type type; }; // An ordered sequence of std:pair is also ordered by std::pair::first. struct with_lookup_by_first { typedef void is_transparent; inline bool operator()(std::pair a, std::pair b) const { return a < b; } inline bool operator()(std::pair a, int first) const { return a.first < first; } inline bool operator()(int first, std::pair b) const { return first < b.first; } }; bool test_heterogeneous_lookup_by_partial_key() { typedef flat_map,int, with_lookup_by_first> map_t; map_t map1; map1[std::pair(0, 1)] = 3; map1[std::pair(0, 2)] = 3; std::pair const first_0_range = map1.equal_range(0); if(2 != (first_0_range.second - first_0_range.first)) return false; if(2 != map1.count(0)) return false; return true; } }}} //namespace boost::container::test int main() { using namespace boost::container::test; //Allocator argument container { flat_map map_((flat_map::allocator_type())); flat_multimap multimap_((flat_multimap::allocator_type())); } //Now test move semantics { test_move >(); test_move >(); } //Now test nth/index_of { flat_map map; flat_multimap mmap; map.insert(std::pair(0, 0)); map.insert(std::pair(1, 0)); map.insert(std::pair(2, 0)); mmap.insert(std::pair(0, 0)); mmap.insert(std::pair(1, 0)); mmap.insert(std::pair(2, 0)); if(!boost::container::test::test_nth_index_of(map)) return 1; if(!boost::container::test::test_nth_index_of(mmap)) return 1; } //////////////////////////////////// // Constructor Template Auto Deduction test //////////////////////////////////// if(!constructor_template_auto_deduction_test()){ return 1; } if (!test::test_heterogeneous_lookup < flat_map , flat_multimap >()) return 1; if (!test::test_heterogeneous_insert < flat_map >()) return 1; if (!test_heterogeneous_lookup_by_partial_key()) return 1; //////////////////////////////////// // Testing allocator implementations //////////////////////////////////// { typedef std::map MyStdMap; typedef std::multimap MyStdMultiMap; if (0 != test::flat_map_test < GetMapContainer::apply::map_type , MyStdMap , GetMapContainer::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer >::apply::map_type , MyStdMap , GetMapContainer >::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer::apply::map_type , MyStdMap , GetMapContainer::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer >::apply::map_type , MyStdMap , GetMapContainer >::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer >::apply::map_type , MyStdMap , GetMapContainer >::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer >::apply::map_type , MyStdMap , GetMapContainer >::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } if (0 != test::flat_map_test < GetMapContainer >::apply::map_type , MyStdMap , GetMapContainer >::apply::multimap_type , MyStdMultiMap>()) { std::cout << "Error in flat_map_test >" << std::endl; return 1; } } if(!boost::container::test::test_map_support_for_initialization_list_for >()) return 1; if (!boost::container::test::test_map_support_for_initialization_list_for >()) return 1; //////////////////////////////////// // Emplace testing //////////////////////////////////// const test::EmplaceOptions MapOptions = (test::EmplaceOptions)(test::EMPLACE_HINT_PAIR | test::EMPLACE_ASSOC_PAIR); if(!boost::container::test::test_emplace, MapOptions>()) return 1; if(!boost::container::test::test_emplace, MapOptions>()) return 1; //////////////////////////////////// // Allocator propagation testing //////////////////////////////////// if(!boost::container::test::test_propagate_allocator()) return 1; if(!boost::container::test::test_propagate_allocator()) return 1; //////////////////////////////////// // Iterator testing //////////////////////////////////// { typedef boost::container::flat_map cont_int; cont_int a; a.insert(cont_int::value_type(0, 9)); a.insert(cont_int::value_type(1, 9)); a.insert(cont_int::value_type(2, 9)); boost::intrusive::test::test_iterator_random< cont_int >(a); if(boost::report_errors() != 0) { return 1; } } { typedef boost::container::flat_multimap cont_int; cont_int a; a.insert(cont_int::value_type(0, 9)); a.insert(cont_int::value_type(1, 9)); a.insert(cont_int::value_type(2, 9)); boost::intrusive::test::test_iterator_random< cont_int >(a); if(boost::report_errors() != 0) { return 1; } } //////////////////////////////////// // has_trivial_destructor_after_move testing //////////////////////////////////// { typedef boost::container::dtl::pair value_t; typedef boost::container::dtl::select1st key_of_value_t; // flat_map, default { typedef boost::container::new_allocator alloc_or_cont_t; typedef boost::container::flat_map cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG( boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_map, default) test failed"); } // flat_map, vector { typedef boost::container::vector alloc_or_cont_t; typedef boost::container::flat_map, alloc_or_cont_t> cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG( boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_map, vector) test failed"); } //Old GCCs have problems (compiler bugs) with std::vector and flat_xxx #if !defined(BOOST_GCC) || (BOOST_GCC >= 50000) // flat_map, std::vector { typedef std::vector alloc_or_cont_t; typedef boost::container::flat_map, alloc_or_cont_t> cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG( boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_map, std::vector) test failed"); } #endif // flat_multimap, default { typedef boost::container::new_allocator alloc_or_cont_t; typedef boost::container::flat_multimap cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG( boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_multimap, default) test failed"); } // flat_multimap, vector { typedef boost::container::vector alloc_or_cont_t; typedef boost::container::flat_multimap, alloc_or_cont_t> cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG( boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_multimap, vector) test failed"); } //Old GCCs have problems (compiler bugs) with std::vector and flat_xxx #if !defined(BOOST_GCC) || (BOOST_GCC >= 50000) // flat_multimap, std::vector { typedef std::vector alloc_or_cont_t; typedef boost::container::flat_multimap, alloc_or_cont_t> cont; typedef boost::container::dtl::flat_tree, alloc_or_cont_t> tree; BOOST_CONTAINER_STATIC_ASSERT_MSG (boost::has_trivial_destructor_after_move::value == boost::has_trivial_destructor_after_move::value , "has_trivial_destructor_after_move(flat_multimap, std::vector) test failed"); } #endif } return 0; }