1e78f53d1SNikolas Klauser // -*- C++ -*- 2e78f53d1SNikolas Klauser //===----------------------------------------------------------------------===// 3e78f53d1SNikolas Klauser // 4e78f53d1SNikolas Klauser // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 5e78f53d1SNikolas Klauser // See https://llvm.org/LICENSE.txt for license information. 6e78f53d1SNikolas Klauser // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 7e78f53d1SNikolas Klauser // 8e78f53d1SNikolas Klauser //===----------------------------------------------------------------------===// 9e78f53d1SNikolas Klauser 10*ce777190SNikolas Klauser #ifndef _LIBCPP___CXX03___THREAD_ID_H 11*ce777190SNikolas Klauser #define _LIBCPP___CXX03___THREAD_ID_H 12e78f53d1SNikolas Klauser 1373fbae83SNikolas Klauser #include <__cxx03/__compare/ordering.h> 1473fbae83SNikolas Klauser #include <__cxx03/__config> 1573fbae83SNikolas Klauser #include <__cxx03/__fwd/functional.h> 1673fbae83SNikolas Klauser #include <__cxx03/__fwd/ostream.h> 1773fbae83SNikolas Klauser #include <__cxx03/__thread/support.h> 18e78f53d1SNikolas Klauser 19e78f53d1SNikolas Klauser #if !defined(_LIBCPP_HAS_NO_PRAGMA_SYSTEM_HEADER) 20e78f53d1SNikolas Klauser # pragma GCC system_header 21e78f53d1SNikolas Klauser #endif 22e78f53d1SNikolas Klauser 23e78f53d1SNikolas Klauser _LIBCPP_BEGIN_NAMESPACE_STD 24e78f53d1SNikolas Klauser 25e78f53d1SNikolas Klauser #ifndef _LIBCPP_HAS_NO_THREADS 26e78f53d1SNikolas Klauser class _LIBCPP_EXPORTED_FROM_ABI __thread_id; 27e78f53d1SNikolas Klauser 28e78f53d1SNikolas Klauser namespace this_thread { 29e78f53d1SNikolas Klauser 30e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI __thread_id get_id() _NOEXCEPT; 31e78f53d1SNikolas Klauser 32e78f53d1SNikolas Klauser } // namespace this_thread 33e78f53d1SNikolas Klauser 34e78f53d1SNikolas Klauser template <> 35e78f53d1SNikolas Klauser struct hash<__thread_id>; 36e78f53d1SNikolas Klauser 37e78f53d1SNikolas Klauser class _LIBCPP_TEMPLATE_VIS __thread_id { 38e78f53d1SNikolas Klauser // FIXME: pthread_t is a pointer on Darwin but a long on Linux. 39e78f53d1SNikolas Klauser // NULL is the no-thread value on Darwin. Someone needs to check 40e78f53d1SNikolas Klauser // on other platforms. We assume 0 works everywhere for now. 41e78f53d1SNikolas Klauser __libcpp_thread_id __id_; 42e78f53d1SNikolas Klauser 43e78f53d1SNikolas Klauser static _LIBCPP_HIDE_FROM_ABI bool 44e78f53d1SNikolas Klauser __lt_impl(__thread_id __x, __thread_id __y) _NOEXCEPT { // id==0 is always less than any other thread_id 45e78f53d1SNikolas Klauser if (__x.__id_ == 0) 46e78f53d1SNikolas Klauser return __y.__id_ != 0; 47e78f53d1SNikolas Klauser if (__y.__id_ == 0) 48e78f53d1SNikolas Klauser return false; 49e78f53d1SNikolas Klauser return __libcpp_thread_id_less(__x.__id_, __y.__id_); 50e78f53d1SNikolas Klauser } 51e78f53d1SNikolas Klauser 52e78f53d1SNikolas Klauser public: 53e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI __thread_id() _NOEXCEPT : __id_(0) {} 54e78f53d1SNikolas Klauser 55e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI void __reset() { __id_ = 0; } 56e78f53d1SNikolas Klauser 57e78f53d1SNikolas Klauser friend _LIBCPP_HIDE_FROM_ABI bool operator==(__thread_id __x, __thread_id __y) _NOEXCEPT; 58e78f53d1SNikolas Klauser # if _LIBCPP_STD_VER <= 17 59e78f53d1SNikolas Klauser friend _LIBCPP_HIDE_FROM_ABI bool operator<(__thread_id __x, __thread_id __y) _NOEXCEPT; 60e78f53d1SNikolas Klauser # else // _LIBCPP_STD_VER <= 17 61e78f53d1SNikolas Klauser friend _LIBCPP_HIDE_FROM_ABI strong_ordering operator<=>(__thread_id __x, __thread_id __y) noexcept; 62e78f53d1SNikolas Klauser # endif // _LIBCPP_STD_VER <= 17 63e78f53d1SNikolas Klauser 64e78f53d1SNikolas Klauser template <class _CharT, class _Traits> 65e78f53d1SNikolas Klauser friend _LIBCPP_HIDE_FROM_ABI basic_ostream<_CharT, _Traits>& 66e78f53d1SNikolas Klauser operator<<(basic_ostream<_CharT, _Traits>& __os, __thread_id __id); 67e78f53d1SNikolas Klauser 68e78f53d1SNikolas Klauser private: 69e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI __thread_id(__libcpp_thread_id __id) : __id_(__id) {} 70e78f53d1SNikolas Klauser 71e78f53d1SNikolas Klauser _LIBCPP_HIDE_FROM_ABI friend __libcpp_thread_id __get_underlying_id(const __thread_id __id) { return __id.__id_; } 72e78f53d1SNikolas Klauser 73e78f53d1SNikolas Klauser friend __thread_id this_thread::get_id() _NOEXCEPT; 74e78f53d1SNikolas Klauser friend class _LIBCPP_EXPORTED_FROM_ABI thread; 75e78f53d1SNikolas Klauser friend struct _LIBCPP_TEMPLATE_VIS hash<__thread_id>; 76e78f53d1SNikolas Klauser }; 77e78f53d1SNikolas Klauser 78e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator==(__thread_id __x, __thread_id __y) _NOEXCEPT { 79e78f53d1SNikolas Klauser // Don't pass id==0 to underlying routines 80e78f53d1SNikolas Klauser if (__x.__id_ == 0) 81e78f53d1SNikolas Klauser return __y.__id_ == 0; 82e78f53d1SNikolas Klauser if (__y.__id_ == 0) 83e78f53d1SNikolas Klauser return false; 84e78f53d1SNikolas Klauser return __libcpp_thread_id_equal(__x.__id_, __y.__id_); 85e78f53d1SNikolas Klauser } 86e78f53d1SNikolas Klauser 87e78f53d1SNikolas Klauser # if _LIBCPP_STD_VER <= 17 88e78f53d1SNikolas Klauser 89e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator!=(__thread_id __x, __thread_id __y) _NOEXCEPT { return !(__x == __y); } 90e78f53d1SNikolas Klauser 91e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator<(__thread_id __x, __thread_id __y) _NOEXCEPT { 92e78f53d1SNikolas Klauser return __thread_id::__lt_impl(__x.__id_, __y.__id_); 93e78f53d1SNikolas Klauser } 94e78f53d1SNikolas Klauser 95e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator<=(__thread_id __x, __thread_id __y) _NOEXCEPT { return !(__y < __x); } 96e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator>(__thread_id __x, __thread_id __y) _NOEXCEPT { return __y < __x; } 97e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI bool operator>=(__thread_id __x, __thread_id __y) _NOEXCEPT { return !(__x < __y); } 98e78f53d1SNikolas Klauser 99e78f53d1SNikolas Klauser # else // _LIBCPP_STD_VER <= 17 100e78f53d1SNikolas Klauser 101e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI strong_ordering operator<=>(__thread_id __x, __thread_id __y) noexcept { 102e78f53d1SNikolas Klauser if (__x == __y) 103e78f53d1SNikolas Klauser return strong_ordering::equal; 104e78f53d1SNikolas Klauser if (__thread_id::__lt_impl(__x, __y)) 105e78f53d1SNikolas Klauser return strong_ordering::less; 106e78f53d1SNikolas Klauser return strong_ordering::greater; 107e78f53d1SNikolas Klauser } 108e78f53d1SNikolas Klauser 109e78f53d1SNikolas Klauser # endif // _LIBCPP_STD_VER <= 17 110e78f53d1SNikolas Klauser 111e78f53d1SNikolas Klauser namespace this_thread { 112e78f53d1SNikolas Klauser 113e78f53d1SNikolas Klauser inline _LIBCPP_HIDE_FROM_ABI __thread_id get_id() _NOEXCEPT { return __libcpp_thread_get_current_id(); } 114e78f53d1SNikolas Klauser 115e78f53d1SNikolas Klauser } // namespace this_thread 116e78f53d1SNikolas Klauser 117e78f53d1SNikolas Klauser #endif // !_LIBCPP_HAS_NO_THREADS 118e78f53d1SNikolas Klauser 119e78f53d1SNikolas Klauser _LIBCPP_END_NAMESPACE_STD 120e78f53d1SNikolas Klauser 121*ce777190SNikolas Klauser #endif // _LIBCPP___CXX03___THREAD_ID_H 122