|  | // 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())) | 
|  | } | 
|  | } | 
|  | } |