2013-08-29 11:45:22 +09:00
|
|
|
/*
|
|
|
|
Copyright (c) 2013 yvt
|
|
|
|
|
|
|
|
This file is part of OpenSpades.
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
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.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with OpenSpades. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
#include "Thread.h"
|
|
|
|
#include "../Imports/SDL.h"
|
|
|
|
#include "AutoLocker.h"
|
|
|
|
#include "Debug.h"
|
|
|
|
#include "ThreadLocalStorage.h"
|
2013-11-27 00:46:12 +09:00
|
|
|
#include "ConcurrentDispatch.h"
|
2013-12-09 17:36:05 +09:00
|
|
|
#include <typeinfo>
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
namespace spades {
|
|
|
|
|
|
|
|
Thread::Thread():
|
2013-11-23 23:08:40 +09:00
|
|
|
runnable(NULL),
|
|
|
|
autoDelete(false){
|
2013-08-18 16:18:06 +09:00
|
|
|
threadInfo = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread::Thread(IRunnable *r):
|
2013-11-27 00:46:12 +09:00
|
|
|
runnable(r),
|
|
|
|
autoDelete(false){
|
2013-08-18 16:18:06 +09:00
|
|
|
threadInfo = NULL;
|
2013-11-27 00:46:12 +09:00
|
|
|
threadId = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
class ThreadCleanuper: public ConcurrentDispatch {
|
|
|
|
std::vector<SDL_Thread *> threads;
|
|
|
|
Mutex mutex;
|
|
|
|
public:
|
|
|
|
void Add(SDL_Thread *thread) {
|
|
|
|
AutoLocker locker(&mutex);
|
|
|
|
threads.push_back(thread);
|
|
|
|
}
|
|
|
|
void Cleanup() {
|
|
|
|
AutoLocker locker(&mutex);
|
|
|
|
for(size_t i = 0; i < threads.size(); i++)
|
|
|
|
SDL_WaitThread(threads[i], NULL);
|
|
|
|
threads.clear();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
static ThreadCleanuper *cleanuper;
|
|
|
|
|
|
|
|
void Thread::InitThreadSystem() {
|
|
|
|
cleanuper = new ThreadCleanuper();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Thread::CleanupExitedThreads() {
|
|
|
|
cleanuper->Cleanup();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
Thread::~Thread() {
|
2013-11-27 00:46:12 +09:00
|
|
|
SDL_Thread *th = NULL;
|
|
|
|
{
|
|
|
|
AutoLocker locker(&lock);
|
|
|
|
th = (SDL_Thread *)threadInfo;
|
|
|
|
if(!th)
|
|
|
|
return;
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
|
2013-11-27 00:46:12 +09:00
|
|
|
// we have to ensure thread handle is destroyed.
|
|
|
|
if(SDL_ThreadID() == threadId) {
|
|
|
|
// thread is deleting itself.
|
|
|
|
// SDL_WaitThread would cause deadlock.
|
|
|
|
cleanuper->Add(th);
|
|
|
|
}else{
|
|
|
|
SDL_WaitThread(th, NULL);
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void Thread::Start() {
|
|
|
|
AutoLocker locker(&lock);
|
|
|
|
if(threadInfo)
|
|
|
|
return;
|
|
|
|
|
2013-12-09 17:36:05 +09:00
|
|
|
const std::type_info& info = typeid(*this);
|
|
|
|
const char *name = info.name();
|
|
|
|
if(name == nullptr)
|
|
|
|
name = "(null)";
|
|
|
|
|
2013-11-27 00:46:12 +09:00
|
|
|
threadId = 0;
|
2013-12-09 17:36:05 +09:00
|
|
|
threadInfo = SDL_CreateThread(InternalRunner, name, this);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void Thread::Join() {
|
2013-11-27 00:46:12 +09:00
|
|
|
if(autoDelete) {
|
|
|
|
SPRaise("Attempted join a thread that is marked for auto deletion.");
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
SDL_Thread *th = NULL;
|
|
|
|
{
|
|
|
|
AutoLocker locker(&lock);
|
|
|
|
th = (SDL_Thread *)threadInfo;
|
|
|
|
if(!th)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
SDL_WaitThread(th, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Thread::IsAlive() {
|
2013-11-27 00:46:12 +09:00
|
|
|
if(autoDelete) {
|
|
|
|
SPRaise("Attempted query the state of a thread that is marked for auto deletion.");
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
return threadInfo != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Thread::InternalRunner(void *th) {
|
2013-11-27 00:46:12 +09:00
|
|
|
Thread *self = (Thread *)th;
|
|
|
|
self->threadId = SDL_ThreadID();
|
2013-08-18 16:18:06 +09:00
|
|
|
try{
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
self->Run();
|
|
|
|
self->Quited();
|
|
|
|
}catch(const std::exception& ex){
|
|
|
|
printf("Exception in thread:\n%s\n", ex.what());
|
|
|
|
throw;
|
|
|
|
}catch(...){
|
|
|
|
printf("Exception in thread: unknown\n");
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
|
|
|
|
// call TLSs' destructors
|
|
|
|
::spades::ThreadExiting();
|
|
|
|
|
|
|
|
::spades::reflection::Backtrace::ThreadExiting();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Thread::Quited() {
|
2014-01-02 03:28:06 +09:00
|
|
|
lock.Lock();
|
2013-08-18 16:18:06 +09:00
|
|
|
threadInfo = NULL;
|
2014-01-02 03:28:06 +09:00
|
|
|
if(autoDelete) {
|
2013-11-23 23:08:40 +09:00
|
|
|
delete this;
|
2014-01-02 03:28:06 +09:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
lock.Unlock();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void Thread::Run() {
|
|
|
|
if(runnable)
|
|
|
|
runnable->Run();
|
|
|
|
}
|
2013-11-23 23:08:40 +09:00
|
|
|
|
|
|
|
void Thread::MarkForAutoDeletion() {
|
2013-11-28 01:03:36 +09:00
|
|
|
lock.Lock();
|
2013-11-23 23:33:15 +09:00
|
|
|
if(threadInfo == NULL){
|
|
|
|
// already exited
|
|
|
|
delete this;
|
|
|
|
return;
|
|
|
|
}
|
2013-11-23 23:08:40 +09:00
|
|
|
autoDelete = true;
|
2013-11-28 01:03:36 +09:00
|
|
|
lock.Unlock();
|
2013-11-23 23:08:40 +09:00
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|