diff options
Diffstat (limited to 'src/lib/bsearch-insert-pos.c')
-rw-r--r-- | src/lib/bsearch-insert-pos.c | 48 |
1 files changed, 48 insertions, 0 deletions
diff --git a/src/lib/bsearch-insert-pos.c b/src/lib/bsearch-insert-pos.c new file mode 100644 index 0000000..dc92f47 --- /dev/null +++ b/src/lib/bsearch-insert-pos.c @@ -0,0 +1,48 @@ +/* Copyright (c) 2005-2018 Dovecot authors, see the included COPYING file */ + +#include "lib.h" +#include "array.h" +#include "bsearch-insert-pos.h" + +#undef bsearch_insert_pos +bool bsearch_insert_pos(const void *key, const void *base, unsigned int nmemb, + size_t size, int (*cmp)(const void *, const void *), + unsigned int *idx_r) +{ + const void *p; + unsigned int idx, left_idx, right_idx; + int ret; + + i_assert(nmemb < INT_MAX); + + idx = 0; left_idx = 0; right_idx = nmemb; + while (left_idx < right_idx) { + idx = (left_idx + right_idx) / 2; + + p = CONST_PTR_OFFSET(base, idx * size); + ret = cmp(key, p); + if (ret > 0) + left_idx = idx+1; + else if (ret < 0) + right_idx = idx; + else { + *idx_r = idx; + return TRUE; + } + } + + if (left_idx > idx) + idx++; + + *idx_r = idx; + return FALSE; +} + +bool array_bsearch_insert_pos_i(const struct array *array, const void *key, + int (*cmp)(const void *, const void *), + unsigned int *idx_r) +{ + return bsearch_insert_pos(key, array->buffer->data, + array_count_i(array), + array->element_size, cmp, idx_r); +} |