mirror of
https://github.com/ultimatepp/ultimatepp.git
synced 2026-05-15 22:02:49 -06:00
124 lines
3.1 KiB
C++
124 lines
3.1 KiB
C++
#ifdef _MULTITHREADED
|
|
|
|
class CoWork : NoCopy {
|
|
struct MJob : Moveable<MJob>, Link<MJob, 2> {
|
|
Function<void ()> fn;
|
|
CoWork *work = NULL;
|
|
};
|
|
|
|
enum { SCHEDULED_MAX = 2048 };
|
|
|
|
public:
|
|
struct Pool {
|
|
MJob *free;
|
|
Link<MJob, 2> jobs;
|
|
MJob slot[SCHEDULED_MAX];
|
|
int waiting_threads;
|
|
Array<Thread> threads;
|
|
bool quit;
|
|
|
|
Mutex lock;
|
|
ConditionVariable waitforjob;
|
|
|
|
void Free(MJob& m);
|
|
void DoJob(MJob& m);
|
|
void PushJob(Function<void ()>&& fn, CoWork *work);
|
|
|
|
void InitThreads(int nthreads);
|
|
void ExitThreads();
|
|
|
|
Pool();
|
|
~Pool();
|
|
|
|
static thread__ bool finlock;
|
|
|
|
bool DoJob();
|
|
static void ThreadRun(int tno);
|
|
};
|
|
|
|
friend struct Pool;
|
|
|
|
static Pool& GetPool();
|
|
|
|
static thread_local int worker_index;
|
|
|
|
ConditionVariable waitforfinish;
|
|
Link<MJob, 2> jobs; // global stack and CoWork stack as double-linked lists
|
|
int todo;
|
|
|
|
// experimental pipe support
|
|
Mutex stepmutex;
|
|
Array<BiVector<Function<void ()>>> step;
|
|
Vector<bool> steprunning;
|
|
|
|
public:
|
|
static bool TrySchedule(Function<void ()>&& fn);
|
|
static bool TrySchedule(const Function<void ()>& fn) { return TrySchedule(clone(fn)); }
|
|
static void Schedule(Function<void ()>&& fn);
|
|
static void Schedule(const Function<void ()>& fn) { return Schedule(clone(fn)); }
|
|
|
|
void Do(Function<void ()>&& fn);
|
|
void Do(const Function<void ()>& fn) { Do(clone(fn)); }
|
|
|
|
CoWork& operator&(const Function<void ()>& fn) { Do(fn); return *this; }
|
|
CoWork& operator&(Function<void ()>&& fn) { Do(pick(fn)); return *this; }
|
|
|
|
void Pipe(int stepi, Function<void ()>&& lambda); // experimental
|
|
|
|
static void FinLock();
|
|
|
|
void Cancel();
|
|
|
|
void Finish();
|
|
|
|
bool IsFinished();
|
|
|
|
static bool IsWorker() { return worker_index >= 0; }
|
|
static int GetWorkerIndex() { return worker_index; }
|
|
static int GetPoolSize();
|
|
static void SetPoolSize(int n);
|
|
|
|
CoWork();
|
|
~CoWork();
|
|
};
|
|
|
|
template <class T>
|
|
class CoWorkerResources {
|
|
int workercount;
|
|
Buffer<T> res;
|
|
|
|
public:
|
|
int GetCount() const { return workercount + 1; }
|
|
T& operator[](int i) { return res[i]; }
|
|
|
|
T& Get() { int i = CoWork::GetWorkerIndex(); return res[i < 0 ? workercount : i]; }
|
|
T& operator~() { return Get(); }
|
|
|
|
T *begin() { return ~res; }
|
|
T *end() { return ~res + GetCount(); }
|
|
|
|
CoWorkerResources() { workercount = CoWork::GetPoolSize(); res.Alloc(GetCount()); }
|
|
|
|
CoWorkerResources(Event<T&> initializer) : CoWorkerResources() {
|
|
for(int i = 0; i < GetCount(); i++)
|
|
initializer(res[i]);
|
|
}
|
|
};
|
|
|
|
#else
|
|
|
|
class CoWork : NoCopy {
|
|
public:
|
|
void Do(Event<> cb) { cb(); }
|
|
CoWork& operator&(Event<> cb) { cb(); return *this; }
|
|
void Finish() {}
|
|
bool IsFinished() { return true; }
|
|
|
|
static void FinLock() {}
|
|
|
|
static bool IsWorker() { return false; }
|
|
static void StartPool(int n) {}
|
|
static void ShutdownPool() {}
|
|
};
|
|
|
|
#endif
|