• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  *
3  * Copyright 2015 Rockchip Electronics Co., LTD.
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at
8  *
9  *      http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  */
17 
18 #define ALOG_TAG "RK_LIST"
19 #include "rk_list.h"
20 #include <mpp_log.h>
21 #include <cerrno>
22 #include <cstring>
23 #include "hdf_log.h"
24 #include "securec.h"
25 
26 #define _RK_LIST_ERROR
27 #ifdef _RK_LIST_DEUBG
28 #define LIST_DEBUG(fmt, ...) mpp_log(fmt, ## __VA_ARGS__)
29 #else
30 #define LIST_DEBUG(fmt, ...) /* not debugging: nothing */
31 #endif
32 
33 
34 typedef struct rk_list_node {
35     rk_list_node*   prev;
36     rk_list_node*   next;
37     RK_U32        key;
38     RK_S32         size;
39 } rk_list_node;
40 
list_node_init(rk_list_node * node)41 static inline void list_node_init(rk_list_node *node)
42 {
43     node->prev = node->next = node;
44 }
45 
list_node_init_with_key_and_size(rk_list_node * node,RK_U32 key,RK_S32 size)46 static inline void list_node_init_with_key_and_size(rk_list_node *node, RK_U32 key, RK_S32 size)
47 {
48     list_node_init(node);
49     node->key   = key;
50     node->size  = size;
51 }
52 
create_list(void * data,RK_S32 size,RK_U32 key)53 static rk_list_node* create_list(void *data, RK_S32 size, RK_U32 key)
54 {
55     rk_list_node *node = (rk_list_node*)malloc(sizeof(rk_list_node) + size);
56     if (node) {
57         void *dst = (void*)(node + 1);
58         list_node_init_with_key_and_size(node, key, size);
59         if (memcpy_s(dst, size, data, size) != EOK) {
60             HDF_LOGE("memcpy_s no");
61         }
62     } else {
63         HDF_LOGE("failed to allocate list node");
64     }
65     return node;
66 }
67 
_rk_list_add(rk_list_node * _new,rk_list_node * prev,rk_list_node * next)68 static inline void _rk_list_add(rk_list_node * _new, rk_list_node * prev, rk_list_node * next)
69 {
70     next->prev = _new;
71     _new->next = next;
72     _new->prev = prev;
73     prev->next = _new;
74 }
75 
rk_list_add(rk_list_node * _new,rk_list_node * head)76 static inline void rk_list_add(rk_list_node *_new, rk_list_node *head)
77 {
78     _rk_list_add(_new, head, head->next);
79 }
80 
rk_list_add_tail(rk_list_node * _new,rk_list_node * head)81 static inline void rk_list_add_tail(rk_list_node *_new, rk_list_node *head)
82 {
83     _rk_list_add(_new, head->prev, head);
84 }
85 
add_at_head(void * data,RK_S32 size)86 RK_S32 rk_list::add_at_head(void *data, RK_S32 size)
87 {
88     RK_S32 ret = -EINVAL;
89     pthread_mutex_lock(&mutex);
90     if (head) {
91         rk_list_node *node = create_list(data, size, 0);
92         if (node) {
93             rk_list_add(node, head);
94             count++;
95             ret = 0;
96         } else {
97             ret = -ENOMEM;
98         }
99     }
100     pthread_mutex_unlock(&mutex);
101     return ret;
102 }
103 
add_at_tail(void * data,RK_S32 size)104 RK_S32 rk_list::add_at_tail(void *data, RK_S32 size)
105 {
106     RK_S32 ret = -EINVAL;
107     pthread_mutex_lock(&mutex);
108     if (head) {
109         rk_list_node *node = create_list(data, size, 0);
110         if (node) {
111             rk_list_add_tail(node, head);
112             count++;
113             ret = 0;
114         } else {
115             ret = -ENOMEM;
116         }
117     }
118     pthread_mutex_unlock(&mutex);
119     return ret;
120 }
121 
release_list(rk_list_node * node,void * data,RK_S32 size)122 static void release_list(rk_list_node*node, void *data, RK_S32 size)
123 {
124     void *src = (void*)(node + 1);
125     if (node->size == size) {
126         if (memcpy_s(data, size, src, size) != EOK) {
127             HDF_LOGE("memcpy_s no");
128         }
129     } else {
130         HDF_LOGE("node size check failed when release_list");
131         size = (size < node->size) ? (size) : (node->size);
132         if (memcpy_s(data, size, src, size) != EOK) {
133             HDF_LOGE("memcpy_s no");
134         }
135     }
136     free(node);
137 }
138 
_rk_list_del(rk_list_node * prev,rk_list_node * next)139 static inline void _rk_list_del(rk_list_node *prev, rk_list_node *next)
140 {
141     next->prev = prev;
142     prev->next = next;
143 }
144 
rk_list_del_init(rk_list_node * node)145 static inline void rk_list_del_init(rk_list_node *node)
146 {
147     _rk_list_del(node->prev, node->next);
148     list_node_init(node);
149 }
150 
_list_del_node_no_lock(rk_list_node * node,void * data,RK_S32 size)151 static inline void _list_del_node_no_lock(rk_list_node *node, void *data, RK_S32 size)
152 {
153     rk_list_del_init(node);
154     release_list(node, data, size);
155 }
156 
del_at_head(void * data,RK_S32 size)157 RK_S32 rk_list::del_at_head(void *data, RK_S32 size)
158 {
159     RK_S32 ret = -EINVAL;
160     pthread_mutex_lock(&mutex);
161     if (head && count) {
162         _list_del_node_no_lock(head->next, data, size);
163         count--;
164         ret = 0;
165     }
166     pthread_mutex_unlock(&mutex);
167     return ret;
168 }
169 
del_at_tail(void * data,RK_S32 size)170 RK_S32 rk_list::del_at_tail(void *data, RK_S32 size)
171 {
172     RK_S32 ret = -EINVAL;
173     pthread_mutex_lock(&mutex);
174     if (head && count) {
175         _list_del_node_no_lock(head->prev, data, size);
176         count--;
177         ret = 0;
178     }
179     pthread_mutex_unlock(&mutex);
180     return ret;
181 }
182 
list_is_empty()183 RK_S32 rk_list::list_is_empty()
184 {
185     pthread_mutex_lock(&mutex);
186     RK_S32 ret = (count == 0);
187     pthread_mutex_unlock(&mutex);
188     return ret;
189 }
190 
list_size()191 RK_S32 rk_list::list_size()
192 {
193     pthread_mutex_lock(&mutex);
194     RK_S32 ret = count;
195     pthread_mutex_unlock(&mutex);
196     return ret;
197 }
198 
add_by_key(void * data,RK_S32 size,RK_U32 * key)199 RK_S32 rk_list::add_by_key(void *data, RK_S32 size, RK_U32 *key)
200 {
201     RK_S32 ret = 0;
202     (void)data;
203     (void)size;
204     (void)key;
205     return ret;
206 }
207 
del_by_key(void * data,RK_S32 size,RK_U32 key)208 RK_S32 rk_list::del_by_key(void *data, RK_S32 size, RK_U32 key)
209 {
210     RK_S32 ret = 0;
211     (void)data;
212     (void)size;
213     (void)key;
214     return ret;
215 }
216 
217 
show_by_key(void * data,RK_U32 key)218 RK_S32 rk_list::show_by_key(void *data, RK_U32 key)
219 {
220     RK_S32 ret = 0;
221     (void)data;
222     (void)key;
223     return ret;
224 }
225 
flush()226 RK_S32 rk_list::flush()
227 {
228     pthread_mutex_lock(&mutex);
229     if (head) {
230         while (count) {
231             rk_list_node* node = head->next;
232             rk_list_del_init(node);
233             if (destroy) {
234                 destroy((void*)(node + 1));
235             }
236             free(node);
237             count--;
238         }
239     }
240     pthread_mutex_unlock(&mutex);
241     return 0;
242 }
243 
rk_list(node_destructor func)244 rk_list::rk_list(node_destructor func)
245     : destroy(nullptr),
246       head(nullptr),
247       count(0)
248 {
249     pthread_mutexattr_t attr;
250     pthread_mutexattr_init(&attr);
251     pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
252     pthread_mutex_init(&mutex, &attr);
253     pthread_mutexattr_destroy(&attr);
254     destroy = func;
255     head = (rk_list_node*)malloc(sizeof(rk_list_node));
256     if (nullptr == head) {
257         HDF_LOGE("failed to allocate list header");
258     } else {
259         list_node_init_with_key_and_size(head, 0, 0);
260     }
261 }
262 
~rk_list()263 rk_list::~rk_list()
264 {
265     flush();
266     if (head) free(head);
267     head = nullptr;
268     destroy = nullptr;
269     pthread_mutex_destroy(&mutex);
270 }
271 
272 #if BUILD_RK_LIST_TEST
273 #include "vpu_mem.h"
274 #include <stdio.h>
275 #include <stdlib.h>
276 #include <string.h>
277 
278 #define LOOP_RK_LIST    600
279 
280 #define COUNT_ADD       100
281 #define COUNT_DEL       99
282 
283 static volatile int err = 0;
284 
rk_list_fifo_test(rk_list * list_0)285 static int rk_list_fifo_test(rk_list *list_0)
286 {
287     int count;
288     VPUMemLinear_t m;
289     for (count = 0; count < COUNT_ADD; count++) {
290         err |= VPUMallocLinear(&m, 100); // 100 size
291         if (err) {
292             printf("VPUMallocLinear in rk_list_fifo_test\n");
293             break;
294         }
295         err |= list_0->add_at_head(&m, sizeof(m));
296         if (err) {
297             printf("add_at_head in rk_list_fifo_test\n");
298             break;
299         }
300     }
301 
302     if (!err) {
303         for (count = 0; count < COUNT_DEL; count++) {
304             err |= list_0->del_at_tail(&m, sizeof(m));
305             if (err) {
306                 printf("del_at_tail in rk_list_fifo_test\n");
307                 break;
308             }
309             err |= VPUFreeLinear(&m);
310             if (err) {
311                 printf("VPUFreeLinear in rk_list_fifo_test\n");
312                 break;
313             }
314         }
315     }
316     return err;
317 }
318 
rk_list_filo_test(rk_list * list_0)319 static int rk_list_filo_test(rk_list *list_0)
320 {
321     int count;
322     VPUMemLinear_t m;
323     for (count = 0; count < COUNT_ADD + COUNT_DEL; count++) {
324         if (count & 1) {
325             err |= list_0->del_at_head(&m, sizeof(m));
326             if (err) {
327                 printf("del_at_head in rk_list_filo_test\n");
328                 break;
329             }
330             err |= VPUFreeLinear(&m);
331             if (err) {
332                 printf("VPUFreeLinear in rk_list_fifo_test\n");
333                 break;
334             }
335         } else {
336             err |= VPUMallocLinear(&m, 100); // 100 size
337             if (err) {
338                 printf("VPUMallocLinear in rk_list_filo_test\n");
339                 break;
340             }
341             err |= list_0->add_at_head(&m, sizeof(m));
342             if (err) {
343                 printf("add_at_head in rk_list_fifo_test\n");
344                 break;
345             }
346         }
347     }
348 
349     return err;
350 }
351 
352 
rk_list_test_loop_0(void * pdata)353 void *rk_list_test_loop_0(void *pdata)
354 {
355     int i;
356     rk_list *list_0 = (rk_list *)pdata;
357 
358     printf("rk_list test 0 loop start\n");
359     for (i = 0; i < LOOP_RK_LIST; i++) {
360         err |= rk_list_filo_test(list_0);
361         if (err) break;
362     }
363 
364     if (err) {
365         printf("thread: found vpu mem operation err %d\n", err);
366     } else {
367         printf("thread: test done and found no err\n");
368     }
369     return nullptr;
370 }
371 
rk_list_test_0()372 int rk_list_test_0()
373 {
374     int i, err = 0;
375     printf("rk_list test 0 FIFO start\n");
376 
377     rk_list *list_0 = new rk_list((node_destructor)VPUFreeLinear);
378 
379     pthread_t mThread;
380     pthread_attr_t attr;
381     pthread_attr_init(&attr);
382     pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_JOINABLE);
383 
384     pthread_create(&mThread, &attr, rk_list_test_loop_0, (void*)list_0);
385     pthread_attr_destroy(&attr);
386 
387     for (i = 0; i < LOOP_RK_LIST; i++) {
388         err |= rk_list_fifo_test(list_0);
389         if (err) break;
390     }
391     if (err) {
392         printf("main  : found rk_list operation err %d\n", err);
393     } else {
394         printf("main  : test done and found no err\n");
395     }
396 
397     void *dummy;
398     pthread_join(mThread, &dummy);
399 
400     printf("rk_list test 0 end size %d\n", list_0->list_size());
401     delete list_0;
402     return err;
403 }
404 
405 #define TOTAL_RK_LIST_TEST_COUNT    1
406 
407 typedef int (*RK_LIST_TEST_FUNC)(void);
408 RK_LIST_TEST_FUNC test_func[TOTAL_RK_LIST_TEST_COUNT] = {
409     rk_list_test_0,
410 };
411 
main(int argc,char * argv[])412 int main(int argc, char *argv[])
413 {
414     int i, start = 0, end = 0;
415     if (argc < 2) { // 2 Conditions
416         end = TOTAL_RK_LIST_TEST_COUNT;
417     } else if (argc == 2) { // 2 Conditions
418         start = atoi(argv[1]);
419         end   = start + 1;
420     } else if (argc == 3) { // 3 Conditions
421         start = atoi(argv[1]);
422         end   = atoi(argv[2]); // argv 2
423     } else {
424         printf("too many argc %d\n", argc);
425         return -1;
426     }
427     if (start < 0 || start > TOTAL_RK_LIST_TEST_COUNT || end < 0 || end > TOTAL_RK_LIST_TEST_COUNT) {
428         printf("invalid input: start %d end %d\n", start, end);
429         return -1;
430     }
431     for (i = start; i < end; i++) {
432         int err = test_func[i]();
433         if (err) {
434             printf("test case %d return err %d\n", i, err);
435             break;
436         }
437     }
438     return 0;
439 }
440 #endif
441 
442