Blender  V2.93
BLI_task_test.cc
Go to the documentation of this file.
1 /* Apache License, Version 2.0 */
2 
3 #include "testing/testing.h"
4 #include <cstring>
5 
6 #include "atomic_ops.h"
7 
8 #include "MEM_guardedalloc.h"
9 
10 #include "BLI_utildefines.h"
11 
12 #include "BLI_listbase.h"
13 #include "BLI_mempool.h"
14 #include "BLI_task.h"
15 
16 #define NUM_ITEMS 10000
17 
18 /* *** Parallel iterations over range of integer values. *** */
19 
20 static void task_range_iter_func(void *userdata, int index, const TaskParallelTLS *__restrict tls)
21 {
22  int *data = (int *)userdata;
23  data[index] = index;
24  *((int *)tls->userdata_chunk) += index;
25  // printf("%d, %d, %d\n", index, data[index], *((int *)tls->userdata_chunk));
26 }
27 
28 static void task_range_iter_reduce_func(const void *__restrict UNUSED(userdata),
29  void *__restrict join_v,
30  void *__restrict userdata_chunk)
31 {
32  int *join = (int *)join_v;
33  int *chunk = (int *)userdata_chunk;
34  *join += *chunk;
35  // printf("%d, %d\n", data[NUM_ITEMS], *((int *)userdata_chunk));
36 }
37 
38 TEST(task, RangeIter)
39 {
40  int data[NUM_ITEMS] = {0};
41  int sum = 0;
42 
44 
45  TaskParallelSettings settings;
47  settings.min_iter_per_thread = 1;
48 
49  settings.userdata_chunk = &sum;
50  settings.userdata_chunk_size = sizeof(sum);
52 
54 
55  /* Those checks should ensure us all items of the listbase were processed once, and only once
56  * as expected. */
57 
58  int expected_sum = 0;
59  for (int i = 0; i < NUM_ITEMS; i++) {
60  EXPECT_EQ(data[i], i);
61  expected_sum += i;
62  }
63  EXPECT_EQ(sum, expected_sum);
64 
66 }
67 
68 /* *** Parallel iterations over mempool items. *** */
69 
70 static void task_mempool_iter_func(void *userdata, MempoolIterData *item)
71 {
72  int *data = (int *)item;
73  int *count = (int *)userdata;
74 
75  EXPECT_TRUE(data != nullptr);
76 
77  *data += 1;
79 }
80 
81 TEST(task, MempoolIter)
82 {
83  int *data[NUM_ITEMS];
86  sizeof(*data[0]), NUM_ITEMS, 32, BLI_MEMPOOL_ALLOW_ITER);
87 
88  int i;
89 
90  /* 'Randomly' add and remove some items from mempool, to create a non-homogenous one. */
91  int num_items = 0;
92  for (i = 0; i < NUM_ITEMS; i++) {
93  data[i] = (int *)BLI_mempool_alloc(mempool);
94  *data[i] = i - 1;
95  num_items++;
96  }
97 
98  for (i = 0; i < NUM_ITEMS; i += 3) {
99  BLI_mempool_free(mempool, data[i]);
100  data[i] = nullptr;
101  num_items--;
102  }
103 
104  for (i = 0; i < NUM_ITEMS; i += 7) {
105  if (data[i] == nullptr) {
106  data[i] = (int *)BLI_mempool_alloc(mempool);
107  *data[i] = i - 1;
108  num_items++;
109  }
110  }
111 
112  for (i = 0; i < NUM_ITEMS - 5; i += 23) {
113  for (int j = 0; j < 5; j++) {
114  if (data[i + j] != nullptr) {
115  BLI_mempool_free(mempool, data[i + j]);
116  data[i + j] = nullptr;
117  num_items--;
118  }
119  }
120  }
121 
122  BLI_task_parallel_mempool(mempool, &num_items, task_mempool_iter_func, true);
123 
124  /* Those checks should ensure us all items of the mempool were processed once, and only once - as
125  * expected. */
126  EXPECT_EQ(num_items, 0);
127  for (i = 0; i < NUM_ITEMS; i++) {
128  if (data[i] != nullptr) {
129  EXPECT_EQ(*data[i], i);
130  }
131  }
132 
133  BLI_mempool_destroy(mempool);
135 }
136 
137 /* *** Parallel iterations over double-linked list items. *** */
138 
139 static void task_listbase_iter_func(void *userdata,
140  void *item,
141  int index,
142  const TaskParallelTLS *__restrict UNUSED(tls))
143 {
144  LinkData *data = (LinkData *)item;
145  int *count = (int *)userdata;
146 
147  data->data = POINTER_FROM_INT(POINTER_AS_INT(data->data) + index);
149 }
150 
151 TEST(task, ListBaseIter)
152 {
153  ListBase list = {nullptr, nullptr};
154  LinkData *items_buffer = (LinkData *)MEM_calloc_arrayN(
155  NUM_ITEMS, sizeof(*items_buffer), __func__);
157 
158  int i;
159 
160  int num_items = 0;
161  for (i = 0; i < NUM_ITEMS; i++) {
162  BLI_addtail(&list, &items_buffer[i]);
163  num_items++;
164  }
165 
166  TaskParallelSettings settings;
168 
169  BLI_task_parallel_listbase(&list, &num_items, task_listbase_iter_func, &settings);
170 
171  /* Those checks should ensure us all items of the listbase were processed once, and only once -
172  * as expected. */
173  EXPECT_EQ(num_items, 0);
174  LinkData *item;
175  for (i = 0, item = (LinkData *)list.first; i < NUM_ITEMS && item != nullptr;
176  i++, item = item->next) {
177  EXPECT_EQ(POINTER_AS_INT(item->data), i);
178  }
179  EXPECT_EQ(NUM_ITEMS, i);
180 
181  MEM_freeN(items_buffer);
183 }
EXPECT_EQ(BLI_expr_pylike_eval(expr, nullptr, 0, &result), EXPR_PYLIKE_INVALID)
void BLI_addtail(struct ListBase *listbase, void *vlink) ATTR_NONNULL(1)
Definition: listbase.c:110
@ BLI_MEMPOOL_ALLOW_ITER
Definition: BLI_mempool.h:85
void BLI_mempool_free(BLI_mempool *pool, void *addr) ATTR_NONNULL(1
BLI_mempool * BLI_mempool_create(unsigned int esize, unsigned int totelem, unsigned int pchunk, unsigned int flag) ATTR_MALLOC ATTR_WARN_UNUSED_RESULT
Definition: BLI_mempool.c:268
void * BLI_mempool_alloc(BLI_mempool *pool) ATTR_MALLOC ATTR_WARN_UNUSED_RESULT ATTR_NONNULL(1)
Definition: BLI_mempool.c:334
void BLI_mempool_destroy(BLI_mempool *pool) ATTR_NONNULL(1)
Definition: BLI_mempool.c:757
void BLI_task_parallel_range(const int start, const int stop, void *userdata, TaskParallelRangeFunc func, const TaskParallelSettings *settings)
Definition: task_range.cc:110
struct MempoolIterData MempoolIterData
Definition: BLI_task.h:223
BLI_INLINE void BLI_parallel_range_settings_defaults(TaskParallelSettings *settings)
Definition: BLI_task.h:231
void BLI_task_parallel_mempool(struct BLI_mempool *mempool, void *userdata, TaskParallelMempoolFunc func, const bool use_threading)
void BLI_task_parallel_listbase(struct ListBase *listbase, void *userdata, TaskParallelIteratorFunc func, const TaskParallelSettings *settings)
static void task_mempool_iter_func(void *userdata, MempoolIterData *item)
static void task_range_iter_func(void *userdata, int index, const TaskParallelTLS *__restrict tls)
static void task_range_iter_reduce_func(const void *__restrict UNUSED(userdata), void *__restrict join_v, void *__restrict userdata_chunk)
static void task_listbase_iter_func(void *userdata, void *item, int index, const TaskParallelTLS *__restrict UNUSED(tls))
TEST(task, RangeIter)
#define NUM_ITEMS
void BLI_threadapi_init(void)
Definition: threads.cc:143
void BLI_threadapi_exit(void)
Definition: threads.cc:151
#define POINTER_FROM_INT(i)
#define UNUSED(x)
#define POINTER_AS_INT(i)
Read Guarded memory(de)allocation.
Provides wrapper around system-specific atomic primitives, and some extensions (faked-atomic operatio...
ATOMIC_INLINE uint32_t atomic_sub_and_fetch_uint32(uint32_t *p, uint32_t x)
static T sum(const btAlignedObjectArray< T > &items)
int count
void(* MEM_freeN)(void *vmemh)
Definition: mallocn.c:41
void *(* MEM_calloc_arrayN)(size_t len, size_t size, const char *str)
Definition: mallocn.c:46
struct blender::compositor::@172::@174 task
unsigned int uint32_t
Definition: stdint.h:83
void * data
Definition: DNA_listBase.h:42
struct LinkData * next
Definition: DNA_listBase.h:41
void * first
Definition: DNA_listBase.h:47
TaskParallelReduceFunc func_reduce
Definition: BLI_task.h:158
size_t userdata_chunk_size
Definition: BLI_task.h:150