Files
smart_ptr/shared_ptr_mt_test.cpp

174 lines
3.3 KiB
C++
Raw Normal View History

2002-02-16 16:09:08 +00:00
#if defined(_MSC_VER) && !defined(__ICL)
#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>
#define BOOST_INCLUDE_MAIN
#include <boost/test/test_tools.hpp>
2002-02-16 16:09:08 +00:00
#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>
extern "C" void* common_thread_routine(void * pv)
{
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
int test_main( int, char ** )
2002-02-16 16:09:08 +00:00
{
2002-02-18 12:39:32 +00:00
std::printf("%s: %d threads, %d iterations, ", title, 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
2002-02-18 12:39:32 +00:00
std::clock_t t = std::clock();
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 i = 0; i < m; ++i)
{
pthread_join(a[i], 0);
}
2002-02-18 12:39:32 +00:00
t = std::clock() - t;
std::printf("%.4f seconds.\n", static_cast<double>(t) / CLOCKS_PER_SEC);
return 0;
2002-02-16 16:09:08 +00:00
}