/* Copyright (c) 2005-2021 Intel Corporation Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ #ifndef NOMINMAX #define NOMINMAX #endif #include #include "common/concurrent_ordered_common.h" //! \file test_concurrent_map.cpp //! \brief Test for [containers.concurrent_map containers.concurrent_multimap] specifications template struct AllowMultimapping> : std::true_type {}; template using MyAllocator = LocalCountingAllocator>>; using move_support_tests::FooWithAssign; using map_type = tbb::concurrent_map, MyAllocator>; using multimap_type = tbb::concurrent_multimap, MyAllocator>; using checked_map_type = tbb::concurrent_map, std::less, MyAllocator>>; using checked_multimap_type = tbb::concurrent_multimap, std::less, MyAllocator>>; using greater_map_type = tbb::concurrent_map, MyAllocator>; using greater_multimap_type = tbb::concurrent_multimap, MyAllocator>; using checked_state_map_type = tbb::concurrent_map, MyAllocator>; using checked_state_multimap_type = tbb::concurrent_multimap, MyAllocator>; template <> struct SpecialTests { static void Test() { SpecialMapTests(); } }; template <> struct SpecialTests { static void Test() { SpecialMultiMapTests(); } }; struct COMapTraits : OrderedMoveTraitsBase { template using container_type = tbb::concurrent_map, Allocator>; template using container_value_type = std::pair; using init_iterator_type = move_support_tests::FooPairIterator; }; // struct COMapTraits struct COMultimapTraits : OrderedMoveTraitsBase { template using container_type = tbb::concurrent_multimap, Allocator>; template using container_value_type = std::pair; using init_iterator_type = move_support_tests::FooPairIterator; }; // struct COMultimapTraits struct OrderedMapTypesTester { template void check( const std::list& lst ) { using key_type = typename ValueType::first_type; using mapped_type = typename ValueType::second_type; TypeTester>(lst); TypeTester>(lst); } }; // struct OrderedMapTypesTester void test_specific_types() { test_map_specific_types(); // Regression test for a problem with excessive requirements of emplace() test_emplace_insert>,std::false_type> (new int, new int); test_emplace_insert>,std::false_type> (new int, new int); } // Regression test for an issue in lock free algorithms // In some cases this test hangs due to broken skip list internal structure on levels > 1 // This issue was resolved by adding index_number into the skip list node void test_cycles_absense() { for (std::size_t execution = 0; execution != 10; ++execution) { tbb::concurrent_multimap mmap; std::vector v(2); int vector_size = int(v.size()); for (int i = 0; i != vector_size; ++i) { v[i] = i; } size_t num_threads = 4; // Can be changed to 2 for debugging utils::NativeParallelFor(num_threads, [&](size_t) { for (int i = 0; i != vector_size; ++i) { mmap.emplace(i, i); } }); for (int i = 0; i != vector_size; ++i) { REQUIRE(mmap.count(i) == num_threads); } } } //! \brief \ref error_guessing TEST_CASE("basic test for concurrent_map with greater compare") { test_basic(); } //! \brief \ref error_guessing TEST_CASE("basic test for concurrent_multimap with greater compare") { test_basic(); } //! \brief \ref resource_usage TEST_CASE("basic test for concurrent_map with elements ctor and dtor check") { Checker checker; test_basic(); } //! \brief \ref resource_usage TEST_CASE("basic test for concurrent_multimap with elements ctor and dtor check") { Checker checker; test_basic(); } //! \brief \ref resource_usage TEST_CASE("basic test for concurrent_map with elements state check") { test_basic(); } //! \brief \ref resource_usage TEST_CASE("basic test for concurrent_multimap with elements state check") { test_basic(); } //! \brief \ref error_guessing TEST_CASE("multithreading support in concurrent_map with greater compare") { test_concurrent(); } //! \brief \ref error_guessing TEST_CASE("multithreading support in concurrent_multimap with greater compare") { test_concurrent(); } //! \brief \ref resource_usage TEST_CASE("multithreading support in concurrent_map with elements ctor and dtor check") { Checker checker; test_concurrent(); } //! \brief \ref resource_usage TEST_CASE("multithreading support in concurrent_multimap with elements ctor and dtor check") { Checker checker; test_concurrent(); } //! \brief \ref resource_usage TEST_CASE("multithreading support in concurrent_map with elements state check") { test_concurrent(); } //! \brief \ref resource_usage TEST_CASE("multithreading support in concurrent_multimap with elements state check") { test_concurrent(); } //! \brief \ref stress \ref error_guessing TEST_CASE("multithreading support in concurrent_multimap no unique keys") { test_concurrent(true); } //! \brief \ref stress \ref error_guessing TEST_CASE("multithreading support in concurrent_multimap with greater compare and no unique keys") { test_concurrent(true); } //! \brief \ref interface \ref error_guessing TEST_CASE("range based for support in concurrent_map") { test_range_based_for_support(); } //! \brief \ref interface \ref error_guessing TEST_CASE("range based for support in concurrent_multimap") { test_range_based_for_support(); } //! \brief \ref regression TEST_CASE("concurrent_map/multimap with specific key/mapped types") { test_specific_types(); } // TODO: add test with scoped_allocator_adaptor with broken macro //! \brief \ref regression TEST_CASE("broken internal structure for multimap") { test_cycles_absense(); } //! \brief \ref error_guessing TEST_CASE("concurrent_map::swap with not always equal allocator") { using not_always_equal_alloc_map_type = tbb::concurrent_map, NotAlwaysEqualAllocator>>; test_swap_not_always_equal_allocator(); } //! \brief \ref error_guessing TEST_CASE("concurrent_multimap::swap with not always equal allocator") { using not_always_equal_alloc_mmap_type = tbb::concurrent_multimap, NotAlwaysEqualAllocator>>; test_swap_not_always_equal_allocator(); } #if TBB_USE_EXCEPTIONS //! \brief \ref error_guessing TEST_CASE("concurrent_map throwing copy constructor") { using exception_map_type = tbb::concurrent_map; test_exception_on_copy_ctor(); } //! \brief \ref error_guessing TEST_CASE("concurrent_multimap throwing copy constructor") { using exception_mmap_type = tbb::concurrent_multimap; test_exception_on_copy_ctor(); } #endif // TBB_USE_EXCEPTIONS #if __TBB_CPP20_CONCEPTS_PRESENT //! \brief \ref error_guessing TEST_CASE("container_range concept for concurrent_map ranges") { static_assert(test_concepts::container_range::range_type>); static_assert(test_concepts::container_range::const_range_type>); } //! \brief \ref error_guessing TEST_CASE("container_range concept for concurrent_multimap ranges") { static_assert(test_concepts::container_range::range_type>); static_assert(test_concepts::container_range::const_range_type>); } #endif // __TBB_CPP20_CONCEPTS_PRESENT