tinygo/testdata/channel.go
Ayke van Laethem 542135c357 compiler,runtime: implement stack-based scheduler
This scheduler is intended to live along the (stackless) coroutine based
scheduler which is needed for WebAssembly and unsupported platforms. The
stack based scheduler is somewhat simpler in implementation as it does
not require full program transform passes and supports things like
function pointers and interface methods out of the box with no changes.

Code size is reduced in most cases, even in the case where no scheduler
scheduler is used at all. I'm not exactly sure why but these changes
likely allowed some further optimizations somewhere. Even RAM is
slightly reduced, perhaps some global was elminated in the process as
well.
2019-08-15 17:31:54 +02:00

177 строки
3 КиБ
Go

package main
import "time"
func main() {
ch := make(chan int)
println("len, cap of channel:", len(ch), cap(ch), ch == nil)
go sender(ch)
n, ok := <-ch
println("recv from open channel:", n, ok)
for n := range ch {
if n == 6 {
time.Sleep(time.Microsecond)
}
println("received num:", n)
}
n, ok = <-ch
println("recv from closed channel:", n, ok)
// Test bigger values
ch2 := make(chan complex128)
go sendComplex(ch2)
println("complex128:", <-ch2)
// Test multi-sender.
ch = make(chan int)
go fastsender(ch, 10)
go fastsender(ch, 23)
go fastsender(ch, 40)
slowreceiver(ch)
// Test multi-receiver.
ch = make(chan int)
go fastreceiver(ch)
go fastreceiver(ch)
go fastreceiver(ch)
slowsender(ch)
// Test iterator style channel.
ch = make(chan int)
go iterator(ch, 100)
sum := 0
for i := range ch {
sum += i
}
println("sum(100):", sum)
// Test simple selects.
go selectDeadlock()
go selectNoOp()
// Test select with a single send operation (transformed into chan send).
ch = make(chan int)
go fastreceiver(ch)
select {
case ch <- 5:
}
close(ch)
time.Sleep(time.Millisecond)
println("did send one")
// Test select with a single recv operation (transformed into chan recv).
select {
case n := <-ch:
println("select one n:", n)
}
// Test select recv with channel that has one entry.
ch = make(chan int)
go func(ch chan int) {
ch <- 55
}(ch)
time.Sleep(time.Millisecond)
select {
case make(chan int) <- 3:
println("unreachable")
case n := <-ch:
println("select n from chan:", n)
case n := <-make(chan int):
println("unreachable:", n)
}
// Test select recv with closed channel.
close(ch)
select {
case make(chan int) <- 3:
println("unreachable")
case n := <-ch:
println("select n from closed chan:", n)
case n := <-make(chan int):
println("unreachable:", n)
}
// Test select send.
ch = make(chan int)
go fastreceiver(ch)
time.Sleep(time.Millisecond)
select {
case ch <- 235:
println("select send")
case n := <-make(chan int):
println("unreachable:", n)
}
close(ch)
// Allow goroutines to exit.
time.Sleep(time.Microsecond)
}
func sender(ch chan int) {
for i := 1; i <= 8; i++ {
if i == 4 {
time.Sleep(time.Microsecond)
println("slept")
}
ch <- i
}
close(ch)
}
func sendComplex(ch chan complex128) {
ch <- 7 + 10.5i
}
func fastsender(ch chan int, n int) {
ch <- n
ch <- n + 1
}
func slowreceiver(ch chan int) {
sum := 0
for i := 0; i < 6; i++ {
sum += <-ch
time.Sleep(time.Microsecond)
}
println("sum of n:", sum)
}
func slowsender(ch chan int) {
for n := 0; n < 6; n++ {
time.Sleep(time.Microsecond)
ch <- 12 + n
}
}
func fastreceiver(ch chan int) {
sum := 0
for i := 0; i < 2; i++ {
n := <-ch
sum += n
}
println("sum:", sum)
}
func iterator(ch chan int, top int) {
for i := 0; i < top; i++ {
ch <- i
}
close(ch)
}
func selectDeadlock() {
println("deadlocking")
select {}
println("unreachable")
}
func selectNoOp() {
println("select no-op")
select {
default:
}
println("after no-op")
}