200 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
		
		
			
		
	
	
			200 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Plaintext
		
	
	
	
	
	
| 
								 | 
							
								#ifndef BOOST_SMART_PTR_DETAIL_QUICK_ALLOCATOR_HPP_INCLUDED
							 | 
						||
| 
								 | 
							
								#define BOOST_SMART_PTR_DETAIL_QUICK_ALLOCATOR_HPP_INCLUDED
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								// MS compatible compilers support #pragma once
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if defined(_MSC_VER) && (_MSC_VER >= 1020)
							 | 
						||
| 
								 | 
							
								# pragma once
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//  detail/quick_allocator.hpp
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								//  Copyright (c) 2003 David Abrahams
							 | 
						||
| 
								 | 
							
								//  Copyright (c) 2003 Peter Dimov
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								// 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)
							 | 
						||
| 
								 | 
							
								//
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/config.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <boost/smart_ptr/detail/lightweight_mutex.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/type_traits/type_with_alignment.hpp>
							 | 
						||
| 
								 | 
							
								#include <boost/type_traits/alignment_of.hpp>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <new>              // ::operator new, ::operator delete
							 | 
						||
| 
								 | 
							
								#include <cstddef>          // std::size_t
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace boost
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								namespace detail
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_> union freeblock
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    typedef typename boost::type_with_alignment<align_>::type aligner_type;
							 | 
						||
| 
								 | 
							
								    aligner_type aligner;
							 | 
						||
| 
								 | 
							
								    char bytes[size];
							 | 
						||
| 
								 | 
							
								    freeblock * next;
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_> struct allocator_impl
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    typedef freeblock<size, align_> block;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // It may seem odd to use such small pages.
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // However, on a typical Windows implementation that uses
							 | 
						||
| 
								 | 
							
								    // the OS allocator, "normal size" pages interact with the
							 | 
						||
| 
								 | 
							
								    // "ordinary" operator new, slowing it down dramatically.
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // 512 byte pages are handled by the small object allocator,
							 | 
						||
| 
								 | 
							
								    // and don't interfere with ::new.
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // The other alternative is to use much bigger pages (1M.)
							 | 
						||
| 
								 | 
							
								    //
							 | 
						||
| 
								 | 
							
								    // It is surprisingly easy to hit pathological behavior by
							 | 
						||
| 
								 | 
							
								    // varying the page size. g++ 2.96 on Red Hat Linux 7.2,
							 | 
						||
| 
								 | 
							
								    // for example, passionately dislikes 496. 512 seems OK.
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#if defined(BOOST_QA_PAGE_SIZE)
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    enum { items_per_page = BOOST_QA_PAGE_SIZE / size };
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#else
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    enum { items_per_page = 512 / size }; // 1048560 / size
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static lightweight_mutex & mutex()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        static freeblock< sizeof( lightweight_mutex ), boost::alignment_of< lightweight_mutex >::value > fbm;
							 | 
						||
| 
								 | 
							
								        static lightweight_mutex * pm = new( &fbm ) lightweight_mutex;
							 | 
						||
| 
								 | 
							
								        return *pm;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static lightweight_mutex * mutex_init;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static block * free;
							 | 
						||
| 
								 | 
							
								    static block * page;
							 | 
						||
| 
								 | 
							
								    static unsigned last;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static inline void * alloc()
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								        lightweight_mutex::scoped_lock lock( mutex() );
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								        if(block * x = free)
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            free = x->next;
							 | 
						||
| 
								 | 
							
								            return x;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        else
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            if(last == items_per_page)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                // "Listen to me carefully: there is no memory leak"
							 | 
						||
| 
								 | 
							
								                // -- Scott Meyers, Eff C++ 2nd Ed Item 10
							 | 
						||
| 
								 | 
							
								                page = ::new block[items_per_page];
							 | 
						||
| 
								 | 
							
								                last = 0;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return &page[last++];
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static inline void * alloc(std::size_t n)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if(n != size) // class-specific new called for a derived object
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            return ::operator new(n);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        else
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								            lightweight_mutex::scoped_lock lock( mutex() );
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								            if(block * x = free)
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                free = x->next;
							 | 
						||
| 
								 | 
							
								                return x;
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            else
							 | 
						||
| 
								 | 
							
								            {
							 | 
						||
| 
								 | 
							
								                if(last == items_per_page)
							 | 
						||
| 
								 | 
							
								                {
							 | 
						||
| 
								 | 
							
								                    page = ::new block[items_per_page];
							 | 
						||
| 
								 | 
							
								                    last = 0;
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								                return &page[last++];
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static inline void dealloc(void * pv)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if(pv != 0) // 18.4.1.1/13
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								            lightweight_mutex::scoped_lock lock( mutex() );
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								            block * pb = static_cast<block *>(pv);
							 | 
						||
| 
								 | 
							
								            pb->next = free;
							 | 
						||
| 
								 | 
							
								            free = pb;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    static inline void dealloc(void * pv, std::size_t n)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if(n != size) // class-specific delete called for a derived object
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								            ::operator delete(pv);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        else if(pv != 0) // 18.4.1.1/13
							 | 
						||
| 
								 | 
							
								        {
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								            lightweight_mutex::scoped_lock lock( mutex() );
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								            block * pb = static_cast<block *>(pv);
							 | 
						||
| 
								 | 
							
								            pb->next = free;
							 | 
						||
| 
								 | 
							
								            free = pb;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#ifdef BOOST_HAS_THREADS
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_>
							 | 
						||
| 
								 | 
							
								  lightweight_mutex * allocator_impl<size, align_>::mutex_init = &allocator_impl<size, align_>::mutex();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_>
							 | 
						||
| 
								 | 
							
								  freeblock<size, align_> * allocator_impl<size, align_>::free = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_>
							 | 
						||
| 
								 | 
							
								  freeblock<size, align_> * allocator_impl<size, align_>::page = 0;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<unsigned size, unsigned align_>
							 | 
						||
| 
								 | 
							
								  unsigned allocator_impl<size, align_>::last = allocator_impl<size, align_>::items_per_page;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								template<class T>
							 | 
						||
| 
								 | 
							
								struct quick_allocator: public allocator_impl< sizeof(T), boost::alignment_of<T>::value >
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								} // namespace detail
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								} // namespace boost
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#endif  // #ifndef BOOST_SMART_PTR_DETAIL_QUICK_ALLOCATOR_HPP_INCLUDED
							 |