1
0
mirror of https://github.com/golang/go synced 2024-11-22 09:44:40 -07:00
go/test/chan/nonblock.go
Ian Lance Taylor 8e985dcda6 test: Reduce race conditions in chan/nonblock.go.
nonblock.go wants to test nonblocking operations on
synchronous channels, so it is inherently racy.  This
introduces loops to make the race conditions much more likely
to succeed when using gccgo.

R=r
CC=golang-dev
https://golang.org/cl/2161043
2010-09-10 15:37:20 -07:00

233 lines
3.5 KiB
Go

// $G $D/$F.go && $L $F.$A && ./$A.out
// Copyright 2009 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.
// Verify channel operations that test for blocking
// Use several sizes and types of operands
package main
import "runtime"
import "time"
func i32receiver(c chan int32, strobe chan bool) {
if <-c != 123 {
panic("i32 value")
}
strobe <- true
}
func i32sender(c chan int32, strobe chan bool) {
c <- 234
strobe <- true
}
func i64receiver(c chan int64, strobe chan bool) {
if <-c != 123456 {
panic("i64 value")
}
strobe <- true
}
func i64sender(c chan int64, strobe chan bool) {
c <- 234567
strobe <- true
}
func breceiver(c chan bool, strobe chan bool) {
if !<-c {
panic("b value")
}
strobe <- true
}
func bsender(c chan bool, strobe chan bool) {
c <- true
strobe <- true
}
func sreceiver(c chan string, strobe chan bool) {
if <-c != "hello" {
panic("s value")
}
strobe <- true
}
func ssender(c chan string, strobe chan bool) {
c <- "hello again"
strobe <- true
}
var ticker = time.Tick(10 * 1000) // 10 us
func sleep() {
<-ticker
<-ticker
runtime.Gosched()
runtime.Gosched()
runtime.Gosched()
}
const maxTries = 10000 // Up to 100ms per test.
func main() {
var i32 int32
var i64 int64
var b bool
var s string
var ok bool
var sync = make(chan bool)
for buffer := 0; buffer < 2; buffer++ {
c32 := make(chan int32, buffer)
c64 := make(chan int64, buffer)
cb := make(chan bool, buffer)
cs := make(chan string, buffer)
i32, ok = <-c32
if ok {
panic("blocked i32sender")
}
i64, ok = <-c64
if ok {
panic("blocked i64sender")
}
b, ok = <-cb
if ok {
panic("blocked bsender")
}
s, ok = <-cs
if ok {
panic("blocked ssender")
}
go i32receiver(c32, sync)
try := 0
for !(c32 <- 123) {
try++
if try > maxTries {
println("i32receiver buffer=", buffer)
panic("fail")
}
sleep()
}
<-sync
go i32sender(c32, sync)
if buffer > 0 {
<-sync
}
try = 0
for i32, ok = <-c32; !ok; i32, ok = <-c32 {
try++
if try > maxTries {
println("i32sender buffer=", buffer)
panic("fail")
}
sleep()
}
if i32 != 234 {
panic("i32sender value")
}
if buffer == 0 {
<-sync
}
go i64receiver(c64, sync)
try = 0
for !(c64 <- 123456) {
try++
if try > maxTries {
panic("i64receiver")
}
sleep()
}
<-sync
go i64sender(c64, sync)
if buffer > 0 {
<-sync
}
try = 0
for i64, ok = <-c64; !ok; i64, ok = <-c64 {
try++
if try > maxTries {
panic("i64sender")
}
sleep()
}
if i64 != 234567 {
panic("i64sender value")
}
if buffer == 0 {
<-sync
}
go breceiver(cb, sync)
try = 0
for !(cb <- true) {
try++
if try > maxTries {
panic("breceiver")
}
sleep()
}
<-sync
go bsender(cb, sync)
if buffer > 0 {
<-sync
}
try = 0
for b, ok = <-cb; !ok; b, ok = <-cb {
try++
if try > maxTries {
panic("bsender")
}
sleep()
}
if !b {
panic("bsender value")
}
if buffer == 0 {
<-sync
}
go sreceiver(cs, sync)
try = 0
for !(cs <- "hello") {
try++
if try > maxTries {
panic("sreceiver")
}
sleep()
}
<-sync
go ssender(cs, sync)
if buffer > 0 {
<-sync
}
try = 0
for s, ok = <-cs; !ok; s, ok = <-cs {
try++
if try > maxTries {
panic("ssender")
}
sleep()
}
if s != "hello again" {
panic("ssender value")
}
if buffer == 0 {
<-sync
}
}
print("PASS\n")
}