Skip to content

Commit 64b0277

Browse files
committed
Auto merge of #29454 - stepancheg:vec-reserve, r=bluss
Before this patch `reserve` function allocated twice as requested amount elements (not twice as capacity). It leaded to unnecessary excessive memory usage in scenarios like this: ``` let mut v = Vec::new(); v.push(17); v.extend(0..10); println!("{}", v.capacity()); ``` `Vec` allocated 22 elements, while it could allocate just 11. `reserve` function must have a property of keeping `push` operation cost (which calls `reserve`) `O(1)`. To achieve this `reserve` must exponentialy grow its capacity when it does reallocation. There's better strategy to implement `reserve`: ``` let new_capacity = max(current_capacity * 2, requested_capacity); ``` This strategy still guarantees that capacity grows at `O(1)` with `reserve`, and fixes the issue with `extend`. Patch imlpements this strategy.
2 parents cc8d398 + 46068c9 commit 64b0277

File tree

1 file changed

+49
-3
lines changed

1 file changed

+49
-3
lines changed

src/liballoc/raw_vec.rs

+49-3
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ use heap;
1515
use super::oom;
1616
use super::boxed::Box;
1717
use core::ops::Drop;
18+
use core::cmp;
1819
use core;
1920

2021
/// A low-level utility for more ergonomically allocating, reallocating, and deallocating a
@@ -360,9 +361,15 @@ impl<T> RawVec<T> {
360361
}
361362

362363
// Nothing we can really do about these checks :(
363-
let new_cap = used_cap.checked_add(needed_extra_cap)
364-
.and_then(|cap| cap.checked_mul(2))
365-
.expect("capacity overflow");
364+
let required_cap = used_cap.checked_add(needed_extra_cap)
365+
.expect("capacity overflow");
366+
367+
// Cannot overflow, because `cap <= isize::MAX`, and type of `cap` is `usize`.
368+
let double_cap = self.cap * 2;
369+
370+
// `double_cap` guarantees exponential growth.
371+
let new_cap = cmp::max(double_cap, required_cap);
372+
366373
let new_alloc_size = new_cap.checked_mul(elem_size).expect("capacity overflow");
367374
// FIXME: may crash and burn on over-reserve
368375
alloc_guard(new_alloc_size);
@@ -486,3 +493,42 @@ fn alloc_guard(alloc_size: usize) {
486493
"capacity overflow");
487494
}
488495
}
496+
497+
498+
#[cfg(test)]
499+
mod tests {
500+
use super::*;
501+
502+
#[test]
503+
fn reserve_does_not_overallocate() {
504+
{
505+
let mut v: RawVec<u32> = RawVec::new();
506+
// First `reserve` allocates like `reserve_exact`
507+
v.reserve(0, 9);
508+
assert_eq!(9, v.cap());
509+
}
510+
511+
{
512+
let mut v: RawVec<u32> = RawVec::new();
513+
v.reserve(0, 7);
514+
assert_eq!(7, v.cap());
515+
// 97 if more than double of 7, so `reserve` should work
516+
// like `reserve_exact`.
517+
v.reserve(7, 90);
518+
assert_eq!(97, v.cap());
519+
}
520+
521+
{
522+
let mut v: RawVec<u32> = RawVec::new();
523+
v.reserve(0, 12);
524+
assert_eq!(12, v.cap());
525+
v.reserve(12, 3);
526+
// 3 is less than half of 12, so `reserve` must grow
527+
// exponentially. At the time of writing this test grow
528+
// factor is 2, so new capacity is 24, however, grow factor
529+
// of 1.5 is OK too. Hence `>= 18` in assert.
530+
assert!(v.cap() >= 12 + 12 / 2);
531+
}
532+
}
533+
534+
}

0 commit comments

Comments
 (0)