From 4651204fb50d404e33391b16793364a2ee2ad5db Mon Sep 17 00:00:00 2001 From: Andy Heninger Date: Thu, 23 Apr 2015 23:59:24 +0000 Subject: [PATCH] ICU-11643 Several threading test cleanups. X-SVN-Rev: 37395 --- icu4c/source/test/intltest/intltest.cpp | 1 + icu4c/source/test/intltest/simplethread.cpp | 214 +---- icu4c/source/test/intltest/simplethread.h | 14 +- icu4c/source/test/intltest/tsmthred.cpp | 886 ++++++-------------- icu4c/source/test/intltest/tzfmttst.cpp | 24 +- 5 files changed, 268 insertions(+), 871 deletions(-) diff --git a/icu4c/source/test/intltest/intltest.cpp b/icu4c/source/test/intltest/intltest.cpp index 63ad6dc912b..3b821540c29 100644 --- a/icu4c/source/test/intltest/intltest.cpp +++ b/icu4c/source/test/intltest/intltest.cpp @@ -561,6 +561,7 @@ void IntlTest::setCaller( IntlTest* callingTest ) verbose = caller->verbose; no_err_msg = caller->no_err_msg; quick = caller->quick; + threadCount = caller->threadCount; testoutfp = caller->testoutfp; LL_indentlevel = caller->LL_indentlevel + indentLevel_offset; numProps = caller->numProps; diff --git a/icu4c/source/test/intltest/simplethread.cpp b/icu4c/source/test/intltest/simplethread.cpp index 4e2f844ac0a..9eac0885944 100644 --- a/icu4c/source/test/intltest/simplethread.cpp +++ b/icu4c/source/test/intltest/simplethread.cpp @@ -1,6 +1,6 @@ /******************************************************************** * COPYRIGHT: - * Copyright (c) 1999-2013, International Business Machines Corporation and + * Copyright (c) 1999-2015, International Business Machines Corporation and * others. All Rights Reserved. ********************************************************************/ @@ -62,15 +62,7 @@ #if defined(POSIX) #define HAVE_IMP -#if (ICU_USE_THREADS == 1) #include -#endif - -#if defined(__hpux) && defined(HPUX_CMA) -# if defined(read) // read being defined as cma_read causes trouble with iostream::read -# undef read -# endif -#endif #if U_PLATFORM == U_PF_OS390 #include @@ -104,31 +96,6 @@ #endif -#if (ICU_USE_THREADS==0) - SimpleThread::SimpleThread() - {} - - SimpleThread::~SimpleThread() - {} - - int32_t - SimpleThread::start() - { return -1; } - - void - SimpleThread::run() - {} - - void - SimpleThread::sleep(int32_t millis) - {} - - UBool - SimpleThread::isRunning() { - return FALSE; - } -#else - #include "unicode/putil.h" /* for mthreadtest*/ @@ -223,77 +190,13 @@ int32_t SimpleThread::start() } -UBool SimpleThread::isRunning() { - // - // Test whether the thread associated with the SimpleThread object is - // still actually running. - // - // NOTE: on Win64 on Itanium processors, a crashes - // occur if the main thread of a process exits concurrently with some - // other thread(s) exiting. To avoid the possibility, we wait until the - // OS indicates that all threads have terminated, rather than waiting - // only until the end of the user's Run function has been reached. - // - // I don't know whether the crashes represent a Windows bug, or whether - // main() programs are supposed to have to wait for their threads. - // +void SimpleThread::join() { Win32ThreadImplementation *imp = (Win32ThreadImplementation*)fImplementation; - - bool success; - DWORD threadExitCode; - if (imp->fHandle == 0) { // No handle, thread must not be running. - return FALSE; + return; } - success = GetExitCodeThread(imp->fHandle, &threadExitCode) != 0; - if (! success) { - // Can't get status, thread must not be running. - return FALSE; - } - return (threadExitCode == STILL_ACTIVE); -} - - -void SimpleThread::sleep(int32_t millis) -{ - ::Sleep(millis); -} - -//----------------------------------------------------------------------------------- -// -// class SimpleThread NULL Implementation -// -//----------------------------------------------------------------------------------- -#elif U_PLATFORM == U_PF_CLASSIC_MACOS - -// since the Mac has no preemptive threading (at least on MacOS 8), only -// cooperative threading, threads are a no-op. We have no yield() calls -// anywhere in the ICU, so we are guaranteed to be thread-safe. - -#define HAVE_IMP - -SimpleThread::SimpleThread() -{} - -SimpleThread::~SimpleThread() -{} - -int32_t -SimpleThread::start() -{ return 0; } - -void -SimpleThread::run() -{} - -void -SimpleThread::sleep(int32_t millis) -{} - -UBool -SimpleThread::isRunning() { - return FALSE; + WaitForSingleObject(imp->fHandle, INFINITE); } #endif @@ -303,22 +206,6 @@ SimpleThread::isRunning() { // // class SimpleThread POSIX implementation // -// A note on the POSIX vs the Windows implementations of this class.. -// On Windows, the main thread must verify that other threads have finished -// before exiting, or crashes occasionally occur. (Seen on Itanium Win64 only) -// The function SimpleThread::isRunning() is used for this purpose. -// -// On POSIX, there is NO reliable non-blocking mechanism to determine -// whether a thread has exited. pthread_kill(thread, 0) almost works, -// but the system can recycle thread ids immediately, so seeing that a -// thread exists with this call could mean that the original thread has -// finished and a new one started with the same ID. Useless. -// -// So we need to do the check with user code, by setting a flag just before -// the thread function returns. A technique that is guaranteed to fail -// on Windows, because it indicates that the thread is done before all -// system level cleanup has happened. -// //----------------------------------------------------------------------------------- #if defined(POSIX) #define HAVE_IMP @@ -326,40 +213,25 @@ SimpleThread::isRunning() { struct PosixThreadImplementation { pthread_t fThread; - UBool fRunning; - UBool fRan; // True if the thread was successfully started }; extern "C" void* SimpleThreadProc(void *arg) { // This is the code that is run in the new separate thread. SimpleThread *This = (SimpleThread *)arg; - This->run(); // Run the user code. - - // The user function has returned. Set the flag indicating that this thread - // is done. Need a mutex for memory barrier purposes only, so that other thread - // will reliably see that the flag has changed. - PosixThreadImplementation *imp = (PosixThreadImplementation*)This->fImplementation; - umtx_lock(NULL); - imp->fRunning = FALSE; - umtx_unlock(NULL); + This->run(); return 0; } SimpleThread::SimpleThread() { PosixThreadImplementation *imp = new PosixThreadImplementation; - imp->fRunning = FALSE; - imp->fRan = FALSE; fImplementation = imp; } SimpleThread::~SimpleThread() { PosixThreadImplementation *imp = (PosixThreadImplementation*)fImplementation; - if (imp->fRan) { - pthread_join(imp->fThread, NULL); - } delete imp; fImplementation = (void *)0xdeadbeef; } @@ -371,16 +243,7 @@ int32_t SimpleThread::start() static UBool attrIsInitialized = FALSE; PosixThreadImplementation *imp = (PosixThreadImplementation*)fImplementation; - imp->fRunning = TRUE; - imp->fRan = TRUE; -#ifdef HPUX_CMA - if (attrIsInitialized == FALSE) { - rc = pthread_attr_create(&attr); - attrIsInitialized = TRUE; - } - rc = pthread_create(&(imp->fThread),attr,&SimpleThreadProc,(void*)this); -#else if (attrIsInitialized == FALSE) { rc = pthread_attr_init(&attr); #if U_PLATFORM == U_PF_OS390 @@ -398,53 +261,18 @@ int32_t SimpleThread::start() #endif attrIsInitialized = TRUE; } - rc = pthread_create(&(imp->fThread),&attr,&SimpleThreadProc,(void*)this); -#endif + rc = pthread_create(&(imp->fThread), &attr, &SimpleThreadProc, (void*)this); if (rc != 0) { // some kind of error occured, the thread did not start. - imp->fRan = FALSE; - imp->fRunning = FALSE; } return rc; } - -UBool -SimpleThread::isRunning() { - // Note: Mutex functions are used here not for synchronization, - // but to force memory barriors to exist, to ensure that one thread - // can see changes made by another when running on processors - // with memory models having weak coherency. +void SimpleThread::join() { PosixThreadImplementation *imp = (PosixThreadImplementation*)fImplementation; - umtx_lock(NULL); - UBool retVal = imp->fRunning; - umtx_unlock(NULL); - return retVal; -} - - -void SimpleThread::sleep(int32_t millis) -{ -#if U_PLATFORM == U_PF_SOLARIS - sigignore(SIGALRM); -#endif - -#ifdef HPUX_CMA - cma_sleep(millis/100); -#elif U_PLATFORM == U_PF_HPUX || U_PLATFORM == U_PF_OS390 - millis *= 1000; - while(millis >= 1000000) { - usleep(999999); - millis -= 1000000; - } - if(millis > 0) { - usleep(millis); - } -#else - usleep(millis * 1000); -#endif + pthread_join(imp->fThread, NULL); } #endif @@ -453,30 +281,4 @@ void SimpleThread::sleep(int32_t millis) #ifndef HAVE_IMP #error No implementation for threads! Cannot test. -0 = 216; //die #endif - -//------------------------------------------------------------------------------------------- -// -// class ThreadWithStatus - a thread that we can check the status and error condition of -// -//------------------------------------------------------------------------------------------- -class ThreadWithStatus : public SimpleThread -{ -public: - UBool getError() { return (fErrors > 0); } - UBool getError(UnicodeString& fillinError) { fillinError = fErrorString; return (fErrors > 0); } - virtual ~ThreadWithStatus(){} -protected: - ThreadWithStatus() : fErrors(0) {} - void error(const UnicodeString &error) { - fErrors++; fErrorString = error; - SimpleThread::errorFunc(); - } - void error() { error("An error occured."); } -private: - int32_t fErrors; - UnicodeString fErrorString; -}; - -#endif // ICU_USE_THREADS diff --git a/icu4c/source/test/intltest/simplethread.h b/icu4c/source/test/intltest/simplethread.h index aa42d0aba46..3f42485bf0b 100644 --- a/icu4c/source/test/intltest/simplethread.h +++ b/icu4c/source/test/intltest/simplethread.h @@ -11,22 +11,16 @@ class U_EXPORT SimpleThread { -public: + public: SimpleThread(); virtual ~SimpleThread(); - int32_t start(void); // start the thread - UBool isRunning(); // return true if a started thread has exited. - void join() {while (isRunning()) sleep(1);}; // Wait until thread finishes. - // TODO: add a real implementation. Ticket #11643 + int32_t start(void); // start the thread. Return 0 if successfull. + void join(); // A thread must be joined before deleting its SimpleThread. virtual void run(void) = 0; // Override this to provide the code to run // in the thread. + private: void *fImplementation; - -public: - static void sleep(int32_t millis); // probably shouldn't go here but oh well. - static void errorFunc(); // Empty function, provides a single convenient place - // to break on errors. }; #endif diff --git a/icu4c/source/test/intltest/tsmthred.cpp b/icu4c/source/test/intltest/tsmthred.cpp index dabf11d0b93..84615e2aade 100644 --- a/icu4c/source/test/intltest/tsmthred.cpp +++ b/icu4c/source/test/intltest/tsmthred.cpp @@ -4,12 +4,6 @@ * others. All Rights Reserved. ********************************************************************/ -#if defined(hpux) -# ifndef _INCLUDE_POSIX_SOURCE -# define _INCLUDE_POSIX_SOURCE -# endif -#endif - #include "simplethread.h" #include "unicode/utypes.h" @@ -32,73 +26,6 @@ #include "unifiedcache.h" #include "uassert.h" -#if U_PLATFORM_USES_ONLY_WIN32_API - /* Prefer native Windows APIs even if POSIX is implemented (i.e., on Cygwin). */ -# undef POSIX -#elif U_PLATFORM_IMPLEMENTS_POSIX -# define POSIX -#else -# undef POSIX -#endif - -/* Needed by z/OS to get usleep */ -#if U_PLATFORM == U_PF_OS390 -#define __DOT1 1 -#define __UU -#ifndef _XPG4_2 -#define _XPG4_2 -#endif -#include -#endif -#if defined(POSIX) - -#define HAVE_IMP - -#if (ICU_USE_THREADS == 1) -#include -#endif - -#if defined(__hpux) && defined(HPUX_CMA) -# if defined(read) // read being defined as cma_read causes trouble with iostream::read -# undef read -# endif -#endif - -/* Define __EXTENSIONS__ for Solaris and old friends in strict mode. */ -#ifndef __EXTENSIONS__ -#define __EXTENSIONS__ -#endif - -#if U_PLATFORM == U_PF_OS390 -#include -#endif - -#if U_PLATFORM != U_PF_OS390 -#include -#endif - -/* Define _XPG4_2 for Solaris and friends. */ -#ifndef _XPG4_2 -#define _XPG4_2 -#endif - -/* Define __USE_XOPEN_EXTENDED for Linux and glibc. */ -#ifndef __USE_XOPEN_EXTENDED -#define __USE_XOPEN_EXTENDED -#endif - -/* Define _INCLUDE_XOPEN_SOURCE_EXTENDED for HP/UX (11?). */ -#ifndef _INCLUDE_XOPEN_SOURCE_EXTENDED -#define _INCLUDE_XOPEN_SOURCE_EXTENDED -#endif - -#include - -#endif -/* HPUX */ -#ifdef sleep -#undef sleep -#endif #define TSMTHREAD_FAIL(msg) errln("%s at file %s, line %d", msg, __FILE__, __LINE__) #define TSMTHREAD_ASSERT(expr) {if (!(expr)) {TSMTHREAD_FAIL("Fail");}} @@ -113,23 +40,6 @@ MultithreadTest::~MultithreadTest() { } - - -#if (ICU_USE_THREADS==0) -void MultithreadTest::runIndexedTest( int32_t index, UBool exec, - const char* &name, char* /*par*/ ) { - if (exec) logln("TestSuite MultithreadTest: "); - - if(index == 0) - name = "NO_THREADED_TESTS"; - else - name = ""; - - if(exec) { logln("MultithreadTest - test DISABLED. ICU_USE_THREADS set to 0, check your configuration if this is a problem.."); - } -} -#else - #include #include #include // tolower, toupper @@ -145,9 +55,6 @@ void MultithreadTest::runIndexedTest( int32_t index, UBool exec, #include "unicode/calendar.h" #include "ucaconf.h" -void SimpleThread::errorFunc() { - // *(char *)0 = 3; // Force entry into a debugger via a crash; -} void MultithreadTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ ) { @@ -235,96 +142,25 @@ void MultithreadTest::runIndexedTest( int32_t index, UBool exec, // TestThreads -- see if threads really work at all. // // Set up N threads pointing at N chars. When they are started, they will -// each sleep 1 second and then set their chars. At the end we make sure they -// are all set. +// set their chars. At the end we make sure they are all set. // //----------------------------------------------------------------------------------- -#define THREADTEST_NRTHREADS 8 -#define ARABICSHAPE_THREADTEST 30 class TestThreadsThread : public SimpleThread { public: TestThreadsThread(char* whatToChange) { fWhatToChange = whatToChange; } - virtual void run() { SimpleThread::sleep(1000); - Mutex m; + virtual void run() { Mutex m; *fWhatToChange = '*'; } private: char *fWhatToChange; }; -//----------------------------------------------------------------------------------- -// -// TestArabicShapeThreads -- see if calls to u_shapeArabic in many threads works successfully -// -// Set up N threads pointing at N chars. When they are started, they will make calls to doTailTest which tests -// u_shapeArabic, if the calls are successful it will the set * chars. -// At the end we make sure all threads managed to run u_shapeArabic successfully. -// This is a unit test for ticket 9473 -// -//----------------------------------------------------------------------------------- -class TestArabicShapeThreads : public SimpleThread -{ -public: - TestArabicShapeThreads(char* whatToChange) { fWhatToChange = whatToChange;} - virtual void run() { - if(doTailTest()==TRUE) - *fWhatToChange = '*'; - } -private: - char *fWhatToChange; - - UBool doTailTest(void) { - static const UChar src[] = { 0x0020, 0x0633, 0 }; - static const UChar dst_old[] = { 0xFEB1, 0x200B,0 }; - static const UChar dst_new[] = { 0xFEB1, 0xFE73,0 }; - UChar dst[3] = { 0x0000, 0x0000,0 }; - int32_t length; - UErrorCode status; - IntlTest inteltst = IntlTest(); - status = U_ZERO_ERROR; - length = u_shapeArabic(src, -1, dst, UPRV_LENGTHOF(dst), - U_SHAPE_LETTERS_SHAPE|U_SHAPE_SEEN_TWOCELL_NEAR, &status); - if(U_FAILURE(status)) { - inteltst.errln("Fail: status %s\n", u_errorName(status)); - return FALSE; - } else if(length!=2) { - inteltst.errln("Fail: len %d expected 3\n", length); - return FALSE; - } else if(u_strncmp(dst,dst_old,UPRV_LENGTHOF(dst))) { - inteltst.errln("Fail: got U+%04X U+%04X expected U+%04X U+%04X\n", - dst[0],dst[1],dst_old[0],dst_old[1]); - return FALSE; - } - - - //"Trying new tail - status = U_ZERO_ERROR; - length = u_shapeArabic(src, -1, dst, UPRV_LENGTHOF(dst), - U_SHAPE_LETTERS_SHAPE|U_SHAPE_SEEN_TWOCELL_NEAR|U_SHAPE_TAIL_NEW_UNICODE, &status); - if(U_FAILURE(status)) { - inteltst.errln("Fail: status %s\n", u_errorName(status)); - return FALSE; - } else if(length!=2) { - inteltst.errln("Fail: len %d expected 3\n", length); - return FALSE; - } else if(u_strncmp(dst,dst_new,UPRV_LENGTHOF(dst))) { - inteltst.errln("Fail: got U+%04X U+%04X expected U+%04X U+%04X\n", - dst[0],dst[1],dst_new[0],dst_new[1]); - return FALSE; - } - - - return TRUE; - -} - - -}; void MultithreadTest::TestThreads() { + static const int32_t THREADTEST_NRTHREADS = 8; char threadTestChars[THREADTEST_NRTHREADS + 1]; SimpleThread *threads[THREADTEST_NRTHREADS]; int32_t numThreadsStarted = 0; @@ -346,273 +182,211 @@ void MultithreadTest::TestThreads() else { numThreadsStarted++; } - SimpleThread::sleep(100); logln(" Subthread started."); } - logln("Waiting for threads to be set.."); - if (numThreadsStarted == 0) { - errln("No threads could be started for testing!"); + if (numThreadsStarted != THREADTEST_NRTHREADS) { + errln("Not all threads could be started for testing!"); return; } - int32_t patience = 40; // seconds to wait - - while(patience--) - { - int32_t count = 0; - umtx_lock(NULL); - for(i=0;ijoin(); + if (threadTestChars[i] != '*') { + errln("%s:%d Thread %d failed.", __FILE__, __LINE__, i); } - umtx_unlock(NULL); - - if(count == THREADTEST_NRTHREADS) - { - logln("->" + UnicodeString(threadTestChars) + "<- Got all threads! cya"); - for(i=0;i" + UnicodeString(threadTestChars) + "<- Waiting.."); - SimpleThread::sleep(500); - } - - errln("->" + UnicodeString(threadTestChars) + "<- PATIENCE EXCEEDED!! Still missing some."); - for(i=0;i do TestArabicShapingThreads <- Firing off threads.. "); - for(i=0;istart() != 0) { - errln("Error starting thread %d", i); - } - else { - numThreadsStarted++; - } - //SimpleThread::sleep(100); - logln(" Subthread started."); - } - - logln("Waiting for threads to be set.."); - if (numThreadsStarted == 0) { - errln("No threads could be started for testing!"); - return; - } - - int32_t patience = 100; // seconds to wait - - while(patience--) - { - int32_t count = 0; - umtx_lock(NULL); - for(i=0;iTestArabicShapingThreads <- Got all threads! cya"); - for(i=0;ierrln("Fail: status %s\n", u_errorName(status)); + return; + } else if(length!=2) { + IntlTest::gTest->errln("Fail: len %d expected 3\n", length); + return; + } else if(u_strncmp(dst,dst_old,UPRV_LENGTHOF(dst))) { + IntlTest::gTest->errln("Fail: got U+%04X U+%04X expected U+%04X U+%04X\n", + dst[0],dst[1],dst_old[0],dst_old[1]); return; } - logln("-> TestArabicShapingThreads <- Waiting.."); - SimpleThread::sleep(500); + + //"Trying new tail + status = U_ZERO_ERROR; + length = u_shapeArabic(src, -1, dst, UPRV_LENGTHOF(dst), + U_SHAPE_LETTERS_SHAPE|U_SHAPE_SEEN_TWOCELL_NEAR|U_SHAPE_TAIL_NEW_UNICODE, &status); + if(U_FAILURE(status)) { + IntlTest::gTest->errln("Fail: status %s\n", u_errorName(status)); + return; + } else if(length!=2) { + IntlTest::gTest->errln("Fail: len %d expected 3\n", length); + return; + } else if(u_strncmp(dst,dst_new,UPRV_LENGTHOF(dst))) { + IntlTest::gTest->errln("Fail: got U+%04X U+%04X expected U+%04X U+%04X\n", + dst[0],dst[1],dst_new[0],dst_new[1]); + return; + } + } + return; +} + + +void MultithreadTest::TestArabicShapingThreads() +{ + TestArabicShapeThreads threads[30]; + + int32_t i; + + logln("-> do TestArabicShapingThreads <- Firing off threads.. "); + for(i=0; i < UPRV_LENGTHOF(threads); i++) { + if (threads[i].start() != 0) { + errln("Error starting thread %d", i); + } } - errln("-> TestArabicShapingThreads <- PATIENCE EXCEEDED!! Still missing some."); - for(i=0;iTestArabicShapingThreads <- Got all threads! cya"); } //----------------------------------------------------------------------- // // TestMutex - a simple (non-stress) test to verify that ICU mutexes -// are actually mutexing. Does not test the use of +// and condition variables are functioning. Does not test the use of // mutexes within ICU services, but rather that the // platform's mutex support is at least superficially there. // //---------------------------------------------------------------------- -static UMutex gTestMutexA = U_MUTEX_INITIALIZER; -static UMutex gTestMutexB = U_MUTEX_INITIALIZER; +static UMutex gTestMutexA = U_MUTEX_INITIALIZER; +static UConditionVar gThreadsCountChanged = U_CONDITION_INITIALIZER; static int gThreadsStarted = 0; static int gThreadsInMiddle = 0; static int gThreadsDone = 0; -static const int TESTMUTEX_THREAD_COUNT = 4; - -static int safeIncr(int &var, int amt) { - // Thread safe (using global mutex) increment of a variable. - // Return the updated value. - // Can also be used as a safe load of a variable by incrementing it by 0. - Mutex m; - var += amt; - return var; -} +static const int TESTMUTEX_THREAD_COUNT = 40; class TestMutexThread : public SimpleThread { public: - virtual void run() - { + virtual void run() { // This is the code that each of the spawned threads runs. - // All of the spawned threads bunch up together at each of the two mutexes - // because the main holds the mutexes until they do. - // - safeIncr(gThreadsStarted, 1); + // All threads move together throught the started - middle - done sequence together, + // waiting for all other threads to reach each point before advancing. umtx_lock(&gTestMutexA); + gThreadsStarted += 1; + umtx_condBroadcast(&gThreadsCountChanged); + while (gThreadsStarted < TESTMUTEX_THREAD_COUNT) { + if (gThreadsInMiddle != 0) { + IntlTest::gTest->errln( + "%s:%d gThreadsInMiddle = %d. Expected 0.", __FILE__, __LINE__, gThreadsInMiddle); + return; + } + umtx_condWait(&gThreadsCountChanged, &gTestMutexA); + } + + gThreadsInMiddle += 1; + umtx_condBroadcast(&gThreadsCountChanged); + while (gThreadsInMiddle < TESTMUTEX_THREAD_COUNT) { + if (gThreadsDone != 0) { + IntlTest::gTest->errln( + "%s:%d gThreadsDone = %d. Expected 0.", __FILE__, __LINE__, gThreadsDone); + return; + } + umtx_condWait(&gThreadsCountChanged, &gTestMutexA); + } + + gThreadsDone += 1; + umtx_condBroadcast(&gThreadsCountChanged); + while (gThreadsDone < TESTMUTEX_THREAD_COUNT) { + umtx_condWait(&gThreadsCountChanged, &gTestMutexA); + } umtx_unlock(&gTestMutexA); - safeIncr(gThreadsInMiddle, 1); - umtx_lock(&gTestMutexB); - umtx_unlock(&gTestMutexB); - safeIncr(gThreadsDone, 1); } }; void MultithreadTest::TestMutex() { - // Start up the test threads. They should all pile up waiting on - // gTestMutexA, which we (the main thread) hold until the test threads - // all get there. gThreadsStarted = 0; gThreadsInMiddle = 0; gThreadsDone = 0; + int32_t i = 0; + TestMutexThread threads[TESTMUTEX_THREAD_COUNT]; umtx_lock(&gTestMutexA); - TestMutexThread *threads[TESTMUTEX_THREAD_COUNT]; - int i; - int32_t numThreadsStarted = 0; for (i=0; istart() != 0) { - errln("Error starting thread %d", i); - } - else { - numThreadsStarted++; + if (threads[i].start() != 0) { + errln("%s:%d Error starting thread %d", __FILE__, __LINE__, i); + return; } } - if (numThreadsStarted == 0) { - errln("No threads could be started for testing!"); + + // Because we are holding gTestMutexA, all of the threads should be blocked + // at the start of their run() function. + if (gThreadsStarted != 0) { + errln("%s:%d gThreadsStarted=%d. Expected 0.", __FILE__, __LINE__, gThreadsStarted); return; } - int patience = 0; - while (safeIncr(gThreadsStarted, 0) != TESTMUTEX_THREAD_COUNT) { - if (patience++ > 24) { - TSMTHREAD_FAIL("Patience Exceeded"); + while (gThreadsInMiddle < TESTMUTEX_THREAD_COUNT) { + if (gThreadsDone != 0) { + errln("%s:%d gThreadsDone=%d. Expected 0.", __FILE__, __LINE__, gThreadsStarted); return; } - SimpleThread::sleep(500); + umtx_condWait(&gThreadsCountChanged, &gTestMutexA); } - // None of the test threads should have advanced past the first mutex. - TSMTHREAD_ASSERT(gThreadsInMiddle==0); - TSMTHREAD_ASSERT(gThreadsDone==0); - // All of the test threads have made it to the first mutex. - // We (the main thread) now let them advance to the second mutex, - // where they should all pile up again. - umtx_lock(&gTestMutexB); + while (gThreadsDone < TESTMUTEX_THREAD_COUNT) { + umtx_condWait(&gThreadsCountChanged, &gTestMutexA); + } umtx_unlock(&gTestMutexA); - patience = 0; - while (safeIncr(gThreadsInMiddle, 0) != TESTMUTEX_THREAD_COUNT) { - if (patience++ > 24) { - TSMTHREAD_FAIL("Patience Exceeded"); - return; - } - SimpleThread::sleep(500); - } - TSMTHREAD_ASSERT(gThreadsDone==0); - - // All test threads made it to the second mutex. - // Now let them proceed from there. They will all terminate. - umtx_unlock(&gTestMutexB); - patience = 0; - while (safeIncr(gThreadsDone, 0) != TESTMUTEX_THREAD_COUNT) { - if (patience++ > 24) { - TSMTHREAD_FAIL("Patience Exceeded"); - return; - } - SimpleThread::sleep(500); - } - - // All threads made it by both mutexes. - for (i=0; i 0); } - UBool getError(UnicodeString& fillinError) { fillinError = fErrorString; return (fErrors > 0); } - virtual ~ThreadWithStatus(){} -protected: - ThreadWithStatus() : fErrors(0) {} - void error(const UnicodeString &error) { - fErrors++; fErrorString = error; - SimpleThread::errorFunc(); - } - void error() { error("An error occured."); } -private: - int32_t fErrors; - UnicodeString fErrorString; -}; - - - -//------------------------------------------------------------------------------------------- -// -// TestMultithreadedIntl. Test ICU Formatting n a multi-threaded environment +// TestMultithreadedIntl. Test ICU Formatting in a multi-threaded environment // //------------------------------------------------------------------------------------------- @@ -646,8 +420,6 @@ UnicodeString showDifference(const UnicodeString& expected, const UnicodeString& } - - //------------------------------------------------------------------------------------------- // // FormatThreadTest - a thread that tests performing a number of numberformats. @@ -811,7 +583,7 @@ UBool U_CALLCONV isAcceptable(void *, const char *, const char *, const UDataInf //static UMTX gDebugMutex; -class FormatThreadTest : public ThreadWithStatus +class FormatThreadTest : public SimpleThread { public: int fNum; @@ -820,7 +592,7 @@ public: LocalPointer fTSF; FormatThreadTest() // constructor is NOT multithread safe. - : ThreadWithStatus(), + : SimpleThread(), fNum(0), fTraceInfo(0), fTSF(NULL), @@ -917,17 +689,19 @@ public: status = U_ZERO_ERROR; LocalPointer formatter(NumberFormat::createInstance(Locale::getEnglish(),status)); if(U_FAILURE(status)) { - error("Error on NumberFormat::createInstance()."); + IntlTest::gTest->errln("%s:%d Error %s on NumberFormat::createInstance().", + __FILE__, __LINE__, u_errorName(status)); goto cleanupAndReturn; } percentFormatter.adoptInstead(NumberFormat::createPercentInstance(Locale::getFrench(),status)); if(U_FAILURE(status)) { - error("Error on NumberFormat::createPercentInstance()."); + IntlTest::gTest->errln("%s:%d Error %s on NumberFormat::createPercentInstance().", + __FILE__, __LINE__, u_errorName(status)); goto cleanupAndReturn; } - for(iteration = 0;!getError() && iterationgetErrors() && iterationformat(kNumberFormatTestData[whichLine].number, output); if(0 != output.compare(kNumberFormatTestData[whichLine].string)) { - error("format().. expected " + kNumberFormatTestData[whichLine].string + IntlTest::gTest->errln("format().. expected " + kNumberFormatTestData[whichLine].string + " got " + output); goto cleanupAndReturn; } @@ -949,7 +723,7 @@ public: percentFormatter->format(kPercentFormatTestData[whichLine].number, output); if(0 != output.compare(kPercentFormatTestData[whichLine].string)) { - error("percent format().. \n" + + IntlTest::gTest->errln("percent format().. \n" + showDifference(kPercentFormatTestData[whichLine].string,output)); goto cleanupAndReturn; } @@ -1012,20 +786,20 @@ public: if(U_FAILURE(status)) { UnicodeString tmp(u_errorName(status)); - error("Failure on message format, pattern=" + patternToCheck + + IntlTest::gTest->errln("Failure on message format, pattern=" + patternToCheck + ", error = " + tmp); goto cleanupAndReturn; } if(result != expected) { - error("PatternFormat: \n" + showDifference(expected,result)); + IntlTest::gTest->errln("PatternFormat: \n" + showDifference(expected,result)); goto cleanupAndReturn; } // test the Thread Safe Format UnicodeString appendErr; if(!fTSF->doStuff(fNum, appendErr, status)) { - error(appendErr); + IntlTest::gTest->errln(appendErr); goto cleanupAndReturn; } } /* end of for loop */ @@ -1033,7 +807,6 @@ public: cleanupAndReturn: - // while (fNum == 4) {SimpleThread::sleep(10000);} // Force a failure by preventing thread from finishing fTraceInfo = 2; } @@ -1045,10 +818,7 @@ private: void MultithreadTest::TestThreadedIntl() { - int i; UnicodeString theErr; - UBool haveDisplayedInfo[kFormatThreadThreads]; - static const int32_t PATIENCE_SECONDS = 45; UErrorCode threadSafeErr = U_ZERO_ERROR; @@ -1060,60 +830,23 @@ void MultithreadTest::TestThreadedIntl() // logln("Spawning: %d threads * %d iterations each.", kFormatThreadThreads, kFormatThreadIterations); - LocalArray tests(new FormatThreadTest[kFormatThreadThreads]); - for(int32_t j = 0; j < kFormatThreadThreads; j++) { + FormatThreadTest tests[kFormatThreadThreads]; + int32_t j; + for(j = 0; j < UPRV_LENGTHOF(tests); j++) { tests[j].fNum = j; int32_t threadStatus = tests[j].start(); if (threadStatus != 0) { - errln("System Error %d starting thread number %d.", threadStatus, j); - SimpleThread::errorFunc(); + errln("%s:%d System Error %d starting thread number %d.", + __FILE__, __LINE__, threadStatus, j); return; } - haveDisplayedInfo[j] = FALSE; } - // Spin, waiting for the test threads to finish. - UBool stillRunning; - UDate startTime, endTime; - startTime = Calendar::getNow(); - double lastComplaint = 0; - do { - /* Spin until the test threads complete. */ - stillRunning = FALSE; - endTime = Calendar::getNow(); - double elapsedSeconds = ((int32_t)(endTime - startTime)/U_MILLIS_PER_SECOND); - if (elapsedSeconds > PATIENCE_SECONDS) { - errln("Patience exceeded. Test is taking too long."); - return; - } else if((elapsedSeconds-lastComplaint) > 2.0) { - infoln("%.1f seconds elapsed (still waiting..)", elapsedSeconds); - lastComplaint = elapsedSeconds; - } - /* - The following sleep must be here because the *BSD operating systems - have a brain dead thread scheduler. They starve the child threads from - CPU time. - */ - SimpleThread::sleep(1); // yield - for(i=0;icompare(lines[i].buff, lines[i].buflen, lines[prev].buff, lines[prev].buflen); if(cmpres != -cmpres2) { - error(UnicodeString("Compare result not symmetrical on line ") + (i + 1)); + IntlTest::gTest->errln(UnicodeString("Compare result not symmetrical on line ") + (i + 1)); break; } if(cmpres != normalizeResult(skres)) { - error(UnicodeString("Difference between coll->compare and sortkey compare on line ") + (i + 1)); + IntlTest::gTest->errln(UnicodeString("Difference between coll->compare and sortkey compare on line ") + (i + 1)); break; } @@ -1214,7 +947,7 @@ public: // which we do via setting strength=identical. } if(res > 0) { - error(UnicodeString("Line is not greater or equal than previous line, for line ") + (i + 1)); + IntlTest::gTest->errln(UnicodeString("Line is not greater or equal than previous line, for line ") + (i + 1)); break; } } @@ -1330,7 +1063,6 @@ void MultithreadTest::TestCollators() coll->setAttribute(UCOL_STRENGTH, isAtLeastUCA62 ? UCOL_IDENTICAL : UCOL_TERTIARY, status); coll->setAttribute(UCOL_ALTERNATE_HANDLING, UCOL_NON_IGNORABLE, status); - int32_t noSpawned = 0; int32_t spawnResult = 0; LocalArray tests(new CollatorThreadTest[kCollatorThreadThreads]); @@ -1344,60 +1076,16 @@ void MultithreadTest::TestCollators() log("%i ", j); spawnResult = tests[j].start(); if(spawnResult != 0) { - infoln("THREAD INFO: Couldn't spawn more than %i threads", noSpawned); - break; - } - noSpawned++; - } - logln("Spawned all"); - if (noSpawned == 0) { - errln("No threads could be spawned."); - return; - } - - for(int32_t patience = kCollatorThreadPatience;patience > 0; patience --) - { - logln("Waiting..."); - - int32_t i; - int32_t terrs = 0; - int32_t completed =0; - - for(i=0;ierrln("%s:%d Original string is corrupt.", __FILE__, __LINE__); break; } - UnicodeString s1 = *fSharedString; + UnicodeString s1 = *gSharedString; s1 += "cat this"; UnicodeString s2(s1); - UnicodeString s3 = *fSharedString; + UnicodeString s3 = *gSharedString; s2 = s3; s3.truncate(12); s2.truncate(0); } - // while (fNum == 4) {SimpleThread::sleep(10000);} // Force a failure by preventing thread from finishing fTraceInfo = 2; } }; +const UnicodeString *StringThreadTest2::gSharedString = NULL; + // ** The actual test function. + void MultithreadTest::TestString() { - int patience; - int terrs = 0; int j; - - UnicodeString *testString = new UnicodeString("This is the original test string."); - - // Not using LocalArray tests[kStringThreadThreads]; - // because we don't always want to delete them. - // See the comments below the cleanupAndReturn label. - StringThreadTest2 *tests[kStringThreadThreads]; - for(j = 0; j < kStringThreadThreads; j++) { - tests[j] = new StringThreadTest2(testString, j); - } + StringThreadTest2::gSharedString = new UnicodeString("This is the original test string."); + StringThreadTest2 tests[kStringThreadThreads]; logln(UnicodeString("Spawning: ") + kStringThreadThreads + " threads * " + kStringThreadIterations + " iterations each."); for(j = 0; j < kStringThreadThreads; j++) { - int32_t threadStatus = tests[j]->start(); + int32_t threadStatus = tests[j].start(); if (threadStatus != 0) { - errln("System Error %d starting thread number %d.", threadStatus, j); - SimpleThread::errorFunc(); - goto cleanupAndReturn; + errln("%s:%d System Error %d starting thread number %d.", __FILE__, __LINE__, threadStatus, j); } } - for(patience = kStringThreadPatience;patience > 0; patience --) - { - logln("Waiting..."); + // Force a failure, to verify test is functioning and can report errors. + // const_cast(StringThreadTest2::gSharedString)->setCharAt(5, 'x'); - int32_t i; - terrs = 0; - int32_t completed =0; - - for(i=0;iisRunning() == FALSE) - { - completed++; - - logln(UnicodeString("Test #") + i + " is complete.. "); - - UnicodeString theErr; - if(tests[i]->getError(theErr)) - { - terrs++; - errln(UnicodeString("#") + i + ": " + theErr); - } - // print out the error, too, if any. - } - } - - if(completed == kStringThreadThreads) - { - logln("Done!"); - if(terrs) { - errln("There were errors."); - } - break; - } - - SimpleThread::sleep(900); + for(j=0; j 0) { - SimpleThread::errorFunc(); - } - -cleanupAndReturn: - if (terrs == 0) { - /* - Don't clean up if there are errors. This prevents crashes if the - threads are still running and using this data. This will only happen - if there is an error with the test, ICU, or the machine is too slow. - It's better to leak than crash. - */ - for(j = 0; j < kStringThreadThreads; j++) { - delete tests[j]; - } - delete testString; - } + delete StringThreadTest2::gSharedString; + StringThreadTest2::gSharedString = NULL; } +// // Test for ticket #10673, race in cache code in AnyTransliterator. // It's difficult to make the original unsafe code actually fail, but // this test will fairly reliably take the code path for races in // populating the cache. +// #if !UCONFIG_NO_TRANSLITERATION +Transliterator *gSharedTranslit = NULL; class TxThread: public SimpleThread { - private: - Transliterator *fSharedTranslit; public: - UBool fSuccess; - TxThread(Transliterator *tx) : fSharedTranslit(tx), fSuccess(FALSE) {}; + TxThread() {}; ~TxThread(); void run(); }; @@ -1568,8 +1193,11 @@ TxThread::~TxThread() {} void TxThread::run() { UnicodeString greekString("\\u03B4\\u03B9\\u03B1\\u03C6\\u03BF\\u03C1\\u03B5\\u03C4\\u03B9\\u03BA\\u03BF\\u03CD\\u03C2"); greekString = greekString.unescape(); - fSharedTranslit->transliterate(greekString); - fSuccess = greekString[0] == 0x64; // 'd'. The whole transliterated string is "diaphoretikous" (accented u). + gSharedTranslit->transliterate(greekString); + if (greekString[0] != 0x64) // 'd'. The whole transliterated string is "diaphoretikous" (accented u). + { + IntlTest::gTest->errln("%s:%d Transliteration failed.", __FILE__, __LINE__); + } } #endif @@ -1582,47 +1210,22 @@ void MultithreadTest::TestAnyTranslit() { dataerrln("File %s, Line %d: Error, status = %s", __FILE__, __LINE__, u_errorName(status)); return; } - TxThread * threads[4]; + gSharedTranslit = tx.getAlias(); + TxThread threads[4]; int32_t i; - for (i=0; i<4; i++) { - threads[i] = new TxThread(tx.getAlias()); - } - for (i=0; i<4; i++) { - threads[i]->start(); - } - int32_t patience = 100; - UBool success; - UBool someThreadRunning; - do { - someThreadRunning = FALSE; - success = TRUE; - for (i=0; i<4; i++) { - if (threads[i]->isRunning()) { - someThreadRunning = TRUE; - SimpleThread::sleep(10); - break; - } else { - if (threads[i]->fSuccess == FALSE) { - success = FALSE; - } - } - } - } while (someThreadRunning && --patience > 0); - - if (patience <= 0) { - errln("File %s, Line %d: Error, one or more threads did not complete.", __FILE__, __LINE__); - } - if (success == FALSE) { - errln("File %s, Line %d: Error, transliteration result incorrect.", __FILE__, __LINE__); + for (i=0; ierrln("File %s, Line %d: Error, gStartedThreads = %d, gFinishedThreads = %d", + IntlTest::gTest->errln("File %s, Line %d: Error, gStartedThreads = %d, gFinishedThreads = %d", __FILE__, __LINE__, gStartedThreads, gFinishedThreads); } umtx_condWait(&gCTConditionVar, &gCTMutex); @@ -1677,7 +1279,6 @@ void CondThread::run() { } void MultithreadTest::TestConditionVariables() { - gThisTest = this; gStartedThreads = 0; gFinishedThreads = 0; int i; @@ -1703,12 +1304,21 @@ void MultithreadTest::TestConditionVariables() { if (!threads[i]->fFinished) { errln("File %s, Line %d: Error, threads[%d]->fFinished == false", __FILE__, __LINE__, i); } + } + + for (i=0; ijoin(); delete threads[i]; } } + +// +// Unified Cache Test +// + static const char *gCacheLocales[] = {"en_US", "en_GB", "fr_FR", "fr"}; -static int32_t gObjectsCreated = 0; +static int32_t gObjectsCreated = 0; // protected by gCTMutex static const int32_t CACHE_LOAD = 3; class UCTMultiThreadItem : public SharedObject { @@ -1727,23 +1337,33 @@ U_NAMESPACE_BEGIN template<> U_EXPORT const UCTMultiThreadItem *LocaleCacheKey::createObject( const void * /*unused*/, UErrorCode & /* status */) const { - // Since multiple threads are hitting the cache for the first time, - // no objects should be created yet. umtx_lock(&gCTMutex); - if (gObjectsCreated != 0) { - gThisTest->errln("Expected no objects to be created yet."); + bool firstObject = (gObjectsCreated == 0); + if (firstObject) { + // Force the first object creation that comes through to wait + // until other have completed. Verifies that cache doesn't + // deadlock when a creation is slow. + + // Note that gObjectsCreated needs to be incremeneted from 0 to 1 + // early, to keep subsequent threads from entering this path. + gObjectsCreated = 1; + while (gObjectsCreated < 3) { + umtx_condWait(&gCTConditionVar, &gCTMutex); + } } umtx_unlock(&gCTMutex); - // Big, expensive object that takes 1 second to create. - SimpleThread::sleep(1000); - - // Log that we created an object. - umtx_lock(&gCTMutex); - ++gObjectsCreated; - umtx_unlock(&gCTMutex); UCTMultiThreadItem *result = new UCTMultiThreadItem(fLoc.getName()); result->addRef(); + + // Log that we created an object. The first object was already counted, + // don't do it again. + umtx_lock(&gCTMutex); + if (!firstObject) { + gObjectsCreated += 1; + } + umtx_condBroadcast(&gCTConditionVar); + umtx_unlock(&gCTMutex); return result; } @@ -1765,15 +1385,9 @@ void UnifiedCacheThread::run() { cache->get(LocaleCacheKey(fLoc), item, status); U_ASSERT(item != NULL); if (uprv_strcmp(fLoc, item->value)) { - gThisTest->errln("Expected %s, got %s", fLoc, item->value); + IntlTest::gTest->errln("Expected %s, got %s", fLoc, item->value); } item->removeRef(); - - // Mark this thread as finished - umtx_lock(&gCTMutex); - ++gFinishedThreads; - umtx_condBroadcast(&gCTConditionVar); - umtx_unlock(&gCTMutex); } void MultithreadTest::TestUnifiedCache() { @@ -1781,7 +1395,6 @@ void MultithreadTest::TestUnifiedCache() { const UnifiedCache *cache = UnifiedCache::getInstance(status); U_ASSERT(cache != NULL); cache->flush(); - gThisTest = this; gFinishedThreads = 0; gObjectsCreated = 0; @@ -1792,14 +1405,13 @@ void MultithreadTest::TestUnifiedCache() { threads[i][j]->start(); } } - // Wait on all the threads to complete verify that LENGTHOF(gCacheLocales) - // objects were created. - umtx_lock(&gCTMutex); - while (gFinishedThreads < CACHE_LOAD*UPRV_LENGTHOF(gCacheLocales)) { - umtx_condWait(&gCTConditionVar, &gCTMutex); + + for (int32_t i=0; ijoin(); + } } assertEquals("Objects created", UPRV_LENGTHOF(gCacheLocales), gObjectsCreated); - umtx_unlock(&gCTMutex); // clean up threads for (int32_t i=0; ijoin(); + } + if (data.numDone != nLocales) { + errln("%s:%d data.numDone = %d, nLocales = %d", __FILE__, __LINE__, data.numDone, nLocales); } for (i = 0; i < nThreads; i++) { @@ -708,7 +699,6 @@ TimeZoneFormatTest::TestTimeRoundTrip(void) { } delete [] threads; -#endif UDate total = 0; logln("### Elapsed time by patterns ###"); for (int32_t i = 0; i < NUM_PATTERNS; i++) {