// Copyright 2007-2010 Baptiste Lepilleur // Distributed under MIT license, or public domain if desired and // recognized in your jurisdiction. // See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE #ifndef JSONCPP_BATCHALLOCATOR_H_INCLUDED #define JSONCPP_BATCHALLOCATOR_H_INCLUDED #include <stdlib.h> #include <assert.h> #ifndef JSONCPP_DOC_EXCLUDE_IMPLEMENTATION namespace Json { /* Fast memory allocator. * * This memory allocator allocates memory for a batch of object (specified by * the page size, the number of object in each page). * * It does not allow the destruction of a single object. All the allocated * objects can be destroyed at once. The memory can be either released or reused * for future allocation. * * The in-place new operator must be used to construct the object using the * pointer returned by allocate. */ template <typename AllocatedType, const unsigned int objectPerAllocation> class BatchAllocator { public: BatchAllocator(unsigned int objectsPerPage = 255) : freeHead_(0), objectsPerPage_(objectsPerPage) { // printf( "Size: %d => %s\n", sizeof(AllocatedType), // typeid(AllocatedType).name() ); assert(sizeof(AllocatedType) * objectPerAllocation >= sizeof(AllocatedType*)); // We must be able to store a slist in the // object free space. assert(objectsPerPage >= 16); batches_ = allocateBatch(0); // allocated a dummy page currentBatch_ = batches_; } ~BatchAllocator() { for (BatchInfo* batch = batches_; batch;) { BatchInfo* nextBatch = batch->next_; free(batch); batch = nextBatch; } } /// allocate space for an array of objectPerAllocation object. /// @warning it is the responsability of the caller to call objects /// constructors. AllocatedType* allocate() { if (freeHead_) // returns node from free list. { AllocatedType* object = freeHead_; freeHead_ = *(AllocatedType**)object; return object; } if (currentBatch_->used_ == currentBatch_->end_) { currentBatch_ = currentBatch_->next_; while (currentBatch_ && currentBatch_->used_ == currentBatch_->end_) currentBatch_ = currentBatch_->next_; if (!currentBatch_) // no free batch found, allocate a new one { currentBatch_ = allocateBatch(objectsPerPage_); currentBatch_->next_ = batches_; // insert at the head of the list batches_ = currentBatch_; } } AllocatedType* allocated = currentBatch_->used_; currentBatch_->used_ += objectPerAllocation; return allocated; } /// Release the object. /// @warning it is the responsability of the caller to actually destruct the /// object. void release(AllocatedType* object) { assert(object != 0); *(AllocatedType**)object = freeHead_; freeHead_ = object; } private: struct BatchInfo { BatchInfo* next_; AllocatedType* used_; AllocatedType* end_; AllocatedType buffer_[objectPerAllocation]; }; // disabled copy constructor and assignement operator. BatchAllocator(const BatchAllocator&); void operator=(const BatchAllocator&); static BatchInfo* allocateBatch(unsigned int objectsPerPage) { const unsigned int mallocSize = sizeof(BatchInfo) - sizeof(AllocatedType) * objectPerAllocation + sizeof(AllocatedType) * objectPerAllocation * objectsPerPage; BatchInfo* batch = static_cast<BatchInfo*>(malloc(mallocSize)); batch->next_ = 0; batch->used_ = batch->buffer_; batch->end_ = batch->buffer_ + objectsPerPage; return batch; } BatchInfo* batches_; BatchInfo* currentBatch_; /// Head of a single linked list within the allocated space of freeed object AllocatedType* freeHead_; unsigned int objectsPerPage_; }; } // namespace Json #endif // ifndef JSONCPP_DOC_INCLUDE_IMPLEMENTATION #endif // JSONCPP_BATCHALLOCATOR_H_INCLUDED