Try to increase robustness of thread pool.

This commit is contained in:
castano 2012-05-03 17:04:53 +00:00
parent 803301ea89
commit c784e03be5
3 changed files with 36 additions and 8 deletions

View File

@ -48,7 +48,6 @@ namespace nv {
nvDebugCheck((intptr_t(ptr) & 3) == 0);
#if POSH_CPU_X86 || POSH_CPU_X86_64
nvCompilerReadBarrier();
uint32 ret = *ptr; // on x86, loads are Acquire
nvCompilerReadBarrier();
return ret;
@ -73,7 +72,6 @@ namespace nv {
nvDebugCheck((intptr_t(&value) & 3) == 0);
#if POSH_CPU_X86 || POSH_CPU_X86_64
nvCompilerWriteBarrier();
*ptr = value; // on x86, stores are Release
nvCompilerWriteBarrier();
#elif POSH_CPU_STRONGARM
@ -87,6 +85,27 @@ namespace nv {
}
template <typename T>
inline void storeReleasePointer(volatile T * pTo, T from)
{
NV_COMPILER_CHECK(sizeof(T) == sizeof(intptr_t));
nvDebugCheck((((intptr_t)pTo) % sizeof(intptr_t)) == 0);
nvDebugCheck((((intptr_t)&from) % sizeof(intptr_t)) == 0);
nvCompilerWriteBarrier();
*pTo = from; // on x86, stores are Release
}
template <typename T>
inline T loadAcquirePointer(volatile T * ptr)
{
NV_COMPILER_CHECK(sizeof(T) == sizeof(intptr_t));
nvDebugCheck((((intptr_t)ptr) % sizeof(intptr_t)) == 0);
T ret = *ptr; // on x86, loads are Acquire
nvCompilerReadBarrier();
return ret;
}
// Atomics. @@ Assuming sequential memory order?
#if NV_CC_MSVC

View File

@ -129,6 +129,10 @@ bool Thread::isRunning () const
DWORD result = WaitForMultipleObjects(count, handles, TRUE, INFINITE);
for (uint i = 0; i < count; i++) {
CloseHandle (threads->p->thread);
threads->p->thread = 0;
}
delete [] handles;
#else*/

View File

@ -3,6 +3,7 @@
#include "ThreadPool.h"
#include "Mutex.h"
#include "Thread.h"
#include "Atomic.h"
#include "nvcore/Utils.h"
@ -49,17 +50,19 @@ AutoPtr<ThreadPool> s_pool;
/*static*/ void ThreadPool::workerFunc(void * arg) {
uint i = toU32((uintptr_t)arg); // This is OK, because workerCount should always be <<< 2^32
uint i = toU32((uintptr_t)arg); // This is OK, because workerCount should always be much smaller than 2^32
while(true)
{
s_pool->startEvents[i].wait();
if (s_pool->func == NULL) {
return; // @@ should we post finish event anyway?
nv::ThreadFunc * func = loadAcquirePointer(&s_pool->func);
if (func == NULL) {
return;
}
s_pool->func(s_pool->arg);
func(s_pool->arg);
s_pool->finishEvents[i].post();
}
@ -102,11 +105,13 @@ void ThreadPool::start(ThreadFunc * func, void * arg)
wait();
// Set our desired function.
this->func = func;
this->arg = arg;
storeReleasePointer(&this->func, func);
storeReleasePointer(&this->arg, arg);
allIdle = false;
nvCompilerWriteBarrier();
// Resume threads.
Event::post(startEvents, workerCount);
}