mirror of
https://github.com/MariaDB/server.git
synced 2025-10-24 00:27:49 +02:00

SHOW ENGINE INNODB MUTEX functionality is completely removed, as are the InnoDB latching order checks. We will enforce innodb_fatal_semaphore_wait_threshold only for dict_sys.mutex and lock_sys.mutex. dict_sys_t::mutex_lock(): A single entry point for dict_sys.mutex. lock_sys_t::mutex_lock(): A single entry point for lock_sys.mutex. FIXME: srv_sys should be removed altogether; it is duplicating tpool functionality. fil_crypt_threads_init(): To prevent SAFE_MUTEX warnings, we must not hold fil_system.mutex. fil_close_all_files(): To prevent SAFE_MUTEX warnings for fil_space_destroy_crypt_data(), we must not hold fil_system.mutex while invoking fil_space_free_low() on a detached tablespace.
1142 lines
28 KiB
C++
1142 lines
28 KiB
C++
/***************************************************************************//**
|
|
|
|
Copyright (c) 2007, 2015, Oracle and/or its affiliates. All Rights Reserved.
|
|
Copyright (c) 2020, MariaDB Corporation.
|
|
|
|
This program is free software; you can redistribute it and/or modify it under
|
|
the terms of the GNU General Public License as published by the Free Software
|
|
Foundation; version 2 of the License.
|
|
|
|
This program is distributed in the hope that it will be useful, but WITHOUT
|
|
ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along with
|
|
this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1335 USA
|
|
|
|
*****************************************************************************/
|
|
/********************************************************************//**
|
|
Red-Black tree implementation
|
|
|
|
(c) 2007 Oracle/Innobase Oy
|
|
|
|
Created 2007-03-20 Sunny Bains
|
|
***********************************************************************/
|
|
|
|
#include "ut0rbt.h"
|
|
#include "ut0new.h"
|
|
|
|
/**********************************************************************//**
|
|
Definition of a red-black tree
|
|
==============================
|
|
|
|
A red-black tree is a binary search tree which has the following
|
|
red-black properties:
|
|
|
|
1. Every node is either red or black.
|
|
2. Every leaf (NULL - in our case tree->nil) is black.
|
|
3. If a node is red, then both its children are black.
|
|
4. Every simple path from a node to a descendant leaf contains the
|
|
same number of black nodes.
|
|
|
|
from (3) above, the implication is that on any path from the root
|
|
to a leaf, red nodes must not be adjacent.
|
|
|
|
However, any number of black nodes may appear in a sequence.
|
|
*/
|
|
|
|
#if defined(IB_RBT_TESTING)
|
|
#warning "Testing enabled!"
|
|
#endif
|
|
|
|
#define ROOT(t) (t->root->left)
|
|
#define SIZEOF_NODE(t) ((sizeof(ib_rbt_node_t) + t->sizeof_value) - 1)
|
|
|
|
#if defined UNIV_DEBUG || defined IB_RBT_TESTING
|
|
/**********************************************************************//**
|
|
Verify that the keys are in order.
|
|
@return TRUE of OK. FALSE if not ordered */
|
|
static
|
|
ibool
|
|
rbt_check_ordering(
|
|
/*===============*/
|
|
const ib_rbt_t* tree) /*!< in: tree to verfify */
|
|
{
|
|
const ib_rbt_node_t* node;
|
|
const ib_rbt_node_t* prev = NULL;
|
|
|
|
/* Iterate over all the nodes, comparing each node with the prev */
|
|
for (node = rbt_first(tree); node; node = rbt_next(tree, prev)) {
|
|
|
|
if (prev) {
|
|
int result;
|
|
|
|
if (tree->cmp_arg) {
|
|
result = tree->compare_with_arg(
|
|
tree->cmp_arg, prev->value,
|
|
node->value);
|
|
} else {
|
|
result = tree->compare(
|
|
prev->value, node->value);
|
|
}
|
|
|
|
if (result >= 0) {
|
|
return(FALSE);
|
|
}
|
|
}
|
|
|
|
prev = node;
|
|
}
|
|
|
|
return(TRUE);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Check that every path from the root to the leaves has the same count.
|
|
Count is expressed in the number of black nodes.
|
|
@return 0 on failure else black height of the subtree */
|
|
static
|
|
ibool
|
|
rbt_count_black_nodes(
|
|
/*==================*/
|
|
const ib_rbt_t* tree, /*!< in: tree to verify */
|
|
const ib_rbt_node_t* node) /*!< in: start of sub-tree */
|
|
{
|
|
ulint result;
|
|
|
|
if (node != tree->nil) {
|
|
ulint left_height = rbt_count_black_nodes(tree, node->left);
|
|
|
|
ulint right_height = rbt_count_black_nodes(tree, node->right);
|
|
|
|
if (left_height == 0
|
|
|| right_height == 0
|
|
|| left_height != right_height) {
|
|
|
|
result = 0;
|
|
} else if (node->color == IB_RBT_RED) {
|
|
|
|
/* Case 3 */
|
|
if (node->left->color != IB_RBT_BLACK
|
|
|| node->right->color != IB_RBT_BLACK) {
|
|
|
|
result = 0;
|
|
} else {
|
|
result = left_height;
|
|
}
|
|
/* Check if it's anything other than RED or BLACK. */
|
|
} else if (node->color != IB_RBT_BLACK) {
|
|
|
|
result = 0;
|
|
} else {
|
|
|
|
result = right_height + 1;
|
|
}
|
|
} else {
|
|
result = 1;
|
|
}
|
|
|
|
return(result);
|
|
}
|
|
#endif /* UNIV_DEBUG || IB_RBT_TESTING */
|
|
|
|
/**********************************************************************//**
|
|
Turn the node's right child's left sub-tree into node's right sub-tree.
|
|
This will also make node's right child it's parent. */
|
|
static
|
|
void
|
|
rbt_rotate_left(
|
|
/*============*/
|
|
const ib_rbt_node_t* nil, /*!< in: nil node of the tree */
|
|
ib_rbt_node_t* node) /*!< in: node to rotate */
|
|
{
|
|
ib_rbt_node_t* right = node->right;
|
|
|
|
node->right = right->left;
|
|
|
|
if (right->left != nil) {
|
|
right->left->parent = node;
|
|
}
|
|
|
|
/* Right's new parent was node's parent. */
|
|
right->parent = node->parent;
|
|
|
|
/* Since root's parent is tree->nil and root->parent->left points
|
|
back to root, we can avoid the check. */
|
|
if (node == node->parent->left) {
|
|
/* Node was on the left of its parent. */
|
|
node->parent->left = right;
|
|
} else {
|
|
/* Node must have been on the right. */
|
|
node->parent->right = right;
|
|
}
|
|
|
|
/* Finally, put node on right's left. */
|
|
right->left = node;
|
|
node->parent = right;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Turn the node's left child's right sub-tree into node's left sub-tree.
|
|
This also make node's left child it's parent. */
|
|
static
|
|
void
|
|
rbt_rotate_right(
|
|
/*=============*/
|
|
const ib_rbt_node_t* nil, /*!< in: nil node of tree */
|
|
ib_rbt_node_t* node) /*!< in: node to rotate */
|
|
{
|
|
ib_rbt_node_t* left = node->left;
|
|
|
|
node->left = left->right;
|
|
|
|
if (left->right != nil) {
|
|
left->right->parent = node;
|
|
}
|
|
|
|
/* Left's new parent was node's parent. */
|
|
left->parent = node->parent;
|
|
|
|
/* Since root's parent is tree->nil and root->parent->left points
|
|
back to root, we can avoid the check. */
|
|
if (node == node->parent->right) {
|
|
/* Node was on the left of its parent. */
|
|
node->parent->right = left;
|
|
} else {
|
|
/* Node must have been on the left. */
|
|
node->parent->left = left;
|
|
}
|
|
|
|
/* Finally, put node on left's right. */
|
|
left->right = node;
|
|
node->parent = left;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Append a node to the tree. */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_tree_add_child(
|
|
/*===============*/
|
|
const ib_rbt_t* tree,
|
|
ib_rbt_bound_t* parent,
|
|
ib_rbt_node_t* node)
|
|
{
|
|
/* Cast away the const. */
|
|
ib_rbt_node_t* last = (ib_rbt_node_t*) parent->last;
|
|
|
|
if (last == tree->root || parent->result < 0) {
|
|
last->left = node;
|
|
} else {
|
|
/* FIXME: We don't handle duplicates (yet)! */
|
|
ut_a(parent->result != 0);
|
|
|
|
last->right = node;
|
|
}
|
|
|
|
node->parent = last;
|
|
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Generic binary tree insert */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_tree_insert(
|
|
/*============*/
|
|
ib_rbt_t* tree,
|
|
const void* key,
|
|
ib_rbt_node_t* node)
|
|
{
|
|
ib_rbt_bound_t parent;
|
|
ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
parent.result = 0;
|
|
parent.last = tree->root;
|
|
|
|
/* Regular binary search. */
|
|
while (current != tree->nil) {
|
|
|
|
parent.last = current;
|
|
|
|
if (tree->cmp_arg) {
|
|
parent.result = tree->compare_with_arg(
|
|
tree->cmp_arg, key, current->value);
|
|
} else {
|
|
parent.result = tree->compare(key, current->value);
|
|
}
|
|
|
|
if (parent.result < 0) {
|
|
current = current->left;
|
|
} else {
|
|
current = current->right;
|
|
}
|
|
}
|
|
|
|
ut_a(current == tree->nil);
|
|
|
|
rbt_tree_add_child(tree, &parent, node);
|
|
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Balance a tree after inserting a node. */
|
|
static
|
|
void
|
|
rbt_balance_tree(
|
|
/*=============*/
|
|
const ib_rbt_t* tree, /*!< in: tree to balance */
|
|
ib_rbt_node_t* node) /*!< in: node that was inserted */
|
|
{
|
|
const ib_rbt_node_t* nil = tree->nil;
|
|
ib_rbt_node_t* parent = node->parent;
|
|
|
|
/* Restore the red-black property. */
|
|
node->color = IB_RBT_RED;
|
|
|
|
while (node != ROOT(tree) && parent->color == IB_RBT_RED) {
|
|
ib_rbt_node_t* grand_parent = parent->parent;
|
|
|
|
if (parent == grand_parent->left) {
|
|
ib_rbt_node_t* uncle = grand_parent->right;
|
|
|
|
if (uncle->color == IB_RBT_RED) {
|
|
|
|
/* Case 1 - change the colors. */
|
|
uncle->color = IB_RBT_BLACK;
|
|
parent->color = IB_RBT_BLACK;
|
|
grand_parent->color = IB_RBT_RED;
|
|
|
|
/* Move node up the tree. */
|
|
node = grand_parent;
|
|
|
|
} else {
|
|
|
|
if (node == parent->right) {
|
|
/* Right is a black node and node is
|
|
to the right, case 2 - move node
|
|
up and rotate. */
|
|
node = parent;
|
|
rbt_rotate_left(nil, node);
|
|
}
|
|
|
|
grand_parent = node->parent->parent;
|
|
|
|
/* Case 3. */
|
|
node->parent->color = IB_RBT_BLACK;
|
|
grand_parent->color = IB_RBT_RED;
|
|
|
|
rbt_rotate_right(nil, grand_parent);
|
|
}
|
|
|
|
} else {
|
|
ib_rbt_node_t* uncle = grand_parent->left;
|
|
|
|
if (uncle->color == IB_RBT_RED) {
|
|
|
|
/* Case 1 - change the colors. */
|
|
uncle->color = IB_RBT_BLACK;
|
|
parent->color = IB_RBT_BLACK;
|
|
grand_parent->color = IB_RBT_RED;
|
|
|
|
/* Move node up the tree. */
|
|
node = grand_parent;
|
|
|
|
} else {
|
|
|
|
if (node == parent->left) {
|
|
/* Left is a black node and node is to
|
|
the right, case 2 - move node up and
|
|
rotate. */
|
|
node = parent;
|
|
rbt_rotate_right(nil, node);
|
|
}
|
|
|
|
grand_parent = node->parent->parent;
|
|
|
|
/* Case 3. */
|
|
node->parent->color = IB_RBT_BLACK;
|
|
grand_parent->color = IB_RBT_RED;
|
|
|
|
rbt_rotate_left(nil, grand_parent);
|
|
}
|
|
}
|
|
|
|
parent = node->parent;
|
|
}
|
|
|
|
/* Color the root black. */
|
|
ROOT(tree)->color = IB_RBT_BLACK;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Find the given node's successor.
|
|
@return successor node or NULL if no successor */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_find_successor(
|
|
/*===============*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
const ib_rbt_node_t* current) /*!< in: this is declared const
|
|
because it can be called via
|
|
rbt_next() */
|
|
{
|
|
const ib_rbt_node_t* nil = tree->nil;
|
|
ib_rbt_node_t* next = current->right;
|
|
|
|
/* Is there a sub-tree to the right that we can follow. */
|
|
if (next != nil) {
|
|
|
|
/* Follow the left most links of the current right child. */
|
|
while (next->left != nil) {
|
|
next = next->left;
|
|
}
|
|
|
|
} else { /* We will have to go up the tree to find the successor. */
|
|
ib_rbt_node_t* parent = current->parent;
|
|
|
|
/* Cast away the const. */
|
|
next = (ib_rbt_node_t*) current;
|
|
|
|
while (parent != tree->root && next == parent->right) {
|
|
next = parent;
|
|
parent = next->parent;
|
|
}
|
|
|
|
next = (parent == tree->root) ? NULL : parent;
|
|
}
|
|
|
|
return(next);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Find the given node's precedecessor.
|
|
@return predecessor node or NULL if no predecesor */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_find_predecessor(
|
|
/*=================*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
const ib_rbt_node_t* current) /*!< in: this is declared const
|
|
because it can be called via
|
|
rbt_prev() */
|
|
{
|
|
const ib_rbt_node_t* nil = tree->nil;
|
|
ib_rbt_node_t* prev = current->left;
|
|
|
|
/* Is there a sub-tree to the left that we can follow. */
|
|
if (prev != nil) {
|
|
|
|
/* Follow the right most links of the current left child. */
|
|
while (prev->right != nil) {
|
|
prev = prev->right;
|
|
}
|
|
|
|
} else { /* We will have to go up the tree to find the precedecessor. */
|
|
ib_rbt_node_t* parent = current->parent;
|
|
|
|
/* Cast away the const. */
|
|
prev = (ib_rbt_node_t*) current;
|
|
|
|
while (parent != tree->root && prev == parent->left) {
|
|
prev = parent;
|
|
parent = prev->parent;
|
|
}
|
|
|
|
prev = (parent == tree->root) ? NULL : parent;
|
|
}
|
|
|
|
return(prev);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Replace node with child. After applying transformations eject becomes
|
|
an orphan. */
|
|
static
|
|
void
|
|
rbt_eject_node(
|
|
/*===========*/
|
|
ib_rbt_node_t* eject, /*!< in: node to eject */
|
|
ib_rbt_node_t* node) /*!< in: node to replace with */
|
|
{
|
|
/* Update the to be ejected node's parent's child pointers. */
|
|
if (eject->parent->left == eject) {
|
|
eject->parent->left = node;
|
|
} else if (eject->parent->right == eject) {
|
|
eject->parent->right = node;
|
|
} else {
|
|
ut_a(0);
|
|
}
|
|
/* eject is now an orphan but otherwise its pointers
|
|
and color are left intact. */
|
|
|
|
node->parent = eject->parent;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Replace a node with another node. */
|
|
static
|
|
void
|
|
rbt_replace_node(
|
|
/*=============*/
|
|
ib_rbt_node_t* replace, /*!< in: node to replace */
|
|
ib_rbt_node_t* node) /*!< in: node to replace with */
|
|
{
|
|
ib_rbt_color_t color = node->color;
|
|
|
|
/* Update the node pointers. */
|
|
node->left = replace->left;
|
|
node->right = replace->right;
|
|
|
|
/* Update the child node pointers. */
|
|
node->left->parent = node;
|
|
node->right->parent = node;
|
|
|
|
/* Make the parent of replace point to node. */
|
|
rbt_eject_node(replace, node);
|
|
|
|
/* Swap the colors. */
|
|
node->color = replace->color;
|
|
replace->color = color;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Detach node from the tree replacing it with one of it's children.
|
|
@return the child node that now occupies the position of the detached node */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_detach_node(
|
|
/*============*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
ib_rbt_node_t* node) /*!< in: node to detach */
|
|
{
|
|
ib_rbt_node_t* child;
|
|
const ib_rbt_node_t* nil = tree->nil;
|
|
|
|
if (node->left != nil && node->right != nil) {
|
|
/* Case where the node to be deleted has two children. */
|
|
ib_rbt_node_t* successor = rbt_find_successor(tree, node);
|
|
|
|
ut_a(successor != nil);
|
|
ut_a(successor->parent != nil);
|
|
ut_a(successor->left == nil);
|
|
|
|
child = successor->right;
|
|
|
|
/* Remove the successor node and replace with its child. */
|
|
rbt_eject_node(successor, child);
|
|
|
|
/* Replace the node to delete with its successor node. */
|
|
rbt_replace_node(node, successor);
|
|
} else {
|
|
ut_a(node->left == nil || node->right == nil);
|
|
|
|
child = (node->left != nil) ? node->left : node->right;
|
|
|
|
/* Replace the node to delete with one of it's children. */
|
|
rbt_eject_node(node, child);
|
|
}
|
|
|
|
/* Reset the node links. */
|
|
node->parent = node->right = node->left = tree->nil;
|
|
|
|
return(child);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Rebalance the right sub-tree after deletion.
|
|
@return node to rebalance if more rebalancing required else NULL */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_balance_right(
|
|
/*==============*/
|
|
const ib_rbt_node_t* nil, /*!< in: rb tree nil node */
|
|
ib_rbt_node_t* parent, /*!< in: parent node */
|
|
ib_rbt_node_t* sibling) /*!< in: sibling node */
|
|
{
|
|
ib_rbt_node_t* node = NULL;
|
|
|
|
ut_a(sibling != nil);
|
|
|
|
/* Case 3. */
|
|
if (sibling->color == IB_RBT_RED) {
|
|
|
|
parent->color = IB_RBT_RED;
|
|
sibling->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_left(nil, parent);
|
|
|
|
sibling = parent->right;
|
|
|
|
ut_a(sibling != nil);
|
|
}
|
|
|
|
/* Since this will violate case 3 because of the change above. */
|
|
if (sibling->left->color == IB_RBT_BLACK
|
|
&& sibling->right->color == IB_RBT_BLACK) {
|
|
|
|
node = parent; /* Parent needs to be rebalanced too. */
|
|
sibling->color = IB_RBT_RED;
|
|
|
|
} else {
|
|
if (sibling->right->color == IB_RBT_BLACK) {
|
|
|
|
ut_a(sibling->left->color == IB_RBT_RED);
|
|
|
|
sibling->color = IB_RBT_RED;
|
|
sibling->left->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_right(nil, sibling);
|
|
|
|
sibling = parent->right;
|
|
ut_a(sibling != nil);
|
|
}
|
|
|
|
sibling->color = parent->color;
|
|
sibling->right->color = IB_RBT_BLACK;
|
|
|
|
parent->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_left(nil, parent);
|
|
}
|
|
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Rebalance the left sub-tree after deletion.
|
|
@return node to rebalance if more rebalancing required else NULL */
|
|
static
|
|
ib_rbt_node_t*
|
|
rbt_balance_left(
|
|
/*=============*/
|
|
const ib_rbt_node_t* nil, /*!< in: rb tree nil node */
|
|
ib_rbt_node_t* parent, /*!< in: parent node */
|
|
ib_rbt_node_t* sibling) /*!< in: sibling node */
|
|
{
|
|
ib_rbt_node_t* node = NULL;
|
|
|
|
ut_a(sibling != nil);
|
|
|
|
/* Case 3. */
|
|
if (sibling->color == IB_RBT_RED) {
|
|
|
|
parent->color = IB_RBT_RED;
|
|
sibling->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_right(nil, parent);
|
|
sibling = parent->left;
|
|
|
|
ut_a(sibling != nil);
|
|
}
|
|
|
|
/* Since this will violate case 3 because of the change above. */
|
|
if (sibling->right->color == IB_RBT_BLACK
|
|
&& sibling->left->color == IB_RBT_BLACK) {
|
|
|
|
node = parent; /* Parent needs to be rebalanced too. */
|
|
sibling->color = IB_RBT_RED;
|
|
|
|
} else {
|
|
if (sibling->left->color == IB_RBT_BLACK) {
|
|
|
|
ut_a(sibling->right->color == IB_RBT_RED);
|
|
|
|
sibling->color = IB_RBT_RED;
|
|
sibling->right->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_left(nil, sibling);
|
|
|
|
sibling = parent->left;
|
|
|
|
ut_a(sibling != nil);
|
|
}
|
|
|
|
sibling->color = parent->color;
|
|
sibling->left->color = IB_RBT_BLACK;
|
|
|
|
parent->color = IB_RBT_BLACK;
|
|
|
|
rbt_rotate_right(nil, parent);
|
|
}
|
|
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Delete the node and rebalance the tree if necessary */
|
|
static
|
|
void
|
|
rbt_remove_node_and_rebalance(
|
|
/*==========================*/
|
|
ib_rbt_t* tree, /*!< in: rb tree */
|
|
ib_rbt_node_t* node) /*!< in: node to remove */
|
|
{
|
|
/* Detach node and get the node that will be used
|
|
as rebalance start. */
|
|
ib_rbt_node_t* child = rbt_detach_node(tree, node);
|
|
|
|
if (node->color == IB_RBT_BLACK) {
|
|
ib_rbt_node_t* last = child;
|
|
|
|
ROOT(tree)->color = IB_RBT_RED;
|
|
|
|
while (child && child->color == IB_RBT_BLACK) {
|
|
ib_rbt_node_t* parent = child->parent;
|
|
|
|
/* Did the deletion cause an imbalance in the
|
|
parents left sub-tree. */
|
|
if (parent->left == child) {
|
|
|
|
child = rbt_balance_right(
|
|
tree->nil, parent, parent->right);
|
|
|
|
} else if (parent->right == child) {
|
|
|
|
child = rbt_balance_left(
|
|
tree->nil, parent, parent->left);
|
|
|
|
} else {
|
|
ut_error;
|
|
}
|
|
|
|
if (child) {
|
|
last = child;
|
|
}
|
|
}
|
|
|
|
ut_a(last);
|
|
|
|
last->color = IB_RBT_BLACK;
|
|
ROOT(tree)->color = IB_RBT_BLACK;
|
|
}
|
|
|
|
/* Note that we have removed a node from the tree. */
|
|
--tree->n_nodes;
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Recursively free the nodes. */
|
|
static
|
|
void
|
|
rbt_free_node(
|
|
/*==========*/
|
|
ib_rbt_node_t* node, /*!< in: node to free */
|
|
ib_rbt_node_t* nil) /*!< in: rb tree nil node */
|
|
{
|
|
if (node != nil) {
|
|
rbt_free_node(node->left, nil);
|
|
rbt_free_node(node->right, nil);
|
|
|
|
ut_free(node);
|
|
}
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Free all the nodes and free the tree. */
|
|
void
|
|
rbt_free(
|
|
/*=====*/
|
|
ib_rbt_t* tree) /*!< in: rb tree to free */
|
|
{
|
|
rbt_free_node(tree->root, tree->nil);
|
|
ut_free(tree->nil);
|
|
ut_free(tree);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Create an instance of a red black tree, whose comparison function takes
|
|
an argument
|
|
@return an empty rb tree */
|
|
ib_rbt_t*
|
|
rbt_create_arg_cmp(
|
|
/*===============*/
|
|
size_t sizeof_value, /*!< in: sizeof data item */
|
|
ib_rbt_arg_compare
|
|
compare, /*!< in: fn to compare items */
|
|
void* cmp_arg) /*!< in: compare fn arg */
|
|
{
|
|
ib_rbt_t* tree;
|
|
|
|
ut_a(cmp_arg);
|
|
|
|
tree = rbt_create(sizeof_value, NULL);
|
|
tree->cmp_arg = cmp_arg;
|
|
tree->compare_with_arg = compare;
|
|
|
|
return(tree);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Create an instance of a red black tree.
|
|
@return an empty rb tree */
|
|
ib_rbt_t*
|
|
rbt_create(
|
|
/*=======*/
|
|
size_t sizeof_value, /*!< in: sizeof data item */
|
|
ib_rbt_compare compare) /*!< in: fn to compare items */
|
|
{
|
|
ib_rbt_t* tree;
|
|
ib_rbt_node_t* node;
|
|
|
|
tree = (ib_rbt_t*) ut_zalloc_nokey(sizeof(*tree));
|
|
|
|
tree->sizeof_value = sizeof_value;
|
|
|
|
/* Create the sentinel (NIL) node. */
|
|
node = tree->nil = (ib_rbt_node_t*) ut_zalloc_nokey(sizeof(*node));
|
|
|
|
node->color = IB_RBT_BLACK;
|
|
node->parent = node->left = node->right = node;
|
|
|
|
/* Create the "fake" root, the real root node will be the
|
|
left child of this node. */
|
|
node = tree->root = (ib_rbt_node_t*) ut_zalloc_nokey(sizeof(*node));
|
|
|
|
node->color = IB_RBT_BLACK;
|
|
node->parent = node->left = node->right = tree->nil;
|
|
|
|
tree->compare = compare;
|
|
|
|
return(tree);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Generic insert of a value in the rb tree.
|
|
@return inserted node */
|
|
const ib_rbt_node_t*
|
|
rbt_insert(
|
|
/*=======*/
|
|
ib_rbt_t* tree, /*!< in: rb tree */
|
|
const void* key, /*!< in: key for ordering */
|
|
const void* value) /*!< in: value of key, this value
|
|
is copied to the node */
|
|
{
|
|
ib_rbt_node_t* node;
|
|
|
|
/* Create the node that will hold the value data. */
|
|
node = (ib_rbt_node_t*) ut_malloc_nokey(SIZEOF_NODE(tree));
|
|
|
|
memcpy(node->value, value, tree->sizeof_value);
|
|
node->parent = node->left = node->right = tree->nil;
|
|
|
|
/* Insert in the tree in the usual way. */
|
|
rbt_tree_insert(tree, key, node);
|
|
rbt_balance_tree(tree, node);
|
|
|
|
++tree->n_nodes;
|
|
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Add a new node to the tree, useful for data that is pre-sorted.
|
|
@return appended node */
|
|
const ib_rbt_node_t*
|
|
rbt_add_node(
|
|
/*=========*/
|
|
ib_rbt_t* tree, /*!< in: rb tree */
|
|
ib_rbt_bound_t* parent, /*!< in: bounds */
|
|
const void* value) /*!< in: this value is copied
|
|
to the node */
|
|
{
|
|
ib_rbt_node_t* node;
|
|
|
|
/* Create the node that will hold the value data */
|
|
node = (ib_rbt_node_t*) ut_malloc_nokey(SIZEOF_NODE(tree));
|
|
|
|
memcpy(node->value, value, tree->sizeof_value);
|
|
node->parent = node->left = node->right = tree->nil;
|
|
|
|
/* If tree is empty */
|
|
if (parent->last == NULL) {
|
|
parent->last = tree->root;
|
|
}
|
|
|
|
/* Append the node, the hope here is that the caller knows
|
|
what s/he is doing. */
|
|
rbt_tree_add_child(tree, parent, node);
|
|
rbt_balance_tree(tree, node);
|
|
|
|
++tree->n_nodes;
|
|
|
|
#if defined UNIV_DEBUG || defined IB_RBT_TESTING
|
|
ut_a(rbt_validate(tree));
|
|
#endif
|
|
return(node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Find a matching node in the rb tree.
|
|
@return NULL if not found else the node where key was found */
|
|
static
|
|
const ib_rbt_node_t*
|
|
rbt_lookup(
|
|
/*=======*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
const void* key) /*!< in: key to use for search */
|
|
{
|
|
const ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
/* Regular binary search. */
|
|
while (current != tree->nil) {
|
|
int result;
|
|
|
|
if (tree->cmp_arg) {
|
|
result = tree->compare_with_arg(
|
|
tree->cmp_arg, key, current->value);
|
|
} else {
|
|
result = tree->compare(key, current->value);
|
|
}
|
|
|
|
if (result < 0) {
|
|
current = current->left;
|
|
} else if (result > 0) {
|
|
current = current->right;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(current != tree->nil ? current : NULL);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Delete a node indentified by key.
|
|
@return TRUE if success FALSE if not found */
|
|
ibool
|
|
rbt_delete(
|
|
/*=======*/
|
|
ib_rbt_t* tree, /*!< in: rb tree */
|
|
const void* key) /*!< in: key to delete */
|
|
{
|
|
ibool deleted = FALSE;
|
|
ib_rbt_node_t* node = (ib_rbt_node_t*) rbt_lookup(tree, key);
|
|
|
|
if (node) {
|
|
rbt_remove_node_and_rebalance(tree, node);
|
|
|
|
ut_free(node);
|
|
deleted = TRUE;
|
|
}
|
|
|
|
return(deleted);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Remove a node from the rb tree, the node is not free'd, that is the
|
|
callers responsibility.
|
|
@return deleted node but without the const */
|
|
ib_rbt_node_t*
|
|
rbt_remove_node(
|
|
/*============*/
|
|
ib_rbt_t* tree, /*!< in: rb tree */
|
|
const ib_rbt_node_t* const_node) /*!< in: node to delete, this
|
|
is a fudge and declared const
|
|
because the caller can access
|
|
only const nodes */
|
|
{
|
|
/* Cast away the const. */
|
|
rbt_remove_node_and_rebalance(tree, (ib_rbt_node_t*) const_node);
|
|
|
|
/* This is to make it easier to do something like this:
|
|
ut_free(rbt_remove_node(node));
|
|
*/
|
|
|
|
return((ib_rbt_node_t*) const_node);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Find the node that has the greatest key that is <= key.
|
|
@return value of result */
|
|
int
|
|
rbt_search(
|
|
/*=======*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
ib_rbt_bound_t* parent, /*!< in: search bounds */
|
|
const void* key) /*!< in: key to search */
|
|
{
|
|
ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
/* Every thing is greater than the NULL root. */
|
|
parent->result = 1;
|
|
parent->last = NULL;
|
|
|
|
while (current != tree->nil) {
|
|
|
|
parent->last = current;
|
|
|
|
if (tree->cmp_arg) {
|
|
parent->result = tree->compare_with_arg(
|
|
tree->cmp_arg, key, current->value);
|
|
} else {
|
|
parent->result = tree->compare(key, current->value);
|
|
}
|
|
|
|
if (parent->result > 0) {
|
|
current = current->right;
|
|
} else if (parent->result < 0) {
|
|
current = current->left;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(parent->result);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Find the node that has the greatest key that is <= key. But use the
|
|
supplied comparison function.
|
|
@return value of result */
|
|
int
|
|
rbt_search_cmp(
|
|
/*===========*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
ib_rbt_bound_t* parent, /*!< in: search bounds */
|
|
const void* key, /*!< in: key to search */
|
|
ib_rbt_compare compare, /*!< in: fn to compare items */
|
|
ib_rbt_arg_compare
|
|
arg_compare) /*!< in: fn to compare items
|
|
with argument */
|
|
{
|
|
ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
/* Every thing is greater than the NULL root. */
|
|
parent->result = 1;
|
|
parent->last = NULL;
|
|
|
|
while (current != tree->nil) {
|
|
|
|
parent->last = current;
|
|
|
|
if (arg_compare) {
|
|
ut_ad(tree->cmp_arg);
|
|
parent->result = arg_compare(
|
|
tree->cmp_arg, key, current->value);
|
|
} else {
|
|
parent->result = compare(key, current->value);
|
|
}
|
|
|
|
if (parent->result > 0) {
|
|
current = current->right;
|
|
} else if (parent->result < 0) {
|
|
current = current->left;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
|
|
return(parent->result);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Return the left most node in the tree. */
|
|
const ib_rbt_node_t*
|
|
rbt_first(
|
|
/*======*/
|
|
/* out leftmost node or NULL */
|
|
const ib_rbt_t* tree) /* in: rb tree */
|
|
{
|
|
ib_rbt_node_t* first = NULL;
|
|
ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
while (current != tree->nil) {
|
|
first = current;
|
|
current = current->left;
|
|
}
|
|
|
|
return(first);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Return the right most node in the tree.
|
|
@return the rightmost node or NULL */
|
|
const ib_rbt_node_t*
|
|
rbt_last(
|
|
/*=====*/
|
|
const ib_rbt_t* tree) /*!< in: rb tree */
|
|
{
|
|
ib_rbt_node_t* last = NULL;
|
|
ib_rbt_node_t* current = ROOT(tree);
|
|
|
|
while (current != tree->nil) {
|
|
last = current;
|
|
current = current->right;
|
|
}
|
|
|
|
return(last);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Return the next node.
|
|
@return node next from current */
|
|
const ib_rbt_node_t*
|
|
rbt_next(
|
|
/*=====*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
const ib_rbt_node_t* current) /*!< in: current node */
|
|
{
|
|
return(current ? rbt_find_successor(tree, current) : NULL);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Return the previous node.
|
|
@return node prev from current */
|
|
const ib_rbt_node_t*
|
|
rbt_prev(
|
|
/*=====*/
|
|
const ib_rbt_t* tree, /*!< in: rb tree */
|
|
const ib_rbt_node_t* current) /*!< in: current node */
|
|
{
|
|
return(current ? rbt_find_predecessor(tree, current) : NULL);
|
|
}
|
|
|
|
/**********************************************************************//**
|
|
Merge the node from dst into src. Return the number of nodes merged.
|
|
@return no. of recs merged */
|
|
ulint
|
|
rbt_merge_uniq(
|
|
/*===========*/
|
|
ib_rbt_t* dst, /*!< in: dst rb tree */
|
|
const ib_rbt_t* src) /*!< in: src rb tree */
|
|
{
|
|
ib_rbt_bound_t parent;
|
|
ulint n_merged = 0;
|
|
const ib_rbt_node_t* src_node = rbt_first(src);
|
|
|
|
if (rbt_empty(src) || dst == src) {
|
|
return(0);
|
|
}
|
|
|
|
for (/* No op */; src_node; src_node = rbt_next(src, src_node)) {
|
|
|
|
if (rbt_search(dst, &parent, src_node->value) != 0) {
|
|
rbt_add_node(dst, &parent, src_node->value);
|
|
++n_merged;
|
|
}
|
|
}
|
|
|
|
return(n_merged);
|
|
}
|
|
|
|
#if defined UNIV_DEBUG || defined IB_RBT_TESTING
|
|
/**********************************************************************//**
|
|
Check that every path from the root to the leaves has the same count and
|
|
the tree nodes are in order.
|
|
@return TRUE if OK FALSE otherwise */
|
|
ibool
|
|
rbt_validate(
|
|
/*=========*/
|
|
const ib_rbt_t* tree) /*!< in: RB tree to validate */
|
|
{
|
|
if (rbt_count_black_nodes(tree, ROOT(tree)) > 0) {
|
|
return(rbt_check_ordering(tree));
|
|
}
|
|
|
|
return(FALSE);
|
|
}
|
|
#endif /* UNIV_DEBUG || IB_RBT_TESTING */
|