16 #ifndef SUMTY_DETAIL_VARIANT_IMPL_HPP
17 #define SUMTY_DETAIL_VARIANT_IMPL_HPP
19 #include "sumty/detail/auto_union.hpp"
20 #include "sumty/detail/traits.hpp"
21 #include "sumty/detail/utils.hpp"
22 #include "sumty/utils.hpp"
27 #include <type_traits>
36 #pragma warning(disable : 4702
)
39 namespace sumty::detail {
43 static inline constexpr uninit_t uninit{};
45 template <
typename Enable,
typename... T>
48 using discrim_t = discriminant_t<
sizeof...(T)>;
50 SUMTY_NO_UNIQ_ADDR auto_union<T...> data_;
54 constexpr void copy_construct(
const auto_union<T...>& data) {
55 if constexpr (I <
sizeof...(T)) {
56 if (discrim_ ==
static_cast<discrim_t>(I)) {
57 if constexpr (std::is_void_v<select_t<I, T...>>) {
58 data_.
template construct<I>();
60 data_.
template construct<I>(data.
template get<I>());
63 copy_construct<I + 1>(data);
69 constexpr void move_construct(auto_union<T...>& data)
noexcept(
70 (
true && ... && traits<T>::is_nothrow_move_constructible)) {
71 if constexpr (I <
sizeof...(T)) {
72 if (discrim_ ==
static_cast<discrim_t>(I)) {
73 if constexpr (std::is_void_v<select_t<I, T...>>) {
74 data_.
template construct<I>();
75 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
76 data_.
template construct<I>(data.
template get<I>());
78 data_.
template construct<I>(std::move(data.
template get<I>()));
81 move_construct<I + 1>(data);
87 constexpr void destroy()
noexcept((
true && ... && traits<T>::is_nothrow_destructible)) {
88 if constexpr (I <
sizeof...(T)) {
89 if (discrim_ ==
static_cast<discrim_t>(I)) {
90 data_.
template destroy<I>();
98 constexpr void copy_assign(
const auto_union<T...>& data) {
99 if constexpr (I <
sizeof...(T)) {
100 if (discrim_ ==
static_cast<discrim_t>(I)) {
101 if constexpr (std::is_void_v<select_t<I, T...>>) {
102 data_.
template construct<I>();
103 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
104 data_.
template construct<I>(data.
template get<I>());
106 data_.
template get<I>() = data.
template get<I>();
109 copy_assign<I + 1>(data);
115 constexpr void move_assign(auto_union<T...>& data)
noexcept(
116 (
true && ... && traits<T>::is_nothrow_move_assignable)) {
117 if constexpr (I <
sizeof...(T)) {
118 if (discrim_ ==
static_cast<discrim_t>(I)) {
119 if constexpr (std::is_void_v<select_t<I, T...>>) {
120 data_.
template construct<I>();
121 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
122 data_.
template construct<I>(data.
template get<I>());
124 data_.
template get<I>() = std::move(data.
template get<I>());
127 move_assign<I + 1>(data);
133 constexpr void same_swap(auto_union<T...>& data)
noexcept(
134 (
true && ... && traits<T>::is_nothrow_swappable)) {
135 if constexpr (I <
sizeof...(T)) {
136 if (discrim_ ==
static_cast<discrim_t>(I)) {
137 if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
138 auto* tmp = &data.
template get<I>();
139 data.
template construct<I>(data_.
template get<I>());
140 data_.
template construct<I>(*tmp);
141 }
else if constexpr (!std::is_void_v<select_t<I, T...>>) {
143 swap(data_.
template get<I>(), data.
template get<I>());
146 same_swap<I + 1>(data);
151 template <size_t I, size_t J>
152 constexpr void diff_swap_impl(variant_impl& other)
noexcept((
154 (traits<T>::is_nothrow_move_constructible && traits<T>::is_nothrow_destructible))) {
155 std::swap(discrim_, other.discrim_);
156 if constexpr (std::is_void_v<select_t<I, T...>>) {
157 if constexpr (std::is_void_v<select_t<J, T...>>) {
159 }
else if constexpr (std::is_lvalue_reference_v<select_t<J, T...>>) {
160 data_.
template construct<J>(other.data_.
template get<J>());
161 other.data_.
template destroy<J>();
162 other.data_.
template construct<I>();
164 data_.
template construct<J>(std::move(other.data_.
template get<J>()));
165 other.data_.
template destroy<J>();
166 other.data_.
template construct<I>();
168 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
169 if constexpr (std::is_void_v<select_t<J, T...>>) {
170 other.data_.
template construct<I>(data_.
template get<I>());
171 data_.
template destroy<I>();
172 data_.
template construct<J>();
173 }
else if constexpr (std::is_lvalue_reference_v<select_t<J, T...>>) {
174 auto& tmp = other.data_.
template get<J>();
175 other.data_.
template construct<I>(data_.
template get<I>());
176 data_.
template construct<J>(tmp);
178 auto& tmp = data_.
template get<I>();
179 data_.
template construct<J>(std::move(other.data_.
template get<J>()));
180 other.data_.
template destroy<J>();
181 other.data_.
template construct<I>(tmp);
184 if constexpr (std::is_void_v<select_t<J, T...>>) {
185 other.data_.
template construct<I>(std::move(data_.
template get<I>()));
186 data_.
template destroy<I>();
187 data_.
template construct<J>();
188 }
else if constexpr (std::is_lvalue_reference_v<select_t<J, T...>>) {
189 auto& tmp = other.data_.
template get<J>();
190 other.data_.
template construct<I>(std::move(data_.
template get<I>()));
191 data_.
template destroy<I>();
192 data_.
template construct<J>(tmp);
194 auto&& tmp = std::move(other.data_.
template get<J>());
195 other.data_.
template destroy<J>();
196 other.data_.
template construct<I>(std::move(data_.
template get<I>()));
197 data_.
template destroy<I>();
198 data_.
template construct<J>(std::move(tmp));
203 template <size_t I, size_t J>
204 constexpr void diff_swap_nested(variant_impl& other)
noexcept((
206 (traits<T>::is_nothrow_move_constructible && traits<T>::is_nothrow_destructible))) {
207 if constexpr (J <
sizeof...(T)) {
208 if (other.discrim_ ==
static_cast<discrim_t>(J)) {
209 diff_swap_impl<I, J>(other);
211 diff_swap_nested<I, J + 1>(other);
217 constexpr void diff_swap(variant_impl& other)
noexcept((
219 (traits<T>::is_nothrow_move_constructible && traits<T>::is_nothrow_destructible))) {
220 if constexpr (I <
sizeof...(T)) {
221 if (discrim_ ==
static_cast<discrim_t>(I)) {
222 diff_swap_nested<I, 0>(other);
224 diff_swap<I + 1>(other);
231 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
233 constexpr variant_impl()
noexcept(
234 traits<first_t<T...>>::is_nothrow_default_constructible)
235 : variant_impl(std::in_place_index<0>) {}
237 constexpr variant_impl(
const variant_impl& other) : discrim_(other.discrim_) {
238 copy_construct<0>(other.data_);
241 constexpr variant_impl(variant_impl&& other)
noexcept(
242 (
true && ... && traits<T>::is_nothrow_move_constructible))
243 : discrim_(other.discrim_) {
244 move_construct<0>(other.data_);
247 template <size_t I,
typename... Args>
249 constexpr explicit(
sizeof...(Args) == 0) variant_impl(
250 [[maybe_unused]] std::in_place_index_t<I> inplace,
251 Args&&... args)
noexcept(traits<select_t<I, T...>>::
252 template is_nothrow_constructible<Args...>)
253 : discrim_(
static_cast<discrim_t>(I)) {
254 data_.
template construct<I>(std::forward<Args>(args)...);
257 constexpr ~variant_impl()
noexcept((
true && ... &&
258 traits<T>::is_nothrow_destructible)) {
262 constexpr variant_impl& operator=(
const variant_impl& rhs) {
264 if (discrim_ == rhs.discrim_) {
265 copy_assign<0>(rhs.data_);
268 discrim_ = rhs.discrim_;
269 copy_construct<0>(rhs.data_);
275 constexpr variant_impl& operator=(variant_impl&& rhs)
noexcept((
277 (traits<T>::is_nothrow_move_assignable &&
278 traits<T>::is_nothrow_move_constructible && traits<T>::is_nothrow_destructible))) {
279 if (discrim_ == rhs.discrim_) {
280 move_assign<0>(rhs.data_);
283 discrim_ = rhs.discrim_;
284 move_construct<0>(rhs.data_);
289 [[nodiscard]]
constexpr size_t index()
const noexcept {
290 return static_cast<size_t>(discrim_);
294 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::reference get() &
noexcept {
295 return data_.
template get<I>();
299 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_reference get()
301 return data_.
template get<I>();
305 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::rvalue_reference get() && {
306 if constexpr (std::is_void_v<select_t<I, T...>>) {
308 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
309 return data_.
template get<I>();
311 return std::move(data_.
template get<I>());
316 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_rvalue_reference get()
318 if constexpr (std::is_void_v<select_t<I, T...>>) {
320 }
else if constexpr (std::is_lvalue_reference_v<select_t<I, T...>>) {
321 return data_.
template get<I>();
323 return std::move(data_.
template get<I>());
328 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::pointer ptr()
noexcept {
329 if constexpr (!std::is_void_v<select_t<I, T...>>) {
330 return &data_.
template get<I>();
337 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_pointer ptr()
339 if constexpr (!std::is_void_v<select_t<I, T...>>) {
340 return &data_.
template get<I>();
346 template <size_t I,
typename... Args>
347 constexpr void emplace(Args&&... args) {
349 data_.
template construct<I>(std::forward<Args>(args)...);
350 discrim_ =
static_cast<discrim_t>(I);
353 template <size_t I,
typename... Args>
354 constexpr void uninit_emplace(Args&&... args) {
355 data_.
template construct<I>(std::forward<Args>(args)...);
356 discrim_ =
static_cast<discrim_t>(I);
359 constexpr void swap(variant_impl& other)
noexcept(
361 (traits<T>::is_nothrow_swappable && traits<T>::is_nothrow_move_constructible &&
362 traits<T>::is_nothrow_destructible))) {
363 if (discrim_ == other.discrim_) {
364 same_swap<0>(other.data_);
372 class variant_impl<
void> {};
374 template <
typename... T>
378 struct empty_tuple<> {};
380 template <
typename... TN>
381 struct empty_tuple<
void, TN...> : empty_tuple<TN...> {
383 constexpr typename traits<select_t<I,
void, TN...>>::reference get()
noexcept {
384 if constexpr (I != 0) {
385 static_cast<empty_tuple<TN...>*>(
this)->
template get<I - 1>();
390 constexpr typename traits<select_t<I,
void, TN...>>::const_reference get()
392 if constexpr (I != 0) {
393 static_cast<
const empty_tuple<TN...>*>(
this)->
template get<I - 1>();
398 template <
typename T0,
typename... TN>
399 requires(std::is_base_of_v<never, std::remove_cvref_t<T0>>)
400 struct empty_tuple<T0, TN...> : empty_tuple<TN...> {
403 constexpr typename traits<select_t<I,
never, TN...>>::reference get()
noexcept {
404 if constexpr (I == 0) {
405 if (std::is_constant_evaluated()) {
410 return *
reinterpret_cast<
never*>(ptr);
413 static_cast<empty_tuple<TN...>*>(
this)->
template get<I - 1>();
419 constexpr typename traits<select_t<I,
never, TN...>>::const_reference get()
421 if constexpr (I == 0) {
422 if (std::is_constant_evaluated()) {
425 const void* ptr =
nullptr;
427 return *
reinterpret_cast<
const never*>(ptr);
430 static_cast<
const empty_tuple<TN...>*>(
this)->
template get<I - 1>();
435 template <
typename T0,
typename... TN>
436 requires(!std::is_void_v<T0> && !std::is_base_of_v<never, std::remove_cvref_t<T0>>)
437 struct empty_tuple<T0, TN...> : empty_tuple<TN...> {
439 static inline T0 head_{};
442 constexpr typename traits<select_t<I, T0, TN...>>::reference get()
noexcept {
443 if constexpr (I == 0) {
446 static_cast<empty_tuple<TN...>*>(
this)->
template get<I - 1>();
451 constexpr typename traits<select_t<I, T0, TN...>>::const_reference get()
453 if constexpr (I == 0) {
456 static_cast<
const empty_tuple<TN...>*>(
this)->
template get<I - 1>();
461 template <
typename T>
462 concept void_or_empty_or_never = std::is_void_v<T> || is_sumty_empty_v<T> ||
463 std::is_base_of_v<never, std::remove_cvref_t<T>>;
465 template <void_or_empty_or_never... T>
467 std::enable_if_t<(
sizeof...(T) > 1) &&
468 !std::is_base_of_v<
never, std::remove_cvref_t<last_t<T...>>> &&
469 !std::is_base_of_v<
never, std::remove_cvref_t<select_t<0, T...>>>>,
472 using discrim_t = discriminant_t<
sizeof...(T)>;
474 discrim_t discrim_{};
475 SUMTY_NO_UNIQ_ADDR empty_tuple<T...> data_{};
479 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
481 constexpr variant_impl()
noexcept =
default;
483 constexpr variant_impl(
const variant_impl&)
noexcept =
default;
485 constexpr variant_impl(variant_impl&&)
noexcept =
default;
487 template <size_t I,
typename... Args>
488 constexpr explicit(
sizeof...(Args) == 0)
490 variant_impl([[maybe_unused]] std::in_place_index_t<I> inplace, Args&&... args)
491 : discrim_(
static_cast<discrim_t>(I)) {
495 discrim_ =
static_cast<discrim_t>(I);
497 if constexpr (
sizeof...(Args) > 0) {
498 std::construct_at(&data_.
template get<I>(), std::forward<Args>(args)...);
502 constexpr ~variant_impl()
noexcept =
default;
504 constexpr variant_impl& operator=(
const variant_impl&) =
default;
506 constexpr variant_impl& operator=(variant_impl&&) =
default;
508 [[nodiscard]]
constexpr size_t index()
const noexcept {
509 return static_cast<size_t>(discrim_);
513 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::reference get()
noexcept {
514 if constexpr (!std::is_void_v<select_t<I, T...>>) {
515 return data_.
template get<I>();
520 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_reference get()
522 if constexpr (!std::is_void_v<select_t<I, T...>>) {
523 return data_.
template get<I>();
528 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::pointer ptr()
noexcept {
529 if constexpr (!std::is_void_v<select_t<I, T...>>) {
530 return &data_.
template get<I>();
535 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_pointer ptr()
537 if constexpr (!std::is_void_v<select_t<I, T...>>) {
538 return &data_.
template get<I>();
542 template <size_t I,
typename... Args>
543 constexpr void emplace([[maybe_unused]] Args&&... args) {
544 discrim_ =
static_cast<discrim_t>(I);
545 if constexpr (
sizeof...(Args) > 0 && !std::is_void_v<select_t<I, T...>>) {
546 std::construct_at(ptr<I>(), std::forward<Args>(args)...);
550 template <size_t I,
typename... Args>
551 constexpr void uninit_emplace([[maybe_unused]] Args&&... args) {
552 discrim_ =
static_cast<discrim_t>(I);
553 if constexpr (
sizeof...(Args) > 0 && !std::is_void_v<select_t<I, T...>>) {
554 std::construct_at(ptr<I>(), std::forward<Args>(args)...);
558 constexpr void swap(variant_impl& other)
noexcept {
559 std::swap(discrim_, other.discrim_);
563 template <
typename T>
564 class variant_impl<std::enable_if_t<!void_or_empty_or_never<T>>, T> {
566 SUMTY_NO_UNIQ_ADDR auto_union<T> data_;
570 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
572 constexpr variant_impl()
noexcept(traits<T>::is_nothrow_default_constructible) {
573 data_.
template construct<0>();
576 constexpr variant_impl(
const variant_impl& other) {
577 data_.
template construct<0>(other.data_.
template get<0>());
580 constexpr variant_impl(variant_impl&& other)
noexcept(
581 traits<T>::is_nothrow_move_constructible) {
582 data_.
template construct<0>(other.data_.
template get<0>());
585 template <
typename... Args>
586 constexpr explicit(
sizeof...(Args) == 0)
588 variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace, Args&&... args) {
589 data_.
template construct<0>(std::forward<Args>(args)...);
592 constexpr ~variant_impl()
noexcept(traits<T>::is_nothrow_destructible) {
593 data_.
template destroy<0>();
596 constexpr variant_impl& operator=(
const variant_impl& rhs) {
598 if constexpr (std::is_lvalue_reference_v<T>) {
599 data_.
template construct<0>(rhs.data_.
template get<0>());
601 data_.
template get<0>() = rhs.data_.
template get<0>();
607 constexpr variant_impl& operator=(variant_impl&& rhs)
noexcept(
608 traits<T>::is_nothrow_move_assignable) {
609 if constexpr (std::is_lvalue_reference_v<T>) {
610 data_.
template construct<0>(rhs.data_.
template get<0>());
612 data_.
template get<0>() = std::move(rhs.data_.
template get<0>());
617 [[nodiscard]]
static constexpr size_t index()
noexcept {
return 0; }
620 [[nodiscard]]
constexpr typename traits<T>::reference get() &
noexcept {
621 return data_.
template get<I>();
625 [[nodiscard]]
constexpr typename traits<T>::const_reference get()
const&
noexcept {
626 return data_.
template get<I>();
630 [[nodiscard]]
constexpr typename traits<T>::rvalue_reference get() && {
631 if constexpr (std::is_void_v<T>) {
633 }
else if constexpr (std::is_lvalue_reference_v<T>) {
634 return data_.
template get<I>();
636 return std::move(data_.
template get<I>());
641 [[nodiscard]]
constexpr typename traits<T>::const_rvalue_reference get()
const&& {
642 if constexpr (std::is_void_v<T>) {
644 }
else if constexpr (std::is_lvalue_reference_v<T>) {
645 return data_.
template get<I>();
647 return std::move(data_.
template get<I>());
652 [[nodiscard]]
constexpr typename traits<T>::pointer ptr()
noexcept {
653 if constexpr (!std::is_void_v<T>) {
654 return &data_.
template get<I>();
661 [[nodiscard]]
constexpr typename traits<T>::const_pointer ptr()
const noexcept {
662 if constexpr (!std::is_void_v<T>) {
663 return &data_.
template get<I>();
669 template <size_t I,
typename... Args>
670 constexpr void emplace(Args&&... args) {
671 data_.
template destroy<0>();
672 data_.
template construct<0>(std::forward<Args>(args)...);
675 template <size_t I,
typename... Args>
676 constexpr void uninit_emplace(Args&&... args) {
677 data_.
template construct<0>(std::forward<Args>(args)...);
680 constexpr void swap(variant_impl& other)
noexcept(traits<T>::is_nothrow_swappable) {
681 if constexpr (std::is_void_v<T>) {
683 }
else if constexpr (std::is_lvalue_reference_v<T>) {
684 auto& tmp = other.data_.
template get<0>();
685 other.data_.
template construct<0>(data_.
template get<0>());
686 data_.
template construct<0>(tmp);
689 swap(data_.
template get<0>(), other.data_.
template get<0>());
694 template <
typename T>
695 class variant_impl<std::enable_if_t<is_sumty_empty_v<T>>, T> {
697 SUMTY_NO_UNIQ_ADDR std::remove_const_t<T> data_;
701 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
703 constexpr variant_impl()
noexcept(traits<T>::is_nothrow_default_constructible)
706 constexpr variant_impl(
const variant_impl& other) =
default;
708 constexpr variant_impl(variant_impl&& other)
noexcept(
709 traits<T>::is_nothrow_move_constructible) =
default;
711 template <
typename... Args>
712 constexpr explicit(
sizeof...(Args) == 0)
714 variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace, Args&&... args)
715 : data_(std::forward<Args>(args)...) {}
717 constexpr ~variant_impl()
noexcept(traits<T>::is_nothrow_destructible) =
default;
719 constexpr variant_impl& operator=(
const variant_impl& rhs) =
default;
721 constexpr variant_impl& operator=(variant_impl&& rhs)
noexcept(
722 traits<T>::is_nothrow_move_assignable) =
default;
724 [[nodiscard]]
static constexpr size_t index()
noexcept {
return 0; }
727 [[nodiscard]]
constexpr typename traits<T>::reference get() &
noexcept {
732 [[nodiscard]]
constexpr typename traits<T>::const_reference get()
const&
noexcept {
737 [[nodiscard]]
constexpr typename traits<T>::rvalue_reference get() && {
738 return std::move(data_);
742 [[nodiscard]]
constexpr typename traits<T>::const_rvalue_reference get()
const&& {
743 return std::move(data_);
747 [[nodiscard]]
constexpr typename traits<T>::pointer ptr()
noexcept {
752 [[nodiscard]]
constexpr typename traits<T>::const_pointer ptr()
const noexcept {
756 template <size_t I,
typename... Args>
757 constexpr void emplace(Args&&... args) {
759 ::
new (&data_) T(std::forward<Args>(args)...);
762 template <size_t I,
typename... Args>
763 constexpr void uninit_emplace(Args&&... args) {
764 ::
new (&data_) T(std::forward<Args>(args)...);
767 constexpr void swap(variant_impl& other)
noexcept(traits<T>::is_nothrow_swappable) {
769 swap(data_, other.data_);
774 class variant_impl<
void,
void> {
776 constexpr variant_impl()
noexcept =
default;
779 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept : variant_impl() {}
781 explicit constexpr variant_impl(
782 [[maybe_unused]] std::in_place_index_t<0> inplace)
noexcept {}
784 template <
typename T>
785 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
787 [[maybe_unused]] T&& value)
noexcept {}
789 [[nodiscard]]
static constexpr size_t index()
noexcept {
return 0; }
792 static constexpr void get()
noexcept {}
795 static constexpr void ptr()
noexcept {}
797 static constexpr void swap([[maybe_unused]] variant_impl& other)
noexcept {}
800 template <
typename T>
801 class variant_impl<
void, T&,
void> {
806 variant_impl() =
delete;
809 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept : data_(
nullptr) {}
811 template <
typename U>
812 requires(std::is_convertible_v<U*, T*>)
813 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
817 explicit constexpr variant_impl(
818 [[maybe_unused]] std::in_place_index_t<1> inplace)
noexcept
821 template <
typename U>
822 explicit constexpr variant_impl([[maybe_unused]] std::in_place_index_t<1> inplace,
824 [[maybe_unused]] U&& value)
noexcept
827 [[nodiscard]]
constexpr size_t index()
const noexcept {
828 return static_cast<size_t>(data_ ==
nullptr);
832 [[nodiscard]]
constexpr typename traits<select_t<I, T&,
void>>::const_reference get()
834 if constexpr (I == 0) {
842 [[nodiscard]]
constexpr typename traits<select_t<I, T&,
void>>::const_pointer ptr()
844 if constexpr (I == 0) {
852 constexpr void emplace()
noexcept {
853 static_assert(I == 1,
"no matching constructor for reference");
857 template <size_t I,
typename U>
859 constexpr void emplace([[maybe_unused]] U&& value)
noexcept {
860 if constexpr (I == 0) {
861 static_assert(std::is_lvalue_reference_v<U>,
862 "no matching constructor for reference");
870 constexpr void uninit_emplace()
noexcept {
874 template <size_t I,
typename U>
875 constexpr void uninit_emplace(U&& value)
noexcept {
876 emplace<I>(std::forward<U>(value));
879 constexpr void swap(variant_impl& other)
noexcept { std::swap(data_, other.data_); }
882 template <
typename T>
883 class variant_impl<
void,
void, T&> {
888 constexpr variant_impl()
noexcept =
default;
891 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept : variant_impl() {}
893 explicit constexpr variant_impl(
894 [[maybe_unused]] std::in_place_index_t<0> inplace)
noexcept
897 template <
typename U>
898 explicit constexpr variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
900 [[maybe_unused]] U&& value)
noexcept
903 template <
typename U>
904 requires(std::is_convertible_v<U*, T*>)
905 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<1> inplace,
909 [[nodiscard]]
constexpr size_t index()
const noexcept {
910 return static_cast<size_t>(data_ !=
nullptr);
914 [[nodiscard]]
constexpr typename traits<select_t<I,
void, T&>>::const_reference get()
916 if constexpr (I == 1) {
924 [[nodiscard]]
constexpr typename traits<select_t<I,
void, T&>>::const_pointer ptr()
926 if constexpr (I == 1) {
934 constexpr void emplace()
noexcept {
935 static_assert(I == 0,
"no matching constructor for reference");
939 template <size_t I,
typename U>
941 constexpr void emplace([[maybe_unused]] U&& value)
noexcept {
942 if constexpr (I == 1) {
943 static_assert(std::is_lvalue_reference_v<U>,
944 "no matching constructor for reference");
952 constexpr void uninit_emplace()
noexcept {
956 template <size_t I,
typename U>
957 constexpr void uninit_emplace(U&& value)
noexcept {
958 emplace<I>(std::forward<U>(value));
961 constexpr void swap(variant_impl& other)
noexcept { std::swap(data_, other.data_); }
964 template <
typename T,
typename U>
965 class variant_impl<std::enable_if_t<is_sumty_empty_v<U>>, T&, U> {
968 SUMTY_NO_UNIQ_ADDR U tail_{};
971 variant_impl() =
delete;
974 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept : head_(
nullptr) {}
976 constexpr variant_impl(
const variant_impl& other)
noexcept =
default;
978 constexpr variant_impl(variant_impl&& other)
noexcept =
default;
980 template <
typename V>
981 requires(std::is_convertible_v<V*, T*>)
982 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
986 template <
typename... Args>
987 explicit(
sizeof...(Args) == 0)
989 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<1> inplace,
991 : head_(
nullptr), tail_(std::forward<Args>(args)...) {}
993 constexpr ~variant_impl()
noexcept =
default;
995 constexpr variant_impl& operator=(
const variant_impl& rhs) =
default;
997 constexpr variant_impl& operator=(variant_impl&& rhs)
noexcept =
default;
999 [[nodiscard]]
constexpr size_t index()
const noexcept {
1000 return static_cast<size_t>(head_ ==
nullptr);
1004 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::reference
1006 if constexpr (I == 0) {
1014 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::const_reference get()
1016 if constexpr (I == 0) {
1024 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::rvalue_reference get() && {
1025 if constexpr (I == 0) {
1028 return std::move(tail_);
1033 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::const_rvalue_reference
1035 if constexpr (I == 0) {
1038 return std::move(tail_);
1043 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::pointer ptr()
noexcept {
1044 if constexpr (I == 0) {
1052 [[nodiscard]]
constexpr typename traits<select_t<I, T&, U>>::const_pointer ptr()
1054 if constexpr (I == 0) {
1061 template <size_t I,
typename... Args>
1062 constexpr void emplace(Args&&... args)
noexcept {
1063 if constexpr (I == 0) {
1065 (
true && ... && std::is_lvalue_reference_v<Args>)&&
sizeof...(Args) == 1,
1066 "no matching constructor for reference");
1067 head_ = std::addressof(std::forward<Args>(args)...);
1070 std::construct_at(&tail_, std::forward<Args>(args)...);
1074 template <size_t I,
typename... Args>
1075 constexpr void uninit_emplace(Args&&... args)
noexcept {
1076 if constexpr (I == 0) {
1078 (
true && ... && std::is_lvalue_reference_v<Args>)&&
sizeof...(Args) == 1,
1079 "no matching constructor for reference");
1080 head_ = std::addressof(std::forward<Args>(args)...);
1083 std::construct_at(&tail_, std::forward<Args>(args)...);
1087 constexpr void swap(variant_impl& other)
noexcept(
1088 std::is_nothrow_swappable_v<U> && std::is_nothrow_move_constructible_v<U> &&
1089 std::is_nothrow_destructible_v<U>) {
1090 if (head_ ==
nullptr) {
1091 if (other.head_ !=
nullptr) {
1092 head_ = other.head_;
1093 other.head_ =
nullptr;
1096 if (other.head_ ==
nullptr) {
1097 other.head_ = head_;
1104 template <
typename T,
typename U>
1105 class variant_impl<std::enable_if_t<is_sumty_empty_v<T>>, T, U&> {
1108 SUMTY_NO_UNIQ_ADDR T head_{};
1111 constexpr variant_impl()
noexcept =
default;
1114 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept : tail_(
nullptr) {}
1116 constexpr variant_impl(
const variant_impl& other)
noexcept =
default;
1118 constexpr variant_impl(variant_impl&& other)
noexcept =
default;
1120 template <
typename... Args>
1121 explicit(
sizeof...(Args) == 0)
1123 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
1125 std::construct_at(&head_, std::forward<Args>(args)...);
1128 template <
typename V>
1129 requires(std::is_convertible_v<V*, U*>)
1130 constexpr variant_impl([[maybe_unused]] std::in_place_index_t<1> inplace,
1134 constexpr ~variant_impl()
noexcept =
default;
1136 constexpr variant_impl& operator=(
const variant_impl& rhs) =
default;
1138 constexpr variant_impl& operator=(variant_impl&& rhs)
noexcept =
default;
1140 [[nodiscard]]
constexpr size_t index()
const noexcept {
1141 return static_cast<size_t>(tail_ !=
nullptr);
1145 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::reference
1147 if constexpr (I == 0) {
1155 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::const_reference get()
1157 if constexpr (I == 0) {
1165 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::rvalue_reference get() && {
1166 if constexpr (I == 0) {
1167 return std::move(head_);
1174 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::const_rvalue_reference
1176 if constexpr (I == 0) {
1177 return std::move(head_);
1184 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::pointer ptr()
noexcept {
1185 if constexpr (I == 0) {
1193 [[nodiscard]]
constexpr typename traits<select_t<I, T, U&>>::const_pointer ptr()
1195 if constexpr (I == 0) {
1202 template <size_t I,
typename... Args>
1203 constexpr void emplace(Args&&... args)
noexcept {
1204 if constexpr (I == 1) {
1206 (
true && ... && std::is_lvalue_reference_v<Args>)&&
sizeof...(Args) == 1,
1207 "no matching constructor for reference");
1208 tail_ = std::addressof(std::forward<Args>(args)...);
1211 std::construct_at(&head_, std::forward<Args>(args)...);
1215 template <size_t I,
typename... Args>
1216 constexpr void uninit_emplace(Args&&... args)
noexcept {
1217 if constexpr (I == 1) {
1219 (
true && ... && std::is_lvalue_reference_v<Args>)&&
sizeof...(Args) == 1,
1220 "no matching constructor for reference");
1221 tail_ = std::addressof(std::forward<Args>(args)...);
1224 std::construct_at(&head_.head, std::forward<Args>(args)...);
1228 constexpr void swap(variant_impl& other)
noexcept(
1229 std::is_nothrow_swappable_v<T> && std::is_nothrow_move_constructible_v<T> &&
1230 std::is_nothrow_destructible_v<T>) {
1231 if (tail_ ==
nullptr) {
1232 if (other.tail_ !=
nullptr) {
1233 tail_ = other.tail_;
1234 other.tail_ =
nullptr;
1237 if (other.tail_ ==
nullptr) {
1238 other.tail_ = tail_;
1245 template <
typename V,
typename T>
1246 struct variant_impl_prepend;
1248 template <
typename T,
typename... TN>
1249 struct variant_impl_prepend<variant_impl<
void, TN...>, T> {
1250 using type = variant_impl<
void, T, TN...>;
1253 template <
typename... T>
1254 struct variant_impl_except_last;
1256 template <
typename T>
1257 struct variant_impl_except_last<T> {
1258 using type = variant_impl<
void>;
1261 template <
typename T0,
typename T1,
typename... TN>
1262 struct variant_impl_except_last<T0, T1, TN...> {
1264 typename variant_impl_prepend<
typename variant_impl_except_last<T1, TN...>::type,
1268 template <
typename... T>
1269 using variant_impl_except_last_t =
typename variant_impl_except_last<T...>::type;
1271 template <
typename... T>
1273 std::enable_if_t<std::is_base_of_v<
never, std::remove_cvref_t<last_t<T...>>> &&
1274 (
sizeof...(T) > 1) &&
1275 !std::is_base_of_v<
never, std::remove_cvref_t<first_t<T...>>>>,
1276 T...> : variant_impl_except_last_t<T...> {
1278 using inner_t = variant_impl_except_last_t<T...>;
1279 using never_t = last_t<T...>;
1281 [[nodiscard]]
constexpr inner_t& inner()
noexcept {
1282 return *
static_cast<inner_t*>(
this);
1285 [[nodiscard]]
constexpr const inner_t& inner()
const noexcept {
1286 return *
static_cast<
const inner_t*>(
this);
1291 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
1293 constexpr variant_impl()
noexcept(std::is_nothrow_default_constructible_v<inner_t>) =
1296 constexpr variant_impl(
const variant_impl&) =
default;
1298 constexpr variant_impl(variant_impl&&)
noexcept(
1299 std::is_nothrow_move_constructible_v<inner_t>) =
default;
1301 template <
typename... Args>
1302 constexpr explicit(
sizeof...(Args) == 0)
1304 variant_impl([[maybe_unused]] std::in_place_index_t<
sizeof...(T)> inplace,
1305 Args&&... args)
noexcept {
1306 [[maybe_unused]] never_t value{std::forward<Args>(args)...};
1309 template <size_t I,
typename... Args>
1310 requires(I <
sizeof...(T) - 1)
1311 constexpr explicit(
sizeof...(Args) == 0)
1313 variant_impl(std::in_place_index_t<I> inplace, Args&&... args)
noexcept(
1314 std::is_nothrow_constructible_v<inner_t, std::in_place_index_t<I>, Args&&...>)
1315 : inner_t(inplace, std::forward<Args>(args)...) {}
1317 constexpr ~variant_impl()
noexcept(std::is_nothrow_destructible_v<inner_t>) =
default;
1319 constexpr variant_impl& operator=(
const variant_impl&) =
default;
1321 constexpr variant_impl& operator=(variant_impl&&)
noexcept(
1322 std::is_nothrow_move_assignable_v<inner_t>) =
default;
1324 [[nodiscard]]
constexpr size_t index()
const noexcept {
return inner().index(); }
1328 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::reference get() &
noexcept {
1329 if constexpr (I ==
sizeof...(T) - 1) {
1330 typename traits<never_t>::pointer ptr =
nullptr;
1333 return inner().
template get<I>();
1339 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_reference get()
1341 if constexpr (I ==
sizeof...(T) - 1) {
1342 typename traits<never_t>::const_pointer ptr =
nullptr;
1345 return inner().
template get<I>();
1350 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::rvalue_reference
1353 if constexpr (I ==
sizeof...(T) - 1) {
1354 typename traits<never_t>::pointer ptr =
nullptr;
1355 return std::move(*ptr);
1357 return std::move(inner()).
template get<I>();
1363 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_rvalue_reference get()
1365 if constexpr (I ==
sizeof...(T) - 1) {
1366 typename traits<never_t>::const_pointer ptr =
nullptr;
1367 return std::move(*ptr);
1369 return std::move(inner()).
template get<I>();
1374 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::pointer ptr()
noexcept {
1375 if constexpr (I ==
sizeof...(T) - 1) {
1378 return inner().
template ptr<I>();
1383 [[nodiscard]]
constexpr typename traits<select_t<I, T...>>::const_pointer ptr()
1385 if constexpr (I ==
sizeof...(T) - 1) {
1388 return inner().
template ptr<I>();
1392 template <size_t I,
typename... Args>
1393 constexpr void emplace(Args&&... args) {
1394 if constexpr (I ==
sizeof...(T) - 1) {
1395 [[maybe_unused]] never_t value{std::forward<Args>(args)...};
1397 inner().
template emplace<I>(std::forward<Args>(args)...);
1401 template <size_t I,
typename... Args>
1402 constexpr void uninit_emplace(Args&&... args) {
1403 if constexpr (I ==
sizeof...(T) - 1) {
1404 [[maybe_unused]] never_t value{std::forward<Args>(args)...};
1406 inner().
template uninit_emplace<I>(std::forward<Args>(args)...);
1410 constexpr void swap(variant_impl& other)
noexcept(
1411 std::is_nothrow_swappable_v<inner_t>) {
1412 inner().swap(other.inner());
1416 template <
typename T0,
typename... TN>
1417 class variant_impl<std::enable_if_t<std::is_base_of_v<
never, std::remove_cvref_t<T0>> &&
1418 (
sizeof...(TN) > 0)>,
1420 TN...> : variant_impl<
void, TN...> {
1422 [[nodiscard]]
constexpr variant_impl<
void, TN...>& inner()
noexcept {
1423 return *
static_cast<variant_impl<
void, TN...>*>(
this);
1426 [[nodiscard]]
constexpr const variant_impl<
void, TN...>& inner()
const noexcept {
1427 return *
static_cast<
const variant_impl<
void, TN...>*>(
this);
1431 static inline constexpr bool okie_dokie =
true;
1434 constexpr variant_impl([[maybe_unused]] uninit_t tag)
noexcept {}
1436 constexpr variant_impl()
noexcept(
1437 std::is_nothrow_default_constructible_v<variant_impl<
void, TN...>>) =
default;
1439 constexpr variant_impl(
const variant_impl&) =
default;
1441 constexpr variant_impl(variant_impl&&)
noexcept(
1442 std::is_nothrow_move_constructible_v<variant_impl<
void, TN...>>) =
default;
1444 template <
typename... Args>
1445 constexpr explicit(
sizeof...(Args) == 0)
1447 variant_impl([[maybe_unused]] std::in_place_index_t<0> inplace,
1448 Args&&... args)
noexcept {
1449 [[maybe_unused]] T0 value{std::forward<Args>(args)...};
1452 template <size_t I,
typename... Args>
1455 constexpr explicit(
sizeof...(Args) == 0) variant_impl(
1456 [[maybe_unused]] std::in_place_index_t<I> inplace,
1457 Args&&... args)
noexcept(std::is_nothrow_constructible_v<variant_impl<
void, TN...>,
1458 std::in_place_index_t<I -
1461 : variant_impl<
void, TN...>(std::in_place_index<I - 1>,
1462 std::forward<Args>(args)...) {}
1464 constexpr ~variant_impl()
noexcept(
1465 std::is_nothrow_destructible_v<variant_impl<
void, TN...>>) =
default;
1467 constexpr variant_impl& operator=(
const variant_impl&) =
default;
1469 constexpr variant_impl& operator=(variant_impl&&)
noexcept(
1470 std::is_nothrow_move_assignable_v<variant_impl<
void, TN...>>) =
default;
1472 [[nodiscard]]
constexpr size_t index()
const noexcept {
return inner().index() + 1; }
1475 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::reference
1478 if constexpr (I == 0) {
1479 typename traits<T0>::pointer ptr =
nullptr;
1482 return inner().
template get<I - 1>();
1488 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::const_reference get()
1490 if constexpr (I == 0) {
1491 typename traits<T0>::const_pointer ptr =
nullptr;
1494 return inner().
template get<I - 1>();
1499 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::rvalue_reference
1502 if constexpr (I == 0) {
1503 typename traits<T0>::pointer ptr =
nullptr;
1504 return std::move(*ptr);
1506 return std::move(inner()).
template get<I - 1>();
1511 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::const_rvalue_reference
1513 get()
const&&
noexcept {
1514 if constexpr (I == 0) {
1515 typename traits<T0>::const_pointer ptr =
nullptr;
1516 return std::move(*ptr);
1518 return std::move(inner()).
template get<I - 1>();
1523 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::pointer
1525 if constexpr (I == 0) {
1528 return inner().
template ptr<I - 1>();
1533 [[nodiscard]]
constexpr typename traits<select_t<I, T0, TN...>>::const_pointer ptr()
1535 if constexpr (I == 0) {
1538 return inner().
template ptr<I - 1>();
1542 template <size_t I,
typename... Args>
1543 constexpr void emplace(Args&&... args) {
1544 if constexpr (I == 0) {
1545 [[maybe_unused]] T0 value{std::forward<Args>(args)...};
1547 inner().
template emplace<I - 1>(std::forward<Args>(args)...);
1551 template <size_t I,
typename... Args>
1552 constexpr void uninit_emplace(Args&&... args) {
1553 if constexpr (I == 0) {
1554 [[maybe_unused]] T0 value{std::forward<Args>(args)...};
1556 inner().
template uninit_emplace<I - 1>(std::forward<Args>(args)...);
1560 constexpr void swap(variant_impl& other)
noexcept(
1561 std::is_nothrow_swappable_v<variant_impl<
void, TN...>>) {
1562 inner().swap(other.inner());
1569 #pragma warning(pop)