[libcxx-commits] [libcxx] [libc++][vector] Fixes shrink_to_fit. (PR #97895)

Mark de Wever via libcxx-commits libcxx-commits at lists.llvm.org
Sun Jul 7 09:40:56 PDT 2024


https://github.com/mordante updated https://github.com/llvm/llvm-project/pull/97895

>From 2e82109050a26db1c9696ba6c355903ae59fcf7c Mon Sep 17 00:00:00 2001
From: Mark de Wever <koraq at xs4all.nl>
Date: Sat, 6 Jul 2024 14:57:15 +0200
Subject: [PATCH 1/2] [libc++][vector] Fixes shrink_to_fit.

This assures shrink_to_fit does not increase the allocated size.

Partly addresses https://github.com/llvm/llvm-project/issues/95161
---
 libcxx/include/vector                         |  6 ++-
 .../vector.capacity/shrink_to_fit.pass.cpp    | 41 +++++++++++++++++++
 2 files changed, 46 insertions(+), 1 deletion(-)

diff --git a/libcxx/include/vector b/libcxx/include/vector
index 299ad8c9b23f2..7c7adb0da7da5 100644
--- a/libcxx/include/vector
+++ b/libcxx/include/vector
@@ -1443,7 +1443,11 @@ _LIBCPP_CONSTEXPR_SINCE_CXX20 void vector<_Tp, _Allocator>::shrink_to_fit() _NOE
 #endif // _LIBCPP_HAS_NO_EXCEPTIONS
       allocator_type& __a = this->__alloc();
       __split_buffer<value_type, allocator_type&> __v(size(), size(), __a);
-      __swap_out_circular_buffer(__v);
+      // The Standard mandates shrink_to_fit() does not increase the capacity.
+      // With equal capacity keep the existing buffer. This avoids extra work
+      // due to swapping the elements.
+      if (__v.capacity() < capacity())
+        __swap_out_circular_buffer(__v);
 #ifndef _LIBCPP_HAS_NO_EXCEPTIONS
     } catch (...) {
     }
diff --git a/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp b/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
index 8851e2a9ed0c7..d0bdf4fddc225 100644
--- a/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
+++ b/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
@@ -71,9 +71,50 @@ TEST_CONSTEXPR_CXX20 bool tests() {
     return true;
 }
 
+#if TEST_STD_VER >= 23
+std::size_t min_bytes = 1000;
+
+template <typename T>
+struct increasing_allocator {
+  using value_type       = T;
+  increasing_allocator() = default;
+  template <typename U>
+  increasing_allocator(const increasing_allocator<U>&) noexcept {}
+  std::allocation_result<T*> 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<T*>(::operator new(allocation_amount)), allocation_amount};
+  }
+  T* allocate(std::size_t n) { return allocate_at_least(n).ptr; }
+  void deallocate(T* p, std::size_t) noexcept { ::operator delete(static_cast<void*>(p)); }
+};
+
+template <typename T, typename U>
+bool operator==(increasing_allocator<T>, increasing_allocator<U>) {
+  return true;
+}
+
+// https://github.com/llvm/llvm-project/issues/95161
+void test_increasing_allocator() {
+  std::vector<int, increasing_allocator<int>> v;
+  v.push_back(1);
+  assert(is_contiguous_container_asan_correct(v));
+  std::size_t capacity = v.capacity();
+  v.shrink_to_fit();
+  assert(v.capacity() == capacity);
+  assert(v.size() == 1);
+  assert(is_contiguous_container_asan_correct(v));
+}
+#endif // TEST_STD_VER >= 23
+
 int main(int, char**)
 {
     tests();
+#if TEST_STD_VER >= 23
+    test_increasing_allocator();
+#endif
 #if TEST_STD_VER > 17
     static_assert(tests());
 #endif

>From 78dedd788a276f18201e1d5c827fb453bb1755a3 Mon Sep 17 00:00:00 2001
From: Mark de Wever <zar-rpg at xs4all.nl>
Date: Sun, 7 Jul 2024 18:40:49 +0200
Subject: [PATCH 2/2] Apply suggestions from code review

Co-authored-by: Mital Ashok <mital.vaja at googlemail.com>
---
 .../sequences/vector/vector.capacity/shrink_to_fit.pass.cpp   | 4 ++--
 1 file changed, 2 insertions(+), 2 deletions(-)

diff --git a/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp b/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
index d0bdf4fddc225..a2f0900cf8110 100644
--- a/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
+++ b/libcxx/test/std/containers/sequences/vector/vector.capacity/shrink_to_fit.pass.cpp
@@ -85,7 +85,7 @@ struct increasing_allocator {
     if (allocation_amount < min_bytes)
       allocation_amount = min_bytes;
     min_bytes += 1000;
-    return {static_cast<T*>(::operator new(allocation_amount)), allocation_amount};
+    return {static_cast<T*>(::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<void*>(p)); }
@@ -103,7 +103,7 @@ void test_increasing_allocator() {
   assert(is_contiguous_container_asan_correct(v));
   std::size_t capacity = v.capacity();
   v.shrink_to_fit();
-  assert(v.capacity() == capacity);
+  assert(v.capacity() <= capacity);
   assert(v.size() == 1);
   assert(is_contiguous_container_asan_correct(v));
 }



More information about the libcxx-commits mailing list