| // run |
| |
| // 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. |
| |
| // Test close(c), receive of closed channel. |
| // |
| // TODO(rsc): Doesn't check behavior of close(c) when there |
| // are blocked senders/receivers. |
| |
| package main |
| |
| import "os" |
| |
| var failed bool |
| |
| type Chan interface { |
| Send(int) |
| Nbsend(int) bool |
| Recv() (int) |
| Nbrecv() (int, bool) |
| Recv2() (int, bool) |
| Nbrecv2() (int, bool, bool) |
| Close() |
| Impl() string |
| } |
| |
| // direct channel operations when possible |
| type XChan chan int |
| |
| func (c XChan) Send(x int) { |
| c <- x |
| } |
| |
| func (c XChan) Nbsend(x int) bool { |
| select { |
| case c <- x: |
| return true |
| default: |
| return false |
| } |
| panic("nbsend") |
| } |
| |
| func (c XChan) Recv() int { |
| return <-c |
| } |
| |
| func (c XChan) Nbrecv() (int, bool) { |
| select { |
| case x := <-c: |
| return x, true |
| default: |
| return 0, false |
| } |
| panic("nbrecv") |
| } |
| |
| func (c XChan) Recv2() (int, bool) { |
| x, ok := <-c |
| return x, ok |
| } |
| |
| func (c XChan) Nbrecv2() (int, bool, bool) { |
| select { |
| case x, ok := <-c: |
| return x, ok, true |
| default: |
| return 0, false, false |
| } |
| panic("nbrecv2") |
| } |
| |
| func (c XChan) Close() { |
| close(c) |
| } |
| |
| func (c XChan) Impl() string { |
| return "(<- operator)" |
| } |
| |
| // indirect operations via select |
| type SChan chan int |
| |
| func (c SChan) Send(x int) { |
| select { |
| case c <- x: |
| } |
| } |
| |
| func (c SChan) Nbsend(x int) bool { |
| select { |
| default: |
| return false |
| case c <- x: |
| return true |
| } |
| panic("nbsend") |
| } |
| |
| func (c SChan) Recv() int { |
| select { |
| case x := <-c: |
| return x |
| } |
| panic("recv") |
| } |
| |
| func (c SChan) Nbrecv() (int, bool) { |
| select { |
| default: |
| return 0, false |
| case x := <-c: |
| return x, true |
| } |
| panic("nbrecv") |
| } |
| |
| func (c SChan) Recv2() (int, bool) { |
| select { |
| case x, ok := <-c: |
| return x, ok |
| } |
| panic("recv") |
| } |
| |
| func (c SChan) Nbrecv2() (int, bool, bool) { |
| select { |
| default: |
| return 0, false, false |
| case x, ok := <-c: |
| return x, ok, true |
| } |
| panic("nbrecv") |
| } |
| |
| func (c SChan) Close() { |
| close(c) |
| } |
| |
| func (c SChan) Impl() string { |
| return "(select)" |
| } |
| |
| // indirect operations via larger selects |
| var dummy = make(chan bool) |
| |
| type SSChan chan int |
| |
| func (c SSChan) Send(x int) { |
| select { |
| case c <- x: |
| case <-dummy: |
| } |
| } |
| |
| func (c SSChan) Nbsend(x int) bool { |
| select { |
| default: |
| return false |
| case <-dummy: |
| case c <- x: |
| return true |
| } |
| panic("nbsend") |
| } |
| |
| func (c SSChan) Recv() int { |
| select { |
| case <-dummy: |
| case x := <-c: |
| return x |
| } |
| panic("recv") |
| } |
| |
| func (c SSChan) Nbrecv() (int, bool) { |
| select { |
| case <-dummy: |
| default: |
| return 0, false |
| case x := <-c: |
| return x, true |
| } |
| panic("nbrecv") |
| } |
| |
| func (c SSChan) Recv2() (int, bool) { |
| select { |
| case <-dummy: |
| case x, ok := <-c: |
| return x, ok |
| } |
| panic("recv") |
| } |
| |
| func (c SSChan) Nbrecv2() (int, bool, bool) { |
| select { |
| case <-dummy: |
| default: |
| return 0, false, false |
| case x, ok := <-c: |
| return x, ok, true |
| } |
| panic("nbrecv") |
| } |
| |
| func (c SSChan) Close() { |
| close(c) |
| } |
| |
| func (c SSChan) Impl() string { |
| return "(select)" |
| } |
| |
| |
| func shouldPanic(f func()) { |
| defer func() { |
| if recover() == nil { |
| panic("did not panic") |
| } |
| }() |
| f() |
| } |
| |
| func test1(c Chan) { |
| for i := 0; i < 3; i++ { |
| // recv a close signal (a zero value) |
| if x := c.Recv(); x != 0 { |
| println("test1: recv on closed:", x, c.Impl()) |
| failed = true |
| } |
| if x, ok := c.Recv2(); x != 0 || ok { |
| println("test1: recv2 on closed:", x, ok, c.Impl()) |
| failed = true |
| } |
| |
| // should work with select: received a value without blocking, so selected == true. |
| x, selected := c.Nbrecv() |
| if x != 0 || !selected { |
| println("test1: recv on closed nb:", x, selected, c.Impl()) |
| failed = true |
| } |
| x, ok, selected := c.Nbrecv2() |
| if x != 0 || ok || !selected { |
| println("test1: recv2 on closed nb:", x, ok, selected, c.Impl()) |
| failed = true |
| } |
| } |
| |
| // send should work with ,ok too: sent a value without blocking, so ok == true. |
| shouldPanic(func() { c.Nbsend(1) }) |
| |
| // the value should have been discarded. |
| if x := c.Recv(); x != 0 { |
| println("test1: recv on closed got non-zero after send on closed:", x, c.Impl()) |
| failed = true |
| } |
| |
| // similarly Send. |
| shouldPanic(func() { c.Send(2) }) |
| if x := c.Recv(); x != 0 { |
| println("test1: recv on closed got non-zero after send on closed:", x, c.Impl()) |
| failed = true |
| } |
| } |
| |
| func testasync1(c Chan) { |
| // should be able to get the last value via Recv |
| if x := c.Recv(); x != 1 { |
| println("testasync1: Recv did not get 1:", x, c.Impl()) |
| failed = true |
| } |
| |
| test1(c) |
| } |
| |
| func testasync2(c Chan) { |
| // should be able to get the last value via Recv2 |
| if x, ok := c.Recv2(); x != 1 || !ok { |
| println("testasync1: Recv did not get 1, true:", x, ok, c.Impl()) |
| failed = true |
| } |
| |
| test1(c) |
| } |
| |
| func testasync3(c Chan) { |
| // should be able to get the last value via Nbrecv |
| if x, selected := c.Nbrecv(); x != 1 || !selected { |
| println("testasync2: Nbrecv did not get 1, true:", x, selected, c.Impl()) |
| failed = true |
| } |
| |
| test1(c) |
| } |
| |
| func testasync4(c Chan) { |
| // should be able to get the last value via Nbrecv2 |
| if x, ok, selected := c.Nbrecv2(); x != 1 || !ok || !selected { |
| println("testasync2: Nbrecv did not get 1, true, true:", x, ok, selected, c.Impl()) |
| failed = true |
| } |
| test1(c) |
| } |
| |
| func closedsync() chan int { |
| c := make(chan int) |
| close(c) |
| return c |
| } |
| |
| func closedasync() chan int { |
| c := make(chan int, 2) |
| c <- 1 |
| close(c) |
| return c |
| } |
| |
| var mks = []func(chan int) Chan { |
| func(c chan int) Chan { return XChan(c) }, |
| func(c chan int) Chan { return SChan(c) }, |
| func(c chan int) Chan { return SSChan(c) }, |
| } |
| |
| var testcloseds = []func(Chan) { |
| testasync1, |
| testasync2, |
| testasync3, |
| testasync4, |
| } |
| |
| func main() { |
| for _, mk := range mks { |
| test1(mk(closedsync())) |
| } |
| |
| for _, testclosed := range testcloseds { |
| for _, mk := range mks { |
| testclosed(mk(closedasync())) |
| } |
| } |
| |
| var ch chan int |
| shouldPanic(func() { |
| close(ch) |
| }) |
| |
| ch = make(chan int) |
| close(ch) |
| shouldPanic(func() { |
| close(ch) |
| }) |
| |
| if failed { |
| os.Exit(1) |
| } |
| } |