liyujie
2025-08-28 d9927380ed7c8366f762049be9f3fee225860833
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
// 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;
}