| //===- llvm/ADT/SmallPtrSet.cpp - 'Normally small' pointer set ------------===// | 
 | // | 
 | //                     The LLVM Compiler Infrastructure | 
 | // | 
 | // This file is distributed under the University of Illinois Open Source | 
 | // License. See LICENSE.TXT for details. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 | // | 
 | // This file implements the SmallPtrSet class.  See SmallPtrSet.h for an | 
 | // overview of the algorithm. | 
 | // | 
 | //===----------------------------------------------------------------------===// | 
 |  | 
 | #include "llvm/ADT/SmallPtrSet.h" | 
 | #include "llvm/Support/MathExtras.h" | 
 | #include <cstdlib> | 
 |  | 
 | using namespace llvm; | 
 |  | 
 | void SmallPtrSetImpl::shrink_and_clear() { | 
 |   assert(!isSmall() && "Can't shrink a small set!"); | 
 |   free(CurArray); | 
 |  | 
 |   // Reduce the number of buckets. | 
 |   CurArraySize = NumElements > 16 ? 1 << (Log2_32_Ceil(NumElements) + 1) : 32; | 
 |   NumElements = NumTombstones = 0; | 
 |  | 
 |   // Install the new array.  Clear all the buckets to empty. | 
 |   CurArray = (const void**)malloc(sizeof(void*) * (CurArraySize+1)); | 
 |   assert(CurArray && "Failed to allocate memory?"); | 
 |   memset(CurArray, -1, CurArraySize*sizeof(void*)); | 
 |    | 
 |   // The end pointer, always valid, is set to a valid element to help the | 
 |   // iterator. | 
 |   CurArray[CurArraySize] = 0; | 
 | } | 
 |  | 
 | bool SmallPtrSetImpl::insert_imp(const void * Ptr) { | 
 |   if (isSmall()) { | 
 |     // Check to see if it is already in the set. | 
 |     for (const void **APtr = SmallArray, **E = SmallArray+NumElements; | 
 |          APtr != E; ++APtr) | 
 |       if (*APtr == Ptr) | 
 |         return false; | 
 |      | 
 |     // Nope, there isn't.  If we stay small, just 'pushback' now. | 
 |     if (NumElements < CurArraySize-1) { | 
 |       SmallArray[NumElements++] = Ptr; | 
 |       return true; | 
 |     } | 
 |     // Otherwise, hit the big set case, which will call grow. | 
 |   } | 
 |    | 
 |   // If more than 3/4 of the array is full, grow. | 
 |   if (NumElements*4 >= CurArraySize*3 || | 
 |       CurArraySize-(NumElements+NumTombstones) < CurArraySize/8) | 
 |     Grow(); | 
 |    | 
 |   // Okay, we know we have space.  Find a hash bucket. | 
 |   void **Bucket = const_cast<void**>(FindBucketFor((void*)Ptr)); | 
 |   if (*Bucket == Ptr) return false; // Already inserted, good. | 
 |    | 
 |   // Otherwise, insert it! | 
 |   if (*Bucket == getTombstoneMarker()) | 
 |     --NumTombstones; | 
 |   *Bucket = (void*)Ptr; | 
 |   ++NumElements;  // Track density. | 
 |   return true; | 
 | } | 
 |  | 
 | bool SmallPtrSetImpl::erase_imp(const void * Ptr) { | 
 |   if (isSmall()) { | 
 |     // Check to see if it is in the set. | 
 |     for (const void **APtr = SmallArray, **E = SmallArray+NumElements; | 
 |          APtr != E; ++APtr) | 
 |       if (*APtr == Ptr) { | 
 |         // If it is in the set, replace this element. | 
 |         *APtr = E[-1]; | 
 |         E[-1] = getEmptyMarker(); | 
 |         --NumElements; | 
 |         return true; | 
 |       } | 
 |      | 
 |     return false; | 
 |   } | 
 |    | 
 |   // Okay, we know we have space.  Find a hash bucket. | 
 |   void **Bucket = const_cast<void**>(FindBucketFor(Ptr)); | 
 |   if (*Bucket != Ptr) return false;  // Not in the set? | 
 |  | 
 |   // Set this as a tombstone. | 
 |   *Bucket = getTombstoneMarker(); | 
 |   --NumElements; | 
 |   ++NumTombstones; | 
 |   return true; | 
 | } | 
 |  | 
 | const void * const *SmallPtrSetImpl::FindBucketFor(const void *Ptr) const { | 
 |   unsigned Bucket = Hash(Ptr); | 
 |   unsigned ArraySize = CurArraySize; | 
 |   unsigned ProbeAmt = 1; | 
 |   const void *const *Array = CurArray; | 
 |   const void *const *Tombstone = 0; | 
 |   while (1) { | 
 |     // Found Ptr's bucket? | 
 |     if (Array[Bucket] == Ptr) | 
 |       return Array+Bucket; | 
 |      | 
 |     // If we found an empty bucket, the pointer doesn't exist in the set. | 
 |     // Return a tombstone if we've seen one so far, or the empty bucket if | 
 |     // not. | 
 |     if (Array[Bucket] == getEmptyMarker()) | 
 |       return Tombstone ? Tombstone : Array+Bucket; | 
 |      | 
 |     // If this is a tombstone, remember it.  If Ptr ends up not in the set, we | 
 |     // prefer to return it than something that would require more probing. | 
 |     if (Array[Bucket] == getTombstoneMarker() && !Tombstone) | 
 |       Tombstone = Array+Bucket;  // Remember the first tombstone found. | 
 |      | 
 |     // It's a hash collision or a tombstone. Reprobe. | 
 |     Bucket = (Bucket + ProbeAmt++) & (ArraySize-1); | 
 |   } | 
 | } | 
 |  | 
 | /// Grow - Allocate a larger backing store for the buckets and move it over. | 
 | /// | 
 | void SmallPtrSetImpl::Grow() { | 
 |   // Allocate at twice as many buckets, but at least 128. | 
 |   unsigned OldSize = CurArraySize; | 
 |   unsigned NewSize = OldSize < 64 ? 128 : OldSize*2; | 
 |    | 
 |   const void **OldBuckets = CurArray; | 
 |   bool WasSmall = isSmall(); | 
 |    | 
 |   // Install the new array.  Clear all the buckets to empty. | 
 |   CurArray = (const void**)malloc(sizeof(void*) * (NewSize+1)); | 
 |   assert(CurArray && "Failed to allocate memory?"); | 
 |   CurArraySize = NewSize; | 
 |   memset(CurArray, -1, NewSize*sizeof(void*)); | 
 |    | 
 |   // The end pointer, always valid, is set to a valid element to help the | 
 |   // iterator. | 
 |   CurArray[NewSize] = 0; | 
 |    | 
 |   // Copy over all the elements. | 
 |   if (WasSmall) { | 
 |     // Small sets store their elements in order. | 
 |     for (const void **BucketPtr = OldBuckets, **E = OldBuckets+NumElements; | 
 |          BucketPtr != E; ++BucketPtr) { | 
 |       const void *Elt = *BucketPtr; | 
 |       *const_cast<void**>(FindBucketFor(Elt)) = const_cast<void*>(Elt); | 
 |     } | 
 |   } else { | 
 |     // Copy over all valid entries. | 
 |     for (const void **BucketPtr = OldBuckets, **E = OldBuckets+OldSize; | 
 |          BucketPtr != E; ++BucketPtr) { | 
 |       // Copy over the element if it is valid. | 
 |       const void *Elt = *BucketPtr; | 
 |       if (Elt != getTombstoneMarker() && Elt != getEmptyMarker()) | 
 |         *const_cast<void**>(FindBucketFor(Elt)) = const_cast<void*>(Elt); | 
 |     } | 
 |      | 
 |     free(OldBuckets); | 
 |     NumTombstones = 0; | 
 |   } | 
 | } | 
 |  | 
 | SmallPtrSetImpl::SmallPtrSetImpl(const SmallPtrSetImpl& that) { | 
 |   // If we're becoming small, prepare to insert into our stack space | 
 |   if (that.isSmall()) { | 
 |     CurArray = &SmallArray[0]; | 
 |   // Otherwise, allocate new heap space (unless we were the same size) | 
 |   } else { | 
 |     CurArray = (const void**)malloc(sizeof(void*) * (that.CurArraySize+1)); | 
 |     assert(CurArray && "Failed to allocate memory?"); | 
 |   } | 
 |    | 
 |   // Copy over the new array size | 
 |   CurArraySize = that.CurArraySize; | 
 |  | 
 |   // Copy over the contents from the other set | 
 |   memcpy(CurArray, that.CurArray, sizeof(void*)*(CurArraySize+1)); | 
 |    | 
 |   NumElements = that.NumElements; | 
 |   NumTombstones = that.NumTombstones; | 
 | } | 
 |  | 
 | /// CopyFrom - implement operator= from a smallptrset that has the same pointer | 
 | /// type, but may have a different small size. | 
 | void SmallPtrSetImpl::CopyFrom(const SmallPtrSetImpl &RHS) { | 
 |   if (isSmall() && RHS.isSmall()) | 
 |     assert(CurArraySize == RHS.CurArraySize && | 
 |            "Cannot assign sets with different small sizes"); | 
 |             | 
 |   // If we're becoming small, prepare to insert into our stack space | 
 |   if (RHS.isSmall()) { | 
 |     if (!isSmall()) | 
 |       free(CurArray); | 
 |     CurArray = &SmallArray[0]; | 
 |   // Otherwise, allocate new heap space (unless we were the same size) | 
 |   } else if (CurArraySize != RHS.CurArraySize) { | 
 |     if (isSmall()) | 
 |       CurArray = (const void**)malloc(sizeof(void*) * (RHS.CurArraySize+1)); | 
 |     else | 
 |       CurArray = (const void**)realloc(CurArray, sizeof(void*)*(RHS.CurArraySize+1)); | 
 |     assert(CurArray && "Failed to allocate memory?"); | 
 |   } | 
 |    | 
 |   // Copy over the new array size | 
 |   CurArraySize = RHS.CurArraySize; | 
 |  | 
 |   // Copy over the contents from the other set | 
 |   memcpy(CurArray, RHS.CurArray, sizeof(void*)*(CurArraySize+1)); | 
 |    | 
 |   NumElements = RHS.NumElements; | 
 |   NumTombstones = RHS.NumTombstones; | 
 | } | 
 |  | 
 | SmallPtrSetImpl::~SmallPtrSetImpl() { | 
 |   if (!isSmall()) | 
 |     free(CurArray); | 
 | } |