xref: /openbmc/linux/tools/testing/selftests/rseq/rseq-abi.h (revision 4f2c0a4acffbec01079c28f839422e64ddeff004)
1*5c105d55SMathieu Desnoyers /* SPDX-License-Identifier: GPL-2.0+ WITH Linux-syscall-note */
2*5c105d55SMathieu Desnoyers #ifndef _RSEQ_ABI_H
3*5c105d55SMathieu Desnoyers #define _RSEQ_ABI_H
4*5c105d55SMathieu Desnoyers 
5*5c105d55SMathieu Desnoyers /*
6*5c105d55SMathieu Desnoyers  * rseq-abi.h
7*5c105d55SMathieu Desnoyers  *
8*5c105d55SMathieu Desnoyers  * Restartable sequences system call API
9*5c105d55SMathieu Desnoyers  *
10*5c105d55SMathieu Desnoyers  * Copyright (c) 2015-2022 Mathieu Desnoyers <mathieu.desnoyers@efficios.com>
11*5c105d55SMathieu Desnoyers  */
12*5c105d55SMathieu Desnoyers 
13*5c105d55SMathieu Desnoyers #include <linux/types.h>
14*5c105d55SMathieu Desnoyers #include <asm/byteorder.h>
15*5c105d55SMathieu Desnoyers 
16*5c105d55SMathieu Desnoyers enum rseq_abi_cpu_id_state {
17*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CPU_ID_UNINITIALIZED			= -1,
18*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CPU_ID_REGISTRATION_FAILED		= -2,
19*5c105d55SMathieu Desnoyers };
20*5c105d55SMathieu Desnoyers 
21*5c105d55SMathieu Desnoyers enum rseq_abi_flags {
22*5c105d55SMathieu Desnoyers 	RSEQ_ABI_FLAG_UNREGISTER = (1 << 0),
23*5c105d55SMathieu Desnoyers };
24*5c105d55SMathieu Desnoyers 
25*5c105d55SMathieu Desnoyers enum rseq_abi_cs_flags_bit {
26*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_PREEMPT_BIT	= 0,
27*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_SIGNAL_BIT	= 1,
28*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_MIGRATE_BIT	= 2,
29*5c105d55SMathieu Desnoyers };
30*5c105d55SMathieu Desnoyers 
31*5c105d55SMathieu Desnoyers enum rseq_abi_cs_flags {
32*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_PREEMPT	=
33*5c105d55SMathieu Desnoyers 		(1U << RSEQ_ABI_CS_FLAG_NO_RESTART_ON_PREEMPT_BIT),
34*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_SIGNAL	=
35*5c105d55SMathieu Desnoyers 		(1U << RSEQ_ABI_CS_FLAG_NO_RESTART_ON_SIGNAL_BIT),
36*5c105d55SMathieu Desnoyers 	RSEQ_ABI_CS_FLAG_NO_RESTART_ON_MIGRATE	=
37*5c105d55SMathieu Desnoyers 		(1U << RSEQ_ABI_CS_FLAG_NO_RESTART_ON_MIGRATE_BIT),
38*5c105d55SMathieu Desnoyers };
39*5c105d55SMathieu Desnoyers 
40*5c105d55SMathieu Desnoyers /*
41*5c105d55SMathieu Desnoyers  * struct rseq_abi_cs is aligned on 4 * 8 bytes to ensure it is always
42*5c105d55SMathieu Desnoyers  * contained within a single cache-line. It is usually declared as
43*5c105d55SMathieu Desnoyers  * link-time constant data.
44*5c105d55SMathieu Desnoyers  */
45*5c105d55SMathieu Desnoyers struct rseq_abi_cs {
46*5c105d55SMathieu Desnoyers 	/* Version of this structure. */
47*5c105d55SMathieu Desnoyers 	__u32 version;
48*5c105d55SMathieu Desnoyers 	/* enum rseq_abi_cs_flags */
49*5c105d55SMathieu Desnoyers 	__u32 flags;
50*5c105d55SMathieu Desnoyers 	__u64 start_ip;
51*5c105d55SMathieu Desnoyers 	/* Offset from start_ip. */
52*5c105d55SMathieu Desnoyers 	__u64 post_commit_offset;
53*5c105d55SMathieu Desnoyers 	__u64 abort_ip;
54*5c105d55SMathieu Desnoyers } __attribute__((aligned(4 * sizeof(__u64))));
55*5c105d55SMathieu Desnoyers 
56*5c105d55SMathieu Desnoyers /*
57*5c105d55SMathieu Desnoyers  * struct rseq_abi is aligned on 4 * 8 bytes to ensure it is always
58*5c105d55SMathieu Desnoyers  * contained within a single cache-line.
59*5c105d55SMathieu Desnoyers  *
60*5c105d55SMathieu Desnoyers  * A single struct rseq_abi per thread is allowed.
61*5c105d55SMathieu Desnoyers  */
62*5c105d55SMathieu Desnoyers struct rseq_abi {
63*5c105d55SMathieu Desnoyers 	/*
64*5c105d55SMathieu Desnoyers 	 * Restartable sequences cpu_id_start field. Updated by the
65*5c105d55SMathieu Desnoyers 	 * kernel. Read by user-space with single-copy atomicity
66*5c105d55SMathieu Desnoyers 	 * semantics. This field should only be read by the thread which
67*5c105d55SMathieu Desnoyers 	 * registered this data structure. Aligned on 32-bit. Always
68*5c105d55SMathieu Desnoyers 	 * contains a value in the range of possible CPUs, although the
69*5c105d55SMathieu Desnoyers 	 * value may not be the actual current CPU (e.g. if rseq is not
70*5c105d55SMathieu Desnoyers 	 * initialized). This CPU number value should always be compared
71*5c105d55SMathieu Desnoyers 	 * against the value of the cpu_id field before performing a rseq
72*5c105d55SMathieu Desnoyers 	 * commit or returning a value read from a data structure indexed
73*5c105d55SMathieu Desnoyers 	 * using the cpu_id_start value.
74*5c105d55SMathieu Desnoyers 	 */
75*5c105d55SMathieu Desnoyers 	__u32 cpu_id_start;
76*5c105d55SMathieu Desnoyers 	/*
77*5c105d55SMathieu Desnoyers 	 * Restartable sequences cpu_id field. Updated by the kernel.
78*5c105d55SMathieu Desnoyers 	 * Read by user-space with single-copy atomicity semantics. This
79*5c105d55SMathieu Desnoyers 	 * field should only be read by the thread which registered this
80*5c105d55SMathieu Desnoyers 	 * data structure. Aligned on 32-bit. Values
81*5c105d55SMathieu Desnoyers 	 * RSEQ_CPU_ID_UNINITIALIZED and RSEQ_CPU_ID_REGISTRATION_FAILED
82*5c105d55SMathieu Desnoyers 	 * have a special semantic: the former means "rseq uninitialized",
83*5c105d55SMathieu Desnoyers 	 * and latter means "rseq initialization failed". This value is
84*5c105d55SMathieu Desnoyers 	 * meant to be read within rseq critical sections and compared
85*5c105d55SMathieu Desnoyers 	 * with the cpu_id_start value previously read, before performing
86*5c105d55SMathieu Desnoyers 	 * the commit instruction, or read and compared with the
87*5c105d55SMathieu Desnoyers 	 * cpu_id_start value before returning a value loaded from a data
88*5c105d55SMathieu Desnoyers 	 * structure indexed using the cpu_id_start value.
89*5c105d55SMathieu Desnoyers 	 */
90*5c105d55SMathieu Desnoyers 	__u32 cpu_id;
91*5c105d55SMathieu Desnoyers 	/*
92*5c105d55SMathieu Desnoyers 	 * Restartable sequences rseq_cs field.
93*5c105d55SMathieu Desnoyers 	 *
94*5c105d55SMathieu Desnoyers 	 * Contains NULL when no critical section is active for the current
95*5c105d55SMathieu Desnoyers 	 * thread, or holds a pointer to the currently active struct rseq_cs.
96*5c105d55SMathieu Desnoyers 	 *
97*5c105d55SMathieu Desnoyers 	 * Updated by user-space, which sets the address of the currently
98*5c105d55SMathieu Desnoyers 	 * active rseq_cs at the beginning of assembly instruction sequence
99*5c105d55SMathieu Desnoyers 	 * block, and set to NULL by the kernel when it restarts an assembly
100*5c105d55SMathieu Desnoyers 	 * instruction sequence block, as well as when the kernel detects that
101*5c105d55SMathieu Desnoyers 	 * it is preempting or delivering a signal outside of the range
102*5c105d55SMathieu Desnoyers 	 * targeted by the rseq_cs. Also needs to be set to NULL by user-space
103*5c105d55SMathieu Desnoyers 	 * before reclaiming memory that contains the targeted struct rseq_cs.
104*5c105d55SMathieu Desnoyers 	 *
105*5c105d55SMathieu Desnoyers 	 * Read and set by the kernel. Set by user-space with single-copy
106*5c105d55SMathieu Desnoyers 	 * atomicity semantics. This field should only be updated by the
107*5c105d55SMathieu Desnoyers 	 * thread which registered this data structure. Aligned on 64-bit.
108*5c105d55SMathieu Desnoyers 	 */
109*5c105d55SMathieu Desnoyers 	union {
110*5c105d55SMathieu Desnoyers 		__u64 ptr64;
111*5c105d55SMathieu Desnoyers 
112*5c105d55SMathieu Desnoyers 		/*
113*5c105d55SMathieu Desnoyers 		 * The "arch" field provides architecture accessor for
114*5c105d55SMathieu Desnoyers 		 * the ptr field based on architecture pointer size and
115*5c105d55SMathieu Desnoyers 		 * endianness.
116*5c105d55SMathieu Desnoyers 		 */
117*5c105d55SMathieu Desnoyers 		struct {
118*5c105d55SMathieu Desnoyers #ifdef __LP64__
119*5c105d55SMathieu Desnoyers 			__u64 ptr;
120*5c105d55SMathieu Desnoyers #elif defined(__BYTE_ORDER) ? (__BYTE_ORDER == __BIG_ENDIAN) : defined(__BIG_ENDIAN)
121*5c105d55SMathieu Desnoyers 			__u32 padding;		/* Initialized to zero. */
122*5c105d55SMathieu Desnoyers 			__u32 ptr;
123*5c105d55SMathieu Desnoyers #else
124*5c105d55SMathieu Desnoyers 			__u32 ptr;
125*5c105d55SMathieu Desnoyers 			__u32 padding;		/* Initialized to zero. */
126*5c105d55SMathieu Desnoyers #endif
127*5c105d55SMathieu Desnoyers 		} arch;
128*5c105d55SMathieu Desnoyers 	} rseq_cs;
129*5c105d55SMathieu Desnoyers 
130*5c105d55SMathieu Desnoyers 	/*
131*5c105d55SMathieu Desnoyers 	 * Restartable sequences flags field.
132*5c105d55SMathieu Desnoyers 	 *
133*5c105d55SMathieu Desnoyers 	 * This field should only be updated by the thread which
134*5c105d55SMathieu Desnoyers 	 * registered this data structure. Read by the kernel.
135*5c105d55SMathieu Desnoyers 	 * Mainly used for single-stepping through rseq critical sections
136*5c105d55SMathieu Desnoyers 	 * with debuggers.
137*5c105d55SMathieu Desnoyers 	 *
138*5c105d55SMathieu Desnoyers 	 * - RSEQ_ABI_CS_FLAG_NO_RESTART_ON_PREEMPT
139*5c105d55SMathieu Desnoyers 	 *     Inhibit instruction sequence block restart on preemption
140*5c105d55SMathieu Desnoyers 	 *     for this thread.
141*5c105d55SMathieu Desnoyers 	 * - RSEQ_ABI_CS_FLAG_NO_RESTART_ON_SIGNAL
142*5c105d55SMathieu Desnoyers 	 *     Inhibit instruction sequence block restart on signal
143*5c105d55SMathieu Desnoyers 	 *     delivery for this thread.
144*5c105d55SMathieu Desnoyers 	 * - RSEQ_ABI_CS_FLAG_NO_RESTART_ON_MIGRATE
145*5c105d55SMathieu Desnoyers 	 *     Inhibit instruction sequence block restart on migration for
146*5c105d55SMathieu Desnoyers 	 *     this thread.
147*5c105d55SMathieu Desnoyers 	 */
148*5c105d55SMathieu Desnoyers 	__u32 flags;
149*5c105d55SMathieu Desnoyers 
150*5c105d55SMathieu Desnoyers 	/*
151*5c105d55SMathieu Desnoyers 	 * Restartable sequences node_id field. Updated by the kernel. Read by
152 	 * user-space with single-copy atomicity semantics. This field should
153 	 * only be read by the thread which registered this data structure.
154 	 * Aligned on 32-bit. Contains the current NUMA node ID.
155 	 */
156 	__u32 node_id;
157 
158 	/*
159 	 * Restartable sequences mm_cid field. Updated by the kernel. Read by
160 	 * user-space with single-copy atomicity semantics. This field should
161 	 * only be read by the thread which registered this data structure.
162 	 * Aligned on 32-bit. Contains the current thread's concurrency ID
163 	 * (allocated uniquely within a memory map).
164 	 */
165 	__u32 mm_cid;
166 
167 	/*
168 	 * Flexible array member at end of structure, after last feature field.
169 	 */
170 	char end[];
171 } __attribute__((aligned(4 * sizeof(__u64))));
172 
173 #endif /* _RSEQ_ABI_H */
174