2015-04-23 15:45:54 -07:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef ANDROID_LINEARALLOCATOR_H
|
|
|
|
#define ANDROID_LINEARALLOCATOR_H
|
|
|
|
|
|
|
|
#include <stddef.h>
|
2015-04-23 15:51:55 -07:00
|
|
|
#include <type_traits>
|
2015-04-23 15:45:54 -07:00
|
|
|
|
2015-10-16 14:23:12 -07:00
|
|
|
#include <vector>
|
|
|
|
|
2015-04-23 15:45:54 -07:00
|
|
|
namespace android {
|
|
|
|
namespace uirenderer {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A memory manager that internally allocates multi-kbyte buffers for placing objects in. It avoids
|
|
|
|
* the overhead of malloc when many objects are allocated. It is most useful when creating many
|
|
|
|
* small objects with a similar lifetime, and doesn't add significant overhead for large
|
|
|
|
* allocations.
|
|
|
|
*/
|
|
|
|
class LinearAllocator {
|
|
|
|
public:
|
|
|
|
LinearAllocator();
|
|
|
|
~LinearAllocator();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reserves and returns a region of memory of at least size 'size', aligning as needed.
|
|
|
|
* Typically this is used in an object's overridden new() method or as a replacement for malloc.
|
|
|
|
*
|
|
|
|
* The lifetime of the returned buffers is tied to that of the LinearAllocator. If calling
|
|
|
|
* delete() on an object stored in a buffer is needed, it should be overridden to use
|
|
|
|
* rewindIfLastAlloc()
|
2016-02-10 16:08:08 -08:00
|
|
|
*
|
|
|
|
* Note that unlike create, for alloc the type is purely for compile-time error
|
|
|
|
* checking and does not affect size.
|
2015-04-23 15:45:54 -07:00
|
|
|
*/
|
2016-02-10 16:08:08 -08:00
|
|
|
template<class T>
|
|
|
|
void* alloc(size_t size) {
|
|
|
|
static_assert(std::is_trivially_destructible<T>::value,
|
|
|
|
"Error, type is non-trivial! did you mean to use create()?");
|
|
|
|
return allocImpl(size);
|
|
|
|
}
|
2015-04-23 15:45:54 -07:00
|
|
|
|
2015-04-23 15:51:55 -07:00
|
|
|
/**
|
2015-12-22 16:32:23 -08:00
|
|
|
* Allocates an instance of the template type with the given construction parameters
|
2015-04-23 15:51:55 -07:00
|
|
|
* and adds it to the automatic destruction list.
|
|
|
|
*/
|
2015-12-22 16:32:23 -08:00
|
|
|
template<class T, typename... Params>
|
2016-02-10 16:08:08 -08:00
|
|
|
T* create(Params&&... params) {
|
|
|
|
T* ret = new (allocImpl(sizeof(T))) T(std::forward<Params>(params)...);
|
|
|
|
if (!std::is_trivially_destructible<T>::value) {
|
|
|
|
auto dtor = [](void* ret) { ((T*)ret)->~T(); };
|
|
|
|
addToDestructionList(dtor, ret);
|
|
|
|
}
|
2015-04-23 15:51:55 -07:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2016-02-10 16:08:08 -08:00
|
|
|
template<class T, typename... Params>
|
|
|
|
T* create_trivial(Params&&... params) {
|
|
|
|
static_assert(std::is_trivially_destructible<T>::value,
|
|
|
|
"Error, called create_trivial on a non-trivial type");
|
|
|
|
return new (allocImpl(sizeof(T))) T(std::forward<Params>(params)...);
|
2015-04-23 15:51:55 -07:00
|
|
|
}
|
|
|
|
|
2016-02-19 15:51:02 -08:00
|
|
|
template<class T>
|
|
|
|
T* create_trivial_array(int count) {
|
|
|
|
static_assert(std::is_trivially_destructible<T>::value,
|
|
|
|
"Error, called create_trivial_array on a non-trivial type");
|
|
|
|
return reinterpret_cast<T*>(allocImpl(sizeof(T) * count));
|
|
|
|
}
|
|
|
|
|
2015-04-23 15:45:54 -07:00
|
|
|
/**
|
|
|
|
* Attempt to deallocate the given buffer, with the LinearAllocator attempting to rewind its
|
2015-04-23 15:51:55 -07:00
|
|
|
* state if possible.
|
2015-04-23 15:45:54 -07:00
|
|
|
*/
|
|
|
|
void rewindIfLastAlloc(void* ptr, size_t allocSize);
|
|
|
|
|
2015-04-23 15:51:55 -07:00
|
|
|
/**
|
|
|
|
* Same as rewindIfLastAlloc(void*, size_t)
|
|
|
|
*/
|
|
|
|
template<class T>
|
|
|
|
void rewindIfLastAlloc(T* ptr) {
|
|
|
|
rewindIfLastAlloc((void*)ptr, sizeof(T));
|
|
|
|
}
|
|
|
|
|
2015-04-23 15:45:54 -07:00
|
|
|
/**
|
|
|
|
* Dump memory usage statistics to the log (allocated and wasted space)
|
|
|
|
*/
|
|
|
|
void dumpMemoryStats(const char* prefix = "");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of bytes used for buffers allocated in the LinearAllocator (does not count space
|
|
|
|
* wasted)
|
|
|
|
*/
|
|
|
|
size_t usedSize() const { return mTotalAllocated - mWastedSpace; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
LinearAllocator(const LinearAllocator& other);
|
|
|
|
|
|
|
|
class Page;
|
2015-04-23 15:51:55 -07:00
|
|
|
typedef void (*Destructor)(void* addr);
|
|
|
|
struct DestructorNode {
|
|
|
|
Destructor dtor;
|
|
|
|
void* addr;
|
|
|
|
DestructorNode* next = nullptr;
|
|
|
|
};
|
2015-04-23 15:45:54 -07:00
|
|
|
|
2016-02-10 16:08:08 -08:00
|
|
|
void* allocImpl(size_t size);
|
|
|
|
|
2015-04-23 15:51:55 -07:00
|
|
|
void addToDestructionList(Destructor, void* addr);
|
|
|
|
void runDestructorFor(void* addr);
|
2015-04-23 15:45:54 -07:00
|
|
|
Page* newPage(size_t pageSize);
|
|
|
|
bool fitsInCurrentPage(size_t size);
|
|
|
|
void ensureNext(size_t size);
|
|
|
|
void* start(Page *p);
|
|
|
|
void* end(Page* p);
|
|
|
|
|
|
|
|
size_t mPageSize;
|
|
|
|
size_t mMaxAllocSize;
|
|
|
|
void* mNext;
|
|
|
|
Page* mCurrentPage;
|
|
|
|
Page* mPages;
|
2015-04-23 15:51:55 -07:00
|
|
|
DestructorNode* mDtorList = nullptr;
|
2015-04-23 15:45:54 -07:00
|
|
|
|
|
|
|
// Memory usage tracking
|
|
|
|
size_t mTotalAllocated;
|
|
|
|
size_t mWastedSpace;
|
|
|
|
size_t mPageCount;
|
|
|
|
size_t mDedicatedPageCount;
|
|
|
|
};
|
|
|
|
|
2015-10-15 17:13:00 -07:00
|
|
|
template <class T>
|
|
|
|
class LinearStdAllocator {
|
|
|
|
public:
|
|
|
|
typedef T value_type; // needed to implement std::allocator
|
|
|
|
typedef T* pointer; // needed to implement std::allocator
|
|
|
|
|
2016-08-29 14:52:43 -07:00
|
|
|
explicit LinearStdAllocator(LinearAllocator& allocator)
|
2015-10-15 17:13:00 -07:00
|
|
|
: linearAllocator(allocator) {}
|
|
|
|
LinearStdAllocator(const LinearStdAllocator& other)
|
|
|
|
: linearAllocator(other.linearAllocator) {}
|
|
|
|
~LinearStdAllocator() {}
|
|
|
|
|
|
|
|
// rebind marks that allocators can be rebound to different types
|
|
|
|
template <class U>
|
|
|
|
struct rebind {
|
|
|
|
typedef LinearStdAllocator<U> other;
|
|
|
|
};
|
|
|
|
// enable allocators to be constructed from other templated types
|
|
|
|
template <class U>
|
2016-08-29 14:52:43 -07:00
|
|
|
LinearStdAllocator(const LinearStdAllocator<U>& other) // NOLINT(implicit)
|
2015-10-15 17:13:00 -07:00
|
|
|
: linearAllocator(other.linearAllocator) {}
|
|
|
|
|
|
|
|
T* allocate(size_t num, const void* = 0) {
|
2016-02-10 16:08:08 -08:00
|
|
|
return (T*)(linearAllocator.alloc<void*>(num * sizeof(T)));
|
2015-10-15 17:13:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
void deallocate(pointer p, size_t num) {
|
|
|
|
// attempt to rewind, but no guarantees
|
|
|
|
linearAllocator.rewindIfLastAlloc(p, num * sizeof(T));
|
|
|
|
}
|
|
|
|
|
|
|
|
// public so template copy constructor can access
|
|
|
|
LinearAllocator& linearAllocator;
|
|
|
|
};
|
|
|
|
|
|
|
|
// return that all specializations of LinearStdAllocator are interchangeable
|
|
|
|
template <class T1, class T2>
|
|
|
|
bool operator== (const LinearStdAllocator<T1>&, const LinearStdAllocator<T2>&) { return true; }
|
|
|
|
template <class T1, class T2>
|
|
|
|
bool operator!= (const LinearStdAllocator<T1>&, const LinearStdAllocator<T2>&) { return false; }
|
|
|
|
|
2015-10-16 14:23:12 -07:00
|
|
|
template <class T>
|
|
|
|
class LsaVector : public std::vector<T, LinearStdAllocator<T>> {
|
|
|
|
public:
|
2016-08-29 14:52:43 -07:00
|
|
|
explicit LsaVector(const LinearStdAllocator<T>& allocator)
|
2015-10-16 14:23:12 -07:00
|
|
|
: std::vector<T, LinearStdAllocator<T>>(allocator) {}
|
|
|
|
};
|
|
|
|
|
2015-04-23 15:45:54 -07:00
|
|
|
}; // namespace uirenderer
|
|
|
|
}; // namespace android
|
|
|
|
|
|
|
|
#endif // ANDROID_LINEARALLOCATOR_H
|