/* * Main authors: * Christian Schulte * * Copyright: * Christian Schulte, 2004 * * Last modified: * $Date: 2006-04-11 15:58:37 +0200 (Tue, 11 Apr 2006) $ by $Author: tack $ * $Revision: 3188 $ * * This file is part of Gecode, the generic constraint * development environment: * http://www.gecode.org * * See the file "LICENSE" for information on usage and * redistribution of this file, and for a * DISCLAIMER OF ALL WARRANTIES. * */ #ifndef __GECODE_SUPPORT_BLOCK_ALLOCATOR_HH__ #define __GECODE_SUPPORT_BLOCK_ALLOCATOR_HH__ #include "gecode/kernel.hh" namespace Gecode { namespace Support { /** * \brief Manage memory organized into block lists (allocator) * * The allocation policy is to free all memory allocated when * the block allocator is deleted. * * Requires \code #include "gecode/support/block-allocator.hh" \endcode * \ingroup FuncSupport */ template class BlockAllocator { private: /// One block of memory class Block { public: static const int blocksize = 512; T b[blocksize]; Block* next; }; /// Pointer to current block Block* b; /// Pointer to next part of block (decreasing) T* n; /// Size of allocated blocks size_t _size; /// Allocate additional block void allocate(void); public: /// Initialize BlockAllocator(void); /// Free all allocated blocks ~BlockAllocator(void); /// Return memory of size required by \a T T* operator()(void); /// Return size of memory required by allocator size_t size(void) const; }; /** * \brief Client for block allocator of type \a T * * Provides memory management for objects of type \a T. * * Requires \code #include "gecode/support/block-allocator.hh" \endcode * \ingroup FuncSupport */ template class BlockClient { public: /// Allocate memory from block allocator \a ba static void* operator new(size_t s, BlockAllocator& ba); /// Noop (memory freed only when block allocator deleted) static void operator delete(void*, BlockAllocator& ba); /// Noop (memory freed only when block allocator deleted) static void operator delete(void*); }; template forceinline BlockAllocator::BlockAllocator(void) { b = reinterpret_cast(Memory::malloc(sizeof(Block))); b->next = NULL; n = &b->b[Block::blocksize]; _size = sizeof(Block); } template forceinline BlockAllocator::~BlockAllocator(void) { while (b != NULL) { Block* f = b; b = b->next; Memory::free(f); } } template forceinline T* BlockAllocator::operator()(void) { T* t = --n; if (t == &b->b[0]) allocate(); return t; } template void BlockAllocator::allocate(void) { // Allocate another block Block* nb = reinterpret_cast(Memory::malloc(sizeof(Block))); nb->next = b; b = nb; n = &nb->b[Block::blocksize]; _size += sizeof(Block); } template forceinline size_t BlockAllocator::size(void) const { return _size; } template forceinline void BlockClient::operator delete(void*, BlockAllocator&) { } template forceinline void BlockClient::operator delete(void*) { } template forceinline void* BlockClient::operator new(size_t s, BlockAllocator& ba) { assert(s == sizeof(T)); return ba(); } }} #endif // STATISTICS: support-any