/////////////////////////////////////////////////////////////////////////////// // // (C) Copyright Ion Gaztanaga 2005-2012. 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/interprocess for documentation. // /////////////////////////////////////////////////////////////////////////////// #ifndef BOOST_INTERPROCESS_EXPAND_BWD_TEST_ALLOCATOR_HPP #define BOOST_INTERPROCESS_EXPAND_BWD_TEST_ALLOCATOR_HPP #if defined (_MSC_VER) # pragma once #endif #include #include #include #include #include #include #include #include #include #include #include #include #include //!\file //!Describes an allocator to test expand capabilities namespace boost { namespace interprocess { 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 ipcdetail::add_reference ::type reference; typedef typename ipcdetail::add_reference ::type const_reference; typedef std::size_t size_type; typedef std::ptrdiff_t difference_type; typedef boost::interprocess::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 *buf, size_type sz, difference_type offset) : mp_buffer(buf), 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(addressof(value)); } const_pointer address(const_reference value) const { return const_pointer(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::interprocess::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::interprocess::expand_bwd)); prefer_in_recvd_out_size = limit_size; if(m_allocations == 0){ if((m_offset + limit_size) > m_size){ assert(0); } ++m_allocations; return mp_buffer + m_offset; } else if(m_allocations == 1){ if(limit_size > m_size){ assert(0); } ++m_allocations; return mp_buffer; } else{ assert(0); throw std::bad_alloc(); } } //!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::interprocess::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 interprocess { } //namespace boost { #include #endif //BOOST_INTERPROCESS_EXPAND_BWD_TEST_ALLOCATOR_HPP