summaryrefslogtreecommitdiff
path: root/vendor/github.com/go-stack/stack/stack-go19_test.go
blob: d7aeea2588008760ce5f05bc907e823bba674969 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// +build go1.9

package stack_test

import (
	"runtime"
	"testing"

	"github.com/go-stack/stack"
)

func TestCallerInlinedPanic(t *testing.T) {
	t.Parallel()

	var line int

	defer func() {
		if recover() != nil {
			var pcs [32]uintptr
			n := runtime.Callers(1, pcs[:])
			frames := runtime.CallersFrames(pcs[:n])
			// count frames to runtime.sigpanic
			panicIdx := 0
			for {
				f, more := frames.Next()
				if f.Function == "runtime.sigpanic" {
					break
				}
				panicIdx++
				if !more {
					t.Fatal("no runtime.sigpanic entry on the stack")
				}
			}

			c := stack.Caller(panicIdx)
			if got, want := c.Frame().Function, "runtime.sigpanic"; got != want {
				t.Errorf("sigpanic frame: got name == %v, want name == %v", got, want)
			}

			c1 := stack.Caller(panicIdx + 1)
			if got, want := c1.Frame().Function, "github.com/go-stack/stack_test.inlinablePanic"; got != want {
				t.Errorf("TestCallerInlinedPanic frame: got name == %v, want name == %v", got, want)
			}
			if got, want := c1.Frame().Line, line; got != want {
				t.Errorf("TestCallerInlinedPanic frame: got line == %v, want line == %v", got, want)
			}
		}
	}()

	doPanic(t, &line)
	t.Fatal("failed to panic")
}

func doPanic(t *testing.T, panicLine *int) {
	_, _, line, ok := runtime.Caller(0)
	*panicLine = line + 11 // adjust to match line of panic below
	if !ok {
		t.Fatal("runtime.Caller(0) failed")
	}
	inlinablePanic()
}

func inlinablePanic() {
	// Initiate a sigpanic.
	var x *uintptr
	_ = *x
}