1*404b540aSrobert // array allocator -*- C++ -*- 2*404b540aSrobert 3*404b540aSrobert // Copyright (C) 2004, 2005, 2006 Free Software Foundation, Inc. 4*404b540aSrobert // 5*404b540aSrobert // This file is part of the GNU ISO C++ Library. This library is free 6*404b540aSrobert // software; you can redistribute it and/or modify it under the 7*404b540aSrobert // terms of the GNU General Public License as published by the 8*404b540aSrobert // Free Software Foundation; either version 2, or (at your option) 9*404b540aSrobert // any later version. 10*404b540aSrobert 11*404b540aSrobert // This library is distributed in the hope that it will be useful, 12*404b540aSrobert // but WITHOUT ANY WARRANTY; without even the implied warranty of 13*404b540aSrobert // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 14*404b540aSrobert // GNU General Public License for more details. 15*404b540aSrobert 16*404b540aSrobert // You should have received a copy of the GNU General Public License along 17*404b540aSrobert // with this library; see the file COPYING. If not, write to the Free 18*404b540aSrobert // Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, 19*404b540aSrobert // USA. 20*404b540aSrobert 21*404b540aSrobert // As a special exception, you may use this file as part of a free software 22*404b540aSrobert // library without restriction. Specifically, if other files instantiate 23*404b540aSrobert // templates or use macros or inline functions from this file, or you compile 24*404b540aSrobert // this file and link it with other files to produce an executable, this 25*404b540aSrobert // file does not by itself cause the resulting executable to be covered by 26*404b540aSrobert // the GNU General Public License. This exception does not however 27*404b540aSrobert // invalidate any other reasons why the executable file might be covered by 28*404b540aSrobert // the GNU General Public License. 29*404b540aSrobert 30*404b540aSrobert /** @file ext/array_allocator.h 31*404b540aSrobert * This file is a GNU extension to the Standard C++ Library. 32*404b540aSrobert */ 33*404b540aSrobert 34*404b540aSrobert #ifndef _ARRAY_ALLOCATOR_H 35*404b540aSrobert #define _ARRAY_ALLOCATOR_H 1 36*404b540aSrobert 37*404b540aSrobert #include <cstddef> 38*404b540aSrobert #include <new> 39*404b540aSrobert #include <bits/functexcept.h> 40*404b540aSrobert #include <tr1/array> 41*404b540aSrobert 42*404b540aSrobert _GLIBCXX_BEGIN_NAMESPACE(__gnu_cxx) 43*404b540aSrobert 44*404b540aSrobert using std::size_t; 45*404b540aSrobert using std::ptrdiff_t; 46*404b540aSrobert 47*404b540aSrobert /// @brief Base class. 48*404b540aSrobert template<typename _Tp> 49*404b540aSrobert class array_allocator_base 50*404b540aSrobert { 51*404b540aSrobert public: 52*404b540aSrobert typedef size_t size_type; 53*404b540aSrobert typedef ptrdiff_t difference_type; 54*404b540aSrobert typedef _Tp* pointer; 55*404b540aSrobert typedef const _Tp* const_pointer; 56*404b540aSrobert typedef _Tp& reference; 57*404b540aSrobert typedef const _Tp& const_reference; 58*404b540aSrobert typedef _Tp value_type; 59*404b540aSrobert 60*404b540aSrobert pointer address(reference __x)61*404b540aSrobert address(reference __x) const { return &__x; } 62*404b540aSrobert 63*404b540aSrobert const_pointer address(const_reference __x)64*404b540aSrobert address(const_reference __x) const { return &__x; } 65*404b540aSrobert 66*404b540aSrobert void deallocate(pointer,size_type)67*404b540aSrobert deallocate(pointer, size_type) 68*404b540aSrobert { 69*404b540aSrobert // Does nothing. 70*404b540aSrobert } 71*404b540aSrobert 72*404b540aSrobert size_type max_size()73*404b540aSrobert max_size() const throw() 74*404b540aSrobert { return size_t(-1) / sizeof(_Tp); } 75*404b540aSrobert 76*404b540aSrobert // _GLIBCXX_RESOLVE_LIB_DEFECTS 77*404b540aSrobert // 402. wrong new expression in [some_] allocator::construct 78*404b540aSrobert void construct(pointer __p,const _Tp & __val)79*404b540aSrobert construct(pointer __p, const _Tp& __val) 80*404b540aSrobert { ::new(__p) value_type(__val); } 81*404b540aSrobert 82*404b540aSrobert void destroy(pointer __p)83*404b540aSrobert destroy(pointer __p) { __p->~_Tp(); } 84*404b540aSrobert }; 85*404b540aSrobert 86*404b540aSrobert /** 87*404b540aSrobert * @brief An allocator that uses previously allocated memory. 88*404b540aSrobert * This memory can be externally, globally, or otherwise allocated. 89*404b540aSrobert */ 90*404b540aSrobert template<typename _Tp, typename _Array = std::tr1::array<_Tp, 1> > 91*404b540aSrobert class array_allocator : public array_allocator_base<_Tp> 92*404b540aSrobert { 93*404b540aSrobert public: 94*404b540aSrobert typedef size_t size_type; 95*404b540aSrobert typedef ptrdiff_t difference_type; 96*404b540aSrobert typedef _Tp* pointer; 97*404b540aSrobert typedef const _Tp* const_pointer; 98*404b540aSrobert typedef _Tp& reference; 99*404b540aSrobert typedef const _Tp& const_reference; 100*404b540aSrobert typedef _Tp value_type; 101*404b540aSrobert typedef _Array array_type; 102*404b540aSrobert 103*404b540aSrobert private: 104*404b540aSrobert array_type* _M_array; 105*404b540aSrobert size_type _M_used; 106*404b540aSrobert 107*404b540aSrobert public: 108*404b540aSrobert template<typename _Tp1, typename _Array1 = _Array> 109*404b540aSrobert struct rebind 110*404b540aSrobert { typedef array_allocator<_Tp1, _Array1> other; }; 111*404b540aSrobert throw()112*404b540aSrobert array_allocator(array_type* __array = NULL) throw() 113*404b540aSrobert : _M_array(__array), _M_used(size_type()) { } 114*404b540aSrobert throw()115*404b540aSrobert array_allocator(const array_allocator& __o) throw() 116*404b540aSrobert : _M_array(__o._M_array), _M_used(__o._M_used) { } 117*404b540aSrobert 118*404b540aSrobert template<typename _Tp1, typename _Array1> array_allocator(const array_allocator<_Tp1,_Array1> &)119*404b540aSrobert array_allocator(const array_allocator<_Tp1, _Array1>&) throw() 120*404b540aSrobert : _M_array(NULL), _M_used(size_type()) { } 121*404b540aSrobert throw()122*404b540aSrobert ~array_allocator() throw() { } 123*404b540aSrobert 124*404b540aSrobert pointer 125*404b540aSrobert allocate(size_type __n, const void* = 0) 126*404b540aSrobert { 127*404b540aSrobert if (_M_array == 0 || _M_used + __n > _M_array->size()) 128*404b540aSrobert std::__throw_bad_alloc(); 129*404b540aSrobert pointer __ret = _M_array->begin() + _M_used; 130*404b540aSrobert _M_used += __n; 131*404b540aSrobert return __ret; 132*404b540aSrobert } 133*404b540aSrobert }; 134*404b540aSrobert 135*404b540aSrobert template<typename _Tp, typename _Array> 136*404b540aSrobert inline bool 137*404b540aSrobert operator==(const array_allocator<_Tp, _Array>&, 138*404b540aSrobert const array_allocator<_Tp, _Array>&) 139*404b540aSrobert { return true; } 140*404b540aSrobert 141*404b540aSrobert template<typename _Tp, typename _Array> 142*404b540aSrobert inline bool 143*404b540aSrobert operator!=(const array_allocator<_Tp, _Array>&, 144*404b540aSrobert const array_allocator<_Tp, _Array>&) 145*404b540aSrobert { return false; } 146*404b540aSrobert 147*404b540aSrobert _GLIBCXX_END_NAMESPACE 148*404b540aSrobert 149*404b540aSrobert #endif 150