boost/signals2/detail/stack_allocator.hpp
/* An allocator which first allocates from the stack, before falling back on usual std::allocator behavior. Used by signals2 to optimize the vector of tracked shared_ptr created during signal invocation. Example usage: static const std::size_t n = 10; stack_storage<T, n> storage; stack_allocator<T, n> a(&storage); std::vector<T, stack_allocator<T, n> > v(a); */ // Copyright Frank Mori Hess 2008. // 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/signals2 for library home page. #ifndef BOOST_SIGNALS2_STACK_ALLOCATOR_HPP #define BOOST_SIGNALS2_STACK_ALLOCATOR_HPP #include <memory> #include <boost/noncopyable.hpp> #include <boost/type_traits/aligned_storage.hpp> #include <boost/type_traits/alignment_of.hpp> namespace boost { namespace signals2 { namespace detail { template<typename T, std::size_t n_stack_elements> class stack_storage: public boost::noncopyable { public: typedef typename boost::aligned_storage<sizeof(T), boost::alignment_of<T>::value>::type storage_type; stack_storage(): is_reserved(false) { } storage_type array[n_stack_elements]; bool is_reserved; }; template<typename T, std::size_t n_stack_elements> class stack_allocator: public std::allocator<T> { typedef std::allocator<T> base_class; public: template<typename U> struct rebind { typedef stack_allocator<U, n_stack_elements> other; }; stack_allocator(stack_storage<T, n_stack_elements> *storage = 0): _storage(storage) { } typename base_class::pointer allocate(typename base_class::size_type n_elements, std::allocator<void>::const_pointer hint = 0) { if(_storage && _storage->is_reserved == false && n_elements <= n_stack_elements) { _storage->is_reserved = true; return reinterpret_cast<typename base_class::pointer>(&_storage->array[0]); } return base_class::allocate(n_elements, hint); } void deallocate(typename base_class::pointer p, typename base_class::size_type n) { if(_storage && p == reinterpret_cast<typename base_class::pointer>(&_storage->array[0])) { _storage->is_reserved = false; }else { base_class::deallocate(p, n); } } private: stack_storage<T, n_stack_elements> *_storage; }; } // namespace detail } // namespace signals2 } // namespace boost #endif // BOOST_SIGNALS2_STACK_ALLOCATOR_HPP