Index: include/core/SkTemplates.h |
diff --git a/include/core/SkTemplates.h b/include/core/SkTemplates.h |
deleted file mode 100644 |
index 17dcd596334bbc0f137c22623eae9866faa93af7..0000000000000000000000000000000000000000 |
--- a/include/core/SkTemplates.h |
+++ /dev/null |
@@ -1,557 +0,0 @@ |
- |
-/* |
- * Copyright 2006 The Android Open Source Project |
- * |
- * Use of this source code is governed by a BSD-style license that can be |
- * found in the LICENSE file. |
- */ |
- |
- |
-#ifndef SkTemplates_DEFINED |
-#define SkTemplates_DEFINED |
- |
-#include "../private/SkTLogic.h" |
-#include "SkMath.h" |
-#include "SkTypes.h" |
-#include <limits.h> |
-#include <new> |
- |
-/** \file SkTemplates.h |
- |
- This file contains light-weight template classes for type-safe and exception-safe |
- resource management. |
-*/ |
- |
-/** |
- * Marks a local variable as known to be unused (to avoid warnings). |
- * Note that this does *not* prevent the local variable from being optimized away. |
- */ |
-template<typename T> inline void sk_ignore_unused_variable(const T&) { } |
- |
-namespace skstd { |
- |
-template <typename T> inline remove_reference_t<T>&& move(T&& t) { |
- return static_cast<remove_reference_t<T>&&>(t); |
-} |
- |
-template <typename T> inline T&& forward(remove_reference_t<T>& t) /*noexcept*/ { |
- return static_cast<T&&>(t); |
-} |
-template <typename T> inline T&& forward(remove_reference_t<T>&& t) /*noexcept*/ { |
- static_assert(!is_lvalue_reference<T>::value, |
- "Forwarding an rvalue reference as an lvalue reference is not allowed."); |
- return static_cast<T&&>(t); |
-} |
- |
-} // namespace skstd |
- |
-///@{ |
-/** SkTConstType<T, CONST>::type will be 'const T' if CONST is true, 'T' otherwise. */ |
-template <typename T, bool CONST> struct SkTConstType { |
- typedef T type; |
-}; |
-template <typename T> struct SkTConstType<T, true> { |
- typedef const T type; |
-}; |
-///@} |
- |
-/** |
- * Returns a pointer to a D which comes immediately after S[count]. |
- */ |
-template <typename D, typename S> static D* SkTAfter(S* ptr, size_t count = 1) { |
- return reinterpret_cast<D*>(ptr + count); |
-} |
- |
-/** |
- * Returns a pointer to a D which comes byteOffset bytes after S. |
- */ |
-template <typename D, typename S> static D* SkTAddOffset(S* ptr, size_t byteOffset) { |
- // The intermediate char* has the same const-ness as D as this produces better error messages. |
- // This relies on the fact that reinterpret_cast can add constness, but cannot remove it. |
- return reinterpret_cast<D*>( |
- reinterpret_cast<typename SkTConstType<char, SkTIsConst<D>::value>::type*>(ptr) + byteOffset |
- ); |
-} |
- |
-/** \class SkAutoTCallVProc |
- |
- Call a function when this goes out of scope. The template uses two |
- parameters, the object, and a function that is to be called in the destructor. |
- If detach() is called, the object reference is set to null. If the object |
- reference is null when the destructor is called, we do not call the |
- function. |
-*/ |
-template <typename T, void (*P)(T*)> class SkAutoTCallVProc : SkNoncopyable { |
-public: |
- SkAutoTCallVProc(T* obj): fObj(obj) {} |
- ~SkAutoTCallVProc() { if (fObj) P(fObj); } |
- |
- operator T*() const { return fObj; } |
- T* operator->() const { SkASSERT(fObj); return fObj; } |
- |
- T* detach() { T* obj = fObj; fObj = NULL; return obj; } |
- void reset(T* obj = NULL) { |
- if (fObj != obj) { |
- if (fObj) { |
- P(fObj); |
- } |
- fObj = obj; |
- } |
- } |
-private: |
- T* fObj; |
-}; |
- |
-/** \class SkAutoTCallIProc |
- |
-Call a function when this goes out of scope. The template uses two |
-parameters, the object, and a function that is to be called in the destructor. |
-If detach() is called, the object reference is set to null. If the object |
-reference is null when the destructor is called, we do not call the |
-function. |
-*/ |
-template <typename T, int (*P)(T*)> class SkAutoTCallIProc : SkNoncopyable { |
-public: |
- SkAutoTCallIProc(T* obj): fObj(obj) {} |
- ~SkAutoTCallIProc() { if (fObj) P(fObj); } |
- |
- operator T*() const { return fObj; } |
- T* operator->() const { SkASSERT(fObj); return fObj; } |
- |
- T* detach() { T* obj = fObj; fObj = NULL; return obj; } |
-private: |
- T* fObj; |
-}; |
- |
-/** \class SkAutoTDelete |
- An SkAutoTDelete<T> is like a T*, except that the destructor of SkAutoTDelete<T> |
- automatically deletes the pointer it holds (if any). That is, SkAutoTDelete<T> |
- owns the T object that it points to. Like a T*, an SkAutoTDelete<T> may hold |
- either NULL or a pointer to a T object. Also like T*, SkAutoTDelete<T> is |
- thread-compatible, and once you dereference it, you get the threadsafety |
- guarantees of T. |
- |
- The size of a SkAutoTDelete is small: sizeof(SkAutoTDelete<T>) == sizeof(T*) |
-*/ |
-template <typename T> class SkAutoTDelete : SkNoncopyable { |
-public: |
- SkAutoTDelete(T* obj = NULL) : fObj(obj) {} |
- ~SkAutoTDelete() { SkDELETE(fObj); } |
- |
- T* get() const { return fObj; } |
- operator T*() const { return fObj; } |
- T& operator*() const { SkASSERT(fObj); return *fObj; } |
- T* operator->() const { SkASSERT(fObj); return fObj; } |
- |
- void reset(T* obj) { |
- if (fObj != obj) { |
- SkDELETE(fObj); |
- fObj = obj; |
- } |
- } |
- |
- /** |
- * Delete the owned object, setting the internal pointer to NULL. |
- */ |
- void free() { |
- SkDELETE(fObj); |
- fObj = NULL; |
- } |
- |
- /** |
- * Transfer ownership of the object to the caller, setting the internal |
- * pointer to NULL. Note that this differs from get(), which also returns |
- * the pointer, but it does not transfer ownership. |
- */ |
- T* detach() { |
- T* obj = fObj; |
- fObj = NULL; |
- return obj; |
- } |
- |
- void swap(SkAutoTDelete* that) { |
- SkTSwap(fObj, that->fObj); |
- } |
- |
-private: |
- T* fObj; |
-}; |
- |
-// Calls ~T() in the destructor. |
-template <typename T> class SkAutoTDestroy : SkNoncopyable { |
-public: |
- SkAutoTDestroy(T* obj = NULL) : fObj(obj) {} |
- ~SkAutoTDestroy() { |
- if (fObj) { |
- fObj->~T(); |
- } |
- } |
- |
- T* get() const { return fObj; } |
- T& operator*() const { SkASSERT(fObj); return *fObj; } |
- T* operator->() const { SkASSERT(fObj); return fObj; } |
- |
-private: |
- T* fObj; |
-}; |
- |
-template <typename T> class SkAutoTDeleteArray : SkNoncopyable { |
-public: |
- SkAutoTDeleteArray(T array[]) : fArray(array) {} |
- ~SkAutoTDeleteArray() { SkDELETE_ARRAY(fArray); } |
- |
- T* get() const { return fArray; } |
- void free() { SkDELETE_ARRAY(fArray); fArray = NULL; } |
- T* detach() { T* array = fArray; fArray = NULL; return array; } |
- |
- void reset(T array[]) { |
- if (fArray != array) { |
- SkDELETE_ARRAY(fArray); |
- fArray = array; |
- } |
- } |
- |
-private: |
- T* fArray; |
-}; |
- |
-/** Allocate an array of T elements, and free the array in the destructor |
- */ |
-template <typename T> class SkAutoTArray : SkNoncopyable { |
-public: |
- SkAutoTArray() { |
- fArray = NULL; |
- SkDEBUGCODE(fCount = 0;) |
- } |
- /** Allocate count number of T elements |
- */ |
- explicit SkAutoTArray(int count) { |
- SkASSERT(count >= 0); |
- fArray = NULL; |
- if (count) { |
- fArray = SkNEW_ARRAY(T, count); |
- } |
- SkDEBUGCODE(fCount = count;) |
- } |
- |
- /** Reallocates given a new count. Reallocation occurs even if new count equals old count. |
- */ |
- void reset(int count) { |
- SkDELETE_ARRAY(fArray); |
- SkASSERT(count >= 0); |
- fArray = NULL; |
- if (count) { |
- fArray = SkNEW_ARRAY(T, count); |
- } |
- SkDEBUGCODE(fCount = count;) |
- } |
- |
- ~SkAutoTArray() { |
- SkDELETE_ARRAY(fArray); |
- } |
- |
- /** Return the array of T elements. Will be NULL if count == 0 |
- */ |
- T* get() const { return fArray; } |
- |
- /** Return the nth element in the array |
- */ |
- T& operator[](int index) const { |
- SkASSERT((unsigned)index < (unsigned)fCount); |
- return fArray[index]; |
- } |
- |
- void swap(SkAutoTArray& other) { |
- SkTSwap(fArray, other.fArray); |
- SkDEBUGCODE(SkTSwap(fCount, other.fCount)); |
- } |
- |
-private: |
- T* fArray; |
- SkDEBUGCODE(int fCount;) |
-}; |
- |
-/** Wraps SkAutoTArray, with room for up to N elements preallocated |
- */ |
-template <int N, typename T> class SkAutoSTArray : SkNoncopyable { |
-public: |
- /** Initialize with no objects */ |
- SkAutoSTArray() { |
- fArray = NULL; |
- fCount = 0; |
- } |
- |
- /** Allocate count number of T elements |
- */ |
- SkAutoSTArray(int count) { |
- fArray = NULL; |
- fCount = 0; |
- this->reset(count); |
- } |
- |
- ~SkAutoSTArray() { |
- this->reset(0); |
- } |
- |
- /** Destroys previous objects in the array and default constructs count number of objects */ |
- void reset(int count) { |
- T* start = fArray; |
- T* iter = start + fCount; |
- while (iter > start) { |
- (--iter)->~T(); |
- } |
- |
- if (fCount != count) { |
- if (fCount > N) { |
- // 'fArray' was allocated last time so free it now |
- SkASSERT((T*) fStorage != fArray); |
- sk_free(fArray); |
- } |
- |
- if (count > N) { |
- const uint64_t size64 = sk_64_mul(count, sizeof(T)); |
- const size_t size = static_cast<size_t>(size64); |
- if (size != size64) { |
- sk_out_of_memory(); |
- } |
- fArray = (T*) sk_malloc_throw(size); |
- } else if (count > 0) { |
- fArray = (T*) fStorage; |
- } else { |
- fArray = NULL; |
- } |
- |
- fCount = count; |
- } |
- |
- iter = fArray; |
- T* stop = fArray + count; |
- while (iter < stop) { |
- SkNEW_PLACEMENT(iter++, T); |
- } |
- } |
- |
- /** Return the number of T elements in the array |
- */ |
- int count() const { return fCount; } |
- |
- /** Return the array of T elements. Will be NULL if count == 0 |
- */ |
- T* get() const { return fArray; } |
- |
- /** Return the nth element in the array |
- */ |
- T& operator[](int index) const { |
- SkASSERT(index < fCount); |
- return fArray[index]; |
- } |
- |
-private: |
- int fCount; |
- T* fArray; |
- // since we come right after fArray, fStorage should be properly aligned |
- char fStorage[N * sizeof(T)]; |
-}; |
- |
-/** Manages an array of T elements, freeing the array in the destructor. |
- * Does NOT call any constructors/destructors on T (T must be POD). |
- */ |
-template <typename T> class SkAutoTMalloc : SkNoncopyable { |
-public: |
- /** Takes ownership of the ptr. The ptr must be a value which can be passed to sk_free. */ |
- explicit SkAutoTMalloc(T* ptr = NULL) { |
- fPtr = ptr; |
- } |
- |
- /** Allocates space for 'count' Ts. */ |
- explicit SkAutoTMalloc(size_t count) { |
- fPtr = (T*)sk_malloc_flags(count * sizeof(T), SK_MALLOC_THROW); |
- } |
- |
- ~SkAutoTMalloc() { |
- sk_free(fPtr); |
- } |
- |
- /** Resize the memory area pointed to by the current ptr preserving contents. */ |
- void realloc(size_t count) { |
- fPtr = reinterpret_cast<T*>(sk_realloc_throw(fPtr, count * sizeof(T))); |
- } |
- |
- /** Resize the memory area pointed to by the current ptr without preserving contents. */ |
- void reset(size_t count) { |
- sk_free(fPtr); |
- fPtr = (T*)sk_malloc_flags(count * sizeof(T), SK_MALLOC_THROW); |
- } |
- |
- T* get() const { return fPtr; } |
- |
- operator T*() { |
- return fPtr; |
- } |
- |
- operator const T*() const { |
- return fPtr; |
- } |
- |
- T& operator[](int index) { |
- return fPtr[index]; |
- } |
- |
- const T& operator[](int index) const { |
- return fPtr[index]; |
- } |
- |
- /** |
- * Transfer ownership of the ptr to the caller, setting the internal |
- * pointer to NULL. Note that this differs from get(), which also returns |
- * the pointer, but it does not transfer ownership. |
- */ |
- T* detach() { |
- T* ptr = fPtr; |
- fPtr = NULL; |
- return ptr; |
- } |
- |
-private: |
- T* fPtr; |
-}; |
- |
-template <size_t N, typename T> class SkAutoSTMalloc : SkNoncopyable { |
-public: |
- SkAutoSTMalloc() : fPtr(fTStorage) {} |
- |
- SkAutoSTMalloc(size_t count) { |
- if (count > N) { |
- fPtr = (T*)sk_malloc_flags(count * sizeof(T), SK_MALLOC_THROW | SK_MALLOC_TEMP); |
- } else { |
- fPtr = fTStorage; |
- } |
- } |
- |
- ~SkAutoSTMalloc() { |
- if (fPtr != fTStorage) { |
- sk_free(fPtr); |
- } |
- } |
- |
- // doesn't preserve contents |
- T* reset(size_t count) { |
- if (fPtr != fTStorage) { |
- sk_free(fPtr); |
- } |
- if (count > N) { |
- fPtr = (T*)sk_malloc_throw(count * sizeof(T)); |
- } else { |
- fPtr = fTStorage; |
- } |
- return fPtr; |
- } |
- |
- T* get() const { return fPtr; } |
- |
- operator T*() { |
- return fPtr; |
- } |
- |
- operator const T*() const { |
- return fPtr; |
- } |
- |
- T& operator[](int index) { |
- return fPtr[index]; |
- } |
- |
- const T& operator[](int index) const { |
- return fPtr[index]; |
- } |
- |
- // Reallocs the array, can be used to shrink the allocation. Makes no attempt to be intelligent |
- void realloc(size_t count) { |
- if (count > N) { |
- if (fPtr == fTStorage) { |
- fPtr = (T*)sk_malloc_throw(count * sizeof(T)); |
- memcpy(fPtr, fTStorage, N * sizeof(T)); |
- } else { |
- fPtr = (T*)sk_realloc_throw(fPtr, count * sizeof(T)); |
- } |
- } else if (fPtr != fTStorage) { |
- fPtr = (T*)sk_realloc_throw(fPtr, count * sizeof(T)); |
- } |
- } |
- |
-private: |
- T* fPtr; |
- union { |
- uint32_t fStorage32[(N*sizeof(T) + 3) >> 2]; |
- T fTStorage[1]; // do NOT want to invoke T::T() |
- }; |
-}; |
- |
-////////////////////////////////////////////////////////////////////////////////////////////////// |
- |
-/** |
- * Pass the object and the storage that was offered during SkInPlaceNewCheck, and this will |
- * safely destroy (and free if it was dynamically allocated) the object. |
- */ |
-template <typename T> void SkInPlaceDeleteCheck(T* obj, void* storage) { |
- if (storage == obj) { |
- obj->~T(); |
- } else { |
- SkDELETE(obj); |
- } |
-} |
- |
-/** |
- * Allocates T, using storage if it is large enough, and allocating on the heap (via new) if |
- * storage is not large enough. |
- * |
- * obj = SkInPlaceNewCheck<Type>(storage, size); |
- * ... |
- * SkInPlaceDeleteCheck(obj, storage); |
- */ |
-template <typename T> T* SkInPlaceNewCheck(void* storage, size_t size) { |
- return (sizeof(T) <= size) ? new (storage) T : SkNEW(T); |
-} |
- |
-template <typename T, typename A1, typename A2, typename A3> |
-T* SkInPlaceNewCheck(void* storage, size_t size, const A1& a1, const A2& a2, const A3& a3) { |
- return (sizeof(T) <= size) ? new (storage) T(a1, a2, a3) : SkNEW_ARGS(T, (a1, a2, a3)); |
-} |
- |
-/** |
- * Reserves memory that is aligned on double and pointer boundaries. |
- * Hopefully this is sufficient for all practical purposes. |
- */ |
-template <size_t N> class SkAlignedSStorage : SkNoncopyable { |
-public: |
- size_t size() const { return N; } |
- void* get() { return fData; } |
- const void* get() const { return fData; } |
- |
-private: |
- union { |
- void* fPtr; |
- double fDouble; |
- char fData[N]; |
- }; |
-}; |
- |
-/** |
- * Reserves memory that is aligned on double and pointer boundaries. |
- * Hopefully this is sufficient for all practical purposes. Otherwise, |
- * we have to do some arcane trickery to determine alignment of non-POD |
- * types. Lifetime of the memory is the lifetime of the object. |
- */ |
-template <int N, typename T> class SkAlignedSTStorage : SkNoncopyable { |
-public: |
- /** |
- * Returns void* because this object does not initialize the |
- * memory. Use placement new for types that require a cons. |
- */ |
- void* get() { return fStorage.get(); } |
- const void* get() const { return fStorage.get(); } |
-private: |
- SkAlignedSStorage<sizeof(T)*N> fStorage; |
-}; |
- |
-#endif |