Skip to content

Commit debc5ab

Browse files
committed
Auto merge of #123089 - Philippe-Cholet:vecdeque_pop_assume_cap, r=<try>
Add invariant to VecDeque::pop_* that len < cap if pop successful Similar to #114370 for VecDeque instead of Vec. I initially come from rust-itertools/itertools#899 where we noticed that `pop_front;push_back;` was slower than expected so `@scottmcm` suggested I file an issue which lead to https://internals.rust-lang.org/t/vecdeque-pop-front-push-back/20483 where **kornel** mentionned #114334 (fixed by #114370). This is my first time with codegen tests, I based the test on what was done for Vec.
2 parents 519d892 + e822937 commit debc5ab

File tree

2 files changed

+81
-0
lines changed

2 files changed

+81
-0
lines changed

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

+6
Original file line numberDiff line numberDiff line change
@@ -1614,6 +1614,9 @@ impl<T, A: Allocator> VecDeque<T, A> {
16141614
let old_head = self.head;
16151615
self.head = self.to_physical_idx(1);
16161616
self.len -= 1;
1617+
unsafe {
1618+
core::intrinsics::assume(self.len < self.capacity());
1619+
}
16171620
Some(unsafe { self.buffer_read(old_head) })
16181621
}
16191622
}
@@ -1638,6 +1641,9 @@ impl<T, A: Allocator> VecDeque<T, A> {
16381641
None
16391642
} else {
16401643
self.len -= 1;
1644+
unsafe {
1645+
core::intrinsics::assume(self.len < self.capacity());
1646+
}
16411647
Some(unsafe { self.buffer_read(self.to_physical_idx(self.len)) })
16421648
}
16431649
}

tests/codegen/vecdeque_pop_push.rs

+75
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
//@ compile-flags: -O
2+
3+
#![crate_type = "lib"]
4+
5+
use std::collections::VecDeque;
6+
7+
#[no_mangle]
8+
// CHECK-LABEL: @noop_back(
9+
pub fn noop_back(v: &mut VecDeque<u8>) {
10+
// CHECK-NOT: reserve_for_push
11+
// CHECK-NOT: call
12+
// CHECK: tail call void @llvm.assume
13+
// CHECK-NOT: reserve_for_push
14+
// CHECK-NOT: call
15+
// CHECK: ret
16+
if let Some(x) = v.pop_back() {
17+
v.push_back(x);
18+
}
19+
}
20+
21+
#[no_mangle]
22+
// CHECK-LABEL: @noop_front(
23+
pub fn noop_front(v: &mut VecDeque<u8>) {
24+
// CHECK-NOT: reserve_for_push
25+
// CHECK-NOT: call
26+
// CHECK: tail call void @llvm.assume
27+
// CHECK-NOT: reserve_for_push
28+
// CHECK-NOT: call
29+
// CHECK: ret
30+
if let Some(x) = v.pop_front() {
31+
v.push_front(x);
32+
}
33+
}
34+
35+
#[no_mangle]
36+
// CHECK-LABEL: @move_byte_front_to_back(
37+
pub fn move_byte_front_to_back(v: &mut VecDeque<u8>) {
38+
// CHECK-NOT: reserve_for_push
39+
// CHECK-NOT: call
40+
// CHECK: tail call void @llvm.assume
41+
// CHECK-NOT: reserve_for_push
42+
// CHECK-NOT: call
43+
// CHECK: ret
44+
if let Some(x) = v.pop_front() {
45+
v.push_back(x);
46+
}
47+
}
48+
49+
#[no_mangle]
50+
// CHECK-LABEL: @move_byte_back_to_front(
51+
pub fn move_byte_back_to_front(v: &mut VecDeque<u8>) {
52+
// CHECK-NOT: reserve_for_push
53+
// CHECK-NOT: call
54+
// CHECK: tail call void @llvm.assume
55+
// CHECK-NOT: reserve_for_push
56+
// CHECK-NOT: call
57+
// CHECK: ret
58+
if let Some(x) = v.pop_back() {
59+
v.push_front(x);
60+
}
61+
}
62+
63+
#[no_mangle]
64+
// CHECK-LABEL: @push_back_byte(
65+
pub fn push_back_byte(v: &mut VecDeque<u8>) {
66+
// CHECK: call {{.*}}reserve_for_push
67+
v.push_back(3);
68+
}
69+
70+
#[no_mangle]
71+
// CHECK-LABEL: @push_front_byte(
72+
pub fn push_front_byte(v: &mut VecDeque<u8>) {
73+
// CHECK: call {{.*}}reserve_for_push
74+
v.push_front(3);
75+
}

0 commit comments

Comments
 (0)