1*71db0c75SAndroid Build Coastguard Worker //===-- Internal header for Linux signals -----------------------*- C++ -*-===//
2*71db0c75SAndroid Build Coastguard Worker //
3*71db0c75SAndroid Build Coastguard Worker // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*71db0c75SAndroid Build Coastguard Worker // See https://llvm.org/LICENSE.txt for license information.
5*71db0c75SAndroid Build Coastguard Worker // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*71db0c75SAndroid Build Coastguard Worker //
7*71db0c75SAndroid Build Coastguard Worker //===----------------------------------------------------------------------===//
8*71db0c75SAndroid Build Coastguard Worker
9*71db0c75SAndroid Build Coastguard Worker #ifndef LLVM_LIBC_SRC_SIGNAL_LINUX_SIGNAL_UTILS_H
10*71db0c75SAndroid Build Coastguard Worker #define LLVM_LIBC_SRC_SIGNAL_LINUX_SIGNAL_UTILS_H
11*71db0c75SAndroid Build Coastguard Worker
12*71db0c75SAndroid Build Coastguard Worker #include "hdr/types/sigset_t.h"
13*71db0c75SAndroid Build Coastguard Worker #include "src/__support/OSUtil/syscall.h" // For internal syscall function.
14*71db0c75SAndroid Build Coastguard Worker #include "src/__support/common.h"
15*71db0c75SAndroid Build Coastguard Worker #include "src/__support/macros/config.h"
16*71db0c75SAndroid Build Coastguard Worker
17*71db0c75SAndroid Build Coastguard Worker #include <signal.h> // sigaction
18*71db0c75SAndroid Build Coastguard Worker #include <stddef.h>
19*71db0c75SAndroid Build Coastguard Worker #include <sys/syscall.h> // For syscall numbers.
20*71db0c75SAndroid Build Coastguard Worker
21*71db0c75SAndroid Build Coastguard Worker namespace LIBC_NAMESPACE_DECL {
22*71db0c75SAndroid Build Coastguard Worker
23*71db0c75SAndroid Build Coastguard Worker // The POSIX definition of struct sigaction and the sigaction data structure
24*71db0c75SAndroid Build Coastguard Worker // expected by the rt_sigaction syscall differ in their definition. So, we
25*71db0c75SAndroid Build Coastguard Worker // define the equivalent of the what the kernel expects to help with making
26*71db0c75SAndroid Build Coastguard Worker // the rt_sigaction syscall.
27*71db0c75SAndroid Build Coastguard Worker //
28*71db0c75SAndroid Build Coastguard Worker // NOTE: Though the kernel definition does not have a union to include the
29*71db0c75SAndroid Build Coastguard Worker // handler taking siginfo_t * argument, one can set sa_handler to sa_sigaction
30*71db0c75SAndroid Build Coastguard Worker // if SA_SIGINFO is set in sa_flags.
31*71db0c75SAndroid Build Coastguard Worker struct KernelSigaction {
32*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE KernelSigaction &operator=(const struct sigaction &sa) {
33*71db0c75SAndroid Build Coastguard Worker sa_flags = sa.sa_flags;
34*71db0c75SAndroid Build Coastguard Worker sa_restorer = sa.sa_restorer;
35*71db0c75SAndroid Build Coastguard Worker sa_mask = sa.sa_mask;
36*71db0c75SAndroid Build Coastguard Worker if (sa_flags & SA_SIGINFO) {
37*71db0c75SAndroid Build Coastguard Worker sa_sigaction = sa.sa_sigaction;
38*71db0c75SAndroid Build Coastguard Worker } else {
39*71db0c75SAndroid Build Coastguard Worker sa_handler = sa.sa_handler;
40*71db0c75SAndroid Build Coastguard Worker }
41*71db0c75SAndroid Build Coastguard Worker return *this;
42*71db0c75SAndroid Build Coastguard Worker }
43*71db0c75SAndroid Build Coastguard Worker
sigactionKernelSigaction44*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE operator struct sigaction() const {
45*71db0c75SAndroid Build Coastguard Worker struct sigaction sa;
46*71db0c75SAndroid Build Coastguard Worker sa.sa_flags = static_cast<int>(sa_flags);
47*71db0c75SAndroid Build Coastguard Worker sa.sa_mask = sa_mask;
48*71db0c75SAndroid Build Coastguard Worker sa.sa_restorer = sa_restorer;
49*71db0c75SAndroid Build Coastguard Worker if (sa_flags & SA_SIGINFO)
50*71db0c75SAndroid Build Coastguard Worker sa.sa_sigaction = sa_sigaction;
51*71db0c75SAndroid Build Coastguard Worker else
52*71db0c75SAndroid Build Coastguard Worker sa.sa_handler = sa_handler;
53*71db0c75SAndroid Build Coastguard Worker return sa;
54*71db0c75SAndroid Build Coastguard Worker }
55*71db0c75SAndroid Build Coastguard Worker
56*71db0c75SAndroid Build Coastguard Worker union {
57*71db0c75SAndroid Build Coastguard Worker void (*sa_handler)(int);
58*71db0c75SAndroid Build Coastguard Worker void (*sa_sigaction)(int, siginfo_t *, void *);
59*71db0c75SAndroid Build Coastguard Worker };
60*71db0c75SAndroid Build Coastguard Worker unsigned long sa_flags;
61*71db0c75SAndroid Build Coastguard Worker void (*sa_restorer)(void);
62*71db0c75SAndroid Build Coastguard Worker // Our public definition of sigset_t matches that of the kernel's definition.
63*71db0c75SAndroid Build Coastguard Worker // So, we can use the public sigset_t type here.
64*71db0c75SAndroid Build Coastguard Worker sigset_t sa_mask;
65*71db0c75SAndroid Build Coastguard Worker };
66*71db0c75SAndroid Build Coastguard Worker
67*71db0c75SAndroid Build Coastguard Worker static constexpr size_t BITS_PER_SIGWORD = sizeof(unsigned long) * 8;
68*71db0c75SAndroid Build Coastguard Worker
full_set()69*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE constexpr sigset_t full_set() { return sigset_t{{-1UL}}; }
70*71db0c75SAndroid Build Coastguard Worker
empty_set()71*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE constexpr sigset_t empty_set() { return sigset_t{{0}}; }
72*71db0c75SAndroid Build Coastguard Worker
73*71db0c75SAndroid Build Coastguard Worker // Set the bit corresponding to |signal| in |set|. Return true on success
74*71db0c75SAndroid Build Coastguard Worker // and false on failure. The function will fail if |signal| is greater than
75*71db0c75SAndroid Build Coastguard Worker // NSIG or negative.
add_signal(sigset_t & set,int signal)76*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE constexpr bool add_signal(sigset_t &set, int signal) {
77*71db0c75SAndroid Build Coastguard Worker if (signal > NSIG || signal <= 0)
78*71db0c75SAndroid Build Coastguard Worker return false;
79*71db0c75SAndroid Build Coastguard Worker size_t n = size_t(signal) - 1;
80*71db0c75SAndroid Build Coastguard Worker size_t word = n / BITS_PER_SIGWORD;
81*71db0c75SAndroid Build Coastguard Worker size_t bit = n % BITS_PER_SIGWORD;
82*71db0c75SAndroid Build Coastguard Worker set.__signals[word] |= (1UL << bit);
83*71db0c75SAndroid Build Coastguard Worker return true;
84*71db0c75SAndroid Build Coastguard Worker }
85*71db0c75SAndroid Build Coastguard Worker
86*71db0c75SAndroid Build Coastguard Worker // Reset the bit corresponding to |signal| in |set|. Return true on success
87*71db0c75SAndroid Build Coastguard Worker // and false on failure. The function will fail if |signal| is greater than
88*71db0c75SAndroid Build Coastguard Worker // NSIG or negative.
delete_signal(sigset_t & set,int signal)89*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE constexpr bool delete_signal(sigset_t &set, int signal) {
90*71db0c75SAndroid Build Coastguard Worker if (signal > NSIG || signal <= 0)
91*71db0c75SAndroid Build Coastguard Worker return false;
92*71db0c75SAndroid Build Coastguard Worker size_t n = size_t(signal) - 1;
93*71db0c75SAndroid Build Coastguard Worker size_t word = n / BITS_PER_SIGWORD;
94*71db0c75SAndroid Build Coastguard Worker size_t bit = n % BITS_PER_SIGWORD;
95*71db0c75SAndroid Build Coastguard Worker set.__signals[word] &= ~(1UL << bit);
96*71db0c75SAndroid Build Coastguard Worker return true;
97*71db0c75SAndroid Build Coastguard Worker }
98*71db0c75SAndroid Build Coastguard Worker
block_all_signals(sigset_t & set)99*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE int block_all_signals(sigset_t &set) {
100*71db0c75SAndroid Build Coastguard Worker sigset_t full = full_set();
101*71db0c75SAndroid Build Coastguard Worker return LIBC_NAMESPACE::syscall_impl<int>(SYS_rt_sigprocmask, SIG_BLOCK, &full,
102*71db0c75SAndroid Build Coastguard Worker &set, sizeof(sigset_t));
103*71db0c75SAndroid Build Coastguard Worker }
104*71db0c75SAndroid Build Coastguard Worker
restore_signals(const sigset_t & set)105*71db0c75SAndroid Build Coastguard Worker LIBC_INLINE int restore_signals(const sigset_t &set) {
106*71db0c75SAndroid Build Coastguard Worker return LIBC_NAMESPACE::syscall_impl<int>(SYS_rt_sigprocmask, SIG_SETMASK,
107*71db0c75SAndroid Build Coastguard Worker &set, nullptr, sizeof(sigset_t));
108*71db0c75SAndroid Build Coastguard Worker }
109*71db0c75SAndroid Build Coastguard Worker
110*71db0c75SAndroid Build Coastguard Worker } // namespace LIBC_NAMESPACE_DECL
111*71db0c75SAndroid Build Coastguard Worker
112*71db0c75SAndroid Build Coastguard Worker #endif // LLVM_LIBC_SRC_SIGNAL_LINUX_SIGNAL_UTILS_H
113