/third_party/gofrontend/libgo/go/runtime/pprof/mprof_test.go

http://github.com/axw/llgo · Go · 101 lines · 73 code · 18 blank · 10 comment · 10 complexity · 7f31a8c9f38c8f215bdd950d085d4912 MD5 · raw file

  1. // Copyright 2014 The Go Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package pprof_test
  5. import (
  6. "bytes"
  7. "fmt"
  8. "regexp"
  9. "runtime"
  10. . "runtime/pprof"
  11. "testing"
  12. "unsafe"
  13. )
  14. var memSink interface{}
  15. func allocateTransient1M() {
  16. for i := 0; i < 1024; i++ {
  17. memSink = &struct{ x [1024]byte }{}
  18. }
  19. }
  20. func allocateTransient2M() {
  21. // prevent inlining
  22. if memSink == nil {
  23. panic("bad")
  24. }
  25. memSink = make([]byte, 2<<20)
  26. }
  27. type Obj32 struct {
  28. link *Obj32
  29. pad [32 - unsafe.Sizeof(uintptr(0))]byte
  30. }
  31. var persistentMemSink *Obj32
  32. func allocatePersistent1K() {
  33. for i := 0; i < 32; i++ {
  34. // Can't use slice because that will introduce implicit allocations.
  35. obj := &Obj32{link: persistentMemSink}
  36. persistentMemSink = obj
  37. }
  38. }
  39. var memoryProfilerRun = 0
  40. func TestMemoryProfiler(t *testing.T) {
  41. // Disable sampling, otherwise it's difficult to assert anything.
  42. oldRate := runtime.MemProfileRate
  43. runtime.MemProfileRate = 1
  44. defer func() {
  45. runtime.MemProfileRate = oldRate
  46. }()
  47. // Allocate a meg to ensure that mcache.next_sample is updated to 1.
  48. for i := 0; i < 1024; i++ {
  49. memSink = make([]byte, 1024)
  50. }
  51. // Do the interesting allocations.
  52. allocateTransient1M()
  53. allocateTransient2M()
  54. allocatePersistent1K()
  55. memSink = nil
  56. runtime.GC() // materialize stats
  57. var buf bytes.Buffer
  58. if err := Lookup("heap").WriteTo(&buf, 1); err != nil {
  59. t.Fatalf("failed to write heap profile: %v", err)
  60. }
  61. memoryProfilerRun++
  62. tests := []string{
  63. fmt.Sprintf(`%v: %v \[%v: %v\] @ 0x[0-9,a-f x]+
  64. # 0x[0-9,a-f]+ pprof_test\.allocatePersistent1K\+0x[0-9,a-f]+ .*/mprof_test\.go:43
  65. # 0x[0-9,a-f]+ runtime_pprof_test\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/mprof_test\.go:66
  66. `, 32*memoryProfilerRun, 1024*memoryProfilerRun, 32*memoryProfilerRun, 1024*memoryProfilerRun),
  67. fmt.Sprintf(`0: 0 \[%v: %v\] @ 0x[0-9,a-f x]+
  68. # 0x[0-9,a-f]+ pprof_test\.allocateTransient1M\+0x[0-9,a-f]+ .*/mprof_test.go:21
  69. # 0x[0-9,a-f]+ runtime_pprof_test\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/mprof_test.go:64
  70. `, (1<<10)*memoryProfilerRun, (1<<20)*memoryProfilerRun),
  71. // This should start with "0: 0" but gccgo's imprecise
  72. // GC means that sometimes the value is not collected.
  73. fmt.Sprintf(`(0|%v): (0|%v) \[%v: %v\] @ 0x[0-9,a-f x]+
  74. # 0x[0-9,a-f]+ pprof_test\.allocateTransient2M\+0x[0-9,a-f]+ .*/mprof_test.go:30
  75. # 0x[0-9,a-f]+ runtime_pprof_test\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/mprof_test.go:65
  76. `, memoryProfilerRun, (2<<20)*memoryProfilerRun, memoryProfilerRun, (2<<20)*memoryProfilerRun),
  77. }
  78. for _, test := range tests {
  79. if !regexp.MustCompile(test).Match(buf.Bytes()) {
  80. t.Fatalf("The entry did not match:\n%v\n\nProfile:\n%v\n", test, buf.String())
  81. }
  82. }
  83. }