| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-27 23:38:01 +00:00
										 |  |  | // Copyright 2006-2008 Daniel James.
 | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  | // 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)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if !defined(BOOST_UNORDERED_TEST_MEMORY_HEADER)
 | 
					
						
							|  |  |  | #define BOOST_UNORDERED_TEST_MEMORY_HEADER
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-12 14:43:40 +00:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <map>
 | 
					
						
							|  |  |  | #include <boost/mpl/apply.hpp>
 | 
					
						
							|  |  |  | #include <boost/assert.hpp>
 | 
					
						
							| 
									
										
										
										
											2008-04-20 21:08:57 +00:00
										 |  |  | #include <boost/unordered/detail/allocator_helpers.hpp>
 | 
					
						
							| 
									
										
										
										
											2008-01-18 19:35:55 +00:00
										 |  |  | #include <boost/mpl/aux_/config/eti.hpp>
 | 
					
						
							| 
									
										
										
										
											2008-01-12 14:43:40 +00:00
										 |  |  | #include "../helpers/test.hpp"
 | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace test | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     namespace detail | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         // This annoymous namespace won't cause ODR violations as I won't
 | 
					
						
							|  |  |  |         // be linking multiple translation units together. I'll probably
 | 
					
						
							|  |  |  |         // move this into a cpp file before a full release, but for now it's
 | 
					
						
							|  |  |  |         // the most convenient way.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         struct memory_area { | 
					
						
							|  |  |  |             void const* start; | 
					
						
							|  |  |  |             void const* end; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             memory_area(void const* s, void const* e) | 
					
						
							|  |  |  |                 : start(s), end(e) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 BOOST_ASSERT(start != end); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         struct memory_track { | 
					
						
							|  |  |  |             explicit memory_track(int tag = -1) : | 
					
						
							|  |  |  |                 constructed_(0), | 
					
						
							|  |  |  |                 tag_(tag) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             int constructed_; | 
					
						
							|  |  |  |             int tag_; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // This is a bit dodgy as it defines overlapping
 | 
					
						
							|  |  |  |         // areas as 'equal', so this isn't a total ordering.
 | 
					
						
							|  |  |  |         // But it is for non-overlapping memory regions - which
 | 
					
						
							|  |  |  |         // is what'll be stored.
 | 
					
						
							|  |  |  |         //
 | 
					
						
							|  |  |  |         // All searches will be for areas entirely contained by
 | 
					
						
							|  |  |  |         // a member of the set - so it should find the area that contains
 | 
					
						
							|  |  |  |         // the region that is searched for.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         struct memory_area_compare { | 
					
						
							|  |  |  |             bool operator()(memory_area const& x, memory_area const& y) const { | 
					
						
							|  |  |  |                 return x.end <= y.start; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-18 19:35:55 +00:00
										 |  |  |         template <class Alloc> | 
					
						
							|  |  |  |         struct allocator_memory_type_gen { | 
					
						
							|  |  |  |             typedef std::map<memory_area, memory_track, memory_area_compare, | 
					
						
							|  |  |  |                 Alloc> type; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(BOOST_MPL_CFG_MSVC_ETI_BUG)
 | 
					
						
							|  |  |  |         template <> | 
					
						
							|  |  |  |         struct allocator_memory_type_gen<int> { | 
					
						
							|  |  |  |             typedef std::map<memory_area, memory_track, memory_area_compare> type; | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-01-18 19:35:55 +00:00
										 |  |  |         template <class Alloc = std::allocator<int> > | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |         struct memory_tracker { | 
					
						
							| 
									
										
										
										
											2008-01-18 19:35:55 +00:00
										 |  |  |             typedef BOOST_DEDUCED_TYPENAME | 
					
						
							|  |  |  |                 boost::unordered_detail::rebind_wrap<Alloc, | 
					
						
							|  |  |  |                     std::pair<memory_area const, memory_track> >::type | 
					
						
							|  |  |  |                 allocator_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             typedef BOOST_DEDUCED_TYPENAME allocator_memory_type_gen<allocator_type>::type | 
					
						
							|  |  |  |                 allocated_memory_type; | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             allocated_memory_type allocated_memory; | 
					
						
							|  |  |  |             unsigned int count_allocators; | 
					
						
							|  |  |  |             unsigned int count_allocations; | 
					
						
							|  |  |  |             unsigned int count_constructions; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             memory_tracker() : | 
					
						
							|  |  |  |                 count_allocators(0), count_allocations(0), | 
					
						
							|  |  |  |                 count_constructions(0) | 
					
						
							|  |  |  |             {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             void allocator_ref() | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if(count_allocators == 0) { | 
					
						
							|  |  |  |                     count_allocations = 0; | 
					
						
							|  |  |  |                     count_constructions = 0; | 
					
						
							|  |  |  |                     allocated_memory.clear(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 ++count_allocators; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             void allocator_unref() | 
					
						
							|  |  |  |             { | 
					
						
							| 
									
										
										
										
											2008-04-17 07:39:24 +00:00
										 |  |  |                 BOOST_CHECK(count_allocators > 0); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                 if(count_allocators > 0) { | 
					
						
							|  |  |  |                     --count_allocators; | 
					
						
							|  |  |  |                     if(count_allocators == 0) { | 
					
						
							|  |  |  |                         bool no_allocations_left = (count_allocations == 0); | 
					
						
							|  |  |  |                         bool no_constructions_left = (count_constructions == 0); | 
					
						
							|  |  |  |                         bool allocated_memory_empty = allocated_memory.empty(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                         // Clearing the data before the checks terminate the tests.
 | 
					
						
							|  |  |  |                         count_allocations = 0; | 
					
						
							|  |  |  |                         count_constructions = 0; | 
					
						
							|  |  |  |                         allocated_memory.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-17 07:39:24 +00:00
										 |  |  |                         BOOST_CHECK(no_allocations_left); | 
					
						
							|  |  |  |                         BOOST_CHECK(no_constructions_left); | 
					
						
							|  |  |  |                         BOOST_CHECK(allocated_memory_empty); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             void track_allocate(void *ptr, std::size_t n, std::size_t size, int tag) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 if(n == 0) { | 
					
						
							|  |  |  |                     BOOST_ERROR("Allocating 0 length array."); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 else { | 
					
						
							|  |  |  |                     ++count_allocations; | 
					
						
							| 
									
										
										
										
											2008-01-12 14:43:40 +00:00
										 |  |  |                     allocated_memory.insert( | 
					
						
							|  |  |  |                         std::pair<memory_area const, memory_track>( | 
					
						
							|  |  |  |                             memory_area(ptr, (char*) ptr + n * size), | 
					
						
							|  |  |  |                             memory_track(tag))); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             void track_deallocate(void* ptr, std::size_t n, std::size_t size, int tag) | 
					
						
							|  |  |  |             { | 
					
						
							|  |  |  |                 BOOST_DEDUCED_TYPENAME allocated_memory_type::iterator pos | 
					
						
							|  |  |  |                     = allocated_memory.find(memory_area(ptr, (char*) ptr + n * size)); | 
					
						
							|  |  |  |                 if(pos == allocated_memory.end()) { | 
					
						
							|  |  |  |                     BOOST_ERROR("Deallocating unknown pointer."); | 
					
						
							|  |  |  |                 } else { | 
					
						
							| 
									
										
										
										
											2008-04-17 07:39:24 +00:00
										 |  |  |                     BOOST_CHECK(pos->first.start == ptr); | 
					
						
							|  |  |  |                     BOOST_CHECK(pos->first.end == (char*) ptr + n * size); | 
					
						
							|  |  |  |                     BOOST_CHECK(pos->second.tag_ == tag); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                     allocated_memory.erase(pos); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-04-17 07:39:24 +00:00
										 |  |  |                 BOOST_CHECK(count_allocations > 0); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                 if(count_allocations > 0) --count_allocations; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merge test improvements, pdf documentation improvements, some implementation
tweaks.
Merged revisions 45747-46030 via svnmerge from 
https://svn.boost.org/svn/boost/branches/unordered/trunk
................
  r45747 | danieljames | 2008-05-25 18:03:30 +0100 (Sun, 25 May 2008) | 1 line
  
  Remove /branches/unordered/dev from svnmerge tracking.
................
  r45874 | danieljames | 2008-05-28 18:54:53 +0100 (Wed, 28 May 2008) | 1 line
  
  Merge from trunk.
................
  r45881 | danieljames | 2008-05-29 00:10:37 +0100 (Thu, 29 May 2008) | 2 lines
  
  Make it possible to use a custom generator in more places.
................
  r45882 | danieljames | 2008-05-29 00:10:56 +0100 (Thu, 29 May 2008) | 2 lines
  
  Remove several unused parameters.
................
  r45883 | danieljames | 2008-05-29 00:11:06 +0100 (Thu, 29 May 2008) | 2 lines
  
  No need to run move_construct_tests1 from move_construct_tests2, already calling it directly.
................
  r45884 | danieljames | 2008-05-29 00:11:15 +0100 (Thu, 29 May 2008) | 2 lines
  
  Remove unused 'construct' as in trunk.
................
  r45885 | danieljames | 2008-05-29 00:11:25 +0100 (Thu, 29 May 2008) | 2 lines
  
  More consistent implementations of insert_impl.
................
  r45886 | danieljames | 2008-05-29 00:11:34 +0100 (Thu, 29 May 2008) | 2 lines
  
  No need to use the single argument version of methods when the variadic version is available. Frees up the non-variadic version for overloading to emulate variadic methods.
................
  r45887 | danieljames | 2008-05-29 00:11:44 +0100 (Thu, 29 May 2008) | 2 lines
  
  Add another test, mainly to avoid an unused parameter warning.
................
  r45888 | danieljames | 2008-05-29 00:11:54 +0100 (Thu, 29 May 2008) | 2 lines
  
  Adjust the pixels per inch of the buckets diagram so it'll fit on the page.
................
  r45889 | danieljames | 2008-05-29 00:15:49 +0100 (Thu, 29 May 2008) | 1 line
  
  Improve pdf output, thanks to John Maddock.
................
  r45890 | danieljames | 2008-05-29 00:17:45 +0100 (Thu, 29 May 2008) | 1 line
  
  Add an svg version of the bucket diagram.
................
  r45891 | danieljames | 2008-05-29 00:18:11 +0100 (Thu, 29 May 2008) | 1 line
  
  Remove the original dia buckets diagram, I'm not using it anymore.
................
  r46025 | danieljames | 2008-06-01 18:39:51 +0100 (Sun, 01 Jun 2008) | 1 line
  
  Initialise merging from doc.
................
  r46030 | danieljames | 2008-06-01 18:54:36 +0100 (Sun, 01 Jun 2008) | 29 lines
  
  Merge in documentation improvements from the doc branch.
  
  Merged revisions 45892-46020 via svnmerge from 
  https://svn.boost.org/svn/boost/branches/doc
  
  ........
    r45892 | danieljames | 2008-05-29 00:30:07 +0100 (Thu, 29 May 2008) | 1 line
    
    Merge latest changes from unordered.
  ........
    r46017 | danieljames | 2008-06-01 18:18:15 +0100 (Sun, 01 Jun 2008) | 2 lines
    
    Use both PNG and SVG versions of the buckets diagram.
  ........
    r46018 | danieljames | 2008-06-01 18:18:27 +0100 (Sun, 01 Jun 2008) | 2 lines
    
    Improved function summary tables in unordered documenations.
  ........
    r46019 | danieljames | 2008-06-01 18:18:39 +0100 (Sun, 01 Jun 2008) | 3 lines
    
    I'd hacked the bibliography to avoid showing the title twice, but this was
    causing problems when generating PDFs, so show the title twice.
  ........
    r46020 | danieljames | 2008-06-01 18:18:52 +0100 (Sun, 01 Jun 2008) | 3 lines
    
    Don't use the compact boostbook refernce style in the standalone documentation
    as it is too wide for printing out.
  ........
................
[SVN r46031]
											
										 
											2008-06-01 18:00:53 +00:00
										 |  |  |             void track_construct(void* /*ptr*/, std::size_t /*size*/, int /*tag*/) | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |             { | 
					
						
							|  |  |  |                 ++count_constructions; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												Merge test improvements, pdf documentation improvements, some implementation
tweaks.
Merged revisions 45747-46030 via svnmerge from 
https://svn.boost.org/svn/boost/branches/unordered/trunk
................
  r45747 | danieljames | 2008-05-25 18:03:30 +0100 (Sun, 25 May 2008) | 1 line
  
  Remove /branches/unordered/dev from svnmerge tracking.
................
  r45874 | danieljames | 2008-05-28 18:54:53 +0100 (Wed, 28 May 2008) | 1 line
  
  Merge from trunk.
................
  r45881 | danieljames | 2008-05-29 00:10:37 +0100 (Thu, 29 May 2008) | 2 lines
  
  Make it possible to use a custom generator in more places.
................
  r45882 | danieljames | 2008-05-29 00:10:56 +0100 (Thu, 29 May 2008) | 2 lines
  
  Remove several unused parameters.
................
  r45883 | danieljames | 2008-05-29 00:11:06 +0100 (Thu, 29 May 2008) | 2 lines
  
  No need to run move_construct_tests1 from move_construct_tests2, already calling it directly.
................
  r45884 | danieljames | 2008-05-29 00:11:15 +0100 (Thu, 29 May 2008) | 2 lines
  
  Remove unused 'construct' as in trunk.
................
  r45885 | danieljames | 2008-05-29 00:11:25 +0100 (Thu, 29 May 2008) | 2 lines
  
  More consistent implementations of insert_impl.
................
  r45886 | danieljames | 2008-05-29 00:11:34 +0100 (Thu, 29 May 2008) | 2 lines
  
  No need to use the single argument version of methods when the variadic version is available. Frees up the non-variadic version for overloading to emulate variadic methods.
................
  r45887 | danieljames | 2008-05-29 00:11:44 +0100 (Thu, 29 May 2008) | 2 lines
  
  Add another test, mainly to avoid an unused parameter warning.
................
  r45888 | danieljames | 2008-05-29 00:11:54 +0100 (Thu, 29 May 2008) | 2 lines
  
  Adjust the pixels per inch of the buckets diagram so it'll fit on the page.
................
  r45889 | danieljames | 2008-05-29 00:15:49 +0100 (Thu, 29 May 2008) | 1 line
  
  Improve pdf output, thanks to John Maddock.
................
  r45890 | danieljames | 2008-05-29 00:17:45 +0100 (Thu, 29 May 2008) | 1 line
  
  Add an svg version of the bucket diagram.
................
  r45891 | danieljames | 2008-05-29 00:18:11 +0100 (Thu, 29 May 2008) | 1 line
  
  Remove the original dia buckets diagram, I'm not using it anymore.
................
  r46025 | danieljames | 2008-06-01 18:39:51 +0100 (Sun, 01 Jun 2008) | 1 line
  
  Initialise merging from doc.
................
  r46030 | danieljames | 2008-06-01 18:54:36 +0100 (Sun, 01 Jun 2008) | 29 lines
  
  Merge in documentation improvements from the doc branch.
  
  Merged revisions 45892-46020 via svnmerge from 
  https://svn.boost.org/svn/boost/branches/doc
  
  ........
    r45892 | danieljames | 2008-05-29 00:30:07 +0100 (Thu, 29 May 2008) | 1 line
    
    Merge latest changes from unordered.
  ........
    r46017 | danieljames | 2008-06-01 18:18:15 +0100 (Sun, 01 Jun 2008) | 2 lines
    
    Use both PNG and SVG versions of the buckets diagram.
  ........
    r46018 | danieljames | 2008-06-01 18:18:27 +0100 (Sun, 01 Jun 2008) | 2 lines
    
    Improved function summary tables in unordered documenations.
  ........
    r46019 | danieljames | 2008-06-01 18:18:39 +0100 (Sun, 01 Jun 2008) | 3 lines
    
    I'd hacked the bibliography to avoid showing the title twice, but this was
    causing problems when generating PDFs, so show the title twice.
  ........
    r46020 | danieljames | 2008-06-01 18:18:52 +0100 (Sun, 01 Jun 2008) | 3 lines
    
    Don't use the compact boostbook refernce style in the standalone documentation
    as it is too wide for printing out.
  ........
................
[SVN r46031]
											
										 
											2008-06-01 18:00:53 +00:00
										 |  |  |             void track_destroy(void* /*ptr*/, std::size_t /*size*/, int /*tag*/) | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |             { | 
					
						
							| 
									
										
										
										
											2008-04-17 07:39:24 +00:00
										 |  |  |                 BOOST_CHECK(count_constructions > 0); | 
					
						
							| 
									
										
										
										
											2008-01-10 22:30:46 +00:00
										 |  |  |                 if(count_constructions > 0) --count_constructions; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |