#include "il2cpp-config.h" #include "utils/MemoryPool.h" #include "utils/Memory.h" #include #include namespace il2cpp { namespace utils { const size_t kPageSize = IL2CPP_PAGE_SIZE; const size_t kDefaultRegionSize = 64 * 1024; // by making all allocations a multiple of this value, we ensure the next // allocation will always be aligned to this value const size_t kMemoryAlignment = 8; static inline size_t MakeMultipleOf(size_t size, size_t alignment) { return (size + alignment - 1) & ~(alignment - 1); } struct MemoryPool::Region { char* start; char* current; size_t size; size_t free; }; MemoryPool::MemoryPool() { AddRegion(kDefaultRegionSize); } MemoryPool::MemoryPool(size_t initialSize) { AddRegion(initialSize); } MemoryPool::~MemoryPool() { for (RegionList::iterator iter = m_Regions.begin(); iter != m_Regions.end(); ++iter) { IL2CPP_FREE((*iter)->start); IL2CPP_FREE(*iter); } m_Regions.clear(); } void* MemoryPool::Malloc(size_t size) { size = MakeMultipleOf(size, kMemoryAlignment); Region* region = m_Regions.back(); if (region->free < size) region = AddRegion(size); IL2CPP_ASSERT(region->free >= size); void* value = region->current; region->current += size; region->free -= size; return value; } void* MemoryPool::Calloc(size_t count, size_t size) { void* ret = Malloc(count * size); return memset(ret, 0, count * size); } MemoryPool::Region* MemoryPool::AddRegion(size_t size) { Region* newRegion = (Region*)IL2CPP_MALLOC(sizeof(Region)); Region* lastFreeRegion = m_Regions.size() > 0 ? m_Regions.back() : NULL; size_t allocationSize; if (lastFreeRegion != NULL && lastFreeRegion->free >= kPageSize) { allocationSize = MakeMultipleOf(size, kPageSize); m_Regions.pop_back(); m_Regions.push_back(newRegion); m_Regions.push_back(lastFreeRegion); } else { allocationSize = std::max(kDefaultRegionSize, MakeMultipleOf(size, kPageSize)); m_Regions.push_back(newRegion); } newRegion->start = newRegion->current = (char*)IL2CPP_MALLOC(allocationSize); newRegion->size = newRegion->free = allocationSize; return newRegion; } } }