⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 thread.cpp

📁 Pegasus is an open-source implementationof the DMTF CIM and WBEM standards. It is designed to be por
💻 CPP
字号:
//%2006//////////////////////////////////////////////////////////////////////////// Copyright (c) 2000, 2001, 2002 BMC Software; Hewlett-Packard Development// Company, L.P.; IBM Corp.; The Open Group; Tivoli Systems.// Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L.P.;// IBM Corp.; EMC Corporation, The Open Group.// Copyright (c) 2004 BMC Software; Hewlett-Packard Development Company, L.P.;// IBM Corp.; EMC Corporation; VERITAS Software Corporation; The Open Group.// Copyright (c) 2005 Hewlett-Packard Development Company, L.P.; IBM Corp.;// EMC Corporation; VERITAS Software Corporation; The Open Group.// Copyright (c) 2006 Hewlett-Packard Development Company, L.P.; IBM Corp.;// EMC Corporation; Symantec Corporation; The Open Group.//// Permission is hereby granted, free of charge, to any person obtaining a copy// of this software and associated documentation files (the "Software"), to// deal in the Software without restriction, including without limitation the// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or// sell copies of the Software, and to permit persons to whom the Software is// furnished to do so, subject to the following conditions:// // THE ABOVE COPYRIGHT NOTICE AND THIS PERMISSION NOTICE SHALL BE INCLUDED IN// ALL COPIES OR SUBSTANTIAL PORTIONS OF THE SOFTWARE. THE SOFTWARE IS PROVIDED// "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT// LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT// HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN// ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.////==============================================================================//// Author: Mike Day (mdday@us.ibm.com)//// Modified By://              Steve Hills (steve.hills@ncr.com)//              Sean Keenan, Hewlett-Packard Company (sean.keenan@hp.com)////%/////////////////////////////////////////////////////////////////////////////#include <Pegasus/Common/Thread.h>#include <Pegasus/Common/ReadWriteSem.h>#include <sys/types.h>#if defined(PEGASUS_PLATFORM_WIN32_IX86_MSVC)#else#include <unistd.h>#endif #if defined(PEGASUS_OS_HPUX) || defined(PEGASUS_OS_LINUX)# include <memory>#endif#include <Pegasus/Common/PegasusAssert.h>#include <iostream>#include <stdio.h>#include <string.h>//#include <malloc>PEGASUS_USING_PEGASUS;PEGASUS_USING_STD;Boolean die = false;ThreadReturnType PEGASUS_THREAD_CDECL reading_thread(void *parm);ThreadReturnType PEGASUS_THREAD_CDECL writing_thread(void *parm);ThreadReturnType PEGASUS_THREAD_CDECL test1_thread( void* parm );ThreadReturnType PEGASUS_THREAD_CDECL test2_thread( void* parm );//ThreadReturnType PEGASUS_THREAD_CDECL test3_thread( void* parm );ThreadReturnType PEGASUS_THREAD_CDECL test4_thread( void* parm );#define  THREAD_NR 500AtomicInt read_count ;AtomicInt write_count ;AtomicInt testval1(0);Boolean verbose = false;struct TestThreadData{    char chars[2];};int main(int argc, char **argv){	int i;	verbose = (getenv("PEGASUS_TEST_VERBOSE")) ? true : false;	// cout << argv[0] << endl;	{	// Test return code	Thread t( test1_thread, 0, false );	t.run();	t.join();	if( t.get_exit() != (ThreadReturnType)32 )	{		cerr << "Error test return code" << endl;		return 1;	}	}	{	// There shouldn't be a memory leak here	Thread* threads[THREAD_NR];	int max_threads = THREAD_NR;		for( i = 0; i < THREAD_NR; i++ )	{		threads[i] = new Thread( test2_thread, 0, false );		TestThreadData* data = NULL;		try {			data = new struct TestThreadData;		} catch (bad_alloc&)		{			cerr << "Not enough memory. Changing the amount of threads used." << endl;			max_threads = i;			delete threads[i];			break;		}		data->chars[0] = 'B';		data->chars[1] = 'E';		threads[i]->put_tsd( "test2", thread_data::default_delete, 2, data );		if (threads[i]->run()!=PEGASUS_THREAD_OK)		{			cerr << "Not enough memory. Changing the amount of threads used." << endl;			max_threads = i;			delete threads[i];			break;		}	}	for( i = 0; i < max_threads; i++ )		threads[i]->join();	for( i = 0; i < max_threads; i++ )		delete threads[i];	// TODO: Programatically check	//Threads::sleep( 10000 );	}	// NOTE: see test3_thread() comments	//{	//// Test proper canceling/thread exit	//testval1 = 0;	//Thread t( test3_thread, 0, false );	//t.run();	//t.cancel();	//t.join();	//if( testval1.get() != 42 )	//{	//	cerr << "Thread probably incorrectly terminated!" << endl;	//	return 1;	//}	//}// Note: A glibc problem on some versions of Linux makes Thread::cancel unsafe#if !defined(PEGASUS_OS_LINUX)	{	// Test deadlocked thread handling	Thread t( test4_thread, 0, false );	t.run();	t.cancel();	if (verbose) cout << argv[0] << " - If this hangs here, there is a thread deadlock handling bug..." << endl;	t.join();	// Shouldn't hang forever	if (verbose) cout << argv[0] << " - Deadlock test finished." << endl;	}#endif   ReadWriteSem *rw = new ReadWriteSem();   Thread *readers[40];   Thread *writers[10];      for(i = 0; i < 40; i++)   {      readers[i] = new Thread(reading_thread, rw, false);      readers[i]->run();   }      for( i = 0; i < 10; i++)   {      writers[i] = new Thread(writing_thread, rw, false);      writers[i]->run();   }   Threads::sleep(20000);    die = true;     for(i = 0; i < 40; i++)   {     readers[i]->join();      delete readers[i];   }   for(i = 0; i < 10; i++)   {      writers[i]->join();      delete writers[i];   }   delete rw;   if (verbose) cout << endl << "read operations: " << read_count.get() << endl;   if (verbose) cout << "write operations: " << write_count.get() << endl;      cout << argv[0] << " +++++ passed all tests" << endl;   return(0);}void deref(void *parm){      Thread *my_handle = (Thread *)parm;   try    {      my_handle->dereference_tsd();   }   catch(IPCException& e)   {      e = e;      cout << "exception dereferencing the tsd " << endl;      abort();   }   return;}void exit_one(void *parm){      Thread *my_handle = (Thread *)parm;   if (verbose) cout << "1";}void exit_two(void *parm){      Thread *my_handle = (Thread *)parm;   if (verbose) cout << "2";}ThreadReturnType PEGASUS_THREAD_CDECL reading_thread(void *parm){   Thread *my_handle = (Thread *)parm;   ReadWriteSem * my_parm = (ReadWriteSem *)my_handle->get_parm();      ThreadType myself = Threads::self();      if (verbose) cout << "r";      const char *keys[] =       {	 "one", "two", "three", "four"      };      try   {      my_handle->cleanup_push(exit_one , my_handle );   }   catch(IPCException& e)   {      e = e;      cout << "Exception while trying to push cleanup handler" << endl;      abort();   }      try   {      my_handle->cleanup_push(exit_two , my_handle );   }      catch(IPCException& e)   {      e = e;      cout << "Exception while trying to push cleanup handler" << endl;      abort();   }      while(die == false)    {      int i = 0;      #ifndef PEGASUS_PLATFORM_ZOS_ZSERIES_IBM      char *my_storage = (char *)calloc(256, sizeof(char));#else      char *my_storage = (char *)::operator new(256);#endif      //    sprintf(my_storage, "%ld", myself + i);      try       {#ifndef PEGASUS_PLATFORM_ZOS_ZSERIES_IBM	 my_handle->put_tsd(keys[i % 4], free, 256, my_storage);#else         my_handle->put_tsd(keys[i % 4], ::operator delete,                            256, my_storage);              #endif      }      catch(IPCException& e)      {      e = e;      cout << "Exception while trying to put local storage: "           << Threads::id(myself).buffer << endl;      abort();      }      try       {	 my_parm->wait_read(myself);      }      catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to get a read lock" << endl;	 abort();      }            read_count++;      if (verbose) cout << "+";      my_handle->sleep(1);      try      {	 my_handle->cleanup_push(deref , my_handle );      }      catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to push cleanup handler" << endl;	 abort();      }      try       {	 my_handle->reference_tsd(keys[i % 4]);      }            catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to reference local storage" << endl;	 abort();      }            try      {	 my_handle->cleanup_pop(true);      }            catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to pop cleanup handler" << endl;	 abort();      }      try       {	 my_parm->unlock_read(myself);      }      catch(IPCException& e)	{	 e = e;	 cout << "Exception while trying to release a read lock" << endl;	 abort();      }            try       {	 my_handle->delete_tsd(keys[i % 4]);      }      catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to delete local storage: "              << Threads::id(myself).buffer << endl;	 abort();      }      i++;   }   my_handle->exit_self((ThreadReturnType)1);   return(0);}ThreadReturnType PEGASUS_THREAD_CDECL writing_thread(void *parm){      Thread *my_handle = (Thread *)parm;   ReadWriteSem * my_parm = (ReadWriteSem *)my_handle->get_parm();      ThreadType myself = Threads::self();      if (verbose) cout << "w";      while(die == false)    {      try       {	 my_parm->wait_write(myself);      }      catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to get a write lock" << endl;	 abort();      }      write_count++;      if (verbose) cout << "*";      my_handle->sleep(1);      try       {	 my_parm->unlock_write(myself);      }      catch(IPCException& e)      {	 e = e;	 cout << "Exception while trying to release a write  lock" << endl;	 abort();      }   }   my_handle->exit_self((ThreadReturnType)1);   return(0);}ThreadReturnType PEGASUS_THREAD_CDECL test1_thread( void* parm ){	Threads::sleep( 1000 );	return( (ThreadReturnType)32 );}ThreadReturnType PEGASUS_THREAD_CDECL test2_thread( void* parm ){	Thread* thread = (Thread*)parm;	TestThreadData* data = (TestThreadData*)thread->reference_tsd("test2");	PEGASUS_TEST_ASSERT (data != NULL);	PEGASUS_TEST_ASSERT (data->chars[0] == 'B');	PEGASUS_TEST_ASSERT (data->chars[1] == 'E');	thread->dereference_tsd();		thread->exit_self( (ThreadReturnType)33 );	return( (ThreadReturnType)32 );}void test3_thread_cleanup1(void*){	testval1 = 42;}// NOTE: I don't think Thread::cleanup_push (and pop) will work// with pthreads because the implicit cancelation point or the // test_cancel() call (which calls pthread_cancel()) will exit the// thread without performing the Thread cleanup routines.//ThreadReturnType PEGASUS_THREAD_CDECL test3_thread( void* parm )//{//	Thread* thread = (Thread*)parm;//	while( true )//	{//		testval1 = 0;//		thread->cleanup_push( test3_thread_cleanup1, 0 );//		Threads::sleep( 2000 );//		thread->test_cancel();//		thread->cleanup_pop( false );//	}//	thread->exit_self( (ThreadReturnType)42 );//	return( (ThreadReturnType)42 );//}ThreadReturnType PEGASUS_THREAD_CDECL test4_thread( void* parm ){	Thread* thread = (Thread*)parm;	// Simulate a deadlocked thread	while( true )	{#if defined(PEGASUS_OS_DARWIN) || defined(PEGASUS_OS_VMS)           //            // sleep is NOT a thread cancellation point           //  for VMS.           // 	   pthread_testcancel();#endif	   Threads::sleep( 2000 );	}	PEGASUS_UNREACHABLE (thread->exit_self( (ThreadReturnType)52 );	return( (ThreadReturnType)52 );)}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -