7#include "polymorphic_optional.hpp"
8#include "../utility/utility.hpp"
9#include "../macros.hpp"
19hi_export_module(hikogui.container.wfree_fifo);
24hi_warning_ignore_msvc(26490);
26hi_export
namespace hi::inline
v1 {
38template<
typename T, std::
size_t SlotSize>
41 static_assert(std::has_single_bit(SlotSize),
"Only power-of-two number of messages size allowed.");
42 static_assert(SlotSize < 65536);
49 constexpr static std::size_t num_slots = fifo_size / slot_size;
61 [[nodiscard]]
bool empty() const noexcept
63 return _head.load(std::memory_order::relaxed) == _tail;
73 template<
typename Func>
76 auto result = get_slot(_tail).invoke_and_reset(std::forward<Func>(func));
89 template<
typename Operation>
90 void take_all(Operation
const& operation)
noexcept
92 while (take_one(operation)) {}
102 template<
typename Message,
typename Func,
typename... Args>
112 auto const offset = _head.fetch_add(slot_size, std::memory_order::relaxed);
113 return get_slot(offset).template wait_emplace_and_invoke<Message>(std::forward<Func>(func), std::forward<Args>(args)...);
116 template<
typename Func,
typename Object>
117 hi_force_inline
auto insert_and_invoke(Func&& func, Object&&
object)
noexcept
119 return emplace_and_invoke<std::decay_t<Object>>(std::forward<Func>(func), std::forward<Object>(
object));
122 template<
typename Message,
typename... Args>
123 hi_force_inline
void emplace(Args&&...args)
noexcept
125 return emplace_and_invoke<Message>([](Message&) ->
void {}, std::forward<Args>(args)...);
128 template<
typename Object>
129 hi_force_inline
void insert(Object &&
object)
noexcept
131 return emplace<std::decay_t<Object>>(std::forward<Object>(
object));
135#if defined(__cpp_lib_hardware_interference_size)
136 constexpr static size_t destructive_interference_size = std::hardware_destructive_interference_size;
138 constexpr static size_t destructive_interference_size = 128;
148 hi_force_inline slot_type& get_slot(uint16_t offset)
noexcept
150 hi_axiom(offset % slot_size == 0);
152 return *std::launder(
153 std::assume_aligned<slot_size>(
reinterpret_cast<slot_type *
>(
reinterpret_cast<char *
>(
this) + offset)));
DOXYGEN BUG.
Definition algorithm_misc.hpp:20
Polymorphic optional.
Definition polymorphic_optional.hpp:43
A wait-free multiple-producer/single-consumer fifo designed for absolute performance.
Definition wfree_fifo.hpp:39
auto take_one(Func &&func) noexcept
Take one message from the fifo slot.
Definition wfree_fifo.hpp:74
void take_all(Operation const &operation) noexcept
Take all message from the queue.
Definition wfree_fifo.hpp:90
bool empty() const noexcept
Check if fifo is empty.
Definition wfree_fifo.hpp:61
hi_force_inline auto emplace_and_invoke(Func &&func, Args &&...args) noexcept
Create an message in-place on the fifo.
Definition wfree_fifo.hpp:103