• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2015, ARM Limited
2 // All rights reserved.
3 //
4 // Redistribution and use in source and binary forms, with or without
5 // modification, are permitted provided that the following conditions are met:
6 //
7 //   * Redistributions of source code must retain the above copyright notice,
8 //     this list of conditions and the following disclaimer.
9 //   * Redistributions in binary form must reproduce the above copyright notice,
10 //     this list of conditions and the following disclaimer in the documentation
11 //     and/or other materials provided with the distribution.
12 //   * Neither the name of ARM Limited nor the names of its contributors may be
13 //     used to endorse or promote products derived from this software without
14 //     specific prior written permission.
15 //
16 // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS CONTRIBUTORS "AS IS" AND
17 // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19 // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
20 // FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
21 // DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
22 // SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
23 // CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
24 // OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
25 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 
27 #ifndef VIXL_INVALSET_H_
28 #define VIXL_INVALSET_H_
29 
30 #include <string.h>
31 
32 #include <algorithm>
33 #include <vector>
34 
35 #include "vixl/globals.h"
36 
37 namespace vixl {
38 
39 // We define a custom data structure template and its iterator as `std`
40 // containers do not fit the performance requirements for some of our use cases.
41 //
42 // The structure behaves like an iterable unordered set with special properties
43 // and restrictions. "InvalSet" stands for "Invalidatable Set".
44 //
45 // Restrictions and requirements:
46 // - Adding an element already present in the set is illegal. In debug mode,
47 //   this is checked at insertion time.
48 // - The templated class `ElementType` must provide comparison operators so that
49 //   `std::sort()` can be used.
50 // - A key must be available to represent invalid elements.
51 // - Elements with an invalid key must compare higher or equal to any other
52 //   element.
53 //
54 // Use cases and performance considerations:
55 // Our use cases present two specificities that allow us to design this
56 // structure to provide fast insertion *and* fast search and deletion
57 // operations:
58 // - Elements are (generally) inserted in order (sorted according to their key).
59 // - A key is available to mark elements as invalid (deleted).
60 // The backing `std::vector` allows for fast insertions. When
61 // searching for an element we ensure the elements are sorted (this is generally
62 // the case) and perform a binary search. When deleting an element we do not
63 // free the associated memory immediately. Instead, an element to be deleted is
64 // marked with the 'invalid' key. Other methods of the container take care of
65 // ignoring entries marked as invalid.
66 // To avoid the overhead of the `std::vector` container when only few entries
67 // are used, a number of elements are preallocated.
68 
69 // 'ElementType' and 'KeyType' are respectively the types of the elements and
70 // their key.  The structure only reclaims memory when safe to do so, if the
71 // number of elements that can be reclaimed is greater than `RECLAIM_FROM` and
72 // greater than `<total number of elements> / RECLAIM_FACTOR.
73 #define TEMPLATE_INVALSET_P_DECL                                               \
74   class ElementType,                                                           \
75   unsigned N_PREALLOCATED_ELEMENTS,                                            \
76   class KeyType,                                                               \
77   KeyType INVALID_KEY,                                                         \
78   size_t RECLAIM_FROM,                                                         \
79   unsigned RECLAIM_FACTOR
80 
81 #define TEMPLATE_INVALSET_P_DEF                                                \
82 ElementType, N_PREALLOCATED_ELEMENTS,                                          \
83 KeyType, INVALID_KEY, RECLAIM_FROM, RECLAIM_FACTOR
84 
85 template<class S> class InvalSetIterator;  // Forward declaration.
86 
87 template<TEMPLATE_INVALSET_P_DECL> class InvalSet {
88  public:
89   InvalSet();
90   ~InvalSet();
91 
92   static const size_t kNPreallocatedElements = N_PREALLOCATED_ELEMENTS;
93   static const KeyType kInvalidKey = INVALID_KEY;
94 
95   // It is illegal to insert an element already present in the set.
96   void insert(const ElementType& element);
97 
98   // Looks for the specified element in the set and - if found - deletes it.
99   void erase(const ElementType& element);
100 
101   // This indicates the number of (valid) elements stored in this set.
102   size_t size() const;
103 
104   // Returns true if no elements are stored in the set.
105   // Note that this does not mean the the backing storage is empty: it can still
106   // contain invalid elements.
107   bool empty() const;
108 
109   void clear();
110 
111   const ElementType min_element();
112 
113   // This returns the key of the minimum element in the set.
114   KeyType min_element_key();
115 
116   static bool IsValid(const ElementType& element);
117   static KeyType Key(const ElementType& element);
118   static void SetKey(ElementType* element, KeyType key);
119 
120  protected:
121   // Returns a pointer to the element in vector_ if it was found, or NULL
122   // otherwise.
123   ElementType* Search(const ElementType& element);
124 
125   // The argument *must* point to an element stored in *this* set.
126   // This function is not allowed to move elements in the backing vector
127   // storage.
128   void EraseInternal(ElementType* element);
129 
130   // The elements in the range searched must be sorted.
131   ElementType* BinarySearch(const ElementType& element,
132                             ElementType* start,
133                             ElementType* end) const;
134 
135   // Sort the elements.
136   enum SortType {
137     // The 'hard' version guarantees that invalid elements are moved to the end
138     // of the container.
139     kHardSort,
140     // The 'soft' version only guarantees that the elements will be sorted.
141     // Invalid elements may still be present anywhere in the set.
142     kSoftSort
143   };
144   void Sort(SortType sort_type);
145 
146   // Delete the elements that have an invalid key. The complexity is linear
147   // with the size of the vector.
148   void Clean();
149 
150   const ElementType Front() const;
151   const ElementType Back() const;
152 
153   // Delete invalid trailing elements and return the last valid element in the
154   // set.
155   const ElementType CleanBack();
156 
157   // Returns a pointer to the start or end of the backing storage.
158   const ElementType* StorageBegin() const;
159   const ElementType* StorageEnd() const;
160   ElementType* StorageBegin();
161   ElementType* StorageEnd();
162 
163   // Returns the index of the element within the backing storage. The element
164   // must belong to the backing storage.
165   size_t ElementIndex(const ElementType* element) const;
166 
167   // Returns the element at the specified index in the backing storage.
168   const ElementType* ElementAt(size_t index) const;
169   ElementType* ElementAt(size_t index);
170 
171   static const ElementType* FirstValidElement(const ElementType* from,
172                                               const ElementType* end);
173 
174   void CacheMinElement();
175   const ElementType CachedMinElement() const;
176 
177   bool ShouldReclaimMemory() const;
178   void ReclaimMemory();
179 
IsUsingVector()180   bool IsUsingVector() const { return vector_ != NULL; }
set_sorted(bool sorted)181   void set_sorted(bool sorted) { sorted_ = sorted; }
182 
183   // We cache some data commonly required by users to improve performance.
184   // We cannot cache pointers to elements as we do not control the backing
185   // storage.
186   bool valid_cached_min_;
187   size_t cached_min_index_;  // Valid iff `valid_cached_min_` is true.
188   KeyType cached_min_key_;         // Valid iff `valid_cached_min_` is true.
189 
190   // Indicates whether the elements are sorted.
191   bool sorted_;
192 
193   // This represents the number of (valid) elements in this set.
194   size_t size_;
195 
196   // The backing storage is either the array of preallocated elements or the
197   // vector. The structure starts by using the preallocated elements, and
198   // transitions (permanently) to using the vector once more than
199   // kNPreallocatedElements are used.
200   // Elements are only invalidated when using the vector. The preallocated
201   // storage always only contains valid elements.
202   ElementType preallocated_[kNPreallocatedElements];
203   std::vector<ElementType>* vector_;
204 
205 #ifdef VIXL_DEBUG
206   // Iterators acquire and release this monitor. While a set is acquired,
207   // certain operations are illegal to ensure that the iterator will
208   // correctly iterate over the elements in the set.
209   int monitor_;
monitor()210   int monitor() const { return monitor_; }
Acquire()211   void Acquire() { monitor_++; }
Release()212   void Release() {
213     monitor_--;
214     VIXL_ASSERT(monitor_ >= 0);
215   }
216 #endif
217 
218   friend class InvalSetIterator<InvalSet<TEMPLATE_INVALSET_P_DEF> >;
219   typedef ElementType _ElementType;
220   typedef KeyType _KeyType;
221 };
222 
223 
224 template<class S> class InvalSetIterator {
225  private:
226   // Redefine types to mirror the associated set types.
227   typedef typename S::_ElementType ElementType;
228   typedef typename S::_KeyType KeyType;
229 
230  public:
231   explicit InvalSetIterator(S* inval_set);
232   ~InvalSetIterator();
233 
234   ElementType* Current() const;
235   void Advance();
236   bool Done() const;
237 
238   // Mark this iterator as 'done'.
239   void Finish();
240 
241   // Delete the current element and advance the iterator to point to the next
242   // element.
243   void DeleteCurrentAndAdvance();
244 
245   static bool IsValid(const ElementType& element);
246   static KeyType Key(const ElementType& element);
247 
248  protected:
249   void MoveToValidElement();
250 
251   // Indicates if the iterator is looking at the vector or at the preallocated
252   // elements.
253   const bool using_vector_;
254   // Used when looking at the preallocated elements, or in debug mode when using
255   // the vector to track how many times the iterator has advanced.
256   size_t index_;
257   typename std::vector<ElementType>::iterator iterator_;
258   S* inval_set_;
259 };
260 
261 
262 template<TEMPLATE_INVALSET_P_DECL>
InvalSet()263 InvalSet<TEMPLATE_INVALSET_P_DEF>::InvalSet()
264   : valid_cached_min_(false),
265     sorted_(true), size_(0), vector_(NULL) {
266 #ifdef VIXL_DEBUG
267   monitor_ = 0;
268 #endif
269 }
270 
271 
272 template<TEMPLATE_INVALSET_P_DECL>
~InvalSet()273 InvalSet<TEMPLATE_INVALSET_P_DEF>::~InvalSet() {
274   VIXL_ASSERT(monitor_ == 0);
275   delete vector_;
276 }
277 
278 
279 template<TEMPLATE_INVALSET_P_DECL>
insert(const ElementType & element)280 void InvalSet<TEMPLATE_INVALSET_P_DEF>::insert(const ElementType& element) {
281   VIXL_ASSERT(monitor() == 0);
282   VIXL_ASSERT(IsValid(element));
283   VIXL_ASSERT(Search(element) == NULL);
284   set_sorted(empty() || (sorted_ && (element > CleanBack())));
285   if (IsUsingVector()) {
286     vector_->push_back(element);
287   } else {
288     if (size_ < kNPreallocatedElements) {
289       preallocated_[size_] = element;
290     } else {
291       // Transition to using the vector.
292       vector_ = new std::vector<ElementType>(preallocated_,
293                                              preallocated_ + size_);
294       vector_->push_back(element);
295     }
296   }
297   size_++;
298 
299   if (valid_cached_min_ && (element < min_element())) {
300     cached_min_index_ = IsUsingVector() ? vector_->size() - 1 : size_ - 1;
301     cached_min_key_ = Key(element);
302     valid_cached_min_ = true;
303   }
304 
305   if (ShouldReclaimMemory()) {
306     ReclaimMemory();
307   }
308 }
309 
310 
311 template<TEMPLATE_INVALSET_P_DECL>
erase(const ElementType & element)312 void InvalSet<TEMPLATE_INVALSET_P_DEF>::erase(const ElementType& element) {
313   VIXL_ASSERT(monitor() == 0);
314   VIXL_ASSERT(IsValid(element));
315   ElementType* local_element = Search(element);
316   if (local_element != NULL) {
317     EraseInternal(local_element);
318   }
319 }
320 
321 
322 template<TEMPLATE_INVALSET_P_DECL>
Search(const ElementType & element)323 ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::Search(
324     const ElementType& element) {
325   VIXL_ASSERT(monitor() == 0);
326   if (empty()) {
327     return NULL;
328   }
329   if (ShouldReclaimMemory()) {
330     ReclaimMemory();
331   }
332   if (!sorted_) {
333     Sort(kHardSort);
334   }
335   if (!valid_cached_min_) {
336     CacheMinElement();
337   }
338   return BinarySearch(element, ElementAt(cached_min_index_), StorageEnd());
339 }
340 
341 
342 template<TEMPLATE_INVALSET_P_DECL>
size()343 size_t InvalSet<TEMPLATE_INVALSET_P_DEF>::size() const {
344   return size_;
345 }
346 
347 
348 template<TEMPLATE_INVALSET_P_DECL>
empty()349 bool InvalSet<TEMPLATE_INVALSET_P_DEF>::empty() const {
350   return size_ == 0;
351 }
352 
353 
354 template<TEMPLATE_INVALSET_P_DECL>
clear()355 void InvalSet<TEMPLATE_INVALSET_P_DEF>::clear() {
356   VIXL_ASSERT(monitor() == 0);
357   size_ = 0;
358   if (IsUsingVector()) {
359     vector_->clear();
360   }
361   set_sorted(true);
362   valid_cached_min_ = false;
363 }
364 
365 
366 template<TEMPLATE_INVALSET_P_DECL>
min_element()367 const ElementType InvalSet<TEMPLATE_INVALSET_P_DEF>::min_element() {
368   VIXL_ASSERT(monitor() == 0);
369   VIXL_ASSERT(!empty());
370   CacheMinElement();
371   return *ElementAt(cached_min_index_);
372 }
373 
374 
375 template<TEMPLATE_INVALSET_P_DECL>
min_element_key()376 KeyType InvalSet<TEMPLATE_INVALSET_P_DEF>::min_element_key() {
377   VIXL_ASSERT(monitor() == 0);
378   if (valid_cached_min_) {
379     return cached_min_key_;
380   } else {
381     return Key(min_element());
382   }
383 }
384 
385 
386 template<TEMPLATE_INVALSET_P_DECL>
IsValid(const ElementType & element)387 bool InvalSet<TEMPLATE_INVALSET_P_DEF>::IsValid(const ElementType& element) {
388   return Key(element) != kInvalidKey;
389 }
390 
391 
392 template<TEMPLATE_INVALSET_P_DECL>
EraseInternal(ElementType * element)393 void InvalSet<TEMPLATE_INVALSET_P_DEF>::EraseInternal(ElementType* element) {
394   // Note that this function must be safe even while an iterator has acquired
395   // this set.
396   VIXL_ASSERT(element != NULL);
397   size_t deleted_index = ElementIndex(element);
398   if (IsUsingVector()) {
399     VIXL_ASSERT((&(vector_->front()) <= element) &&
400                 (element <= &(vector_->back())));
401     SetKey(element, kInvalidKey);
402   } else {
403     VIXL_ASSERT((preallocated_ <= element) &&
404                 (element < (preallocated_ + kNPreallocatedElements)));
405     ElementType* end = preallocated_ + kNPreallocatedElements;
406     size_t copy_size = sizeof(*element) * (end - element - 1);
407     memmove(element, element + 1, copy_size);
408   }
409   size_--;
410 
411   if (valid_cached_min_ &&
412       (deleted_index == cached_min_index_)) {
413     if (sorted_ && !empty()) {
414       const ElementType* min = FirstValidElement(element, StorageEnd());
415       cached_min_index_ = ElementIndex(min);
416       cached_min_key_ = Key(*min);
417       valid_cached_min_ = true;
418     } else {
419       valid_cached_min_ = false;
420     }
421   }
422 }
423 
424 
425 template<TEMPLATE_INVALSET_P_DECL>
BinarySearch(const ElementType & element,ElementType * start,ElementType * end)426 ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::BinarySearch(
427     const ElementType& element, ElementType* start, ElementType* end) const {
428   if (start == end) {
429     return NULL;
430   }
431   VIXL_ASSERT(sorted_);
432   VIXL_ASSERT(start < end);
433   VIXL_ASSERT(!empty());
434 
435   // Perform a binary search through the elements while ignoring invalid
436   // elements.
437   ElementType* elements = start;
438   size_t low = 0;
439   size_t high = (end - start) - 1;
440   while (low < high) {
441     // Find valid bounds.
442     while (!IsValid(elements[low]) && (low < high)) ++low;
443     while (!IsValid(elements[high]) && (low < high)) --high;
444     VIXL_ASSERT(low <= high);
445     // Avoid overflow when computing the middle index.
446     size_t middle = low / 2 + high / 2 + (low & high & 1);
447     if ((middle == low) || (middle == high)) {
448       break;
449     }
450     while (!IsValid(elements[middle]) && (middle < high - 1)) ++middle;
451     while (!IsValid(elements[middle]) && (low + 1 < middle)) --middle;
452     if (!IsValid(elements[middle])) {
453       break;
454     }
455     if (elements[middle] < element) {
456       low = middle;
457     } else {
458       high = middle;
459     }
460   }
461 
462   if (elements[low] == element) return &elements[low];
463   if (elements[high] == element) return &elements[high];
464   return NULL;
465 }
466 
467 
468 template<TEMPLATE_INVALSET_P_DECL>
Sort(SortType sort_type)469 void InvalSet<TEMPLATE_INVALSET_P_DEF>::Sort(SortType sort_type) {
470   VIXL_ASSERT(monitor() == 0);
471   if (sort_type == kSoftSort) {
472     if (sorted_) {
473       return;
474     }
475   }
476   if (empty()) {
477     return;
478   }
479 
480   Clean();
481   std::sort(StorageBegin(), StorageEnd());
482 
483   set_sorted(true);
484   cached_min_index_ = 0;
485   cached_min_key_ = Key(Front());
486   valid_cached_min_ = true;
487 }
488 
489 
490 template<TEMPLATE_INVALSET_P_DECL>
Clean()491 void InvalSet<TEMPLATE_INVALSET_P_DEF>::Clean() {
492   VIXL_ASSERT(monitor() == 0);
493   if (empty() || !IsUsingVector()) {
494     return;
495   }
496   // Manually iterate through the vector storage to discard invalid elements.
497   ElementType* start = &(vector_->front());
498   ElementType* end = start + vector_->size();
499   ElementType* c = start;
500   ElementType* first_invalid;
501   ElementType* first_valid;
502   ElementType* next_invalid;
503 
504   while (c < end && IsValid(*c)) { c++; }
505   first_invalid = c;
506 
507   while (c < end) {
508     while (c < end && !IsValid(*c)) { c++; }
509     first_valid = c;
510     while (c < end && IsValid(*c)) { c++; }
511     next_invalid = c;
512 
513     ptrdiff_t n_moved_elements = (next_invalid - first_valid);
514     memmove(first_invalid, first_valid,  n_moved_elements * sizeof(*c));
515     first_invalid = first_invalid + n_moved_elements;
516     c = next_invalid;
517   }
518 
519   // Delete the trailing invalid elements.
520   vector_->erase(vector_->begin() + (first_invalid - start), vector_->end());
521   VIXL_ASSERT(vector_->size() == size_);
522 
523   if (sorted_) {
524     valid_cached_min_ = true;
525     cached_min_index_ = 0;
526     cached_min_key_ = Key(*ElementAt(0));
527   } else {
528     valid_cached_min_ = false;
529   }
530 }
531 
532 
533 template<TEMPLATE_INVALSET_P_DECL>
Front()534 const ElementType InvalSet<TEMPLATE_INVALSET_P_DEF>::Front() const {
535   VIXL_ASSERT(!empty());
536   return IsUsingVector() ? vector_->front() : preallocated_[0];
537 }
538 
539 
540 template<TEMPLATE_INVALSET_P_DECL>
Back()541 const ElementType InvalSet<TEMPLATE_INVALSET_P_DEF>::Back() const {
542   VIXL_ASSERT(!empty());
543   return IsUsingVector() ? vector_->back() : preallocated_[size_ - 1];
544 }
545 
546 
547 template<TEMPLATE_INVALSET_P_DECL>
CleanBack()548 const ElementType InvalSet<TEMPLATE_INVALSET_P_DEF>::CleanBack() {
549   VIXL_ASSERT(monitor() == 0);
550   if (IsUsingVector()) {
551     // Delete the invalid trailing elements.
552     typename std::vector<ElementType>::reverse_iterator it = vector_->rbegin();
553     while (!IsValid(*it)) {
554       it++;
555     }
556     vector_->erase(it.base(), vector_->end());
557   }
558   return Back();
559 }
560 
561 
562 template<TEMPLATE_INVALSET_P_DECL>
StorageBegin()563 const ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::StorageBegin() const {
564   return IsUsingVector() ? &(vector_->front()) : preallocated_;
565 }
566 
567 
568 template<TEMPLATE_INVALSET_P_DECL>
StorageEnd()569 const ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::StorageEnd() const {
570   return IsUsingVector() ? &(vector_->back()) + 1 : preallocated_ + size_;
571 }
572 
573 
574 template<TEMPLATE_INVALSET_P_DECL>
StorageBegin()575 ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::StorageBegin() {
576   return IsUsingVector() ? &(vector_->front()) : preallocated_;
577 }
578 
579 
580 template<TEMPLATE_INVALSET_P_DECL>
StorageEnd()581 ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::StorageEnd() {
582   return IsUsingVector() ? &(vector_->back()) + 1 : preallocated_ + size_;
583 }
584 
585 
586 template<TEMPLATE_INVALSET_P_DECL>
ElementIndex(const ElementType * element)587 size_t InvalSet<TEMPLATE_INVALSET_P_DEF>::ElementIndex(
588     const ElementType* element) const {
589   VIXL_ASSERT((StorageBegin() <= element) && (element < StorageEnd()));
590   return element - StorageBegin();
591 }
592 
593 
594 template<TEMPLATE_INVALSET_P_DECL>
ElementAt(size_t index)595 const ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::ElementAt(
596     size_t index) const {
597   VIXL_ASSERT(
598       (IsUsingVector() && (index < vector_->size())) || (index < size_));
599   return StorageBegin() + index;
600 }
601 
602 template<TEMPLATE_INVALSET_P_DECL>
ElementAt(size_t index)603 ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::ElementAt(size_t index) {
604   VIXL_ASSERT(
605       (IsUsingVector() && (index < vector_->size())) || (index < size_));
606   return StorageBegin() + index;
607 }
608 
609 template<TEMPLATE_INVALSET_P_DECL>
FirstValidElement(const ElementType * from,const ElementType * end)610 const ElementType* InvalSet<TEMPLATE_INVALSET_P_DEF>::FirstValidElement(
611     const ElementType* from, const ElementType* end) {
612   while ((from < end) && !IsValid(*from)) {
613     from++;
614   }
615   return from;
616 }
617 
618 
619 template<TEMPLATE_INVALSET_P_DECL>
CacheMinElement()620 void InvalSet<TEMPLATE_INVALSET_P_DEF>::CacheMinElement() {
621   VIXL_ASSERT(monitor() == 0);
622   VIXL_ASSERT(!empty());
623 
624   if (valid_cached_min_) {
625     return;
626   }
627 
628   if (sorted_) {
629     const ElementType* min = FirstValidElement(StorageBegin(), StorageEnd());
630     cached_min_index_ = ElementIndex(min);
631     cached_min_key_ = Key(*min);
632     valid_cached_min_ = true;
633   } else {
634     Sort(kHardSort);
635   }
636   VIXL_ASSERT(valid_cached_min_);
637 }
638 
639 
640 template<TEMPLATE_INVALSET_P_DECL>
ShouldReclaimMemory()641 bool InvalSet<TEMPLATE_INVALSET_P_DEF>::ShouldReclaimMemory() const {
642   if (!IsUsingVector()) {
643     return false;
644   }
645   size_t n_invalid_elements = vector_->size() - size_;
646   return (n_invalid_elements > RECLAIM_FROM) &&
647          (n_invalid_elements > vector_->size() / RECLAIM_FACTOR);
648 }
649 
650 
651 template<TEMPLATE_INVALSET_P_DECL>
ReclaimMemory()652 void InvalSet<TEMPLATE_INVALSET_P_DEF>::ReclaimMemory() {
653   VIXL_ASSERT(monitor() == 0);
654   Clean();
655 }
656 
657 
658 template<class S>
InvalSetIterator(S * inval_set)659 InvalSetIterator<S>::InvalSetIterator(S* inval_set)
660     : using_vector_((inval_set != NULL) && inval_set->IsUsingVector()),
661       index_(0),
662       inval_set_(inval_set) {
663   if (inval_set != NULL) {
664     inval_set->Sort(S::kSoftSort);
665 #ifdef VIXL_DEBUG
666     inval_set->Acquire();
667 #endif
668     if (using_vector_) {
669       iterator_ = typename std::vector<ElementType>::iterator(
670           inval_set_->vector_->begin());
671     }
672     MoveToValidElement();
673   }
674 }
675 
676 
677 template<class S>
~InvalSetIterator()678 InvalSetIterator<S>::~InvalSetIterator() {
679 #ifdef VIXL_DEBUG
680   if (inval_set_ != NULL) {
681     inval_set_->Release();
682   }
683 #endif
684 }
685 
686 
687 template<class S>
Current()688 typename S::_ElementType* InvalSetIterator<S>::Current() const {
689   VIXL_ASSERT(!Done());
690   if (using_vector_) {
691     return &(*iterator_);
692   } else {
693     return &(inval_set_->preallocated_[index_]);
694   }
695 }
696 
697 
698 template<class S>
Advance()699 void InvalSetIterator<S>::Advance() {
700   VIXL_ASSERT(!Done());
701   if (using_vector_) {
702     iterator_++;
703 #ifdef VIXL_DEBUG
704     index_++;
705 #endif
706     MoveToValidElement();
707   } else {
708     index_++;
709   }
710 }
711 
712 
713 template<class S>
Done()714 bool InvalSetIterator<S>::Done() const {
715   if (using_vector_) {
716     bool done = (iterator_ == inval_set_->vector_->end());
717     VIXL_ASSERT(done == (index_ == inval_set_->size()));
718     return done;
719   } else {
720     return index_ == inval_set_->size();
721   }
722 }
723 
724 
725 template<class S>
Finish()726 void InvalSetIterator<S>::Finish() {
727   VIXL_ASSERT(inval_set_->sorted_);
728   if (using_vector_) {
729     iterator_ = inval_set_->vector_->end();
730   }
731   index_ = inval_set_->size();
732 }
733 
734 
735 template<class S>
DeleteCurrentAndAdvance()736 void InvalSetIterator<S>::DeleteCurrentAndAdvance() {
737   if (using_vector_) {
738     inval_set_->EraseInternal(&(*iterator_));
739     MoveToValidElement();
740   } else {
741     inval_set_->EraseInternal(inval_set_->preallocated_ + index_);
742   }
743 }
744 
745 
746 template<class S>
IsValid(const ElementType & element)747 bool InvalSetIterator<S>::IsValid(const ElementType& element) {
748   return S::IsValid(element);
749 }
750 
751 
752 template<class S>
Key(const ElementType & element)753 typename S::_KeyType InvalSetIterator<S>::Key(const ElementType& element) {
754   return S::Key(element);
755 }
756 
757 
758 template<class S>
MoveToValidElement()759 void InvalSetIterator<S>::MoveToValidElement() {
760   if (using_vector_) {
761     while ((iterator_ != inval_set_->vector_->end()) && !IsValid(*iterator_)) {
762       iterator_++;
763     }
764   } else {
765     VIXL_ASSERT(inval_set_->empty() || IsValid(inval_set_->preallocated_[0]));
766     // Nothing to do.
767   }
768 }
769 
770 #undef TEMPLATE_INVALSET_P_DECL
771 #undef TEMPLATE_INVALSET_P_DEF
772 
773 }  // namespace vixl
774 
775 #endif  // VIXL_INVALSET_H_
776