The FreeRADIUS server $Id: 15bac2a4c627c01d1aa2047687b3418955ac7f00 $
|
Red/black tree implementation. More...
#include <freeradius-devel/util/misc.h>
#include <stdbool.h>
#include <stdint.h>
Go to the source code of this file.
Data Structures | |
struct | fr_rb_iter_inorder_t |
Iterator structure for in-order traversal of an rbtree. More... | |
struct | fr_rb_iter_postorder_t |
Iterator structure for post-order traversal of an rbtree. More... | |
struct | fr_rb_iter_preorder_t |
Iterator structure for pre-order traversal of an rbtree. More... | |
struct | fr_rb_node_s |
struct | fr_rb_tree_s |
The main red black tree structure. More... | |
Macros | |
#define | fr_rb_alloc(_ctx, _data_cmp, _data_free) _fr_rb_alloc(_ctx, -1, NULL, _data_cmp, _data_free) |
Allocs a red black tree. | |
#define | fr_rb_init(_tree, _node_ctx, _data_cmp, _data_free) _fr_rb_init(_tree, _node_ctx, -1, NULL, _data_cmp, _data_free) |
Initialises a red black tree. | |
#define | fr_rb_inline_alloc(_ctx, _type, _field, _data_cmp, _data_free) |
Allocs a red black tree. | |
#define | fr_rb_inline_init(_tree, _type, _field, _data_cmp, _data_free) |
Initialises a red black tree. | |
#define | fr_rb_inline_talloc_alloc(_ctx, _type, _field, _data_cmp, _data_free) |
Allocs a red black that verifies elements are of a specific talloc type. | |
#define | fr_rb_inline_talloc_init(_tree, _type, _field, _data_cmp, _data_free) |
Initialises a red black that verifies elements are of a specific talloc type. | |
#define | fr_rb_inorder_foreach(_tree, _type, _iter) |
#define | fr_rb_postorder_foreach(_tree, _type, _iter) |
#define | fr_rb_preorder_foreach(_tree, _type, _iter) |
#define | fr_rb_talloc_alloc(_ctx, _type, _data_cmp, _data_free) _fr_rb_alloc(_ctx, -1, #_type, _data_cmp, _data_free) |
Allocs a red black that verifies elements are of a specific talloc type. | |
#define | fr_rb_talloc_init(_tree, _node_ctx, _type, _data_cmp, _data_free) _fr_rb_init(_tree, _node_ctx, -1, #_type, _data_cmp, _data_free) |
Initialises a red black that verifies elements are of a specific talloc type. | |
Typedefs | |
typedef struct fr_rb_node_s | fr_rb_node_t |
typedef struct fr_rb_tree_s | fr_rb_tree_t |
typedef fr_rb_node_t *(* | rb_node_alloc_t) (fr_rb_tree_t const *tree, void *data) |
Callback used to alloc rbnodes. | |
typedef void(* | rb_node_free_t) (fr_rb_tree_t const *tree, fr_rb_node_t *node, bool free_data) |
Callback used to free rbnodes. | |
Enumerations | |
enum | fr_rb_colour_t { BLACK , RED } |
Functions | |
fr_rb_tree_t * | _fr_rb_alloc (TALLOC_CTX *ctx, ssize_t offset, char const *type, fr_cmp_t data_cmp, fr_free_t data_free) |
Alloc a new RED-BLACK tree. | |
int | _fr_rb_init (fr_rb_tree_t *tree, TALLOC_CTX *node_ctx, ssize_t offset, char const *type, fr_cmp_t data_cmp, fr_free_t data_free) |
Initialise a new RED-BLACK tree. | |
bool | fr_rb_delete (fr_rb_tree_t *tree, void const *data) |
Remove node and free data (if a free function was specified) | |
bool | fr_rb_delete_by_inline_node (fr_rb_tree_t *tree, fr_rb_node_t *node) |
Remove node and free data (if a free function was specified) | |
void * | fr_rb_find (fr_rb_tree_t const *tree, void const *data) |
Find an element in the tree, returning the data, not the node. | |
int | fr_rb_find_or_insert (void **found, fr_rb_tree_t *tree, void const *data)) |
Attempt to find current data in the tree, if it does not exist insert it. | |
void * | fr_rb_first (fr_rb_tree_t *tree) |
int | fr_rb_flatten_inorder (TALLOC_CTX *ctx, void **out[], fr_rb_tree_t *tree) |
int | fr_rb_flatten_postorder (TALLOC_CTX *ctx, void **out[], fr_rb_tree_t *tree) |
int | fr_rb_flatten_preorder (TALLOC_CTX *ctx, void **out[], fr_rb_tree_t *tree) |
bool | fr_rb_insert (fr_rb_tree_t *tree, void const *data) |
Insert data into a tree. | |
void | fr_rb_iter_delete_inorder (fr_rb_iter_inorder_t *iter) |
Remove the current node from the tree. | |
void * | fr_rb_iter_init_inorder (fr_rb_iter_inorder_t *iter, fr_rb_tree_t *tree) |
Initialise an in-order iterator. | |
void * | fr_rb_iter_init_postorder (fr_rb_iter_postorder_t *iter, fr_rb_tree_t *tree) |
Initialise a post-order iterator. | |
void * | fr_rb_iter_init_preorder (fr_rb_iter_preorder_t *iter, fr_rb_tree_t *tree) |
Initialise a pre-order iterator. | |
void * | fr_rb_iter_next_inorder (fr_rb_iter_inorder_t *iter) |
Return the next node. | |
void * | fr_rb_iter_next_postorder (fr_rb_iter_postorder_t *iter) |
Return the next node. | |
void * | fr_rb_iter_next_preorder (fr_rb_iter_preorder_t *iter) |
Return the next node. | |
void * | fr_rb_last (fr_rb_tree_t *tree) |
static bool | fr_rb_node_inline_in_tree (fr_rb_node_t const *node) |
Check to see if an item is in a tree by examining its inline fr_rb_node_t. | |
uint32_t | fr_rb_num_elements (fr_rb_tree_t *tree) |
Return how many nodes there are in a tree. | |
void * | fr_rb_remove (fr_rb_tree_t *tree, void const *data) |
Remove an entry from the tree, without freeing the data. | |
void * | fr_rb_remove_by_inline_node (fr_rb_tree_t *tree, fr_rb_node_t *node) |
Remove an entry from the tree, using the node structure, without freeing the data. | |
int | fr_rb_replace (void **old, fr_rb_tree_t *tree, void const *data)) |
Replace old data with new data, OR insert if there is no old. | |
struct fr_rb_iter_inorder_t |
Data Fields | ||
---|---|---|
fr_rb_node_t * | next | if non-NULL, next node cached by fr_rb_iter_delete() |
fr_rb_node_t * | node | current node–set to NULL (not NIL) by fr_rb_iter_delete() |
fr_rb_tree_t * | tree | Tree being iterated over. |
struct fr_rb_iter_postorder_t |
Data Fields | ||
---|---|---|
fr_rb_node_t * | node | current node |
fr_rb_tree_t * | tree | Tree being iterated over. |
struct fr_rb_iter_preorder_t |
Data Fields | ||
---|---|---|
fr_rb_node_t * | node | current node |
fr_rb_tree_t * | tree | Tree being iterated over. |
struct fr_rb_node_s |
Data Fields | ||
---|---|---|
bool | being_freed | Disable frees if we're currently calling a free function. |
fr_rb_colour_t | colour | Node colour (BLACK, RED) |
void * | data | data stored in node |
fr_rb_node_t * | left | Left child. |
fr_rb_node_t * | parent | Parent. |
fr_rb_node_t * | right | Right child. |
struct fr_rb_tree_s |
Data Fields | ||
---|---|---|
bool | being_freed | Prevent double frees in talloc_destructor. |
fr_cmp_t | data_cmp | Callback to compare node data. |
fr_free_t | data_free | Callback to free node data. |
uint32_t | magic | |
rb_node_alloc_t | node_alloc | Callback to allocate a new node. |
TALLOC_CTX * | node_ctx | Talloc ctx to allocate nodes in. |
rb_node_free_t | node_free | Callback to free a node. |
uint32_t | num_elements | How many elements are inside the tree. |
uint16_t | offset | Where's the fr_rb_node_t is located in the structure being inserted. |
fr_rb_node_t * | root | Root of the rbtree. |
char const * | type | Talloc type to check elements against. |
#define fr_rb_alloc | ( | _ctx, | |
_data_cmp, | |||
_data_free | |||
) | _fr_rb_alloc(_ctx, -1, NULL, _data_cmp, _data_free) |
Allocs a red black tree.
This variant allocates an fr_rb_node_t on the heap. This allows the data structure to be inserted into multiple trees.
[in] | _ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_init | ( | _tree, | |
_node_ctx, | |||
_data_cmp, | |||
_data_free | |||
) | _fr_rb_init(_tree, _node_ctx, -1, NULL, _data_cmp, _data_free) |
Initialises a red black tree.
This variant initiates an fr_rb_node_t on the heap. This allows the data structure to be inserted into multiple trees.
[out] | _tree | to initialise. |
[in] | _node_ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_inline_alloc | ( | _ctx, | |
_type, | |||
_field, | |||
_data_cmp, | |||
_data_free | |||
) |
Allocs a red black tree.
This variant stores fr_rb_node_t data inline with the data structure to avoid allocating fr_rb_node_t on the heap.
It is suitable for use where the data structure will only be inserted into a fixed set of trees.
[in] | _ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _field | Containing the fr_rb_node_t within item being stored. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_inline_init | ( | _tree, | |
_type, | |||
_field, | |||
_data_cmp, | |||
_data_free | |||
) |
Initialises a red black tree.
This variant stores fr_rb_node_t data inline with the data structure to avoid initiating fr_rb_node_t on the heap.
It is suitable for use where the data structure will only be inserted into a fixed set of trees.
[out] | _tree | to initialise. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _field | Containing the fr_rb_node_t within item being stored. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_inline_talloc_alloc | ( | _ctx, | |
_type, | |||
_field, | |||
_data_cmp, | |||
_data_free | |||
) |
Allocs a red black that verifies elements are of a specific talloc type.
This variant stores fr_rb_node_t data inline with the data structure to avoid allocating fr_rb_node_t on the heap.
It is suitable for use where the data structure will only be inserted into a fixed set of trees.
[in] | _ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _field | Containing the fr_rb_node_t within item being stored. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_inline_talloc_init | ( | _tree, | |
_type, | |||
_field, | |||
_data_cmp, | |||
_data_free | |||
) |
Initialises a red black that verifies elements are of a specific talloc type.
This variant stores fr_rb_node_t data inline with the data structure to avoid initiating fr_rb_node_t on the heap.
It is suitable for use where the data structure will only be inserted into a fixed set of trees.
[out] | _tree | to initialise. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _field | Containing the fr_rb_node_t within item being stored. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_inorder_foreach | ( | _tree, | |
_type, | |||
_iter | |||
) |
#define fr_rb_postorder_foreach | ( | _tree, | |
_type, | |||
_iter | |||
) |
#define fr_rb_preorder_foreach | ( | _tree, | |
_type, | |||
_iter | |||
) |
#define fr_rb_talloc_alloc | ( | _ctx, | |
_type, | |||
_data_cmp, | |||
_data_free | |||
) | _fr_rb_alloc(_ctx, -1, #_type, _data_cmp, _data_free) |
Allocs a red black that verifies elements are of a specific talloc type.
This variant allocates an fr_rb_node_t on the heap. This allows the data structure to be inserted into multiple trees.
[in] | _ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
#define fr_rb_talloc_init | ( | _tree, | |
_node_ctx, | |||
_type, | |||
_data_cmp, | |||
_data_free | |||
) | _fr_rb_init(_tree, _node_ctx, -1, #_type, _data_cmp, _data_free) |
Initialises a red black that verifies elements are of a specific talloc type.
This variant allocates an fr_rb_node_t on the heap. This allows the data structure to be inserted into multiple trees.
[out] | _tree | to initialise. |
[in] | _node_ctx | to tie tree lifetime to. If ctx is freed, tree will free any nodes, calling the free function if set. |
[in] | _type | of item being stored in the tree, e.g. fr_value_box_t. |
[in] | _data_cmp | Callback to compare node data. |
[in] | _data_free | Optional function used to free data if tree nodes are deleted or replaced. |
typedef struct fr_rb_node_s fr_rb_node_t |
typedef struct fr_rb_tree_s fr_rb_tree_t |
typedef fr_rb_node_t *(* rb_node_alloc_t) (fr_rb_tree_t const *tree, void *data) |
typedef void(* rb_node_free_t) (fr_rb_tree_t const *tree, fr_rb_node_t *node, bool free_data) |
enum fr_rb_colour_t |
fr_rb_tree_t * _fr_rb_alloc | ( | TALLOC_CTX * | ctx, |
ssize_t | offset, | ||
char const * | type, | ||
fr_cmp_t | data_cmp, | ||
fr_free_t | data_free | ||
) |
Alloc a new RED-BLACK tree.
[in] | ctx | to allocate the tree in. Only the tree is allocated in this context, the memory for the fr_rb_node_t is allocated as part of the data being inserted into the tree. |
[in] | offset | offsetof the fr_rb_node_t field in the data being inserted. If < 0, nodes will be allocated on the heap. |
[in] | type | Talloc type of structures being inserted, may be NULL. |
[in] | data_cmp | Comparator function for ordering data in the tree. |
[in] | data_free | Free function to call whenever data is deleted or replaced. |
Definition at line 202 of file rb.c.
int _fr_rb_init | ( | fr_rb_tree_t * | tree, |
TALLOC_CTX * | node_ctx, | ||
ssize_t | offset, | ||
char const * | type, | ||
fr_cmp_t | data_cmp, | ||
fr_free_t | data_free | ||
) |
Initialise a new RED-BLACK tree.
[out] | tree | to initialise. |
[in] | node_ctx | the ctx used to allocate fr_rb_node_t if the tree isn't using inline fr_rb_node_t. |
[in] | offset | offsetof the fr_rb_node_t field in the data being inserted. If < 0, nodes will be allocated on the heap. |
[in] | type | Talloc type of structures being inserted, may be NULL. |
[in] | data_cmp | Comparator function for ordering data in the tree. |
[in] | data_free | Free function to call whenever data is deleted or replaced. |
Definition at line 147 of file rb.c.
bool fr_rb_delete | ( | fr_rb_tree_t * | tree, |
void const * | data | ||
) |
Remove node and free data (if a free function was specified)
[in] | tree | to remove data from. |
[in] | data | to remove/free. |
Definition at line 741 of file rb.c.
bool fr_rb_delete_by_inline_node | ( | fr_rb_tree_t * | tree, |
fr_rb_node_t * | node | ||
) |
Remove node and free data (if a free function was specified)
This function can help where multiple items may be in the tree with the same comparator value.
[in] | tree | to remove data from. |
[in] | node | to remove/free. |
Definition at line 767 of file rb.c.
void * fr_rb_find | ( | fr_rb_tree_t const * | tree, |
void const * | data | ||
) |
int fr_rb_find_or_insert | ( | void ** | found, |
fr_rb_tree_t * | tree, | ||
void const * | data | ||
) |
Attempt to find current data in the tree, if it does not exist insert it.
[out] | found | Pre-existing data we found. |
[in] | tree | to search/insert into. |
[in] | data | to find. |
Definition at line 598 of file rb.c.
void * fr_rb_first | ( | fr_rb_tree_t * | tree | ) |
int fr_rb_flatten_inorder | ( | TALLOC_CTX * | ctx, |
void ** | out[], | ||
fr_rb_tree_t * | tree | ||
) |
int fr_rb_flatten_postorder | ( | TALLOC_CTX * | ctx, |
void ** | out[], | ||
fr_rb_tree_t * | tree | ||
) |
int fr_rb_flatten_preorder | ( | TALLOC_CTX * | ctx, |
void ** | out[], | ||
fr_rb_tree_t * | tree | ||
) |
bool fr_rb_insert | ( | fr_rb_tree_t * | tree, |
void const * | data | ||
) |
void fr_rb_iter_delete_inorder | ( | fr_rb_iter_inorder_t * | iter | ) |
Remove the current node from the tree.
[in] | iter | previously initialised with fr_rb_iter_init_inorder |
Definition at line 898 of file rb.c.
void * fr_rb_iter_init_inorder | ( | fr_rb_iter_inorder_t * | iter, |
fr_rb_tree_t * | tree | ||
) |
void * fr_rb_iter_init_postorder | ( | fr_rb_iter_postorder_t * | iter, |
fr_rb_tree_t * | tree | ||
) |
void * fr_rb_iter_init_preorder | ( | fr_rb_iter_preorder_t * | iter, |
fr_rb_tree_t * | tree | ||
) |
void * fr_rb_iter_next_inorder | ( | fr_rb_iter_inorder_t * | iter | ) |
Return the next node.
[in] | iter | previously initialised with fr_rb_iter_init_inorder |
Definition at line 850 of file rb.c.
void * fr_rb_iter_next_postorder | ( | fr_rb_iter_postorder_t * | iter | ) |
Return the next node.
[in] | iter | previously initialised with fr_rb_iter_init_postorder |
Definition at line 1025 of file rb.c.
void * fr_rb_iter_next_preorder | ( | fr_rb_iter_preorder_t * | iter | ) |
Return the next node.
[in] | iter | previously initialised with fr_rb_iter_init_preorder |
Definition at line 941 of file rb.c.
void * fr_rb_last | ( | fr_rb_tree_t * | tree | ) |
|
inlinestatic |
Check to see if an item is in a tree by examining its inline fr_rb_node_t.
This works because we use NIL sentinels to represent the absence of a child or parent. When the node is initialised all these fields should be NULL and when it's removed from the tree, the "free" function for inline nodes also sets all of these back to NULL.
[in] | node | to check. |
Definition at line 314 of file rb.h.
uint32_t fr_rb_num_elements | ( | fr_rb_tree_t * | tree | ) |
void * fr_rb_remove | ( | fr_rb_tree_t * | tree, |
void const * | data | ||
) |
Remove an entry from the tree, without freeing the data.
[in] | tree | to remove data from. |
[in] | data | to remove. |
Definition at line 695 of file rb.c.
void * fr_rb_remove_by_inline_node | ( | fr_rb_tree_t * | tree, |
fr_rb_node_t * | node | ||
) |
Remove an entry from the tree, using the node structure, without freeing the data.
This function can help where multiple items may be in the tree with the same comparator value.
[in] | tree | to remove data from. |
[in] | node | to remove. |
Definition at line 722 of file rb.c.
int fr_rb_replace | ( | void ** | old, |
fr_rb_tree_t * | tree, | ||
void const * | data | ||
) |
Replace old data with new data, OR insert if there is no old.
[out] | old | data that was replaced. If this argument is not NULL, then the old data will not be freed, even if a free function is configured. |
[in] | tree | to insert data into. |
[in] | data | to replace. |
Definition at line 647 of file rb.c.