1 /**************************************************************************
2  *
3  * Copyright 2010 LunarG, Inc.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a
7  * copy of this software and associated documentation files (the
8  * "Software"), to deal in the Software without restriction, including
9  * without limitation the rights to use, copy, modify, merge, publish,
10  * distribute, sub license, and/or sell copies of the Software, and to
11  * permit persons to whom the Software is furnished to do so, subject to
12  * the following conditions:
13  *
14  * The above copyright notice and this permission notice (including the
15  * next paragraph) shall be included in all copies or substantial portions
16  * of the Software.
17  *
18  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
21  * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
24  * DEALINGS IN THE SOFTWARE.
25  *
26  **************************************************************************/
27 
28 
29 #include <stdlib.h>
30 #include <string.h>
31 
32 #include "egllog.h"
33 #include "eglarray.h"
34 
35 
36 /**
37  * Grow the size of the array.
38  */
39 static EGLBoolean
40 _eglGrowArray(_EGLArray *array)
41 {
42    EGLint new_size;
43    void **elems;
44 
45    new_size = array->MaxSize;
46    while (new_size <= array->Size)
47       new_size *= 2;
48 
49    elems = realloc(array->Elements, new_size * sizeof(array->Elements[0]));
50    if (!elems) {
51       _eglLog(_EGL_DEBUG, "failed to grow %s array to %d",
52             array->Name, new_size);
53       return EGL_FALSE;
54    }
55 
56    array->Elements = elems;
57    array->MaxSize = new_size;
58 
59    return EGL_TRUE;
60 }
61 
62 
63 /**
64  * Create an array.
65  */
66 _EGLArray *
67 _eglCreateArray(const char *name, EGLint init_size)
68 {
69    _EGLArray *array;
70 
71    array = calloc(1, sizeof(*array));
72    if (array) {
73       array->Name = name;
74       array->MaxSize = (init_size > 0) ? init_size : 1;
75       if (!_eglGrowArray(array)) {
76          free(array);
77          array = NULL;
78       }
79    }
80 
81    return array;
82 }
83 
84 
85 /**
86  * Destroy an array, optionally free the data.
87  */
88 void
89 _eglDestroyArray(_EGLArray *array, void (*free_cb)(void *))
90 {
91    if (free_cb) {
92       EGLint i;
93       for (i = 0; i < array->Size; i++)
94          free_cb(array->Elements[i]);
95    }
96    free(array->Elements);
97    free(array);
98 }
99 
100 
101 /**
102  * Append a element to an array.
103  */
104 void
105 _eglAppendArray(_EGLArray *array, void *elem)
106 {
107    if (array->Size >= array->MaxSize && !_eglGrowArray(array))
108       return;
109 
110    array->Elements[array->Size++] = elem;
111 }
112 
113 
114 /**
115  * Erase an element from an array.
116  */
117 void
118 _eglEraseArray(_EGLArray *array, EGLint i, void (*free_cb)(void *))
119 {
120    if (free_cb)
121       free_cb(array->Elements[i]);
122    if (i < array->Size - 1) {
123       memmove(&array->Elements[i], &array->Elements[i + 1],
124             (array->Size - i - 1) * sizeof(array->Elements[0]));
125    }
126    array->Size--;
127 }
128 
129 
130 /**
131  * Find in an array for the given element.
132  */
133 void *
134 _eglFindArray(_EGLArray *array, void *elem)
135 {
136    EGLint i;
137 
138    if (!array)
139       return NULL;
140 
141    for (i = 0; i < array->Size; i++)
142       if (array->Elements[i] == elem)
143          return elem;
144    return NULL;
145 }
146 
147 
148 /**
149  * Filter an array and return the number of filtered elements.
150  */
151 EGLint
152 _eglFilterArray(_EGLArray *array, void **data, EGLint size,
153                 _EGLArrayForEach filter, void *filter_data)
154 {
155    EGLint count = 0, i;
156 
157    if (!array)
158       return 0;
159 
160    if (filter) {
161       for (i = 0; i < array->Size; i++) {
162          if (filter(array->Elements[i], filter_data)) {
163             if (data && count < size)
164                data[count] = array->Elements[i];
165             count++;
166          }
167          if (data && count >= size)
168             break;
169       }
170    }
171    else {
172       if (data) {
173          count = (size < array->Size) ? size : array->Size;
174          memcpy(data, array->Elements, count * sizeof(array->Elements[0]));
175       }
176       else {
177          count = array->Size;
178       }
179    }
180 
181    return count;
182 }
183 
184 
185 /**
186  * Flatten an array by converting array elements into another form and store
187  * them in a buffer.
188  */
189 EGLint
190 _eglFlattenArray(_EGLArray *array, void *buffer, EGLint elem_size, EGLint size,
191                  _EGLArrayForEach flatten)
192 {
193    EGLint i, count;
194 
195    if (!array)
196       return 0;
197 
198    count = array->Size;
199    if (buffer) {
200       /* clamp size to 0 */
201       if (size < 0)
202          size = 0;
203       /* do not exceed buffer size */
204       if (count > size)
205          count = size;
206       for (i = 0; i < count; i++)
207          flatten(array->Elements[i],
208                (void *) ((char *) buffer + elem_size * i));
209    }
210 
211    return count;
212 }
213