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