mirror of
https://github.com/golang/go
synced 2024-11-19 02:04:42 -07:00
f7e51c1320
If non-Go code calls sigaltstack before a signal is received, use sigaltstack to determine the current signal stack and set the gsignal stack to use it. This makes the Go runtime more robust in the face of non-Go code. We still can't handle a disabled signal stack or a signal triggered with SA_ONSTACK clear, but we now give clear errors for those cases. Fixes #7227. Update #9896. Change-Id: Icb1607e01fd6461019b6d77d940e59b3aed4d258 Reviewed-on: https://go-review.googlesource.com/18102 Run-TryBot: Ian Lance Taylor <iant@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Mikio Hara <mikioh.mikioh@gmail.com>
72 lines
1.6 KiB
Go
72 lines
1.6 KiB
Go
// Copyright 2015 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.
|
|
|
|
// +build !windows
|
|
|
|
// Test that the Go runtime still works if C code changes the signal stack.
|
|
|
|
package cgotest
|
|
|
|
/*
|
|
#include <signal.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
|
|
static stack_t oss;
|
|
static char signalStack[SIGSTKSZ];
|
|
|
|
static void changeSignalStack() {
|
|
stack_t ss;
|
|
memset(&ss, 0, sizeof ss);
|
|
ss.ss_sp = signalStack;
|
|
ss.ss_flags = 0;
|
|
ss.ss_size = SIGSTKSZ;
|
|
if (sigaltstack(&ss, &oss) < 0) {
|
|
perror("sigaltstack");
|
|
abort();
|
|
}
|
|
}
|
|
|
|
static void restoreSignalStack() {
|
|
#if defined(__x86_64__) && defined(__APPLE__)
|
|
// The Darwin C library enforces a minimum that the kernel does not.
|
|
// This is OK since we allocated this much space in mpreinit,
|
|
// it was just removed from the buffer by stackalloc.
|
|
oss.ss_size = MINSIGSTKSZ;
|
|
#endif
|
|
if (sigaltstack(&oss, NULL) < 0) {
|
|
perror("sigaltstack restore");
|
|
abort();
|
|
}
|
|
}
|
|
|
|
static int zero() {
|
|
return 0;
|
|
}
|
|
*/
|
|
import "C"
|
|
|
|
import (
|
|
"runtime"
|
|
"testing"
|
|
)
|
|
|
|
func testSigaltstack(t *testing.T) {
|
|
switch {
|
|
case runtime.GOOS == "solaris", runtime.GOOS == "darwin" && (runtime.GOARCH == "arm" || runtime.GOARCH == "arm64"):
|
|
t.Skipf("switching signal stack not implemented on %s/s", runtime.GOOS, runtime.GOARCH)
|
|
}
|
|
|
|
C.changeSignalStack()
|
|
defer C.restoreSignalStack()
|
|
defer func() {
|
|
if recover() == nil {
|
|
t.Error("did not see expected panic")
|
|
}
|
|
}()
|
|
v := 1 / int(C.zero())
|
|
t.Errorf("unexpected success of division by zero == %d", v)
|
|
}
|