//===- BinTree.h ----------------------------------------------------------===//
//
// The MCLinker Project
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#ifndef MCLD_ADT_BINTREE_H_
#define MCLD_ADT_BINTREE_H_
#include "mcld/ADT/TreeAllocator.h"
#include "mcld/ADT/TreeBase.h"
#include "mcld/Support/Compiler.h"
#include <cstddef>
#include <iterator>
#include <memory>
#include <queue>
#include <stack>
namespace mcld {
template <class DataType>
class BinaryTree;
class DFSIterator : public TreeIteratorBase {
public:
DFSIterator() : TreeIteratorBase() {}
explicit DFSIterator(NodeBase* X) : TreeIteratorBase(X) {
if (hasRightChild())
m_Stack.push(m_pNode->right);
if (hasLeftChild())
m_Stack.push(m_pNode->left);
}
virtual ~DFSIterator() {}
void advance() {
if (m_Stack.empty()) { // reach the end
m_pNode = m_pNode->right; // should be root
return;
}
m_pNode = m_Stack.top();
m_Stack.pop();
if (hasRightChild())
m_Stack.push(m_pNode->right);
if (hasLeftChild())
m_Stack.push(m_pNode->left);
}
private:
std::stack<NodeBase*> m_Stack;
};
class BFSIterator : public TreeIteratorBase {
public:
BFSIterator() : TreeIteratorBase() {}
explicit BFSIterator(NodeBase* X) : TreeIteratorBase(X) {
if (hasRightChild())
m_Queue.push(m_pNode->right);
if (hasLeftChild())
m_Queue.push(m_pNode->left);
}
virtual ~BFSIterator() {}
void advance() {
if (m_Queue.empty()) { // reach the end
m_pNode = m_pNode->right; // should be root
return;
}
m_pNode = m_Queue.front();
m_Queue.pop();
if (hasRightChild())
m_Queue.push(m_pNode->right);
if (hasLeftChild())
m_Queue.push(m_pNode->left);
}
private:
std::queue<NodeBase*> m_Queue;
};
template <class DataType, class Traits, class IteratorType>
class PolicyIteratorBase : public IteratorType {
public:
typedef DataType value_type;
typedef Traits traits;
typedef typename traits::pointer pointer;
typedef typename traits::reference reference;
typedef PolicyIteratorBase<value_type, Traits, IteratorType> Self;
typedef Node<value_type> node_type;
typedef typename traits::nonconst_traits nonconst_traits;
typedef typename traits::const_traits const_traits;
typedef PolicyIteratorBase<value_type, nonconst_traits, IteratorType>
iterator;
typedef PolicyIteratorBase<value_type, const_traits, IteratorType>
const_iterator;
typedef std::forward_iterator_tag iterator_category;
typedef size_t size_type;
typedef ptrdiff_t difference_type;
public:
PolicyIteratorBase() : IteratorType() {}
PolicyIteratorBase(const iterator& X) : IteratorType(X.m_pNode) {}
explicit PolicyIteratorBase(NodeBase* X) : IteratorType(X) {}
virtual ~PolicyIteratorBase() {}
// ----- operators ----- //
pointer operator*() const {
return static_cast<node_type*>(IteratorType::m_pNode)->data;
}
reference operator->() const {
return *static_cast<node_type*>(IteratorType::m_pNode)->data;
}
bool hasData() const {
return (!IteratorType::isRoot() &&
(0 != static_cast<node_type*>(IteratorType::m_pNode)->data));
}
};
template <class DataType, class Traits, class IteratorType>
class PolicyIterator
: public PolicyIteratorBase<DataType, Traits, IteratorType> {
public:
typedef PolicyIterator<DataType, Traits, IteratorType> Self;
typedef PolicyIteratorBase<DataType, Traits, IteratorType> Base;
typedef PolicyIterator<DataType,
typename Traits::nonconst_traits,
IteratorType> iterator;
typedef PolicyIterator<DataType, typename Traits::const_traits, IteratorType>
const_iterator;
public:
PolicyIterator() : Base() {}
PolicyIterator(const iterator& X) : Base(X.m_pNode) {}
explicit PolicyIterator(NodeBase* X) : Base(X) {}
virtual ~PolicyIterator() {}
Self& operator++() {
IteratorType::advance();
return *this;
}
Self operator++(int) {
Self tmp = *this;
IteratorType::advance();
return tmp;
}
};
template <class DataType>
class BinaryTree;
/** \class TreeIterator
* \brief TreeIterator provides full functions of binary tree's iterator.
*
* TreeIterator is designed to compatible with STL iterators.
* TreeIterator is bi-directional. Incremental direction means to move
* rightward, and decremental direction is leftward.
*
* @see TreeIteratorBase
*/
template <class DataType, class Traits>
struct TreeIterator : public TreeIteratorBase {
public:
typedef DataType value_type;
typedef Traits traits;
typedef typename traits::pointer pointer;
typedef typename traits::reference reference;
typedef TreeIterator<value_type, Traits> Self;
typedef Node<value_type> node_type;
typedef typename traits::nonconst_traits nonconst_traits;
typedef TreeIterator<value_type, nonconst_traits> iterator;
typedef typename traits::const_traits const_traits;
typedef TreeIterator<value_type, const_traits> const_iterator;
typedef std::bidirectional_iterator_tag iterator_category;
typedef size_t size_type;
typedef ptrdiff_t difference_type;
public:
TreeIterator() : TreeIteratorBase() {}
TreeIterator(const iterator& X) : TreeIteratorBase(X.m_pNode) {}
~TreeIterator() {}
// ----- operators ----- //
pointer operator*() const { return static_cast<node_type*>(m_pNode)->data; }
reference operator->() const {
return *static_cast<node_type*>(m_pNode)->data;
}
bool isRoot() const { return (m_pNode->right == m_pNode); }
bool hasData() const {
return (!isRoot() && (0 != static_cast<node_type*>(m_pNode)->data));
}
Self& operator++() {
this->move<TreeIteratorBase::Rightward>();
return *this;
}
Self operator++(int) {
Self tmp = *this;
this->move<TreeIteratorBase::Rightward>();
return tmp;
}
Self& operator--() {
this->move<TreeIteratorBase::Leftward>();
return *this;
}
Self operator--(int) {
Self tmp = *this;
this->move<TreeIteratorBase::Leftward>();
return tmp;
}
explicit TreeIterator(NodeBase* X) : TreeIteratorBase(X) {}
};
/** \class BinaryTreeBase
* \brief BinaryTreeBase gives root node and memory management.
*
* The memory management of nodes in is hidden by BinaryTreeBase.
* BinaryTreeBase also provides the basic functions for merging a tree and
* inserton of a node.
*
* @see BinaryTree
*/
template <class DataType>
class BinaryTreeBase {
public:
typedef Node<DataType> NodeType;
protected:
/// TreeImpl - TreeImpl records the root node and the number of nodes
//
// +---> Root(end) <---+
// | |left |
// | begin |
// | / \ |
// | Left Right |
// +---/ \-----+
//
class TreeImpl : public NodeFactory<DataType> {
typedef typename NodeFactory<DataType>::iterator iterator;
typedef typename NodeFactory<DataType>::const_iterator const_iterator;
public:
NodeBase node;
public:
TreeImpl() : NodeFactory<DataType>() { node.left = node.right = &node; }
~TreeImpl() {}
/// summon - change the final edges of pClient to our root
void summon(TreeImpl& pClient) {
if (this == &pClient)
return;
iterator data;
iterator dEnd = pClient.end();
for (data = pClient.begin(); data != dEnd; ++data) {
if ((*data).left == &pClient.node)
(*data).left = &node;
if ((*data).right == &pClient.node)
(*data).right = &node;
}
}
}; // TreeImpl
protected:
/// m_Root is a special object who responses:
// - the pointer of root
// - the simple factory of nodes.
TreeImpl m_Root;
protected:
NodeType* createNode() {
NodeType* result = m_Root.produce();
result->left = result->right = &m_Root.node;
return result;
}
void destroyNode(NodeType* pNode) {
pNode->left = pNode->right = 0;
pNode->data = 0;
m_Root.deallocate(pNode);
}
public:
BinaryTreeBase() : m_Root() {}
virtual ~BinaryTreeBase() {}
size_t size() const { return m_Root.size(); }
bool empty() const { return m_Root.empty(); }
protected:
void clear() { m_Root.clear(); }
private:
DISALLOW_COPY_AND_ASSIGN(BinaryTreeBase);
};
/** \class BinaryTree
* \brief An abstract data type of binary tree.
*
* @see mcld::InputTree
*/
template <class DataType>
class BinaryTree : public BinaryTreeBase<DataType> {
public:
typedef size_t size_type;
typedef ptrdiff_t difference_type;
typedef DataType value_type;
typedef value_type* pointer;
typedef value_type& reference;
typedef const value_type* const_pointer;
typedef const value_type& const_reference;
typedef BinaryTree<DataType> Self;
typedef TreeIterator<value_type, NonConstTraits<value_type> > iterator;
typedef TreeIterator<value_type, ConstTraits<value_type> > const_iterator;
typedef PolicyIterator<value_type, NonConstTraits<value_type>, DFSIterator>
dfs_iterator;
typedef PolicyIterator<value_type, ConstTraits<value_type>, DFSIterator>
const_dfs_iterator;
typedef PolicyIterator<value_type, NonConstTraits<value_type>, BFSIterator>
bfs_iterator;
typedef PolicyIterator<value_type, ConstTraits<value_type>, BFSIterator>
const_bfs_iterator;
protected:
typedef Node<value_type> node_type;
public:
// ----- constructors and destructor ----- //
BinaryTree() : BinaryTreeBase<DataType>() {}
~BinaryTree() {}
// ----- iterators ----- //
bfs_iterator bfs_begin() {
return bfs_iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
bfs_iterator bfs_end() {
return bfs_iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
const_bfs_iterator bfs_begin() const {
return const_bfs_iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
const_bfs_iterator bfs_end() const {
return const_bfs_iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
dfs_iterator dfs_begin() {
return dfs_iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
dfs_iterator dfs_end() {
return dfs_iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
const_dfs_iterator dfs_begin() const {
return const_dfs_iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
const_dfs_iterator dfs_end() const {
return const_dfs_iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
iterator root() { return iterator(&(BinaryTreeBase<DataType>::m_Root.node)); }
const_iterator root() const {
return const_iterator(&(BinaryTreeBase<DataType>::m_Root.node));
}
iterator begin() {
return iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
iterator end() {
return iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
const_iterator begin() const {
return const_iterator(BinaryTreeBase<DataType>::m_Root.node.left);
}
const_iterator end() const {
return const_iterator(BinaryTreeBase<DataType>::m_Root.node.right);
}
// ----- modifiers ----- //
/// join - create a leaf node and merge it in the tree.
// This version of join determines the direction on compilation time.
// @param DIRECT the direction of the connecting edge of the parent node.
// @param position the parent node
// @param value the value being pushed.
template <size_t DIRECT>
BinaryTree& join(TreeIteratorBase& pPosition, const DataType& pValue) {
node_type* node = BinaryTreeBase<DataType>::createNode();
node->data = const_cast<DataType*>(&pValue);
if (pPosition.isRoot())
pPosition.hook<TreeIteratorBase::Leftward>(node);
else
pPosition.hook<DIRECT>(node);
return *this;
}
/// merge - merge the tree
// @param DIRECT the direction of the connecting edge of the parent node.
// @param position the parent node
// @param the tree being joined.
// @return the joined tree
template <size_t DIRECT>
BinaryTree& merge(TreeIteratorBase& pPosition, BinaryTree& pTree) {
if (this == &pTree)
return *this;
if (!pTree.empty()) {
pPosition.hook<DIRECT>(pTree.m_Root.node.left);
BinaryTreeBase<DataType>::m_Root.summon(
pTree.BinaryTreeBase<DataType>::m_Root);
BinaryTreeBase<DataType>::m_Root.delegate(pTree.m_Root);
pTree.m_Root.node.left = pTree.m_Root.node.right = &pTree.m_Root.node;
}
return *this;
}
};
} // namespace mcld
#endif // MCLD_ADT_BINTREE_H_