Skip to content

Commit

Permalink
runtime: disable a signal by restoring the original disposition
Browse files Browse the repository at this point in the history
Fixes #13034.
Fixes #13042.
Update #9896.

Change-Id: I189f381090223dd07086848aac2d69d2c00d80c4
Reviewed-on: https://go-review.googlesource.com/18062
Reviewed-by: Russ Cox <rsc@golang.org>
  • Loading branch information
ianlancetaylor committed Jan 5, 2016
1 parent 6c8a141 commit a7d2b4d
Show file tree
Hide file tree
Showing 9 changed files with 486 additions and 25 deletions.
153 changes: 153 additions & 0 deletions misc/cgo/testcarchive/main3.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,153 @@
// 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 os/signal.Notify and os/signal.Reset.
// This is a lot like misc/cgo/testcshared/main5.c.

#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sched.h>

#include "libgo3.h"

static void die(const char* msg) {
perror(msg);
exit(EXIT_FAILURE);
}

static volatile sig_atomic_t sigioSeen;

static void ioHandler(int signo, siginfo_t* info, void* ctxt) {
sigioSeen = 1;
}

int main(int argc, char** argv) {
int verbose;
struct sigaction sa;
int i;

verbose = argc > 2;
setvbuf(stdout, NULL, _IONBF, 0);

if (verbose) {
printf("calling sigaction\n");
}

memset(&sa, 0, sizeof sa);
sa.sa_sigaction = ioHandler;
if (sigemptyset(&sa.sa_mask) < 0) {
die("sigemptyset");
}
sa.sa_flags = SA_SIGINFO;
if (sigaction(SIGIO, &sa, NULL) < 0) {
die("sigaction");
}

// At this point there should not be a Go signal handler
// installed for SIGIO.

if (verbose) {
printf("raising SIGIO\n");
}

if (raise(SIGIO) < 0) {
die("raise");
}

if (verbose) {
printf("waiting for sigioSeen\n");
}

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

sigioSeen = 0;

// Tell the Go code to catch SIGIO.

if (verbose) {
printf("calling CatchSIGIO\n");
}

CatchSIGIO();

if (verbose) {
printf("raising SIGIO\n");
}

if (raise(SIGIO) < 0) {
die("raise");
}

if (verbose) {
printf("calling SawSIGIO\n");
}

if (!SawSIGIO()) {
fprintf(stderr, "Go handler did not see SIGIO\n");
exit(EXIT_FAILURE);
}

if (sigioSeen != 0) {
fprintf(stderr, "C handler saw SIGIO when only Go handler should have\n");
exit(EXIT_FAILURE);
}

// Tell the Go code to stop catching SIGIO.

if (verbose) {
printf("calling ResetSIGIO\n");
}

ResetSIGIO();

if (verbose) {
printf("raising SIGIO\n");
}

if (raise(SIGIO) < 0) {
die("raise");
}

if (verbose) {
printf("calling SawSIGIO\n");
}

if (SawSIGIO()) {
fprintf(stderr, "Go handler saw SIGIO after Reset\n");
exit(EXIT_FAILURE);
}

if (verbose) {
printf("waiting for sigioSeen\n");
}

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

printf("PASS\n");
return 0;
}
44 changes: 44 additions & 0 deletions misc/cgo/testcarchive/src/libgo3/libgo3.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
// 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 "C"

import (
"os"
"os/signal"
"syscall"
"time"
)

// The channel used to read SIGIO signals.
var sigioChan chan os.Signal

// CatchSIGIO starts catching SIGIO signals.
//export CatchSIGIO
func CatchSIGIO() {
sigioChan = make(chan os.Signal, 1)
signal.Notify(sigioChan, syscall.SIGIO)
}

// ResetSIGIO stops catching SIGIO signals.
//export ResetSIGIO
func ResetSIGIO() {
signal.Reset(syscall.SIGIO)
}

// SawSIGIO returns whether we saw a SIGIO within a brief pause.
//export SawSIGIO
func SawSIGIO() C.int {
select {
case <-sigioChan:
return 1
case <-time.After(100 * time.Millisecond):
return 0
}
}

func main() {
}
18 changes: 13 additions & 5 deletions misc/cgo/testcarchive/test.bash
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ status=0
GOPATH=$(pwd) go install -buildmode=c-archive libgo
$(go env CC) $(go env GOGCCFLAGS) $ccargs -o testp main.c pkg/$(go env GOOS)_$(go env GOARCH)/libgo.a
if ! $bin arg1 arg2; then
echo "FAIL test1"
echo "FAIL test1a"
status=1
fi
rm -f libgo.a libgo.h testp
Expand All @@ -41,32 +41,40 @@ rm -f libgo.a libgo.h testp
GOPATH=$(pwd) go build -buildmode=c-archive src/libgo/libgo.go
$(go env CC) $(go env GOGCCFLAGS) $ccargs -o testp main.c libgo.a
if ! $bin arg1 arg2; then
echo "FAIL test2"
echo "FAIL test1b"
status=1
fi
rm -f libgo.a libgo.h testp

GOPATH=$(pwd) go build -buildmode=c-archive -o libgo.a libgo
$(go env CC) $(go env GOGCCFLAGS) $ccargs -o testp main.c libgo.a
if ! $bin arg1 arg2; then
echo "FAIL test3"
echo "FAIL test1c"
status=1
fi
rm -rf libgo.a libgo.h testp pkg

case "$(go env GOOS)/$(go env GOARCH)" in
"darwin/arm" | "darwin/arm64")
echo "Skipping test4; see https://golang.org/issue/13701"
echo "Skipping test2; see https://golang.org/issue/13701"
;;
*)
GOPATH=$(pwd) go build -buildmode=c-archive -o libgo2.a libgo2
$(go env CC) $(go env GOGCCFLAGS) $ccargs -o testp main2.c libgo2.a
if ! $bin; then
echo "FAIL test4"
echo "FAIL test2"
status=1
fi
rm -rf libgo2.a libgo2.h testp pkg
;;
esac

GOPATH=$(pwd) go build -buildmode=c-archive -o libgo3.a libgo3
$(go env CC) $(go env GOGCCFLAGS) $ccargs -o testp main3.c libgo3.a
if ! $bin; then
echo "FAIL test3"
status=1
fi
rm -rf libgo3.a libgo3.h testp pkg

exit $status
Loading

0 comments on commit a7d2b4d

Please sign in to comment.