2015-08-21 18:09:53 +00:00
|
|
|
/*
|
|
|
|
Bullet Continuous Collision Detection and Physics Library
|
|
|
|
Copyright (c) 2010 Erwin Coumans http://bulletphysics.org
|
|
|
|
|
|
|
|
This software is provided 'as-is', without any express or implied warranty.
|
|
|
|
In no event will the authors be held liable for any damages arising from the use of this software.
|
|
|
|
Permission is granted to anyone to use this software for any purpose,
|
|
|
|
including commercial applications, and to alter it and redistribute it freely,
|
|
|
|
subject to the following restrictions:
|
|
|
|
|
|
|
|
1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
|
|
|
|
2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
|
|
|
|
3. This notice may not be removed or altered from any source distribution.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/// ThreadingDemo shows how to use the cross platform thread support interface.
|
|
|
|
/// You can start threads and perform a blocking wait for completion
|
|
|
|
/// Under Windows it uses Win32 Threads. On Mac and Linux it uses pthreads. On PlayStation 3 Cell SPU it uses SPURS.
|
|
|
|
|
|
|
|
/// June 2010
|
|
|
|
/// New: critical section/barriers and non-blocking polling for completion
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
void SampleThreadFunc(void* userPtr, void* lsMemory);
|
|
|
|
void* SamplelsMemoryFunc();
|
2019-11-21 01:06:05 +00:00
|
|
|
void SamplelsMemoryReleaseFunc(void* ptr);
|
2015-08-21 18:09:53 +00:00
|
|
|
#include <stdio.h>
|
|
|
|
//#include "BulletMultiThreaded/PlatformDefinitions.h"
|
|
|
|
|
|
|
|
#ifndef _WIN32
|
|
|
|
#include "b3PosixThreadSupport.h"
|
|
|
|
|
|
|
|
b3ThreadSupportInterface* createThreadSupport(int numThreads)
|
|
|
|
{
|
|
|
|
b3PosixThreadSupport::ThreadConstructionInfo constructionInfo("testThreads",
|
2018-09-23 21:17:31 +00:00
|
|
|
SampleThreadFunc,
|
|
|
|
SamplelsMemoryFunc,
|
2019-11-21 01:06:05 +00:00
|
|
|
SamplelsMemoryReleaseFunc,
|
2018-09-23 21:17:31 +00:00
|
|
|
numThreads);
|
|
|
|
b3ThreadSupportInterface* threadSupport = new b3PosixThreadSupport(constructionInfo);
|
2015-08-21 18:09:53 +00:00
|
|
|
|
|
|
|
return threadSupport;
|
|
|
|
}
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
#elif defined(_WIN32)
|
2015-08-21 18:09:53 +00:00
|
|
|
#include "b3Win32ThreadSupport.h"
|
|
|
|
|
|
|
|
b3ThreadSupportInterface* createThreadSupport(int numThreads)
|
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
b3Win32ThreadSupport::Win32ThreadConstructionInfo threadConstructionInfo("testThreads", SampleThreadFunc, SamplelsMemoryFunc, numThreads);
|
2015-08-21 18:09:53 +00:00
|
|
|
b3Win32ThreadSupport* threadSupport = new b3Win32ThreadSupport(threadConstructionInfo);
|
|
|
|
return threadSupport;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
struct SampleArgs
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
SampleArgs()
|
2018-09-23 21:17:31 +00:00
|
|
|
: m_fakeWork(1)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
b3CriticalSection* m_cs;
|
|
|
|
float m_fakeWork;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct SampleThreadLocalStorage
|
|
|
|
{
|
|
|
|
int threadId;
|
|
|
|
};
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
void SampleThreadFunc(void* userPtr, void* lsMemory)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
printf("thread started\n");
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
SampleThreadLocalStorage* localStorage = (SampleThreadLocalStorage*)lsMemory;
|
2015-08-21 18:09:53 +00:00
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
SampleArgs* args = (SampleArgs*)userPtr;
|
2015-08-21 18:09:53 +00:00
|
|
|
int workLeft = true;
|
|
|
|
while (workLeft)
|
|
|
|
{
|
|
|
|
args->m_cs->lock();
|
|
|
|
int count = args->m_cs->getSharedParam(0);
|
2018-09-23 21:17:31 +00:00
|
|
|
args->m_cs->setSharedParam(0, count - 1);
|
2015-08-21 18:09:53 +00:00
|
|
|
args->m_cs->unlock();
|
2018-09-23 21:17:31 +00:00
|
|
|
if (count > 0)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
printf("thread %d processed number %d\n", localStorage->threadId, count);
|
2015-08-21 18:09:53 +00:00
|
|
|
}
|
|
|
|
//do some fake work
|
2018-09-23 21:17:31 +00:00
|
|
|
for (int i = 0; i < 1000000; i++)
|
|
|
|
args->m_fakeWork = b3Scalar(1.21) * args->m_fakeWork;
|
|
|
|
workLeft = count > 0;
|
2015-08-21 18:09:53 +00:00
|
|
|
}
|
|
|
|
printf("finished\n");
|
|
|
|
//do nothing
|
|
|
|
}
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
void* SamplelsMemoryFunc()
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
//don't create local store memory, just return 0
|
|
|
|
return new SampleThreadLocalStorage;
|
|
|
|
}
|
|
|
|
|
2019-11-21 01:06:05 +00:00
|
|
|
void SamplelsMemoryReleaseFunc(void* ptr)
|
|
|
|
{
|
|
|
|
SampleThreadLocalStorage* p = (SampleThreadLocalStorage*)ptr;
|
|
|
|
delete p;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
int main(int argc, char** argv)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
int numThreads = 8;
|
|
|
|
|
|
|
|
b3ThreadSupportInterface* threadSupport = createThreadSupport(numThreads);
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
for (int i = 0; i < threadSupport->getNumTasks(); i++)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
SampleThreadLocalStorage* storage = (SampleThreadLocalStorage*)threadSupport->getThreadLocalMemory(i);
|
|
|
|
b3Assert(storage);
|
|
|
|
storage->threadId = i;
|
|
|
|
}
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
SampleArgs args;
|
2015-08-21 18:09:53 +00:00
|
|
|
args.m_cs = threadSupport->createCriticalSection();
|
2018-09-23 21:17:31 +00:00
|
|
|
args.m_cs->setSharedParam(0, 100);
|
2015-08-21 18:09:53 +00:00
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
int arg0, arg1;
|
2015-08-21 18:09:53 +00:00
|
|
|
int i;
|
2018-09-23 21:17:31 +00:00
|
|
|
for (i = 0; i < numThreads; i++)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
threadSupport->runTask(B3_THREAD_SCHEDULE_TASK, (void*)&args, i);
|
2015-08-21 18:09:53 +00:00
|
|
|
}
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
bool blockingWait = false;
|
2015-08-21 18:09:53 +00:00
|
|
|
if (blockingWait)
|
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
for (i = 0; i < numThreads; i++)
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
threadSupport->waitForResponse(&arg0, &arg1);
|
|
|
|
printf("finished waiting for response: %d %d\n", arg0, arg1);
|
2015-08-21 18:09:53 +00:00
|
|
|
}
|
2018-09-23 21:17:31 +00:00
|
|
|
}
|
|
|
|
else
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
int numActiveThreads = numThreads;
|
|
|
|
while (numActiveThreads)
|
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
if (threadSupport->isTaskCompleted(&arg0, &arg1, 0))
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
|
|
|
numActiveThreads--;
|
2018-09-23 21:17:31 +00:00
|
|
|
printf("numActiveThreads = %d\n", numActiveThreads);
|
|
|
|
}
|
|
|
|
else
|
2015-08-21 18:09:53 +00:00
|
|
|
{
|
2018-09-23 21:17:31 +00:00
|
|
|
// printf("polling..");
|
2015-08-21 18:09:53 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-09-23 21:17:31 +00:00
|
|
|
printf("stopping threads\n");
|
2015-08-21 18:09:53 +00:00
|
|
|
|
|
|
|
delete threadSupport;
|
|
|
|
printf("Press ENTER to quit\n");
|
2015-08-21 22:18:18 +00:00
|
|
|
//getchar();
|
2015-08-21 18:09:53 +00:00
|
|
|
return 0;
|
|
|
|
}
|