| // $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. |
| |
| // Try to tickle stack splitting bugs by doing |
| // go, defer, and closure calls at different stack depths. |
| |
| package main |
| |
| type T [20]int |
| |
| func g(c chan int, t T) { |
| s := 0 |
| for i := 0; i < len(t); i++ { |
| s += t[i] |
| } |
| c <- s |
| } |
| |
| func d(t T) { |
| s := 0 |
| for i := 0; i < len(t); i++ { |
| s += t[i] |
| } |
| if s != len(t) { |
| println("bad defer", s) |
| panic("fail") |
| } |
| } |
| |
| var c = make(chan int) |
| var t T |
| var b = []byte{1, 2, 3, 4, 5, 6, 7, 8, 9, 10} |
| |
| func recur(n int) { |
| ss := string(b) |
| if len(ss) != len(b) { |
| panic("bad []byte -> string") |
| } |
| go g(c, t) |
| s := <-c |
| if s != len(t) { |
| println("bad go", s) |
| panic("fail") |
| } |
| f := func(t T) int { |
| s := 0 |
| for i := 0; i < len(t); i++ { |
| s += t[i] |
| } |
| s += n |
| return s |
| } |
| s = f(t) |
| if s != len(t)+n { |
| println("bad func", s, "at level", n) |
| panic("fail") |
| } |
| if n > 0 { |
| recur(n - 1) |
| } |
| defer d(t) |
| } |
| |
| func main() { |
| for i := 0; i < len(t); i++ { |
| t[i] = 1 |
| } |
| recur(10000) |
| } |