aboutsummaryrefslogtreecommitdiff
path: root/test/inline_callers.go
diff options
context:
space:
mode:
authorDavid Lazar <lazard@golang.org>2017-03-06 14:48:36 -0500
committerDavid Lazar <lazard@golang.org>2017-03-29 17:22:08 +0000
commitee97216a1787a979911d43c0c5c582b5492a2205 (patch)
tree636c8c346cee5da888df84c5498dd2f0f984fee6 /test/inline_callers.go
parentf3f5b10e06f6bb29731e9213dd8745a9b1857568 (diff)
downloadgo-ee97216a1787a979911d43c0c5c582b5492a2205.tar.gz
go-ee97216a1787a979911d43c0c5c582b5492a2205.zip
runtime: handle inlined calls in runtime.Callers
The `skip` argument passed to runtime.Caller and runtime.Callers should be interpreted as the number of logical calls to skip (rather than the number of physical stack frames to skip). This changes runtime.Callers to skip inlined calls in addition to physical stack frames. The result value of runtime.Callers is a slice of program counters ([]uintptr) representing physical stack frames. If the `skip` parameter to runtime.Callers skips part-way into a physical frame, there is no convenient way to encode that in the resulting slice. To avoid changing the API in an incompatible way, our solution is to store the number of skipped logical calls of the first frame in the _second_ uintptr returned by runtime.Callers. Since this number is a small integer, we encode it as a valid PC value into a small symbol called: runtime.skipPleaseUseCallersFrames For example, if f() calls g(), g() calls `runtime.Callers(2, pcs)`, and g() is inlined into f, then the frame for f will be partially skipped, resulting in the following slice: pcs = []uintptr{pc_in_f, runtime.skipPleaseUseCallersFrames+1, ...} We store the skip PC in pcs[1] instead of pcs[0] so that `pcs[i:]` will truncate the captured stack trace rather than grow it for all i. Updates #19348. Change-Id: I1c56f89ac48c29e6f52a5d085567c6d77d499cf1 Reviewed-on: https://go-review.googlesource.com/37854 Run-TryBot: David Lazar <lazard@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Austin Clements <austin@google.com>
Diffstat (limited to 'test/inline_callers.go')
-rw-r--r--test/inline_callers.go72
1 files changed, 72 insertions, 0 deletions
diff --git a/test/inline_callers.go b/test/inline_callers.go
new file mode 100644
index 0000000000..c387362fa8
--- /dev/null
+++ b/test/inline_callers.go
@@ -0,0 +1,72 @@
+// run -gcflags -l=4
+
+// Copyright 2017 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 main
+
+import (
+ "log"
+ "runtime"
+)
+
+var skip int
+var npcs int
+var pcs = make([]uintptr, 32)
+
+func f() {
+ g()
+}
+
+func g() {
+ h()
+}
+
+func h() {
+ npcs = runtime.Callers(skip, pcs)
+}
+
+func testCallers(skp int) (frames []string) {
+ skip = skp
+ f()
+ for i := 0; i < npcs; i++ {
+ fn := runtime.FuncForPC(pcs[i])
+ frames = append(frames, fn.Name())
+ if fn.Name() == "main.main" {
+ break
+ }
+ }
+ return
+}
+
+var expectedFrames [][]string = [][]string{
+ 0: {"runtime.Callers", "main.testCallers", "main.main"},
+ 1: {"main.testCallers", "main.main"},
+ 2: {"main.testCallers", "runtime.skipPleaseUseCallersFrames", "main.main"},
+ 3: {"main.testCallers", "runtime.skipPleaseUseCallersFrames", "main.main"},
+ 4: {"main.testCallers", "runtime.skipPleaseUseCallersFrames", "main.main"},
+ 5: {"main.main"},
+}
+
+func same(xs, ys []string) bool {
+ if len(xs) != len(ys) {
+ return false
+ }
+ for i := range xs {
+ if xs[i] != ys[i] {
+ return false
+ }
+ }
+ return true
+}
+
+func main() {
+ for i := 0; i <= 5; i++ {
+ frames := testCallers(i)
+ expected := expectedFrames[i]
+ if !same(frames, expected) {
+ log.Fatalf("testCallers(%d):\n got %v\n want %v", i, frames, expected)
+ }
+ }
+}