// -*- C++ -*- #ifndef HL_OBSTACKREAP_H #define HL_OBSTACKREAP_H #include /* ObstackReap layers obstack functionality on top of reaps. */ #if WIN32 #include #endif #include "utility/dynarray.h" namespace ObstackReapNS { #if 0 template class DynamicArray { public: DynamicArray() : internalArray (0), internalArrayLength (0) {} ~DynamicArray() { clear(); } // clear deletes everything in the array. inline void clear() { if (internalArray) { delete [] internalArray; internalArray = 0; internalArrayLength = 0; //printf ("\ninternalArrayLength %x = %d\n", this, internalArrayLength); } } // Read-only access to an array element; // asserts that this index is in range. inline const ObjType& operator[] (int index) const { assert (index < internalArrayLength); assert (index >= 0); return internalArray[index]; } // Access a particular array index by reference, // growing the array if necessary. inline ObjType& operator[] (int index) { assert (index >= 0); if (index >= internalArrayLength) { // This index is beyond the current size of the array. // Grow the array by doubling and copying the old array into the new. const int newSize = index * 2 + 1; ObjType * arr = new ObjType[newSize]; // printf ("grow! %d to %d\n", internalArrayLength, newSize); #if MALLOC_TRACE printf ("m %x %d\n", arr, newSize * sizeof(ObjType)); #endif if (internalArray) { memcpy (arr, internalArray, internalArrayLength * sizeof(ObjType)); delete [] internalArray; #if MALLOC_TRACE printf ("f %x\n", internalArray); #endif } internalArray = arr; internalArrayLength = newSize; // printf ("\ninternalArrayLength %x = %d\n", this, internalArrayLength); } return internalArray[index]; } // trim informs the array that it is now only nelts long // as far as the client is concerned. This may trigger // shrinking of the array. inline void trim (int nelts) { // Halve the array if the number of elements // drops below one-fourth of the array size. if (internalArray) { if (nelts * 4 < internalArrayLength) { const int newSize = nelts * 2; ObjType * arr = new ObjType[newSize]; // printf ("trim! %d to %d\n", internalArrayLength, newSize); #if MALLOC_TRACE printf ("m %x %d\n", arr, newSize * sizeof(ObjType)); #endif memcpy (arr, internalArray, sizeof(ObjType) * nelts); delete [] internalArray; #if MALLOC_TRACE printf ("f %x\n", internalArray); #endif internalArray = arr; internalArrayLength = newSize; } assert (nelts <= internalArrayLength); } } private: // The pointer to the current array. ObjType * internalArray; // The length of the internal array, in elements. int internalArrayLength; }; #endif template class DynStack { public: DynStack() : numItems (0) { items[0] = 0; } int length() const { return numItems; } inline void push (OBJTYPE * ptr) { numItems++; items[numItems] = ptr; } inline OBJTYPE * pop() { OBJTYPE * ptr = 0; if (numItems > 0) { ptr = items[numItems]; numItems--; // The array has shrunk, so potentially trim it. items.trim (numItems + 1); } return ptr; } inline OBJTYPE * top() { OBJTYPE * ptr = NULL; if (numItems > 0) { ptr = items[numItems]; } return ptr; } inline void clear() { items.clear(); } private: // The number of items recorded above. // 0 == no items. // 1 == items[1] has the single item, etc. // i.e., we waste entry zero. int numItems; // The array of remembered objects. HL::DynamicArray items; }; }; // Layers on top of a reap. template class ObstackReap { public: ObstackReap() { currentReap = new ReapType; initCurrentObject(); } ~ObstackReap() { ReapType * r; while ((r = reapStack.pop())) { delete r; } delete currentReap; delete currentObject; } inline void * malloc (size_t sz); inline void freeAfter (void * ptr); inline void freeAll(); inline void * getObjectBase(); inline void finalize(); inline void * grow (size_t sz); private: inline void initCurrentObject(); // Hide free. void free (void *); enum { INITIAL_OBJECT_SIZE = 8 * sizeof(double) }; void * currentObject; char * currentObjectPosition; size_t currentObjectSize; size_t actualObjectSize; bool isCurrentObjectExposed; ReapType * currentReap; ObstackReapNS::DynStack reapStack; }; template void ObstackReap::initCurrentObject() { currentObject = currentReap->malloc (INITIAL_OBJECT_SIZE); currentObjectPosition = (char *) currentObject; currentObjectSize = 0; actualObjectSize = INITIAL_OBJECT_SIZE; isCurrentObjectExposed = false; } template void * ObstackReap::malloc (size_t sz) { if (!isCurrentObjectExposed) { return currentReap->malloc (sz); } else { void * ptr = currentReap->realloc (currentObject, sz); reapStack.push (currentReap); currentReap = new ReapType; initCurrentObject(); return ptr; } } template void ObstackReap::freeAfter (void * ptr) { while (currentReap && (!currentReap->find(ptr))) { delete currentReap; currentReap = reapStack.pop(); } } template void ObstackReap::freeAll() { while (currentReap) { delete currentReap; currentReap = reapStack.pop(); } currentReap = new ReapType; } template inline void * ObstackReap::getObjectBase() { isCurrentObjectExposed = true; return currentObject; } template inline void ObstackReap::finalize() { if (isCurrentObjectExposed) { reapStack.push (currentReap); currentReap = new ReapType; } initCurrentObject(); } template inline void * ObstackReap::grow (size_t sz) { const int requestedObjectSize = currentObjectSize + sz; if (requestedObjectSize > actualObjectSize) { void * ptr = currentReap->realloc (currentObject, sz); currentObjectPosition = (char *) ptr + (currentObjectPosition - (char *) currentObject); if (isCurrentObjectExposed) { reapStack.push (currentReap); currentReap = new ReapType; } currentObject = ptr; } // Because calling grow can result in a new object, // the current object can be considered no longer exposed (if it was before). isCurrentObjectExposed = false; currentObjectSize += sz; char * oldPosition = currentObjectPosition; currentObjectPosition += sz; return oldPosition; } #endif