2012-02-10 22:19:24 -07:00
|
|
|
// 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"
|
2013-09-13 12:19:23 -06:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
. "runtime"
|
2014-02-20 14:18:05 -07:00
|
|
|
"runtime/debug"
|
2013-09-13 12:19:23 -06:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2012-02-10 22:19:24 -07:00
|
|
|
"testing"
|
2014-02-20 14:18:05 -07:00
|
|
|
"unsafe"
|
2012-02-10 22:19:24 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
var errf error
|
|
|
|
|
|
|
|
func errfn() error {
|
|
|
|
return errf
|
|
|
|
}
|
|
|
|
|
|
|
|
func errfn1() error {
|
|
|
|
return io.EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIfaceCmp100(b *testing.B) {
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
for j := 0; j < 100; j++ {
|
|
|
|
if errfn() == io.EOF {
|
|
|
|
b.Fatal("bad comparison")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkIfaceCmpNil100(b *testing.B) {
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
for j := 0; j < 100; j++ {
|
|
|
|
if errfn1() == nil {
|
|
|
|
b.Fatal("bad comparison")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-12-22 12:54:39 -07:00
|
|
|
|
|
|
|
func BenchmarkDefer(b *testing.B) {
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
defer1()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func defer1() {
|
|
|
|
defer func(x, y, z int) {
|
|
|
|
if recover() != nil || x != 1 || y != 2 || z != 3 {
|
|
|
|
panic("bad recover")
|
|
|
|
}
|
|
|
|
}(1, 2, 3)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkDefer10(b *testing.B) {
|
|
|
|
for i := 0; i < b.N/10; i++ {
|
|
|
|
defer2()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func defer2() {
|
|
|
|
for i := 0; i < 10; i++ {
|
|
|
|
defer func(x, y, z int) {
|
|
|
|
if recover() != nil || x != 1 || y != 2 || z != 3 {
|
|
|
|
panic("bad recover")
|
|
|
|
}
|
|
|
|
}(1, 2, 3)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkDeferMany(b *testing.B) {
|
|
|
|
for i := 0; i < b.N; i++ {
|
|
|
|
defer func(x, y, z int) {
|
|
|
|
if recover() != nil || x != 1 || y != 2 || z != 3 {
|
|
|
|
panic("bad recover")
|
|
|
|
}
|
|
|
|
}(1, 2, 3)
|
|
|
|
}
|
|
|
|
}
|
2013-09-13 12:19:23 -06:00
|
|
|
|
|
|
|
// The profiling signal handler needs to know whether it is executing runtime.gogo.
|
|
|
|
// The constant RuntimeGogoBytes in arch_*.h gives the size of the function;
|
|
|
|
// we don't have a way to obtain it from the linker (perhaps someday).
|
|
|
|
// Test that the constant matches the size determined by 'go tool nm -S'.
|
|
|
|
// The value reported will include the padding between runtime.gogo and the
|
|
|
|
// next function in memory. That's fine.
|
|
|
|
func TestRuntimeGogoBytes(t *testing.T) {
|
2013-12-17 20:17:47 -07:00
|
|
|
// TODO(brainman): delete when issue 6973 is fixed.
|
|
|
|
if GOOS == "windows" {
|
|
|
|
t.Skip("skipping broken test on windows")
|
|
|
|
}
|
2013-09-13 12:19:23 -06:00
|
|
|
dir, err := ioutil.TempDir("", "go-build")
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("failed to create temp directory: %v", err)
|
|
|
|
}
|
|
|
|
defer os.RemoveAll(dir)
|
|
|
|
|
|
|
|
out, err := exec.Command("go", "build", "-o", dir+"/hello", "../../../test/helloworld.go").CombinedOutput()
|
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("building hello world: %v\n%s", err, out)
|
|
|
|
}
|
|
|
|
|
2013-12-16 10:59:30 -07:00
|
|
|
out, err = exec.Command("go", "tool", "nm", "-size", dir+"/hello").CombinedOutput()
|
2013-09-13 12:19:23 -06:00
|
|
|
if err != nil {
|
|
|
|
t.Fatalf("go tool nm: %v\n%s", err, out)
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, line := range strings.Split(string(out), "\n") {
|
|
|
|
f := strings.Fields(line)
|
|
|
|
if len(f) == 4 && f[3] == "runtime.gogo" {
|
|
|
|
size, _ := strconv.Atoi(f[1])
|
|
|
|
if GogoBytes() != int32(size) {
|
|
|
|
t.Fatalf("RuntimeGogoBytes = %d, should be %d", GogoBytes(), size)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Fatalf("go tool nm did not report size for runtime.gogo")
|
|
|
|
}
|
2014-01-06 10:53:55 -07:00
|
|
|
|
|
|
|
// golang.org/issue/7063
|
|
|
|
func TestStopCPUProfilingWithProfilerOff(t *testing.T) {
|
|
|
|
SetCPUProfileRate(0)
|
|
|
|
}
|
2014-02-20 14:18:05 -07:00
|
|
|
|
|
|
|
func TestSetPanicOnFault(t *testing.T) {
|
2014-02-27 09:26:26 -07:00
|
|
|
// This currently results in a fault in the signal trampoline on
|
|
|
|
// dragonfly/386 - see issue 7421.
|
|
|
|
if GOOS == "dragonfly" && GOARCH == "386" {
|
|
|
|
t.Skip("skipping test on dragonfly/386")
|
|
|
|
}
|
|
|
|
|
2014-02-20 14:18:05 -07:00
|
|
|
old := debug.SetPanicOnFault(true)
|
|
|
|
defer debug.SetPanicOnFault(old)
|
|
|
|
|
|
|
|
defer func() {
|
|
|
|
if err := recover(); err == nil {
|
|
|
|
t.Fatalf("did not find error in recover")
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
|
|
|
var p *int
|
|
|
|
p = (*int)(unsafe.Pointer(^uintptr(0)))
|
|
|
|
println(*p)
|
|
|
|
t.Fatalf("still here - should have faulted")
|
|
|
|
}
|