Source file src/sync/waitgroup.go
1 // Copyright 2011 The Go Authors. All rights reserved. 2 // Use of this source code is governed by a BSD-style 3 // license that can be found in the LICENSE file. 4 5 package sync 6 7 import ( 8 "internal/race" 9 "sync/atomic" 10 "unsafe" 11 ) 12 13 // A WaitGroup is a counting semaphore typically used to wait 14 // for a group of goroutines to finish. 15 // 16 // The main goroutine calls [WaitGroup.Add] to set (or increase) the number of 17 // goroutines to wait for. Then each of the goroutines 18 // runs and calls [WaitGroup.Done] when finished. At the same time, 19 // [WaitGroup.Wait] can be used to block until all goroutines have finished. 20 // 21 // This is a typical pattern of WaitGroup usage to 22 // synchronize 3 goroutines, each calling the function f: 23 // 24 // var wg sync.WaitGroup 25 // for range 3 { 26 // wg.Add(1) 27 // go func() { 28 // defer wg.Done() 29 // f() 30 // }() 31 // } 32 // wg.Wait() 33 // 34 // For convenience, the [WaitGroup.Go] method simplifies this pattern to: 35 // 36 // var wg sync.WaitGroup 37 // for range 3 { 38 // wg.Go(f) 39 // } 40 // wg.Wait() 41 // 42 // A WaitGroup must not be copied after first use. 43 // 44 // In the terminology of [the Go memory model], a call to [WaitGroup.Done] 45 // “synchronizes before” the return of any Wait call that it unblocks. 46 // 47 // [the Go memory model]: https://go.dev/ref/mem 48 type WaitGroup struct { 49 noCopy noCopy 50 51 state atomic.Uint64 // high 32 bits are counter, low 32 bits are waiter count. 52 sema uint32 53 } 54 55 // Add adds delta, which may be negative, to the [WaitGroup] counter. 56 // If the counter becomes zero, all goroutines blocked on [WaitGroup.Wait] are released. 57 // If the counter goes negative, Add panics. 58 // 59 // Note that calls with a positive delta that occur when the counter is zero 60 // must happen before a Wait. Calls with a negative delta, or calls with a 61 // positive delta that start when the counter is greater than zero, may happen 62 // at any time. 63 // Typically this means the calls to Add should execute before the statement 64 // creating the goroutine or other event to be waited for. 65 // If a WaitGroup is reused to wait for several independent sets of events, 66 // new Add calls must happen after all previous Wait calls have returned. 67 // See the WaitGroup example. 68 func (wg *WaitGroup) Add(delta int) { 69 if race.Enabled { 70 if delta < 0 { 71 // Synchronize decrements with Wait. 72 race.ReleaseMerge(unsafe.Pointer(wg)) 73 } 74 race.Disable() 75 defer race.Enable() 76 } 77 state := wg.state.Add(uint64(delta) << 32) 78 v := int32(state >> 32) 79 w := uint32(state) 80 if race.Enabled && delta > 0 && v == int32(delta) { 81 // The first increment must be synchronized with Wait. 82 // Need to model this as a read, because there can be 83 // several concurrent wg.counter transitions from 0. 84 race.Read(unsafe.Pointer(&wg.sema)) 85 } 86 if v < 0 { 87 panic("sync: negative WaitGroup counter") 88 } 89 if w != 0 && delta > 0 && v == int32(delta) { 90 panic("sync: WaitGroup misuse: Add called concurrently with Wait") 91 } 92 if v > 0 || w == 0 { 93 return 94 } 95 // This goroutine has set counter to 0 when waiters > 0. 96 // Now there can't be concurrent mutations of state: 97 // - Adds must not happen concurrently with Wait, 98 // - Wait does not increment waiters if it sees counter == 0. 99 // Still do a cheap sanity check to detect WaitGroup misuse. 100 if wg.state.Load() != state { 101 panic("sync: WaitGroup misuse: Add called concurrently with Wait") 102 } 103 // Reset waiters count to 0. 104 wg.state.Store(0) 105 for ; w != 0; w-- { 106 runtime_Semrelease(&wg.sema, false, 0) 107 } 108 } 109 110 // Done decrements the [WaitGroup] counter by one. 111 func (wg *WaitGroup) Done() { 112 wg.Add(-1) 113 } 114 115 // Wait blocks until the [WaitGroup] counter is zero. 116 func (wg *WaitGroup) Wait() { 117 if race.Enabled { 118 race.Disable() 119 } 120 for { 121 state := wg.state.Load() 122 v := int32(state >> 32) 123 w := uint32(state) 124 if v == 0 { 125 // Counter is 0, no need to wait. 126 if race.Enabled { 127 race.Enable() 128 race.Acquire(unsafe.Pointer(wg)) 129 } 130 return 131 } 132 // Increment waiters count. 133 if wg.state.CompareAndSwap(state, state+1) { 134 if race.Enabled && w == 0 { 135 // Wait must be synchronized with the first Add. 136 // Need to model this is as a write to race with the read in Add. 137 // As a consequence, can do the write only for the first waiter, 138 // otherwise concurrent Waits will race with each other. 139 race.Write(unsafe.Pointer(&wg.sema)) 140 } 141 runtime_SemacquireWaitGroup(&wg.sema) 142 if wg.state.Load() != 0 { 143 panic("sync: WaitGroup is reused before previous Wait has returned") 144 } 145 if race.Enabled { 146 race.Enable() 147 race.Acquire(unsafe.Pointer(wg)) 148 } 149 return 150 } 151 } 152 } 153 154 // Go calls f in a new goroutine and adds that task to the WaitGroup. 155 // When f returns, the task is removed from the WaitGroup. 156 // 157 // If the WaitGroup is empty, Go must happen before a [WaitGroup.Wait]. 158 // Typically, this simply means Go is called to start tasks before Wait is called. 159 // If the WaitGroup is not empty, Go may happen at any time. 160 // This means a goroutine started by Go may itself call Go. 161 // If a WaitGroup is reused to wait for several independent sets of tasks, 162 // new Go calls must happen after all previous Wait calls have returned. 163 // 164 // In the terminology of [the Go memory model](https://go.dev/ref/mem), 165 // the return from f "synchronizes before" the return of any Wait call that it unblocks. 166 func (wg *WaitGroup) Go(f func()) { 167 wg.Add(1) 168 go func() { 169 defer wg.Done() 170 f() 171 }() 172 } 173