1
0
mirror of https://github.com/DataDog/go-profiler-notes.git synced 2025-07-12 23:50:13 +02:00
Files
Felix Geisendörfer 6dd649f6b8 small change
2021-03-26 14:29:15 +01:00

69 lines
1.4 KiB
Go

package main
import (
"fmt"
"runtime"
"testing"
"time"
)
func BenchmarkProfilerGoroutines(b *testing.B) {
max := 1024 * 1024
buf := make([]runtime.StackRecord, max*2)
for g := 1; g <= max; g = g * 2 {
g := g
name := fmt.Sprintf("%d goroutines", g)
b.Run(name, func(b *testing.B) {
initalRoutines := runtime.NumGoroutine()
readyCh := make(chan struct{})
stopCh := make(chan struct{})
for i := 0; i < g; i++ {
go atStackDepth(16, func() {
defer func() { stopCh <- struct{}{} }()
readyCh <- struct{}{}
})
<-readyCh
}
b.ResetTimer()
for i := 0; i < b.N; i++ {
n, ok := runtime.GoroutineProfile(buf)
if !ok {
b.Logf("GoroutineProfile not ok")
} else if gotRoutines := n - initalRoutines; gotRoutines != g {
b.Logf("want %d goroutines, but got %d on iteration %d", g, gotRoutines, i)
}
}
b.StopTimer()
for i := 0; i < g; i++ {
<-stopCh
}
start := time.Now()
for i := 0; ; i++ {
if runtime.NumGoroutine() == initalRoutines {
break
}
time.Sleep(20 * time.Millisecond)
if time.Since(start) > 10*time.Second {
b.Fatalf("%d goroutines still running, want %d", runtime.NumGoroutine(), initalRoutines)
}
}
})
}
}
func atStackDepth(depth int, fn func()) {
pcs := make([]uintptr, depth+10)
n := runtime.Callers(1, pcs)
if n > depth {
panic("depth exceeded")
} else if n < depth {
atStackDepth(depth, fn)
return
}
fn()
}