1
0
mirror of https://github.com/golang/go synced 2024-11-26 08:38:01 -07:00
go/misc/cgo/testcarchive/testdata/main5.c
Ian Lance Taylor d05ce23756 misc/cgo/testcarchive: permit SIGQUIT for TestSignalForwardingExternal
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>
2022-08-19 23:07:11 +00:00

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;
}