24RCSIDH(slab_h,
"$Id: 0f2c2eb0f32433205fc0a3a884c6d3856d931398 $")
30#include <freeradius-devel/util/dlist.h>
31#include <freeradius-devel/util/event.h>
35#define FR_SLAB_CONFIG_CONF_PARSER \
36 { FR_CONF_OFFSET("min", fr_slab_config_t, min_elements), .dflt = "10" }, \
37 { FR_CONF_OFFSET("max", fr_slab_config_t, max_elements), .dflt = "100" }, \
38 { FR_CONF_OFFSET("cleanup_interval", fr_slab_config_t, interval), .dflt = "30s" }, \
75#define FR_SLAB_TYPES(_name, _type) \
76 FR_DLIST_TYPES(_name ## _slab) \
77 FR_DLIST_TYPES(_name ## _slab_element) \
79 typedef int (*_type ## _slab_free_t)(_type *elem, void *uctx); \
80 typedef int (*_type ## _slab_alloc_t)(_type *elem, void *uctx); \
81 typedef int (*_type ## _slab_reserve_t)(_type *elem, void *uctx); \
84 FR_DLIST_HEAD(_name ## _slab) reserved; \
85 FR_DLIST_HEAD(_name ## _slab) avail; \
86 fr_event_list_t *el; \
88 fr_slab_config_t config; \
89 unsigned int in_use; \
90 unsigned int high_water_mark; \
91 _type ## _slab_alloc_t alloc; \
92 _type ## _slab_reserve_t reserve; \
96 } _name ## _slab_list_t; \
99 FR_DLIST_ENTRY(_name ## _slab) entry; \
100 _name ## _slab_list_t *list; \
103 FR_DLIST_HEAD(_name ## _slab_element) reserved; \
104 FR_DLIST_HEAD(_name ## _slab_element) avail; \
105 } _name ## _slab_t; \
109 FR_DLIST_ENTRY(_name ## _slab_element) entry; \
111 _name ## _slab_t *slab; \
112 _type ## _slab_free_t free; \
114 } _name ## _slab_element_t;
124#define FR_SLAB_FUNCS(_name, _type) \
125 FR_DLIST_FUNCS(_name ## _slab, _name ## _slab_t, entry) \
126 FR_DLIST_FUNCS(_name ## _slab_element, _name ## _slab_element_t, entry) \
128DIAG_OFF(unused-function) \
136 static void _ ## _name ## _slab_cleanup(fr_timer_list_t *tl, UNUSED fr_time_t now, void *uctx) \
138 _name ## _slab_list_t *slab_list = talloc_get_type_abort(uctx, _name ## _slab_list_t); \
139 _name ## _slab_t *slab = NULL, *next_slab = NULL; \
140 unsigned int to_clear, cleared = 0; \
141 to_clear = (slab_list->high_water_mark - slab_list->in_use) / 2; \
142 if ((slab_list->in_use + to_clear) < slab_list->config.min_elements) \
143 to_clear = slab_list->high_water_mark - slab_list->config.min_elements; \
144 if (to_clear < slab_list->config.elements_per_slab) goto finish; \
145 slab = _name ## _slab_head(&slab_list->avail); \
147 next_slab = _name ## _slab_next(&slab_list->avail, slab); \
148 if (_name ## _slab_element_num_elements(&slab->reserved) > 0) goto next; \
149 _name ## _slab_remove(&slab_list->avail, slab); \
150 cleared += _name ## _slab_element_num_elements(&slab->avail); \
151 to_clear -= _name ## _slab_element_num_elements(&slab->avail); \
152 _name ## _slab_element_talloc_free(&slab->avail); \
154 if (to_clear < slab_list->config.elements_per_slab) break; \
158 slab_list->high_water_mark -= cleared; \
160 (void) fr_timer_in(slab_list, tl, &slab_list->ev, slab_list->config.interval, false, \
161 _ ## _name ## _slab_cleanup, slab_list); \
179 static inline _name ## _slab_list_t *_name ## _slab_list_alloc(TALLOC_CTX *ctx, \
180 fr_event_list_t *el, \
181 fr_slab_config_t const *config, \
182 _type ## _slab_alloc_t alloc, \
183 _type ## _slab_reserve_t reserve, \
185 bool release_reset, \
188 _name ## _slab_list_t *slab; \
189 MEM(slab = talloc_zero(ctx, _name ## _slab_list_t)); \
191 slab->config = *config; \
192 if (slab->config.elements_per_slab == 0) { \
193 slab->config.elements_per_slab = (config->min_elements ? config->min_elements : 1); \
195 slab->alloc = alloc; \
196 slab->reserve = reserve; \
198 slab->release_reset = release_reset; \
199 slab->reserve_mru = reserve_mru; \
200 _name ## _slab_init(&slab->reserved); \
201 _name ## _slab_init(&slab->avail); \
203 if (unlikely(fr_timer_in(slab, el->tl, &slab->ev, config->interval, false, _ ## _name ## _slab_cleanup, slab) < 0)) { \
217 static int _ ## _name ## _slab_free(_name ## _slab_t *slab) \
219 slab->being_freed = true; \
236 static int _ ## _type ## _element_free(_name ## _slab_element_t *element) \
238 _name ## _slab_t *slab; \
239 if (element->in_use && element->free) element->free(( _type *)element, element->uctx); \
240 if (!element->slab) return 0; \
241 slab = element->slab; \
242 if (element->in_use) { \
243 fr_assert_msg(slab->being_freed || slab->list->config.allow_direct_free, \
244 "in-use slab element freed with talloc_free - use _slab_release()"); \
245 slab->list->in_use--; \
246 element->in_use = false; \
247 _name ## _slab_element_remove(&slab->reserved, element); \
249 _name ## _slab_element_remove(&slab->avail, element); \
263 static inline CC_HINT(nonnull) _type *_name ## _slab_reserve(_name ## _slab_list_t *slab_list) \
265 _name ## _slab_t *slab; \
266 _name ## _slab_element_t *element = NULL; \
268 slab = slab_list->reserve_mru ? _name ## _slab_tail(&slab_list->avail) : \
269 _name ## _slab_head(&slab_list->avail); \
270 if (!slab && ((_name ## _slab_num_elements(&slab_list->reserved) * \
271 slab_list->config.elements_per_slab) < slab_list->config.max_elements)) { \
272 _name ## _slab_element_t *new_element; \
273 unsigned int count, elems; \
275 elems = slab_list->config.elements_per_slab * (1 + slab_list->config.num_children); \
276 elem_size = slab_list->config.elements_per_slab * (sizeof(_name ## _slab_element_t) + \
277 slab_list->config.child_pool_size); \
278 MEM(slab = talloc_zero_pooled_object(slab_list, _name ## _slab_t, elems, elem_size)); \
279 talloc_set_destructor(slab, _ ## _name ## _slab_free); \
280 _name ## _slab_element_init(&slab->avail); \
281 _name ## _slab_element_init(&slab->reserved); \
282 _name ## _slab_insert_head(&slab_list->avail, slab); \
283 slab->list = slab_list; \
284 for (count = 0; count < slab_list->config.elements_per_slab; count++) { \
285 if (slab_list->config.num_children > 0) { \
286 MEM(new_element = talloc_zero_pooled_object(slab, _name ## _slab_element_t, \
287 slab_list->config.num_children, \
288 slab_list->config.child_pool_size)); \
290 MEM(new_element = talloc_zero(slab, _name ## _slab_element_t)); \
292 talloc_set_type(new_element, _type); \
293 talloc_set_destructor(new_element, _ ## _type ## _element_free); \
294 _name ## _slab_element_insert_tail(&slab->avail, new_element); \
295 new_element->slab = slab; \
301 if (slab_list->alloc) { \
302 _name ## _slab_element_t *prev = NULL; \
303 new_element = NULL; \
304 while ((new_element = _name ## _slab_element_next(&slab->avail, new_element))) { \
305 if (slab_list->alloc((_type *)new_element, slab_list->uctx) < 0) { \
306 prev = _name ## _slab_element_remove(&slab->avail, new_element); \
307 talloc_free(new_element); \
308 new_element = prev; \
313 slab_list->high_water_mark += _name ## _slab_element_num_elements(&slab->avail); \
315 if (!slab && slab_list->config.at_max_fail) return NULL; \
316 if (slab) element = slab_list->reserve_mru ? _name ## _slab_element_pop_tail(&slab->avail) : \
317 _name ## _slab_element_pop_head(&slab->avail); \
319 _name ## _slab_element_insert_tail(&slab->reserved, element); \
320 if (_name ## _slab_element_num_elements(&slab->avail) == 0) { \
321 _name ## _slab_remove(&slab_list->avail, slab); \
322 _name ## _slab_insert_tail(&slab_list->reserved, slab); \
324 element->in_use = true; \
325 slab_list->in_use++; \
327 MEM(element = talloc_zero(slab_list, _name ## _slab_element_t)); \
328 talloc_set_type(element, _type); \
329 talloc_set_destructor(element, _ ## _type ## _element_free); \
330 if (slab_list->alloc) slab_list->alloc((_type *)element, slab_list->uctx); \
332 if (slab_list->reserve) slab_list->reserve((_type *)element, slab_list->uctx); \
333 return (_type *)element; \
342 static inline CC_HINT(nonnull(1,2)) void _name ## _slab_element_set_destructor(_type *elem, _type ## _slab_free_t func, void *uctx) \
344 _name ## _slab_element_t *element = (_name ## _slab_element_t *)elem; \
345 element->free = func; \
346 element->uctx = uctx; \
356 static inline CC_HINT(nonnull) void _name ## _slab_release(_type *elem) \
358 _name ## _slab_element_t *element = (_name ## _slab_element_t *)elem; \
359 _name ## _slab_t *slab = element->slab; \
360 if (element->free) element->free(elem, element->uctx); \
362 _name ## _slab_list_t *slab_list; \
363 slab_list = slab->list; \
364 _name ## _slab_element_remove(&slab->reserved, element); \
365 if (slab_list->release_reset){ \
366 talloc_free_children(element); \
367 memset(&element->elem, 0, sizeof(_type)); \
368 element->free = NULL; \
369 element->uctx = NULL; \
371 _name ## _slab_element_insert_tail(&slab->avail, element); \
372 if (_name ## _slab_element_num_elements(&slab->avail) == 1) { \
373 _name ## _slab_remove(&slab_list->reserved, slab); \
374 _name ## _slab_insert_tail(&slab_list->avail, slab); \
376 slab_list->in_use--; \
377 element->in_use = false; \
380 talloc_free(element); \
383 static inline CC_HINT(nonnull) unsigned int _name ## _slab_num_elements_used(_name ## _slab_list_t *slab_list) \
385 return slab_list->in_use; \
388 static inline CC_HINT(nonnull) unsigned int _name ## _slab_num_allocated(_name ## _slab_list_t *slab_list) \
390 return _name ## _slab_num_elements(&slab_list->reserved) + \
391 _name ## _slab_num_elements(&slab_list->avail); \
393DIAG_ON(unused-function)
fr_time_delta_t interval
Interval between slab cleanup events being fired.
unsigned int min_elements
Minimum number of elements to keep allocated.
unsigned int max_elements
Maximum number of elements to allocate using slabs.
bool at_max_fail
Should requests for additional elements fail when the number in use has reached max_elements.
unsigned int elements_per_slab
Number of elements to allocate per slab.
unsigned int num_children
How many child allocations are expected off each element.
size_t child_pool_size
Size of pool space to be allocated to each element.
bool allow_direct_free
Allow in-use elements to be freed with talloc_free.
Tuneable parameters for slabs.
A time delta, a difference in time measured in nanoseconds.