/////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2005-2013. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) // // See http://www.boost.org/libs/container for documentation. // /////////////////////////////////////////////////////////////////////////////// #ifndef BOOST_CONTAINER_EXPAND_BWD_TEST_ALLOCATOR_HPP #define BOOST_CONTAINER_EXPAND_BWD_TEST_ALLOCATOR_HPP #ifndef BOOST_CONFIG_HPP # include #endif #if defined(BOOST_HAS_PRAGMA_ONCE) # pragma once #endif #include #include #include #include #include #include #include #include #include #include #include #include #include namespace boost { namespace container { namespace test { //This allocator just allows two allocations. The first one will return //mp_buffer + m_offset configured in the constructor. The second one //will return mp_buffer. template class expand_bwd_test_allocator { private: typedef expand_bwd_test_allocator self_t; typedef void * aux_pointer_t; typedef const void * cvoid_ptr; template expand_bwd_test_allocator& operator=(const expand_bwd_test_allocator&); expand_bwd_test_allocator& operator=(const expand_bwd_test_allocator&); public: typedef T value_type; typedef T * pointer; typedef const T * const_pointer; typedef typename dtl::add_reference ::type reference; typedef typename dtl::add_reference ::type const_reference; typedef std::size_t size_type; typedef std::ptrdiff_t difference_type; typedef boost::container::dtl::version_type version; //Dummy multiallocation chain struct multiallocation_chain{}; template struct rebind { typedef expand_bwd_test_allocator other; }; //!Constructor from the segment manager. Never throws expand_bwd_test_allocator(T *buffer, size_type sz, difference_type offset) : mp_buffer(buffer), m_size(sz) , m_offset(offset), m_allocations(0){ } //!Constructor from other expand_bwd_test_allocator. Never throws expand_bwd_test_allocator(const expand_bwd_test_allocator &other) : mp_buffer(other.mp_buffer), m_size(other.m_size) , m_offset(other.m_offset), m_allocations(0){ } //!Constructor from related expand_bwd_test_allocator. Never throws template expand_bwd_test_allocator(const expand_bwd_test_allocator &other) : mp_buffer(other.mp_buffer), m_size(other.m_size) , m_offset(other.m_offset), m_allocations(0){ } pointer address(reference value) { return pointer(dtl::addressof(value)); } const_pointer address(const_reference value) const { return const_pointer(dtl::addressof(value)); } pointer allocate(size_type , cvoid_ptr hint = 0) { (void)hint; return 0; } void deallocate(const pointer &, size_type) {} template void construct(pointer ptr, const Convertible &value) { new((void*)ptr) value_type(value); } void destroy(pointer ptr) { (*ptr).~value_type(); } size_type max_size() const { return m_size; } friend void swap(self_t &alloc1, self_t &alloc2) { boost::adl_move_swap(alloc1.mp_buffer, alloc2.mp_buffer); boost::adl_move_swap(alloc1.m_size, alloc2.m_size); boost::adl_move_swap(alloc1.m_offset, alloc2.m_offset); } //Experimental version 2 expand_bwd_test_allocator functions pointer allocation_command(boost::container::allocation_type command, size_type limit_size,size_type &prefer_in_recvd_out_size,pointer &reuse) { (void)reuse; (void)command; //This allocator only expands backwards! assert(m_allocations == 0 || (command & boost::container::expand_bwd)); prefer_in_recvd_out_size = limit_size; if(m_allocations == 0){ if((m_offset + limit_size) > m_size){ assert(0); } ++m_allocations; reuse = 0; return (mp_buffer + m_offset); } else if(m_allocations == 1){ if(limit_size > m_size){ assert(0); } ++m_allocations; return mp_buffer; } else{ throw_bad_alloc(); return mp_buffer; } } //!Returns maximum the number of objects the previously allocated memory //!pointed by p can hold. size_type size(const pointer &p) const { (void)p; return m_size; } //!Allocates just one object. Memory allocated with this function //!must be deallocated only with deallocate_one(). //!Throws boost::container::bad_alloc if there is no enough memory pointer allocate_one() { return this->allocate(1); } //!Deallocates memory previously allocated with allocate_one(). //!You should never use deallocate_one to deallocate memory allocated //!with other functions different from allocate_one(). Never throws void deallocate_one(const pointer &p) { return this->deallocate(p, 1); } pointer mp_buffer; size_type m_size; difference_type m_offset; char m_allocations; }; //!Equality test for same type of expand_bwd_test_allocator template inline bool operator==(const expand_bwd_test_allocator &, const expand_bwd_test_allocator &) { return false; } //!Inequality test for same type of expand_bwd_test_allocator template inline bool operator!=(const expand_bwd_test_allocator &, const expand_bwd_test_allocator &) { return true; } } //namespace test { } //namespace container { } //namespace boost { #include #endif //BOOST_CONTAINER_EXPAND_BWD_TEST_ALLOCATOR_HPP