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 #include "SkDeque.h"
11 
12 struct SkDeque::Block {
13     Block*  fNext;
14     Block*  fPrev;
15     char*   fBegin; // start of used section in this chunk
16     char*   fEnd;   // end of used section in this chunk
17     char*   fStop;  // end of the allocated chunk
18 
startSkDeque::Block19     char*       start() { return (char*)(this + 1); }
startSkDeque::Block20     const char* start() const { return (const char*)(this + 1); }
21 
initSkDeque::Block22     void init(size_t size) {
23         fNext   = fPrev = nullptr;
24         fBegin  = fEnd = nullptr;
25         fStop   = (char*)this + size;
26     }
27 };
28 
SkDeque(size_t elemSize,int allocCount)29 SkDeque::SkDeque(size_t elemSize, int allocCount)
30         : fElemSize(elemSize)
31         , fInitialStorage(nullptr)
32         , fCount(0)
33         , fAllocCount(allocCount) {
34     SkASSERT(allocCount >= 1);
35     fFrontBlock = fBackBlock = nullptr;
36     fFront = fBack = nullptr;
37 }
38 
SkDeque(size_t elemSize,void * storage,size_t storageSize,int allocCount)39 SkDeque::SkDeque(size_t elemSize, void* storage, size_t storageSize, int allocCount)
40         : fElemSize(elemSize)
41         , fInitialStorage(storage)
42         , fCount(0)
43         , fAllocCount(allocCount) {
44     SkASSERT(storageSize == 0 || storage != nullptr);
45     SkASSERT(allocCount >= 1);
46 
47     if (storageSize >= sizeof(Block) + elemSize) {
48         fFrontBlock = (Block*)storage;
49         fFrontBlock->init(storageSize);
50     } else {
51         fFrontBlock = nullptr;
52     }
53     fBackBlock = fFrontBlock;
54     fFront = fBack = nullptr;
55 }
56 
~SkDeque()57 SkDeque::~SkDeque() {
58     Block* head = fFrontBlock;
59     Block* initialHead = (Block*)fInitialStorage;
60 
61     while (head) {
62         Block* next = head->fNext;
63         if (head != initialHead) {
64             this->freeBlock(head);
65         }
66         head = next;
67     }
68 }
69 
push_front()70 void* SkDeque::push_front() {
71     fCount += 1;
72 
73     if (nullptr == fFrontBlock) {
74         fFrontBlock = this->allocateBlock(fAllocCount);
75         fBackBlock = fFrontBlock;     // update our linklist
76     }
77 
78     Block*  first = fFrontBlock;
79     char*   begin;
80 
81     if (nullptr == first->fBegin) {
82     INIT_CHUNK:
83         first->fEnd = first->fStop;
84         begin = first->fStop - fElemSize;
85     } else {
86         begin = first->fBegin - fElemSize;
87         if (begin < first->start()) {    // no more room in this chunk
88             // should we alloc more as we accumulate more elements?
89             first = this->allocateBlock(fAllocCount);
90             first->fNext = fFrontBlock;
91             fFrontBlock->fPrev = first;
92             fFrontBlock = first;
93             goto INIT_CHUNK;
94         }
95     }
96 
97     first->fBegin = begin;
98 
99     if (nullptr == fFront) {
100         SkASSERT(nullptr == fBack);
101         fFront = fBack = begin;
102     } else {
103         SkASSERT(fBack);
104         fFront = begin;
105     }
106 
107     return begin;
108 }
109 
push_back()110 void* SkDeque::push_back() {
111     fCount += 1;
112 
113     if (nullptr == fBackBlock) {
114         fBackBlock = this->allocateBlock(fAllocCount);
115         fFrontBlock = fBackBlock; // update our linklist
116     }
117 
118     Block*  last = fBackBlock;
119     char*   end;
120 
121     if (nullptr == last->fBegin) {
122     INIT_CHUNK:
123         last->fBegin = last->start();
124         end = last->fBegin + fElemSize;
125     } else {
126         end = last->fEnd + fElemSize;
127         if (end > last->fStop) {  // no more room in this chunk
128             // should we alloc more as we accumulate more elements?
129             last = this->allocateBlock(fAllocCount);
130             last->fPrev = fBackBlock;
131             fBackBlock->fNext = last;
132             fBackBlock = last;
133             goto INIT_CHUNK;
134         }
135     }
136 
137     last->fEnd = end;
138     end -= fElemSize;
139 
140     if (nullptr == fBack) {
141         SkASSERT(nullptr == fFront);
142         fFront = fBack = end;
143     } else {
144         SkASSERT(fFront);
145         fBack = end;
146     }
147 
148     return end;
149 }
150 
pop_front()151 void SkDeque::pop_front() {
152     SkASSERT(fCount > 0);
153     fCount -= 1;
154 
155     Block*  first = fFrontBlock;
156 
157     SkASSERT(first != nullptr);
158 
159     if (first->fBegin == nullptr) {  // we were marked empty from before
160         first = first->fNext;
161         first->fPrev = nullptr;
162         this->freeBlock(fFrontBlock);
163         fFrontBlock = first;
164         SkASSERT(first != nullptr);    // else we popped too far
165     }
166 
167     char* begin = first->fBegin + fElemSize;
168     SkASSERT(begin <= first->fEnd);
169 
170     if (begin < fFrontBlock->fEnd) {
171         first->fBegin = begin;
172         SkASSERT(first->fBegin);
173         fFront = first->fBegin;
174     } else {
175         first->fBegin = first->fEnd = nullptr;  // mark as empty
176         if (nullptr == first->fNext) {
177             fFront = fBack = nullptr;
178         } else {
179             SkASSERT(first->fNext->fBegin);
180             fFront = first->fNext->fBegin;
181         }
182     }
183 }
184 
pop_back()185 void SkDeque::pop_back() {
186     SkASSERT(fCount > 0);
187     fCount -= 1;
188 
189     Block* last = fBackBlock;
190 
191     SkASSERT(last != nullptr);
192 
193     if (last->fEnd == nullptr) {  // we were marked empty from before
194         last = last->fPrev;
195         last->fNext = nullptr;
196         this->freeBlock(fBackBlock);
197         fBackBlock = last;
198         SkASSERT(last != nullptr);  // else we popped too far
199     }
200 
201     char* end = last->fEnd - fElemSize;
202     SkASSERT(end >= last->fBegin);
203 
204     if (end > last->fBegin) {
205         last->fEnd = end;
206         SkASSERT(last->fEnd);
207         fBack = last->fEnd - fElemSize;
208     } else {
209         last->fBegin = last->fEnd = nullptr;    // mark as empty
210         if (nullptr == last->fPrev) {
211             fFront = fBack = nullptr;
212         } else {
213             SkASSERT(last->fPrev->fEnd);
214             fBack = last->fPrev->fEnd - fElemSize;
215         }
216     }
217 }
218 
numBlocksAllocated() const219 int SkDeque::numBlocksAllocated() const {
220     int numBlocks = 0;
221 
222     for (const Block* temp = fFrontBlock; temp; temp = temp->fNext) {
223         ++numBlocks;
224     }
225 
226     return numBlocks;
227 }
228 
allocateBlock(int allocCount)229 SkDeque::Block* SkDeque::allocateBlock(int allocCount) {
230     Block* newBlock = (Block*)sk_malloc_throw(sizeof(Block) + allocCount * fElemSize);
231     newBlock->init(sizeof(Block) + allocCount * fElemSize);
232     return newBlock;
233 }
234 
freeBlock(Block * block)235 void SkDeque::freeBlock(Block* block) {
236     sk_free(block);
237 }
238 
239 ///////////////////////////////////////////////////////////////////////////////
240 
Iter()241 SkDeque::Iter::Iter() : fCurBlock(nullptr), fPos(nullptr), fElemSize(0) {}
242 
Iter(const SkDeque & d,IterStart startLoc)243 SkDeque::Iter::Iter(const SkDeque& d, IterStart startLoc) {
244     this->reset(d, startLoc);
245 }
246 
247 // Due to how reset and next work, next actually returns the current element
248 // pointed to by fPos and then updates fPos to point to the next one.
next()249 void* SkDeque::Iter::next() {
250     char* pos = fPos;
251 
252     if (pos) {   // if we were valid, try to move to the next setting
253         char* next = pos + fElemSize;
254         SkASSERT(next <= fCurBlock->fEnd);
255         if (next == fCurBlock->fEnd) { // exhausted this chunk, move to next
256             do {
257                 fCurBlock = fCurBlock->fNext;
258             } while (fCurBlock != nullptr && fCurBlock->fBegin == nullptr);
259             next = fCurBlock ? fCurBlock->fBegin : nullptr;
260         }
261         fPos = next;
262     }
263     return pos;
264 }
265 
266 // Like next, prev actually returns the current element pointed to by fPos and
267 // then makes fPos point to the previous element.
prev()268 void* SkDeque::Iter::prev() {
269     char* pos = fPos;
270 
271     if (pos) {   // if we were valid, try to move to the prior setting
272         char* prev = pos - fElemSize;
273         SkASSERT(prev >= fCurBlock->fBegin - fElemSize);
274         if (prev < fCurBlock->fBegin) { // exhausted this chunk, move to prior
275             do {
276                 fCurBlock = fCurBlock->fPrev;
277             } while (fCurBlock != nullptr && fCurBlock->fEnd == nullptr);
278             prev = fCurBlock ? fCurBlock->fEnd - fElemSize : nullptr;
279         }
280         fPos = prev;
281     }
282     return pos;
283 }
284 
285 // reset works by skipping through the spare blocks at the start (or end)
286 // of the doubly linked list until a non-empty one is found. The fPos
287 // member is then set to the first (or last) element in the block. If
288 // there are no elements in the deque both fCurBlock and fPos will come
289 // out of this routine nullptr.
reset(const SkDeque & d,IterStart startLoc)290 void SkDeque::Iter::reset(const SkDeque& d, IterStart startLoc) {
291     fElemSize = d.fElemSize;
292 
293     if (kFront_IterStart == startLoc) {
294         // initialize the iterator to start at the front
295         fCurBlock = d.fFrontBlock;
296         while (fCurBlock && nullptr == fCurBlock->fBegin) {
297             fCurBlock = fCurBlock->fNext;
298         }
299         fPos = fCurBlock ? fCurBlock->fBegin : nullptr;
300     } else {
301         // initialize the iterator to start at the back
302         fCurBlock = d.fBackBlock;
303         while (fCurBlock && nullptr == fCurBlock->fEnd) {
304             fCurBlock = fCurBlock->fPrev;
305         }
306         fPos = fCurBlock ? fCurBlock->fEnd - fElemSize : nullptr;
307     }
308 }
309