runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import "fmt"
|
|
|
|
|
|
|
|
/*
|
2015-12-16 13:16:17 -07:00
|
|
|
#cgo CFLAGS: -pthread
|
|
|
|
#cgo LDFLAGS: -pthread
|
|
|
|
|
runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
#include <signal.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
2015-12-16 13:16:17 -07:00
|
|
|
#include <string.h>
|
|
|
|
#include <pthread.h>
|
runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
|
|
|
|
int *p;
|
|
|
|
static void sigsegv() {
|
|
|
|
*p = 1;
|
|
|
|
fprintf(stderr, "ERROR: C SIGSEGV not thrown on caught?.\n");
|
|
|
|
exit(2);
|
|
|
|
}
|
|
|
|
|
2015-12-16 13:16:17 -07:00
|
|
|
static void segvhandler(int signum) {
|
runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
if (signum == SIGSEGV) {
|
|
|
|
exit(0); // success
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-16 13:16:17 -07:00
|
|
|
static volatile sig_atomic_t sigioSeen;
|
|
|
|
|
|
|
|
// Use up some stack space.
|
|
|
|
static void recur(int i, char *p) {
|
|
|
|
char a[1024];
|
|
|
|
|
|
|
|
*p = '\0';
|
|
|
|
if (i > 0) {
|
|
|
|
recur(i - 1, a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void iohandler(int signum) {
|
|
|
|
char a[1024];
|
|
|
|
|
|
|
|
recur(4, a);
|
|
|
|
sigioSeen = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void* sigioThread(void* arg __attribute__ ((unused))) {
|
|
|
|
raise(SIGIO);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sigioOnThread() {
|
|
|
|
pthread_t tid;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
pthread_create(&tid, NULL, sigioThread, NULL);
|
|
|
|
pthread_join(tid, NULL);
|
|
|
|
|
|
|
|
// Wait until the signal has been delivered.
|
|
|
|
i = 0;
|
|
|
|
while (!sigioSeen) {
|
|
|
|
if (sched_yield() < 0) {
|
|
|
|
perror("sched_yield");
|
|
|
|
}
|
|
|
|
i++;
|
|
|
|
if (i > 10000) {
|
|
|
|
fprintf(stderr, "looping too long waiting for signal\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
static void __attribute__ ((constructor)) sigsetup(void) {
|
|
|
|
struct sigaction act;
|
2015-12-16 13:16:17 -07:00
|
|
|
|
|
|
|
memset(&act, 0, sizeof act);
|
|
|
|
act.sa_handler = segvhandler;
|
|
|
|
sigaction(SIGSEGV, &act, NULL);
|
|
|
|
|
|
|
|
act.sa_handler = iohandler;
|
|
|
|
sigaction(SIGIO, &act, NULL);
|
runtime: signal forwarding
Forward signals to signal handlers installed before Go installs its own,
under certain circumstances. In particular, as iant@ suggests, signals are
forwarded iff:
(1) a non-SIG_DFL signal handler existed before Go, and
(2) signal is synchronous (i.e., one of SIGSEGV, SIGBUS, SIGFPE), and
(3a) signal occured on a non-Go thread, or
(3b) signal occurred on a Go thread but in CGo code.
Supported only on Linux, for now.
Change-Id: I403219ee47b26cf65da819fb86cf1ec04d3e25f5
Reviewed-on: https://go-review.googlesource.com/8712
Reviewed-by: Ian Lance Taylor <iant@golang.org>
2015-04-09 12:12:12 -06:00
|
|
|
}
|
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
|
|
|
|
var p *byte
|
|
|
|
|
|
|
|
func f() (ret bool) {
|
|
|
|
defer func() {
|
|
|
|
if recover() == nil {
|
|
|
|
fmt.Errorf("ERROR: couldn't raise SIGSEGV in Go.")
|
|
|
|
C.exit(2)
|
|
|
|
}
|
|
|
|
ret = true
|
|
|
|
}()
|
|
|
|
*p = 1
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
// Test that the signal originating in Go is handled (and recovered) by Go.
|
|
|
|
if !f() {
|
|
|
|
fmt.Errorf("couldn't recover from SIGSEGV in Go.")
|
|
|
|
C.exit(2)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Test that the signal originating in C is handled by C.
|
|
|
|
C.sigsegv()
|
|
|
|
}
|