mirror of
https://github.com/moparisthebest/hexchat
synced 2024-11-13 12:55:01 -05:00
235 lines
3.5 KiB
C
235 lines
3.5 KiB
C
/*
|
|
This is used for quick userlist insertion and lookup. It's not really
|
|
a tree, but it could be :)
|
|
*/
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "tree.h"
|
|
|
|
#define ARRAY_GROW 32
|
|
|
|
struct _tree
|
|
{
|
|
int elements;
|
|
int array_size;
|
|
void **array;
|
|
tree_cmp_func *cmp;
|
|
void *data;
|
|
};
|
|
|
|
tree *
|
|
tree_new (tree_cmp_func *cmp, void *data)
|
|
{
|
|
tree *t = calloc (1, sizeof (tree));
|
|
t->cmp = cmp;
|
|
t->data = data;
|
|
return t;
|
|
}
|
|
|
|
void
|
|
tree_destroy (tree *t)
|
|
{
|
|
if (t)
|
|
{
|
|
if (t->array)
|
|
free (t->array);
|
|
free (t);
|
|
}
|
|
}
|
|
|
|
static int
|
|
tree_find_insertion_pos (tree *t, void *key, int *done)
|
|
{
|
|
int c, u, l, idx;
|
|
|
|
if (t->elements < 1)
|
|
{
|
|
*done = 1;
|
|
t->array[0] = key;
|
|
t->elements++;
|
|
return 0;
|
|
}
|
|
|
|
if (t->elements < 2)
|
|
{
|
|
*done = 1;
|
|
c = t->cmp (key, t->array[0], t->data);
|
|
if (c == 0)
|
|
return -1;
|
|
t->elements++;
|
|
if (c > 0)
|
|
{
|
|
t->array[1] = key;
|
|
return 1;
|
|
}
|
|
t->array[1] = t->array[0];
|
|
t->array[0] = key;
|
|
return 0;
|
|
}
|
|
|
|
*done = 0;
|
|
|
|
c = t->cmp (key, t->array[0], t->data);
|
|
if (c < 0)
|
|
return 0; /* prepend */
|
|
|
|
c = t->cmp (key, t->array[t->elements - 1], t->data);
|
|
if (c > 0)
|
|
return t->elements; /* append */
|
|
|
|
l = 0;
|
|
u = t->elements - 1;
|
|
while (1)
|
|
{
|
|
idx = (l + u) / 2;
|
|
c = t->cmp (key, t->array[idx], t->data);
|
|
|
|
if (0 > c)
|
|
u = idx;
|
|
else if (0 < c && 0 > t->cmp (key, t->array[idx+1], t->data))
|
|
return idx + 1;
|
|
else if (c == 0)
|
|
return -1;
|
|
else
|
|
l = idx + 1;
|
|
}
|
|
}
|
|
|
|
static void
|
|
tree_insert_at_pos (tree *t, void *key, int pos)
|
|
{
|
|
int post_bytes;
|
|
|
|
/* append is easy */
|
|
if (pos != t->elements)
|
|
{
|
|
post_bytes = (t->elements - pos) * sizeof (void *);
|
|
memmove (&t->array[pos + 1], &t->array[pos], post_bytes);
|
|
}
|
|
|
|
t->array[pos] = key;
|
|
t->elements++;
|
|
}
|
|
|
|
static void *
|
|
mybsearch (const void *key, void **array, size_t nmemb,
|
|
int (*compar) (const void *, const void *, void *data), void *data, int *pos)
|
|
{
|
|
int l, u, idx;
|
|
int comparison;
|
|
|
|
l = 0;
|
|
u = nmemb;
|
|
while (l < u)
|
|
{
|
|
idx = (l + u) / 2;
|
|
comparison = (*compar) (key, array[idx], data);
|
|
if (comparison < 0)
|
|
u = idx;
|
|
else if (comparison > 0)
|
|
l = idx + 1;
|
|
else
|
|
{
|
|
*pos = idx;
|
|
return array[idx];
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void *
|
|
tree_find (tree *t, void *key, tree_cmp_func *cmp, void *data, int *pos)
|
|
{
|
|
if (!t || !t->array)
|
|
return NULL;
|
|
|
|
return mybsearch (key, &t->array[0], t->elements, cmp, data, pos);
|
|
}
|
|
|
|
void
|
|
tree_remove_at_pos (tree *t, int pos)
|
|
{
|
|
int post_bytes;
|
|
|
|
t->elements--;
|
|
if (pos != t->elements)
|
|
{
|
|
post_bytes = (t->elements - pos) * sizeof (void *);
|
|
memmove (&t->array[pos], &t->array[pos + 1], post_bytes);
|
|
}
|
|
}
|
|
|
|
int
|
|
tree_remove (tree *t, void *key, int *pos)
|
|
{
|
|
void *data;
|
|
|
|
data = tree_find (t, key, t->cmp, t->data, pos);
|
|
if (!data)
|
|
return 0;
|
|
|
|
tree_remove_at_pos (t, *pos);
|
|
return 1;
|
|
}
|
|
|
|
void
|
|
tree_foreach (tree *t, tree_traverse_func *func, void *data)
|
|
{
|
|
int j;
|
|
|
|
if (!t || !t->array)
|
|
return;
|
|
|
|
for (j = 0; j < t->elements; j++)
|
|
{
|
|
if (!func (t->array[j], data))
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
tree_grow (tree *t)
|
|
{
|
|
if (t->array_size < t->elements + 1)
|
|
{
|
|
int new_size = t->array_size + ARRAY_GROW;
|
|
|
|
t->array = realloc (t->array, sizeof (void *) * new_size);
|
|
t->array_size = new_size;
|
|
}
|
|
|
|
}
|
|
|
|
int
|
|
tree_insert (tree *t, void *key)
|
|
{
|
|
int pos, done;
|
|
|
|
if (!t)
|
|
return -1;
|
|
|
|
tree_grow (t);
|
|
pos = tree_find_insertion_pos (t, key, &done);
|
|
if (!done && pos != -1)
|
|
tree_insert_at_pos (t, key, pos);
|
|
|
|
return pos;
|
|
}
|
|
|
|
void
|
|
tree_append (tree *t, void *key)
|
|
{
|
|
tree_grow (t);
|
|
tree_insert_at_pos (t, key, t->elements);
|
|
}
|
|
|
|
int tree_size (tree *t)
|
|
{
|
|
return t->elements;
|
|
}
|
|
|