/*
Bacula® - The Network Backup Solution
Copyright (C) 2005-2007 Free Software Foundation Europe e.V.
The main author of Bacula is Kern Sibbald, with contributions from
many others, a complete list can be found in the file AUTHORS.
This program is Free Software; you can redistribute it and/or
modify it under the terms of version two of the GNU General Public
License as published by the Free Software Foundation and included
in the file 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-1301, USA.
Bacula® is a registered trademark of John Walker.
The licensor of Bacula is the Free Software Foundation Europe
(FSFE), Fiduciary Program, Sumatrastrasse 25, 8006 Zürich,
Switzerland, email:ftf@fsfeurope.org.
*/
/*
* Version $Id: rblist.h 4992 2007-06-07 14:46:43Z kerns $
*/
/* ========================================================================
*
* red-black binary tree routines -- rblist.h
*
* Kern Sibbald, MMV
*
*/
#define M_ABORT 1
/*
* There is a lot of extra casting here to work around the fact
* that some compilers (Sun and Visual C++) do not accept
* (bnode *) as an lvalue on the left side of an equal.
*
* Loop var through each member of list
*/
#ifdef HAVE_TYPEOF
#define foreach_rblist(var, tree) \
for (((var)=(typeof(var))(tree)->first()); (var); ((var)=(typeof(var))(tree)->next(var)))
#else
#define foreach_rblist(var, tree) \
for ((*((void **)&(var))=(void*)((tree)->first())); (var); (*((void **)&(var))=(void*)((tree)->next(var))))
#endif
struct rblink {
void *parent;
void *left;
void *right;
bool red;
};
class rblist : public SMARTALLOC {
void *head;
int16_t loffset;
uint32_t num_items;
bool down;
void left_rotate(void *item);
void right_rotate(void *item);
public:
rblist(void *item, rblink *link);
rblist(void);
~rblist(void) { destroy(); }
void init(void *item, rblink *link);
void set_parent(void *item, void *parent);
void set_left(void *item, void *left);
void set_right(void *item, void *right);
void set_red(void *item, bool red);
void *parent(const void *item) const;
void *left(const void *item) const;
void *right(const void *item) const;
bool red(const void *item) const;
void *insert(void *item, int compare(void *item1, void *item2));
void *search(void *item, int compare(void *item1, void *item2));
void *first(void);
void *next(void *item);
void *any(void *item);
void remove(void *item);
bool empty(void) const;
int size(void) const;
void destroy(void);
};
inline rblist::rblist(void *item, rblink *link)
{
init(item, link);
}
/* Constructor with link at head of item */
inline rblist::rblist(void): head(0), loffset(0), num_items(0)
{
}
/*
* This allows us to do explicit initialization,
* allowing us to mix C++ classes inside malloc'ed
* C structures. Define before called in constructor.
*/
inline void rblist::init(void *item, rblink *link)
{
head = NULL;
loffset = (int)((char *)link - (char *)item);
if (loffset < 0 || loffset > 5000) {
Emsg0(M_ABORT, 0, "Improper rblist initialization.\n");
}
num_items = 0;
}
inline void rblist::set_parent(void *item, void *parent)
{
((rblink *)(((char *)item)+loffset))->parent = parent;
}
inline void rblist::set_left(void *item, void *left)
{
((rblink *)(((char *)item)+loffset))->left = left;
}
inline void rblist::set_right(void *item, void *right)
{
((rblink *)(((char *)item)+loffset))->right = right;
}
inline void rblist::set_red(void *item, bool red)
{
((rblink *)(((char *)item)+loffset))->red = red;
}
inline bool rblist::empty(void) const
{
return head == NULL;
}
inline int rblist::size() const
{
return num_items;
}
inline void *rblist::parent(const void *item) const
{
return ((rblink *)(((char *)item)+loffset))->parent;
}
inline void *rblist::left(const void *item) const
{
return ((rblink *)(((char *)item)+loffset))->left;
}
inline void *rblist::right(const void *item) const
{
return ((rblink *)(((char *)item)+loffset))->right;
}
inline bool rblist::red(const void *item) const
{
return ((rblink *)(((char *)item)+loffset))->red;
}
syntax highlighted by Code2HTML, v. 0.9.1