54c9c975f1
This is a port of https://golang.org/cl/18150 to the gccgo runtime. The previous behaviour of installing the signal handlers in a separate thread meant that Go initialization raced with non-Go initialization if the non-Go initialization also wanted to install signal handlers. Make installing signal handlers synchronous so that the process-wide behavior is predictable. Reviewed-on: https://go-review.googlesource.com/19494 From-SVN: r233393
177 lines
3.4 KiB
C
177 lines
3.4 KiB
C
// Copyright 2012 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.
|
|
|
|
// +build darwin dragonfly freebsd linux netbsd openbsd solaris
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include "runtime.h"
|
|
#include "defs.h"
|
|
#include "signal_unix.h"
|
|
|
|
extern SigTab runtime_sigtab[];
|
|
|
|
void
|
|
runtime_initsig(bool preinit)
|
|
{
|
|
int32 i;
|
|
SigTab *t;
|
|
|
|
// For c-archive/c-shared this is called by go-libmain.c with
|
|
// preinit == true.
|
|
if(runtime_isarchive && !preinit)
|
|
return;
|
|
|
|
// First call: basic setup.
|
|
for(i = 0; runtime_sigtab[i].sig != -1; i++) {
|
|
t = &runtime_sigtab[i];
|
|
if((t->flags == 0) || (t->flags & SigDefault))
|
|
continue;
|
|
|
|
t->fwdsig = runtime_getsig(i);
|
|
|
|
// For some signals, we respect an inherited SIG_IGN handler
|
|
// rather than insist on installing our own default handler.
|
|
// Even these signals can be fetched using the os/signal package.
|
|
switch(t->sig) {
|
|
case SIGHUP:
|
|
case SIGINT:
|
|
if(t->fwdsig == GO_SIG_IGN) {
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if(runtime_isarchive && (t->flags&SigPanic) == 0)
|
|
continue;
|
|
|
|
t->flags |= SigHandling;
|
|
runtime_setsig(i, runtime_sighandler, true);
|
|
}
|
|
}
|
|
|
|
void
|
|
runtime_sigenable(uint32 sig)
|
|
{
|
|
int32 i;
|
|
SigTab *t;
|
|
|
|
t = nil;
|
|
for(i = 0; runtime_sigtab[i].sig != -1; i++) {
|
|
if(runtime_sigtab[i].sig == (int32)sig) {
|
|
t = &runtime_sigtab[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(t == nil)
|
|
return;
|
|
|
|
if((t->flags & SigNotify) && !(t->flags & SigHandling)) {
|
|
t->flags |= SigHandling;
|
|
t->fwdsig = runtime_getsig(i);
|
|
runtime_setsig(i, runtime_sighandler, true);
|
|
}
|
|
}
|
|
|
|
void
|
|
runtime_sigdisable(uint32 sig)
|
|
{
|
|
int32 i;
|
|
SigTab *t;
|
|
|
|
t = nil;
|
|
for(i = 0; runtime_sigtab[i].sig != -1; i++) {
|
|
if(runtime_sigtab[i].sig == (int32)sig) {
|
|
t = &runtime_sigtab[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(t == nil)
|
|
return;
|
|
|
|
if((sig == SIGHUP || sig == SIGINT) && t->fwdsig == GO_SIG_IGN) {
|
|
t->flags &= ~SigHandling;
|
|
runtime_setsig(i, t->fwdsig, true);
|
|
}
|
|
}
|
|
|
|
void
|
|
runtime_sigignore(uint32 sig)
|
|
{
|
|
int32 i;
|
|
SigTab *t;
|
|
|
|
t = nil;
|
|
for(i = 0; runtime_sigtab[i].sig != -1; i++) {
|
|
if(runtime_sigtab[i].sig == (int32)sig) {
|
|
t = &runtime_sigtab[i];
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(t == nil)
|
|
return;
|
|
|
|
if((t->flags & SigNotify) != 0) {
|
|
t->flags &= ~SigHandling;
|
|
runtime_setsig(i, GO_SIG_IGN, true);
|
|
}
|
|
}
|
|
|
|
void
|
|
runtime_resetcpuprofiler(int32 hz)
|
|
{
|
|
struct itimerval it;
|
|
|
|
runtime_memclr((byte*)&it, sizeof it);
|
|
if(hz == 0) {
|
|
runtime_setitimer(ITIMER_PROF, &it, nil);
|
|
} else {
|
|
it.it_interval.tv_sec = 0;
|
|
it.it_interval.tv_usec = 1000000 / hz;
|
|
it.it_value = it.it_interval;
|
|
runtime_setitimer(ITIMER_PROF, &it, nil);
|
|
}
|
|
runtime_m()->profilehz = hz;
|
|
}
|
|
|
|
void
|
|
runtime_unblocksignals(void)
|
|
{
|
|
sigset_t sigset_none;
|
|
sigemptyset(&sigset_none);
|
|
pthread_sigmask(SIG_SETMASK, &sigset_none, nil);
|
|
}
|
|
|
|
void
|
|
runtime_crash(void)
|
|
{
|
|
int32 i;
|
|
|
|
#ifdef GOOS_darwin
|
|
// OS X core dumps are linear dumps of the mapped memory,
|
|
// from the first virtual byte to the last, with zeros in the gaps.
|
|
// Because of the way we arrange the address space on 64-bit systems,
|
|
// this means the OS X core file will be >128 GB and even on a zippy
|
|
// workstation can take OS X well over an hour to write (uninterruptible).
|
|
// Save users from making that mistake.
|
|
if(sizeof(void*) == 8)
|
|
return;
|
|
#endif
|
|
|
|
runtime_unblocksignals();
|
|
for(i = 0; runtime_sigtab[i].sig != -1; i++)
|
|
if(runtime_sigtab[i].sig == SIGABRT)
|
|
break;
|
|
runtime_setsig(i, GO_SIG_DFL, false);
|
|
runtime_raise(SIGABRT);
|
|
}
|
|
|
|
void
|
|
runtime_raise(int32 sig)
|
|
{
|
|
raise(sig);
|
|
}
|