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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
// Copyright 2012 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 runtime_test
import (
"io/ioutil"
"os"
"os/exec"
"path/filepath"
"runtime"
"testing"
"text/template"
)
type crashTest struct {
Cgo bool
}
// This test is a separate program, because it is testing
// both main (m0) and non-main threads (m).
func testCrashHandler(t *testing.T, ct *crashTest) {
if runtime.GOOS == "freebsd" || runtime.GOOS == "netbsd" {
// TODO(brainman): do not know why this test fails on freebsd
// TODO(jsing): figure out why this causes delayed failures
// on NetBSD - http://golang.org/issue/3954
t.Logf("skipping test on %q", runtime.GOOS)
return
}
st := template.Must(template.New("crashSource").Parse(crashSource))
dir, err := ioutil.TempDir("", "go-build")
if err != nil {
t.Fatalf("failed to create temp directory: %v", err)
}
defer os.RemoveAll(dir)
src := filepath.Join(dir, "main.go")
f, err := os.Create(src)
if err != nil {
t.Fatalf("failed to create %v: %v", src, err)
}
err = st.Execute(f, ct)
if err != nil {
f.Close()
t.Fatalf("failed to execute template: %v", err)
}
f.Close()
got, err := exec.Command("go", "run", src).CombinedOutput()
if err != nil {
t.Fatalf("program exited with error: %v\n%v", err, string(got))
}
want := "main: recovered done\nnew-thread: recovered done\nsecond-new-thread: recovered done\nmain-again: recovered done\n"
if string(got) != string(want) {
t.Fatalf("expected %q, but got %q", string(want), string(got))
}
}
func TestCrashHandler(t *testing.T) {
testCrashHandler(t, &crashTest{Cgo: false})
}
const crashSource = `
package main
import (
"fmt"
"runtime"
)
{{if .Cgo}}
import "C"
{{end}}
func test(name string) {
defer func() {
if x := recover(); x != nil {
fmt.Printf(" recovered")
}
fmt.Printf(" done\n")
}()
fmt.Printf("%s:", name)
var s *string
_ = *s
fmt.Print("SHOULD NOT BE HERE")
}
func testInNewThread(name string) {
c := make(chan bool)
go func() {
runtime.LockOSThread()
test(name)
c <- true
}()
<-c
}
func main() {
runtime.LockOSThread()
test("main")
testInNewThread("new-thread")
testInNewThread("second-new-thread")
test("main-again")
}
`
|