// Copyright 2018 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 freebsd,amd64 #include <errno.h> #include <stddef.h> #include <stdint.h> #include <string.h> #include <signal.h> #include "libcgo.h" // go_sigaction_t is a C version of the sigactiont struct from // os_freebsd.go. This definition — and its conversion to and from struct // sigaction — are specific to freebsd/amd64. typedef struct { uint32_t __bits[_SIG_WORDS]; } go_sigset_t; typedef struct { uintptr_t handler; int32_t flags; go_sigset_t mask; } go_sigaction_t; int32_t x_cgo_sigaction(intptr_t signum, const go_sigaction_t *goact, go_sigaction_t *oldgoact) { int32_t ret; struct sigaction act; struct sigaction oldact; size_t i; _cgo_tsan_acquire(); memset(&act, 0, sizeof act); memset(&oldact, 0, sizeof oldact); if (goact) { if (goact->flags & SA_SIGINFO) { act.sa_sigaction = (void(*)(int, siginfo_t*, void*))(goact->handler); } else { act.sa_handler = (void(*)(int))(goact->handler); } sigemptyset(&act.sa_mask); for (i = 0; i < 8 * sizeof(goact->mask); i++) { if (goact->mask.__bits[i/32] & ((uint32_t)(1)<<(i&31))) { sigaddset(&act.sa_mask, i+1); } } act.sa_flags = goact->flags; } ret = sigaction(signum, goact ? &act : NULL, oldgoact ? &oldact : NULL); if (ret == -1) { // runtime.sigaction expects _cgo_sigaction to return errno on error. _cgo_tsan_release(); return errno; } if (oldgoact) { if (oldact.sa_flags & SA_SIGINFO) { oldgoact->handler = (uintptr_t)(oldact.sa_sigaction); } else { oldgoact->handler = (uintptr_t)(oldact.sa_handler); } for (i = 0 ; i < _SIG_WORDS; i++) { oldgoact->mask.__bits[i] = 0; } for (i = 0; i < 8 * sizeof(oldgoact->mask); i++) { if (sigismember(&oldact.sa_mask, i+1) == 1) { oldgoact->mask.__bits[i/32] |= (uint32_t)(1)<<(i&31); } } oldgoact->flags = oldact.sa_flags; } _cgo_tsan_release(); return ret; }