1841132efSMarek Kurdej //===----------------------------------------------------------------------===//
2841132efSMarek Kurdej //
3841132efSMarek Kurdej // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4841132efSMarek Kurdej // See https://llvm.org/LICENSE.txt for license information.
5841132efSMarek Kurdej // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6841132efSMarek Kurdej //
7841132efSMarek Kurdej //===----------------------------------------------------------------------===//
8841132efSMarek Kurdej 
9841132efSMarek Kurdej // <string>
10841132efSMarek Kurdej 
11*425620ccSNikolas Klauser // void reserve(size_type res_arg); // constexpr since C++20
12841132efSMarek Kurdej 
13841132efSMarek Kurdej // This test relies on https://llvm.org/PR45368 being fixed, which isn't in
14841132efSMarek Kurdej // older Apple dylibs
15b7042b73SLouis Dionne // XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx{{10.9|10.10|10.11|10.12|10.13|10.14|10.15|11.0}}
16841132efSMarek Kurdej 
17841132efSMarek Kurdej #include <string>
18841132efSMarek Kurdej #include <stdexcept>
19841132efSMarek Kurdej #include <cassert>
20841132efSMarek Kurdej 
21841132efSMarek Kurdej #include "test_macros.h"
22841132efSMarek Kurdej #include "min_allocator.h"
23841132efSMarek Kurdej 
24841132efSMarek Kurdej template <class S>
2585e9b268SNikolas Klauser TEST_CONSTEXPR_CXX20 void
test(typename S::size_type min_cap,typename S::size_type erased_index,typename S::size_type res_arg)26841132efSMarek Kurdej test(typename S::size_type min_cap, typename S::size_type erased_index, typename S::size_type res_arg)
27841132efSMarek Kurdej {
28841132efSMarek Kurdej     S s(min_cap, 'a');
29841132efSMarek Kurdej     s.erase(erased_index);
30841132efSMarek Kurdej     assert(s.size() == erased_index);
31841132efSMarek Kurdej     assert(s.capacity() >= min_cap); // Check that we really have at least this capacity.
32841132efSMarek Kurdej 
33841132efSMarek Kurdej #if TEST_STD_VER > 17
34841132efSMarek Kurdej     typename S::size_type old_cap = s.capacity();
35841132efSMarek Kurdej #endif
36841132efSMarek Kurdej     S s0 = s;
37841132efSMarek Kurdej     if (res_arg <= s.max_size())
38841132efSMarek Kurdej     {
39841132efSMarek Kurdej         s.reserve(res_arg);
40841132efSMarek Kurdej         LIBCPP_ASSERT(s.__invariants());
41841132efSMarek Kurdej         assert(s == s0);
42841132efSMarek Kurdej         assert(s.capacity() >= res_arg);
43841132efSMarek Kurdej         assert(s.capacity() >= s.size());
44841132efSMarek Kurdej #if TEST_STD_VER > 17
45841132efSMarek Kurdej         assert(s.capacity() >= old_cap); // reserve never shrinks as of P0966 (C++20)
46841132efSMarek Kurdej #endif
47841132efSMarek Kurdej     }
48841132efSMarek Kurdej #ifndef TEST_HAS_NO_EXCEPTIONS
4985e9b268SNikolas Klauser     else if (!TEST_IS_CONSTANT_EVALUATED)
50841132efSMarek Kurdej     {
51841132efSMarek Kurdej         try
52841132efSMarek Kurdej         {
53841132efSMarek Kurdej             s.reserve(res_arg);
54841132efSMarek Kurdej             LIBCPP_ASSERT(s.__invariants());
55841132efSMarek Kurdej             assert(false);
56841132efSMarek Kurdej         }
57841132efSMarek Kurdej         catch (std::length_error&)
58841132efSMarek Kurdej         {
59841132efSMarek Kurdej             assert(res_arg > s.max_size());
60841132efSMarek Kurdej         }
61841132efSMarek Kurdej     }
62841132efSMarek Kurdej #endif
63841132efSMarek Kurdej }
64841132efSMarek Kurdej 
test()65*425620ccSNikolas Klauser TEST_CONSTEXPR_CXX20 bool test() {
66841132efSMarek Kurdej   {
67841132efSMarek Kurdej     typedef std::string S;
68841132efSMarek Kurdej     {
69841132efSMarek Kurdej       test<S>(0, 0, 5);
70841132efSMarek Kurdej       test<S>(0, 0, 10);
71841132efSMarek Kurdej       test<S>(0, 0, 50);
72841132efSMarek Kurdej     }
73841132efSMarek Kurdej     {
74841132efSMarek Kurdej       test<S>(100, 50, 5);
75841132efSMarek Kurdej       test<S>(100, 50, 10);
76841132efSMarek Kurdej       test<S>(100, 50, 50);
77841132efSMarek Kurdej       test<S>(100, 50, 100);
78841132efSMarek Kurdej       test<S>(100, 50, 1000);
79841132efSMarek Kurdej       test<S>(100, 50, S::npos);
80841132efSMarek Kurdej     }
81841132efSMarek Kurdej   }
82841132efSMarek Kurdej #if TEST_STD_VER >= 11
83841132efSMarek Kurdej   {
84841132efSMarek Kurdej     typedef std::basic_string<char, std::char_traits<char>, min_allocator<char>> S;
85841132efSMarek Kurdej     {
86841132efSMarek Kurdej       test<S>(0, 0, 5);
87841132efSMarek Kurdej       test<S>(0, 0, 10);
88841132efSMarek Kurdej       test<S>(0, 0, 50);
89841132efSMarek Kurdej     }
90841132efSMarek Kurdej     {
91841132efSMarek Kurdej       test<S>(100, 50, 5);
92841132efSMarek Kurdej       test<S>(100, 50, 10);
93841132efSMarek Kurdej       test<S>(100, 50, 50);
94841132efSMarek Kurdej       test<S>(100, 50, 100);
95841132efSMarek Kurdej       test<S>(100, 50, 1000);
96841132efSMarek Kurdej       test<S>(100, 50, S::npos);
97841132efSMarek Kurdej     }
98841132efSMarek Kurdej   }
99841132efSMarek Kurdej #endif
100841132efSMarek Kurdej 
101e85018b7SNikolas Klauser   return true;
102e85018b7SNikolas Klauser }
103e85018b7SNikolas Klauser 
main(int,char **)104e85018b7SNikolas Klauser int main(int, char**)
105e85018b7SNikolas Klauser {
106e85018b7SNikolas Klauser   test();
107e85018b7SNikolas Klauser #if TEST_STD_VER > 17
108*425620ccSNikolas Klauser   static_assert(test());
109e85018b7SNikolas Klauser #endif
110e85018b7SNikolas Klauser 
111841132efSMarek Kurdej   return 0;
112841132efSMarek Kurdej }
113