16 #ifndef SUMTY_DETAIL_AUTO_UNION_HPP
17 #define SUMTY_DETAIL_AUTO_UNION_HPP
19 #include "sumty/detail/traits.hpp"
20 #include "sumty/detail/utils.hpp"
21 #include "sumty/utils.hpp"
25 #include <type_traits>
31 namespace sumty::detail {
33 template <
typename... T>
38 constexpr auto_union()
noexcept {}
40 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
42 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
44 constexpr ~auto_union()
noexcept {}
46 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
50 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
55 template <
typename T0,
typename... TN>
56 union auto_union<T0, TN...> {
57 SUMTY_NO_UNIQ_ADDR std::remove_const_t<T0> head_;
58 SUMTY_NO_UNIQ_ADDR auto_union<TN...> tail_;
60 constexpr auto_union()
noexcept {}
62 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
64 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
66 constexpr ~auto_union()
noexcept {}
68 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
72 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
77 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0, TN...>>::reference
79 if constexpr (IDX == 0) {
82 return tail_.
template get<IDX - 1>();
87 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0, TN...>>::const_reference get()
89 if constexpr (IDX == 0) {
92 return tail_.
template get<IDX - 1>();
96 template <size_t IDX,
typename... Args>
97 void construct(Args&&... args) {
98 if constexpr (IDX == 0) {
99 std::construct_at(&head_, std::forward<Args>(args)...);
101 tail_.
template construct<IDX - 1>(std::forward<Args>(args)...);
105 template <size_t IDX>
107 if constexpr (IDX == 0) {
108 std::destroy_at(&head_);
110 tail_.
template destroy<IDX - 1>();
115 template <
typename T0,
typename... TN>
116 union auto_union<T0&&, TN...> {
117 SUMTY_NO_UNIQ_ADDR std::remove_const_t<T0> head_;
118 SUMTY_NO_UNIQ_ADDR auto_union<TN...> tail_;
120 constexpr auto_union()
noexcept {}
122 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
124 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
126 constexpr ~auto_union()
noexcept {}
128 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
132 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
136 template <size_t IDX>
137 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0, TN...>>::reference
139 if constexpr (IDX == 0) {
142 return tail_.
template get<IDX - 1>();
146 template <size_t IDX>
147 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0, TN...>>::const_reference get()
149 if constexpr (IDX == 0) {
152 return tail_.
template get<IDX - 1>();
156 template <size_t IDX,
typename... Args>
157 void construct(Args&&... args) {
158 if constexpr (IDX == 0) {
159 std::construct_at(&head_, std::forward<Args>(args)...);
161 tail_.
template construct<IDX - 1>(std::forward<Args>(args)...);
165 template <size_t IDX>
167 if constexpr (IDX == 0) {
168 std::destroy_at(&head_);
170 tail_.
template destroy<IDX - 1>();
175 template <
typename T0,
typename... TN>
176 union auto_union<T0&, TN...> {
178 SUMTY_NO_UNIQ_ADDR auto_union<TN...> tail_;
180 constexpr auto_union()
noexcept {}
182 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
184 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
186 constexpr ~auto_union()
noexcept {}
188 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
192 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
196 template <size_t IDX>
197 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0&, TN...>>::reference
199 if constexpr (IDX == 0) {
202 return tail_.
template get<IDX - 1>();
206 template <size_t IDX>
207 [[nodiscard]]
constexpr typename traits<select_t<IDX, T0&, TN...>>::const_reference
208 get()
const noexcept {
209 if constexpr (IDX == 0) {
212 return tail_.
template get<IDX - 1>();
216 template <size_t IDX,
typename... Args>
217 void construct([[maybe_unused]] Args&&... args) {
218 if constexpr (IDX == 0) {
219 if constexpr (!std::is_base_of_v<
never,
220 std::remove_cvref_t<first_t<Args...>>>) {
221 head_ = std::addressof(std::forward<Args>(args)...);
224 tail_.
template construct<IDX - 1>(std::forward<Args>(args)...);
228 template <size_t IDX>
230 if constexpr (IDX != 0) { tail_.
template destroy<IDX - 1>(); }
234 template <
typename... TN>
235 union auto_union<
void, TN...> {
236 SUMTY_NO_UNIQ_ADDR auto_union<TN...> tail_;
238 constexpr auto_union()
noexcept {}
240 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
242 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
244 constexpr ~auto_union()
noexcept {}
246 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
250 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
254 template <size_t IDX>
255 [[nodiscard]]
constexpr typename traits<select_t<IDX,
void, TN...>>::reference
257 if constexpr (IDX != 0) {
258 return tail_.
template get<IDX - 1>();
264 template <size_t IDX>
265 [[nodiscard]]
constexpr typename traits<select_t<IDX,
void, TN...>>::const_reference
266 get()
const noexcept {
267 if constexpr (IDX != 0) {
268 return tail_.
template get<IDX - 1>();
274 template <size_t IDX,
typename... Args>
275 void construct([[maybe_unused]] Args&&... args) {
276 if constexpr (IDX != 0) {
277 tail_.
template construct<IDX - 1>(std::forward<Args>(args)...);
281 template <size_t IDX>
283 if constexpr (IDX != 0) { tail_.
template destroy<IDX - 1>(); }
287 template <
typename... TN>
288 union auto_union<
never, TN...> {
289 SUMTY_NO_UNIQ_ADDR auto_union<TN...> tail_;
291 constexpr auto_union()
noexcept {}
293 constexpr auto_union([[maybe_unused]]
const auto_union& other)
noexcept {}
295 constexpr auto_union([[maybe_unused]] auto_union&& other)
noexcept {}
297 constexpr ~auto_union()
noexcept {}
299 constexpr auto_union& operator=([[maybe_unused]]
const auto_union& rhs)
noexcept {
303 constexpr auto_union& operator=([[maybe_unused]] auto_union&& rhs)
noexcept {
307 template <size_t IDX>
308 [[nodiscard]]
constexpr typename traits<select_t<IDX,
never, TN...>>::reference
310 if constexpr (IDX != 0) {
311 if (std::is_constant_evaluated()) {
316 return *
reinterpret_cast<
never*>(ptr);
323 template <size_t IDX>
324 [[nodiscard]]
constexpr typename traits<select_t<IDX,
never, TN...>>::const_reference
325 get()
const noexcept {
326 if constexpr (IDX != 0) {
327 if (std::is_constant_evaluated()) {
330 const void* ptr =
nullptr;
332 return *
reinterpret_cast<
const never*>(ptr);
339 template <size_t IDX,
typename... Args>
340 void construct([[maybe_unused]] Args&&... args) {
341 if constexpr (IDX == 0) {
345 [[maybe_unused]]
never value{std::forward<Args>(args)...};
347 tail_.
template construct<IDX - 1>(std::forward<Args>(args)...);
351 template <size_t IDX>
353 if constexpr (IDX != 0) { tail_.
template destroy<IDX - 1>(); }