| /* |
| * Copyright 2011 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #ifndef SkTArray_DEFINED |
| #define SkTArray_DEFINED |
| |
| #include <new> |
| #include "SkTypes.h" |
| #include "SkTemplates.h" |
| |
| template <typename T, bool MEM_COPY = false> class SkTArray; |
| |
| namespace SkTArrayExt { |
| |
| template<typename T> |
| inline void copy(SkTArray<T, true>* self, const T* array) { |
| memcpy(self->fMemArray, array, self->fCount * sizeof(T)); |
| } |
| template<typename T> |
| inline void copyAndDelete(SkTArray<T, true>* self, char* newMemArray) { |
| memcpy(newMemArray, self->fMemArray, self->fCount * sizeof(T)); |
| } |
| |
| template<typename T> |
| inline void copy(SkTArray<T, false>* self, const T* array) { |
| for (int i = 0; i < self->fCount; ++i) { |
| new (self->fItemArray + i) T(array[i]); |
| } |
| } |
| template<typename T> |
| inline void copyAndDelete(SkTArray<T, false>* self, char* newMemArray) { |
| for (int i = 0; i < self->fCount; ++i) { |
| new (newMemArray + sizeof(T) * i) T(self->fItemArray[i]); |
| self->fItemArray[i].~T(); |
| } |
| } |
| |
| } |
| |
| /** When MEM_COPY is true T will be bit copied when moved. |
| When MEM_COPY is false, T will be copy constructed / destructed. |
| In all cases T's constructor will be called on allocation, |
| and its destructor will be called from this object's destructor. |
| */ |
| template <typename T, bool MEM_COPY> class SkTArray { |
| public: |
| /** |
| * Creates an empty array with no initial storage |
| */ |
| SkTArray() { |
| fCount = 0; |
| fReserveCount = gMIN_ALLOC_COUNT; |
| fAllocCount = 0; |
| fMemArray = NULL; |
| fPreAllocMemArray = NULL; |
| } |
| |
| /** |
| * Creates an empty array that will preallocate space for reserveCount |
| * elements. |
| */ |
| explicit SkTArray(int reserveCount) { |
| this->init(NULL, 0, NULL, reserveCount); |
| } |
| |
| /** |
| * Copies one array to another. The new array will be heap allocated. |
| */ |
| explicit SkTArray(const SkTArray& array) { |
| this->init(array.fItemArray, array.fCount, NULL, 0); |
| } |
| |
| /** |
| * Creates a SkTArray by copying contents of a standard C array. The new |
| * array will be heap allocated. Be careful not to use this constructor |
| * when you really want the (void*, int) version. |
| */ |
| SkTArray(const T* array, int count) { |
| this->init(array, count, NULL, 0); |
| } |
| |
| /** |
| * assign copy of array to this |
| */ |
| SkTArray& operator =(const SkTArray& array) { |
| for (int i = 0; i < fCount; ++i) { |
| fItemArray[i].~T(); |
| } |
| fCount = 0; |
| checkRealloc((int)array.count()); |
| fCount = array.count(); |
| SkTArrayExt::copy(this, static_cast<const T*>(array.fMemArray)); |
| return *this; |
| } |
| |
| virtual ~SkTArray() { |
| for (int i = 0; i < fCount; ++i) { |
| fItemArray[i].~T(); |
| } |
| if (fMemArray != fPreAllocMemArray) { |
| sk_free(fMemArray); |
| } |
| } |
| |
| /** |
| * Resets to count() == 0 |
| */ |
| void reset() { this->pop_back_n(fCount); } |
| |
| /** |
| * Number of elements in the array. |
| */ |
| int count() const { return fCount; } |
| |
| /** |
| * Is the array empty. |
| */ |
| bool empty() const { return !fCount; } |
| |
| /** |
| * Adds 1 new default-constructed T value and returns in by reference. Note |
| * the reference only remains valid until the next call that adds or removes |
| * elements. |
| */ |
| T& push_back() { |
| checkRealloc(1); |
| new ((char*)fMemArray+sizeof(T)*fCount) T; |
| ++fCount; |
| return fItemArray[fCount-1]; |
| } |
| |
| /** |
| * Version of above that uses a copy constructor to initialize the new item |
| */ |
| T& push_back(const T& t) { |
| checkRealloc(1); |
| new ((char*)fMemArray+sizeof(T)*fCount) T(t); |
| ++fCount; |
| return fItemArray[fCount-1]; |
| } |
| |
| /** |
| * Allocates n more default T values, and returns the address of the start |
| * of that new range. Note: this address is only valid until the next API |
| * call made on the array that might add or remove elements. |
| */ |
| T* push_back_n(int n) { |
| SkASSERT(n >= 0); |
| checkRealloc(n); |
| for (int i = 0; i < n; ++i) { |
| new (fItemArray + fCount + i) T; |
| } |
| fCount += n; |
| return fItemArray + fCount - n; |
| } |
| |
| /** |
| * Version of above that uses a copy constructor to initialize all n items |
| * to the same T. |
| */ |
| T* push_back_n(int n, const T& t) { |
| SkASSERT(n >= 0); |
| checkRealloc(n); |
| for (int i = 0; i < n; ++i) { |
| new (fItemArray + fCount + i) T(t); |
| } |
| fCount += n; |
| return fItemArray + fCount - n; |
| } |
| |
| /** |
| * Version of above that uses a copy constructor to initialize the n items |
| * to separate T values. |
| */ |
| T* push_back_n(int n, const T t[]) { |
| SkASSERT(n >= 0); |
| checkRealloc(n); |
| for (int i = 0; i < n; ++i) { |
| new (fItemArray + fCount + i) T(t[i]); |
| } |
| fCount += n; |
| return fItemArray + fCount - n; |
| } |
| |
| /** |
| * Removes the last element. Not safe to call when count() == 0. |
| */ |
| void pop_back() { |
| SkASSERT(fCount > 0); |
| --fCount; |
| fItemArray[fCount].~T(); |
| checkRealloc(0); |
| } |
| |
| /** |
| * Removes the last n elements. Not safe to call when count() < n. |
| */ |
| void pop_back_n(int n) { |
| SkASSERT(n >= 0); |
| SkASSERT(fCount >= n); |
| fCount -= n; |
| for (int i = 0; i < n; ++i) { |
| fItemArray[i].~T(); |
| } |
| checkRealloc(0); |
| } |
| |
| /** |
| * Pushes or pops from the back to resize. Pushes will be default |
| * initialized. |
| */ |
| void resize_back(int newCount) { |
| SkASSERT(newCount >= 0); |
| |
| if (newCount > fCount) { |
| push_back_n(newCount - fCount); |
| } else if (newCount < fCount) { |
| pop_back_n(fCount - newCount); |
| } |
| } |
| |
| /** |
| * Get the i^th element. |
| */ |
| T& operator[] (int i) { |
| SkASSERT(i < fCount); |
| SkASSERT(i >= 0); |
| return fItemArray[i]; |
| } |
| |
| const T& operator[] (int i) const { |
| SkASSERT(i < fCount); |
| SkASSERT(i >= 0); |
| return fItemArray[i]; |
| } |
| |
| /** |
| * equivalent to operator[](0) |
| */ |
| T& front() { SkASSERT(fCount > 0); return fItemArray[0];} |
| |
| const T& front() const { SkASSERT(fCount > 0); return fItemArray[0];} |
| |
| /** |
| * equivalent to operator[](count() - 1) |
| */ |
| T& back() { SkASSERT(fCount); return fItemArray[fCount - 1];} |
| |
| const T& back() const { SkASSERT(fCount > 0); return fItemArray[fCount - 1];} |
| |
| /** |
| * equivalent to operator[](count()-1-i) |
| */ |
| T& fromBack(int i) { |
| SkASSERT(i >= 0); |
| SkASSERT(i < fCount); |
| return fItemArray[fCount - i - 1]; |
| } |
| |
| const T& fromBack(int i) const { |
| SkASSERT(i >= 0); |
| SkASSERT(i < fCount); |
| return fItemArray[fCount - i - 1]; |
| } |
| |
| protected: |
| /** |
| * Creates an empty array that will use the passed storage block until it |
| * is insufficiently large to hold the entire array. |
| */ |
| template <int N> |
| SkTArray(SkAlignedSTStorage<N,T>* storage) { |
| this->init(NULL, 0, storage->get(), N); |
| } |
| |
| /** |
| * Copy another array, using preallocated storage if preAllocCount >= |
| * array.count(). Otherwise storage will only be used when array shrinks |
| * to fit. |
| */ |
| template <int N> |
| SkTArray(const SkTArray& array, SkAlignedSTStorage<N,T>* storage) { |
| this->init(array.fItemArray, array.fCount, storage->get(), N); |
| } |
| |
| /** |
| * Copy a C array, using preallocated storage if preAllocCount >= |
| * count. Otherwise storage will only be used when array shrinks |
| * to fit. |
| */ |
| template <int N> |
| SkTArray(const T* array, int count, SkAlignedSTStorage<N,T>* storage) { |
| this->init(array, count, storage->get(), N); |
| } |
| |
| void init(const T* array, int count, |
| void* preAllocStorage, int preAllocOrReserveCount) { |
| SkASSERT(count >= 0); |
| SkASSERT(preAllocOrReserveCount >= 0); |
| fCount = count; |
| fReserveCount = (preAllocOrReserveCount > 0) ? |
| preAllocOrReserveCount : |
| gMIN_ALLOC_COUNT; |
| fPreAllocMemArray = preAllocStorage; |
| if (fReserveCount >= fCount && |
| NULL != preAllocStorage) { |
| fAllocCount = fReserveCount; |
| fMemArray = preAllocStorage; |
| } else { |
| fAllocCount = SkMax32(fCount, fReserveCount); |
| fMemArray = sk_malloc_throw(fAllocCount * sizeof(T)); |
| } |
| |
| SkTArrayExt::copy(this, array); |
| } |
| |
| private: |
| |
| static const int gMIN_ALLOC_COUNT = 8; |
| |
| inline void checkRealloc(int delta) { |
| SkASSERT(fCount >= 0); |
| SkASSERT(fAllocCount >= 0); |
| |
| SkASSERT(-delta <= fCount); |
| |
| int newCount = fCount + delta; |
| int newAllocCount = fAllocCount; |
| |
| if (newCount > fAllocCount || newCount < (fAllocCount / 3)) { |
| // whether we're growing or shrinking, we leave at least 50% extra space for future |
| // growth (clamped to the reserve count). |
| newAllocCount = SkMax32(newCount + ((newCount + 1) >> 1), fReserveCount); |
| } |
| if (newAllocCount != fAllocCount) { |
| |
| fAllocCount = newAllocCount; |
| char* newMemArray; |
| |
| if (fAllocCount == fReserveCount && NULL != fPreAllocMemArray) { |
| newMemArray = (char*) fPreAllocMemArray; |
| } else { |
| newMemArray = (char*) sk_malloc_throw(fAllocCount*sizeof(T)); |
| } |
| |
| SkTArrayExt::copyAndDelete<T>(this, newMemArray); |
| |
| if (fMemArray != fPreAllocMemArray) { |
| sk_free(fMemArray); |
| } |
| fMemArray = newMemArray; |
| } |
| } |
| |
| template<typename X> friend void SkTArrayExt::copy(SkTArray<X, true>* that, const X*); |
| template<typename X> friend void SkTArrayExt::copyAndDelete(SkTArray<X, true>* that, char*); |
| |
| template<typename X> friend void SkTArrayExt::copy(SkTArray<X, false>* that, const X*); |
| template<typename X> friend void SkTArrayExt::copyAndDelete(SkTArray<X, false>* that, char*); |
| |
| int fReserveCount; |
| int fCount; |
| int fAllocCount; |
| void* fPreAllocMemArray; |
| union { |
| T* fItemArray; |
| void* fMemArray; |
| }; |
| }; |
| |
| /** |
| * Subclass of SkTArray that contains a preallocated memory block for the array. |
| */ |
| template <int N, typename T, bool DATA_TYPE = false> |
| class SkSTArray : public SkTArray<T, DATA_TYPE> { |
| private: |
| typedef SkTArray<T, DATA_TYPE> INHERITED; |
| |
| public: |
| SkSTArray() : INHERITED(&fStorage) { |
| } |
| |
| SkSTArray(const SkSTArray& array) |
| : INHERITED(array, &fStorage) { |
| } |
| |
| explicit SkSTArray(const INHERITED& array) |
| : INHERITED(array, &fStorage) { |
| } |
| |
| SkSTArray(const T* array, int count) |
| : INHERITED(array, count, &fStorage) { |
| } |
| |
| SkSTArray& operator= (const SkSTArray& array) { |
| return *this = *(const INHERITED*)&array; |
| } |
| |
| SkSTArray& operator= (const INHERITED& array) { |
| INHERITED::operator=(array); |
| return *this; |
| } |
| |
| private: |
| SkAlignedSTStorage<N,T> fStorage; |
| }; |
| |
| #endif |