Commit 460d2186 authored by Yoni Fogel's avatar Yoni Fogel

Addresses #825

We rebalance at most one subtree per delete.

Insert and delete modified slightly to use tail recursion when simple.

git-svn-id: file:///svn/tokudb@3943 c7de825b-a66e-492c-adef-691d508d4ae1
parent c7d6778a
...@@ -231,10 +231,6 @@ static inline BOOL will_need_rebalance(OMT omt, node_idx n_idx, int leftmod, int ...@@ -231,10 +231,6 @@ static inline BOOL will_need_rebalance(OMT omt, node_idx n_idx, int leftmod, int
(1+weight_right < (1+1+weight_left)/2)); (1+weight_right < (1+1+weight_left)/2));
} }
static inline void maybe_rebalance(OMT omt, node_idx *n_idxp) {
if (will_need_rebalance(omt, *n_idxp, 0, 0)) rebalance(omt, n_idxp);
}
static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_int32_t index, node_idx **rebalance_idx) { static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_int32_t index, node_idx **rebalance_idx) {
if (*n_idxp==NODE_NULL) { if (*n_idxp==NODE_NULL) {
assert(index==0); assert(index==0);
...@@ -248,6 +244,7 @@ static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_ ...@@ -248,6 +244,7 @@ static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_
} else { } else {
node_idx idx = *n_idxp; node_idx idx = *n_idxp;
OMT_NODE n = omt->nodes+idx; OMT_NODE n = omt->nodes+idx;
n->weight++;
if (index <= nweight(omt, n->left)) { if (index <= nweight(omt, n->left)) {
if (*rebalance_idx==NULL && will_need_rebalance(omt, idx, 1, 0)) { if (*rebalance_idx==NULL && will_need_rebalance(omt, idx, 1, 0)) {
*rebalance_idx = n_idxp; *rebalance_idx = n_idxp;
...@@ -260,7 +257,6 @@ static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_ ...@@ -260,7 +257,6 @@ static inline void insert_internal(OMT omt, node_idx *n_idxp, OMTVALUE value, u_
u_int32_t sub_index = index-nweight(omt, n->left)-1; u_int32_t sub_index = index-nweight(omt, n->left)-1;
insert_internal(omt, &n->right, value, sub_index, rebalance_idx); insert_internal(omt, &n->right, value, sub_index, rebalance_idx);
} }
n->weight++;
} }
} }
...@@ -292,12 +288,15 @@ int toku_omt_set_at (OMT omt, OMTVALUE value, u_int32_t index) { ...@@ -292,12 +288,15 @@ int toku_omt_set_at (OMT omt, OMTVALUE value, u_int32_t index) {
return 0; return 0;
} }
static inline void delete_internal(OMT omt, node_idx *n_idxp, u_int32_t index, OMTVALUE *vp) { static inline void delete_internal(OMT omt, node_idx *n_idxp, u_int32_t index, OMTVALUE *vp, node_idx **rebalance_idx) {
assert(*n_idxp!=NODE_NULL); assert(*n_idxp!=NODE_NULL);
OMT_NODE n = omt->nodes+*n_idxp; OMT_NODE n = omt->nodes+*n_idxp;
if (index < nweight(omt, n->left)) { if (index < nweight(omt, n->left)) {
delete_internal(omt, &n->left, index, vp);
n->weight--; n->weight--;
if (*rebalance_idx==NULL && will_need_rebalance(omt, *n_idxp, -1, 0)) {
*rebalance_idx = n_idxp;
}
delete_internal(omt, &n->left, index, vp, rebalance_idx);
} else if (index == nweight(omt, n->left)) { } else if (index == nweight(omt, n->left)) {
if (n->left==NODE_NULL) { if (n->left==NODE_NULL) {
u_int32_t idx = *n_idxp; u_int32_t idx = *n_idxp;
...@@ -312,15 +311,20 @@ static inline void delete_internal(OMT omt, node_idx *n_idxp, u_int32_t index, O ...@@ -312,15 +311,20 @@ static inline void delete_internal(OMT omt, node_idx *n_idxp, u_int32_t index, O
} 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.
delete_internal(omt, &n->right, 0, &zv); if (*rebalance_idx==NULL && will_need_rebalance(omt, *n_idxp, 0, -1)) {
*rebalance_idx = n_idxp;
}
delete_internal(omt, &n->right, 0, &zv, rebalance_idx);
n->value = zv; n->value = zv;
n->weight--; n->weight--;
} }
} else { } else {
delete_internal(omt, &n->right, index-nweight(omt, n->left)-1, vp);
n->weight--; n->weight--;
if (*rebalance_idx==NULL && will_need_rebalance(omt, *n_idxp, 0, -1)) {
*rebalance_idx = n_idxp;
}
delete_internal(omt, &n->right, index-nweight(omt, n->left)-1, vp, rebalance_idx);
} }
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) {
...@@ -328,7 +332,9 @@ int toku_omt_delete_at(OMT omt, u_int32_t index) { ...@@ -328,7 +332,9 @@ int toku_omt_delete_at(OMT omt, u_int32_t index) {
int r; int r;
if (index>=nweight(omt, omt->root)) return ERANGE; if (index>=nweight(omt, omt->root)) return ERANGE;
if ((r=maybe_resize_and_rebuild(omt, -1+nweight(omt, omt->root), MAYBE_REBUILD))) 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); node_idx* rebalance_idx = NULL;
delete_internal(omt, &omt->root, index, &v, &rebalance_idx);
if (rebalance_idx) rebalance(omt, rebalance_idx);
return 0; return 0;
} }
......
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