Skip to content

Commit a2f8f62

Browse files
committed
Auto merge of #81335 - thomwiggers:no-panic-shrink-to, r=Mark-Simulacrum
Trying to shrink_to greater than capacity should be no-op Per the discussion in #56431, `shrink_to` shouldn't panic if you try to make a vector shrink to a capacity greater than its current capacity.
2 parents 613ef74 + d069c58 commit a2f8f62

File tree

8 files changed

+15
-32
lines changed

8 files changed

+15
-32
lines changed

library/alloc/src/collections/binary_heap.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -870,8 +870,7 @@ impl<T> BinaryHeap<T> {
870870
/// The capacity will remain at least as large as both the length
871871
/// and the supplied value.
872872
///
873-
/// Panics if the current capacity is smaller than the supplied
874-
/// minimum capacity.
873+
/// If the current capacity is less than the lower limit, this is a no-op.
875874
///
876875
/// # Examples
877876
///

library/alloc/src/collections/vec_deque/mod.rs

+4-6
Original file line numberDiff line numberDiff line change
@@ -761,8 +761,7 @@ impl<T> VecDeque<T> {
761761
/// The capacity will remain at least as large as both the length
762762
/// and the supplied value.
763763
///
764-
/// Panics if the current capacity is smaller than the supplied
765-
/// minimum capacity.
764+
/// If the current capacity is less than the lower limit, this is a no-op.
766765
///
767766
/// # Examples
768767
///
@@ -780,10 +779,9 @@ impl<T> VecDeque<T> {
780779
/// ```
781780
#[unstable(feature = "shrink_to", reason = "new API", issue = "56431")]
782781
pub fn shrink_to(&mut self, min_capacity: usize) {
783-
assert!(self.capacity() >= min_capacity, "Tried to shrink to a larger capacity");
784-
785-
// +1 since the ringbuffer always leaves one space empty
786-
// len + 1 can't overflow for an existing, well-formed ringbuffer.
782+
let min_capacity = cmp::min(min_capacity, self.capacity());
783+
// We don't have to worry about an overflow as neither `self.len()` nor `self.capacity()`
784+
// can ever be `usize::MAX`. +1 as the ringbuffer always leaves one space empty.
787785
let target_cap = cmp::max(cmp::max(min_capacity, self.len()) + 1, MINIMUM_CAPACITY + 1)
788786
.next_power_of_two();
789787

library/alloc/src/string.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -1036,8 +1036,7 @@ impl String {
10361036
/// The capacity will remain at least as large as both the length
10371037
/// and the supplied value.
10381038
///
1039-
/// Panics if the current capacity is smaller than the supplied
1040-
/// minimum capacity.
1039+
/// If the current capacity is less than the lower limit, this is a no-op.
10411040
///
10421041
/// # Examples
10431042
///

library/alloc/src/vec/mod.rs

+6-6
Original file line numberDiff line numberDiff line change
@@ -321,7 +321,7 @@ mod spec_extend;
321321
/// ensures no unnecessary allocations or deallocations occur. Emptying a `Vec`
322322
/// and then filling it back up to the same [`len`] should incur no calls to
323323
/// the allocator. If you wish to free up unused memory, use
324-
/// [`shrink_to_fit`].
324+
/// [`shrink_to_fit`] or [`shrink_to`].
325325
///
326326
/// [`push`] and [`insert`] will never (re)allocate if the reported capacity is
327327
/// sufficient. [`push`] and [`insert`] *will* (re)allocate if
@@ -360,6 +360,7 @@ mod spec_extend;
360360
/// [`String`]: crate::string::String
361361
/// [`&str`]: type@str
362362
/// [`shrink_to_fit`]: Vec::shrink_to_fit
363+
/// [`shrink_to`]: Vec::shrink_to
363364
/// [`capacity`]: Vec::capacity
364365
/// [`mem::size_of::<T>`]: core::mem::size_of
365366
/// [`len`]: Vec::len
@@ -909,10 +910,7 @@ impl<T, A: Allocator> Vec<T, A> {
909910
/// The capacity will remain at least as large as both the length
910911
/// and the supplied value.
911912
///
912-
/// # Panics
913-
///
914-
/// Panics if the current capacity is smaller than the supplied
915-
/// minimum capacity.
913+
/// If the current capacity is less than the lower limit, this is a no-op.
916914
///
917915
/// # Examples
918916
///
@@ -929,7 +927,9 @@ impl<T, A: Allocator> Vec<T, A> {
929927
#[doc(alias = "realloc")]
930928
#[unstable(feature = "shrink_to", reason = "new API", issue = "56431")]
931929
pub fn shrink_to(&mut self, min_capacity: usize) {
932-
self.buf.shrink_to_fit(cmp::max(self.len, min_capacity));
930+
if self.capacity() > min_capacity {
931+
self.buf.shrink_to_fit(cmp::max(self.len, min_capacity));
932+
}
933933
}
934934

935935
/// Converts the vector into [`Box<[T]>`][owned slice].

library/std/src/collections/hash/map.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -658,8 +658,7 @@ where
658658
/// down no lower than the supplied limit while maintaining the internal rules
659659
/// and possibly leaving some space in accordance with the resize policy.
660660
///
661-
/// Panics if the current capacity is smaller than the supplied
662-
/// minimum capacity.
661+
/// If the current capacity is less than the lower limit, this is a no-op.
663662
///
664663
/// # Examples
665664
///
@@ -679,7 +678,6 @@ where
679678
#[inline]
680679
#[unstable(feature = "shrink_to", reason = "new API", issue = "56431")]
681680
pub fn shrink_to(&mut self, min_capacity: usize) {
682-
assert!(self.capacity() >= min_capacity, "Tried to shrink to a larger capacity");
683681
self.base.shrink_to(min_capacity);
684682
}
685683

library/std/src/collections/hash/set.rs

+1-3
Original file line numberDiff line numberDiff line change
@@ -462,9 +462,7 @@ where
462462
/// down no lower than the supplied limit while maintaining the internal rules
463463
/// and possibly leaving some space in accordance with the resize policy.
464464
///
465-
/// Panics if the current capacity is smaller than the supplied
466-
/// minimum capacity.
467-
///
465+
/// If the current capacity is less than the lower limit, this is a no-op.
468466
/// # Examples
469467
///
470468
/// ```

library/std/src/ffi/os_str.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -304,8 +304,7 @@ impl OsString {
304304
/// The capacity will remain at least as large as both the length
305305
/// and the supplied value.
306306
///
307-
/// Panics if the current capacity is smaller than the supplied
308-
/// minimum capacity.
307+
/// If the current capacity is less than the lower limit, this is a no-op.
309308
///
310309
/// # Examples
311310
///

src/test/codegen/vec-shrink-panic.rs

-8
Original file line numberDiff line numberDiff line change
@@ -26,11 +26,3 @@ pub fn issue75636<'a>(iter: &[&'a str]) -> Box<[&'a str]> {
2626
// CHECK-NOT: panic
2727
iter.iter().copied().collect()
2828
}
29-
30-
// Sanity-check that we do see a possible panic for an arbitrary `Vec::shrink_to`.
31-
// CHECK-LABEL: @shrink_to
32-
#[no_mangle]
33-
pub fn shrink_to(vec: &mut Vec<u32>) {
34-
// CHECK: panic
35-
vec.shrink_to(42);
36-
}

0 commit comments

Comments
 (0)