/src/runtime/pprof/mprof_test.go

https://bitbucket.org/bnat6582/go192 · Go · 112 lines · 81 code · 22 blank · 9 comment · 8 complexity · 1cc19eedaec8bcba63e44ad618b77b6e 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
  5. import (
  6. "bytes"
  7. "fmt"
  8. "reflect"
  9. "regexp"
  10. "runtime"
  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. //go:noinline
  21. func allocateTransient2M() {
  22. memSink = make([]byte, 2<<20)
  23. }
  24. type Obj32 struct {
  25. link *Obj32
  26. pad [32 - unsafe.Sizeof(uintptr(0))]byte
  27. }
  28. var persistentMemSink *Obj32
  29. func allocatePersistent1K() {
  30. for i := 0; i < 32; i++ {
  31. // Can't use slice because that will introduce implicit allocations.
  32. obj := &Obj32{link: persistentMemSink}
  33. persistentMemSink = obj
  34. }
  35. }
  36. // Allocate transient memory using reflect.Call.
  37. func allocateReflectTransient() {
  38. memSink = make([]byte, 2<<20)
  39. }
  40. func allocateReflect() {
  41. rv := reflect.ValueOf(allocateReflectTransient)
  42. rv.Call(nil)
  43. }
  44. var memoryProfilerRun = 0
  45. func TestMemoryProfiler(t *testing.T) {
  46. // Disable sampling, otherwise it's difficult to assert anything.
  47. oldRate := runtime.MemProfileRate
  48. runtime.MemProfileRate = 1
  49. defer func() {
  50. runtime.MemProfileRate = oldRate
  51. }()
  52. // Allocate a meg to ensure that mcache.next_sample is updated to 1.
  53. for i := 0; i < 1024; i++ {
  54. memSink = make([]byte, 1024)
  55. }
  56. // Do the interesting allocations.
  57. allocateTransient1M()
  58. allocateTransient2M()
  59. allocatePersistent1K()
  60. allocateReflect()
  61. memSink = nil
  62. runtime.GC() // materialize stats
  63. var buf bytes.Buffer
  64. if err := Lookup("heap").WriteTo(&buf, 1); err != nil {
  65. t.Fatalf("failed to write heap profile: %v", err)
  66. }
  67. memoryProfilerRun++
  68. tests := []string{
  69. fmt.Sprintf(`%v: %v \[%v: %v\] @ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+
  70. # 0x[0-9,a-f]+ runtime/pprof\.allocatePersistent1K\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test\.go:40
  71. # 0x[0-9,a-f]+ runtime/pprof\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test\.go:74
  72. `, 32*memoryProfilerRun, 1024*memoryProfilerRun, 32*memoryProfilerRun, 1024*memoryProfilerRun),
  73. fmt.Sprintf(`0: 0 \[%v: %v\] @ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+
  74. # 0x[0-9,a-f]+ runtime/pprof\.allocateTransient1M\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test.go:21
  75. # 0x[0-9,a-f]+ runtime/pprof\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test.go:72
  76. `, (1<<10)*memoryProfilerRun, (1<<20)*memoryProfilerRun),
  77. fmt.Sprintf(`0: 0 \[%v: %v\] @ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+ 0x[0-9,a-f]+
  78. # 0x[0-9,a-f]+ runtime/pprof\.allocateTransient2M\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test.go:27
  79. # 0x[0-9,a-f]+ runtime/pprof\.TestMemoryProfiler\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test.go:73
  80. `, memoryProfilerRun, (2<<20)*memoryProfilerRun),
  81. fmt.Sprintf(`0: 0 \[%v: %v\] @( 0x[0-9,a-f]+)+
  82. # 0x[0-9,a-f]+ runtime/pprof\.allocateReflectTransient\+0x[0-9,a-f]+ .*/runtime/pprof/mprof_test.go:48
  83. `, memoryProfilerRun, (2<<20)*memoryProfilerRun),
  84. }
  85. for _, test := range tests {
  86. if !regexp.MustCompile(test).Match(buf.Bytes()) {
  87. t.Fatalf("The entry did not match:\n%v\n\nProfile:\n%v\n", test, buf.String())
  88. }
  89. }
  90. }