Commit 54c29298 authored by Yoni Fogel's avatar Yoni Fogel

Closes #825

OMT now uses block allocs instead of a single malloc for each node.

git-svn-id: file:///svn/tokudb@3940 c7de825b-a66e-492c-adef-691d508d4ae1
parent 1dde5d83
...@@ -2,178 +2,298 @@ ...@@ -2,178 +2,298 @@
#include <errno.h> #include <errno.h>
#include <sys/types.h> #include <sys/types.h>
#include <stdint.h>
typedef void *OMTVALUE; typedef void *OMTVALUE;
#include "omt.h" #include "omt.h"
#include "../newbrt/memory.h" #include "../newbrt/memory.h"
#include "../newbrt/toku_assert.h" #include "../newbrt/toku_assert.h"
#include "../include/db.h" #include "../include/db.h"
#include "../newbrt/brttypes.h"
typedef u_int32_t node_idx;
static const node_idx NODE_NULL = UINT32_MAX;
typedef struct omt_node *OMT_NODE; typedef struct omt_node *OMT_NODE;
struct omt_node { struct omt_node {
u_int32_t weight; // how many values below us (including this node) u_int32_t weight; /* Size of subtree rooted at this node (including this one). */
OMT_NODE left, right; node_idx left; /* Index of left subtree. */
OMTVALUE value; node_idx right; /* Index of right subtree. */
OMTVALUE value; /* The value stored in the node. */
}; };
struct omt { struct omt {
OMT_NODE root; node_idx root;
u_int32_t tmparray_size; u_int32_t node_capacity;
OMT_NODE *tmparray; OMT_NODE nodes;
node_idx free_idx;
u_int32_t tmparray_size;
node_idx* tmparray;
}; };
int toku_omt_create (OMT *omtp) { static int omt_create_internal(OMT *omtp, u_int32_t num_starting_nodes) {
if (num_starting_nodes < 2) num_starting_nodes = 2;
OMT MALLOC(result); OMT MALLOC(result);
if (result==NULL) return errno; if (result==NULL) return errno;
result->root=NULL; result->root=NODE_NULL;
result->tmparray_size = 4; result->node_capacity = num_starting_nodes*2;
MALLOC_N(result->node_capacity, result->nodes);
if (result->nodes==NULL) {
toku_free(result);
return errno;
}
result->tmparray_size = num_starting_nodes*2;
MALLOC_N(result->tmparray_size, result->tmparray); MALLOC_N(result->tmparray_size, result->tmparray);
if (result->tmparray==0) { if (result->tmparray==NULL) {
toku_free(result->nodes);
toku_free(result); toku_free(result);
return errno; return errno;
} }
result->free_idx = 0;
*omtp = result; *omtp = result;
return 0; return 0;
} }
static inline u_int32_t nweight (OMT_NODE n) { int toku_omt_create (OMT *omtp) {
if (n==NULL) return 0; return omt_create_internal(omtp, 2);
else return n->weight;
} }
static inline void fill_array_from_omt_nodes_tree (OMT_NODE *array, OMT_NODE tree) { void toku_omt_destroy(OMT *omtp) {
if (tree==NULL) return; OMT omt=*omtp;
fill_array_from_omt_nodes_tree(array, tree->left); toku_free(omt->nodes);
array[nweight(tree->left)] = tree; toku_free(omt->tmparray);
fill_array_from_omt_nodes_tree(array+nweight(tree->left)+1, tree->right); toku_free(omt);
*omtp=NULL;
} }
static inline void rebuild_from_sorted_array_of_omt_nodes(OMT_NODE *np, OMT_NODE *nodes, u_int32_t numvalues) { static inline u_int32_t nweight(OMT omt, node_idx idx) {
if (idx==NODE_NULL) return 0;
else return (omt->nodes+idx)->weight;
}
u_int32_t toku_omt_size(OMT V) {
return nweight(V, V->root);
}
static inline node_idx omt_node_malloc(OMT omt) {
assert(omt->free_idx < omt->node_capacity);
return omt->free_idx++;
}
static inline void omt_node_free(OMT omt, node_idx idx) {
assert(idx < omt->node_capacity);
}
static inline void fill_array_with_subtree_values(OMT omt, OMTVALUE *array, node_idx tree_idx) {
if (tree_idx==NODE_NULL) return;
OMT_NODE tree = omt->nodes+tree_idx;
fill_array_with_subtree_values(omt, array, tree->left);
array[nweight(omt, tree->left)] = tree->value;
fill_array_with_subtree_values(omt, array+nweight(omt, tree->left)+1, tree->right);
}
// Example: numvalues=4, halfway=2, left side is values of size 2
// right side is values+3 of size 1
// numvalues=3, halfway=1, left side is values of size 1
// right side is values+2 of size 1
// numvalues=2, halfway=1, left side is values of size 1
// right side is values+2 of size 0
// numvalues=1, halfway=0, left side is values of size 0
// right side is values of size 0.
static inline void create_from_sorted_array_internal(OMT omt, node_idx *n_idxp,
OMTVALUE *values, u_int32_t numvalues) {
if (numvalues==0) { if (numvalues==0) {
*np=NULL; *n_idxp = NODE_NULL;
} else { } else {
u_int32_t halfway = numvalues/2; u_int32_t halfway = numvalues/2;
OMT_NODE newnode = nodes[halfway]; node_idx newidx = omt_node_malloc(omt);
newnode->weight = numvalues; OMT_NODE newnode = omt->nodes+newidx;
newnode->weight = numvalues;
newnode->value = values[halfway];
create_from_sorted_array_internal(omt, &newnode->left, values, halfway);
create_from_sorted_array_internal(omt, &newnode->right, values+halfway+1, numvalues-(halfway+1));
*n_idxp = newidx;
}
}
int toku_omt_create_from_sorted_array(OMT *omtp, OMTVALUE *values, u_int32_t numvalues) {
OMT omt = NULL;
int r;
if ((r = omt_create_internal(&omt, numvalues))) return r;
create_from_sorted_array_internal(omt, &omt->root, values, numvalues);
*omtp=omt;
return 0;
}
enum build_choice { MAYBE_REBUILD, JUST_RESIZE };
static inline int maybe_resize_and_rebuild(OMT omt, u_int32_t n, enum build_choice choice) {
node_idx *new_tmparray = NULL;
OMT_NODE new_nodes = NULL;
OMTVALUE *tmp_values = NULL;
int r = ENOSYS;
u_int32_t new_size = 2*n;
if (omt->tmparray_size<n ||
(omt->tmparray_size/4 >= n && n>=2)) {
/* Malloc and free instead of realloc (saves the memcpy). */
MALLOC_N(new_size, new_tmparray);
if (new_tmparray==NULL) { r = errno; goto cleanup; }
}
/* Rebuild/realloc the nodes array iff any of the following:
* The array is smaller than the number of elements we want.
* We are increasing the number of elements and there is no free space.
* The array is too large. */
u_int32_t num_nodes = nweight(omt, omt->root);
if ((omt->node_capacity/4 >= n && n>=2) ||
(omt->free_idx>=omt->node_capacity && num_nodes<n) ||
(omt->node_capacity<n)) {
if (choice==MAYBE_REBUILD) {
MALLOC_N(num_nodes, tmp_values);
if (tmp_values==NULL) { r = errno; goto cleanup;}
}
MALLOC_N(new_size, new_nodes);
if (new_nodes==NULL) { r = errno; goto cleanup; }
}
/* Nothing can fail now. Atomically update both sizes. */
if (new_tmparray) {
toku_free(omt->tmparray);
omt->tmparray = new_tmparray;
omt->tmparray_size = new_size;
}
if (new_nodes) {
/* Rebuild the tree in the new array, leftshifted, in preorder */
if (choice==MAYBE_REBUILD) {
fill_array_with_subtree_values(omt, tmp_values, omt->root);
}
toku_free(omt->nodes);
omt->nodes = new_nodes;
omt->node_capacity = new_size;
omt->free_idx = 0; /* Allocating from mempool starts over. */
if (choice==MAYBE_REBUILD) {
create_from_sorted_array_internal(omt, &omt->root, tmp_values, num_nodes);
}
}
r = 0;
cleanup:
if (r!=0) {
if (new_tmparray) toku_free(new_tmparray);
if (new_nodes) toku_free(new_nodes);
}
if (tmp_values) toku_free(tmp_values);
return r;
}
static inline void fill_array_with_subtree_idxs(OMT omt, node_idx *array, node_idx tree_idx) {
if (tree_idx==NODE_NULL) return;
OMT_NODE tree = omt->nodes+tree_idx;
fill_array_with_subtree_idxs(omt, array, tree->left);
array[nweight(omt, tree->left)] = tree_idx;
fill_array_with_subtree_idxs(omt, array+nweight(omt, tree->left)+1, tree->right);
}
/* Reuses existing OMT_NODE structures (used for rebalancing). */
static inline void rebuild_subtree_from_idxs(OMT omt, node_idx *n_idxp, node_idx *idxs,
u_int32_t numvalues) {
if (numvalues==0) {
*n_idxp=NODE_NULL;
} else {
u_int32_t halfway = numvalues/2;
node_idx newidx = idxs[halfway];
OMT_NODE newnode = omt->nodes+newidx;
newnode->weight = numvalues;
// value is already in there. // value is already in there.
rebuild_from_sorted_array_of_omt_nodes(&newnode->left, nodes, halfway); rebuild_subtree_from_idxs(omt, &newnode->left, idxs, halfway);
rebuild_from_sorted_array_of_omt_nodes(&newnode->right, nodes+halfway+1, numvalues-(halfway+1)); rebuild_subtree_from_idxs(omt, &newnode->right, idxs+halfway+1, numvalues-(halfway+1));
*np = newnode; *n_idxp = newidx;
} }
} }
static inline void maybe_rebalance (OMT omt, OMT_NODE *np) { static inline void maybe_rebalance(OMT omt, node_idx *n_idxp) {
OMT_NODE n = *np; node_idx idx = *n_idxp;
if (n==0) return; if (idx==NODE_NULL) return;
OMT_NODE n = omt->nodes+idx;
// one of the 1's is for the root. // one of the 1's is for the root.
// the other is to take ceil(n/2) // the other is to take ceil(n/2)
if (((1+nweight(n->left)) < (1+1+nweight(n->right))/2) if (((1+nweight(omt, n->left)) < (1+1+nweight(omt, n->right))/2)
|| ||
((1+nweight(n->right)) < (1+1+nweight(n->left))/2)) { ((1+nweight(omt, n->right)) < (1+1+nweight(omt, n->left))/2)) {
// Must rebalance the tree. // Must rebalance the subtree.
fill_array_from_omt_nodes_tree(omt->tmparray, *np); fill_array_with_subtree_idxs(omt, omt->tmparray, idx);
rebuild_from_sorted_array_of_omt_nodes(np, omt->tmparray, nweight(*np)); rebuild_subtree_from_idxs(omt, n_idxp, omt->tmparray, n->weight);
} }
} }
static inline int insert_internal (OMT omt, OMT_NODE *np, OMTVALUE value, u_int32_t index) { static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_int32_t index) {
if (*np==0) { if (*n_idxp==NODE_NULL) {
assert(index==0); assert(index==0);
OMT_NODE MALLOC(newnode); node_idx newidx = omt_node_malloc(omt);
if (newnode==0) return errno; OMT_NODE newnode = omt->nodes+newidx;
newnode->weight = 1; newnode->weight = 1;
newnode->left = NULL; newnode->left = NODE_NULL;
newnode->right = NULL; newnode->right = NODE_NULL;
newnode->value = value; newnode->value = value;
*np = newnode; *n_idxp = newidx;
return 0;
} else { } else {
OMT_NODE n=*np; node_idx idx = *n_idxp;
int r; OMT_NODE n = omt->nodes+idx;
if (index <= nweight(n->left)) { if (index <= nweight(omt, n->left)) {
if ((r = insert_internal(omt, &n->left, value, index))) return r; insert_internal(omt, &n->left, value, index);
} else { } else {
if ((r = insert_internal(omt, &n->right, value, index-nweight(n->left)-1))) return r; u_int32_t sub_index = index-nweight(omt, n->left)-1;
insert_internal(omt, &n->right, value, sub_index);
} }
n->weight++; n->weight++;
maybe_rebalance(omt, np); maybe_rebalance(omt, n_idxp);
return 0;
}
}
static inline int make_sure_array_is_sized_ok (OMT omt, u_int32_t n) {
u_int32_t new_size;
if (omt->tmparray_size < n) {
new_size = 2*n;
do_realloc: ;
OMT_NODE *newarray = toku_realloc(omt->tmparray, new_size * sizeof(*newarray));
if (newarray==0) return errno;
omt->tmparray = newarray;
omt->tmparray_size = new_size;
} else if (omt->tmparray_size/4 > n && n>=2) {
new_size = 2*n;
goto do_realloc;
} }
return 0;
} }
int toku_omt_insert_at (OMT omt, OMTVALUE value, u_int32_t index) { int toku_omt_insert_at(OMT omt, OMTVALUE value, u_int32_t index) {
int r; int r;
if (index>nweight(omt->root)) return ERANGE; if (index>nweight(omt, omt->root)) return ERANGE;
if ((r=make_sure_array_is_sized_ok(omt, 1+nweight(omt->root)))) return r; if ((r=maybe_resize_and_rebuild(omt, 1+nweight(omt, omt->root), MAYBE_REBUILD))) return r;
return insert_internal(omt, &omt->root, value, index); insert_internal(omt, &omt->root, value, index);
return 0;
} }
static inline void set_at_internal (OMT_NODE n, OMTVALUE v, u_int32_t index) { static inline void set_at_internal(OMT omt, node_idx n_idx, OMTVALUE v, u_int32_t index) {
assert(n); assert(n_idx!=NODE_NULL);
if (index<nweight(n->left)) OMT_NODE n = omt->nodes+n_idx;
set_at_internal(n->left, v, index); if (index<nweight(omt, n->left))
else if (index==nweight(n->left)) { set_at_internal(omt, n->left, v, index);
else if (index==nweight(omt, n->left)) {
n->value = v; n->value = v;
} else { } else {
set_at_internal(n->right, v, index-nweight(n->left)-1); set_at_internal(omt, n->right, v, index-nweight(omt, n->left)-1);
} }
} }
int toku_omt_set_at (OMT omt, OMTVALUE value, u_int32_t index) { int toku_omt_set_at (OMT omt, OMTVALUE value, u_int32_t index) {
if (index>=nweight(omt->root)) return ERANGE; if (index>=nweight(omt, omt->root)) return ERANGE;
set_at_internal(omt->root, value, index); set_at_internal(omt, omt->root, value, index);
return 0;
}
int toku_omt_insert(OMT omt, OMTVALUE value, int(*h)(OMTVALUE, void*v), void *v, u_int32_t *index) {
int r;
u_int32_t idx;
r = toku_omt_find_zero(omt, h, v, NULL, &idx);
if (r==0) {
if (index) *index = idx;
return DB_KEYEXIST;
}
if (r!=DB_NOTFOUND) return r;
if ((r = toku_omt_insert_at(omt, value, idx))) return r;
if (index) *index = idx;
return 0; return 0;
} }
static inline void delete_internal (OMT omt, OMT_NODE *np, u_int32_t index, OMTVALUE *vp) { static inline void delete_internal(OMT omt, node_idx *n_idxp, u_int32_t index, OMTVALUE *vp) {
OMT_NODE n=*np; assert(*n_idxp!=NODE_NULL);
if (index < nweight(n->left)) { OMT_NODE n = omt->nodes+*n_idxp;
if (index < nweight(omt, n->left)) {
delete_internal(omt, &n->left, index, vp); delete_internal(omt, &n->left, index, vp);
n->weight--; n->weight--;
} else if (index == nweight(n->left)) { } else if (index == nweight(omt, n->left)) {
if (n->left==NULL) { if (n->left==NODE_NULL) {
*np = n->right; u_int32_t idx = *n_idxp;
*vp = n->value; *n_idxp = n->right;
toku_free(n); *vp = n->value;
} else if (n->right==NULL) { omt_node_free(omt, idx);
*np = n->left; } else if (n->right==NODE_NULL) {
*vp = n->value; u_int32_t idx = *n_idxp;
toku_free(n); *n_idxp = n->left;
*vp = n->value;
omt_node_free(omt, idx);
} else { } else {
OMTVALUE zv; OMTVALUE zv;
// delete the successor of index, get the value, and store it here. // delete the successor of index, get the value, and store it here.
...@@ -182,53 +302,86 @@ static inline void delete_internal (OMT omt, OMT_NODE *np, u_int32_t index, OMTV ...@@ -182,53 +302,86 @@ static inline void delete_internal (OMT omt, OMT_NODE *np, u_int32_t index, OMTV
n->weight--; n->weight--;
} }
} else { } else {
delete_internal(omt, &n->right, index-nweight(n->left)-1, vp); delete_internal(omt, &n->right, index-nweight(omt, n->left)-1, vp);
n->weight--; n->weight--;
} }
maybe_rebalance(omt, np); maybe_rebalance(omt, n_idxp);
} }
int toku_omt_delete_at(OMT omt, u_int32_t index) { int toku_omt_delete_at(OMT omt, u_int32_t index) {
OMTVALUE v; OMTVALUE v;
int r; int r;
if (index>=nweight(omt->root)) return ERANGE; if (index>=nweight(omt, omt->root)) return ERANGE;
if ((r=make_sure_array_is_sized_ok(omt, -1+nweight(omt->root)))) return r; if ((r=maybe_resize_and_rebuild(omt, -1+nweight(omt, omt->root), MAYBE_REBUILD))) return r;
delete_internal(omt, &omt->root, index, &v); delete_internal(omt, &omt->root, index, &v);
return 0; return 0;
} }
static inline int fetch_internal (OMT_NODE n, u_int32_t i, OMTVALUE *v) { static inline void fetch_internal(OMT V, node_idx idx, u_int32_t i, OMTVALUE *v) {
if (i < nweight(n->left)) { OMT_NODE n = V->nodes+idx;
return fetch_internal(n->left, i, v); if (i < nweight(V, n->left)) {
} else if (i == nweight(n->left)) { fetch_internal(V, n->left, i, v);
} else if (i == nweight(V, n->left)) {
*v = n->value; *v = n->value;
return 0;
} else { } else {
return fetch_internal(n->right, i-nweight(n->left)-1, v); fetch_internal(V, n->right, i-nweight(V, n->left)-1, v);
} }
} }
int toku_omt_fetch (OMT V, u_int32_t i, OMTVALUE *v) { int toku_omt_fetch(OMT V, u_int32_t i, OMTVALUE *v) {
if (i>=nweight(V->root)) return ERANGE; if (i>=nweight(V, V->root)) return ERANGE;
return fetch_internal(V->root, i, v); fetch_internal(V, V->root, i, v);
return 0;
}
static inline int iterate_internal(OMT omt, node_idx n_idx, u_int32_t idx,
int (*f)(OMTVALUE, u_int32_t, void*), void*v) {
int r;
if (n_idx==NODE_NULL) return 0;
OMT_NODE n = omt->nodes+n_idx;
if ((r=iterate_internal(omt, n->left, idx, f, v))) return r;
if ((r=f(n->value, idx+nweight(omt, n->left), v))) return r;
return iterate_internal(omt, n->right, idx+nweight(omt, n->left)+1, f, v);
}
int toku_omt_iterate(OMT omt, int (*f)(OMTVALUE, u_int32_t, void*), void*v) {
return iterate_internal(omt, omt->root, 0, f, v);
} }
static inline int find_internal_zero (OMT_NODE n, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) { int toku_omt_insert(OMT omt, OMTVALUE value, int(*h)(OMTVALUE, void*v), void *v, u_int32_t *index) {
if (n==NULL) { int r;
u_int32_t idx;
r = toku_omt_find_zero(omt, h, v, NULL, &idx);
if (r==0) {
if (index) *index = idx;
return DB_KEYEXIST;
}
if (r!=DB_NOTFOUND) return r;
if ((r = toku_omt_insert_at(omt, value, idx))) return r;
if (index) *index = idx;
return 0;
}
static inline int find_internal_zero(OMT omt, node_idx n_idx, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) {
if (n_idx==NODE_NULL) {
if (index!=NULL) (*index)=0; if (index!=NULL) (*index)=0;
return DB_NOTFOUND; return DB_NOTFOUND;
} }
OMT_NODE n = omt->nodes+n_idx;
int hv = h(n->value, extra); int hv = h(n->value, extra);
if (hv<0) { if (hv<0) {
int r = find_internal_zero(n->right, h, extra, value, index); int r = find_internal_zero(omt, n->right, h, extra, value, index);
if (index!=NULL) (*index) += nweight(n->left)+1; if (index!=NULL) (*index) += nweight(omt, n->left)+1;
return r; return r;
} else if (hv>0) { } else if (hv>0) {
return find_internal_zero(n->left, h, extra, value, index); return find_internal_zero(omt, n->left, h, extra, value, index);
} else { } else {
int r = find_internal_zero(n->left, h, extra, value, index); int r = find_internal_zero(omt, n->left, h, extra, value, index);
if (r==DB_NOTFOUND) { if (r==DB_NOTFOUND) {
if (index!=NULL) *index = nweight(n->left); if (index!=NULL) *index = nweight(omt, n->left);
if (value!=NULL) *value = n->value; if (value!=NULL) *value = n->value;
r = 0; r = 0;
} }
...@@ -236,43 +389,45 @@ static inline int find_internal_zero (OMT_NODE n, int (*h)(OMTVALUE, void*extra) ...@@ -236,43 +389,45 @@ static inline int find_internal_zero (OMT_NODE n, int (*h)(OMTVALUE, void*extra)
} }
} }
int toku_omt_find_zero (OMT t, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) { int toku_omt_find_zero(OMT V, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) {
return find_internal_zero(t->root, h, extra, value, index); return find_internal_zero(V, V->root, h, extra, value, index);
} }
// If direction <0 then find the largest i such that h(V_i,extra)<0. // If direction <0 then find the largest i such that h(V_i,extra)<0.
static inline int find_internal_minus (OMT_NODE n, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) { static inline int find_internal_minus(OMT omt, node_idx n_idx, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) {
if (n==NULL) return DB_NOTFOUND; if (n_idx==NODE_NULL) return DB_NOTFOUND;
OMT_NODE n = omt->nodes+n_idx;
int hv = h(n->value, extra); int hv = h(n->value, extra);
if (hv<0) { if (hv<0) {
int r = find_internal_minus(n->right, h, extra, value, index); int r = find_internal_minus(omt, n->right, h, extra, value, index);
if (r==0 && index!=NULL) (*index) += nweight(n->left)+1; if (r==0 && index!=NULL) (*index) += nweight(omt, n->left)+1;
else if (r==DB_NOTFOUND) { else if (r==DB_NOTFOUND) {
if (index!=NULL) *index = nweight(n->left); if (index!=NULL) *index = nweight(omt, n->left);
if (value!=NULL) *value = n->value; if (value!=NULL) *value = n->value;
r = 0; r = 0;
} }
return r; return r;
} else { } else {
return find_internal_minus(n->left, h, extra, value, index); return find_internal_minus(omt, n->left, h, extra, value, index);
} }
} }
// If direction >0 then find the smallest i such that h(V_i,extra)>0. // If direction >0 then find the smallest i such that h(V_i,extra)>0.
static inline int find_internal_plus (OMT_NODE n, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) { static inline int find_internal_plus(OMT omt, node_idx n_idx, int (*h)(OMTVALUE, void*extra), void*extra, OMTVALUE *value, u_int32_t *index) {
if (n==NULL) return DB_NOTFOUND; if (n_idx==NODE_NULL) return DB_NOTFOUND;
OMT_NODE n = omt->nodes+n_idx;
int hv = h(n->value, extra); int hv = h(n->value, extra);
if (hv>0) { if (hv>0) {
int r = find_internal_plus(n->left, h, extra, value, index); int r = find_internal_plus(omt, n->left, h, extra, value, index);
if (r==DB_NOTFOUND) { if (r==DB_NOTFOUND) {
if (index!=NULL) *index = nweight(n->left); if (index!=NULL) *index = nweight(omt, n->left);
if (value!=NULL) *value = n->value; if (value!=NULL) *value = n->value;
r = 0; r = 0;
} }
return r; return r;
} else { } else {
int r = find_internal_plus(n->right, h, extra, value, index); int r = find_internal_plus(omt, n->right, h, extra, value, index);
if (r==0 && index!=NULL) (*index) += nweight(n->left)+1; if (r==0 && index!=NULL) (*index) += nweight(omt, n->left)+1;
return r; return r;
} }
} }
...@@ -281,137 +436,57 @@ int toku_omt_find(OMT V, int (*h)(OMTVALUE, void*extra), void*extra, int directi ...@@ -281,137 +436,57 @@ int toku_omt_find(OMT V, int (*h)(OMTVALUE, void*extra), void*extra, int directi
if (direction==0) { if (direction==0) {
abort(); abort();
} else if (direction<0) { } else if (direction<0) {
return find_internal_minus(V->root, h, extra, value, index); return find_internal_minus(V, V->root, h, extra, value, index);
} else {
return find_internal_plus(V->root, h, extra, value, index);
}
}
static inline void free_omt_nodes (OMT_NODE n) {
if (n==0) return;
free_omt_nodes(n->left);
free_omt_nodes(n->right);
toku_free(n);
}
// Example: numvalues=4, halfway=2, left side is values of size 2
// right side is values+3 of size 1
// numvalues=3, halfway=1, left side is values of size 1
// right side is values+2 of size 1
// numvalues=2, halfway=1, left side is values of size 1
// right side is values+2 of size 0
// numvalues=1, halfway=0, left side is values of size 0
// right side is values of size 0.
static inline int create_from_sorted_array_internal(OMT_NODE *np, OMTVALUE *values, u_int32_t numvalues) {
if (numvalues==0) {
*np=NULL;
return 0;
} else { } else {
int r; return find_internal_plus( V, V->root, h, extra, value, index);
u_int32_t halfway = numvalues/2;
OMT_NODE MALLOC(newnode);
if (newnode==NULL) return errno;
newnode->weight = numvalues;
newnode->value = values[halfway];
if ((r = create_from_sorted_array_internal(&newnode->left, values, halfway))) {
toku_free(newnode);
return r;
}
if ((r = create_from_sorted_array_internal(&newnode->right, values+halfway+1, numvalues-(halfway+1)))) {
free_omt_nodes(newnode->left);
toku_free(newnode);
return r;
}
*np = newnode;
return 0;
} }
} }
int toku_omt_create_from_sorted_array(OMT *omtp, OMTVALUE *values, u_int32_t numvalues) {
OMT omt = NULL;
int r;
if ((r = toku_omt_create(&omt))) return r;
if ((r = create_from_sorted_array_internal(&omt->root, values, numvalues))) {
toku_omt_destroy(&omt);
return r;
}
if ((r=make_sure_array_is_sized_ok(omt, numvalues))) {
toku_omt_destroy(&omt);
return r;
}
*omtp=omt;
return 0;
}
void toku_omt_destroy(OMT *omtp) {
OMT omt=*omtp;
free_omt_nodes(omt->root);
toku_free(omt->tmparray);
toku_free(omt);
*omtp=NULL;
}
u_int32_t toku_omt_size(OMT V) {
return nweight(V->root);
}
static inline int iterate_internal(OMT_NODE n, u_int32_t idx, int (*f)(OMTVALUE, u_int32_t, void*), void*v) {
int r;
if (n==NULL) return 0;
if ((r=iterate_internal(n->left, idx, f, v))) return r;
if ((r=f(n->value, idx+nweight(n->left), v))) return r;
return iterate_internal(n->right, idx+nweight(n->left)+1, f, v);
}
int toku_omt_iterate(OMT omt, int (*f)(OMTVALUE, u_int32_t, void*), void*v) {
return iterate_internal(omt->root, 0, f, v);
}
int toku_omt_split_at(OMT omt, OMT *newomtp, u_int32_t index) { int toku_omt_split_at(OMT omt, OMT *newomtp, u_int32_t index) {
if (index>nweight(omt->root)) return ERANGE; int r = ENOSYS;
int r; OMT newomt = NULL;
u_int32_t newsize = toku_omt_size(omt)-index; OMTVALUE *tmp_values = NULL;
OMT newomt = NULL; if (index>nweight(omt, omt->root)) { r = ERANGE; goto cleanup; }
if ((r = toku_omt_create(&newomt))) return r; u_int32_t newsize = nweight(omt, omt->root)-index;
if ((r = make_sure_array_is_sized_ok(newomt, newsize))) { if ((r = omt_create_internal(&newomt, newsize))) goto cleanup;
fail: MALLOC_N(nweight(omt, omt->root), tmp_values);
toku_omt_destroy(&newomt); if (tmp_values==NULL) { r = errno; goto cleanup; }
return r; fill_array_with_subtree_values(omt, tmp_values, omt->root);
}
OMT_NODE *MALLOC_N(toku_omt_size(omt), nodes);
if (nodes==0) {
r = errno;
goto fail;
}
// Modify omt's array at the last possible moment, since after this nothing can fail. // Modify omt's array at the last possible moment, since after this nothing can fail.
if ((r = make_sure_array_is_sized_ok(omt, index))) { if ((r = maybe_resize_and_rebuild(omt, index, TRUE))) goto cleanup;
toku_free(nodes); create_from_sorted_array_internal(omt, &omt->root, tmp_values, index);
goto fail; create_from_sorted_array_internal(newomt, &newomt->root, tmp_values+index, newsize);
}
fill_array_from_omt_nodes_tree(nodes, omt->root);
rebuild_from_sorted_array_of_omt_nodes(&newomt->root, nodes+index, newsize);
rebuild_from_sorted_array_of_omt_nodes(&omt->root, nodes, index);
toku_free(nodes);
*newomtp = newomt; *newomtp = newomt;
return 0; r = 0;
cleanup:
if (r!=0) {
if (newomt) toku_omt_destroy(&newomt);
}
if (tmp_values) toku_free(tmp_values);
return r;
} }
int toku_omt_merge(OMT leftomt, OMT rightomt, OMT *newomtp) { int toku_omt_merge(OMT leftomt, OMT rightomt, OMT *newomtp) {
int r; int r = ENOSYS;
OMT newomt = NULL; OMT newomt = NULL;
OMTVALUE *tmp_values = NULL;
u_int32_t newsize = toku_omt_size(leftomt)+toku_omt_size(rightomt); u_int32_t newsize = toku_omt_size(leftomt)+toku_omt_size(rightomt);
if ((r = toku_omt_create(&newomt))) return r; if ((r = omt_create_internal(&newomt, newsize))) goto cleanup;
if ((r = make_sure_array_is_sized_ok(newomt, newsize))) { MALLOC_N(newsize, tmp_values);
toku_omt_destroy(&newomt); if (tmp_values==NULL) { r = errno; goto cleanup; }
return r;
} fill_array_with_subtree_values(leftomt, tmp_values, leftomt->root);
fill_array_from_omt_nodes_tree(newomt->tmparray, leftomt->root); fill_array_with_subtree_values(rightomt, tmp_values+toku_omt_size(leftomt), rightomt->root);
fill_array_from_omt_nodes_tree(newomt->tmparray+toku_omt_size(leftomt), rightomt->root); create_from_sorted_array_internal(newomt, &newomt->root, tmp_values, newsize);
rebuild_from_sorted_array_of_omt_nodes(&newomt->root, newomt->tmparray, newsize);
leftomt->root = rightomt->root = NULL;
toku_omt_destroy(&leftomt); toku_omt_destroy(&leftomt);
toku_omt_destroy(&rightomt); toku_omt_destroy(&rightomt);
*newomtp = newomt; *newomtp = newomt;
return 0; r = 0;
cleanup:
if (r!=0) {
if (newomt) toku_omt_destroy(&newomt);
}
if (tmp_values) toku_free(tmp_values);
return r;
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment