| // Copyright 2012 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package race_test |
| |
| import ( |
| "sync" |
| "testing" |
| "time" |
| ) |
| |
| func TestRaceMutexRWMutex(t *testing.T) { |
| var mu1 sync.Mutex |
| var mu2 sync.RWMutex |
| var x int16 = 0 |
| ch := make(chan bool, 2) |
| go func() { |
| mu1.Lock() |
| defer mu1.Unlock() |
| x = 1 |
| ch <- true |
| }() |
| go func() { |
| mu2.Lock() |
| x = 2 |
| mu2.Unlock() |
| ch <- true |
| }() |
| <-ch |
| <-ch |
| } |
| |
| func TestNoRaceRWMutex(t *testing.T) { |
| var mu sync.RWMutex |
| var x, y int64 = 0, 1 |
| ch := make(chan bool, 2) |
| go func() { |
| mu.Lock() |
| defer mu.Unlock() |
| x = 2 |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y = x |
| mu.RUnlock() |
| ch <- true |
| }() |
| <-ch |
| <-ch |
| } |
| |
| func TestRaceRWMutexMultipleReaders(t *testing.T) { |
| var mu sync.RWMutex |
| var x, y int64 = 0, 1 |
| ch := make(chan bool, 4) |
| go func() { |
| mu.Lock() |
| defer mu.Unlock() |
| x = 2 |
| ch <- true |
| }() |
| // Use three readers so that no matter what order they're |
| // scheduled in, two will be on the same side of the write |
| // lock above. |
| go func() { |
| mu.RLock() |
| y = x + 1 |
| mu.RUnlock() |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y = x + 2 |
| mu.RUnlock() |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y = x + 3 |
| mu.RUnlock() |
| ch <- true |
| }() |
| <-ch |
| <-ch |
| <-ch |
| <-ch |
| _ = y |
| } |
| |
| func TestNoRaceRWMutexMultipleReaders(t *testing.T) { |
| var mu sync.RWMutex |
| x := int64(0) |
| ch := make(chan bool, 4) |
| go func() { |
| mu.Lock() |
| defer mu.Unlock() |
| x = 2 |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y := x + 1 |
| _ = y |
| mu.RUnlock() |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y := x + 2 |
| _ = y |
| mu.RUnlock() |
| ch <- true |
| }() |
| go func() { |
| mu.RLock() |
| y := x + 3 |
| _ = y |
| mu.RUnlock() |
| ch <- true |
| }() |
| <-ch |
| <-ch |
| <-ch |
| <-ch |
| } |
| |
| func TestNoRaceRWMutexTransitive(t *testing.T) { |
| var mu sync.RWMutex |
| x := int64(0) |
| ch := make(chan bool, 2) |
| go func() { |
| mu.RLock() |
| _ = x |
| mu.RUnlock() |
| ch <- true |
| }() |
| go func() { |
| time.Sleep(1e7) |
| mu.RLock() |
| _ = x |
| mu.RUnlock() |
| ch <- true |
| }() |
| time.Sleep(2e7) |
| mu.Lock() |
| x = 42 |
| mu.Unlock() |
| <-ch |
| <-ch |
| } |