| // 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 |
| |
| /* |
| #include <signal.h> |
| #include <unistd.h> |
| #include <stdlib.h> |
| #include <stdio.h> |
| |
| // Raise SIGPIPE. |
| static void CRaiseSIGPIPE() { |
| int fds[2]; |
| |
| if (pipe(fds) == -1) { |
| perror("pipe"); |
| exit(EXIT_FAILURE); |
| } |
| // Close the reader end |
| close(fds[0]); |
| // Write to the writer end to provoke a SIGPIPE |
| if (write(fds[1], "some data", 9) != -1) { |
| fprintf(stderr, "write to a closed pipe succeeded\n"); |
| exit(EXIT_FAILURE); |
| } |
| close(fds[1]); |
| } |
| */ |
| import "C" |
| |
| import ( |
| "fmt" |
| "os" |
| "runtime" |
| ) |
| |
| // RunGoroutines starts some goroutines that don't do anything. |
| // The idea is to get some threads going, so that a signal will be delivered |
| // to a thread started by Go. |
| //export RunGoroutines |
| func RunGoroutines() { |
| for i := 0; i < 4; i++ { |
| go func() { |
| runtime.LockOSThread() |
| select {} |
| }() |
| } |
| } |
| |
| // Block blocks the current thread while running Go code. |
| //export Block |
| func Block() { |
| select {} |
| } |
| |
| var P *byte |
| |
| // TestSEGV makes sure that an invalid address turns into a run-time Go panic. |
| //export TestSEGV |
| func TestSEGV() { |
| defer func() { |
| if recover() == nil { |
| fmt.Fprintln(os.Stderr, "no panic from segv") |
| os.Exit(1) |
| } |
| }() |
| *P = 0 |
| fmt.Fprintln(os.Stderr, "continued after segv") |
| os.Exit(1) |
| } |
| |
| // Noop ensures that the Go runtime is initialized. |
| //export Noop |
| func Noop() { |
| } |
| |
| // Raise SIGPIPE. |
| //export GoRaiseSIGPIPE |
| func GoRaiseSIGPIPE() { |
| C.CRaiseSIGPIPE() |
| } |
| |
| func main() { |
| } |