mirror of
https://github.com/golang/go
synced 2024-11-18 07:14:44 -07:00
a409356c54
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. Fixes #53907 Change-Id: Iaefb964c2be4a815c11c507fa89648f8a7740ba9 Reviewed-on: https://go-review.googlesource.com/c/go/+/419014 Run-TryBot: Ian Lance Taylor <iant@golang.org> Auto-Submit: Ian Lance Taylor <iant@google.com> Reviewed-by: Austin Clements <austin@google.com> TryBot-Result: Gopher Robot <gobot@golang.org> Reviewed-by: Bryan Mills <bcmills@google.com> Reviewed-by: Ian Lance Taylor <iant@google.com> Run-TryBot: 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;
|
|
}
|