about summary refs log tree commit diff
path: root/src/liballoc
diff options
context:
space:
mode:
authorRalf Jung <post@ralfj.de>2020-03-30 13:31:16 +0200
committerRalf Jung <post@ralfj.de>2020-03-30 13:34:03 +0200
commit5bbaac357dd85092ed0fb822947df7a4d60c1db9 (patch)
tree12ab35b2de11f357ee0cef3711e926fd25237660 /src/liballoc
parent8f479e362fbfcb31e83396ef850ab5219a32821e (diff)
downloadrust-5bbaac357dd85092ed0fb822947df7a4d60c1db9.tar.gz
rust-5bbaac357dd85092ed0fb822947df7a4d60c1db9.zip
fix and test aliasing in swap_remove
Diffstat (limited to 'src/liballoc')
-rw-r--r--src/liballoc/tests/vec.rs5
-rw-r--r--src/liballoc/vec.rs7
2 files changed, 9 insertions, 3 deletions
diff --git a/src/liballoc/tests/vec.rs b/src/liballoc/tests/vec.rs
index a90bc58cbfd..6321e7154e7 100644
--- a/src/liballoc/tests/vec.rs
+++ b/src/liballoc/tests/vec.rs
@@ -1378,6 +1378,11 @@ fn test_stable_pointers() {
     v.remove(1);
     v.pop().unwrap();
     assert_eq!(*v0, 13);
+    v.push(1);
+    v.swap_remove(1);
+    assert_eq!(v.len(), 2);
+    v.swap_remove(1); // swap_remove the last element
+    assert_eq!(*v0, 13);
 
     // Appending
     v.append(&mut vec![27, 19]);
diff --git a/src/liballoc/vec.rs b/src/liballoc/vec.rs
index dc2a246f817..c600a6b649f 100644
--- a/src/liballoc/vec.rs
+++ b/src/liballoc/vec.rs
@@ -963,12 +963,13 @@ impl<T> Vec<T> {
     #[inline]
     #[stable(feature = "rust1", since = "1.0.0")]
     pub fn swap_remove(&mut self, index: usize) -> T {
+        assert!(index < self.len);
         unsafe {
             // We replace self[index] with the last element. Note that if the
-            // bounds check on hole succeeds there must be a last element (which
+            // bounds check above succeeds there must be a last element (which
             // can be self[index] itself).
-            let hole: *mut T = &mut self[index];
-            let last = ptr::read(self.get_unchecked(self.len - 1));
+            let last = ptr::read(self.as_ptr().add(self.len - 1));
+            let hole: *mut T = self.as_mut_ptr().add(index);
             self.len -= 1;
             ptr::replace(hole, last)
         }