Skip to content

Commit ba79ac7

Browse files
committed
Fix potential buffer overflow in insert_many
Fixes #252.
1 parent 0b2b4e5 commit ba79ac7

File tree

2 files changed

+40
-21
lines changed

2 files changed

+40
-21
lines changed

src/lib.rs

Lines changed: 27 additions & 21 deletions
Original file line numberDiff line numberDiff line change
@@ -1009,7 +1009,7 @@ impl<A: Array> SmallVec<A> {
10091009
/// Insert multiple elements at position `index`, shifting all following elements toward the
10101010
/// back.
10111011
pub fn insert_many<I: IntoIterator<Item = A::Item>>(&mut self, index: usize, iterable: I) {
1012-
let iter = iterable.into_iter();
1012+
let mut iter = iterable.into_iter();
10131013
if index == self.len() {
10141014
return self.extend(iter);
10151015
}
@@ -1019,11 +1019,13 @@ impl<A: Array> SmallVec<A> {
10191019
assert!(index + lower_size_bound >= index); // Protect against overflow
10201020
self.reserve(lower_size_bound);
10211021

1022+
let mut num_added = 0;
1023+
let old_len = self.len();
1024+
assert!(index <= old_len);
1025+
10221026
unsafe {
1023-
let old_len = self.len();
1024-
assert!(index <= old_len);
10251027
let start = self.as_mut_ptr();
1026-
let mut ptr = start.add(index);
1028+
let ptr = start.add(index);
10271029

10281030
// Move the trailing elements.
10291031
ptr::copy(ptr, ptr.add(lower_size_bound), old_len - index);
@@ -1036,26 +1038,16 @@ impl<A: Array> SmallVec<A> {
10361038
len: old_len + lower_size_bound,
10371039
};
10381040

1039-
let mut num_added = 0;
1040-
for element in iter {
1041-
let mut cur = ptr.add(num_added);
1042-
if num_added >= lower_size_bound {
1043-
// Iterator provided more elements than the hint. Move trailing items again.
1044-
self.reserve(1);
1045-
let start = self.as_mut_ptr();
1046-
ptr = start.add(index);
1047-
cur = ptr.add(num_added);
1048-
ptr::copy(cur, cur.add(1), old_len - index);
1049-
1050-
guard.start = start;
1051-
guard.len += 1;
1052-
guard.skip.end += 1;
1053-
}
1041+
while num_added < lower_size_bound {
1042+
let element = match iter.next() {
1043+
Some(x) => x,
1044+
None => break,
1045+
};
1046+
let cur = ptr.add(num_added);
10541047
ptr::write(cur, element);
10551048
guard.skip.start += 1;
10561049
num_added += 1;
10571050
}
1058-
mem::forget(guard);
10591051

10601052
if num_added < lower_size_bound {
10611053
// Iterator provided fewer elements than the hint
@@ -1066,12 +1058,26 @@ impl<A: Array> SmallVec<A> {
10661058
);
10671059
}
10681060

1061+
// There are no more duplicate or uninitialized slots, so the guard is not needed.
10691062
self.set_len(old_len + num_added);
1063+
mem::forget(guard);
1064+
}
1065+
1066+
unsafe {
1067+
for element in iter {
1068+
// Iterator provided more elements than the hint. Move trailing items again.
1069+
self.reserve(1);
1070+
let cur = self.as_mut_ptr().add(index).add(num_added);
1071+
ptr::copy(cur, cur.add(1), old_len - index);
1072+
ptr::write(cur, element);
1073+
self.set_len(self.len() + 1);
1074+
num_added += 1;
1075+
}
10701076
}
10711077

10721078
struct DropOnPanic<T> {
10731079
start: *mut T,
1074-
skip: Range<usize>,
1080+
skip: Range<usize>, // Space we copied-out-of, but haven't written-to yet.
10751081
len: usize,
10761082
}
10771083

src/tests.rs

Lines changed: 13 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -905,3 +905,16 @@ fn empty_macro() {
905905
fn zero_size_items() {
906906
SmallVec::<[(); 0]>::new().push(());
907907
}
908+
909+
#[test]
910+
fn test_insert_many_overflow() {
911+
let mut v: SmallVec<[u8; 1]> = SmallVec::new();
912+
v.push(123);
913+
914+
// Prepare an iterator with small lower bound
915+
let iter = (0u8..5).filter(|n| n % 2 == 0);
916+
assert_eq!(iter.size_hint().0, 0);
917+
918+
v.insert_many(0, iter);
919+
assert_eq!(&*v, &[0, 2, 4, 123]);
920+
}

0 commit comments

Comments
 (0)