aboutsummaryrefslogblamecommitdiff
path: root/src/common/libkogata/btree.c
blob: dc5f11f30498faeb2d5af02de09db02fbd420333 (plain) (tree)
1
2
3
4

                          
 
                         





















                                                             
                           






                      

                                                                 
 

                                               
 




                                                     











                                
                                           



                             
                                           




                                                               
                                                  








                                                      

                               



                 
                                                   








                                                     

                               



                 
                                                  


                                                       
                                         

                                                       
                                          







                         

                                                                              
 



                                                               
         
                                  
 


                                                  





                                                                      
                               
 
                                               




                    








                                                                                     
         


                                                                               
 












                                                                     
                        
                                                                                              
 

                                                   
 


                                                         
 


                                                             
 
                                                                                                               
         


                                                     

 








                                                                                       
         


                                                                                                  
 












                                                                      
                        






                                                                                                
                 
 













                                                                                                               

 



                              









                                                                             
         

                                               
 
                                                           




                             

                                                                              
 












                                                                     

                             
                                                  


                      

                                                                              
 












                                                                    

                             
                                                  


                      
 






                                                        
                                              




                                                                                        
 


                                                       
                                  




                                                        
         
 
                                                                  
                                               


                                   
#include <kogata/malloc.h>
#include <kogata/debug.h>

#include <kogata/btree.h>

typedef struct btree_item {
	void *key, *val;

	int height;
	struct btree_item *left, *right;
} btree_item_t;

struct btree {
	key_cmp_fun_t cf;
	kv_iter_fun_t releasef;

	btree_item_t *root;
	int nitems;
};

btree_t* create_btree(key_cmp_fun_t cf, kv_iter_fun_t relf) {
	btree_t* t = (btree_t*)malloc(sizeof(btree_t));

	if (t == 0) return 0;

	t->cf = cf;
	t->releasef = relf;

	t->root = 0;
	t->nitems = 0;

	return t;
}

void _btree_delete_item_rec(btree_item_t *i,kv_iter_fun_t relf) {
	if (i == 0) return;

	_btree_delete_item_rec(i->left, relf);
	_btree_delete_item_rec(i->right, relf);

	if (relf) relf(i->key, i->val);
	free(i);
}
void delete_btree(btree_t *t) {
	_btree_delete_item_rec(t->root, t->releasef);

	free(t);
}

size_t btree_count(btree_t *i) {
	return i->nitems;
}

// ============== //
// TREE ROTATIONS //
// ============== //

static inline int height(btree_item_t *i) {
	if (i == 0) return 0;
	return i->height;
}

void btree_recalc_height(btree_item_t *i) {
	if (i == 0) return;

	i->height = MAX(height(i->left), height(i->right)) + 1;
}

btree_item_t* btree_rotate_left(btree_item_t *i) {
	// I(X(a, b), Y) -> a X b I Y -> X(a, I(b, Y))

	btree_item_t *x = i->left;
	if (x == 0) return i;
	btree_item_t *b = x->right;

	x->right = i;
	i->left = b;

	btree_recalc_height(i);
	btree_recalc_height(x);

	return x;
}

btree_item_t* btree_rotate_right(btree_item_t *i) {
	// I(X, Y(a,b)) -> X I a Y b -> Y(I(X, a), b)

	btree_item_t *y = i->right;
	if (y == 0) return i;
	btree_item_t *a = y->left;

	y->left = i;
	i->right = a;

	btree_recalc_height(i);
	btree_recalc_height(y);

	return y;
}

btree_item_t* btree_equilibrate(btree_item_t *i) {
	if (i == 0) return 0;

	while (height(i->left) - height(i->right) >= 2)
		i = btree_rotate_left(i);

	while (height(i->right) - height(i->left) >= 2)
		i = btree_rotate_right(i);
	
	return i;
}

// =================== //
// ADDING AND DELETING //
// =================== //

btree_item_t* _btree_insert(btree_t *t, btree_item_t *root, btree_item_t *i) {
	if (root == 0) return i;

	if (t->cf(i->key, root->key) <= 0) {
		root->left = _btree_insert(t, root->left, i);
	} else {
		root->right = _btree_insert(t, root->right, i);
	}
	btree_recalc_height(root);

	return btree_equilibrate(root);
}
bool btree_add(btree_t *t, void* key, void* val) {
	btree_item_t *x = (btree_item_t*)malloc(sizeof(btree_item_t));
	if (x == 0) return false;

	x->key = key;
	x->val = val;
	x->left = x->right = 0;
	btree_recalc_height(x);

	t->root = _btree_insert(t, t->root, x);
	t->nitems++;

	return true;
}

btree_item_t *_btree_extract_smallest(btree_item_t *i, btree_item_t **out_smallest) {
	ASSERT(i != 0);
	if (i->left == 0) {
		*out_smallest = i;
		return i->right;
	} else {
		i->left = _btree_extract_smallest(i->left, out_smallest);
		btree_recalc_height(i);
		return btree_equilibrate(i);
	}
}
btree_item_t *_btree_remove_aux(btree_t *t, btree_item_t *i, const void* key) {
	if (i == 0) return 0;

	int c = t->cf(key, i->key);
	if (c < 0) {
		i->left = _btree_remove_aux(t, i->left, key);
		return btree_equilibrate(i);
	} else if (c > 0) {
		i->right = _btree_remove_aux(t, i->right, key);
		return btree_equilibrate(i);
	} else {
		// remove item i

		btree_item_t *new_i;
		if (i->right == 0 || i->left == 0) {
			new_i = (i->right == 0 ? i->left : i->right);
		} else {
			btree_item_t *new_i_right = _btree_extract_smallest(i->right, &new_i);

			new_i->left = i->left;
			new_i->right = new_i_right;

			btree_recalc_height(new_i);
			new_i = btree_equilibrate(new_i);
		}

		if (t->releasef) t->releasef(i->key, i->val);
		free(i);
		t->nitems--;

		return _btree_remove_aux(t, new_i, key);	// loop because several elements may correspond
	}
}
void btree_remove(btree_t *t, const void* key) {
	t->root = _btree_remove_aux(t, t->root, key);
}

btree_item_t *_btree_extract_smallest_v(btree_item_t *i, btree_item_t **out_smallest) {
	ASSERT(i != 0);
	if (i->left == 0) {
		*out_smallest = i;
		return i->right;
	} else {
		i->left = _btree_extract_smallest_v(i->left, out_smallest);
		btree_recalc_height(i);
		return btree_equilibrate(i);
	}
}
btree_item_t *_btree_remove_aux_v(btree_t *t, btree_item_t *i, const void* key, const void* val) {
	if (i == 0) return 0;

	int c = t->cf(key, i->key);
	if (c < 0) {
		i->left = _btree_remove_aux_v(t, i->left, key, val);
		return btree_equilibrate(i);
	} else if (c > 0) {
		i->right = _btree_remove_aux_v(t, i->right, key, val);
		return btree_equilibrate(i);
	} else if (i->val == val) {
		// remove item i

		btree_item_t *new_i;
		if (i->right == 0 || i->left == 0) {
			new_i = (i->right == 0 ? i->left : i->right);
		} else {
			btree_item_t *new_i_right = _btree_extract_smallest_v(i->right, &new_i);

			new_i->left = i->left;
			new_i->right = new_i_right;

			btree_recalc_height(new_i);
			new_i = btree_equilibrate(new_i);
		}

		if (t->releasef) t->releasef(i->key, i->val);
		free(i);
		t->nitems--;

		return _btree_remove_aux_v(t, new_i, key, val);	// loop because several elements may correspond
	} else {
		i->left = _btree_remove_aux_v(t, i->left, key, val);
		i->right = _btree_remove_aux_v(t, i->right, key, val);
		btree_recalc_height(i);
		return btree_equilibrate(i);
	}
}
void btree_remove_v(btree_t *t, const void* key, const void* val) {
	t->root = _btree_remove_aux_v(t, t->root, key, val);
}

// ======================== //
// LOOKING UP AND ITERATING //
// ======================== //

btree_item_t *_btree_find_aux(btree_t *t, btree_item_t *i, const void* key) {
	if (i == 0) return 0;

	int c = t->cf(key, i->key);
	if (c == 0) {
		return i;
	} else if (c < 0) {
		return _btree_find_aux(t, i->left, key);
	} else {
		return _btree_find_aux(t, i->right, key);
	}
}
void* btree_find(btree_t *t, const void* key) {

	btree_item_t *i = _btree_find_aux(t, t->root, key);

	if (i == 0) return 0;
	return i->val;
}

btree_item_t *_btree_lower_aux(btree_t *t, btree_item_t *i, const void* key) {
	if (i == 0) return 0;

	int c = t->cf(key, i->key);
	if (c == 0) {
		return i;
	} else if (c < 0) {
		return _btree_lower_aux(t, i->left, key);
	} else {
		btree_item_t *r = _btree_lower_aux(t, i->right, key);
		if (r == 0) r = i;
		return r;
	}
}
void* btree_lower(btree_t *t, const void* key, void** actual_key) {
	btree_item_t *i = _btree_lower_aux(t, t->root, key);

	if (i == 0) return 0;
	if (actual_key != 0) *actual_key = i->key;
	return i->val;
}

btree_item_t *_btree_upper_aux(btree_t *t, btree_item_t *i, const void* key) {
	if (i == 0) return 0;

	int c = t->cf(key, i->key);
	if (c == 0) {
		return i;
	} else if (c < 0) {
		btree_item_t *r = _btree_upper_aux(t, i->left, key);
		if (r == 0) r = i;
		return r;
	} else {
		return _btree_upper_aux(t, i->right, key);
	}
}
void* btree_upper(btree_t *t, const void* key, void** actual_key) {
	btree_item_t *i = _btree_upper_aux(t, t->root, key);

	if (i == 0) return 0;
	if (actual_key != 0) *actual_key = i->key;
	return i->val;
}


void _btree_iter_aux(btree_item_t *i, kv_iter_fun_t f) {
	if (i == 0) return;

	_btree_iter_aux(i->left, f);
	f(i->key, i->val);
	_btree_iter_aux(i->right, f);
}
void btree_iter(btree_t *t, kv_iter_fun_t f) {
	_btree_iter_aux(t->root, f);
}

void _btree_iter_on_aux(btree_t *t, btree_item_t *i, const void* key, kv_iter_fun_t f) {
	if (i == 0) return;

	int c = t->cf(key, i->key);
	if (c == 0) {
		_btree_iter_on_aux(t, i->left, key, f);
		f(i->key, i->val);
		_btree_iter_on_aux(t, i->right, key, f);
	} else if (c < 0) {
		_btree_iter_on_aux(t, i->left, key, f);
	} else {
		_btree_iter_on_aux(t, i->right, key, f);
	}
}
void btree_iter_on(btree_t *t, const void* key, kv_iter_fun_t f) {
	_btree_iter_on_aux(t, t->root, key, f);
}

/* vim: set ts=4 sw=4 tw=0 noet :*/