From 9c056f44f8114f45e151f9538486d80d629d50ea Mon Sep 17 00:00:00 2001 From: Jean-Roland Date: Fri, 20 Dec 2024 10:18:00 +0100 Subject: [PATCH] feat: switch tree to sorted array --- include/zenoh-pico/collections/lru_cache.h | 15 +- include/zenoh-pico/config.h | 9 - include/zenoh-pico/config.h.in | 9 - src/collections/lru_cache.c | 219 ++++++--------------- tests/z_lru_cache_test.c | 9 +- 5 files changed, 74 insertions(+), 187 deletions(-) diff --git a/include/zenoh-pico/collections/lru_cache.h b/include/zenoh-pico/collections/lru_cache.h index f60eca4c1..1bed89b27 100644 --- a/include/zenoh-pico/collections/lru_cache.h +++ b/include/zenoh-pico/collections/lru_cache.h @@ -15,9 +15,10 @@ #ifndef ZENOH_PICO_COLLECTIONS_LRUCACHE_H #define ZENOH_PICO_COLLECTIONS_LRUCACHE_H +#include #include -#include "zenoh-pico/collections/element.h" +#include "zenoh-pico/utils/result.h" #ifdef __cplusplus extern "C" { @@ -34,13 +35,11 @@ typedef void _z_lru_cache_node_t; * A least recently used cache implementation */ typedef struct _z_lru_cache_t { - size_t capacity; // Max number of node - size_t len; // Number of node - _z_lru_cache_node_t *head; // List head - _z_lru_cache_node_t *tail; // List tail -#if Z_FEATURE_CACHE_TREE == 1 - _z_lru_cache_node_t *root; // Tree root -#endif + size_t capacity; // Max number of node + size_t len; // Number of node + _z_lru_cache_node_t *head; // List head + _z_lru_cache_node_t *tail; // List tail + _z_lru_cache_node_t **slist; // Sorted node list } _z_lru_cache_t; _z_lru_cache_t _z_lru_cache_init(size_t capacity); diff --git a/include/zenoh-pico/config.h b/include/zenoh-pico/config.h index d5883e121..3542c7d1d 100644 --- a/include/zenoh-pico/config.h +++ b/include/zenoh-pico/config.h @@ -180,15 +180,6 @@ */ #define Z_RX_CACHE_SIZE 10 -/** - * Use binary tree to speed up rx cache search - */ -#if Z_RX_CACHE_SIZE > 20 -#define Z_FEATURE_CACHE_TREE 1 -#else -#define Z_FEATURE_CACHE_TREE 0 -#endif - /** * Default get timeout in milliseconds. */ diff --git a/include/zenoh-pico/config.h.in b/include/zenoh-pico/config.h.in index 2586d7e9e..a3748c432 100644 --- a/include/zenoh-pico/config.h.in +++ b/include/zenoh-pico/config.h.in @@ -180,15 +180,6 @@ */ #define Z_RX_CACHE_SIZE 10 -/** - * Use binary tree to speed up rx cache search - */ -#if Z_RX_CACHE_SIZE > 20 -#define Z_FEATURE_CACHE_TREE 1 -#else -#define Z_FEATURE_CACHE_TREE 0 -#endif - /** * Default get timeout in milliseconds. */ diff --git a/src/collections/lru_cache.c b/src/collections/lru_cache.c index 81505755e..aa0f62bd9 100644 --- a/src/collections/lru_cache.c +++ b/src/collections/lru_cache.c @@ -23,14 +23,10 @@ #include "zenoh-pico/utils/pointers.h" #include "zenoh-pico/utils/result.h" -// Nodes are chained both as a binary tree for lookup and as double linked list for lru insertion/deletion. +// Nodes are chained as double linked list for lru insertion/deletion. typedef struct _z_lru_cache_node_data_t { _z_lru_cache_node_t *prev; // List previous node _z_lru_cache_node_t *next; // List next node -#if Z_FEATURE_CACHE_TREE == 1 - _z_lru_cache_node_t *left; // Tree left child - _z_lru_cache_node_t *right; // Tree right child -#endif } _z_lru_cache_node_data_t; #define NODE_DATA_SIZE sizeof(_z_lru_cache_node_data_t) @@ -106,176 +102,79 @@ static void _z_lru_cache_update_list(_z_lru_cache_t *cache, _z_lru_cache_node_t _z_lru_cache_insert_list_node(cache, node); } -// Tree functions -#if Z_FEATURE_CACHE_TREE == 1 -static _z_lru_cache_node_t *_z_lru_cache_insert_tree_node_rec(_z_lru_cache_node_t *previous, _z_lru_cache_node_t *node, - _z_lru_val_cmp_f compare) { - if (previous == NULL) { - return node; - } - _z_lru_cache_node_data_t *prev_data = _z_lru_cache_node_data(previous); - int res = compare(_z_lru_cache_node_value(previous), _z_lru_cache_node_value(node)); - if (res < 0) { - prev_data->left = _z_lru_cache_insert_tree_node_rec(prev_data->left, node, compare); - } else if (res > 0) { - prev_data->right = _z_lru_cache_insert_tree_node_rec(prev_data->right, node, compare); - } - // == 0 shouldn't happen - else { - _Z_ERROR("Comparison equality shouldn't happen during tree insertion"); - } - return previous; -} - -static void _z_lru_cache_insert_tree_node(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, _z_lru_val_cmp_f compare) { - cache->root = _z_lru_cache_insert_tree_node_rec(cache->root, node, compare); -} - -static _z_lru_cache_node_t *_z_lru_cache_search_tree_rec(_z_lru_cache_node_t *node, void *value, - _z_lru_val_cmp_f compare) { - // Check value - if (node == NULL) { - return NULL; - } - // Compare keys - int res = compare(_z_lru_cache_node_value(node), value); - // Node has key - if (res == 0) { - return node; - } - _z_lru_cache_node_data_t *node_data = _z_lru_cache_node_data(node); - // Key is greater than node's key - if (res > 0) { - return _z_lru_cache_search_tree_rec(node_data->right, value, compare); - } - // Key is smaller than node's key - return _z_lru_cache_search_tree_rec(node_data->left, value, compare); -} - -static _z_lru_cache_node_t *_z_lru_cache_search_tree(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f compare) { - return _z_lru_cache_search_tree_rec(cache->root, value, compare); -} - -static _z_lru_cache_node_t *_z_lru_cache_get_tree_node_successor(_z_lru_cache_node_t *node) { - _z_lru_cache_node_data_t *node_data = _z_lru_cache_node_data(node); - node = node_data->right; - node_data = _z_lru_cache_node_data(node); - while ((node != NULL) && (node_data->left != NULL)) { - node = node_data->left; - node_data = _z_lru_cache_node_data(node); +// Sorted list function +static _z_lru_cache_node_t *_z_lru_cache_search_slist(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f compare, + size_t *idx) { + ssize_t l_idx = 0; + ssize_t h_idx = (ssize_t)cache->len - 1; + while (l_idx <= h_idx) { + ssize_t curr_idx = (l_idx + h_idx) / 2; + int res = compare(_z_lru_cache_node_value(cache->slist[curr_idx]), value); + if (res == 0) { + *idx = (size_t)curr_idx; + return cache->slist[curr_idx]; + } else if (res < 0) { + l_idx = curr_idx + 1; + } else { + h_idx = curr_idx - 1; + } } - return node; + return NULL; } -static void _z_lru_cache_transfer_tree_node(_z_lru_cache_node_t *dst, _z_lru_cache_node_t *src, size_t value_size) { - void *src_val = _z_lru_cache_node_value(src); - void *dst_val = _z_lru_cache_node_value(dst); - _z_lru_cache_node_data_t *src_data = _z_lru_cache_node_data(src); - _z_lru_cache_node_data_t *dst_data = _z_lru_cache_node_data(dst); - // Copy content - memcpy(dst_val, src_val, value_size); - dst_data->prev = src_data->prev; - dst_data->next = src_data->next; - // Update list - if (src_data->prev != NULL) { - _z_lru_cache_node_data_t *prev_data = _z_lru_cache_node_data(src_data->prev); - prev_data->next = dst; - } - if (src_data->next != NULL) { - _z_lru_cache_node_data_t *next_data = _z_lru_cache_node_data(src_data->next); - next_data->prev = dst; +static void _z_lru_cache_sort_slist(_z_lru_cache_node_t **slist, size_t slist_size, _z_lru_val_cmp_f compare) { + for (size_t i = 1; i < slist_size; i++) { + _z_lru_cache_node_t *node = slist[i]; + void *node_val = _z_lru_cache_node_value(node); + ssize_t j = (ssize_t)i - 1; + while ((j >= 0) && (compare(_z_lru_cache_node_value(slist[j]), node_val) > 0)) { + slist[j + 1] = slist[j]; + j--; + } + slist[j + 1] = node; } } -static _z_lru_cache_node_t *_z_lru_cache_delete_tree_node_rec(_z_lru_cache_node_t *node, void *value, size_t value_size, - _z_lru_val_cmp_f compare) { - // No node - if (node == NULL) { - return NULL; - } - // Compare value - _z_lru_cache_node_data_t *node_data = _z_lru_cache_node_data(node); - void *node_val = _z_lru_cache_node_value(node); - int res = compare(node_val, value); - if (res < 0) { - node_data->left = _z_lru_cache_delete_tree_node_rec(node_data->left, value, value_size, compare); - } else if (res > 0) { - node_data->right = _z_lru_cache_delete_tree_node_rec(node_data->right, value, value_size, compare); - } else { // Node found - // No child / one child - if (node_data->left == NULL) { - _z_lru_cache_node_t *temp = node_data->right; - z_free(node); - return temp; - } else if (node_data->right == NULL) { - _z_lru_cache_node_t *temp = node_data->left; - z_free(node); - return temp; - } - // Two children - _z_lru_cache_node_t *succ = _z_lru_cache_get_tree_node_successor(node); - // Transfer successor data to node - _z_lru_cache_transfer_tree_node(node, succ, value_size); - node_data->right = - _z_lru_cache_delete_tree_node_rec(node_data->right, _z_lru_cache_node_value(succ), value_size, compare); +static void _z_lru_cache_insert_slist(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, _z_lru_val_cmp_f compare, + size_t *idx) { + if (idx != NULL) { + assert(*idx < cache->capacity); + cache->slist[*idx] = node; + } else { + assert(cache->len < cache->capacity); + assert(cache->slist[cache->len] == NULL); + cache->slist[cache->len] = node; } - return node; + _z_lru_cache_sort_slist(cache->slist, cache->len + 1, compare); } -static void _z_lru_cache_delete_tree_node(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, size_t value_size, - _z_lru_val_cmp_f compare) { - cache->root = _z_lru_cache_delete_tree_node_rec(cache->root, _z_lru_cache_node_value(node), value_size, compare); +static size_t _z_lru_cache_delete_slist(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, _z_lru_val_cmp_f compare) { + size_t idx = 0; + (void)_z_lru_cache_search_slist(cache, _z_lru_cache_node_value(node), compare, &idx); + return idx; } -#else -_z_lru_cache_node_t *_z_lru_cache_search_list(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f compare) { - if (cache->head == NULL) { - return NULL; - } - _z_lru_cache_node_data_t *node = cache->head; - // Parse list - while (node != NULL) { - _z_lru_cache_node_data_t *node_data = _z_lru_cache_node_data(node); - void *node_val = _z_lru_cache_node_value(node); - if (compare(node_val, value) == 0) { - return node; - } - node = node_data->next; - } - return NULL; -} -#endif // Main static functions -static void _z_lru_cache_delete_last(_z_lru_cache_t *cache, size_t value_size, _z_lru_val_cmp_f compare) { +static size_t _z_lru_cache_delete_last(_z_lru_cache_t *cache, _z_lru_val_cmp_f compare) { _z_lru_cache_node_t *last = cache->tail; assert(last != NULL); _z_lru_cache_remove_list_node(cache, last); -#if Z_FEATURE_CACHE_TREE == 1 - _z_lru_cache_delete_tree_node(cache, last, value_size, compare); -#else - _ZP_UNUSED(compare); - _ZP_UNUSED(value_size); + size_t idx = _z_lru_cache_delete_slist(cache, last, compare); z_free(last); -#endif cache->len--; + return idx; } -static void _z_lru_cache_insert_node(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, _z_lru_val_cmp_f compare) { +static void _z_lru_cache_insert_node(_z_lru_cache_t *cache, _z_lru_cache_node_t *node, _z_lru_val_cmp_f compare, + size_t *idx) { _z_lru_cache_insert_list_node(cache, node); -#if Z_FEATURE_CACHE_TREE == 1 - _z_lru_cache_insert_tree_node(cache, node, compare); -#else - _ZP_UNUSED(compare); -#endif + _z_lru_cache_insert_slist(cache, node, compare, idx); cache->len++; } static _z_lru_cache_node_t *_z_lru_cache_search_node(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f compare) { -#if Z_FEATURE_CACHE_TREE == 1 - return _z_lru_cache_search_tree(cache, value, compare); -#else - return _z_lru_cache_search_list(cache, value, compare); -#endif + size_t idx = 0; + return _z_lru_cache_search_slist(cache, value, compare, &idx); } // Public functions @@ -286,7 +185,7 @@ _z_lru_cache_t _z_lru_cache_init(size_t capacity) { } void *_z_lru_cache_get(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f compare) { - // Lookup tree if node exists. + // Lookup if node exists. _z_lru_cache_node_t *node = _z_lru_cache_search_node(cache, value, compare); if (node == NULL) { return NULL; @@ -298,23 +197,35 @@ void *_z_lru_cache_get(_z_lru_cache_t *cache, void *value, _z_lru_val_cmp_f comp z_result_t _z_lru_cache_insert(_z_lru_cache_t *cache, void *value, size_t value_size, _z_lru_val_cmp_f compare) { assert(cache->capacity > 0); + // Init slist + if (cache->slist == NULL) { + cache->slist = (_z_lru_cache_node_t **)z_malloc(cache->capacity * sizeof(void *)); + if (cache->slist == NULL) { + return _Z_ERR_SYSTEM_OUT_OF_MEMORY; + } + memset(cache->slist, 0, cache->capacity * sizeof(void *)); + } // Create node _z_lru_cache_node_t *node = _z_lru_cache_node_create(value, value_size); if (node == NULL) { return _Z_ERR_SYSTEM_OUT_OF_MEMORY; } // Check capacity + size_t *idx = NULL; + size_t del_idx = 0; if (cache->len == cache->capacity) { // Delete lru entry - _z_lru_cache_delete_last(cache, value_size, compare); + del_idx = _z_lru_cache_delete_last(cache, compare); + idx = &del_idx; } // Update the cache - _z_lru_cache_insert_node(cache, node, compare); + _z_lru_cache_insert_node(cache, node, compare, idx); return _Z_RES_OK; } void _z_lru_cache_delete(_z_lru_cache_t *cache) { _z_lru_cache_node_data_t *node = cache->head; + z_free(cache->slist); // Parse list while (node != NULL) { _z_lru_cache_node_t *tmp = node; diff --git a/tests/z_lru_cache_test.c b/tests/z_lru_cache_test.c index 64dc4e7e8..daa9a5345 100644 --- a/tests/z_lru_cache_test.c +++ b/tests/z_lru_cache_test.c @@ -191,12 +191,7 @@ void *stop_task(void *ctx) { *stop_flag = true; return NULL; } -// Results -// size 10: 287720681 list, 294626805 tree (1.024) -// size 20: 207124561 list, 282312820 tree (1.363) -// size 50: 104164238 list, 212367113 tree (2.039) -// size 100: 51666425 list, 198541036 tree (3.843) -// size 1000: 5838653 list, 122717170 tree (21.018) + void test_search_benchmark(void) { _dummy_lru_cache_t dcache = _dummy_lru_cache_init(10); _dummy_t data[10] = {0}; @@ -219,7 +214,7 @@ void test_search_benchmark(void) { get_cnt++; assert(res != NULL); } - printf("Mode: %d, Get count: %ld\n", Z_FEATURE_CACHE_TREE, get_cnt); + printf("Get count: %ld\n", get_cnt); } #endif