| |
| |
| |
|
|
| package debug_test |
|
|
| import ( |
| "os" |
| "runtime" |
| . "runtime/debug" |
| "testing" |
| ) |
|
|
| func TestWriteHeapDumpNonempty(t *testing.T) { |
| if runtime.GOOS == "js" { |
| t.Skipf("WriteHeapDump is not available on %s.", runtime.GOOS) |
| } |
| f, err := os.CreateTemp("", "heapdumptest") |
| if err != nil { |
| t.Fatalf("TempFile failed: %v", err) |
| } |
| defer os.Remove(f.Name()) |
| defer f.Close() |
| WriteHeapDump(f.Fd()) |
| fi, err := f.Stat() |
| if err != nil { |
| t.Fatalf("Stat failed: %v", err) |
| } |
| const minSize = 1 |
| if size := fi.Size(); size < minSize { |
| t.Fatalf("Heap dump size %d bytes, expected at least %d bytes", size, minSize) |
| } |
| } |
|
|
| type Obj struct { |
| x, y int |
| } |
|
|
| func objfin(x *Obj) { |
| |
| } |
|
|
| func TestWriteHeapDumpFinalizers(t *testing.T) { |
| if runtime.GOOS == "js" { |
| t.Skipf("WriteHeapDump is not available on %s.", runtime.GOOS) |
| } |
| f, err := os.CreateTemp("", "heapdumptest") |
| if err != nil { |
| t.Fatalf("TempFile failed: %v", err) |
| } |
| defer os.Remove(f.Name()) |
| defer f.Close() |
|
|
| |
| println("allocating objects") |
| x := &Obj{} |
| runtime.SetFinalizer(x, objfin) |
| y := &Obj{} |
| runtime.SetFinalizer(y, objfin) |
|
|
| |
| println("starting gc") |
| runtime.GC() |
|
|
| |
| println("starting dump") |
| WriteHeapDump(f.Fd()) |
| println("done dump") |
| } |
|
|
| type G[T any] struct{} |
| type I interface { |
| M() |
| } |
|
|
| |
| func (g G[T]) M() {} |
|
|
| var dummy I = G[int]{} |
| var dummy2 I = G[G[int]]{} |
|
|
| func TestWriteHeapDumpTypeName(t *testing.T) { |
| if runtime.GOOS == "js" { |
| t.Skipf("WriteHeapDump is not available on %s.", runtime.GOOS) |
| } |
| f, err := os.CreateTemp("", "heapdumptest") |
| if err != nil { |
| t.Fatalf("TempFile failed: %v", err) |
| } |
| defer os.Remove(f.Name()) |
| defer f.Close() |
| WriteHeapDump(f.Fd()) |
| dummy.M() |
| dummy2.M() |
| } |
|
|