| // Copyright 2013 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. |
| |
| // Only works on systems with syscall.Close. |
| // We need a fast system call to provoke the race, |
| // and Close(-1) is nearly universally fast. |
| |
| //go:build aix || darwin || dragonfly || freebsd || linux || netbsd || openbsd || plan9 |
| |
| package runtime_test |
| |
| import ( |
| "runtime" |
| "sync" |
| "sync/atomic" |
| "syscall" |
| "testing" |
| ) |
| |
| func TestGoroutineProfile(t *testing.T) { |
| // GoroutineProfile used to use the wrong starting sp for |
| // goroutines coming out of system calls, causing possible |
| // crashes. |
| defer runtime.GOMAXPROCS(runtime.GOMAXPROCS(100)) |
| |
| var stop uint32 |
| defer atomic.StoreUint32(&stop, 1) // in case of panic |
| |
| var wg sync.WaitGroup |
| for i := 0; i < 4; i++ { |
| wg.Add(1) |
| go func() { |
| for atomic.LoadUint32(&stop) == 0 { |
| syscall.Close(-1) |
| } |
| wg.Done() |
| }() |
| } |
| |
| max := 10000 |
| if testing.Short() { |
| max = 100 |
| } |
| stk := make([]runtime.StackRecord, 128) |
| for n := 0; n < max; n++ { |
| _, ok := runtime.GoroutineProfile(stk) |
| if !ok { |
| t.Fatalf("GoroutineProfile failed") |
| } |
| } |
| |
| // If the program didn't crash, we passed. |
| atomic.StoreUint32(&stop, 1) |
| wg.Wait() |
| } |