//===----------------------------------------------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20 // // containers extract() &&; #include #include #include #include #include #include #include "MinSequenceContainer.h" #include "../helpers.h" #include "test_macros.h" #include "min_allocator.h" template concept CanExtract = requires(T&& t) { std::forward(t).extract(); }; static_assert(CanExtract&&>); static_assert(!CanExtract&>); static_assert(!CanExtract const&>); static_assert(!CanExtract const&&>); template void test() { using M = std::flat_map, KeyContainer, ValueContainer>; M m = M({1, 2, 3}, {4, 5, 6}); std::same_as auto containers = std::move(m).extract(); auto expected_keys = {1, 2, 3}; auto expected_values = {4, 5, 6}; assert(std::ranges::equal(containers.keys, expected_keys)); assert(std::ranges::equal(containers.values, expected_values)); check_invariant(m); LIBCPP_ASSERT(m.empty()); LIBCPP_ASSERT(m.keys().size() == 0); LIBCPP_ASSERT(m.values().size() == 0); } int main(int, char**) { test, std::vector>(); test, std::vector>(); test, MinSequenceContainer>(); test>, std::vector>>(); { // extracted object maintains invariant if one of underlying container does not clear after move using M = std::flat_map, std::vector, CopyOnlyVector>; M m = M({1, 2, 3}, {1, 2, 3}); std::same_as auto containers = std::move(m).extract(); assert(containers.keys.size() == 3); assert(containers.values.size() == 3); check_invariant(m); LIBCPP_ASSERT(m.empty()); LIBCPP_ASSERT(m.keys().size() == 0); LIBCPP_ASSERT(m.values().size() == 0); } { #ifndef TEST_HAS_NO_EXCEPTIONS using KeyContainer = std::vector; using ValueContainer = ThrowOnMoveContainer; using M = std::flat_map; M m; m.emplace(1, 1); m.emplace(2, 2); try { auto c = std::move(m).extract(); assert(false); } catch (int) { check_invariant(m); // In libc++, we try to erase the key after value emplacement failure. // and after erasure failure, we clear the flat_map LIBCPP_ASSERT(m.size() == 0); } #endif } return 0; }