blob: 810a7117edbce5d4dda578986b1a2762dd4fa2d4 [file] [log] [blame]
// Copyright (c) 2015, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#ifndef RUNTIME_VM_THREAD_BARRIER_H_
#define RUNTIME_VM_THREAD_BARRIER_H_
#include "vm/globals.h"
#include "vm/lockers.h"
#include "vm/os_thread.h"
namespace dart {
// Thread barrier with:
// * fixed (at construction) number n of participating threads {T1,T2,T3,...,Tn}
// * unknown number of rounds.
// Requirements:
// * there is some R such that each participating thread makes
// R calls to Sync() followed by its one and only call to Exit().
// Guarantees:
// * for any two threads Ti and Tj and round number r <= R,
// everything done by Ti before its r'th call to Sync() happens before
// everything done by Tj after its r'th call to Sync().
// Note:
// * it's not required that the thread that constructs the barrier participates.
//
// Example usage with 3 threads (1 controller + 2 workers) and 3 rounds:
//
// T1:
// ThreadBarrier barrier(3);
// Dart::thread_pool()->Run( T2:
// new FooTask(&barrier)); fooSetup();
// Dart::thread_pool()->Run( ... T3:
// new BarTask(&barrier)); ... barSetup();
// barrier.Sync(); barrier_->Sync(); barrier_->Sync();
// /* Both tasks have finished setup */ ... ...
// prepareWorkForTasks(); ... ...
// barrier.Sync(); barrier_->Sync(); barrier_->Sync();
// /* Idle while tasks are working */ fooWork(); barWork();
// barrier.Sync(); barrier_->Sync(); barrier_->Sync();
// collectResultsFromTasks(); barrier_->Exit(); barrier_->Exit();
// barrier.Exit();
//
// Note that the calls to Sync() "line up" in time, but there is no such
// guarantee for Exit().
//
class ThreadBarrier {
public:
explicit ThreadBarrier(intptr_t num_threads, intptr_t initial = 0)
: ref_count_(num_threads),
monitor_(),
participating_(initial),
remaining_(initial),
generation_(0) {}
bool TryEnter() {
MonitorLocker ml(&monitor_);
if (generation_ != 0) {
return false;
}
remaining_++;
participating_++;
return true;
}
void Sync() {
MonitorLocker ml(&monitor_);
const intptr_t g = generation_;
remaining_--;
if (remaining_ == 0) {
// I'm last, advance to the next generation and wake the others.
generation_++;
remaining_ = participating_;
ml.NotifyAll();
} else {
// Waiting for others.
while (g == generation_) {
ml.Wait();
}
}
}
void Release() {
intptr_t old = ref_count_.fetch_sub(1, std::memory_order_acq_rel);
ASSERT(old > 0);
if (old == 1) {
delete this;
}
}
private:
std::atomic<intptr_t> ref_count_;
Monitor monitor_;
intptr_t participating_;
intptr_t remaining_;
intptr_t generation_;
DISALLOW_COPY_AND_ASSIGN(ThreadBarrier);
};
} // namespace dart
#endif // RUNTIME_VM_THREAD_BARRIER_H_