Home | History | Annotate | Download | only in private
      1 
      2 /*
      3  * Copyright 2006 The Android Open Source Project
      4  *
      5  * Use of this source code is governed by a BSD-style license that can be
      6  * found in the LICENSE file.
      7  */
      8 
      9 
     10 #ifndef SkTDArray_DEFINED
     11 #define SkTDArray_DEFINED
     12 
     13 #include "SkTypes.h"
     14 #include "SkMalloc.h"
     15 
     16 template <typename T> class SkTDArray {
     17 public:
     18     SkTDArray() : fArray(nullptr), fReserve(0), fCount(0) {}
     19     SkTDArray(const T src[], int count) {
     20         SkASSERT(src || count == 0);
     21 
     22         fReserve = fCount = 0;
     23         fArray = NULL;
     24         if (count) {
     25             fArray = (T*)sk_malloc_throw(count * sizeof(T));
     26             memcpy(fArray, src, sizeof(T) * count);
     27             fReserve = fCount = count;
     28         }
     29     }
     30     SkTDArray(const SkTDArray<T>& src) : fArray(nullptr), fReserve(0), fCount(0) {
     31         SkTDArray<T> tmp(src.fArray, src.fCount);
     32         this->swap(tmp);
     33     }
     34     SkTDArray(SkTDArray<T>&& src) : fArray(nullptr), fReserve(0), fCount(0) {
     35         this->swap(src);
     36     }
     37     ~SkTDArray() {
     38         sk_free(fArray);
     39     }
     40 
     41     SkTDArray<T>& operator=(const SkTDArray<T>& src) {
     42         if (this != &src) {
     43             if (src.fCount > fReserve) {
     44                 SkTDArray<T> tmp(src.fArray, src.fCount);
     45                 this->swap(tmp);
     46             } else {
     47                 sk_careful_memcpy(fArray, src.fArray, sizeof(T) * src.fCount);
     48                 fCount = src.fCount;
     49             }
     50         }
     51         return *this;
     52     }
     53     SkTDArray<T>& operator=(SkTDArray<T>&& src) {
     54         if (this != &src) {
     55             this->swap(src);
     56             src.reset();
     57         }
     58         return *this;
     59     }
     60 
     61     friend bool operator==(const SkTDArray<T>& a, const SkTDArray<T>& b) {
     62         return  a.fCount == b.fCount &&
     63                 (a.fCount == 0 ||
     64                  !memcmp(a.fArray, b.fArray, a.fCount * sizeof(T)));
     65     }
     66     friend bool operator!=(const SkTDArray<T>& a, const SkTDArray<T>& b) {
     67         return !(a == b);
     68     }
     69 
     70     void swap(SkTDArray<T>& other) {
     71         SkTSwap(fArray, other.fArray);
     72         SkTSwap(fReserve, other.fReserve);
     73         SkTSwap(fCount, other.fCount);
     74     }
     75 
     76     // The deleter that ought to be used for a std:: smart pointer that takes ownership from
     77     // release().
     78     struct Deleter {
     79         void operator()(const void* p) { sk_free((void*)p); }
     80     };
     81 
     82     /** Return a ptr to the array of data, to be freed with sk_free. This also
     83         resets the SkTDArray to be empty.
     84      */
     85     T* release() {
     86         T* array = fArray;
     87         fArray = NULL;
     88         fReserve = fCount = 0;
     89         return array;
     90     }
     91 
     92     bool isEmpty() const { return fCount == 0; }
     93 
     94     /**
     95      *  Return the number of elements in the array
     96      */
     97     int count() const { return fCount; }
     98 
     99     /**
    100      *  Return the total number of elements allocated.
    101      *  reserved() - count() gives you the number of elements you can add
    102      *  without causing an allocation.
    103      */
    104     int reserved() const { return fReserve; }
    105 
    106     /**
    107      *  return the number of bytes in the array: count * sizeof(T)
    108      */
    109     size_t bytes() const { return fCount * sizeof(T); }
    110 
    111     T*  begin() { return fArray; }
    112     const T*  begin() const { return fArray; }
    113     T*  end() { return fArray ? fArray + fCount : NULL; }
    114     const T*  end() const { return fArray ? fArray + fCount : NULL; }
    115 
    116     T&  operator[](int index) {
    117         SkASSERT(index < fCount);
    118         return fArray[index];
    119     }
    120     const T&  operator[](int index) const {
    121         SkASSERT(index < fCount);
    122         return fArray[index];
    123     }
    124 
    125     T&  getAt(int index)  {
    126         return (*this)[index];
    127     }
    128     const T&  getAt(int index) const {
    129         return (*this)[index];
    130     }
    131 
    132     void reset() {
    133         if (fArray) {
    134             sk_free(fArray);
    135             fArray = NULL;
    136             fReserve = fCount = 0;
    137         } else {
    138             SkASSERT(fReserve == 0 && fCount == 0);
    139         }
    140     }
    141 
    142     void rewind() {
    143         // same as setCount(0)
    144         fCount = 0;
    145     }
    146 
    147     /**
    148      *  Sets the number of elements in the array.
    149      *  If the array does not have space for count elements, it will increase
    150      *  the storage allocated to some amount greater than that required.
    151      *  It will never shrink the storage.
    152      */
    153     void setCount(int count) {
    154         SkASSERT(count >= 0);
    155         if (count > fReserve) {
    156             this->resizeStorageToAtLeast(count);
    157         }
    158         fCount = count;
    159     }
    160 
    161     void setReserve(int reserve) {
    162         if (reserve > fReserve) {
    163             this->resizeStorageToAtLeast(reserve);
    164         }
    165     }
    166 
    167     T* prepend() {
    168         this->adjustCount(1);
    169         memmove(fArray + 1, fArray, (fCount - 1) * sizeof(T));
    170         return fArray;
    171     }
    172 
    173     T* append() {
    174         return this->append(1, NULL);
    175     }
    176     T* append(int count, const T* src = NULL) {
    177         int oldCount = fCount;
    178         if (count)  {
    179             SkASSERT(src == NULL || fArray == NULL ||
    180                     src + count <= fArray || fArray + oldCount <= src);
    181 
    182             this->adjustCount(count);
    183             if (src) {
    184                 memcpy(fArray + oldCount, src, sizeof(T) * count);
    185             }
    186         }
    187         return fArray + oldCount;
    188     }
    189 
    190     T* appendClear() {
    191         T* result = this->append();
    192         *result = 0;
    193         return result;
    194     }
    195 
    196     T* insert(int index) {
    197         return this->insert(index, 1, NULL);
    198     }
    199     T* insert(int index, int count, const T* src = NULL) {
    200         SkASSERT(count);
    201         SkASSERT(index <= fCount);
    202         size_t oldCount = fCount;
    203         this->adjustCount(count);
    204         T* dst = fArray + index;
    205         memmove(dst + count, dst, sizeof(T) * (oldCount - index));
    206         if (src) {
    207             memcpy(dst, src, sizeof(T) * count);
    208         }
    209         return dst;
    210     }
    211 
    212     void remove(int index, int count = 1) {
    213         SkASSERT(index + count <= fCount);
    214         fCount = fCount - count;
    215         memmove(fArray + index, fArray + index + count, sizeof(T) * (fCount - index));
    216     }
    217 
    218     void removeShuffle(int index) {
    219         SkASSERT(index < fCount);
    220         int newCount = fCount - 1;
    221         fCount = newCount;
    222         if (index != newCount) {
    223             memcpy(fArray + index, fArray + newCount, sizeof(T));
    224         }
    225     }
    226 
    227     template <typename S> int select(S&& selector) const {
    228         const T* iter = fArray;
    229         const T* stop = fArray + fCount;
    230 
    231         for (; iter < stop; iter++) {
    232             if (selector(*iter)) {
    233                 return SkToInt(iter - fArray);
    234             }
    235         }
    236         return -1;
    237     }
    238 
    239     int find(const T& elem) const {
    240         const T* iter = fArray;
    241         const T* stop = fArray + fCount;
    242 
    243         for (; iter < stop; iter++) {
    244             if (*iter == elem) {
    245                 return SkToInt(iter - fArray);
    246             }
    247         }
    248         return -1;
    249     }
    250 
    251     int rfind(const T& elem) const {
    252         const T* iter = fArray + fCount;
    253         const T* stop = fArray;
    254 
    255         while (iter > stop) {
    256             if (*--iter == elem) {
    257                 return SkToInt(iter - stop);
    258             }
    259         }
    260         return -1;
    261     }
    262 
    263     /**
    264      * Returns true iff the array contains this element.
    265      */
    266     bool contains(const T& elem) const {
    267         return (this->find(elem) >= 0);
    268     }
    269 
    270     /**
    271      * Copies up to max elements into dst. The number of items copied is
    272      * capped by count - index. The actual number copied is returned.
    273      */
    274     int copyRange(T* dst, int index, int max) const {
    275         SkASSERT(max >= 0);
    276         SkASSERT(!max || dst);
    277         if (index >= fCount) {
    278             return 0;
    279         }
    280         int count = SkMin32(max, fCount - index);
    281         memcpy(dst, fArray + index, sizeof(T) * count);
    282         return count;
    283     }
    284 
    285     void copy(T* dst) const {
    286         this->copyRange(dst, 0, fCount);
    287     }
    288 
    289     // routines to treat the array like a stack
    290     T*       push() { return this->append(); }
    291     void     push(const T& elem) { *this->append() = elem; }
    292     const T& top() const { return (*this)[fCount - 1]; }
    293     T&       top() { return (*this)[fCount - 1]; }
    294     void     pop(T* elem) { SkASSERT(fCount > 0); if (elem) *elem = (*this)[fCount - 1]; --fCount; }
    295     void     pop() { SkASSERT(fCount > 0); --fCount; }
    296 
    297     void deleteAll() {
    298         T*  iter = fArray;
    299         T*  stop = fArray + fCount;
    300         while (iter < stop) {
    301             delete *iter;
    302             iter += 1;
    303         }
    304         this->reset();
    305     }
    306 
    307     void freeAll() {
    308         T*  iter = fArray;
    309         T*  stop = fArray + fCount;
    310         while (iter < stop) {
    311             sk_free(*iter);
    312             iter += 1;
    313         }
    314         this->reset();
    315     }
    316 
    317     void unrefAll() {
    318         T*  iter = fArray;
    319         T*  stop = fArray + fCount;
    320         while (iter < stop) {
    321             (*iter)->unref();
    322             iter += 1;
    323         }
    324         this->reset();
    325     }
    326 
    327     void safeUnrefAll() {
    328         T*  iter = fArray;
    329         T*  stop = fArray + fCount;
    330         while (iter < stop) {
    331             SkSafeUnref(*iter);
    332             iter += 1;
    333         }
    334         this->reset();
    335     }
    336 
    337     void visitAll(void visitor(T&)) {
    338         T* stop = this->end();
    339         for (T* curr = this->begin(); curr < stop; curr++) {
    340             if (*curr) {
    341                 visitor(*curr);
    342             }
    343         }
    344     }
    345 
    346 #ifdef SK_DEBUG
    347     void validate() const {
    348         SkASSERT((fReserve == 0 && fArray == NULL) ||
    349                  (fReserve > 0 && fArray != NULL));
    350         SkASSERT(fCount <= fReserve);
    351     }
    352 #endif
    353 
    354     void shrinkToFit() {
    355         fReserve = fCount;
    356         fArray = (T*)sk_realloc_throw(fArray, fReserve * sizeof(T));
    357     }
    358 
    359 private:
    360     T*      fArray;
    361     int     fReserve;
    362     int     fCount;
    363 
    364     /**
    365      *  Adjusts the number of elements in the array.
    366      *  This is the same as calling setCount(count() + delta).
    367      */
    368     void adjustCount(int delta) {
    369         this->setCount(fCount + delta);
    370     }
    371 
    372     /**
    373      *  Increase the storage allocation such that it can hold (fCount + extra)
    374      *  elements.
    375      *  It never shrinks the allocation, and it may increase the allocation by
    376      *  more than is strictly required, based on a private growth heuristic.
    377      *
    378      *  note: does NOT modify fCount
    379      */
    380     void resizeStorageToAtLeast(int count) {
    381         SkASSERT(count > fReserve);
    382         fReserve = count + 4;
    383         fReserve += fReserve / 4;
    384         fArray = (T*)sk_realloc_throw(fArray, fReserve * sizeof(T));
    385     }
    386 };
    387 
    388 #endif
    389