@@ -60,13 +60,13 @@ use std::hash::{Hash, Hasher};
60
60
use std:: hint:: unreachable_unchecked;
61
61
#[ cfg( feature = "std" ) ]
62
62
use std:: io;
63
- use std:: iter:: { repeat, FromIterator , IntoIterator } ;
63
+ use std:: iter:: { repeat, FromIterator , FusedIterator , IntoIterator } ;
64
64
#[ cfg( feature = "serde" ) ]
65
65
use std:: marker:: PhantomData ;
66
66
use std:: mem;
67
67
use std:: mem:: MaybeUninit ;
68
- use std:: ops;
69
- use std:: ptr;
68
+ use std:: ops:: { self , Bound , RangeBounds } ;
69
+ use std:: ptr:: { self , NonNull } ;
70
70
use std:: slice:: { self , SliceIndex } ;
71
71
72
72
/// Creates a [`SmallVec`] containing the arguments.
@@ -182,15 +182,30 @@ unsafe fn deallocate<T>(ptr: *mut T, capacity: usize) {
182
182
/// Returned from [`SmallVec::drain`][1].
183
183
///
184
184
/// [1]: struct.SmallVec.html#method.drain
185
- pub struct Drain < ' a , T : ' a > {
186
- iter : slice:: IterMut < ' a , T > ,
185
+ pub struct Drain < ' a , T : ' a + Array > {
186
+ tail_start : usize ,
187
+ tail_len : usize ,
188
+ iter : slice:: Iter < ' a , T :: Item > ,
189
+ vec : NonNull < SmallVec < T > > ,
187
190
}
188
191
189
- impl < ' a , T : ' a > Iterator for Drain < ' a , T > {
190
- type Item = T ;
192
+ impl < ' a , T : ' a + Array > fmt:: Debug for Drain < ' a , T >
193
+ where
194
+ T :: Item : fmt:: Debug ,
195
+ {
196
+ fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
197
+ f. debug_tuple ( "Drain" ) . field ( & self . iter . as_slice ( ) ) . finish ( )
198
+ }
199
+ }
200
+
201
+ unsafe impl < ' a , T : Sync + Array > Sync for Drain < ' a , T > { }
202
+ unsafe impl < ' a , T : Send + Array > Send for Drain < ' a , T > { }
203
+
204
+ impl < ' a , T : ' a + Array > Iterator for Drain < ' a , T > {
205
+ type Item = T :: Item ;
191
206
192
207
#[ inline]
193
- fn next ( & mut self ) -> Option < T > {
208
+ fn next ( & mut self ) -> Option < T :: Item > {
194
209
self . iter
195
210
. next ( )
196
211
. map ( |reference| unsafe { ptr:: read ( reference) } )
@@ -202,21 +217,43 @@ impl<'a, T: 'a> Iterator for Drain<'a, T> {
202
217
}
203
218
}
204
219
205
- impl < ' a , T : ' a > DoubleEndedIterator for Drain < ' a , T > {
220
+ impl < ' a , T : ' a + Array > DoubleEndedIterator for Drain < ' a , T > {
206
221
#[ inline]
207
- fn next_back ( & mut self ) -> Option < T > {
222
+ fn next_back ( & mut self ) -> Option < T :: Item > {
208
223
self . iter
209
224
. next_back ( )
210
225
. map ( |reference| unsafe { ptr:: read ( reference) } )
211
226
}
212
227
}
213
228
214
- impl < ' a , T > ExactSizeIterator for Drain < ' a , T > { }
229
+ impl < ' a , T : Array > ExactSizeIterator for Drain < ' a , T > {
230
+ #[ inline]
231
+ fn len ( & self ) -> usize {
232
+ self . iter . len ( )
233
+ }
234
+ }
215
235
216
- impl < ' a , T : ' a > Drop for Drain < ' a , T > {
236
+ impl < ' a , T : Array > FusedIterator for Drain < ' a , T > { }
237
+
238
+ impl < ' a , T : ' a + Array > Drop for Drain < ' a , T > {
217
239
fn drop ( & mut self ) {
218
- // Destroy the remaining elements.
219
- for _ in self . by_ref ( ) { }
240
+ self . for_each ( drop) ;
241
+
242
+ if self . tail_len > 0 {
243
+ unsafe {
244
+ let source_vec = self . vec . as_mut ( ) ;
245
+
246
+ // memmove back untouched tail, update to new length
247
+ let start = source_vec. len ( ) ;
248
+ let tail = self . tail_start ;
249
+ if tail != start {
250
+ let src = source_vec. as_ptr ( ) . add ( tail) ;
251
+ let dst = source_vec. as_mut_ptr ( ) . add ( start) ;
252
+ ptr:: copy ( src, dst, self . tail_len ) ;
253
+ }
254
+ source_vec. set_len ( start + self . tail_len ) ;
255
+ }
256
+ }
220
257
}
221
258
}
222
259
@@ -548,18 +585,50 @@ impl<A: Array> SmallVec<A> {
548
585
self . capacity > A :: size ( )
549
586
}
550
587
551
- /// Empty the vector and return an iterator over its former contents.
552
- pub fn drain ( & mut self ) -> Drain < A :: Item > {
553
- unsafe {
554
- let ptr = self . as_mut_ptr ( ) ;
588
+ /// Creates a draining iterator that removes the specified range in the vector
589
+ /// and yields the removed items.
590
+ ///
591
+ /// Note 1: The element range is removed even if the iterator is only
592
+ /// partially consumed or not consumed at all.
593
+ ///
594
+ /// Note 2: It is unspecified how many elements are removed from the vector
595
+ /// if the `Drain` value is leaked.
596
+ ///
597
+ /// # Panics
598
+ ///
599
+ /// Panics if the starting point is greater than the end point or if
600
+ /// the end point is greater than the length of the vector.
601
+ pub fn drain < R > ( & mut self , range : R ) -> Drain < A >
602
+ where
603
+ R : RangeBounds < usize > ,
604
+ {
605
+ use Bound :: * ;
555
606
556
- let current_len = self . len ( ) ;
557
- self . set_len ( 0 ) ;
607
+ let len = self . len ( ) ;
608
+ let start = match range. start_bound ( ) {
609
+ Included ( & n) => n,
610
+ Excluded ( & n) => n + 1 ,
611
+ Unbounded => 0 ,
612
+ } ;
613
+ let end = match range. end_bound ( ) {
614
+ Included ( & n) => n + 1 ,
615
+ Excluded ( & n) => n,
616
+ Unbounded => len,
617
+ } ;
618
+
619
+ assert ! ( start <= end) ;
620
+ assert ! ( end <= len) ;
621
+
622
+ unsafe {
623
+ self . set_len ( start) ;
558
624
559
- let slice = slice:: from_raw_parts_mut ( ptr , current_len ) ;
625
+ let range_slice = slice:: from_raw_parts_mut ( self . as_mut_ptr ( ) . add ( start ) , end - start ) ;
560
626
561
627
Drain {
562
- iter : slice. iter_mut ( ) ,
628
+ tail_start : end,
629
+ tail_len : len - end,
630
+ iter : range_slice. iter ( ) ,
631
+ vec : NonNull :: from ( self ) ,
563
632
}
564
633
}
565
634
}
@@ -1697,26 +1766,36 @@ mod tests {
1697
1766
fn drain ( ) {
1698
1767
let mut v: SmallVec < [ u8 ; 2 ] > = SmallVec :: new ( ) ;
1699
1768
v. push ( 3 ) ;
1700
- assert_eq ! ( v. drain( ) . collect:: <Vec <_>>( ) , & [ 3 ] ) ;
1769
+ assert_eq ! ( v. drain( .. ) . collect:: <Vec <_>>( ) , & [ 3 ] ) ;
1701
1770
1702
1771
// spilling the vec
1703
1772
v. push ( 3 ) ;
1704
1773
v. push ( 4 ) ;
1705
1774
v. push ( 5 ) ;
1706
- assert_eq ! ( v. drain( ) . collect:: <Vec <_>>( ) , & [ 3 , 4 , 5 ] ) ;
1775
+ let old_capacity = v. capacity ( ) ;
1776
+ assert_eq ! ( v. drain( 1 ..) . collect:: <Vec <_>>( ) , & [ 4 , 5 ] ) ;
1777
+ // drain should not change the capacity
1778
+ assert_eq ! ( v. capacity( ) , old_capacity) ;
1707
1779
}
1708
1780
1709
1781
#[ test]
1710
1782
fn drain_rev ( ) {
1711
1783
let mut v: SmallVec < [ u8 ; 2 ] > = SmallVec :: new ( ) ;
1712
1784
v. push ( 3 ) ;
1713
- assert_eq ! ( v. drain( ) . rev( ) . collect:: <Vec <_>>( ) , & [ 3 ] ) ;
1785
+ assert_eq ! ( v. drain( .. ) . rev( ) . collect:: <Vec <_>>( ) , & [ 3 ] ) ;
1714
1786
1715
1787
// spilling the vec
1716
1788
v. push ( 3 ) ;
1717
1789
v. push ( 4 ) ;
1718
1790
v. push ( 5 ) ;
1719
- assert_eq ! ( v. drain( ) . rev( ) . collect:: <Vec <_>>( ) , & [ 5 , 4 , 3 ] ) ;
1791
+ assert_eq ! ( v. drain( ..) . rev( ) . collect:: <Vec <_>>( ) , & [ 5 , 4 , 3 ] ) ;
1792
+ }
1793
+
1794
+ #[ test]
1795
+ fn drain_forget ( ) {
1796
+ let mut v: SmallVec < [ u8 ; 1 ] > = smallvec ! [ 0 , 1 , 2 , 3 , 4 , 5 , 6 , 7 ] ;
1797
+ std:: mem:: forget ( v. drain ( 2 ..5 ) ) ;
1798
+ assert_eq ! ( v. len( ) , 2 ) ;
1720
1799
}
1721
1800
1722
1801
#[ test]
@@ -2157,7 +2236,8 @@ mod tests {
2157
2236
fn test_exact_size_iterator ( ) {
2158
2237
let mut vec = SmallVec :: < [ u32 ; 2 ] > :: from ( & [ 1 , 2 , 3 ] [ ..] ) ;
2159
2238
assert_eq ! ( vec. clone( ) . into_iter( ) . len( ) , 3 ) ;
2160
- assert_eq ! ( vec. drain( ) . len( ) , 3 ) ;
2239
+ assert_eq ! ( vec. drain( ..2 ) . len( ) , 2 ) ;
2240
+ assert_eq ! ( vec. into_iter( ) . len( ) , 1 ) ;
2161
2241
}
2162
2242
2163
2243
#[ test]
0 commit comments