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.
|
|
|
|
//
|
|
|
|
// Permission to copy, use, modify, sell and distribute this software
|
|
|
|
// is granted provided this copyright notice appears in all copies.
|
|
|
|
// This software is provided "as is" without express or implied
|
|
|
|
// warranty, and with no claim as to its suitability for any purpose.
|
|
|
|
//
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
2002-08-16 16:41:16 +00:00
|
|
|
int main()
|
2002-02-16 16:09:08 +00:00
|
|
|
{
|
2003-02-08 16:05:46 +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
|
|
|
|
2002-02-16 18:45:20 +00:00
|
|
|
boost::shared_ptr<int> pi(new int(42));
|
2002-02-16 16:09:08 +00:00
|
|
|
|
2003-02-08 16:05:46 +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) );
|
|
|
|
}
|
|
|
|
|
2003-02-08 16:05:46 +00:00
|
|
|
for(int j = 0; j < m; ++j)
|
2002-02-16 16:09:08 +00:00
|
|
|
{
|
2003-02-08 16:05:46 +00:00
|
|
|
pthread_join(a[j], 0);
|
2002-02-16 16:09:08 +00:00
|
|
|
}
|
2002-02-16 18:45:20 +00:00
|
|
|
|
2003-02-08 16:05:46 +00:00
|
|
|
t = clock() - t;
|
2002-02-18 12:39:32 +00:00
|
|
|
|
2003-02-08 16:05:46 +00:00
|
|
|
printf("\n\n%.3f seconds.\n", static_cast<double>(t) / CLOCKS_PER_SEC);
|
2002-02-18 12:39:32 +00:00
|
|
|
|
2002-02-16 18:45:20 +00:00
|
|
|
return 0;
|
2002-02-16 16:09:08 +00:00
|
|
|
}
|