//===- GCFactory.h --------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef MCLD_SUPPORT_GCFACTORY_H_ #define MCLD_SUPPORT_GCFACTORY_H_ #include "mcld/ADT/TypeTraits.h" #include "mcld/Support/Allocators.h" #include <assert.h> #include <cstddef> #include <iterator> namespace mcld { /** \class DataIteratorBase * \brief DataIteratorBase provides the basic functions of DataIterator * @see DataIterator */ template <typename ChunkType> struct DataIteratorBase { public: ChunkType* m_pChunk; unsigned int m_Pos; public: DataIteratorBase(ChunkType* X, unsigned int pPos) : m_pChunk(X), m_Pos(pPos) {} inline void advance() { ++m_Pos; if ((m_Pos == m_pChunk->bound) && (0 == m_pChunk->next)) return; if (m_Pos == m_pChunk->bound) { m_pChunk = m_pChunk->next; m_Pos = 0; } } bool operator==(const DataIteratorBase& y) const { return ((this->m_pChunk == y.m_pChunk) && (this->m_Pos == y.m_Pos)); } bool operator!=(const DataIteratorBase& y) const { return ((this->m_pChunk != y.m_pChunk) || (this->m_Pos != y.m_Pos)); } }; /** \class DataIterator * \brief DataIterator provides STL compatible iterator for allocators */ template <typename ChunkType, class Traits> class DataIterator : public DataIteratorBase<ChunkType> { public: typedef typename ChunkType::value_type value_type; typedef Traits traits; typedef typename traits::pointer pointer; typedef typename traits::reference reference; typedef DataIterator<ChunkType, Traits> Self; typedef DataIteratorBase<ChunkType> Base; typedef typename traits::nonconst_traits nonconst_traits; typedef DataIterator<ChunkType, nonconst_traits> iterator; typedef typename traits::const_traits const_traits; typedef DataIterator<ChunkType, const_traits> const_iterator; typedef std::forward_iterator_tag iterator_category; typedef size_t size_type; typedef ptrdiff_t difference_type; public: DataIterator() : Base(NULL, 0) {} DataIterator(ChunkType* pChunk, unsigned int pPos) : Base(pChunk, pPos) {} DataIterator(const DataIterator& pCopy) : Base(pCopy.m_pChunk, pCopy.m_Pos) {} ~DataIterator() {} // ----- operators ----- // reference operator*() { assert(this->m_pChunk != NULL && "data iterator goes to a invalid position"); return this->m_pChunk->data[Base::m_Pos]; } Self& operator++() { this->Base::advance(); return *this; } Self operator++(int) { Self tmp = *this; this->Base::advance(); return tmp; } }; template <typename Alloc> class GCFactoryBase : public Alloc { public: typedef DataIterator<typename Alloc::chunk_type, NonConstTraits<typename Alloc::value_type> > iterator; typedef DataIterator<typename Alloc::chunk_type, ConstTraits<typename Alloc::value_type> > const_iterator; typedef typename Alloc::value_type value_type; typedef typename Alloc::pointer pointer; typedef typename Alloc::reference reference; typedef typename Alloc::size_type size_type; protected: GCFactoryBase() : Alloc(), m_NumAllocData(0) {} explicit GCFactoryBase(size_t pNum) : Alloc(pNum), m_NumAllocData(0) {} public: virtual ~GCFactoryBase() { Alloc::clear(); } // ----- modifiers ----- // value_type* allocate(size_t N) { value_type* result = Alloc::allocate(N); if (result != NULL) m_NumAllocData += N; return result; } value_type* allocate() { ++m_NumAllocData; return Alloc::allocate(); } void deallocate(pointer& pPtr, size_type N) { Alloc::deallocate(pPtr, N); if (pPtr == NULL) m_NumAllocData -= N; } void deallocate(pointer& pPtr) { Alloc::deallocate(pPtr); if (pPtr == NULL) --m_NumAllocData; } void reset() { Alloc::reset(); m_NumAllocData = 0; } // ----- iterators ----- // iterator begin() { return iterator(Alloc::m_pRoot, 0); } const_iterator begin() const { return const_iterator(Alloc::m_pRoot, 0); } iterator end() { return (Alloc::m_pCurrent) == 0 ? begin() : iterator(Alloc::m_pCurrent, Alloc::m_pCurrent->bound); } const_iterator end() const { return (Alloc::m_pCurrent) == 0 ? begin() : const_iterator(Alloc::m_pCurrent, Alloc::m_pCurrent->bound); } // ----- observers ----- // bool empty() const { return Alloc::empty(); } unsigned int capacity() const { return Alloc::max_size(); } unsigned int size() const { return m_NumAllocData; } protected: unsigned int m_NumAllocData; }; /** \class GCFactory * \brief GCFactory provides a factory that guaratees to remove all allocated * data. */ template <typename DataType, size_t ChunkSize> class GCFactory : public GCFactoryBase<LinearAllocator<DataType, ChunkSize> > { public: GCFactory() : GCFactoryBase<LinearAllocator<DataType, ChunkSize> >() {} }; template <typename DataType> class GCFactory<DataType, 0> : public GCFactoryBase<LinearAllocator<DataType, 0> > { public: explicit GCFactory(size_t pNum) : GCFactoryBase<LinearAllocator<DataType, 0> >(pNum) {} }; } // namespace mcld #endif // MCLD_SUPPORT_GCFACTORY_H_