xref: /xnu-8020.121.3/libsyscall/os/tsd.h (revision fdd8201d7b966f0c3ea610489d29bd841d358941)
1*fdd8201dSApple OSS Distributions /*
2*fdd8201dSApple OSS Distributions  * Copyright (c) 2012 Apple Inc. All rights reserved.
3*fdd8201dSApple OSS Distributions  *
4*fdd8201dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_START@
5*fdd8201dSApple OSS Distributions  *
6*fdd8201dSApple OSS Distributions  * This file contains Original Code and/or Modifications of Original Code
7*fdd8201dSApple OSS Distributions  * as defined in and that are subject to the Apple Public Source License
8*fdd8201dSApple OSS Distributions  * Version 2.0 (the 'License'). You may not use this file except in
9*fdd8201dSApple OSS Distributions  * compliance with the License. The rights granted to you under the License
10*fdd8201dSApple OSS Distributions  * may not be used to create, or enable the creation or redistribution of,
11*fdd8201dSApple OSS Distributions  * unlawful or unlicensed copies of an Apple operating system, or to
12*fdd8201dSApple OSS Distributions  * circumvent, violate, or enable the circumvention or violation of, any
13*fdd8201dSApple OSS Distributions  * terms of an Apple operating system software license agreement.
14*fdd8201dSApple OSS Distributions  *
15*fdd8201dSApple OSS Distributions  * Please obtain a copy of the License at
16*fdd8201dSApple OSS Distributions  * http://www.opensource.apple.com/apsl/ and read it before using this file.
17*fdd8201dSApple OSS Distributions  *
18*fdd8201dSApple OSS Distributions  * The Original Code and all software distributed under the License are
19*fdd8201dSApple OSS Distributions  * distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
20*fdd8201dSApple OSS Distributions  * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
21*fdd8201dSApple OSS Distributions  * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
22*fdd8201dSApple OSS Distributions  * FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
23*fdd8201dSApple OSS Distributions  * Please see the License for the specific language governing rights and
24*fdd8201dSApple OSS Distributions  * limitations under the License.
25*fdd8201dSApple OSS Distributions  *
26*fdd8201dSApple OSS Distributions  * @APPLE_OSREFERENCE_LICENSE_HEADER_END@
27*fdd8201dSApple OSS Distributions  */
28*fdd8201dSApple OSS Distributions 
29*fdd8201dSApple OSS Distributions #ifndef OS_TSD_H
30*fdd8201dSApple OSS Distributions #define OS_TSD_H
31*fdd8201dSApple OSS Distributions 
32*fdd8201dSApple OSS Distributions /* The low nine slots of the TSD are reserved for libsyscall usage. */
33*fdd8201dSApple OSS Distributions #define __TSD_RESERVED_BASE 0
34*fdd8201dSApple OSS Distributions #define __TSD_RESERVED_MAX 9
35*fdd8201dSApple OSS Distributions 
36*fdd8201dSApple OSS Distributions #define __TSD_THREAD_SELF 0
37*fdd8201dSApple OSS Distributions #define __TSD_ERRNO 1
38*fdd8201dSApple OSS Distributions #define __TSD_MIG_REPLY 2
39*fdd8201dSApple OSS Distributions #define __TSD_MACH_THREAD_SELF 3
40*fdd8201dSApple OSS Distributions #define __TSD_THREAD_QOS_CLASS 4
41*fdd8201dSApple OSS Distributions #define __TSD_RETURN_TO_KERNEL 5
42*fdd8201dSApple OSS Distributions /* slot 6 is reserved for Windows/WINE compatibility reasons */
43*fdd8201dSApple OSS Distributions #define __TSD_PTR_MUNGE 7
44*fdd8201dSApple OSS Distributions #define __TSD_MACH_SPECIAL_REPLY 8
45*fdd8201dSApple OSS Distributions #define __TSD_SEMAPHORE_CACHE 9
46*fdd8201dSApple OSS Distributions 
47*fdd8201dSApple OSS Distributions 
48*fdd8201dSApple OSS Distributions #define __TPIDR_CPU_NUM_MASK 0x0000000000000fff
49*fdd8201dSApple OSS Distributions #define __TPIDR_CPU_NUM_SHIFT 0
50*fdd8201dSApple OSS Distributions 
51*fdd8201dSApple OSS Distributions #ifndef __ASSEMBLER__
52*fdd8201dSApple OSS Distributions 
53*fdd8201dSApple OSS Distributions #include <stdint.h>
54*fdd8201dSApple OSS Distributions #include <TargetConditionals.h>
55*fdd8201dSApple OSS Distributions 
56*fdd8201dSApple OSS Distributions #ifdef __arm__
57*fdd8201dSApple OSS Distributions #include <arm/arch.h>
58*fdd8201dSApple OSS Distributions #endif
59*fdd8201dSApple OSS Distributions 
60*fdd8201dSApple OSS Distributions extern void _thread_set_tsd_base(void *tsd_base);
61*fdd8201dSApple OSS Distributions 
62*fdd8201dSApple OSS Distributions /*
63*fdd8201dSApple OSS Distributions  * The implementation details of this function are not ABI and are subject to change,
64*fdd8201dSApple OSS Distributions  * do not copy them in another project
65*fdd8201dSApple OSS Distributions  */
66*fdd8201dSApple OSS Distributions __attribute__((always_inline))
67*fdd8201dSApple OSS Distributions static __inline__ unsigned int
_os_cpu_number(void)68*fdd8201dSApple OSS Distributions _os_cpu_number(void)
69*fdd8201dSApple OSS Distributions {
70*fdd8201dSApple OSS Distributions #if defined(__arm__)
71*fdd8201dSApple OSS Distributions 	uintptr_t p;
72*fdd8201dSApple OSS Distributions 	__asm__ __volatile__ ("mrc	p15, 0, %[p], c13, c0, 3" : [p] "=&r" (p));
73*fdd8201dSApple OSS Distributions 	return (unsigned int)(p & 0x3ul);
74*fdd8201dSApple OSS Distributions #elif defined(__arm64__)
75*fdd8201dSApple OSS Distributions 	uint64_t p;
76*fdd8201dSApple OSS Distributions 	__asm__ __volatile__ ("mrs %0, TPIDR_EL0" : "=r" (p));
77*fdd8201dSApple OSS Distributions 	return (p & __TPIDR_CPU_NUM_MASK) >> __TPIDR_CPU_NUM_SHIFT;
78*fdd8201dSApple OSS Distributions #elif defined(__x86_64__) || defined(__i386__)
79*fdd8201dSApple OSS Distributions 	struct { uintptr_t p1, p2; } p;
80*fdd8201dSApple OSS Distributions 	__asm__ __volatile__ ("sidt %[p]" : [p] "=&m" (p));
81*fdd8201dSApple OSS Distributions 	return (unsigned int)(p.p1 & 0xfff);
82*fdd8201dSApple OSS Distributions #else
83*fdd8201dSApple OSS Distributions #error _os_cpu_number not implemented on this architecture
84*fdd8201dSApple OSS Distributions #endif
85*fdd8201dSApple OSS Distributions }
86*fdd8201dSApple OSS Distributions 
87*fdd8201dSApple OSS Distributions #if defined(__i386__) || defined(__x86_64__)
88*fdd8201dSApple OSS Distributions 
89*fdd8201dSApple OSS Distributions #if defined(__has_attribute)
90*fdd8201dSApple OSS Distributions #if __has_attribute(address_space)
91*fdd8201dSApple OSS Distributions #define OS_GS_RELATIVE  __attribute__((address_space(256)))
92*fdd8201dSApple OSS Distributions #endif
93*fdd8201dSApple OSS Distributions #endif
94*fdd8201dSApple OSS Distributions 
95*fdd8201dSApple OSS Distributions #ifdef OS_GS_RELATIVE
96*fdd8201dSApple OSS Distributions #define _os_tsd_get_base() ((void * OS_GS_RELATIVE *)0)
97*fdd8201dSApple OSS Distributions #else
98*fdd8201dSApple OSS Distributions __attribute__((always_inline))
99*fdd8201dSApple OSS Distributions static __inline__ void*
_os_tsd_get_direct(unsigned long slot)100*fdd8201dSApple OSS Distributions _os_tsd_get_direct(unsigned long slot)
101*fdd8201dSApple OSS Distributions {
102*fdd8201dSApple OSS Distributions 	void *ret;
103*fdd8201dSApple OSS Distributions 	__asm__("mov %%gs:%1, %0" : "=r" (ret) : "m" (*(void **)(slot * sizeof(void *))));
104*fdd8201dSApple OSS Distributions 	return ret;
105*fdd8201dSApple OSS Distributions }
106*fdd8201dSApple OSS Distributions 
107*fdd8201dSApple OSS Distributions __attribute__((always_inline))
108*fdd8201dSApple OSS Distributions static __inline__ int
_os_tsd_set_direct(unsigned long slot,void * val)109*fdd8201dSApple OSS Distributions _os_tsd_set_direct(unsigned long slot, void *val)
110*fdd8201dSApple OSS Distributions {
111*fdd8201dSApple OSS Distributions #if defined(__i386__) && defined(__PIC__)
112*fdd8201dSApple OSS Distributions 	__asm__("movl %1, %%gs:%0" : "=m" (*(void **)(slot * sizeof(void *))) : "rn" (val));
113*fdd8201dSApple OSS Distributions #elif defined(__i386__) && !defined(__PIC__)
114*fdd8201dSApple OSS Distributions 	__asm__("movl %1, %%gs:%0" : "=m" (*(void **)(slot * sizeof(void *))) : "ri" (val));
115*fdd8201dSApple OSS Distributions #else
116*fdd8201dSApple OSS Distributions 	__asm__("movq %1, %%gs:%0" : "=m" (*(void **)(slot * sizeof(void *))) : "rn" (val));
117*fdd8201dSApple OSS Distributions #endif
118*fdd8201dSApple OSS Distributions 	return 0;
119*fdd8201dSApple OSS Distributions }
120*fdd8201dSApple OSS Distributions #endif
121*fdd8201dSApple OSS Distributions 
122*fdd8201dSApple OSS Distributions #elif defined(__arm__) || defined(__arm64__)
123*fdd8201dSApple OSS Distributions 
124*fdd8201dSApple OSS Distributions __attribute__((always_inline, const))
125*fdd8201dSApple OSS Distributions static __inline__ void**
_os_tsd_get_base(void)126*fdd8201dSApple OSS Distributions _os_tsd_get_base(void)
127*fdd8201dSApple OSS Distributions {
128*fdd8201dSApple OSS Distributions #if defined(__arm__)
129*fdd8201dSApple OSS Distributions 	uintptr_t tsd;
130*fdd8201dSApple OSS Distributions 	__asm__("mrc p15, 0, %0, c13, c0, 3\n"
131*fdd8201dSApple OSS Distributions                 "bic %0, %0, #0x3\n" : "=r" (tsd));
132*fdd8201dSApple OSS Distributions 	/* lower 2-bits contain CPU number */
133*fdd8201dSApple OSS Distributions #elif defined(__arm64__)
134*fdd8201dSApple OSS Distributions 	/*
135*fdd8201dSApple OSS Distributions 	 * <rdar://73762648> Do not use __builtin_arm_rsr64("TPIDRRO_EL0")
136*fdd8201dSApple OSS Distributions 	 * so that the "const" attribute takes effect and repeated use
137*fdd8201dSApple OSS Distributions 	 * is coalesced properly.
138*fdd8201dSApple OSS Distributions 	 */
139*fdd8201dSApple OSS Distributions 	uint64_t tsd;
140*fdd8201dSApple OSS Distributions 	__asm__ ("mrs %0, TPIDRRO_EL0" : "=r" (tsd));
141*fdd8201dSApple OSS Distributions #endif
142*fdd8201dSApple OSS Distributions 
143*fdd8201dSApple OSS Distributions 	return (void**)(uintptr_t)tsd;
144*fdd8201dSApple OSS Distributions }
145*fdd8201dSApple OSS Distributions #define _os_tsd_get_base()  _os_tsd_get_base()
146*fdd8201dSApple OSS Distributions 
147*fdd8201dSApple OSS Distributions #else
148*fdd8201dSApple OSS Distributions #error _os_tsd_get_base not implemented on this architecture
149*fdd8201dSApple OSS Distributions #endif
150*fdd8201dSApple OSS Distributions 
151*fdd8201dSApple OSS Distributions #ifdef _os_tsd_get_base
152*fdd8201dSApple OSS Distributions __attribute__((always_inline))
153*fdd8201dSApple OSS Distributions static __inline__ void*
_os_tsd_get_direct(unsigned long slot)154*fdd8201dSApple OSS Distributions _os_tsd_get_direct(unsigned long slot)
155*fdd8201dSApple OSS Distributions {
156*fdd8201dSApple OSS Distributions 	return _os_tsd_get_base()[slot];
157*fdd8201dSApple OSS Distributions }
158*fdd8201dSApple OSS Distributions 
159*fdd8201dSApple OSS Distributions __attribute__((always_inline))
160*fdd8201dSApple OSS Distributions static __inline__ int
_os_tsd_set_direct(unsigned long slot,void * val)161*fdd8201dSApple OSS Distributions _os_tsd_set_direct(unsigned long slot, void *val)
162*fdd8201dSApple OSS Distributions {
163*fdd8201dSApple OSS Distributions 	_os_tsd_get_base()[slot] = val;
164*fdd8201dSApple OSS Distributions 	return 0;
165*fdd8201dSApple OSS Distributions }
166*fdd8201dSApple OSS Distributions #endif
167*fdd8201dSApple OSS Distributions 
168*fdd8201dSApple OSS Distributions __attribute__((always_inline, const))
169*fdd8201dSApple OSS Distributions static __inline__ uintptr_t
_os_ptr_munge_token(void)170*fdd8201dSApple OSS Distributions _os_ptr_munge_token(void)
171*fdd8201dSApple OSS Distributions {
172*fdd8201dSApple OSS Distributions 	return (uintptr_t)_os_tsd_get_direct(__TSD_PTR_MUNGE);
173*fdd8201dSApple OSS Distributions }
174*fdd8201dSApple OSS Distributions 
175*fdd8201dSApple OSS Distributions __attribute__((always_inline, const))
176*fdd8201dSApple OSS Distributions static __inline__ uintptr_t
_os_ptr_munge(uintptr_t ptr)177*fdd8201dSApple OSS Distributions _os_ptr_munge(uintptr_t ptr)
178*fdd8201dSApple OSS Distributions {
179*fdd8201dSApple OSS Distributions 	return ptr ^ _os_ptr_munge_token();
180*fdd8201dSApple OSS Distributions }
181*fdd8201dSApple OSS Distributions #define _OS_PTR_MUNGE(_ptr) _os_ptr_munge((uintptr_t)(_ptr))
182*fdd8201dSApple OSS Distributions #define _OS_PTR_UNMUNGE(_ptr) _os_ptr_munge((uintptr_t)(_ptr))
183*fdd8201dSApple OSS Distributions 
184*fdd8201dSApple OSS Distributions #else // __ASSEMBLER__
185*fdd8201dSApple OSS Distributions 
186*fdd8201dSApple OSS Distributions #define _OS_TSD_OFFSET(_key) \
187*fdd8201dSApple OSS Distributions 	((__POINTER_WIDTH__/__CHAR_BIT__)*_key)
188*fdd8201dSApple OSS Distributions 
189*fdd8201dSApple OSS Distributions #if defined(__i386__) || defined(__x86_64__)
190*fdd8201dSApple OSS Distributions 
191*fdd8201dSApple OSS Distributions #define _OS_PTR_MUNGE(_reg) \
192*fdd8201dSApple OSS Distributions 	xor %gs:_OS_TSD_OFFSET(__TSD_PTR_MUNGE), _reg
193*fdd8201dSApple OSS Distributions 
194*fdd8201dSApple OSS Distributions #define _OS_PTR_UNMUNGE(_reg) \
195*fdd8201dSApple OSS Distributions 	_OS_PTR_MUNGE(_reg)
196*fdd8201dSApple OSS Distributions 
197*fdd8201dSApple OSS Distributions #elif defined(__arm__) || defined(__arm64__)
198*fdd8201dSApple OSS Distributions 
199*fdd8201dSApple OSS Distributions #if defined(__arm__)
200*fdd8201dSApple OSS Distributions 
201*fdd8201dSApple OSS Distributions #define _OS_PTR_MUNGE_TOKEN(_reg, _token) \
202*fdd8201dSApple OSS Distributions 	mrc p15, 0, _reg, c13, c0, 3; \
203*fdd8201dSApple OSS Distributions 	bic	_reg, _reg, #3; \
204*fdd8201dSApple OSS Distributions 	ldr	_token, [ _reg,  #_OS_TSD_OFFSET(__TSD_PTR_MUNGE) ]
205*fdd8201dSApple OSS Distributions 
206*fdd8201dSApple OSS Distributions #elif defined(__arm64__)
207*fdd8201dSApple OSS Distributions 
208*fdd8201dSApple OSS Distributions #define _OS_PTR_MUNGE_TOKEN(_reg, _token) \
209*fdd8201dSApple OSS Distributions 	mrs _reg, TPIDRRO_EL0 %% \
210*fdd8201dSApple OSS Distributions 	ldr	_token, [ _reg,  #_OS_TSD_OFFSET(__TSD_PTR_MUNGE) ]
211*fdd8201dSApple OSS Distributions 
212*fdd8201dSApple OSS Distributions #endif // defined(__arm64__)
213*fdd8201dSApple OSS Distributions 
214*fdd8201dSApple OSS Distributions #define _OS_PTR_MUNGE(_regdest, _regsrc, _token) \
215*fdd8201dSApple OSS Distributions 	eor _regdest, _regsrc, _token
216*fdd8201dSApple OSS Distributions 
217*fdd8201dSApple OSS Distributions #define _OS_PTR_UNMUNGE(_regdest, _regsrc, _token) \
218*fdd8201dSApple OSS Distributions 	_OS_PTR_MUNGE(_regdest, _regsrc, _token)
219*fdd8201dSApple OSS Distributions 
220*fdd8201dSApple OSS Distributions #endif // defined(__arm__) || defined(__arm64__)
221*fdd8201dSApple OSS Distributions 
222*fdd8201dSApple OSS Distributions #endif // __ASSEMBLER__
223*fdd8201dSApple OSS Distributions 
224*fdd8201dSApple OSS Distributions #endif // OS_TSD_H
225