Source file src/sync/waitgroup_test.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_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 // Will block if barrier fails to unlock someone.
    36  	}
    37  }
    38  
    39  func TestWaitGroup(t *testing.T) {
    40  	wg1 := &WaitGroup{}
    41  	wg2 := &WaitGroup{}
    42  
    43  	// Run the same test a few times to ensure barrier is in a proper state.
    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  	// Run this test for about 1ms.
    65  	for i := 0; i < 1000; i++ {
    66  		wg := &WaitGroup{}
    67  		n := new(int32)
    68  		// spawn goroutine 1
    69  		wg.Add(1)
    70  		go func() {
    71  			atomic.AddInt32(n, 1)
    72  			wg.Done()
    73  		}()
    74  		// spawn goroutine 2
    75  		wg.Add(1)
    76  		go func() {
    77  			atomic.AddInt32(n, 1)
    78  			wg.Done()
    79  		}()
    80  		// Wait for goroutine 1 and 2
    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