//===----------------------------------------------------------------------===// // // 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 // //===----------------------------------------------------------------------===// // // void shrink_to_fit(); // constexpr since C++20 #include #include #include "test_macros.h" #include "min_allocator.h" #include "asan_testing.h" template TEST_CONSTEXPR_CXX20 void test(S s) { typename S::size_type old_cap = s.capacity(); S s0 = s; s.shrink_to_fit(); LIBCPP_ASSERT(s.__invariants()); assert(s == s0); assert(s.capacity() <= old_cap); assert(s.capacity() >= s.size()); LIBCPP_ASSERT(is_string_asan_correct(s)); } template TEST_CONSTEXPR_CXX20 void test_string() { S s; test(s); s.assign(10, 'a'); s.erase(5); test(s); s.assign(50, 'a'); s.erase(5); test(s); s.assign(100, 'a'); s.erase(50); test(s); s.assign(100, 'a'); for (int i = 0; i <= 9; ++i) { s.erase(90 - 10 * i); test(s); } } TEST_CONSTEXPR_CXX20 bool test() { test_string(); #if TEST_STD_VER >= 11 test_string, min_allocator>>(); test_string, safe_allocator>>(); #endif return true; } #if TEST_STD_VER >= 23 std::size_t min_bytes = 1000; template struct increasing_allocator { using value_type = T; increasing_allocator() = default; template increasing_allocator(const increasing_allocator&) noexcept {} std::allocation_result allocate_at_least(std::size_t n) { std::size_t allocation_amount = n * sizeof(T); if (allocation_amount < min_bytes) allocation_amount = min_bytes; min_bytes += 1000; return {static_cast(::operator new(allocation_amount)), allocation_amount / sizeof(T)}; } T* allocate(std::size_t n) { return allocate_at_least(n).ptr; } void deallocate(T* p, std::size_t) noexcept { ::operator delete(static_cast(p)); } }; template bool operator==(increasing_allocator, increasing_allocator) { return true; } // https://github.com/llvm/llvm-project/issues/95161 void test_increasing_allocator() { std::basic_string, increasing_allocator> s{ "String does not fit in the internal buffer"}; std::size_t capacity = s.capacity(); std::size_t size = s.size(); s.shrink_to_fit(); assert(s.capacity() <= capacity); assert(s.size() == size); LIBCPP_ASSERT(is_string_asan_correct(s)); } #endif // TEST_STD_VER >= 23 int main(int, char**) { test(); #if TEST_STD_VER >= 23 test_increasing_allocator(); #endif #if TEST_STD_VER > 17 static_assert(test()); #endif return 0; }