mirror of
https://github.com/golang/go
synced 2024-11-14 09:10:27 -07:00
956879dd0b
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>
78 lines
1.4 KiB
Go
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()))
|
|
}
|
|
}
|
|
}
|