// 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 stack splitting code. // 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") } } func f0() { // likely to make a new stack for f0, // because the call to f1 puts 3000 bytes // in our frame. f1() } func f1() [3000]byte { // likely to make a new stack for f1, // because 3000 bytes were used by f0 // and we need 3000 more for the call // to f2. if the call to morestack in f1 // does not pass the frame size, the new // stack (default size 5k) will not be big // enough for the frame, and the morestack // check in f2 will die, if we get that far // without faulting. f2() return [3000]byte{} } func f2() [3000]byte { // just take up space return [3000]byte{} } 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) f0() 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(8000) }