mirror of
				https://github.com/boostorg/smart_ptr.git
				synced 2025-10-26 13:21:39 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			182 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			182 lines
		
	
	
		
			3.5 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #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>
 | |
| 
 | |
| #include <vector>
 | |
| #include <memory>
 | |
| #include <stdexcept>
 | |
| 
 | |
| #include <cstdio>
 | |
| #include <ctime>
 | |
| 
 | |
| // 'portable' thread framework
 | |
| 
 | |
| class abstract_thread
 | |
| {
 | |
| public:
 | |
| 
 | |
|     virtual ~abstract_thread() {}
 | |
|     virtual void run() = 0;
 | |
| };
 | |
| 
 | |
| #if !defined(BOOST_HAS_PTHREADS) && defined(BOOST_HAS_WINTHREADS)
 | |
| 
 | |
| char const * title = "Using Windows threads";
 | |
| 
 | |
| #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
 | |
| 
 | |
| char const * title = "Using POSIX threads";
 | |
| 
 | |
| #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
 | |
| 
 | |
| #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 test_main( int, char ** )
 | |
| {
 | |
|     std::printf("%s: %s, %d threads, %d iterations: ", title, implementation, m, n);
 | |
| 
 | |
|     boost::shared_ptr<int> pi(new int(42));
 | |
| 
 | |
|     std::clock_t t = std::clock();
 | |
| 
 | |
|     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);
 | |
|     }
 | |
| 
 | |
|     t = std::clock() - t;
 | |
| 
 | |
|     std::printf("\n\n%.3f seconds.\n", static_cast<double>(t) / CLOCKS_PER_SEC);
 | |
| 
 | |
|     return 0;
 | |
| }
 |