diff options
Diffstat (limited to 'linden/indra/llcommon/lllinkedqueue.h')
-rw-r--r-- | linden/indra/llcommon/lllinkedqueue.h | 310 |
1 files changed, 310 insertions, 0 deletions
diff --git a/linden/indra/llcommon/lllinkedqueue.h b/linden/indra/llcommon/lllinkedqueue.h new file mode 100644 index 0000000..699667a --- /dev/null +++ b/linden/indra/llcommon/lllinkedqueue.h | |||
@@ -0,0 +1,310 @@ | |||
1 | /** | ||
2 | * @file lllinkedqueue.h | ||
3 | * @brief Declaration of linked queue classes. | ||
4 | * | ||
5 | * Copyright (c) 2003-2007, Linden Research, Inc. | ||
6 | * | ||
7 | * The source code in this file ("Source Code") is provided by Linden Lab | ||
8 | * to you under the terms of the GNU General Public License, version 2.0 | ||
9 | * ("GPL"), unless you have obtained a separate licensing agreement | ||
10 | * ("Other License"), formally executed by you and Linden Lab. Terms of | ||
11 | * the GPL can be found in doc/GPL-license.txt in this distribution, or | ||
12 | * online at http://secondlife.com/developers/opensource/gplv2 | ||
13 | * | ||
14 | * There are special exceptions to the terms and conditions of the GPL as | ||
15 | * it is applied to this Source Code. View the full text of the exception | ||
16 | * in the file doc/FLOSS-exception.txt in this software distribution, or | ||
17 | * online at http://secondlife.com/developers/opensource/flossexception | ||
18 | * | ||
19 | * By copying, modifying or distributing this software, you acknowledge | ||
20 | * that you have read and understood your obligations described above, | ||
21 | * and agree to abide by those obligations. | ||
22 | * | ||
23 | * ALL LINDEN LAB SOURCE CODE IS PROVIDED "AS IS." LINDEN LAB MAKES NO | ||
24 | * WARRANTIES, EXPRESS, IMPLIED OR OTHERWISE, REGARDING ITS ACCURACY, | ||
25 | * COMPLETENESS OR PERFORMANCE. | ||
26 | */ | ||
27 | |||
28 | #ifndef LL_LLLINKEDQUEUE_H | ||
29 | #define LL_LLLINKEDQUEUE_H | ||
30 | |||
31 | #include "llerror.h" | ||
32 | |||
33 | // node that actually contains the data | ||
34 | template <class DATA_TYPE> class LLLinkedQueueNode | ||
35 | { | ||
36 | public: | ||
37 | DATA_TYPE mData; | ||
38 | LLLinkedQueueNode *mNextp; | ||
39 | LLLinkedQueueNode *mPrevp; | ||
40 | |||
41 | |||
42 | public: | ||
43 | LLLinkedQueueNode(); | ||
44 | LLLinkedQueueNode(const DATA_TYPE data); | ||
45 | |||
46 | // destructor does not, by default, destroy associated data | ||
47 | // however, the mDatap must be NULL to ensure that we aren't causing memory leaks | ||
48 | ~LLLinkedQueueNode(); | ||
49 | }; | ||
50 | |||
51 | |||
52 | |||
53 | template <class DATA_TYPE> class LLLinkedQueue | ||
54 | { | ||
55 | |||
56 | public: | ||
57 | LLLinkedQueue(); | ||
58 | |||
59 | // destructor destroys list and nodes, but not data in nodes | ||
60 | ~LLLinkedQueue(); | ||
61 | |||
62 | // Puts at end of FIFO | ||
63 | void push(const DATA_TYPE data); | ||
64 | |||
65 | // Takes off front of FIFO | ||
66 | BOOL pop(DATA_TYPE &data); | ||
67 | BOOL peek(DATA_TYPE &data); | ||
68 | |||
69 | void reset(); | ||
70 | |||
71 | S32 getLength() const; | ||
72 | |||
73 | BOOL isEmpty() const; | ||
74 | |||
75 | BOOL remove(const DATA_TYPE data); | ||
76 | |||
77 | BOOL checkData(const DATA_TYPE data) const; | ||
78 | |||
79 | private: | ||
80 | // add node to end of list | ||
81 | // set mCurrentp to mQueuep | ||
82 | void addNodeAtEnd(LLLinkedQueueNode<DATA_TYPE> *nodep); | ||
83 | |||
84 | private: | ||
85 | LLLinkedQueueNode<DATA_TYPE> mHead; // head node | ||
86 | LLLinkedQueueNode<DATA_TYPE> mTail; // tail node | ||
87 | S32 mLength; | ||
88 | }; | ||
89 | |||
90 | |||
91 | // | ||
92 | // Nodes | ||
93 | // | ||
94 | |||
95 | template <class DATA_TYPE> | ||
96 | LLLinkedQueueNode<DATA_TYPE>::LLLinkedQueueNode() : | ||
97 | mData(), mNextp(NULL), mPrevp(NULL) | ||
98 | { } | ||
99 | |||
100 | template <class DATA_TYPE> | ||
101 | LLLinkedQueueNode<DATA_TYPE>::LLLinkedQueueNode(const DATA_TYPE data) : | ||
102 | mData(data), mNextp(NULL), mPrevp(NULL) | ||
103 | { } | ||
104 | |||
105 | template <class DATA_TYPE> | ||
106 | LLLinkedQueueNode<DATA_TYPE>::~LLLinkedQueueNode() | ||
107 | { } | ||
108 | |||
109 | |||
110 | // | ||
111 | // Queue itself | ||
112 | // | ||
113 | |||
114 | template <class DATA_TYPE> | ||
115 | LLLinkedQueue<DATA_TYPE>::LLLinkedQueue() | ||
116 | : mHead(), | ||
117 | mTail(), | ||
118 | mLength(0) | ||
119 | { } | ||
120 | |||
121 | |||
122 | // destructor destroys list and nodes, but not data in nodes | ||
123 | template <class DATA_TYPE> | ||
124 | LLLinkedQueue<DATA_TYPE>::~LLLinkedQueue() | ||
125 | { | ||
126 | reset(); | ||
127 | } | ||
128 | |||
129 | |||
130 | // put data into a node and stick it at the end of the list | ||
131 | template <class DATA_TYPE> | ||
132 | void LLLinkedQueue<DATA_TYPE>::push(const DATA_TYPE data) | ||
133 | { | ||
134 | // make the new node | ||
135 | LLLinkedQueueNode<DATA_TYPE> *nodep = new LLLinkedQueueNode<DATA_TYPE>(data); | ||
136 | |||
137 | addNodeAtEnd(nodep); | ||
138 | } | ||
139 | |||
140 | |||
141 | // search the list starting at mHead.mNextp and remove the link with mDatap == data | ||
142 | // set mCurrentp to mQueuep, or NULL if mQueuep points to node with mDatap == data | ||
143 | // return TRUE if found, FALSE if not found | ||
144 | template <class DATA_TYPE> | ||
145 | BOOL LLLinkedQueue<DATA_TYPE>::remove(const DATA_TYPE data) | ||
146 | { | ||
147 | BOOL b_found = FALSE; | ||
148 | |||
149 | LLLinkedQueueNode<DATA_TYPE> *currentp = mHead.mNextp; | ||
150 | |||
151 | while (currentp) | ||
152 | { | ||
153 | if (currentp->mData == data) | ||
154 | { | ||
155 | b_found = TRUE; | ||
156 | |||
157 | // if there is a next one, fix it | ||
158 | if (currentp->mNextp) | ||
159 | { | ||
160 | currentp->mNextp->mPrevp = currentp->mPrevp; | ||
161 | } | ||
162 | else // we are at end of list | ||
163 | { | ||
164 | mTail.mPrevp = currentp->mPrevp; | ||
165 | } | ||
166 | |||
167 | // if there is a previous one, fix it | ||
168 | if (currentp->mPrevp) | ||
169 | { | ||
170 | currentp->mPrevp->mNextp = currentp->mNextp; | ||
171 | } | ||
172 | else // we are at beginning of list | ||
173 | { | ||
174 | mHead.mNextp = currentp->mNextp; | ||
175 | } | ||
176 | |||
177 | // remove the node | ||
178 | delete currentp; | ||
179 | mLength--; | ||
180 | break; | ||
181 | } | ||
182 | currentp = currentp->mNextp; | ||
183 | } | ||
184 | |||
185 | return b_found; | ||
186 | } | ||
187 | |||
188 | |||
189 | // remove all nodes from the list but do not delete associated data | ||
190 | template <class DATA_TYPE> | ||
191 | void LLLinkedQueue<DATA_TYPE>::reset() | ||
192 | { | ||
193 | LLLinkedQueueNode<DATA_TYPE> *currentp; | ||
194 | LLLinkedQueueNode<DATA_TYPE> *nextp; | ||
195 | currentp = mHead.mNextp; | ||
196 | |||
197 | while (currentp) | ||
198 | { | ||
199 | nextp = currentp->mNextp; | ||
200 | delete currentp; | ||
201 | currentp = nextp; | ||
202 | } | ||
203 | |||
204 | // reset mHead and mCurrentp | ||
205 | mHead.mNextp = NULL; | ||
206 | mTail.mPrevp = NULL; | ||
207 | mLength = 0; | ||
208 | } | ||
209 | |||
210 | template <class DATA_TYPE> | ||
211 | S32 LLLinkedQueue<DATA_TYPE>::getLength() const | ||
212 | { | ||
213 | return mLength; | ||
214 | } | ||
215 | |||
216 | template <class DATA_TYPE> | ||
217 | BOOL LLLinkedQueue<DATA_TYPE>::isEmpty() const | ||
218 | { | ||
219 | return mLength <= 0; | ||
220 | } | ||
221 | |||
222 | // check to see if data is in list | ||
223 | // set mCurrentp and mQueuep to the target of search if found, otherwise set mCurrentp to mQueuep | ||
224 | // return TRUE if found, FALSE if not found | ||
225 | template <class DATA_TYPE> | ||
226 | BOOL LLLinkedQueue<DATA_TYPE>::checkData(const DATA_TYPE data) const | ||
227 | { | ||
228 | LLLinkedQueueNode<DATA_TYPE> *currentp = mHead.mNextp; | ||
229 | |||
230 | while (currentp) | ||
231 | { | ||
232 | if (currentp->mData == data) | ||
233 | { | ||
234 | return TRUE; | ||
235 | } | ||
236 | currentp = currentp->mNextp; | ||
237 | } | ||
238 | return FALSE; | ||
239 | } | ||
240 | |||
241 | template <class DATA_TYPE> | ||
242 | BOOL LLLinkedQueue<DATA_TYPE>::pop(DATA_TYPE &data) | ||
243 | { | ||
244 | LLLinkedQueueNode<DATA_TYPE> *currentp; | ||
245 | |||
246 | currentp = mHead.mNextp; | ||
247 | if (!currentp) | ||
248 | { | ||
249 | return FALSE; | ||
250 | } | ||
251 | |||
252 | mHead.mNextp = currentp->mNextp; | ||
253 | if (currentp->mNextp) | ||
254 | { | ||
255 | currentp->mNextp->mPrevp = currentp->mPrevp; | ||
256 | } | ||
257 | else | ||
258 | { | ||
259 | mTail.mPrevp = currentp->mPrevp; | ||
260 | } | ||
261 | |||
262 | data = currentp->mData; | ||
263 | delete currentp; | ||
264 | mLength--; | ||
265 | return TRUE; | ||
266 | } | ||
267 | |||
268 | template <class DATA_TYPE> | ||
269 | BOOL LLLinkedQueue<DATA_TYPE>::peek(DATA_TYPE &data) | ||
270 | { | ||
271 | LLLinkedQueueNode<DATA_TYPE> *currentp; | ||
272 | |||
273 | currentp = mHead.mNextp; | ||
274 | if (!currentp) | ||
275 | { | ||
276 | return FALSE; | ||
277 | } | ||
278 | data = currentp->mData; | ||
279 | return TRUE; | ||
280 | } | ||
281 | |||
282 | |||
283 | ////////////////////////////////////////////////////////////////////////////////////////// | ||
284 | // private members | ||
285 | ////////////////////////////////////////////////////////////////////////////////////////// | ||
286 | |||
287 | |||
288 | // add node to end of list | ||
289 | // set mCurrentp to mQueuep | ||
290 | template <class DATA_TYPE> | ||
291 | void LLLinkedQueue<DATA_TYPE>::addNodeAtEnd(LLLinkedQueueNode<DATA_TYPE> *nodep) | ||
292 | { | ||
293 | // add the node to the end of the list | ||
294 | nodep->mNextp = NULL; | ||
295 | nodep->mPrevp = mTail.mPrevp; | ||
296 | mTail.mPrevp = nodep; | ||
297 | |||
298 | // if there's something in the list, fix its back pointer | ||
299 | if (nodep->mPrevp) | ||
300 | { | ||
301 | nodep->mPrevp->mNextp = nodep; | ||
302 | } | ||
303 | else // otherwise fix the head node | ||
304 | { | ||
305 | mHead.mNextp = nodep; | ||
306 | } | ||
307 | mLength++; | ||
308 | } | ||
309 | |||
310 | #endif | ||