190 lines
6.4 KiB
C++
190 lines
6.4 KiB
C++
|
//////////////////////////////////////////////////////////////////////////////
|
||
|
//
|
||
|
// (C) Copyright Ion Gaztanaga 2007-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.
|
||
|
//
|
||
|
//////////////////////////////////////////////////////////////////////////////
|
||
|
|
||
|
#ifdef _MSC_VER
|
||
|
#pragma warning (disable : 4512)
|
||
|
#endif
|
||
|
|
||
|
#include <boost/container/detail/dlmalloc.hpp>
|
||
|
#include <boost/container/throw_exception.hpp>
|
||
|
|
||
|
#define BOOST_INTERPROCESS_VECTOR_ALLOC_STATS
|
||
|
|
||
|
#include <iostream> //std::cout, std::endl
|
||
|
#include <typeinfo> //typeid
|
||
|
#include <cassert> //assert
|
||
|
|
||
|
#include <boost/move/detail/nsec_clock.hpp>
|
||
|
using boost::move_detail::cpu_timer;
|
||
|
using boost::move_detail::cpu_times;
|
||
|
using boost::move_detail::nanosecond_type;
|
||
|
|
||
|
using namespace boost::container;
|
||
|
|
||
|
template <class POD>
|
||
|
void allocation_timing_test(std::size_t num_iterations, std::size_t num_elements)
|
||
|
{
|
||
|
size_t capacity = 0;
|
||
|
std::size_t numalloc = 0, numexpand = 0;
|
||
|
|
||
|
std::cout
|
||
|
<< " ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ \n"
|
||
|
<< " Iterations/Elements: " << num_iterations << "/" << num_elements << '\n'
|
||
|
<< " ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ \n"
|
||
|
<< std::endl;
|
||
|
|
||
|
|
||
|
allocation_type malloc_types[] = { BOOST_CONTAINER_EXPAND_BWD, BOOST_CONTAINER_EXPAND_FWD, BOOST_CONTAINER_ALLOCATE_NEW };
|
||
|
const char * malloc_names[] = { "Backwards expansion", "Forward expansion", "New allocation" };
|
||
|
for(size_t i = 0; i < sizeof(malloc_types)/sizeof(allocation_type); ++i){
|
||
|
numalloc = 0; numexpand = 0;
|
||
|
const allocation_type m_mode = malloc_types[i];
|
||
|
const char *malloc_name = malloc_names[i];
|
||
|
|
||
|
cpu_timer timer;
|
||
|
timer.resume();
|
||
|
|
||
|
for(std::size_t r = 0; r != num_iterations; ++r){
|
||
|
void *first_mem = 0;
|
||
|
if(m_mode != BOOST_CONTAINER_EXPAND_FWD)
|
||
|
first_mem = dlmalloc_malloc(sizeof(POD)*num_elements*3/2);
|
||
|
void *addr = dlmalloc_malloc(1*sizeof(POD));
|
||
|
if(m_mode == BOOST_CONTAINER_EXPAND_FWD)
|
||
|
first_mem = dlmalloc_malloc(sizeof(POD)*num_elements*3/2);
|
||
|
capacity = dlmalloc_size(addr)/sizeof(POD);
|
||
|
dlmalloc_free(first_mem);
|
||
|
++numalloc;
|
||
|
|
||
|
BOOST_CONTAINER_TRY{
|
||
|
dlmalloc_command_ret_t ret;
|
||
|
for(size_t e = capacity + 1; e < num_elements; ++e){
|
||
|
size_t received_size;
|
||
|
size_t min = (capacity+1)*sizeof(POD);
|
||
|
size_t max = (capacity*3/2)*sizeof(POD);
|
||
|
if(min > max)
|
||
|
max = min;
|
||
|
ret = dlmalloc_allocation_command
|
||
|
( m_mode, sizeof(POD)
|
||
|
, min, max, &received_size, addr);
|
||
|
if(!ret.first){
|
||
|
throw_runtime_error("!ret.first)");
|
||
|
}
|
||
|
if(!ret.second){
|
||
|
assert(m_mode == BOOST_CONTAINER_ALLOCATE_NEW);
|
||
|
if(m_mode != BOOST_CONTAINER_ALLOCATE_NEW){
|
||
|
std::cout << "m_mode != BOOST_CONTAINER_ALLOCATE_NEW!" << std::endl;
|
||
|
return;
|
||
|
}
|
||
|
dlmalloc_free(addr);
|
||
|
addr = ret.first;
|
||
|
++numalloc;
|
||
|
}
|
||
|
else{
|
||
|
assert(m_mode != BOOST_CONTAINER_ALLOCATE_NEW);
|
||
|
if(m_mode == BOOST_CONTAINER_ALLOCATE_NEW){
|
||
|
std::cout << "m_mode == BOOST_CONTAINER_ALLOCATE_NEW!" << std::endl;
|
||
|
return;
|
||
|
}
|
||
|
++numexpand;
|
||
|
}
|
||
|
capacity = received_size/sizeof(POD);
|
||
|
addr = ret.first;
|
||
|
e = capacity + 1;
|
||
|
}
|
||
|
dlmalloc_free(addr);
|
||
|
}
|
||
|
BOOST_CONTAINER_CATCH(...){
|
||
|
dlmalloc_free(addr);
|
||
|
BOOST_CONTAINER_RETHROW;
|
||
|
}
|
||
|
BOOST_CONTAINER_CATCH_END
|
||
|
}
|
||
|
|
||
|
assert( dlmalloc_allocated_memory() == 0);
|
||
|
if(dlmalloc_allocated_memory()!= 0){
|
||
|
std::cout << "Memory leak!" << std::endl;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
timer.stop();
|
||
|
nanosecond_type nseconds = timer.elapsed().wall;
|
||
|
|
||
|
std::cout << " Malloc type: " << malloc_name
|
||
|
<< std::endl
|
||
|
<< " allocation ns: "
|
||
|
<< float(nseconds)/float(num_iterations*num_elements)
|
||
|
<< std::endl
|
||
|
<< " capacity - alloc calls (new/expand): "
|
||
|
<< (std::size_t)capacity << " - "
|
||
|
<< (float(numalloc) + float(numexpand))/float(num_iterations)
|
||
|
<< "(" << float(numalloc)/float(num_iterations) << "/" << float(numexpand)/float(num_iterations) << ")"
|
||
|
<< std::endl << std::endl;
|
||
|
dlmalloc_trim(0);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
template<unsigned N>
|
||
|
struct char_holder
|
||
|
{
|
||
|
char ints_[N];
|
||
|
};
|
||
|
|
||
|
template<class POD>
|
||
|
int allocation_loop()
|
||
|
{
|
||
|
std::cout << std::endl
|
||
|
<< "-------------------------------------------\n"
|
||
|
<< "-------------------------------------------\n"
|
||
|
<< " Type(sizeof): " << typeid(POD).name() << " (" << sizeof(POD) << ")\n"
|
||
|
<< "-------------------------------------------\n"
|
||
|
<< "-------------------------------------------\n"
|
||
|
<< std::endl;
|
||
|
|
||
|
//#define SINGLE_TEST
|
||
|
#define SIMPLE_IT
|
||
|
#ifdef SINGLE_TEST
|
||
|
#ifdef NDEBUG
|
||
|
std::size_t numrep [] = { 50000 };
|
||
|
#else
|
||
|
std::size_t numrep [] = { 5000 };
|
||
|
#endif
|
||
|
std::size_t numele [] = { 100 };
|
||
|
#elif defined(SIMPLE_IT)
|
||
|
std::size_t numrep [] = { 3 };
|
||
|
std::size_t numele [] = { 100 };
|
||
|
#else
|
||
|
#ifdef NDEBUG
|
||
|
std::size_t numrep [] = { /*10000, */10000, 100000, 1000000 };
|
||
|
#else
|
||
|
std::size_t numrep [] = { /*10000, */1000, 10000, 100000 };
|
||
|
#endif
|
||
|
std::size_t numele [] = { /*10000, */1000, 100, 10 };
|
||
|
#endif
|
||
|
|
||
|
for(std::size_t i = 0; i < sizeof(numele)/sizeof(numele[0]); ++i){
|
||
|
allocation_timing_test<POD>(numrep[i], numele[i]);
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int main()
|
||
|
{
|
||
|
dlmalloc_mallopt( (-3)//M_MMAP_THRESHOLD
|
||
|
, 100*10000000);
|
||
|
//allocation_loop<char_holder<4> >();
|
||
|
//allocation_loop<char_holder<6> >();
|
||
|
allocation_loop<char_holder<8> >();
|
||
|
allocation_loop<char_holder<12> >();
|
||
|
//allocation_loop<char_holder<14> >();
|
||
|
allocation_loop<char_holder<24> >();
|
||
|
return 0;
|
||
|
}
|