85 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			V
		
	
	
			
		
		
	
	
			85 lines
		
	
	
		
			2.3 KiB
		
	
	
	
		
			V
		
	
	
| // Copyright (c) 2019 Alexander Medvednikov. All rights reserved.
 | |
| // Use of this source code is governed by an MIT license
 | |
| // that can be found in the LICENSE file.
 | |
| module sync
 | |
| 
 | |
| [trusted]
 | |
| fn C.atomic_fetch_add_u32(voidptr, u32) u32
 | |
| 
 | |
| [trusted]
 | |
| fn C.atomic_load_u32(voidptr) u32
 | |
| 
 | |
| [trusted]
 | |
| fn C.atomic_compare_exchange_weak_u32(voidptr, voidptr, u32) bool
 | |
| 
 | |
| // WaitGroup
 | |
| // Do not copy an instance of WaitGroup, use a ref instead.
 | |
| //
 | |
| // usage: in main thread:
 | |
| // `wg := sync.new_waitgroup()
 | |
| // `wg.add(nr_jobs)` before starting jobs with `go ...`
 | |
| // `wg.wait()` to wait for all jobs to have finished
 | |
| //
 | |
| // in each parallel job:
 | |
| // `wg.done()` when finished
 | |
| //
 | |
| // [init_with=new_waitgroup] // TODO: implement support for init_with struct attribute, and disallow WaitGroup{} from outside the sync.new_waitgroup() function.
 | |
| [heap]
 | |
| struct WaitGroup {
 | |
| mut:
 | |
| 	task_count u32       // current task count - reading/writing should be atomic
 | |
| 	wait_count u32       // current wait count - reading/writing should be atomic
 | |
| 	sem        Semaphore // This blocks wait() until tast_countreleased by add()
 | |
| }
 | |
| 
 | |
| pub fn new_waitgroup() &WaitGroup {
 | |
| 	mut wg := WaitGroup{}
 | |
| 	wg.init()
 | |
| 	return &wg
 | |
| }
 | |
| 
 | |
| pub fn (mut wg WaitGroup) init() {
 | |
| 	wg.sem.init(0)
 | |
| }
 | |
| 
 | |
| // 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) {
 | |
| 	old_nrjobs := int(C.atomic_fetch_add_u32(&wg.task_count, u32(delta)))
 | |
| 	new_nrjobs := old_nrjobs + delta
 | |
| 	mut num_waiters := C.atomic_load_u32(&wg.wait_count)
 | |
| 	if new_nrjobs < 0 {
 | |
| 		panic('Negative number of jobs in waitgroup')
 | |
| 	}
 | |
| 
 | |
| 	if new_nrjobs == 0 && num_waiters > 0 {
 | |
| 		// clear waiters
 | |
| 		for !C.atomic_compare_exchange_weak_u32(&wg.wait_count, &num_waiters, 0) {
 | |
| 			if num_waiters == 0 {
 | |
| 				return
 | |
| 			}
 | |
| 		}
 | |
| 		for (num_waiters > 0) {
 | |
| 			wg.sem.post()
 | |
| 			num_waiters--
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // done is a convenience fn for add(-1)
 | |
| pub fn (mut wg WaitGroup) done() {
 | |
| 	wg.add(-1)
 | |
| }
 | |
| 
 | |
| // wait blocks until all tasks are done (task count becomes zero)
 | |
| pub fn (mut wg WaitGroup) wait() {
 | |
| 	nrjobs := int(C.atomic_load_u32(&wg.task_count))
 | |
| 	if nrjobs == 0 {
 | |
| 		// no need to wait
 | |
| 		return
 | |
| 	}
 | |
| 	C.atomic_fetch_add_u32(&wg.wait_count, 1)
 | |
| 	wg.sem.wait() // blocks until task_count becomes 0
 | |
| }
 |