Files
boost_smart_ptr/test/shared_ptr_mt_test.cpp

182 lines
3.4 KiB
C++
Raw Normal View History

2003-11-28 15:35:21 +00:00
#include <boost/config.hpp>
#if defined(BOOST_MSVC)
2002-02-16 16:09:08 +00:00
#pragma warning(disable: 4786) // identifier truncated in debug info
#pragma warning(disable: 4710) // function not inlined
#pragma warning(disable: 4711) // function selected for automatic inline expansion
#pragma warning(disable: 4514) // unreferenced inline removed
#endif
//
// shared_ptr_mt_test.cpp - tests shared_ptr with multiple threads
//
// Copyright (c) 2002 Peter Dimov and Multi Media Ltd.
//
// 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)
2002-02-16 16:09:08 +00:00
//
#include <boost/shared_ptr.hpp>
#include <boost/bind.hpp>
#include <vector>
#include <memory>
#include <stdexcept>
#include <cstdio>
2002-02-18 12:39:32 +00:00
#include <ctime>
2002-02-16 16:09:08 +00:00
// 'portable' thread framework
class abstract_thread
{
public:
virtual ~abstract_thread() {}
virtual void run() = 0;
};
#if !defined(BOOST_HAS_PTHREADS) && defined(BOOST_HAS_WINTHREADS)
2002-02-18 12:39:32 +00:00
char const * title = "Using Windows threads";
2002-02-16 16:09:08 +00:00
#include <windows.h>
#include <process.h>
typedef HANDLE pthread_t;
unsigned __stdcall common_thread_routine(void * pv)
{
abstract_thread * pt = static_cast<abstract_thread *>(pv);
pt->run();
delete pt;
return 0;
}
int pthread_create(pthread_t * thread, void const *, unsigned (__stdcall * start_routine) (void*), void* arg)
{
HANDLE h = (HANDLE)_beginthreadex(0, 0, start_routine, arg, 0, 0);
if(h != 0)
{
*thread = h;
return 0;
}
else
{
return 1; // return errno;
}
}
int pthread_join(pthread_t thread, void ** /*value_ptr*/)
{
::WaitForSingleObject(thread, INFINITE);
::CloseHandle(thread);
return 0;
}
#else
2002-02-18 12:39:32 +00:00
char const * title = "Using POSIX threads";
2002-02-16 16:09:08 +00:00
#include <pthread.h>
2002-03-15 22:03:56 +00:00
extern "C" void * common_thread_routine(void * pv)
2002-02-16 16:09:08 +00:00
{
abstract_thread * pt = static_cast<abstract_thread *>(pv);
pt->run();
delete pt;
return 0;
}
#endif
//
template<class F> class thread: public abstract_thread
{
public:
explicit thread(F f): f_(f)
{
}
void run()
{
f_();
}
private:
F f_;
};
template<class F> pthread_t createThread(F f)
{
std::auto_ptr<abstract_thread> p(new thread<F>(f));
pthread_t r;
if(pthread_create(&r, 0, common_thread_routine, p.get()) == 0)
{
p.release();
return r;
}
throw std::runtime_error("createThread failed.");
}
//
int const n = 1024 * 1024;
void test(boost::shared_ptr<int> const & pi)
{
std::vector< boost::shared_ptr<int> > v;
for(int i = 0; i < n; ++i)
{
v.push_back(pi);
}
}
int const m = 16; // threads
2002-03-15 22:03:56 +00:00
#if defined(BOOST_LWM_USE_CRITICAL_SECTION)
char const * implementation = "critical section";
#elif defined(BOOST_LWM_USE_PTHREADS)
char const * implementation = "pthread_mutex";
#else
char const * implementation = "spinlock";
#endif
int main()
2002-02-16 16:09:08 +00:00
{
using namespace std; // printf, clock_t, clock
printf("%s: %s, %d threads, %d iterations: ", title, implementation, m, n);
2002-02-16 16:09:08 +00:00
boost::shared_ptr<int> pi(new int(42));
2002-02-16 16:09:08 +00:00
clock_t t = clock();
2002-02-18 12:39:32 +00:00
2002-02-16 16:09:08 +00:00
pthread_t a[m];
for(int i = 0; i < m; ++i)
{
a[i] = createThread( boost::bind(test, pi) );
}
for(int j = 0; j < m; ++j)
2002-02-16 16:09:08 +00:00
{
pthread_join(a[j], 0);
2002-02-16 16:09:08 +00:00
}
t = clock() - t;
2002-02-18 12:39:32 +00:00
printf("\n\n%.3f seconds.\n", static_cast<double>(t) / CLOCKS_PER_SEC);
2002-02-18 12:39:32 +00:00
return 0;
2002-02-16 16:09:08 +00:00
}