]> git.saurik.com Git - apple/xnu.git/blobdiff - libsyscall/os/tsd.h
xnu-4903.221.2.tar.gz
[apple/xnu.git] / libsyscall / os / tsd.h
index 0e064b954ca519cfb456e6b3732b722c22b07b7d..474c97aec2976bba7491dc0bd2e3c457e3acac1e 100644 (file)
@@ -29,8 +29,6 @@
 #ifndef OS_TSD_H
 #define OS_TSD_H
 
-#include <stdint.h>
-
 /* The low nine slots of the TSD are reserved for libsyscall usage. */
 #define __TSD_RESERVED_BASE 0
 #define __TSD_RESERVED_MAX 9
 #define __TSD_THREAD_SELF 0
 #define __TSD_ERRNO 1
 #define __TSD_MIG_REPLY 2
+#define __TSD_MACH_THREAD_SELF 3
+#define __TSD_THREAD_QOS_CLASS 4
+#define __TSD_RETURN_TO_KERNEL 5
+/* slot 6 is reserved for Windows/WINE compatibility reasons */
+#define __TSD_PTR_MUNGE 7
+#define __TSD_MACH_SPECIAL_REPLY 8
 #define __TSD_SEMAPHORE_CACHE 9
 
+#ifndef __ASSEMBLER__
+
+#include <stdint.h>
+
+#ifdef __arm__
+#include <arm/arch.h>
+#endif
+
+extern void _thread_set_tsd_base(void *tsd_base);
 
 __attribute__((always_inline))
 static __inline__ unsigned int
 _os_cpu_number(void)
 {
-       /* Not yet implemented */
-       return 0;
+#if defined(__arm__) && defined(_ARM_ARCH_6)
+       uintptr_t p;
+       __asm__("mrc    p15, 0, %[p], c13, c0, 3" : [p] "=&r" (p));
+       return (unsigned int)(p & 0x3ul);
+#elif defined(__arm64__)
+       uint64_t p;
+       __asm__("mrs    %[p], TPIDRRO_EL0" : [p] "=&r" (p));
+       return (unsigned int)p & 0x7;
+#elif defined(__x86_64__) || defined(__i386__)
+       struct { uintptr_t p1, p2; } p;
+       __asm__("sidt %[p]" : [p] "=&m" (p));
+       return (unsigned int)(p.p1 & 0xfff);
+#else
+#error _os_cpu_number not implemented on this architecture
+#endif
 }
 
 #if defined(__i386__) || defined(__x86_64__)
@@ -84,6 +110,28 @@ _os_tsd_set_direct(unsigned long slot, void *val)
 }
 #endif
 
+#elif defined(__arm__) || defined(__arm64__)
+
+__attribute__((always_inline, pure))
+static __inline__ void**
+_os_tsd_get_base(void)
+{
+#if defined(__arm__) && defined(_ARM_ARCH_6)
+       uintptr_t tsd;
+       __asm__("mrc p15, 0, %0, c13, c0, 3" : "=r" (tsd));
+       tsd &= ~0x3ul; /* lower 2-bits contain CPU number */
+#elif defined(__arm__) && defined(_ARM_ARCH_5)
+       register uintptr_t tsd asm ("r9");
+#elif defined(__arm64__)
+       uint64_t tsd;
+       __asm__("mrs %0, TPIDRRO_EL0" : "=r" (tsd));
+       tsd &= ~0x7ull;
+#endif
+
+       return (void**)(uintptr_t)tsd;
+}
+#define _os_tsd_get_base()  _os_tsd_get_base()
+
 #else
 #error _os_tsd_get_base not implemented on this architecture
 #endif
@@ -105,6 +153,61 @@ _os_tsd_set_direct(unsigned long slot, void *val)
 }
 #endif
 
-extern void _thread_set_tsd_base(void *tsd_base);
+__attribute__((always_inline, pure))
+static __inline__ uintptr_t
+_os_ptr_munge_token(void)
+{
+       return (uintptr_t)_os_tsd_get_direct(__TSD_PTR_MUNGE);
+}
 
-#endif
+__attribute__((always_inline, pure))
+static __inline__ uintptr_t
+_os_ptr_munge(uintptr_t ptr)
+{
+       return ptr ^ _os_ptr_munge_token();
+}
+#define _OS_PTR_MUNGE(_ptr) _os_ptr_munge((uintptr_t)(_ptr))
+#define _OS_PTR_UNMUNGE(_ptr) _os_ptr_munge((uintptr_t)(_ptr))
+
+#else // __ASSEMBLER__
+
+#define _OS_TSD_OFFSET(_key) \
+       ((__POINTER_WIDTH__/__CHAR_BIT__)*_key)
+
+#if defined(__i386__) || defined(__x86_64__)
+
+#define _OS_PTR_MUNGE(_reg) \
+       xor %gs:_OS_TSD_OFFSET(__TSD_PTR_MUNGE), _reg
+
+#define _OS_PTR_UNMUNGE(_reg) \
+       _OS_PTR_MUNGE(_reg)
+
+#elif defined(__arm__) || defined(__arm64__)
+
+#if defined(__arm__)
+
+#define _OS_PTR_MUNGE_TOKEN(_reg, _token) \
+       mrc p15, 0, _reg, c13, c0, 3; \
+       bic     _reg, _reg, #3; \
+       ldr     _token, [ _reg,  #_OS_TSD_OFFSET(__TSD_PTR_MUNGE) ]
+
+#elif defined(__arm64__)
+
+#define _OS_PTR_MUNGE_TOKEN(_reg, _token) \
+       mrs _reg, TPIDRRO_EL0 %% \
+       and     _reg, _reg, #~0x7 %% \
+       ldr     _token, [ _reg,  #_OS_TSD_OFFSET(__TSD_PTR_MUNGE) ]
+
+#endif // defined(__arm64__)
+
+#define _OS_PTR_MUNGE(_regdest, _regsrc, _token) \
+       eor _regdest, _regsrc, _token
+
+#define _OS_PTR_UNMUNGE(_regdest, _regsrc, _token) \
+       _OS_PTR_MUNGE(_regdest, _regsrc, _token)
+
+#endif // defined(__arm__) || defined(__arm64__)
+
+#endif // __ASSEMBLER__
+
+#endif // OS_TSD_H