waitgroup: remove races, make platform independent, document
parent
e65879d076
commit
640bbbae85
|
@ -1,41 +1,59 @@
|
||||||
// Copyright (c) 2019-2020 Alexander Medvednikov. All rights reserved.
|
// Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
|
||||||
// Use of this source code is governed by an MIT license
|
// Use of this source code is governed by an MIT license
|
||||||
// that can be found in the LICENSE file.
|
// that can be found in the LICENSE file.
|
||||||
module sync
|
module sync
|
||||||
|
|
||||||
|
// WaitGroup implementation. wait() blocks until all tasks complete.
|
||||||
|
// Do not copy an instance of WaitGroup, use a ref instead.
|
||||||
|
//
|
||||||
|
// Two mutexes are required so that wait() doesn't unblock on done()/add() before
|
||||||
|
// task_count becomes zero.
|
||||||
|
//
|
||||||
// [init_with=new_waitgroup] // TODO: implement support for init_with struct attribute, and disallow WaitGroup{} from outside the sync.new_waitgroup() function.
|
// [init_with=new_waitgroup] // TODO: implement support for init_with struct attribute, and disallow WaitGroup{} from outside the sync.new_waitgroup() function.
|
||||||
[ref_only]
|
[ref_only]
|
||||||
pub struct WaitGroup {
|
struct WaitGroup {
|
||||||
mut:
|
mut:
|
||||||
mu &Mutex = &Mutex(0)
|
task_count int // current task count
|
||||||
active int
|
task_count_mutex &Mutex = &Mutex(0) // This mutex protects the task_count count in add()
|
||||||
|
wait_blocker &Mutex = &Mutex(0) // This mutex blocks the wait() until released by add()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_waitgroup() &WaitGroup {
|
pub fn new_waitgroup() &WaitGroup {
|
||||||
return &WaitGroup{mu: sync.new_mutex() }
|
return &WaitGroup{
|
||||||
|
task_count_mutex: new_mutex()
|
||||||
|
wait_blocker: new_mutex()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// add increments (+ve delta) or decrements (-ve delta) task count by delta
|
||||||
|
// and unblocks any wait() calls if task count becomes zero.
|
||||||
|
// add panics if task count drops below zero.
|
||||||
pub fn (mut wg WaitGroup) add(delta int) {
|
pub fn (mut wg WaitGroup) add(delta int) {
|
||||||
wg.mu.lock()
|
// protect task_count
|
||||||
wg.active += delta
|
wg.task_count_mutex.lock()
|
||||||
wg.mu.unlock()
|
defer {
|
||||||
if wg.active < 0 {
|
wg.task_count_mutex.unlock()
|
||||||
|
}
|
||||||
|
// If task_count likely to leave zero, set wait() to block
|
||||||
|
if wg.task_count == 0 {
|
||||||
|
wg.wait_blocker.lock()
|
||||||
|
}
|
||||||
|
wg.task_count += delta
|
||||||
|
if wg.task_count < 0 {
|
||||||
panic('Negative number of jobs in waitgroup')
|
panic('Negative number of jobs in waitgroup')
|
||||||
}
|
}
|
||||||
|
if wg.task_count == 0 { // if no more task_count tasks
|
||||||
|
wg.wait_blocker.unlock() // unblock wait()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// done is a convenience fn for add(-1)
|
||||||
pub fn (mut wg WaitGroup) done() {
|
pub fn (mut wg WaitGroup) done() {
|
||||||
wg.add(-1)
|
wg.add(-1)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn (wg &WaitGroup) wait() {
|
// wait blocks until all tasks are done (task count becomes zero)
|
||||||
for wg.active > 0 {
|
pub fn (mut wg WaitGroup) wait() {
|
||||||
// Do not remove this, busy empty loops are optimized
|
wg.wait_blocker.lock() // blocks until task_count becomes 0
|
||||||
// with -prod by some compilers, see issue #2874
|
wg.wait_blocker.unlock() // allow other wait()s to unblock or reuse wait group
|
||||||
$if windows {
|
|
||||||
C.Sleep(1)
|
|
||||||
} $else {
|
|
||||||
C.usleep(1000)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue