Source file
src/sync/waitgroup_test.go
1
2
3
4
5 package sync_test
6
7 import (
8 . "sync"
9 "sync/atomic"
10 "testing"
11 )
12
13 func testWaitGroup(t *testing.T, wg1 *WaitGroup, wg2 *WaitGroup) {
14 n := 16
15 wg1.Add(n)
16 wg2.Add(n)
17 exited := make(chan bool, n)
18 for i := 0; i != n; i++ {
19 go func() {
20 wg1.Done()
21 wg2.Wait()
22 exited <- true
23 }()
24 }
25 wg1.Wait()
26 for i := 0; i != n; i++ {
27 select {
28 case <-exited:
29 t.Fatal("WaitGroup released group too soon")
30 default:
31 }
32 wg2.Done()
33 }
34 for i := 0; i != n; i++ {
35 <-exited
36 }
37 }
38
39 func TestWaitGroup(t *testing.T) {
40 wg1 := &WaitGroup{}
41 wg2 := &WaitGroup{}
42
43
44 for i := 0; i != 8; i++ {
45 testWaitGroup(t, wg1, wg2)
46 }
47 }
48
49 func TestWaitGroupMisuse(t *testing.T) {
50 defer func() {
51 err := recover()
52 if err != "sync: negative WaitGroup counter" {
53 t.Fatalf("Unexpected panic: %#v", err)
54 }
55 }()
56 wg := &WaitGroup{}
57 wg.Add(1)
58 wg.Done()
59 wg.Done()
60 t.Fatal("Should panic")
61 }
62
63 func TestWaitGroupRace(t *testing.T) {
64
65 for i := 0; i < 1000; i++ {
66 wg := &WaitGroup{}
67 n := new(int32)
68
69 wg.Add(1)
70 go func() {
71 atomic.AddInt32(n, 1)
72 wg.Done()
73 }()
74
75 wg.Add(1)
76 go func() {
77 atomic.AddInt32(n, 1)
78 wg.Done()
79 }()
80
81 wg.Wait()
82 if atomic.LoadInt32(n) != 2 {
83 t.Fatal("Spurious wakeup from Wait")
84 }
85 }
86 }
87
88 func TestWaitGroupAlign(t *testing.T) {
89 type X struct {
90 x byte
91 wg WaitGroup
92 }
93 var x X
94 x.wg.Add(1)
95 go func(x *X) {
96 x.wg.Done()
97 }(&x)
98 x.wg.Wait()
99 }
100
101 func TestWaitGroupGo(t *testing.T) {
102 wg := &WaitGroup{}
103 var i int
104 wg.Go(func() {
105 i++
106 })
107 wg.Wait()
108 if i != 1 {
109 t.Fatalf("got %d, want 1", i)
110 }
111 }
112
113 func BenchmarkWaitGroupUncontended(b *testing.B) {
114 type PaddedWaitGroup struct {
115 WaitGroup
116 pad [128]uint8
117 }
118 b.RunParallel(func(pb *testing.PB) {
119 var wg PaddedWaitGroup
120 for pb.Next() {
121 wg.Add(1)
122 wg.Done()
123 wg.Wait()
124 }
125 })
126 }
127
128 func benchmarkWaitGroupAddDone(b *testing.B, localWork int) {
129 var wg WaitGroup
130 b.RunParallel(func(pb *testing.PB) {
131 foo := 0
132 for pb.Next() {
133 wg.Add(1)
134 for i := 0; i < localWork; i++ {
135 foo *= 2
136 foo /= 2
137 }
138 wg.Done()
139 }
140 _ = foo
141 })
142 }
143
144 func BenchmarkWaitGroupAddDone(b *testing.B) {
145 benchmarkWaitGroupAddDone(b, 0)
146 }
147
148 func BenchmarkWaitGroupAddDoneWork(b *testing.B) {
149 benchmarkWaitGroupAddDone(b, 100)
150 }
151
152 func benchmarkWaitGroupWait(b *testing.B, localWork int) {
153 var wg WaitGroup
154 b.RunParallel(func(pb *testing.PB) {
155 foo := 0
156 for pb.Next() {
157 wg.Wait()
158 for i := 0; i < localWork; i++ {
159 foo *= 2
160 foo /= 2
161 }
162 }
163 _ = foo
164 })
165 }
166
167 func BenchmarkWaitGroupWait(b *testing.B) {
168 benchmarkWaitGroupWait(b, 0)
169 }
170
171 func BenchmarkWaitGroupWaitWork(b *testing.B) {
172 benchmarkWaitGroupWait(b, 100)
173 }
174
175 func BenchmarkWaitGroupActuallyWait(b *testing.B) {
176 b.ReportAllocs()
177 b.RunParallel(func(pb *testing.PB) {
178 for pb.Next() {
179 var wg WaitGroup
180 wg.Add(1)
181 go func() {
182 wg.Done()
183 }()
184 wg.Wait()
185 }
186 })
187 }
188
View as plain text