mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-13 20:09:59 +00:00
787 lines
27 KiB
C
787 lines
27 KiB
C
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
// $Id: tree.h,v 1.13 2006/11/18 03:12:35 dmouldin Exp $
|
||
|
//
|
||
|
// Visual Leak Detector - Red-black Tree Template
|
||
|
// Copyright (c) 2005-2006 Dan Moulding
|
||
|
//
|
||
|
// This library is free software; you can redistribute it and/or
|
||
|
// modify it under the terms of the GNU Lesser General Public
|
||
|
// License as published by the Free Software Foundation; either
|
||
|
// version 2.1 of the License, or (at your option) any later version.
|
||
|
//
|
||
|
// This library 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
|
||
|
// Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public
|
||
|
// License along with this library; if not, write to the Free Software
|
||
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||
|
//
|
||
|
// See COPYING.txt for the full terms of the GNU Lesser General Public License.
|
||
|
//
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#pragma once
|
||
|
|
||
|
#ifndef VLDBUILD
|
||
|
#error \
|
||
|
"This header should only be included by Visual Leak Detector when building it from source. \
|
||
|
Applications should never include this header."
|
||
|
#endif
|
||
|
|
||
|
#include "vldheap.h" // Provides internal new and delete operators.
|
||
|
|
||
|
#define TREE_DEFAULT_RESERVE 32 // By default, trees reserve enough space, in advance, for this many nodes.
|
||
|
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// The Tree Template Class
|
||
|
//
|
||
|
// This data structure is the internal data structure behind the lightweight
|
||
|
// STL-like container classes. This is a red-black tree which provides for
|
||
|
// fast insert, find, and erase operations.
|
||
|
//
|
||
|
// The binary tree nodes are overlaid on top of larger chunks of allocated
|
||
|
// memory (called chunks) which are arranged in a simple linked list. This
|
||
|
// allows the tree to grow (add nodes) dynamically without incurring a heap
|
||
|
// hit each time a new node is added.
|
||
|
//
|
||
|
// The Tree class provides member functions which make it easily adaptable to
|
||
|
// an STL-like interface so that it can be used as the backend for STL-like
|
||
|
// container classes.
|
||
|
//
|
||
|
template <typename T>
|
||
|
class Tree
|
||
|
{
|
||
|
public:
|
||
|
// This is a red-black tree.
|
||
|
enum color_e {
|
||
|
red,
|
||
|
black
|
||
|
};
|
||
|
|
||
|
// The node is the basic data structure which the tree is built from.
|
||
|
typedef struct node_s {
|
||
|
color_e color; // The node's color.
|
||
|
T key; // The node's value, by which nodes are sorted.
|
||
|
union {
|
||
|
struct node_s *left; // For nodes in the tree, the node's left child.
|
||
|
struct node_s *next; // For nodes in the free list, the next node on the free list.
|
||
|
};
|
||
|
struct node_s *parent; // The node's parent.
|
||
|
struct node_s *right; // The node's right child.
|
||
|
} node_t;
|
||
|
|
||
|
// Reserve capacity for the tree is allocated in large chunks with room for
|
||
|
// many nodes.
|
||
|
typedef struct chunk_s {
|
||
|
struct chunk_s *next; // Pointer to the next node in the chunk list.
|
||
|
node_t *nodes; // Pointer to an array (of variable size) where nodes are stored.
|
||
|
} chunk_t;
|
||
|
|
||
|
// Constructor
|
||
|
Tree ()
|
||
|
{
|
||
|
m_freelist = NULL;
|
||
|
InitializeCriticalSection(&m_lock);
|
||
|
m_nil.color = black;
|
||
|
m_nil.key = T();
|
||
|
m_nil.left = &m_nil;
|
||
|
m_nil.parent = &m_nil;
|
||
|
m_nil.right = &m_nil;
|
||
|
m_reserve = TREE_DEFAULT_RESERVE;
|
||
|
m_root = &m_nil;
|
||
|
m_store = NULL;
|
||
|
m_storetail = NULL;
|
||
|
}
|
||
|
|
||
|
// Copy constructor - The sole purpose of this constructor's existence is
|
||
|
// to ensure that trees are not being inadvertently copied.
|
||
|
Tree (const Tree& source)
|
||
|
{
|
||
|
assert(FALSE); // Do not make copies of trees!
|
||
|
}
|
||
|
|
||
|
// Destructor
|
||
|
~Tree ()
|
||
|
{
|
||
|
chunk_t *cur;
|
||
|
chunk_t *temp;
|
||
|
|
||
|
// Free all the chunks in the chunk list.
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
cur = m_store;
|
||
|
while (cur != NULL) {
|
||
|
temp = cur;
|
||
|
cur = cur->next;
|
||
|
delete [] temp->nodes;
|
||
|
delete temp;
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
DeleteCriticalSection(&m_lock);
|
||
|
}
|
||
|
|
||
|
// operator = - Assignment operator. For efficiency, we want to avoid ever
|
||
|
// making copies of Trees (only pointer passing or reference passing
|
||
|
// should be performed). The sole purpose of this assignment operator is
|
||
|
// to ensure that no copying is being done inadvertently.
|
||
|
//
|
||
|
Tree<T>& operator = (const Tree<T> &other)
|
||
|
{
|
||
|
// Don't make copies of Trees!
|
||
|
assert(FALSE);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
// begin - Obtains a pointer to the first node (the node with the smallest
|
||
|
// key value) in the tree.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a pointer to the first node in the tree.
|
||
|
//
|
||
|
typename Tree::node_t* begin () const
|
||
|
{
|
||
|
node_t *cur;
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
if (m_root == &m_nil) {
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
cur = m_root;
|
||
|
while (cur->left != &m_nil) {
|
||
|
cur = cur->left;
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
|
||
|
return cur;
|
||
|
}
|
||
|
|
||
|
// erase - Erases the specified node from the tree. Note that this does
|
||
|
// not cause the key associated with the erased node to be freed. The
|
||
|
// caller is responsible for freeing any dynamically allocated memory
|
||
|
// associated with the key.
|
||
|
//
|
||
|
// - node (IN): Pointer to the node to erase from the tree.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID erase (typename Tree::node_t *node)
|
||
|
{
|
||
|
node_t *child;
|
||
|
node_t *cur;
|
||
|
node_t *erasure;
|
||
|
node_t *sibling;
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
|
||
|
if ((node->left == &m_nil) || (node->right == &m_nil)) {
|
||
|
// The node to be erased has less than two children. It can be directly
|
||
|
// removed from the tree.
|
||
|
erasure = node;
|
||
|
}
|
||
|
else {
|
||
|
// The node to be erased has two children. It can only be removed
|
||
|
// indirectly. The actual node will stay where it is, but it's contents
|
||
|
// will be replaced by it's in-order successor's contents. The successor
|
||
|
// node will then be erased. Find the successor.
|
||
|
erasure = node->right;
|
||
|
while (erasure->left != &m_nil) {
|
||
|
erasure = erasure->left;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Select the child node which will replace the node to be erased.
|
||
|
if (erasure->left != &m_nil) {
|
||
|
child = erasure->left;
|
||
|
}
|
||
|
else {
|
||
|
child = erasure->right;
|
||
|
}
|
||
|
|
||
|
// Replace the node to be erased with the selected child.
|
||
|
child->parent = erasure->parent;
|
||
|
if (child->parent == &m_nil) {
|
||
|
// The root of the tree is being erased. The child becomes root.
|
||
|
m_root = child;
|
||
|
}
|
||
|
else {
|
||
|
if (erasure == erasure->parent->left) {
|
||
|
erasure->parent->left = child;
|
||
|
}
|
||
|
else {
|
||
|
erasure->parent->right = child;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (erasure != node) {
|
||
|
// The node being erased from the tree is the successor of the actual
|
||
|
// node to be erased. Replace the contents of the node to be erased
|
||
|
// with the successor's contents.
|
||
|
node->key = erasure->key;
|
||
|
}
|
||
|
|
||
|
if (erasure->color == black) {
|
||
|
// The node being erased from the tree is black. Restructuring of the
|
||
|
// tree may be needed so that black-height is maintained.
|
||
|
cur = child;
|
||
|
while ((cur != m_root) && (cur->color == black)) {
|
||
|
if (cur == cur->parent->left) {
|
||
|
// Current node is a left child.
|
||
|
sibling = cur->parent->right;
|
||
|
if (sibling->color == red) {
|
||
|
// Sibling is red. Rotate sibling up and color it black.
|
||
|
sibling->color = black;
|
||
|
cur->parent->color = red;
|
||
|
_rotateleft(cur->parent);
|
||
|
sibling = cur->parent->right;
|
||
|
}
|
||
|
if ((sibling->left->color == black) && (sibling->right->color == black)) {
|
||
|
// Both of sibling's children are black. Color sibling red.
|
||
|
sibling->color = red;
|
||
|
cur = cur->parent;
|
||
|
}
|
||
|
else {
|
||
|
// At least one of sibling's children is red.
|
||
|
if (sibling->right->color == black) {
|
||
|
sibling->left->color = black;
|
||
|
sibling->color = red;
|
||
|
_rotateright(sibling);
|
||
|
sibling = cur->parent->right;
|
||
|
}
|
||
|
sibling->color = cur->parent->color;
|
||
|
cur->parent->color = black;
|
||
|
sibling->right->color = black;
|
||
|
_rotateleft(cur->parent);
|
||
|
cur = m_root;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// Current node is a right child.
|
||
|
sibling = cur->parent->left;
|
||
|
if (sibling->color == red) {
|
||
|
// Sibling is red. Rotate sibling up and color it black.
|
||
|
sibling->color = black;
|
||
|
cur->parent->color = red;
|
||
|
_rotateright(cur->parent);
|
||
|
sibling = cur->parent->left;
|
||
|
}
|
||
|
if ((sibling->left->color == black) && (sibling->right->color == black)) {
|
||
|
// Both of sibling's children are black. Color sibling red.
|
||
|
sibling->color = red;
|
||
|
cur = cur->parent;
|
||
|
}
|
||
|
else {
|
||
|
// At least one of sibling's children is red.
|
||
|
if (sibling->left->color == black) {
|
||
|
sibling->right->color = black;
|
||
|
sibling->color = red;
|
||
|
_rotateleft(sibling);
|
||
|
sibling = cur->parent->left;
|
||
|
}
|
||
|
sibling->color = cur->parent->color;
|
||
|
cur->parent->color = black;
|
||
|
sibling->left->color = black;
|
||
|
_rotateright(cur->parent);
|
||
|
cur = m_root;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
cur->color = black;
|
||
|
}
|
||
|
|
||
|
// Put the erased node onto the free list.
|
||
|
erasure->next = m_freelist;
|
||
|
m_freelist = erasure;
|
||
|
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
}
|
||
|
|
||
|
// erase - Erases the specified key from the tree. Note that this does
|
||
|
// not cause the key associated with the erased node to be freed. The
|
||
|
// caller is responsible for freeing any dynamically allocated memory
|
||
|
// associated with the key.
|
||
|
//
|
||
|
// - key (IN): The key to erase from the tree. This value is treated as
|
||
|
// the key for sorting within the tree. It must therefore be of a type
|
||
|
// which supports the "<" operator.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID erase (const T &key)
|
||
|
{
|
||
|
node_t *node;
|
||
|
|
||
|
// Find the node to erase.
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
node = m_root;
|
||
|
while (node != &m_nil) {
|
||
|
if (node->key < key) {
|
||
|
// Go right.
|
||
|
node = node->right;
|
||
|
}
|
||
|
else if (key < node->key) {
|
||
|
// Go left.
|
||
|
node = node->left;
|
||
|
}
|
||
|
else {
|
||
|
// Found it.
|
||
|
erase(node);
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
|
||
|
// 'key' is not in the tree.
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// find - Obtains a pointer to the node corresponding to the specified key.
|
||
|
//
|
||
|
// - key (IN): The value to search for in the tree. This value is treated
|
||
|
// as the key for sorting within the tree. It must therefore be of a
|
||
|
// type which supports the "<" operator.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a pointer to the node corresponding to the specified key. If
|
||
|
// the key is not in the tree, then "find" returns NULL.
|
||
|
//
|
||
|
typename Tree::node_t* find (const T &key) const
|
||
|
{
|
||
|
node_t *cur;
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
cur = m_root;
|
||
|
while (cur != &m_nil) {
|
||
|
if (cur->key < key) {
|
||
|
// Go right.
|
||
|
cur = cur->right;
|
||
|
}
|
||
|
else if (key < cur->key) {
|
||
|
// Go left.
|
||
|
cur = cur->left;
|
||
|
}
|
||
|
else {
|
||
|
// Found it.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return cur;
|
||
|
}
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
|
||
|
// 'key' is not in the tree.
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
// insert - Inserts a new key into the tree.
|
||
|
//
|
||
|
// - key (IN): The key to insert into the tree. This value is treated as
|
||
|
// the key for sorting within the tree. It must therefore be of a type
|
||
|
// which supports the "<" operator.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a pointer to the node corresponding to the newly inserted
|
||
|
// key. If an attempt is made to insert a key which is already in the
|
||
|
// tree, then NULL is returned and the new key is not inserted.
|
||
|
//
|
||
|
typename Tree::node_t* insert (const T &key)
|
||
|
{
|
||
|
node_t *cur;
|
||
|
node_t *node;
|
||
|
node_t *parent;
|
||
|
node_t *uncle;
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
|
||
|
// Find the location where the new node should be inserted..
|
||
|
cur = m_root;
|
||
|
parent = &m_nil;
|
||
|
while (cur != &m_nil) {
|
||
|
parent = cur;
|
||
|
if (cur->key < key) {
|
||
|
// Go right.
|
||
|
cur = cur->right;
|
||
|
}
|
||
|
else if (key < cur->key) {
|
||
|
// Go left.
|
||
|
cur = cur->left;
|
||
|
}
|
||
|
else {
|
||
|
// Keys in the tree must be unique.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Obtain a new node from the free list.
|
||
|
if (m_freelist == NULL) {
|
||
|
// Allocate additional storage.
|
||
|
reserve(m_reserve);
|
||
|
}
|
||
|
node = m_freelist;
|
||
|
m_freelist = m_freelist->next;
|
||
|
|
||
|
// Initialize the new node and insert it.
|
||
|
node->color = red;
|
||
|
node->key = key;
|
||
|
node->left = &m_nil;
|
||
|
node->parent = parent;
|
||
|
node->right = &m_nil;
|
||
|
if (parent == &m_nil) {
|
||
|
// The tree is empty. The new node becomes root.
|
||
|
m_root = node;
|
||
|
}
|
||
|
else {
|
||
|
if (parent->key < key) {
|
||
|
// New node is a right child.
|
||
|
parent->right = node;
|
||
|
}
|
||
|
else {
|
||
|
// New node is a left child.
|
||
|
parent->left = node;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Rebalance and/or adjust the tree, if necessary.
|
||
|
cur = node;
|
||
|
while (cur->parent->color == red) {
|
||
|
// Double-red violation. Rebalancing/adjustment needed.
|
||
|
if (cur->parent == cur->parent->parent->left) {
|
||
|
// Parent is the left child. Uncle is the right child.
|
||
|
uncle = cur->parent->parent->right;
|
||
|
if (uncle->color == red) {
|
||
|
// Uncle is red. Recolor.
|
||
|
cur->parent->parent->color = red;
|
||
|
cur->parent->color = black;
|
||
|
uncle->color = black;
|
||
|
cur = cur->parent->parent;
|
||
|
}
|
||
|
else {
|
||
|
// Uncle is black. Restructure.
|
||
|
if (cur == cur->parent->right) {
|
||
|
cur = cur->parent;
|
||
|
_rotateleft(cur);
|
||
|
}
|
||
|
cur->parent->color = black;
|
||
|
cur->parent->parent->color = red;
|
||
|
_rotateright(cur->parent->parent);
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// Parent is the right child. Uncle is the left child.
|
||
|
uncle = cur->parent->parent->left;
|
||
|
if (uncle->color == red) {
|
||
|
// Uncle is red. Recolor.
|
||
|
cur->parent->parent->color = red;
|
||
|
cur->parent->color = black;
|
||
|
uncle->color = black;
|
||
|
cur = cur->parent->parent;
|
||
|
}
|
||
|
else {
|
||
|
// Uncle is black. Restructure.
|
||
|
if (cur == cur->parent->left) {
|
||
|
cur = cur->parent;
|
||
|
_rotateright(cur);
|
||
|
}
|
||
|
cur->parent->color = black;
|
||
|
cur->parent->parent->color = red;
|
||
|
_rotateleft(cur->parent->parent);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// The root node is always colored black.
|
||
|
m_root->color = black;
|
||
|
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
|
||
|
return node;
|
||
|
}
|
||
|
|
||
|
// next - Obtains a pointer to the in-order successor of the specified
|
||
|
// node.
|
||
|
//
|
||
|
// - node (IN): Pointer to the node whose in-order successor to retrieve.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a pointer to the node's in-order successor. If the specified
|
||
|
// node corresponds to the largest value in the tree, then the specified
|
||
|
// node has no successor and "next" will return NULL.
|
||
|
//
|
||
|
typename Tree::node_t* next (typename Tree::node_t *node) const
|
||
|
{
|
||
|
node_t* cur;
|
||
|
|
||
|
if (node == NULL) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
if (node->right != &m_nil) {
|
||
|
// 'node' has a right child. Successor is the far left node in
|
||
|
// the right subtree.
|
||
|
cur = node->right;
|
||
|
while (cur->left != &m_nil) {
|
||
|
cur = cur->left;
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return cur;
|
||
|
}
|
||
|
else if (node->parent != &m_nil) {
|
||
|
// 'node' has no right child, but does have a parent.
|
||
|
if (node == node->parent->left) {
|
||
|
// 'node' is a left child; node's parent is successor.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return node->parent;
|
||
|
}
|
||
|
else {
|
||
|
// 'node' is a right child.
|
||
|
cur = node;
|
||
|
// Go up the tree until we find a parent to the right.
|
||
|
while (cur->parent != &m_nil) {
|
||
|
if (cur == cur->parent->right) {
|
||
|
cur = cur->parent;
|
||
|
continue;
|
||
|
}
|
||
|
else {
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return cur->parent;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// There is no parent greater than 'node'. 'node' is the
|
||
|
// maximum node.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// 'node' is root and root is the maximum node.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// prev - Obtains a pointer to the in-order predecessor of the specified
|
||
|
// node.
|
||
|
//
|
||
|
// - node (IN): Pointer to the node whose in-order predecessor to retrieve.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns a pointer to the node's in-order predecessor. If the specified
|
||
|
// node corresponds to the smallest value in the tree, then the specified
|
||
|
// node has no predecessor and "prev" will return NULL.
|
||
|
//
|
||
|
typename Tree::node_t* prev (typename Tree::node_t *node) const
|
||
|
{
|
||
|
node_t* cur;
|
||
|
|
||
|
if (node == NULL) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
if (node->left != &m_nil) {
|
||
|
// 'node' has left child. Predecessor is the far right node in the
|
||
|
// left subtree.
|
||
|
cur = node->left;
|
||
|
while (cur->right != &m_nil) {
|
||
|
cur = cur->right;
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return cur;
|
||
|
}
|
||
|
else if (node->parent != & m_nil) {
|
||
|
// 'node' has no left child, but does have a parent.
|
||
|
if (node == node->parent->right) {
|
||
|
// 'node' is a right child; node's parent is predecessor.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return node->parent;
|
||
|
}
|
||
|
else {
|
||
|
// 'node is a left child.
|
||
|
cur = node;
|
||
|
// Go up the tree until we find a parent to the left.
|
||
|
while (cur->parent != &m_nil) {
|
||
|
if (cur == cur->parent->left) {
|
||
|
cur = cur->parent;
|
||
|
continue;
|
||
|
}
|
||
|
else {
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return cur->parent;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// There is no parent less than 'node'. 'node' is the minimum
|
||
|
// node.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
else {
|
||
|
// 'node' is root and root is the minimum node.
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
return NULL;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// reserve - Reserves storage for a number of nodes in advance and/or sets
|
||
|
// the number of nodes for which the tree will automatically reserve
|
||
|
// storage when the tree needs to "grow" to accomodate new values being
|
||
|
// inserted into the tree. If this function is not called to set the
|
||
|
// reserve size to a specific value, then a pre-determined default value
|
||
|
// will be used. If this function is called when the tree currently has
|
||
|
// no reserve storage, then in addition to setting the tree's reserve
|
||
|
// value, it will also cause the tree to immediately reserve the
|
||
|
// specified amount of storage.
|
||
|
//
|
||
|
// - count (IN): The number of individual nodes' worth of storage to
|
||
|
// reserve.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// Returns the previously defined reserve value.
|
||
|
//
|
||
|
UINT32 reserve (UINT32 count)
|
||
|
{
|
||
|
chunk_t *chunk;
|
||
|
UINT32 index;
|
||
|
UINT32 oldreserve = m_reserve;
|
||
|
|
||
|
if (count != m_reserve) {
|
||
|
if (count < 1) {
|
||
|
// Minimum reserve size is 1.
|
||
|
m_reserve = 1;
|
||
|
}
|
||
|
else {
|
||
|
m_reserve = count;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
EnterCriticalSection(&m_lock);
|
||
|
if (m_freelist == NULL) {
|
||
|
// Allocate additional storage.
|
||
|
// Link a new chunk into the chunk list.
|
||
|
chunk = new Tree::chunk_t;
|
||
|
chunk->nodes = new Tree::node_s [m_reserve];
|
||
|
chunk->next = NULL;
|
||
|
if (m_store == NULL) {
|
||
|
m_store = chunk;
|
||
|
}
|
||
|
else {
|
||
|
m_storetail->next = chunk;
|
||
|
}
|
||
|
m_storetail = chunk;
|
||
|
|
||
|
// Link the individual nodes together to form a new free list.
|
||
|
for (index = 0; index < m_reserve - 1; index++) {
|
||
|
chunk->nodes[index].next = &chunk->nodes[index + 1];
|
||
|
}
|
||
|
chunk->nodes[index].next = NULL;
|
||
|
m_freelist = chunk->nodes;
|
||
|
}
|
||
|
LeaveCriticalSection(&m_lock);
|
||
|
|
||
|
return oldreserve;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
// _rotateleft: Rotates a pair of nodes counter-clockwise so that the parent
|
||
|
// node becomes the left child and the right child becomes the parent.
|
||
|
//
|
||
|
// - parent (IN): Pointer to the parent to rotate about.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID _rotateleft (typename Tree::node_t *parent)
|
||
|
{
|
||
|
node_t *child = parent->right;
|
||
|
|
||
|
// Reassign the child's left subtree to the parent.
|
||
|
parent->right = child->left;
|
||
|
if (child->left != &m_nil) {
|
||
|
child->left->parent = parent;
|
||
|
}
|
||
|
|
||
|
// Reassign the child/parent relationship.
|
||
|
child->parent = parent->parent;
|
||
|
if (parent->parent == &m_nil) {
|
||
|
// The child becomes the new root node.
|
||
|
m_root = child;
|
||
|
}
|
||
|
else {
|
||
|
// Point the grandparent at the child.
|
||
|
if (parent == parent->parent->left) {
|
||
|
parent->parent->left = child;
|
||
|
}
|
||
|
else {
|
||
|
parent->parent->right = child;
|
||
|
}
|
||
|
}
|
||
|
child->left = parent;
|
||
|
parent->parent = child;
|
||
|
}
|
||
|
|
||
|
// _rotateright - Rotates a pair of nodes clockwise so that the parent node
|
||
|
// becomes the right child and the left child becomes the parent.
|
||
|
//
|
||
|
// - parent (IN): Pointer to the parent to rotate about.
|
||
|
//
|
||
|
// Return Value:
|
||
|
//
|
||
|
// None.
|
||
|
//
|
||
|
VOID _rotateright (typename Tree::node_t *parent)
|
||
|
{
|
||
|
node_t *child = parent->left;
|
||
|
|
||
|
// Reassign the child's right subtree to the parent.
|
||
|
parent->left = child->right;
|
||
|
if (child->right != &m_nil) {
|
||
|
child->right->parent = parent;
|
||
|
}
|
||
|
|
||
|
// Reassign the child/parent relationship.
|
||
|
child->parent = parent->parent;
|
||
|
if (parent->parent == &m_nil) {
|
||
|
// The child becomes the new root node.
|
||
|
m_root = child;
|
||
|
}
|
||
|
else {
|
||
|
// Point the grandparent at the child.
|
||
|
if (parent == parent->parent->left) {
|
||
|
parent->parent->left = child;
|
||
|
}
|
||
|
else {
|
||
|
parent->parent->right = child;
|
||
|
}
|
||
|
}
|
||
|
child->right = parent;
|
||
|
parent->parent = child;
|
||
|
}
|
||
|
|
||
|
// Private data members.
|
||
|
node_t *m_freelist; // Pointer to the list of free nodes (reserve storage).
|
||
|
mutable CRITICAL_SECTION m_lock; // Protects the tree's integrity against concurrent accesses.
|
||
|
node_t m_nil; // The tree's nil node. All leaf nodes point to this.
|
||
|
UINT32 m_reserve; // The size (in nodes) of the chunks of reserve storage.
|
||
|
node_t *m_root; // Pointer to the tree's root node.
|
||
|
chunk_t *m_store; // Pointer to the start of the chunk list.
|
||
|
chunk_t *m_storetail; // Pointer to the end of the chunk list.
|
||
|
};
|