mirror of
https://github.com/golang/go
synced 2024-11-08 13:56:21 -07:00
d05ce23756
Occasionally the signal will be sent to a Go thread, which will cause the program to exit with SIGQUIT rather than SIGSEGV. Add TestSignalForwardingGo to test the case where the signal is expected to be delivered to a Go thread. This is a roll forward of CL 419014 which was rolled back in CL 424954. This CL differs from 419014 in that it skips TestSignalForwardingGo on darwin-amd64. Fixes #53907 Change-Id: I5df3fd610c068df3bd48d9b3d7a9379248b97999 Reviewed-on: https://go-review.googlesource.com/c/go/+/425002 Run-TryBot: Ian Lance Taylor <iant@google.com> Reviewed-by: Bryan Mills <bcmills@google.com> Run-TryBot: Ian Lance Taylor <iant@golang.org> TryBot-Result: Gopher Robot <gobot@golang.org> Reviewed-by: Ian Lance Taylor <iant@google.com> Auto-Submit: Ian Lance Taylor <iant@google.com>
106 lines
2.0 KiB
C
106 lines
2.0 KiB
C
// 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.
|
|
|
|
// Test for verifying that the Go runtime properly forwards
|
|
// signals when non-Go signals are raised.
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/time.h>
|
|
#include <sys/select.h>
|
|
|
|
#include "libgo2.h"
|
|
|
|
int *nilp;
|
|
|
|
int main(int argc, char** argv) {
|
|
int verbose;
|
|
int test;
|
|
|
|
if (argc < 2) {
|
|
printf("Missing argument\n");
|
|
return 1;
|
|
}
|
|
|
|
test = atoi(argv[1]);
|
|
|
|
verbose = (argc > 2);
|
|
|
|
Noop();
|
|
|
|
switch (test) {
|
|
case 1: {
|
|
if (verbose) {
|
|
printf("attempting segfault\n");
|
|
}
|
|
|
|
*nilp = 0;
|
|
break;
|
|
}
|
|
|
|
case 2: {
|
|
struct timeval tv;
|
|
|
|
if (verbose) {
|
|
printf("attempting external signal test\n");
|
|
}
|
|
|
|
fprintf(stderr, "OK\n");
|
|
fflush(stderr);
|
|
|
|
// The program should be interrupted before
|
|
// this sleep finishes. We use select rather
|
|
// than sleep because in older versions of
|
|
// glibc the sleep function does some signal
|
|
// fiddling to handle SIGCHLD. If this
|
|
// program is fiddling signals just when the
|
|
// test program sends the signal, the signal
|
|
// may be delivered to a Go thread which will
|
|
// break this test.
|
|
tv.tv_sec = 60;
|
|
tv.tv_usec = 0;
|
|
select(0, NULL, NULL, NULL, &tv);
|
|
|
|
break;
|
|
}
|
|
case 3: {
|
|
if (verbose) {
|
|
printf("attempting SIGPIPE\n");
|
|
}
|
|
|
|
int fd[2];
|
|
if (pipe(fd) != 0) {
|
|
printf("pipe(2) failed\n");
|
|
return 0;
|
|
}
|
|
// Close the reading end.
|
|
close(fd[0]);
|
|
// Expect that write(2) fails (EPIPE)
|
|
if (write(fd[1], "some data", 9) != -1) {
|
|
printf("write(2) unexpectedly succeeded\n");
|
|
return 0;
|
|
}
|
|
printf("did not receive SIGPIPE\n");
|
|
return 0;
|
|
}
|
|
case 4: {
|
|
fprintf(stderr, "OK\n");
|
|
fflush(stderr);
|
|
|
|
if (verbose) {
|
|
printf("calling Block\n");
|
|
}
|
|
Block();
|
|
}
|
|
default:
|
|
printf("Unknown test: %d\n", test);
|
|
return 0;
|
|
}
|
|
|
|
printf("FAIL\n");
|
|
return 0;
|
|
}
|