4#if __has_include(<stop_token>)
5 #if __cpp_lib_jthread >= 201911L
6 #define OSSIA_HAS_STD_JTHREAD 1
8 #elif ((_LIBCPP_VERSION >= 18100) && (_LIBCPP_VERSION < 99999)) || (_LIBCPP_VERSION >= 180100)
9 #if defined(_LIBCPP_HAS_NO_EXPERIMENTAL_STOP_TOKEN)
10 #error Rebuild with -fexperimental-library, clang 18 ships headers which are incompatible with this file but hides half of them behind that flag
12 #define OSSIA_HAS_STD_JTHREAD 1
18#if OSSIA_HAS_STD_JTHREAD
22#include <ossia/detail/audio_spin_mutex.hpp>
44struct __stop_callback_base
46 void (*__callback_)(__stop_callback_base*) =
nullptr;
48 __stop_callback_base* __next_ =
nullptr;
49 __stop_callback_base** __prev_ =
nullptr;
50 bool* __isRemoved_ =
nullptr;
51 std::atomic<bool> __callbackFinishedExecuting_{
false};
53 void __execute() noexcept { __callback_(
this); }
58 ~__stop_callback_base() =
default;
64 void __add_token_reference() noexcept
66 __state_.fetch_add(__token_ref_increment, std::memory_order_relaxed);
69 void __remove_token_reference() noexcept
72 = __state_.fetch_sub(__token_ref_increment, std::memory_order_acq_rel);
73 if(__oldState < (__token_ref_increment + __token_ref_increment))
79 void __add_source_reference() noexcept
81 __state_.fetch_add(__source_ref_increment, std::memory_order_relaxed);
84 void __remove_source_reference() noexcept
87 = __state_.fetch_sub(__source_ref_increment, std::memory_order_acq_rel);
88 if(__oldState < (__token_ref_increment + __source_ref_increment))
94 bool __request_stop() noexcept
97 if(!__try_lock_and_signal_until_signalled())
105 __signallingThread_ = std::this_thread::get_id();
107 while(__head_ !=
nullptr)
110 auto* __cb = __head_;
111 __head_ = __cb->__next_;
112 const bool anyMore = __head_ !=
nullptr;
115 __head_->__prev_ = &__head_;
118 __cb->__prev_ =
nullptr;
131 bool __isRemoved =
false;
132 __cb->__isRemoved_ = &__isRemoved;
138 __cb->__isRemoved_ =
nullptr;
139 __cb->__callbackFinishedExecuting_.store(
true, std::memory_order_release);
159 bool __is_stop_requested() noexcept
161 return __is_stop_requested(__state_.load(std::memory_order_acquire));
164 bool __is_stop_requestable() noexcept
166 return __is_stop_requestable(__state_.load(std::memory_order_acquire));
169 bool __try_add_callback(
170 __stop_callback_base* __cb,
bool __incrementRefCountIfSuccessful)
noexcept
172 std::uint64_t __oldState;
179 ossia_rwlock_pause();
181 __oldState = __state_.load(std::memory_order_acquire);
183 if(__is_stop_requested(__oldState))
188 else if(!__is_stop_requestable(__oldState))
192 }
while(__is_locked(__oldState));
193 }
while(!__state_.compare_exchange_weak(
194 __oldState, __oldState | __locked_flag, std::memory_order_acquire));
197 __cb->__next_ = __head_;
198 if(__cb->__next_ !=
nullptr)
200 __cb->__next_->__prev_ = &__cb->__next_;
202 __cb->__prev_ = &__head_;
205 if(__incrementRefCountIfSuccessful)
207 __unlock_and_increment_token_ref_count();
218 void __remove_callback(__stop_callback_base* __cb)
noexcept
222 if(__cb->__prev_ !=
nullptr)
226 *__cb->__prev_ = __cb->__next_;
227 if(__cb->__next_ !=
nullptr)
229 __cb->__next_->__prev_ = __cb->__prev_;
232 __unlock_and_decrement_token_ref_count();
242 if(__signallingThread_ == std::this_thread::get_id())
246 if(__cb->__isRemoved_ !=
nullptr)
251 *__cb->__isRemoved_ =
true;
258 while(!__cb->__callbackFinishedExecuting_.load(std::memory_order_acquire))
260 ossia_rwlock_pause();
264 __remove_token_reference();
268 static bool __is_locked(std::uint64_t __state)
noexcept
270 return (__state & __locked_flag) != 0;
273 static bool __is_stop_requested(std::uint64_t __state)
noexcept
275 return (__state & __stop_requested_flag) != 0;
278 static bool __is_stop_requestable(std::uint64_t __state)
noexcept
282 return __is_stop_requested(__state) || (__state >= __source_ref_increment);
285 bool __try_lock_and_signal_until_signalled() noexcept
287 std::uint64_t __oldState = __state_.load(std::memory_order_acquire);
290 if(__is_stop_requested(__oldState))
292 while(__is_locked(__oldState))
294 ossia_rwlock_pause();
295 __oldState = __state_.load(std::memory_order_acquire);
296 if(__is_stop_requested(__oldState))
299 }
while(!__state_.compare_exchange_weak(
300 __oldState, __oldState | __stop_requested_flag | __locked_flag,
301 std::memory_order_acq_rel, std::memory_order_acquire));
305 void __lock() noexcept
307 auto __oldState = __state_.load(std::memory_order_relaxed);
310 while(__is_locked(__oldState))
312 ossia_rwlock_pause();
313 __oldState = __state_.load(std::memory_order_relaxed);
315 }
while(!__state_.compare_exchange_weak(
316 __oldState, __oldState | __locked_flag, std::memory_order_acquire,
317 std::memory_order_relaxed));
320 void __unlock() noexcept
322 __state_.fetch_sub(__locked_flag, std::memory_order_release);
325 void __unlock_and_increment_token_ref_count() noexcept
327 __state_.fetch_sub(__locked_flag - __token_ref_increment, std::memory_order_release);
330 void __unlock_and_decrement_token_ref_count() noexcept
332 auto __oldState = __state_.fetch_sub(
333 __locked_flag + __token_ref_increment, std::memory_order_acq_rel);
336 if(__oldState < (__locked_flag + __token_ref_increment + __token_ref_increment))
342 static constexpr std::uint64_t __stop_requested_flag = 1u;
343 static constexpr std::uint64_t __locked_flag = 2u;
344 static constexpr std::uint64_t __token_ref_increment = 4u;
345 static constexpr std::uint64_t __source_ref_increment =
static_cast<std::uint64_t
>(1u)
352 std::atomic<std::uint64_t> __state_{__source_ref_increment};
353 __stop_callback_base* __head_ =
nullptr;
354 std::thread::id __signallingThread_{};
362template <
typename _Callback>
369 explicit nostopstate_t() =
default;
371inline constexpr nostopstate_t nostopstate{};
382 stop_token() noexcept
388 stop_token(
const stop_token& __it) noexcept
389 : __state_(__it.__state_)
391 if(__state_ !=
nullptr)
393 __state_->__add_token_reference();
397 stop_token(stop_token&& __it) noexcept
398 : __state_(std::exchange(__it.__state_,
nullptr))
404 if(__state_ !=
nullptr)
406 __state_->__remove_token_reference();
410 stop_token& operator=(
const stop_token& __it)
noexcept
412 if(__state_ != __it.__state_)
414 stop_token __tmp{__it};
420 stop_token& operator=(stop_token&& __it)
noexcept
422 stop_token __tmp{std::move(__it)};
427 void swap(stop_token& __it)
noexcept { std::swap(__state_, __it.__state_); }
430 [[nodiscard]]
bool stop_requested() const noexcept
432 return __state_ !=
nullptr && __state_->__is_stop_requested();
435 [[nodiscard]]
bool stop_possible() const noexcept
437 return __state_ !=
nullptr && __state_->__is_stop_requestable();
440 [[nodiscard]]
friend bool
441 operator==(
const stop_token& __a,
const stop_token& __b)
noexcept
443 return __a.__state_ == __b.__state_;
445 [[nodiscard]]
friend bool
446 operator!=(
const stop_token& __a,
const stop_token& __b)
noexcept
448 return __a.__state_ != __b.__state_;
452 friend class stop_source;
453 template <
typename _Callback>
454 friend class stop_callback;
456 explicit stop_token(__stop_state* __state) noexcept
459 if(__state_ !=
nullptr)
461 __state_->__add_token_reference();
465 __stop_state* __state_;
476 : __state_(new __stop_state())
480 explicit stop_source(nostopstate_t) noexcept
487 if(__state_ !=
nullptr)
489 __state_->__remove_source_reference();
493 stop_source(
const stop_source& __other) noexcept
494 : __state_(__other.__state_)
496 if(__state_ !=
nullptr)
498 __state_->__add_source_reference();
502 stop_source(stop_source&& __other) noexcept
503 : __state_(std::exchange(__other.__state_,
nullptr))
507 stop_source& operator=(stop_source&& __other)
noexcept
509 stop_source __tmp{std::move(__other)};
514 stop_source& operator=(
const stop_source& __other)
noexcept
516 if(__state_ != __other.__state_)
518 stop_source __tmp{__other};
524 [[nodiscard]]
bool stop_requested() const noexcept
526 return __state_ !=
nullptr && __state_->__is_stop_requested();
529 [[nodiscard]]
bool stop_possible() const noexcept {
return __state_ !=
nullptr; }
531 bool request_stop() noexcept
533 if(__state_ !=
nullptr)
535 return __state_->__request_stop();
540 [[nodiscard]] stop_token get_token() const noexcept {
return stop_token{__state_}; }
542 void swap(stop_source& __other)
noexcept { std::swap(__state_, __other.__state_); }
544 [[nodiscard]]
friend bool
545 operator==(
const stop_source& __a,
const stop_source& __b)
noexcept
547 return __a.__state_ == __b.__state_;
549 [[nodiscard]]
friend bool
550 operator!=(
const stop_source& __a,
const stop_source& __b)
noexcept
552 return __a.__state_ != __b.__state_;
556 __stop_state* __state_;
563template <
typename _Callback>
565class [[nodiscard]] stop_callback :
private __stop_callback_base
568 using callback_type = _Callback;
571 typename _CB, std::enable_if_t<std::is_constructible_v<_Callback, _CB>,
int> = 0>
573 explicit stop_callback(
const stop_token& __token, _CB&& __cb)
noexcept(
574 std::is_nothrow_constructible_v<_Callback, _CB>)
575 : __stop_callback_base{[](__stop_callback_base* __that) noexcept {
576 static_cast<stop_callback*>(__that)->__execute();
579 , __cb_(
static_cast<_CB&&
>(__cb))
581 if(__token.__state_ !=
nullptr && __token.__state_->__try_add_callback(
this,
true))
583 __state_ = __token.__state_;
588 typename _CB, std::enable_if_t<std::is_constructible_v<_Callback, _CB>,
int> = 0>
590 explicit stop_callback(stop_token&& __token, _CB&& __cb)
noexcept(
591 std::is_nothrow_constructible_v<_Callback, _CB>)
592 : __stop_callback_base{[](__stop_callback_base* __that) noexcept {
593 static_cast<stop_callback*>(__that)->__execute();
596 , __cb_(
static_cast<_CB&&
>(__cb))
598 if(__token.__state_ !=
nullptr && __token.__state_->__try_add_callback(
this,
false))
600 __state_ = std::exchange(__token.__state_,
nullptr);
607 if(__inExecute_.load())
609 std::cerr <<
"*** OOPS: ~stop_callback() while callback executed\n";
612 if(__state_ !=
nullptr)
614 __state_->__remove_callback(
this);
618 stop_callback& operator=(
const stop_callback&) =
delete;
619 stop_callback& operator=(stop_callback&&) =
delete;
620 stop_callback(
const stop_callback&) =
delete;
621 stop_callback(stop_callback&&) =
delete;
624 void __execute() noexcept
629 __inExecute_.store(
true);
631 __inExecute_.store(
false);
637 __stop_state* __state_;
640 std::atomic<bool> __inExecute_{
false};
644template <
typename _Callback>
645stop_callback(stop_token, _Callback) -> stop_callback<_Callback>;
653#include <type_traits>
672 using id = ::std::thread::id;
673 using native_handle_type = ::std::thread::native_handle_type;
681 typename Callable, typename... Args,
683 = ::std::enable_if_t<!::std::is_same_v<::std::decay_t<Callable>, jthread>>>
684 explicit jthread(Callable&& cb, Args&&... args);
687 jthread(const jthread&) = delete;
688 jthread(jthread&&) noexcept = default;
689 jthread& operator=(const jthread&) = delete;
690 jthread& operator=(jthread&&) noexcept;
693 void swap(jthread&) noexcept;
694 bool joinable() const noexcept;
698 id get_id() const noexcept;
699 native_handle_type native_handle();
702 static
unsigned hardware_concurrency() noexcept
704 return ::std::thread::hardware_concurrency();
710 [[nodiscard]] stop_source get_stop_source() noexcept;
711 [[nodiscard]] stop_token get_stop_token() const noexcept;
712 bool request_stop() noexcept {
return get_stop_source().request_stop(); }
720 stop_source _stopSource;
721 ::std::thread _thread{};
731inline jthread::jthread() noexcept
732 : _stopSource{nostopstate}
738template <
typename Callable,
typename... Args,
typename>
739inline jthread::jthread(Callable&& cb, Args&&... args)
743 [](stop_token st, auto&& cb, auto&&... args) {
745 if constexpr(std::is_invocable_v<Callable, stop_token, Args...>)
749 ::std::forward<
decltype(cb)>(cb), std::move(st),
750 ::std::forward<
decltype(args)>(args)...);
756 ::std::forward<
decltype(cb)>(cb), ::std::forward<
decltype(args)>(args)...);
759 _stopSource.get_token(),
760 ::std::forward<Callable>(cb),
761 ::std::forward<Args>(args)...
767inline jthread& jthread::operator=(jthread&& t)
noexcept
775 _thread = std::move(t._thread);
776 _stopSource = std::move(t._stopSource);
781inline jthread::~jthread()
791inline bool jthread::joinable() const noexcept
793 return _thread.joinable();
795inline void jthread::join()
799inline void jthread::detach()
803inline typename jthread::id jthread::get_id() const noexcept
805 return _thread.get_id();
807inline typename jthread::native_handle_type jthread::native_handle()
809 return _thread.native_handle();
812inline stop_source jthread::get_stop_source() noexcept
816inline stop_token jthread::get_stop_token() const noexcept
818 return _stopSource.get_token();
821inline void jthread::swap(jthread& t)
noexcept
823 std::swap(_stopSource, t._stopSource);
824 std::swap(_thread, t._thread);