diff options
Diffstat (limited to 'src/cmd/compile/internal/gc/util.go')
-rw-r--r-- | src/cmd/compile/internal/gc/util.go | 103 |
1 files changed, 103 insertions, 0 deletions
diff --git a/src/cmd/compile/internal/gc/util.go b/src/cmd/compile/internal/gc/util.go new file mode 100644 index 0000000..58be2f8 --- /dev/null +++ b/src/cmd/compile/internal/gc/util.go @@ -0,0 +1,103 @@ +// Copyright 2015 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. + +package gc + +import ( + "os" + "runtime" + "runtime/pprof" +) + +// Line returns n's position as a string. If n has been inlined, +// it uses the outermost position where n has been inlined. +func (n *Node) Line() string { + return linestr(n.Pos) +} + +var atExitFuncs []func() + +func atExit(f func()) { + atExitFuncs = append(atExitFuncs, f) +} + +func Exit(code int) { + for i := len(atExitFuncs) - 1; i >= 0; i-- { + f := atExitFuncs[i] + atExitFuncs = atExitFuncs[:i] + f() + } + os.Exit(code) +} + +var ( + blockprofile string + cpuprofile string + memprofile string + memprofilerate int64 + traceprofile string + traceHandler func(string) + mutexprofile string +) + +func startProfile() { + if cpuprofile != "" { + f, err := os.Create(cpuprofile) + if err != nil { + Fatalf("%v", err) + } + if err := pprof.StartCPUProfile(f); err != nil { + Fatalf("%v", err) + } + atExit(pprof.StopCPUProfile) + } + if memprofile != "" { + if memprofilerate != 0 { + runtime.MemProfileRate = int(memprofilerate) + } + f, err := os.Create(memprofile) + if err != nil { + Fatalf("%v", err) + } + atExit(func() { + // Profile all outstanding allocations. + runtime.GC() + // compilebench parses the memory profile to extract memstats, + // which are only written in the legacy pprof format. + // See golang.org/issue/18641 and runtime/pprof/pprof.go:writeHeap. + const writeLegacyFormat = 1 + if err := pprof.Lookup("heap").WriteTo(f, writeLegacyFormat); err != nil { + Fatalf("%v", err) + } + }) + } else { + // Not doing memory profiling; disable it entirely. + runtime.MemProfileRate = 0 + } + if blockprofile != "" { + f, err := os.Create(blockprofile) + if err != nil { + Fatalf("%v", err) + } + runtime.SetBlockProfileRate(1) + atExit(func() { + pprof.Lookup("block").WriteTo(f, 0) + f.Close() + }) + } + if mutexprofile != "" { + f, err := os.Create(mutexprofile) + if err != nil { + Fatalf("%v", err) + } + startMutexProfiling() + atExit(func() { + pprof.Lookup("mutex").WriteTo(f, 0) + f.Close() + }) + } + if traceprofile != "" && traceHandler != nil { + traceHandler(traceprofile) + } +} |