cond_test.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256
  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. package sync_test
  5. import (
  6. . "sync"
  7. "runtime"
  8. "testing"
  9. )
  10. func TestCondSignal(t *testing.T) {
  11. var m Mutex
  12. c := NewCond(&m)
  13. n := 2
  14. running := make(chan bool, n)
  15. awake := make(chan bool, n)
  16. for i := 0; i < n; i++ {
  17. go func() {
  18. m.Lock()
  19. running <- true
  20. c.Wait()
  21. awake <- true
  22. m.Unlock()
  23. }()
  24. }
  25. for i := 0; i < n; i++ {
  26. <-running // Wait for everyone to run.
  27. }
  28. for n > 0 {
  29. select {
  30. case <-awake:
  31. t.Fatal("goroutine not asleep")
  32. default:
  33. }
  34. m.Lock()
  35. c.Signal()
  36. m.Unlock()
  37. <-awake // Will deadlock if no goroutine wakes up
  38. select {
  39. case <-awake:
  40. t.Fatal("too many goroutines awake")
  41. default:
  42. }
  43. n--
  44. }
  45. c.Signal()
  46. }
  47. func TestCondSignalGenerations(t *testing.T) {
  48. var m Mutex
  49. c := NewCond(&m)
  50. n := 100
  51. running := make(chan bool, n)
  52. awake := make(chan int, n)
  53. for i := 0; i < n; i++ {
  54. go func(i int) {
  55. m.Lock()
  56. running <- true
  57. c.Wait()
  58. awake <- i
  59. m.Unlock()
  60. }(i)
  61. if i > 0 {
  62. a := <-awake
  63. if a != i-1 {
  64. t.Fatalf("wrong goroutine woke up: want %d, got %d", i-1, a)
  65. }
  66. }
  67. <-running
  68. m.Lock()
  69. c.Signal()
  70. m.Unlock()
  71. }
  72. }
  73. func TestCondBroadcast(t *testing.T) {
  74. var m Mutex
  75. c := NewCond(&m)
  76. n := 200
  77. running := make(chan int, n)
  78. awake := make(chan int, n)
  79. exit := false
  80. for i := 0; i < n; i++ {
  81. go func(g int) {
  82. m.Lock()
  83. for !exit {
  84. running <- g
  85. c.Wait()
  86. awake <- g
  87. }
  88. m.Unlock()
  89. }(i)
  90. }
  91. for i := 0; i < n; i++ {
  92. for i := 0; i < n; i++ {
  93. <-running // Will deadlock unless n are running.
  94. }
  95. if i == n-1 {
  96. m.Lock()
  97. exit = true
  98. m.Unlock()
  99. }
  100. select {
  101. case <-awake:
  102. t.Fatal("goroutine not asleep")
  103. default:
  104. }
  105. m.Lock()
  106. c.Broadcast()
  107. m.Unlock()
  108. seen := make([]bool, n)
  109. for i := 0; i < n; i++ {
  110. g := <-awake
  111. if seen[g] {
  112. t.Fatal("goroutine woke up twice")
  113. }
  114. seen[g] = true
  115. }
  116. }
  117. select {
  118. case <-running:
  119. t.Fatal("goroutine did not exit")
  120. default:
  121. }
  122. c.Broadcast()
  123. }
  124. func TestRace(t *testing.T) {
  125. x := 0
  126. c := NewCond(&Mutex{})
  127. done := make(chan bool)
  128. go func() {
  129. c.L.Lock()
  130. x = 1
  131. c.Wait()
  132. if x != 2 {
  133. t.Fatal("want 2")
  134. }
  135. x = 3
  136. c.Signal()
  137. c.L.Unlock()
  138. done <- true
  139. }()
  140. go func() {
  141. c.L.Lock()
  142. for {
  143. if x == 1 {
  144. x = 2
  145. c.Signal()
  146. break
  147. }
  148. c.L.Unlock()
  149. runtime.Gosched()
  150. c.L.Lock()
  151. }
  152. c.L.Unlock()
  153. done <- true
  154. }()
  155. go func() {
  156. c.L.Lock()
  157. for {
  158. if x == 2 {
  159. c.Wait()
  160. if x != 3 {
  161. t.Fatal("want 3")
  162. }
  163. break
  164. }
  165. if x == 3 {
  166. break
  167. }
  168. c.L.Unlock()
  169. runtime.Gosched()
  170. c.L.Lock()
  171. }
  172. c.L.Unlock()
  173. done <- true
  174. }()
  175. <-done
  176. <-done
  177. <-done
  178. }
  179. func TestCondCopy(t *testing.T) {
  180. defer func() {
  181. err := recover()
  182. if err == nil || err.(string) != "sync.Cond is copied" {
  183. t.Fatalf("got %v, expect sync.Cond is copied", err)
  184. }
  185. }()
  186. c := Cond{L: &Mutex{}}
  187. c.Signal()
  188. c2 := c
  189. c2.Signal()
  190. }
  191. func BenchmarkCond1(b *testing.B) {
  192. benchmarkCond(b, 1)
  193. }
  194. func BenchmarkCond2(b *testing.B) {
  195. benchmarkCond(b, 2)
  196. }
  197. func BenchmarkCond4(b *testing.B) {
  198. benchmarkCond(b, 4)
  199. }
  200. func BenchmarkCond8(b *testing.B) {
  201. benchmarkCond(b, 8)
  202. }
  203. func BenchmarkCond16(b *testing.B) {
  204. benchmarkCond(b, 16)
  205. }
  206. func BenchmarkCond32(b *testing.B) {
  207. benchmarkCond(b, 32)
  208. }
  209. func benchmarkCond(b *testing.B, waiters int) {
  210. c := NewCond(&Mutex{})
  211. done := make(chan bool)
  212. id := 0
  213. for routine := 0; routine < waiters+1; routine++ {
  214. go func() {
  215. for i := 0; i < b.N; i++ {
  216. c.L.Lock()
  217. if id == -1 {
  218. c.L.Unlock()
  219. break
  220. }
  221. id++
  222. if id == waiters+1 {
  223. id = 0
  224. c.Broadcast()
  225. } else {
  226. c.Wait()
  227. }
  228. c.L.Unlock()
  229. }
  230. c.L.Lock()
  231. id = -1
  232. c.Broadcast()
  233. c.L.Unlock()
  234. done <- true
  235. }()
  236. }
  237. for routine := 0; routine < waiters+1; routine++ {
  238. <-done
  239. }
  240. }