• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2     Copyright 2010 Google Inc.
3 
4     Licensed under the Apache License, Version 2.0 (the "License");
5     you may not use this file except in compliance with the License.
6     You may obtain a copy of the License at
7 
8          http://www.apache.org/licenses/LICENSE-2.0
9 
10     Unless required by applicable law or agreed to in writing, software
11     distributed under the License is distributed on an "AS IS" BASIS,
12     WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13     See the License for the specific language governing permissions and
14     limitations under the License.
15  */
16 
17 
18 #ifndef GrTDArray_DEFINED
19 #define GrTDArray_DEFINED
20 
21 #include "GrTypes.h"
22 
GrInitialArrayAllocationCount()23 static int GrInitialArrayAllocationCount() {
24     return 4;
25 }
26 
GrNextArrayAllocationCount(int count)27 static int GrNextArrayAllocationCount(int count) {
28     return count + ((count + 1) >> 1);
29 }
30 
31 template <typename T> class GrTDArray {
32 public:
GrTDArray()33     GrTDArray() : fArray(NULL), fAllocated(0), fCount(0) {}
GrTDArray(const GrTDArray & src)34     GrTDArray(const GrTDArray& src) {
35         fCount = fAllocated = src.fCount;
36         fArray = (T*)GrMalloc(fAllocated * sizeof(T));
37         memcpy(fArray, src.fArray, fCount * sizeof(T));
38     }
~GrTDArray()39     ~GrTDArray() {
40         if (fArray) {
41             GrFree(fArray);
42         }
43     }
44 
isEmpty()45     bool isEmpty() const { return 0 == fCount; }
count()46     int count() const { return fCount; }
47 
at(int index)48     const T& at(int index) const {
49         GrAssert((unsigned)index < (unsigned)fCount);
50         return fArray[index];
51     }
at(int index)52     T& at(int index) {
53         GrAssert((unsigned)index < (unsigned)fCount);
54         return fArray[index];
55     }
56 
57     const T& operator[](int index) const { return this->at(index); }
58     T& operator[](int index) { return this->at(index); }
59 
60     GrTDArray& operator=(const GrTDArray& src) {
61         if (fAllocated < src.fCount) {
62             fAllocated = src.fCount;
63             GrFree(fArray);
64             fArray = (T*)GrMalloc(fAllocated * sizeof(T));
65         }
66         fCount = src.fCount;
67         memcpy(fArray, src.fArray, fCount * sizeof(T));
68         return *this;
69     }
70 
reset()71     void reset() {
72         if (fArray) {
73             GrFree(fArray);
74             fArray = NULL;
75         }
76         fAllocated = fCount = 0;
77     }
78 
begin()79     T* begin() const { return fArray; }
end()80     T* end() const { return fArray + fCount; }
back()81     T* back() const { GrAssert(fCount); return fArray + (fCount - 1); }
82 
prepend()83     T* prepend() {
84         this->growAt(0);
85         return fArray;
86     }
87 
append()88     T* append() {
89         this->growAt(fCount);
90         return fArray + fCount - 1;
91     }
92 
93     /**
94      *  index may be [0..count], so that you can insert at the end (like append)
95      */
insert(int index)96     T* insert(int index) {
97         GrAssert((unsigned)index <= (unsigned)fCount);
98         this->growAt(index);
99         return fArray + index;
100     }
101 
remove(int index)102     void remove(int index) {
103         GrAssert((unsigned)index < (unsigned)fCount);
104         fCount -= 1;
105         if (index < fCount) {
106             int remaining = fCount - index;
107             memmove(fArray + index, fArray + index + 1, remaining * sizeof(T));
108         }
109     }
110 
removeShuffle(int index)111     void removeShuffle(int index) {
112         GrAssert((unsigned)index < (unsigned)fCount);
113         fCount -= 1;
114         if (index < fCount) {
115             memmove(fArray + index, fArray + fCount, sizeof(T));
116         }
117     }
118 
119     // Utility iterators
120 
121     /**
122      *  Calls GrFree() on each element. Assumes each is NULL or was allocated
123      *  with GrMalloc().
124      */
freeAll()125     void freeAll() {
126         T* stop = this->end();
127         for (T* curr = this->begin(); curr < stop; curr++) {
128             GrFree(*curr);
129         }
130         this->reset();
131     }
132 
133     /**
134      *  Calls delete on each element. Assumes each is NULL or was allocated
135      *  with new.
136      */
deleteAll()137     void deleteAll() {
138         T* stop = this->end();
139         for (T* curr = this->begin(); curr < stop; curr++) {
140             delete *curr;
141         }
142         this->reset();
143     }
144 
145     /**
146      *  Calls GrSafeUnref() on each element. Assumes each is NULL or is a
147      *  subclass of GrRefCnt.
148      */
unrefAll()149     void unrefAll() {
150         T* stop = this->end();
151         for (T* curr = this->begin(); curr < stop; curr++) {
152             GrSafeUnref(*curr);
153         }
154         this->reset();
155     }
156 
visit(void visitor (T &))157     void visit(void visitor(T&)) const {
158         T* stop = this->end();
159         for (T* curr = this->begin(); curr < stop; curr++) {
160             if (*curr) {
161                 visitor(*curr);
162             }
163         }
164     }
165 
find(const T & elem)166     int find(const T& elem) const {
167         int count = this->count();
168         T* curr = this->begin();
169         for (int i = 0; i < count; i++) {
170             if (elem == curr[i]) {
171                 return i;
172             }
173         }
174         return -1;
175     }
176 
177     friend bool operator==(const GrTDArray<T>& a, const GrTDArray<T>& b) {
178         return a.count() == b.count() &&
179                (0 == a.count() ||
180                 0 == memcmp(a.begin(), b.begin(), a.count() * sizeof(T)));
181     }
182     friend bool operator!=(const GrTDArray<T>& a, const GrTDArray<T>& b) {
183         return !(a == b);
184     }
185 
186 private:
187     T*  fArray;
188     int fAllocated, fCount;
189 
190     // growAt will increment fCount, reallocate fArray (as needed), and slide
191     // the contents of fArray to make a hole for new data at index.
growAt(int index)192     void growAt(int index) {
193         GrAssert(fCount <= fAllocated);
194         if (0 == fAllocated) {
195             fAllocated = GrInitialArrayAllocationCount();
196             fArray = (T*)GrMalloc(fAllocated * sizeof(T));
197         } else if (fCount == fAllocated) {
198             fAllocated = GrNextArrayAllocationCount(fAllocated);
199             T* newArray = (T*)GrMalloc(fAllocated * sizeof(T));
200             memcpy(newArray, fArray, index * sizeof(T));
201             memcpy(newArray + index + 1, fArray + index,
202                    (fCount - index) * sizeof(T));
203             GrFree(fArray);
204             fArray = newArray;
205         } else {
206             // check that we're not just appending
207             if (index < fCount) {
208                 memmove(fArray + index + 1, fArray + index,
209                         (fCount - index) * sizeof(T));
210             }
211         }
212         GrAssert(fCount < fAllocated);
213         fCount += 1;
214     }
215 };
216 
217 extern void* GrTDArray_growAt(void*, int* allocated, int& count, int index,
218                               size_t);
219 
220 
221 #endif
222 
223