xref: /llvm-project/libcxx/include/__cxx03/__thread/id.h (revision ce7771902dc50d900de639d499a60486b83f70e0)
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