Blender V4.3
BLI_linklist_stack.h
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2001-2002 NaN Holding BV. All rights reserved.
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
5#pragma once
6
19/* -------------------------------------------------------------------- */
25#define BLI_LINKSTACK_DECLARE(var, type) \
26 LinkNode *var; \
27 BLI_mempool *var##_pool_; \
28 type var##_type_
29
30#define BLI_LINKSTACK_INIT(var) \
31 { \
32 var = NULL; \
33 var##_pool_ = BLI_mempool_create(sizeof(LinkNode), 0, 64, BLI_MEMPOOL_NOP); \
34 } \
35 (void)0
36
37#define BLI_LINKSTACK_SIZE(var) BLI_mempool_len(var##_pool_)
38
39/* Check for `decltype()` or `typeof()` support. */
40#if defined(__cplusplus)
41# define BLI_LINKSTACK_PUSH(var, ptr) \
42 (CHECK_TYPE_INLINE_NONCONST(ptr, decltype(var##_type_)), \
43 BLI_linklist_prepend_pool(&(var), ptr, var##_pool_))
44# define BLI_LINKSTACK_POP(var) \
45 (decltype(var##_type_))(var ? BLI_linklist_pop_pool(&(var), var##_pool_) : NULL)
46# define BLI_LINKSTACK_POP_DEFAULT(var, r) \
47 (decltype(var##_type_))(var ? BLI_linklist_pop_pool(&(var), var##_pool_) : r)
48#elif defined(__GNUC__)
49# define BLI_LINKSTACK_PUSH(var, ptr) \
50 (CHECK_TYPE_INLINE_NONCONST(ptr, typeof(var##_type_)), \
51 BLI_linklist_prepend_pool(&(var), ptr, var##_pool_))
52# define BLI_LINKSTACK_POP(var) \
53 (typeof(var##_type_))(var ? BLI_linklist_pop_pool(&(var), var##_pool_) : NULL)
54# define BLI_LINKSTACK_POP_DEFAULT(var, r) \
55 (typeof(var##_type_))(var ? BLI_linklist_pop_pool(&(var), var##_pool_) : r)
56#else /* Non GCC/C++. */
57# define BLI_LINKSTACK_PUSH(var, ptr) (BLI_linklist_prepend_pool(&(var), ptr, var##_pool_))
58# define BLI_LINKSTACK_POP(var) (var ? BLI_linklist_pop_pool(&(var), var##_pool_) : NULL)
59# define BLI_LINKSTACK_POP_DEFAULT(var, r) (var ? BLI_linklist_pop_pool(&(var), var##_pool_) : r)
60#endif
61
62#define BLI_LINKSTACK_SWAP(var_a, var_b) \
63 { \
64 CHECK_TYPE_PAIR(var_a##_type_, var_b##_type_); \
65 SWAP(LinkNode *, var_a, var_b); \
66 SWAP(BLI_mempool *, var_a##_pool_, var_b##_pool_); \
67 } \
68 (void)0
69
70#define BLI_LINKSTACK_FREE(var) \
71 { \
72 BLI_mempool_destroy(var##_pool_); \
73 var##_pool_ = NULL; \
74 (void)var##_pool_; \
75 var = NULL; \
76 (void)var; \
77 (void)&(var##_type_); \
78 } \
79 (void)0
80
81#include "BLI_linklist.h"
82#include "BLI_mempool.h"
83
86/* -------------------------------------------------------------------- */
95#ifdef __GNUC__
96# define _BLI_SMALLSTACK_CAST(var) (typeof(_##var##_type))
97#else
98# define _BLI_SMALLSTACK_CAST(var)
99#endif
100
101#define _BLI_SMALLSTACK_FAKEUSER(var) (void)(&(_##var##_type))
102
103#define BLI_SMALLSTACK_DECLARE(var, type) \
104 LinkNode *_##var##_stack = NULL, *_##var##_free = NULL, *_##var##_temp = NULL; \
105 type _##var##_type
106
107#define BLI_SMALLSTACK_PUSH(var, data) \
108 { \
109 CHECK_TYPE_PAIR(data, _##var##_type); \
110 if (_##var##_free) { \
111 _##var##_temp = _##var##_free; \
112 _##var##_free = _##var##_free->next; \
113 } \
114 else { \
115 _##var##_temp = (LinkNode *)alloca(sizeof(LinkNode)); \
116 } \
117 _##var##_temp->next = _##var##_stack; \
118 _##var##_temp->link = data; \
119 _##var##_stack = _##var##_temp; \
120 _BLI_SMALLSTACK_FAKEUSER(var); \
121 } \
122 (void)0
123
124/* internal use, no null check */
125#define _BLI_SMALLSTACK_DEL_EX(var_src, var_dst) \
126 (void)(_BLI_SMALLSTACK_FAKEUSER(var_src), \
127 _BLI_SMALLSTACK_FAKEUSER(var_dst), \
128 (_##var_src##_temp = _##var_src##_stack->next), \
129 (_##var_src##_stack->next = _##var_dst##_free), \
130 (_##var_dst##_free = _##var_src##_stack), \
131 (_##var_src##_stack = _##var_src##_temp))
132
133#define _BLI_SMALLSTACK_DEL(var) _BLI_SMALLSTACK_DEL_EX(var, var)
134
135/* check for typeof() */
136#define BLI_SMALLSTACK_POP(var) \
137 (_BLI_SMALLSTACK_CAST(var)( \
138 (_##var##_stack) ? (_BLI_SMALLSTACK_DEL(var), (_##var##_free->link)) : NULL))
139
140/* support to put the free-node into another stack */
141#define BLI_SMALLSTACK_POP_EX(var_src, var_dst) \
142 (_BLI_SMALLSTACK_CAST(var_src)( \
143 (_##var_src##_stack) ? \
144 (_BLI_SMALLSTACK_DEL_EX(var_src, var_dst), (_##var_dst##_free->link)) : \
145 NULL))
146
147#define BLI_SMALLSTACK_PEEK(var) \
148 (_BLI_SMALLSTACK_CAST(var)((_##var##_stack) ? _##var##_stack->link : NULL))
149
150#define BLI_SMALLSTACK_IS_EMPTY(var) ((_BLI_SMALLSTACK_CAST(var) _##var##_stack) == NULL)
151
152/* fill in a lookup table */
153#define BLI_SMALLSTACK_AS_TABLE(var, data) \
154 { \
155 LinkNode *_##var##_iter; \
156 unsigned int i; \
157 for (_##var##_iter = _##var##_stack, i = 0; _##var##_iter; \
158 _##var##_iter = _##var##_iter->next, i++) \
159 { \
160 *(void **)&(data)[i] = _##var##_iter->link; \
161 } \
162 } \
163 ((void)0)
164
165/* loop over stack members last-added-first */
166#define BLI_SMALLSTACK_ITER_BEGIN(var, item) \
167 { \
168 LinkNode *_##var##_iter; \
169 for (_##var##_iter = _##var##_stack; _##var##_iter; _##var##_iter = _##var##_iter->next) { \
170 item = _BLI_SMALLSTACK_CAST(var)(_##var##_iter->link);
171
172#define BLI_SMALLSTACK_ITER_END \
173 } \
174 } \
175 (void)0
176
177#define BLI_SMALLSTACK_SWAP(var_a, var_b) \
178 { \
179 CHECK_TYPE_PAIR(_##var_a##_type, _##var_b##_type); \
180 SWAP(LinkNode *, _##var_a##_stack, _##var_b##_stack); \
181 SWAP(LinkNode *, _##var_a##_free, _##var_b##_free); \
182 } \
183 (void)0
184