/* ScummVM - Graphic Adventure Engine * * ScummVM is the legal property of its developers, whose names * are too numerous to list here. Please refer to the COPYRIGHT * file distributed with this source distribution. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. * * $URL$ * $Id$ * */ #ifndef COMMON_MEMORYPOOL_H #define COMMON_MEMORYPOOL_H #include "common/scummsys.h" #include "common/array.h" namespace Common { class MemoryPool { protected: MemoryPool(const MemoryPool&); MemoryPool& operator=(const MemoryPool&); struct Page { void *start; size_t numChunks; }; size_t _chunkSize; Array _pages; void *_next; size_t _chunksPerPage; void allocPage(); void addPageToPool(const Page &page); bool isPointerInPage(void *ptr, const Page &page); public: MemoryPool(size_t chunkSize); ~MemoryPool(); void *allocChunk(); void freeChunk(void *ptr); void freeUnusedPages(); size_t getChunkSize() const { return _chunkSize; } }; template class FixedSizeMemoryPool : public MemoryPool { private: enum { REAL_CHUNK_SIZE = (CHUNK_SIZE + sizeof(void*) - 1) & (~(sizeof(void*) - 1)) }; byte _storage[NUM_INTERNAL_CHUNKS * REAL_CHUNK_SIZE]; public: FixedSizeMemoryPool() : MemoryPool(CHUNK_SIZE) { assert(REAL_CHUNK_SIZE == _chunkSize); // Insert some static storage Page internalPage = { _storage, NUM_INTERNAL_CHUNKS }; addPageToPool(internalPage); } }; template class FixedSizeMemoryPool : public MemoryPool { public: FixedSizeMemoryPool() : MemoryPool(CHUNK_SIZE) {} }; template class ObjectPool : public FixedSizeMemoryPool { public: void deleteChunk(T *ptr) { ptr->~T(); freeChunk(ptr); } }; } // End of namespace Common // Provide a custom placement new operator, using an arbitrary // MemoryPool. // // This *should* work with all C++ implementations, but may not. // // For details on using placement new for custom allocators, see e.g. // inline void* operator new(size_t nbytes, Common::MemoryPool& pool) { assert(nbytes <= pool.getChunkSize()); return pool.allocChunk(); } inline void operator delete(void* p, Common::MemoryPool& pool) { pool.freeChunk(p); } #endif