| /* | 
 |  * Copyright 2012, The Android Open Source Project | 
 |  * | 
 |  * Redistribution and use in source and binary forms, with or without | 
 |  * modification, are permitted provided that the following conditions | 
 |  * are met: | 
 |  *  * Redistributions of source code must retain the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer. | 
 |  *  * Redistributions in binary form must reproduce the above copyright | 
 |  *    notice, this list of conditions and the following disclaimer in the | 
 |  *    documentation and/or other materials provided with the distribution. | 
 |  * | 
 |  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY | 
 |  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE | 
 |  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR | 
 |  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR | 
 |  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, | 
 |  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, | 
 |  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | 
 |  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY | 
 |  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | 
 |  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
 |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
 |  */ | 
 |  | 
 | #define LOG_TAG "LinearAllocator" | 
 | #define LOG_NDEBUG 1 | 
 |  | 
 | #include <stdlib.h> | 
 | #include <utils/LinearAllocator.h> | 
 | #include <utils/Log.h> | 
 |  | 
 |  | 
 | // The ideal size of a page allocation (these need to be multiples of 8) | 
 | #define INITIAL_PAGE_SIZE ((size_t)4096) // 4kb | 
 | #define MAX_PAGE_SIZE ((size_t)131072) // 128kb | 
 |  | 
 | // The maximum amount of wasted space we can have per page | 
 | // Allocations exceeding this will have their own dedicated page | 
 | // If this is too low, we will malloc too much | 
 | // Too high, and we may waste too much space | 
 | // Must be smaller than INITIAL_PAGE_SIZE | 
 | #define MAX_WASTE_SIZE ((size_t)1024) | 
 |  | 
 | #if ALIGN_DOUBLE | 
 | #define ALIGN_SZ (sizeof(double)) | 
 | #else | 
 | #define ALIGN_SZ (sizeof(int)) | 
 | #endif | 
 |  | 
 | #define ALIGN(x) ((x + ALIGN_SZ - 1 ) & ~(ALIGN_SZ - 1)) | 
 | #define ALIGN_PTR(p) ((void*)(ALIGN((size_t)p))) | 
 |  | 
 | #if LOG_NDEBUG | 
 | #define ADD_ALLOCATION(size) | 
 | #define RM_ALLOCATION(size) | 
 | #else | 
 | #include <utils/Thread.h> | 
 | #include <utils/Timers.h> | 
 | static size_t s_totalAllocations = 0; | 
 | static nsecs_t s_nextLog = 0; | 
 | static android::Mutex s_mutex; | 
 |  | 
 | static void _logUsageLocked() { | 
 |     nsecs_t now = systemTime(SYSTEM_TIME_MONOTONIC); | 
 |     if (now > s_nextLog) { | 
 |         s_nextLog = now + milliseconds_to_nanoseconds(10); | 
 |         ALOGV("Total memory usage: %zu kb", s_totalAllocations / 1024); | 
 |     } | 
 | } | 
 |  | 
 | static void _addAllocation(size_t size) { | 
 |     android::AutoMutex lock(s_mutex); | 
 |     s_totalAllocations += size; | 
 |     _logUsageLocked(); | 
 | } | 
 |  | 
 | #define ADD_ALLOCATION(size) _addAllocation(size); | 
 | #define RM_ALLOCATION(size) _addAllocation(-size); | 
 | #endif | 
 |  | 
 | #define min(x,y) (((x) < (y)) ? (x) : (y)) | 
 |  | 
 | namespace android { | 
 |  | 
 | class LinearAllocator::Page { | 
 | public: | 
 |     Page* next() { return mNextPage; } | 
 |     void setNext(Page* next) { mNextPage = next; } | 
 |  | 
 |     Page() | 
 |         : mNextPage(0) | 
 |     {} | 
 |  | 
 |     void* operator new(size_t /*size*/, void* buf) { return buf; } | 
 |  | 
 |     void* start() { | 
 |         return (void*) (((size_t)this) + sizeof(Page)); | 
 |     } | 
 |  | 
 |     void* end(int pageSize) { | 
 |         return (void*) (((size_t)start()) + pageSize); | 
 |     } | 
 |  | 
 | private: | 
 |     Page(const Page& /*other*/) {} | 
 |     Page* mNextPage; | 
 | }; | 
 |  | 
 | LinearAllocator::LinearAllocator() | 
 |     : mPageSize(INITIAL_PAGE_SIZE) | 
 |     , mMaxAllocSize(MAX_WASTE_SIZE) | 
 |     , mNext(0) | 
 |     , mCurrentPage(0) | 
 |     , mPages(0) | 
 |     , mTotalAllocated(0) | 
 |     , mWastedSpace(0) | 
 |     , mPageCount(0) | 
 |     , mDedicatedPageCount(0) {} | 
 |  | 
 | LinearAllocator::~LinearAllocator(void) { | 
 |     Page* p = mPages; | 
 |     while (p) { | 
 |         Page* next = p->next(); | 
 |         p->~Page(); | 
 |         free(p); | 
 |         RM_ALLOCATION(mPageSize); | 
 |         p = next; | 
 |     } | 
 | } | 
 |  | 
 | void* LinearAllocator::start(Page* p) { | 
 |     return ALIGN_PTR(((size_t*)p) + sizeof(Page)); | 
 | } | 
 |  | 
 | void* LinearAllocator::end(Page* p) { | 
 |     return ((char*)p) + mPageSize; | 
 | } | 
 |  | 
 | bool LinearAllocator::fitsInCurrentPage(size_t size) { | 
 |     return mNext && ((char*)mNext + size) <= end(mCurrentPage); | 
 | } | 
 |  | 
 | void LinearAllocator::ensureNext(size_t size) { | 
 |     if (fitsInCurrentPage(size)) return; | 
 |  | 
 |     if (mCurrentPage && mPageSize < MAX_PAGE_SIZE) { | 
 |         mPageSize = min(MAX_PAGE_SIZE, mPageSize * 2); | 
 |         mPageSize = ALIGN(mPageSize); | 
 |     } | 
 |     mWastedSpace += mPageSize; | 
 |     Page* p = newPage(mPageSize); | 
 |     if (mCurrentPage) { | 
 |         mCurrentPage->setNext(p); | 
 |     } | 
 |     mCurrentPage = p; | 
 |     if (!mPages) { | 
 |         mPages = mCurrentPage; | 
 |     } | 
 |     mNext = start(mCurrentPage); | 
 | } | 
 |  | 
 | void* LinearAllocator::alloc(size_t size) { | 
 |     size = ALIGN(size); | 
 |     if (size > mMaxAllocSize && !fitsInCurrentPage(size)) { | 
 |         ALOGV("Exceeded max size %zu > %zu", size, mMaxAllocSize); | 
 |         // Allocation is too large, create a dedicated page for the allocation | 
 |         Page* page = newPage(size); | 
 |         mDedicatedPageCount++; | 
 |         page->setNext(mPages); | 
 |         mPages = page; | 
 |         if (!mCurrentPage) | 
 |             mCurrentPage = mPages; | 
 |         return start(page); | 
 |     } | 
 |     ensureNext(size); | 
 |     void* ptr = mNext; | 
 |     mNext = ((char*)mNext) + size; | 
 |     mWastedSpace -= size; | 
 |     return ptr; | 
 | } | 
 |  | 
 | void LinearAllocator::rewindIfLastAlloc(void* ptr, size_t allocSize) { | 
 |     // Don't bother rewinding across pages | 
 |     allocSize = ALIGN(allocSize); | 
 |     if (ptr >= start(mCurrentPage) && ptr < end(mCurrentPage) | 
 |             && ptr == ((char*)mNext - allocSize)) { | 
 |         mTotalAllocated -= allocSize; | 
 |         mWastedSpace += allocSize; | 
 |         mNext = ptr; | 
 |     } | 
 | } | 
 |  | 
 | LinearAllocator::Page* LinearAllocator::newPage(size_t pageSize) { | 
 |     pageSize = ALIGN(pageSize + sizeof(LinearAllocator::Page)); | 
 |     ADD_ALLOCATION(pageSize); | 
 |     mTotalAllocated += pageSize; | 
 |     mPageCount++; | 
 |     void* buf = malloc(pageSize); | 
 |     return new (buf) Page(); | 
 | } | 
 |  | 
 | static const char* toSize(size_t value, float& result) { | 
 |     if (value < 2000) { | 
 |         result = value; | 
 |         return "B"; | 
 |     } | 
 |     if (value < 2000000) { | 
 |         result = value / 1024.0f; | 
 |         return "KB"; | 
 |     } | 
 |     result = value / 1048576.0f; | 
 |     return "MB"; | 
 | } | 
 |  | 
 | void LinearAllocator::dumpMemoryStats(const char* prefix) { | 
 |     float prettySize; | 
 |     const char* prettySuffix; | 
 |     prettySuffix = toSize(mTotalAllocated, prettySize); | 
 |     ALOGD("%sTotal allocated: %.2f%s", prefix, prettySize, prettySuffix); | 
 |     prettySuffix = toSize(mWastedSpace, prettySize); | 
 |     ALOGD("%sWasted space: %.2f%s (%.1f%%)", prefix, prettySize, prettySuffix, | 
 |           (float) mWastedSpace / (float) mTotalAllocated * 100.0f); | 
 |     ALOGD("%sPages %zu (dedicated %zu)", prefix, mPageCount, mDedicatedPageCount); | 
 | } | 
 |  | 
 | }; // namespace android |