1 /* Restartable Sequences exported symbols.  Linux header.
2    Copyright (C) 2021 Free Software Foundation, Inc.
3 
4    The GNU C Library is free software; you can redistribute it and/or
5    modify it under the terms of the GNU Lesser General Public
6    License as published by the Free Software Foundation; either
7    version 2.1 of the License, or (at your option) any later version.
8 
9    The GNU C Library is distributed in the hope that it will be useful,
10    but WITHOUT ANY WARRANTY; without even the implied warranty of
11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12    Lesser General Public License for more details.
13 
14    You should have received a copy of the GNU Lesser General Public
15    License along with the GNU C Library; if not, see
16    <https://www.gnu.org/licenses/>.  */
17 
18 #ifndef _SYS_RSEQ_H
19 #define _SYS_RSEQ_H	1
20 
21 /* Architecture-specific rseq signature.  */
22 #include <bits/rseq.h>
23 
24 #include <stdint.h>
25 #include <sys/cdefs.h>
26 #include <bits/endian.h>
27 
28 #ifdef __has_include
29 # if __has_include ("linux/rseq.h")
30 #  define __GLIBC_HAVE_KERNEL_RSEQ
31 # endif
32 #else
33 # include <linux/version.h>
34 # if LINUX_VERSION_CODE >= KERNEL_VERSION (4, 18, 0)
35 #  define __GLIBC_HAVE_KERNEL_RSEQ
36 # endif
37 #endif
38 
39 #ifdef __GLIBC_HAVE_KERNEL_RSEQ
40 /* We use the structures declarations from the kernel headers.  */
41 # include <linux/rseq.h>
42 #else /* __GLIBC_HAVE_KERNEL_RSEQ */
43 /* We use a copy of the include/uapi/linux/rseq.h kernel header.  */
44 
45 enum rseq_cpu_id_state
46   {
47     RSEQ_CPU_ID_UNINITIALIZED = -1,
48     RSEQ_CPU_ID_REGISTRATION_FAILED = -2,
49   };
50 
51 enum rseq_flags
52   {
53     RSEQ_FLAG_UNREGISTER = (1 << 0),
54   };
55 
56 enum rseq_cs_flags_bit
57   {
58     RSEQ_CS_FLAG_NO_RESTART_ON_PREEMPT_BIT = 0,
59     RSEQ_CS_FLAG_NO_RESTART_ON_SIGNAL_BIT = 1,
60     RSEQ_CS_FLAG_NO_RESTART_ON_MIGRATE_BIT = 2,
61   };
62 
63 enum rseq_cs_flags
64   {
65     RSEQ_CS_FLAG_NO_RESTART_ON_PREEMPT =
66       (1U << RSEQ_CS_FLAG_NO_RESTART_ON_PREEMPT_BIT),
67     RSEQ_CS_FLAG_NO_RESTART_ON_SIGNAL =
68       (1U << RSEQ_CS_FLAG_NO_RESTART_ON_SIGNAL_BIT),
69     RSEQ_CS_FLAG_NO_RESTART_ON_MIGRATE =
70       (1U << RSEQ_CS_FLAG_NO_RESTART_ON_MIGRATE_BIT),
71   };
72 
73 /* struct rseq_cs is aligned on 32 bytes to ensure it is always
74    contained within a single cache-line.  It is usually declared as
75    link-time constant data.  */
76 struct rseq_cs
77   {
78     /* Version of this structure.  */
79     uint32_t version;
80     /* enum rseq_cs_flags.  */
81     uint32_t flags;
82     uint64_t start_ip;
83     /* Offset from start_ip.  */
84     uint64_t post_commit_offset;
85     uint64_t abort_ip;
86   } __attribute__ ((__aligned__ (32)));
87 
88 /* struct rseq is aligned on 32 bytes to ensure it is always
89    contained within a single cache-line.
90 
91    A single struct rseq per thread is allowed.  */
92 struct rseq
93   {
94     /* Restartable sequences cpu_id_start field.  Updated by the
95        kernel.  Read by user-space with single-copy atomicity
96        semantics.  This field should only be read by the thread which
97        registered this data structure.  Aligned on 32-bit.  Always
98        contains a value in the range of possible CPUs, although the
99        value may not be the actual current CPU (e.g. if rseq is not
100        initialized).  This CPU number value should always be compared
101        against the value of the cpu_id field before performing a rseq
102        commit or returning a value read from a data structure indexed
103        using the cpu_id_start value.  */
104     uint32_t cpu_id_start;
105     /* Restartable sequences cpu_id field.  Updated by the kernel.
106        Read by user-space with single-copy atomicity semantics.  This
107        field should only be read by the thread which registered this
108        data structure.  Aligned on 32-bit.  Values
109        RSEQ_CPU_ID_UNINITIALIZED and RSEQ_CPU_ID_REGISTRATION_FAILED
110        have a special semantic: the former means "rseq uninitialized",
111        and latter means "rseq initialization failed".  This value is
112        meant to be read within rseq critical sections and compared
113        with the cpu_id_start value previously read, before performing
114        the commit instruction, or read and compared with the
115        cpu_id_start value before returning a value loaded from a data
116        structure indexed using the cpu_id_start value.  */
117     uint32_t cpu_id;
118     /* Restartable sequences rseq_cs field.
119 
120        Contains NULL when no critical section is active for the current
121        thread, or holds a pointer to the currently active struct rseq_cs.
122 
123        Updated by user-space, which sets the address of the currently
124        active rseq_cs at the beginning of assembly instruction sequence
125        block, and set to NULL by the kernel when it restarts an assembly
126        instruction sequence block, as well as when the kernel detects that
127        it is preempting or delivering a signal outside of the range
128        targeted by the rseq_cs.  Also needs to be set to NULL by user-space
129        before reclaiming memory that contains the targeted struct rseq_cs.
130 
131        Read and set by the kernel.  Set by user-space with single-copy
132        atomicity semantics.  This field should only be updated by the
133        thread which registered this data structure.  Aligned on 64-bit.  */
134     union
135       {
136         uint64_t ptr64;
137 # ifdef __LP64__
138         uint64_t ptr;
139 # else /* __LP64__ */
140         struct
141           {
142 #if __BYTE_ORDER == __BIG_ENDIAN
143             uint32_t padding; /* Initialized to zero.  */
144             uint32_t ptr32;
145 #  else /* LITTLE */
146             uint32_t ptr32;
147             uint32_t padding; /* Initialized to zero.  */
148 #  endif /* ENDIAN */
149           } ptr;
150 # endif /* __LP64__ */
151       } rseq_cs;
152 
153     /* Restartable sequences flags field.
154 
155        This field should only be updated by the thread which
156        registered this data structure.  Read by the kernel.
157        Mainly used for single-stepping through rseq critical sections
158        with debuggers.
159 
160        - RSEQ_CS_FLAG_NO_RESTART_ON_PREEMPT
161            Inhibit instruction sequence block restart on preemption
162            for this thread.
163        - RSEQ_CS_FLAG_NO_RESTART_ON_SIGNAL
164            Inhibit instruction sequence block restart on signal
165            delivery for this thread.
166        - RSEQ_CS_FLAG_NO_RESTART_ON_MIGRATE
167            Inhibit instruction sequence block restart on migration for
168            this thread.  */
169     uint32_t flags;
170   } __attribute__ ((__aligned__ (32)));
171 
172 #endif /* __GLIBC_HAVE_KERNEL_RSEQ */
173 
174 /* Offset from the thread pointer to the rseq area.  */
175 extern const int __rseq_offset;
176 
177 /* Size of the registered rseq area.  0 if the registration was
178    unsuccessful.  */
179 extern const unsigned int __rseq_size;
180 
181 /* Flags used during rseq registration.  */
182 extern const unsigned int __rseq_flags;
183 
184 #endif /* sys/rseq.h */
185