1 #ifndef _DEPOOLARRAY_H
2 #define _DEPOOLARRAY_H
3 /*-------------------------------------------------------------------------
4  * drawElements Memory Pool Library
5  * --------------------------------
6  *
7  * Copyright 2014 The Android Open Source Project
8  *
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  *
13  *      http://www.apache.org/licenses/LICENSE-2.0
14  *
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  *
21  *//*!
22  * \file
23  * \brief Memory pool array class.
24  *//*--------------------------------------------------------------------*/
25 
26 #include "deDefs.h"
27 #include "deMemPool.h"
28 
29 enum
30 {
31 	DE_ARRAY_ELEMENTS_PER_PAGE_LOG2	= 4		/*!< \internal 16 */
32 };
33 
34 /*--------------------------------------------------------------------*//*!
35  * \internal
36  * \brief Type-independent version of the array template class.
37  *//*--------------------------------------------------------------------*/
38 typedef struct dePoolArray_s
39 {
40 	deMemPool*		pool;				/*!< Pool from which all memory is allocated from.	*/
41 
42 	int				elementSize;		/*!< Size of the element (in bytes).				*/
43 	int				numElements;		/*!< Number of elements in the array.				*/
44 	int				capacity;			/*!< Number of allocated elements in the array.		*/
45 
46 	int				pageTableCapacity;	/*!< Size of the page table.						*/
47 	void**			pageTable;			/*!< Pointer to the page table.						*/
48 } dePoolArray;
49 
50 DE_BEGIN_EXTERN_C
51 
52 dePoolArray*	dePoolArray_create			(deMemPool* pool, int elementSize);
53 deBool			dePoolArray_reserve			(dePoolArray* arr, int capacity);
54 deBool			dePoolArray_setSize			(dePoolArray* arr, int size);
55 
56 void			dePoolArray_selfTest		(void);
57 
58 DE_END_EXTERN_C
59 
60 /*--------------------------------------------------------------------*//*!
61  * \brief Declare a template pool array class.
62  * \param TYPENAME	Type name of the declared array.
63  * \param VALUETYPE	Type of the value contained in the array.
64  *
65  * This macro declares a pool array with all the necessary functions for
66  * operating with it. All allocated memory is taken from the memory pool
67  * given to the constructor.
68  *
69  * The array is implemented by having a set of pages (which store the
70  * elements) and a page table with pointers to each of them. The pages
71  * are allocated individually whenever they are needed, but the page
72  * table grows exponentially. This keeps the memory overhead for large
73  * arrays very small. On the other hand, the relative overhead for very
74  * small arrays is prohibitive (the minimum allocation is 16 elements).
75  *
76  * The functions for operating the array are:
77  * \todo [petri] Figure out how to comment these in Doxygen-style.
78  *
79  * \code
80  * Array*   Array_create            (deMemPool* pool);
81  * int      Array_getNumElements    (const Array* array);
82  * deBool   Array_reserve           (Array* array, int size);
83  * deBool   Array_setSize           (Array* array, int size);
84  * void		Array_reset				(Array* array);
85  * Element  Array_get               (Array* array, int ndx);
86  * deBool   Array_set               (Array* array, int ndx, Element elem);
87  * deBool   Array_pushBack          (Array* array, Element elem);
88  * Element  Array_popBack           (Array* array);
89  * void     Array_swap              (Array* array, int aNdx, int bNdx);
90  * \endcode
91 *//*--------------------------------------------------------------------*/
92 #define DE_DECLARE_POOL_ARRAY(TYPENAME, VALUETYPE)		\
93     \
94 typedef struct TYPENAME##_s					\
95 {											\
96 	deMemPool*			pool;				\
97 											\
98 	int					elementSize;		\
99 	int					numElements;		\
100 	int					capacity;			\
101 											\
102 	int					pageTableCapacity;	\
103 	DE_PTR_TYPE(VALUETYPE)*	pageTable;		\
104 } TYPENAME; /* NOLINT(TYPENAME) */			\
105 \
106 DE_INLINE TYPENAME*	TYPENAME##_create			(deMemPool* pool);															\
107 DE_INLINE int		TYPENAME##_getNumElements	(const TYPENAME* arr)									DE_UNUSED_FUNCTION;	\
108 DE_INLINE deBool	TYPENAME##_reserve			(DE_PTR_TYPE(TYPENAME) arr, int capacity)				DE_UNUSED_FUNCTION;	\
109 DE_INLINE deBool	TYPENAME##_setSize			(DE_PTR_TYPE(TYPENAME) arr, int size)					DE_UNUSED_FUNCTION;	\
110 DE_INLINE void		TYPENAME##_reset			(DE_PTR_TYPE(TYPENAME) arr)								DE_UNUSED_FUNCTION;	\
111 DE_INLINE VALUETYPE	TYPENAME##_get				(const TYPENAME* arr, int ndx)							DE_UNUSED_FUNCTION;	\
112 DE_INLINE void		TYPENAME##_set				(DE_PTR_TYPE(TYPENAME) arr, int ndx, VALUETYPE elem)	DE_UNUSED_FUNCTION;	\
113 DE_INLINE deBool	TYPENAME##_pushBack			(DE_PTR_TYPE(TYPENAME) arr, VALUETYPE elem)				DE_UNUSED_FUNCTION;	\
114 DE_INLINE VALUETYPE	TYPENAME##_popBack			(DE_PTR_TYPE(TYPENAME) arr)								DE_UNUSED_FUNCTION;	\
115 DE_INLINE deBool	TYPENAME##_copy				(DE_PTR_TYPE(TYPENAME) dst, const TYPENAME* src)		DE_UNUSED_FUNCTION;	\
116 DE_INLINE void		TYPENAME##_swap				(DE_PTR_TYPE(TYPENAME) arr, int aNdx, int bNdx)			DE_UNUSED_FUNCTION;	\
117 \
118 DE_INLINE TYPENAME* TYPENAME##_create (deMemPool* pool)    \
119 {    \
120 	return (TYPENAME*)dePoolArray_create(pool, sizeof(VALUETYPE));    \
121 }    \
122 \
123 DE_INLINE int TYPENAME##_getNumElements (const TYPENAME* arr)    \
124 {    \
125 	return arr->numElements;    \
126 }    \
127 \
128 DE_INLINE deBool TYPENAME##_reserve (DE_PTR_TYPE(TYPENAME) arr, int capacity)    \
129 {    \
130 	if (capacity > arr->capacity)    \
131 		return dePoolArray_reserve((dePoolArray*)arr, capacity);    \
132 	return  DE_TRUE;    \
133 }    \
134 \
135 DE_INLINE deBool TYPENAME##_setSize (DE_PTR_TYPE(TYPENAME) arr, int size)    \
136 {    \
137 	if (size > arr->capacity)    \
138 		return dePoolArray_setSize((dePoolArray*)arr, size);    \
139 \
140 	arr->numElements = size;    \
141 	return DE_TRUE;    \
142 }    \
143 \
144 DE_INLINE void TYPENAME##_reset (DE_PTR_TYPE(TYPENAME) arr)    \
145 {    \
146 	arr->numElements = 0;    \
147 }    \
148 \
149 DE_INLINE VALUETYPE TYPENAME##_get (const TYPENAME* arr, int ndx)    \
150 {    \
151 	DE_ASSERT(ndx >= 0 && ndx < arr->numElements);    \
152 	{    \
153 		int pageNdx	= (ndx >> DE_ARRAY_ELEMENTS_PER_PAGE_LOG2);    \
154 		int subNdx	= ndx & ((1 << DE_ARRAY_ELEMENTS_PER_PAGE_LOG2) - 1);    \
155 		return ((VALUETYPE*)arr->pageTable[pageNdx])[subNdx];    \
156 	}    \
157 }    \
158 \
159 DE_INLINE void TYPENAME##_set (DE_PTR_TYPE(TYPENAME) arr, int ndx, VALUETYPE elem)    \
160 {    \
161 	DE_ASSERT(ndx >= 0 && ndx < arr->numElements);    \
162 	{    \
163 		int pageNdx	= (ndx >> DE_ARRAY_ELEMENTS_PER_PAGE_LOG2);    \
164 		int subNdx	= ndx & ((1 << DE_ARRAY_ELEMENTS_PER_PAGE_LOG2) - 1);    \
165 		((VALUETYPE*)arr->pageTable[pageNdx])[subNdx] = elem;    \
166 	}    \
167 }    \
168 \
169 DE_INLINE deBool TYPENAME##_pushBack (DE_PTR_TYPE(TYPENAME) arr, VALUETYPE elem)    \
170 {    \
171 	if ((arr->numElements + 1 >= arr->capacity) && !TYPENAME##_reserve(arr, arr->numElements + 1)) \
172 		return DE_FALSE; \
173 	arr->numElements++; \
174 	TYPENAME##_set(arr, arr->numElements - 1, elem); \
175 	return DE_TRUE;    \
176 }    \
177 \
178 DE_INLINE VALUETYPE TYPENAME##_popBack (DE_PTR_TYPE(TYPENAME) arr)    \
179 {    \
180 	int ndx		= arr->numElements - 1; \
181 	int pageNdx	= (ndx >> DE_ARRAY_ELEMENTS_PER_PAGE_LOG2);    \
182 	int subNdx	= ndx & ((1 << DE_ARRAY_ELEMENTS_PER_PAGE_LOG2) - 1);    \
183 	DE_ASSERT(arr->numElements > 0); \
184 	arr->numElements--; \
185 	/* \note We access a value which is out-of-bounds, but we know it to be safe. */ \
186 	return ((VALUETYPE*)arr->pageTable[pageNdx])[subNdx];    \
187 }    \
188 \
189 DE_INLINE deBool TYPENAME##_copy (DE_PTR_TYPE(TYPENAME) dst, const TYPENAME* src)		\
190 {																			\
191 	DE_ASSERT(dst && src);													\
192 	{																		\
193 		int numElements = src->numElements;									\
194 		int ndx;															\
195 		if (!TYPENAME##_setSize(dst, numElements))							\
196 			return DE_FALSE;												\
197 		for (ndx = 0; ndx < numElements; ndx++)								\
198 			TYPENAME##_set(dst, ndx, TYPENAME##_get(src, ndx));				\
199 	}																		\
200 	return DE_TRUE;															\
201 }																			\
202 \
203 DE_INLINE void TYPENAME##_swap (DE_PTR_TYPE(TYPENAME) arr, int aNdx, int bNdx)	\
204 {	\
205 	VALUETYPE tmp = TYPENAME##_get(arr, aNdx);	\
206 	TYPENAME##_set(arr, aNdx, TYPENAME##_get(arr, bNdx));	\
207 	TYPENAME##_set(arr, bNdx, tmp);	\
208 }	\
209 \
210 struct TYPENAME##Dummy_s { int dummy; }
211 
212 /*--------------------------------------------------------------------*//*!
213  * \brief Declare a sort function for an array.
214  * \param TYPENAME	Type name of the declared array.
215  * \param VALUETYPE	Type of the value contained in the array.
216  * \param SORTNAME	Name for this specific sort.
217  * \param CMPFUNC	Comparison function for sorting.
218  *
219  * This macro declares a sort function for an array declared using
220  * DE_DECLARE_POOL_ARRAY macro.
221  *
222  * Sorting algorithm is heap sort since it requires constant amount of
223  * auxiliary space and is in-place sort. Worst-case run-time is O(n log n)
224  * and sort is NOT stable.
225  *
226  * CMPFUNC is used to compare elements in array. It must accept two
227  * parameters and return negative integer if first is smaller than, 0 if
228  * both are equal and positive integer if first is larger than second.
229  *
230  * The functions for sorting array are:
231  * \todo [petri] Figure out how to comment these in Doxygen-style.
232  *
233  * \code
234  * void		Array_sortName			(Array* array);
235  * void		Array_sortNameHeapify	(Array* array);
236  * void		Array_sortNameShiftDown	(Array* array, int start, int end);
237  * \endcode
238 *//*--------------------------------------------------------------------*/
239 #define DE_DECLARE_POOL_ARRAY_SORT(TYPENAME, VALUETYPE, SORTNAME, CMPFUNC)	\
240 \
241 DE_INLINE void TYPENAME##_##SORTNAME##ShiftDown (DE_PTR_TYPE(TYPENAME) arr, int startNdx, int endNdx)	\
242 {	\
243 	int rootNdx = startNdx;	\
244 	\
245 	while (rootNdx * 2 + 1 <= endNdx)	\
246 	{	\
247 		int childNdx = rootNdx * 2 + 1;	\
248 		\
249 		if ((childNdx + 1 <= endNdx) && (CMPFUNC(TYPENAME##_get(arr, childNdx), TYPENAME##_get(arr, childNdx + 1)) < 0))	\
250 			childNdx += 1;	\
251 		\
252 		if (CMPFUNC(TYPENAME##_get(arr, rootNdx), TYPENAME##_get(arr, childNdx)) < 0)	\
253 		{	\
254 			TYPENAME##_swap(arr, rootNdx, childNdx);	\
255 			rootNdx = childNdx;	\
256 		}	\
257 		else	\
258 			break;	\
259 	}	\
260 }	\
261 \
262 DE_INLINE void TYPENAME##_##SORTNAME##Heapify (DE_PTR_TYPE(TYPENAME) arr)	\
263 {	\
264 	int startNdx = (TYPENAME##_getNumElements(arr) - 2) / 2;	\
265 	\
266 	while (startNdx >= 0)	\
267 	{	\
268 		TYPENAME##_##SORTNAME##ShiftDown(arr, startNdx, TYPENAME##_getNumElements(arr) - 1);	\
269 		startNdx -= 1;	\
270 	}	\
271 }	\
272 \
273 DE_INLINE void TYPENAME##_##SORTNAME (DE_PTR_TYPE(TYPENAME) arr)	\
274 {	\
275 	int endNdx = TYPENAME##_getNumElements(arr) - 1;	\
276 	\
277 	TYPENAME##_##SORTNAME##Heapify(arr);	\
278 	\
279 	while (endNdx > 0)	\
280 	{	\
281 		TYPENAME##_swap(arr, endNdx, 0);	\
282 		endNdx -= 1;	\
283 		TYPENAME##_##SORTNAME##ShiftDown(arr, 0, endNdx);	\
284 	}	\
285 }	\
286 \
287 struct TYPENAME##SORTNAME##Dummy_s { int dummy; }
288 
289 /* Basic array types. */
290 
291 DE_DECLARE_POOL_ARRAY(deIntArray, int);
292 DE_DECLARE_POOL_ARRAY(deInt8Array, deInt8);
293 DE_DECLARE_POOL_ARRAY(deUint8Array, deUint8);
294 DE_DECLARE_POOL_ARRAY(deInt16Array, deInt16);
295 DE_DECLARE_POOL_ARRAY(deUint16Array, deUint16);
296 DE_DECLARE_POOL_ARRAY(deInt32Array, deInt32);
297 DE_DECLARE_POOL_ARRAY(deUint32Array, deUint32);
298 
299 #endif /* _DEPOOLARRAY_H */
300