@@ -220,6 +220,27 @@ class SmallVectorTemplateCommon
220
220
}
221
221
void assertSafeToEmplace () {}
222
222
223
+ // / Reserve enough space to add one element, and return the updated element
224
+ // / pointer in case it was a reference to the storage.
225
+ template <class U >
226
+ static const T *reserveForParamAndGetAddressImpl (U *This, const T &Elt,
227
+ size_t N) {
228
+ size_t NewSize = This->size () + N;
229
+ if (LLVM_LIKELY (NewSize <= This->capacity ()))
230
+ return &Elt;
231
+
232
+ bool ReferencesStorage = false ;
233
+ int64_t Index = -1 ;
234
+ if (!U::TakesParamByValue) {
235
+ if (LLVM_UNLIKELY (This->isReferenceToStorage (&Elt))) {
236
+ ReferencesStorage = true ;
237
+ Index = &Elt - This->begin ();
238
+ }
239
+ }
240
+ This->grow (NewSize);
241
+ return ReferencesStorage ? This->begin () + Index : &Elt;
242
+ }
243
+
223
244
public:
224
245
using size_type = size_t ;
225
246
using difference_type = ptrdiff_t ;
@@ -303,7 +324,12 @@ template <typename T, bool = (is_trivially_copy_constructible<T>::value) &&
303
324
(is_trivially_move_constructible<T>::value) &&
304
325
std::is_trivially_destructible<T>::value>
305
326
class SmallVectorTemplateBase : public SmallVectorTemplateCommon <T> {
327
+ friend class SmallVectorTemplateCommon <T>;
328
+
306
329
protected:
330
+ static constexpr bool TakesParamByValue = false ;
331
+ using ValueParamT = const T &;
332
+
307
333
SmallVectorTemplateBase (size_t Size ) : SmallVectorTemplateCommon<T>(Size ) {}
308
334
309
335
static void destroy_range (T *S, T *E) {
@@ -333,20 +359,32 @@ class SmallVectorTemplateBase : public SmallVectorTemplateCommon<T> {
333
359
// / element, or MinSize more elements if specified.
334
360
void grow (size_t MinSize = 0 );
335
361
362
+ // / Reserve enough space to add one element, and return the updated element
363
+ // / pointer in case it was a reference to the storage.
364
+ const T *reserveForParamAndGetAddress (const T &Elt, size_t N = 1 ) {
365
+ return this ->reserveForParamAndGetAddressImpl (this , Elt, N);
366
+ }
367
+
368
+ // / Reserve enough space to add one element, and return the updated element
369
+ // / pointer in case it was a reference to the storage.
370
+ T *reserveForParamAndGetAddress (T &Elt, size_t N = 1 ) {
371
+ return const_cast <T *>(
372
+ this ->reserveForParamAndGetAddressImpl (this , Elt, N));
373
+ }
374
+
375
+ static T &&forward_value_param(T &&V) { return std::move (V); }
376
+ static const T &forward_value_param (const T &V) { return V; }
377
+
336
378
public:
337
379
void push_back (const T &Elt) {
338
- this ->assertSafeToAdd (&Elt);
339
- if (LLVM_UNLIKELY (this ->size () >= this ->capacity ()))
340
- this ->grow ();
341
- ::new ((void *) this ->end ()) T (Elt);
380
+ const T *EltPtr = reserveForParamAndGetAddress (Elt);
381
+ ::new ((void *)this ->end ()) T (*EltPtr);
342
382
this ->set_size (this ->size () + 1 );
343
383
}
344
384
345
385
void push_back (T &&Elt) {
346
- this ->assertSafeToAdd (&Elt);
347
- if (LLVM_UNLIKELY (this ->size () >= this ->capacity ()))
348
- this ->grow ();
349
- ::new ((void *) this ->end ()) T (::std::move (Elt));
386
+ T *EltPtr = reserveForParamAndGetAddress (Elt);
387
+ ::new ((void *)this ->end ()) T (::std::move (*EltPtr));
350
388
this ->set_size (this ->size () + 1 );
351
389
}
352
390
@@ -396,7 +434,18 @@ void SmallVectorTemplateBase<T, TriviallyCopyable>::grow(size_t MinSize) {
396
434
// / skipping destruction.
397
435
template <typename T>
398
436
class SmallVectorTemplateBase <T, true > : public SmallVectorTemplateCommon<T> {
437
+ friend class SmallVectorTemplateCommon <T>;
438
+
399
439
protected:
440
+ // / True if it's cheap enough to take parameters by value. Doing so avoids
441
+ // / overhead related to mitigations for reference invalidation.
442
+ static constexpr bool TakesParamByValue = sizeof (T) <= 2 * sizeof (void *);
443
+
444
+ // / Either const T& or T, depending on whether it's cheap enough to take
445
+ // / parameters by value.
446
+ using ValueParamT =
447
+ typename std::conditional<TakesParamByValue, T, const T &>::type;
448
+
400
449
SmallVectorTemplateBase (size_t Size ) : SmallVectorTemplateCommon<T>(Size ) {}
401
450
402
451
// No need to do a destroy loop for POD's.
@@ -437,12 +486,26 @@ class SmallVectorTemplateBase<T, true> : public SmallVectorTemplateCommon<T> {
437
486
// / least one more element or MinSize if specified.
438
487
void grow (size_t MinSize = 0 ) { this ->grow_pod (MinSize, sizeof (T)); }
439
488
489
+ // / Reserve enough space to add one element, and return the updated element
490
+ // / pointer in case it was a reference to the storage.
491
+ const T *reserveForParamAndGetAddress (const T &Elt, size_t N = 1 ) {
492
+ return this ->reserveForParamAndGetAddressImpl (this , Elt, N);
493
+ }
494
+
495
+ // / Reserve enough space to add one element, and return the updated element
496
+ // / pointer in case it was a reference to the storage.
497
+ T *reserveForParamAndGetAddress (T &Elt, size_t N = 1 ) {
498
+ return const_cast <T *>(
499
+ this ->reserveForParamAndGetAddressImpl (this , Elt, N));
500
+ }
501
+
502
+ // / Copy \p V or return a reference, depending on \a ValueParamT.
503
+ static ValueParamT forward_value_param (ValueParamT V) { return V; }
504
+
440
505
public:
441
- void push_back (const T &Elt) {
442
- this ->assertSafeToAdd (&Elt);
443
- if (LLVM_UNLIKELY (this ->size () >= this ->capacity ()))
444
- this ->grow ();
445
- memcpy (reinterpret_cast <void *>(this ->end ()), &Elt, sizeof (T));
506
+ void push_back (ValueParamT Elt) {
507
+ const T *EltPtr = reserveForParamAndGetAddress (Elt);
508
+ memcpy (reinterpret_cast <void *>(this ->end ()), EltPtr, sizeof (T));
446
509
this ->set_size (this ->size () + 1 );
447
510
}
448
511
@@ -462,6 +525,9 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
462
525
using size_type = typename SuperClass::size_type;
463
526
464
527
protected:
528
+ using SmallVectorTemplateBase<T>::TakesParamByValue;
529
+ using ValueParamT = typename SuperClass::ValueParamT;
530
+
465
531
// Default ctor - Initialize to empty.
466
532
explicit SmallVectorImpl (unsigned N)
467
533
: SmallVectorTemplateBase<T>(N) {}
@@ -502,7 +568,7 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
502
568
// / Like resize, but \ref T is POD, the new values won't be initialized.
503
569
void resize_for_overwrite (size_type N) { resizeImpl<true >(N); }
504
570
505
- void resize (size_type N, const T & NV) {
571
+ void resize (size_type N, ValueParamT NV) {
506
572
if (N == this ->size ())
507
573
return ;
508
574
@@ -511,11 +577,8 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
511
577
return ;
512
578
}
513
579
514
- this ->assertSafeToReferenceAfterResize (&NV, N);
515
- if (this ->capacity () < N)
516
- this ->grow (N);
517
- std::uninitialized_fill (this ->end (), this ->begin () + N, NV);
518
- this ->set_size (N);
580
+ // N > this->size(). Defer to append.
581
+ this ->append (N - this ->size (), NV);
519
582
}
520
583
521
584
void reserve (size_type N) {
@@ -551,12 +614,9 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
551
614
}
552
615
553
616
// / Append \p NumInputs copies of \p Elt to the end.
554
- void append (size_type NumInputs, const T &Elt) {
555
- this ->assertSafeToAdd (&Elt, NumInputs);
556
- if (NumInputs > this ->capacity () - this ->size ())
557
- this ->grow (this ->size ()+NumInputs);
558
-
559
- std::uninitialized_fill_n (this ->end (), NumInputs, Elt);
617
+ void append (size_type NumInputs, ValueParamT Elt) {
618
+ const T *EltPtr = this ->reserveForParamAndGetAddress (Elt, NumInputs);
619
+ std::uninitialized_fill_n (this ->end (), NumInputs, *EltPtr);
560
620
this ->set_size (this ->size () + NumInputs);
561
621
}
562
622
@@ -622,31 +682,35 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
622
682
623
683
private:
624
684
template <class ArgType > iterator insert_one_impl (iterator I, ArgType &&Elt) {
685
+ // Callers ensure that ArgType is derived from T.
686
+ static_assert (
687
+ std::is_same<std::remove_const_t <std::remove_reference_t <ArgType>>,
688
+ T>::value,
689
+ " ArgType must be derived from T!" );
690
+
625
691
if (I == this ->end ()) { // Important special case for empty vector.
626
692
this ->push_back (::std::forward<ArgType>(Elt));
627
693
return this ->end ()-1 ;
628
694
}
629
695
630
696
assert (this ->isReferenceToStorage (I) && " Insertion iterator is out of bounds." );
631
697
632
- // Check that adding an element won't invalidate Elt.
633
- this ->assertSafeToAdd (&Elt);
634
-
635
- if (this ->size () >= this ->capacity ()) {
636
- size_t EltNo = I-this ->begin ();
637
- this ->grow ();
638
- I = this ->begin ()+EltNo;
639
- }
698
+ // Grow if necessary.
699
+ size_t Index = I - this ->begin ();
700
+ std::remove_reference_t <ArgType> *EltPtr =
701
+ this ->reserveForParamAndGetAddress (Elt);
702
+ I = this ->begin () + Index;
640
703
641
704
::new ((void *) this ->end ()) T (::std::move (this ->back ()));
642
705
// Push everything else over.
643
706
std::move_backward (I, this ->end ()-1 , this ->end ());
644
707
this ->set_size (this ->size () + 1 );
645
708
646
709
// If we just moved the element we're inserting, be sure to update
647
- // the reference.
648
- std::remove_reference_t <ArgType> *EltPtr = &Elt;
649
- if (this ->isReferenceToRange (EltPtr, I, this ->end ()))
710
+ // the reference (never happens if TakesParamByValue).
711
+ static_assert (!TakesParamByValue || std::is_same<ArgType, T>::value,
712
+ " ArgType must be 'T' when taking by value!" );
713
+ if (!TakesParamByValue && this ->isReferenceToRange (EltPtr, I, this ->end ()))
650
714
++EltPtr;
651
715
652
716
*I = ::std::forward<ArgType>(*EltPtr);
@@ -655,12 +719,14 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
655
719
656
720
public:
657
721
iterator insert (iterator I, T &&Elt) {
658
- return insert_one_impl (I, std::move (Elt));
722
+ return insert_one_impl (I, this -> forward_value_param ( std::move (Elt) ));
659
723
}
660
724
661
- iterator insert (iterator I, const T &Elt) { return insert_one_impl (I, Elt); }
725
+ iterator insert (iterator I, const T &Elt) {
726
+ return insert_one_impl (I, this ->forward_value_param (Elt));
727
+ }
662
728
663
- iterator insert (iterator I, size_type NumToInsert, const T & Elt) {
729
+ iterator insert (iterator I, size_type NumToInsert, ValueParamT Elt) {
664
730
// Convert iterator to elt# to avoid invalidating iterator when we reserve()
665
731
size_t InsertElt = I - this ->begin ();
666
732
@@ -671,11 +737,9 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
671
737
672
738
assert (this ->isReferenceToStorage (I) && " Insertion iterator is out of bounds." );
673
739
674
- // Check that adding NumToInsert elements won't invalidate Elt.
675
- this ->assertSafeToAdd (&Elt, NumToInsert);
676
-
677
- // Ensure there is enough space.
678
- reserve (this ->size () + NumToInsert);
740
+ // Ensure there is enough space, and get the (maybe updated) address of
741
+ // Elt.
742
+ const T *EltPtr = this ->reserveForParamAndGetAddress (Elt, NumToInsert);
679
743
680
744
// Uninvalidate the iterator.
681
745
I = this ->begin ()+InsertElt;
@@ -692,7 +756,12 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
692
756
// Copy the existing elements that get replaced.
693
757
std::move_backward (I, OldEnd-NumToInsert, OldEnd);
694
758
695
- std::fill_n (I, NumToInsert, Elt);
759
+ // If we just moved the element we're inserting, be sure to update
760
+ // the reference (never happens if TakesParamByValue).
761
+ if (!TakesParamByValue && I <= EltPtr && EltPtr < this ->end ())
762
+ EltPtr += NumToInsert;
763
+
764
+ std::fill_n (I, NumToInsert, *EltPtr);
696
765
return I;
697
766
}
698
767
@@ -705,11 +774,16 @@ class SmallVectorImpl : public SmallVectorTemplateBase<T> {
705
774
size_t NumOverwritten = OldEnd-I;
706
775
this ->uninitialized_move (I, OldEnd, this ->end ()-NumOverwritten);
707
776
777
+ // If we just moved the element we're inserting, be sure to update
778
+ // the reference (never happens if TakesParamByValue).
779
+ if (!TakesParamByValue && I <= EltPtr && EltPtr < this ->end ())
780
+ EltPtr += NumToInsert;
781
+
708
782
// Replace the overwritten part.
709
- std::fill_n (I, NumOverwritten, Elt );
783
+ std::fill_n (I, NumOverwritten, *EltPtr );
710
784
711
785
// Insert the non-overwritten middle part.
712
- std::uninitialized_fill_n (OldEnd, NumToInsert- NumOverwritten, Elt );
786
+ std::uninitialized_fill_n (OldEnd, NumToInsert - NumOverwritten, *EltPtr );
713
787
return I;
714
788
}
715
789
0 commit comments