1
0
mirror of https://github.com/golang/go synced 2024-11-26 21:21:34 -07:00
go/test/inline_caller.go
Keith Randall 956879dd0b runtime: make FuncForPC return the innermost inlined frame
Returning the innermost frame instead of the outermost
makes code that walks the results of runtime.Caller{,s}
still work correctly in the presence of mid-stack inlining.

Fixes #29582

Change-Id: I2392e3dd5636eb8c6f58620a61cef2194fe660a7
Reviewed-on: https://go-review.googlesource.com/c/156364
Run-TryBot: Keith Randall <khr@golang.org>
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2019-01-08 21:54:04 +00:00

78 lines
1.4 KiB
Go

// 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 (
"fmt"
"runtime"
)
type frame struct {
pc uintptr
file string
line int
ok bool
}
var (
skip int
globalFrame frame
)
func f() {
g() // line 27
}
func g() {
h() // line 31
}
func h() {
x := &globalFrame
x.pc, x.file, x.line, x.ok = runtime.Caller(skip) // line 36
}
//go:noinline
func testCaller(skp int) frame {
skip = skp
f() // line 42
frame := globalFrame
if !frame.ok {
panic(fmt.Sprintf("skip=%d runtime.Caller failed", skp))
}
return frame
}
type wantFrame struct {
funcName string
line int
}
// -1 means don't care
var expected = []wantFrame{
0: {"main.h", 36},
1: {"main.g", 31},
2: {"main.f", 27},
3: {"main.testCaller", 42},
4: {"main.main", 68},
5: {"runtime.main", -1},
6: {"runtime.goexit", -1},
}
func main() {
for i := 0; i <= 6; i++ {
frame := testCaller(i) // line 68
fn := runtime.FuncForPC(frame.pc)
if expected[i].line >= 0 && frame.line != expected[i].line {
panic(fmt.Sprintf("skip=%d expected line %d, got line %d", i, expected[i].line, frame.line))
}
if fn.Name() != expected[i].funcName {
panic(fmt.Sprintf("skip=%d expected function %s, got %s", i, expected[i].funcName, fn.Name()))
}
}
}