openspades/Sources/Core/ConcurrentDispatch.cpp

386 lines
8.4 KiB
C++
Raw Normal View History

2013-08-29 11:45:22 +09:00
/*
Copyright (c) 2013 yvt
2016-11-20 19:13:00 +09:00
2013-08-29 11:45:22 +09:00
This file is part of OpenSpades.
2016-11-20 19:13:00 +09:00
2013-08-29 11:45:22 +09:00
OpenSpades is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
2016-11-20 19:13:00 +09:00
2013-08-29 11:45:22 +09:00
OpenSpades is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
2016-11-20 19:13:00 +09:00
2013-08-29 11:45:22 +09:00
You should have received a copy of the GNU General Public License
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
2016-11-20 19:13:00 +09:00
2013-08-29 11:45:22 +09:00
*/
2013-08-18 16:18:06 +09:00
#include <list>
#include <sys/types.h>
#include <memory>
#include <Imports/SDL.h>
2013-08-18 16:18:06 +09:00
#if defined(__APPLE__)
#include <sys/sysctl.h>
2013-10-07 20:22:06 +09:00
#else
2013-08-18 16:18:06 +09:00
#if defined(WIN32)
#include <windows.h>
#else
2013-09-24 23:07:53 +02:00
#ifndef _MSC_VER
2013-08-18 16:18:06 +09:00
#include <unistd.h>
2013-08-28 00:45:23 +02:00
#endif
#if defined(__linux__)
2013-09-26 12:43:50 -04:00
#include <sys/sysinfo.h>
#endif
2013-10-07 20:22:06 +09:00
#endif
#endif
2013-10-07 20:22:06 +09:00
#include "ConcurrentDispatch.h"
#include "Debug.h"
#include "Exception.h"
#include "Settings.h"
#include "Thread.h"
#include <OpenSpades.h>
2013-08-18 16:18:06 +09:00
#include "ThreadLocalStorage.h"
2016-11-19 21:03:51 +09:00
DEFINE_SPADES_SETTING(core_numDispatchQueueThreads, "auto");
2013-08-18 16:18:06 +09:00
static int GetNumCores() {
#ifdef WIN32
2016-11-20 19:13:00 +09:00
SYSTEM_INFO sysinfo;
GetSystemInfo(&sysinfo);
return sysinfo.dwNumberOfProcessors;
2013-08-18 16:18:06 +09:00
#elif defined(__APPLE__)
2016-11-20 19:13:00 +09:00
int nm[2];
size_t len = 4;
uint32_t count;
nm[0] = CTL_HW;
nm[1] = HW_AVAILCPU;
2016-11-20 19:13:00 +09:00
sysctl(nm, 2, &count, &len, NULL, 0);
if (count < 1) {
2016-11-20 19:13:00 +09:00
nm[1] = HW_NCPU;
sysctl(nm, 2, &count, &len, NULL, 0);
if (count < 1) {
count = 1;
}
2016-11-20 19:13:00 +09:00
}
return count;
#elif defined(__linux__)
2016-11-20 19:13:00 +09:00
return get_nprocs();
2013-08-18 16:18:06 +09:00
#else
2016-11-20 19:13:00 +09:00
return sysconf(_SC_NPROCESSORS_ONLN);
2013-08-18 16:18:06 +09:00
#endif
}
namespace spades {
2016-11-20 19:13:00 +09:00
struct SyncQueueEntry {
2013-08-18 16:18:06 +09:00
SDL_cond *doneCond;
SDL_mutex *doneMutex;
ConcurrentDispatch *dispatch;
bool done;
bool released;
2016-11-20 19:13:00 +09:00
SyncQueueEntry(ConcurrentDispatch *disp)
: doneCond(SDL_CreateCond()),
doneMutex(SDL_CreateMutex()),
dispatch(disp),
done(false),
released(false) {}
2013-08-18 16:18:06 +09:00
~SyncQueueEntry() {
SDL_DestroyCond(doneCond);
SDL_DestroyMutex(doneMutex);
if (released) {
2013-08-18 16:18:06 +09:00
dispatch->entry = NULL;
delete dispatch;
}
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void Done() {
SDL_LockMutex(doneMutex);
done = true;
SDL_CondBroadcast(doneCond);
if (released) {
2013-08-18 16:18:06 +09:00
delete this;
return;
}
SDL_UnlockMutex(doneMutex);
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void Release() {
SDL_LockMutex(doneMutex);
released = true;
if (done) {
2013-08-18 16:18:06 +09:00
delete this;
return;
}
SDL_UnlockMutex(doneMutex);
}
2016-11-20 19:13:00 +09:00
void Join() {
2013-08-18 16:18:06 +09:00
SDL_LockMutex(doneMutex);
while (!done) {
2013-08-18 16:18:06 +09:00
SDL_CondWait(doneCond, doneMutex);
}
SDL_UnlockMutex(doneMutex);
}
};
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
class SynchronizedQueue {
std::list<SyncQueueEntry *> entries;
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
SDL_cond *pushCond;
SDL_mutex *pushMutex;
2013-08-18 16:18:06 +09:00
public:
SynchronizedQueue() {
pushMutex = SDL_CreateMutex();
pushCond = SDL_CreateCond();
}
~SynchronizedQueue() {
SDL_DestroyMutex(pushMutex);
SDL_DestroyCond(pushCond);
}
2016-11-20 19:13:00 +09:00
void Push(SyncQueueEntry *entry) {
2013-08-18 16:18:06 +09:00
SDL_LockMutex(pushMutex);
try {
2013-08-18 16:18:06 +09:00
entries.push_back(entry);
} catch (...) {
2013-08-18 16:18:06 +09:00
SDL_UnlockMutex(pushMutex);
throw;
}
SDL_CondSignal(pushCond);
SDL_UnlockMutex(pushMutex);
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
SyncQueueEntry *Wait() {
SDL_LockMutex(pushMutex);
while (entries.empty()) {
2013-08-18 16:18:06 +09:00
SDL_CondWait(pushCond, pushMutex);
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
SyncQueueEntry *ent = entries.front();
entries.pop_front();
SDL_UnlockMutex(pushMutex);
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
return ent;
}
2016-11-20 19:13:00 +09:00
SyncQueueEntry *Poll() {
2013-08-18 16:18:06 +09:00
SDL_LockMutex(pushMutex);
if (!entries.empty()) {
2013-08-18 16:18:06 +09:00
SyncQueueEntry *ent = entries.front();
entries.pop_front();
SDL_UnlockMutex(pushMutex);
return ent;
}
SDL_UnlockMutex(pushMutex);
return NULL;
}
};
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
static AutoDeletedThreadLocalStorage<DispatchQueue> threadQueue("threadDispatchQueue");
static DispatchQueue *sdlQueue = NULL;
2016-11-20 19:13:00 +09:00
DispatchQueue::DispatchQueue() {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
internal = new SynchronizedQueue();
}
DispatchQueue::~DispatchQueue() {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
delete internal;
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
DispatchQueue *DispatchQueue::GetThreadQueue() {
SPADES_MARK_FUNCTION();
DispatchQueue *q = threadQueue;
if (!q) {
2013-08-18 16:18:06 +09:00
q = new DispatchQueue();
threadQueue = q;
}
return q;
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void DispatchQueue::ProcessQueue() {
SPADES_MARK_FUNCTION();
SyncQueueEntry *ent;
while ((ent = internal->Poll()) != NULL) {
2013-08-18 16:18:06 +09:00
ent->dispatch->Execute();
}
Thread::CleanupExitedThreads();
2013-08-18 16:18:06 +09:00
}
2016-11-20 19:13:00 +09:00
void DispatchQueue::EnterEventLoop() noexcept {
while (true) {
2013-08-18 16:18:06 +09:00
SyncQueueEntry *ent = internal->Wait();
ent->dispatch->ExecuteProtected();
}
}
2016-11-20 19:13:00 +09:00
void DispatchQueue::MarkSDLVideoThread() { sdlQueue = this; }
2016-11-20 19:13:00 +09:00
namespace {
struct GlobalDispatchThreadPool {
SynchronizedQueue globalQueue;
std::vector<std::unique_ptr<DispatchThread>> threads;
GlobalDispatchThreadPool();
~GlobalDispatchThreadPool();
};
std::unique_ptr<GlobalDispatchThreadPool> globalThreadPool;
}
// Cannot define this in an anonymous namespace since this is referred to by
// `ConcurrentDispatch`'s friend class declaration
class DispatchThread : public Thread {
2013-08-18 16:18:06 +09:00
public:
DispatchThread(GlobalDispatchThreadPool &pool) : pool{pool} {
}
void Run() noexcept override {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
while (true) {
SyncQueueEntry *ent = pool.globalQueue.Wait();
if (ent->dispatch == nullptr) {
// Exit request
return;
}
2013-08-18 16:18:06 +09:00
ent->dispatch->ExecuteProtected();
}
}
private:
GlobalDispatchThreadPool &pool;
2013-08-18 16:18:06 +09:00
};
GlobalDispatchThreadPool::GlobalDispatchThreadPool() {
int cnt = GetNumCores();
if (!("auto" == core_numDispatchQueueThreads)) {
cnt = core_numDispatchQueueThreads;
}
2016-11-20 19:13:00 +09:00
SPLog("Creating %d dispatch thread(s)", cnt);
for (int i = 0; i < cnt; i++) {
DispatchThread *t = new DispatchThread(*this);
threads.emplace_back(t);
t->Start();
}
}
GlobalDispatchThreadPool::~GlobalDispatchThreadPool() {
// From this point no more dispatches are accepted
for (const auto &_: threads) {
(void) _; // unused!
globalQueue.Push(new SyncQueueEntry(nullptr));
}
// Destroy all `DispatchThread`s
threads.clear();
// When `Thread`s' destructors are called, they'll wait until the execution of
// the thread is completed. So at this point all `DispatchThread`s have finished execution
// and there'll be no threads accessing the `globalQueue`, thus it's safe to delete it.
}
2016-11-20 19:13:00 +09:00
ConcurrentDispatch::ConcurrentDispatch() : entry(NULL), runnable(NULL) {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
}
ConcurrentDispatch::ConcurrentDispatch(std::string name)
2017-11-18 21:15:07 +00:00
: name(name), entry(NULL), runnable(NULL) {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
}
2016-11-20 19:13:00 +09:00
ConcurrentDispatch::~ConcurrentDispatch() {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
Join();
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void ConcurrentDispatch::Execute() {
SPADES_MARK_FUNCTION();
SyncQueueEntry *ent = entry;
if (!ent) {
2013-08-18 16:18:06 +09:00
SPRaise("Attempted to execute dispatch '%s' without entry", name.c_str());
}
try {
2013-08-18 16:18:06 +09:00
Run();
} catch (...) {
2013-08-18 16:18:06 +09:00
ent->Done();
throw;
}
ent->Done();
}
2016-11-20 19:13:00 +09:00
void ConcurrentDispatch::ExecuteProtected() noexcept {
try {
2013-08-18 16:18:06 +09:00
Execute();
} catch (const std::exception &ex) {
2013-08-18 16:18:06 +09:00
fprintf(stderr, "-- UNHANDLED CONCURRENT DISPATCH EXCEPTION ---\n");
fprintf(stderr, "%s\n", ex.what());
} catch (...) {
2013-08-18 16:18:06 +09:00
fprintf(stderr, "-- UNHANDLED CONCURRENT DISPATCH EXCEPTION ---\n");
fprintf(stderr, "(no information provided)\n");
}
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void ConcurrentDispatch::Start() {
SPADES_MARK_FUNCTION();
if (entry) {
2013-08-18 16:18:06 +09:00
SPRaise("Attempted to start dispatch '%s' when it's already started", name.c_str());
} else {
// should we atomically initialize globalThreadPool? maybe not
if (!globalThreadPool) {
globalThreadPool.reset(new GlobalDispatchThreadPool());
2013-08-18 16:18:06 +09:00
}
entry = new SyncQueueEntry(this);
globalThreadPool->globalQueue.Push(entry);
2013-08-18 16:18:06 +09:00
}
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void ConcurrentDispatch::StartOn(DispatchQueue *queue) {
SPADES_MARK_FUNCTION();
if (entry) {
2013-08-18 16:18:06 +09:00
SPRaise("Attempted to start dispatch '%s' when it's already started", name.c_str());
} else {
2013-08-18 16:18:06 +09:00
entry = new SyncQueueEntry(this);
queue->internal->Push(entry);
2016-11-20 19:13:00 +09:00
if (queue == sdlQueue) {
2013-08-18 16:18:06 +09:00
SDL_Event evt;
memset(&evt, 0, sizeof(evt));
evt.type = SDL_USEREVENT;
SDL_PushEvent(&evt);
}
}
}
2016-11-20 19:13:00 +09:00
2013-08-18 16:18:06 +09:00
void ConcurrentDispatch::Join() {
SPADES_MARK_FUNCTION();
if (!entry) {
} else {
2013-08-18 16:18:06 +09:00
entry->Join();
delete entry;
entry = NULL;
}
}
2016-11-20 19:13:00 +09:00
void ConcurrentDispatch::Release() {
2013-08-18 16:18:06 +09:00
SPADES_MARK_FUNCTION();
if (entry) {
SyncQueueEntry *ent = entry;
2013-08-18 16:18:06 +09:00
ent->Release();
}
}
2016-11-20 19:13:00 +09:00
void ConcurrentDispatch::Run() {
if (runnable)
2013-08-18 16:18:06 +09:00
runnable->Run();
}
}