//===----------------------------------------------------------------------===// // // The LLVM Compiler Infrastructure // // This file is dual licensed under the MIT and the University of Illinois Open // Source Licenses. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // void swap(set& c) // noexcept(!allocator_type::propagate_on_container_swap::value || // __is_nothrow_swappable::value); // // In C++17, the standard says that swap shall have: // noexcept(allocator_traits::is_always_equal::value && // noexcept(swap(declval(), declval()))); // This tests a conforming extension #include #include #include "MoveOnly.h" #include "test_allocator.h" template struct some_comp { typedef T value_type; some_comp() {} some_comp(const some_comp&) {} void deallocate(void*, unsigned) {} typedef std::true_type propagate_on_container_swap; }; template struct some_comp2 { typedef T value_type; some_comp2() {} some_comp2(const some_comp2&) {} void deallocate(void*, unsigned) {} typedef std::true_type propagate_on_container_swap; }; #if TEST_STD_VER >= 14 template void swap(some_comp2&, some_comp2&) noexcept {} #endif template struct some_alloc { typedef T value_type; some_alloc() {} some_alloc(const some_alloc&); void deallocate(void*, unsigned) {} typedef std::true_type propagate_on_container_swap; }; template struct some_alloc2 { typedef T value_type; some_alloc2() {} some_alloc2(const some_alloc2&); void deallocate(void*, unsigned) {} typedef std::false_type propagate_on_container_swap; typedef std::true_type is_always_equal; }; template struct some_alloc3 { typedef T value_type; some_alloc3() {} some_alloc3(const some_alloc3&); void deallocate(void*, unsigned) {} typedef std::false_type propagate_on_container_swap; typedef std::false_type is_always_equal; }; int main() { #if __has_feature(cxx_noexcept) { typedef std::set C; C c1, c2; static_assert(noexcept(swap(c1, c2)), ""); } { typedef std::set, test_allocator> C; C c1, c2; static_assert(noexcept(swap(c1, c2)), ""); } { typedef std::set, other_allocator> C; C c1, c2; static_assert(noexcept(swap(c1, c2)), ""); } { typedef std::set> C; C c1, c2; static_assert(!noexcept(swap(c1, c2)), ""); } #if TEST_STD_VER >= 14 { // POCS allocator, throwable swap for comp typedef std::set, some_alloc > C; C c1, c2; static_assert(!noexcept(swap(c1, c2)), ""); } { // always equal allocator, throwable swap for comp typedef std::set, some_alloc2> C; C c1, c2; static_assert(!noexcept(swap(c1, c2)), ""); } { // POCS allocator, nothrow swap for comp typedef std::set, some_alloc > C; C c1, c2; static_assert( noexcept(swap(c1, c2)), ""); } { // always equal allocator, nothrow swap for comp typedef std::set, some_alloc2> C; C c1, c2; static_assert( noexcept(swap(c1, c2)), ""); } { // NOT always equal allocator, nothrow swap for comp typedef std::set, some_alloc3> C; C c1, c2; static_assert( noexcept(swap(c1, c2)), ""); } #endif #endif }