1baf6f918Svarconst //===----------------------------------------------------------------------===//
2baf6f918Svarconst //
3baf6f918Svarconst // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4baf6f918Svarconst // See https://llvm.org/LICENSE.txt for license information.
5baf6f918Svarconst // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6baf6f918Svarconst //
7baf6f918Svarconst //===----------------------------------------------------------------------===//
8baf6f918Svarconst
9baf6f918Svarconst // UNSUPPORTED: c++03, c++11, c++14, c++17, c++20
10baf6f918Svarconst
11baf6f918Svarconst // template<container-compatible-range<charT> R>
12baf6f918Svarconst // constexpr basic_string& assign_range(R&& rg); // C++23
13baf6f918Svarconst
14baf6f918Svarconst #include <string>
15baf6f918Svarconst
16baf6f918Svarconst #include "../../../../containers/sequences/insert_range_sequence_containers.h"
17baf6f918Svarconst #include "test_macros.h"
18*9ed20568STacet #include "asan_testing.h"
19baf6f918Svarconst
20baf6f918Svarconst // Tested cases:
21baf6f918Svarconst // - different kinds of assignments (assigning an {empty/one-element/mid-sized/long range} to an
22baf6f918Svarconst // {empty/one-element/full} container);
23baf6f918Svarconst // - an exception is thrown when allocating new elements.
24baf6f918Svarconst
test_constexpr()25baf6f918Svarconst constexpr bool test_constexpr() {
26baf6f918Svarconst for_all_iterators_and_allocators_constexpr<char, const char*>([]<class Iter, class Sent, class Alloc>() {
27346a2990SStephan T. Lavavej test_sequence_assign_range<std::basic_string<char, std::char_traits<char>, Alloc>, Iter, Sent>(
28346a2990SStephan T. Lavavej []([[maybe_unused]] auto&& c) { LIBCPP_ASSERT(c.__invariants()); });
29baf6f918Svarconst });
30baf6f918Svarconst
31baf6f918Svarconst return true;
32baf6f918Svarconst }
33baf6f918Svarconst
main(int,char **)34baf6f918Svarconst int main(int, char**) {
35baf6f918Svarconst static_assert(test_constraints_assign_range<std::basic_string, char, int>());
36baf6f918Svarconst
37baf6f918Svarconst for_all_iterators_and_allocators<char, const char*>([]<class Iter, class Sent, class Alloc>() {
38346a2990SStephan T. Lavavej test_sequence_assign_range<std::basic_string<char, std::char_traits<char>, Alloc>, Iter, Sent>(
39346a2990SStephan T. Lavavej []([[maybe_unused]] auto&& c) { LIBCPP_ASSERT(c.__invariants()); });
40baf6f918Svarconst });
41baf6f918Svarconst static_assert(test_constexpr());
42baf6f918Svarconst
43baf6f918Svarconst { // Check that `assign_range` returns a reference to the string.
44baf6f918Svarconst std::string c;
45a40bada9SBrendan Emery static_assert(std::is_lvalue_reference_v<decltype(c.assign_range(FullContainer_Begin_EmptyRange<char>.input))>);
46baf6f918Svarconst assert(&c.assign_range(FullContainer_Begin_EmptyRange<char>.input) == &c);
47*9ed20568STacet LIBCPP_ASSERT(is_string_asan_correct(c));
48baf6f918Svarconst assert(&c.assign_range(FullContainer_Begin_OneElementRange<char>.input) == &c);
49*9ed20568STacet LIBCPP_ASSERT(is_string_asan_correct(c));
50baf6f918Svarconst assert(&c.assign_range(FullContainer_Begin_MidRange<char>.input) == &c);
51*9ed20568STacet LIBCPP_ASSERT(is_string_asan_correct(c));
52baf6f918Svarconst assert(&c.assign_range(FullContainer_Begin_LongRange<char>.input) == &c);
53*9ed20568STacet LIBCPP_ASSERT(is_string_asan_correct(c));
54baf6f918Svarconst }
55baf6f918Svarconst
56baf6f918Svarconst // Note: `test_assign_range_exception_safety_throwing_copy` doesn't apply because copying chars cannot throw.
57baf6f918Svarconst {
58baf6f918Svarconst #if !defined(TEST_HAS_NO_EXCEPTIONS)
59baf6f918Svarconst // Note: the input string must be long enough to prevent SSO, otherwise the allocator won't be used.
60baf6f918Svarconst std::string in(64, 'a');
61baf6f918Svarconst
62baf6f918Svarconst try {
63baf6f918Svarconst ThrowingAllocator<char> alloc;
64baf6f918Svarconst
65baf6f918Svarconst globalMemCounter.reset();
66baf6f918Svarconst std::basic_string<char, std::char_traits<char>, ThrowingAllocator<char>> c(alloc);
67baf6f918Svarconst c.assign_range(in);
68baf6f918Svarconst assert(false); // The function call above should throw.
69baf6f918Svarconst
70baf6f918Svarconst } catch (int) {
71baf6f918Svarconst assert(globalMemCounter.new_called == globalMemCounter.delete_called);
72baf6f918Svarconst }
73baf6f918Svarconst #endif
74baf6f918Svarconst }
75baf6f918Svarconst
76baf6f918Svarconst return 0;
77baf6f918Svarconst }
78