Source file
src/runtime/synctest.go
1
2
3
4
5 package runtime
6
7 import (
8 "unsafe"
9 )
10
11
12 type synctestGroup struct {
13 mu mutex
14 timers timers
15 now int64
16 root *g
17 waiter *g
18 waiting bool
19
20
21
22
23
24
25
26
27
28
29
30
31 total int
32 running int
33 active int
34 }
35
36
37
38 func (sg *synctestGroup) changegstatus(gp *g, oldval, newval uint32) {
39
40
41
42
43
44
45
46
47 totalDelta := 0
48 wasRunning := true
49 switch oldval {
50 case _Gdead:
51 wasRunning = false
52 totalDelta++
53 case _Gwaiting:
54 if gp.waitreason.isIdleInSynctest() {
55 wasRunning = false
56 }
57 }
58 isRunning := true
59 switch newval {
60 case _Gdead:
61 isRunning = false
62 totalDelta--
63 case _Gwaiting:
64 if gp.waitreason.isIdleInSynctest() {
65 isRunning = false
66 }
67 }
68
69
70 if wasRunning == isRunning && totalDelta == 0 {
71 return
72 }
73
74 lock(&sg.mu)
75 sg.total += totalDelta
76 if wasRunning != isRunning {
77 if isRunning {
78 sg.running++
79 } else {
80 sg.running--
81 if raceenabled && newval != _Gdead {
82
83
84 racereleasemergeg(gp, sg.raceaddr())
85 }
86 }
87 }
88 if sg.total < 0 {
89 fatal("total < 0")
90 }
91 if sg.running < 0 {
92 fatal("running < 0")
93 }
94 wake := sg.maybeWakeLocked()
95 unlock(&sg.mu)
96 if wake != nil {
97 goready(wake, 0)
98 }
99 }
100
101
102
103 func (sg *synctestGroup) incActive() {
104 lock(&sg.mu)
105 sg.active++
106 unlock(&sg.mu)
107 }
108
109
110 func (sg *synctestGroup) decActive() {
111 lock(&sg.mu)
112 sg.active--
113 if sg.active < 0 {
114 throw("active < 0")
115 }
116 wake := sg.maybeWakeLocked()
117 unlock(&sg.mu)
118 if wake != nil {
119 goready(wake, 0)
120 }
121 }
122
123
124 func (sg *synctestGroup) maybeWakeLocked() *g {
125 if sg.running > 0 || sg.active > 0 {
126 return nil
127 }
128
129
130
131
132
133
134
135
136
137 sg.active++
138 next := sg.timers.wakeTime()
139 if next > 0 && next <= sg.now {
140
141 return sg.root
142 }
143 if gp := sg.waiter; gp != nil {
144
145 return gp
146 }
147
148
149 return sg.root
150 }
151
152 func (sg *synctestGroup) raceaddr() unsafe.Pointer {
153
154
155
156
157 return unsafe.Pointer(sg)
158 }
159
160
161 func synctestRun(f func()) {
162 if debug.asynctimerchan.Load() != 0 {
163 panic("synctest.Run not supported with asynctimerchan!=0")
164 }
165
166 gp := getg()
167 if gp.syncGroup != nil {
168 panic("synctest.Run called from within a synctest bubble")
169 }
170 gp.syncGroup = &synctestGroup{
171 total: 1,
172 running: 1,
173 root: gp,
174 }
175 const synctestBaseTime = 946684800000000000
176 gp.syncGroup.now = synctestBaseTime
177 gp.syncGroup.timers.syncGroup = gp.syncGroup
178 lockInit(&gp.syncGroup.mu, lockRankSynctest)
179 lockInit(&gp.syncGroup.timers.mu, lockRankTimers)
180 defer func() {
181 gp.syncGroup = nil
182 }()
183
184 fv := *(**funcval)(unsafe.Pointer(&f))
185 newproc(fv)
186
187 sg := gp.syncGroup
188 lock(&sg.mu)
189 sg.active++
190 for {
191 unlock(&sg.mu)
192 systemstack(func() {
193
194
195 curg := gp.m.curg
196 gp.m.curg = nil
197 gp.syncGroup.timers.check(gp.syncGroup.now)
198 gp.m.curg = curg
199 })
200 gopark(synctestidle_c, nil, waitReasonSynctestRun, traceBlockSynctest, 0)
201 lock(&sg.mu)
202 if sg.active < 0 {
203 throw("active < 0")
204 }
205 next := sg.timers.wakeTime()
206 if next == 0 {
207 break
208 }
209 if next < sg.now {
210 throw("time went backwards")
211 }
212 sg.now = next
213 }
214
215 total := sg.total
216 unlock(&sg.mu)
217 if raceenabled {
218
219
220 raceacquireg(gp, gp.syncGroup.raceaddr())
221 }
222 if total != 1 {
223 panic("deadlock: all goroutines in bubble are blocked")
224 }
225 if gp.timer != nil && gp.timer.isFake {
226
227
228 throw("synctest root goroutine has a fake timer")
229 }
230 }
231
232 func synctestidle_c(gp *g, _ unsafe.Pointer) bool {
233 lock(&gp.syncGroup.mu)
234 canIdle := true
235 if gp.syncGroup.running == 0 && gp.syncGroup.active == 1 {
236
237 canIdle = false
238 } else {
239 gp.syncGroup.active--
240 }
241 unlock(&gp.syncGroup.mu)
242 return canIdle
243 }
244
245
246 func synctestWait() {
247 gp := getg()
248 if gp.syncGroup == nil {
249 panic("goroutine is not in a bubble")
250 }
251 lock(&gp.syncGroup.mu)
252
253
254
255 if gp.syncGroup.waiting {
256 unlock(&gp.syncGroup.mu)
257 panic("wait already in progress")
258 }
259 gp.syncGroup.waiting = true
260 unlock(&gp.syncGroup.mu)
261 gopark(synctestwait_c, nil, waitReasonSynctestWait, traceBlockSynctest, 0)
262
263 lock(&gp.syncGroup.mu)
264 gp.syncGroup.active--
265 if gp.syncGroup.active < 0 {
266 throw("active < 0")
267 }
268 gp.syncGroup.waiter = nil
269 gp.syncGroup.waiting = false
270 unlock(&gp.syncGroup.mu)
271
272
273
274 if raceenabled {
275 raceacquireg(gp, gp.syncGroup.raceaddr())
276 }
277 }
278
279 func synctestwait_c(gp *g, _ unsafe.Pointer) bool {
280 lock(&gp.syncGroup.mu)
281 if gp.syncGroup.running == 0 && gp.syncGroup.active == 0 {
282
283 throw("running == 0 && active == 0")
284 }
285 gp.syncGroup.waiter = gp
286 unlock(&gp.syncGroup.mu)
287 return true
288 }
289
290
291 func synctest_acquire() any {
292 if sg := getg().syncGroup; sg != nil {
293 sg.incActive()
294 return sg
295 }
296 return nil
297 }
298
299
300 func synctest_release(sg any) {
301 sg.(*synctestGroup).decActive()
302 }
303
304
305 func synctest_inBubble(sg any, f func()) {
306 gp := getg()
307 if gp.syncGroup != nil {
308 panic("goroutine is already bubbled")
309 }
310 gp.syncGroup = sg.(*synctestGroup)
311 defer func() {
312 gp.syncGroup = nil
313 }()
314 f()
315 }
316
View as plain text