17 #include <bsoncxx/config/prelude.hpp>
19 #if defined(BSONCXX_POLY_USE_MNMLSTC)
21 #include <core/optional.hpp>
27 using ::core::in_place;
28 using ::core::in_place_t;
29 using ::core::make_optional;
30 using ::core::nullopt;
31 using ::core::nullopt_t;
32 using ::core::optional;
38 #elif defined(BSONCXX_POLY_USE_BOOST)
40 #include <boost/none.hpp>
41 #include <boost/optional/optional.hpp>
42 #include <boost/optional/optional_io.hpp>
48 #if BOOST_VERSION >= 106300
49 using in_place_t = ::boost::in_place_init_t;
50 const in_place_t in_place{::boost::in_place_init};
53 using ::boost::optional;
54 using nullopt_t = ::boost::none_t;
56 const nullopt_t nullopt{::boost::none};
57 using ::boost::make_optional;
63 #elif defined(BSONCXX_POLY_USE_STD)
71 using ::std::in_place;
72 using ::std::in_place_t;
73 using ::std::make_optional;
75 using ::std::nullopt_t;
76 using ::std::optional;
82 #elif defined(BSONCXX_POLY_USE_IMPLS)
87 #include <initializer_list>
90 #include <type_traits>
93 #include <bsoncxx/stdx/operators.hpp>
94 #include <bsoncxx/stdx/type_traits.hpp>
109 template <
typename T>
116 class bad_optional_access :
public std::exception {
118 const char* what() const noexcept
override {
119 return "bad_optional_access()";
124 explicit constexpr nullopt_t(std::nullptr_t) noexcept {}
127 static constexpr nullopt_t nullopt{0};
129 static constexpr
struct in_place_t {
135 [[noreturn]]
inline void terminate_disengaged_optional(
const char* what) noexcept {
136 (void)std::fprintf(stderr,
"%s: Invalid attempted use of disengaged optional<T>\n", what);
140 [[noreturn]]
inline void throw_bad_optional() {
141 throw bad_optional_access();
144 template <
typename T>
145 struct optional_base_class;
148 template <
typename T>
149 std::true_type not_an_optional_f(
const T&);
152 template <
typename T>
153 std::false_type not_an_optional_f(
const optional<T>&);
156 template <
typename T>
157 struct not_an_optional : decltype(not_an_optional_f(std::declval<T const&>())) {};
159 template <
typename T,
typename Ucvr,
typename U>
160 struct enable_opt_conversion
161 : bsoncxx::detail::conjunction<
162 std::is_constructible<T, Ucvr>,
163 bsoncxx::detail::disjunction<
164 std::is_same<T, bool>,
165 bsoncxx::detail::negation<
166 bsoncxx::detail::conjunction<std::is_constructible<T, optional<U>&>,
167 std::is_constructible<T, optional<U> const&>,
168 std::is_constructible<T, optional<U>&&>,
169 std::is_constructible<T, optional<U> const&&>,
170 std::is_convertible<optional<U>&, T>,
171 std::is_convertible<optional<U> const&, T>,
172 std::is_convertible<optional<U>&&, T>,
173 std::is_convertible<optional<U> const&&, T>>>>> {};
175 template <
typename From,
typename To>
176 struct enable_opt_value_conversion
177 : bsoncxx::detail::conjunction<
178 std::is_constructible<To, From&&>,
179 bsoncxx::detail::negation<bsoncxx::detail::is_alike<From, in_place_t>>,
180 bsoncxx::detail::negation<bsoncxx::detail::is_alike<From, optional<To>>>,
181 bsoncxx::detail::disjunction<
182 bsoncxx::detail::negation<bsoncxx::detail::is_alike<To, bool>>,
183 detail::not_an_optional<bsoncxx::detail::remove_cvref_t<From>>>> {};
187 template <
typename T>
188 class optional : bsoncxx::detail::equality_operators,
189 bsoncxx::detail::ordering_operators,
190 public detail::optional_base_class<T>
::type {
193 using value_type = T;
195 using reference = bsoncxx::detail::add_lvalue_reference_t<T>;
197 using const_reference =
198 bsoncxx::detail::add_lvalue_reference_t<bsoncxx::detail::add_const_t<T>>;
200 using rvalue_reference = bsoncxx::detail::add_rvalue_reference_t<T>;
202 using const_rvalue_reference =
203 bsoncxx::detail::add_rvalue_reference_t<bsoncxx::detail::add_const_t<T>>;
205 using pointer = bsoncxx::detail::add_pointer_t<T>;
207 using const_pointer = bsoncxx::detail::add_pointer_t<const T>;
210 optional() =
default;
211 constexpr optional(nullopt_t) noexcept {}
214 optional(
const optional&) =
default;
215 optional(optional&&) =
default;
217 optional& operator=(
const optional&) =
default;
218 optional& operator=(optional&&) =
default;
219 ~optional() =
default;
222 template <
typename... Args>
223 bsoncxx_cxx14_constexpr
explicit optional(in_place_t, Args&&... args) noexcept(
224 noexcept(T(BSONCXX_FWD(args)...))) {
225 this->emplace(BSONCXX_FWD(args)...);
228 template <
typename U,
typename... Args>
229 bsoncxx_cxx14_constexpr
explicit optional(
231 std::initializer_list<U> il,
232 Args&&... args) noexcept(noexcept(T(il, BSONCXX_FWD(args)...))) {
233 this->emplace(il, BSONCXX_FWD(args)...);
240 bsoncxx::detail::requires_t<int,
241 detail::enable_opt_value_conversion<U&&, T>,
242 bsoncxx::detail::negation<std::is_convertible<U&&, T>>> = 0>
243 bsoncxx_cxx14_constexpr
explicit optional(U&& arg) noexcept(
244 std::is_nothrow_constructible<T, U&&>::value)
245 : optional(in_place, BSONCXX_FWD(arg)) {}
249 template <
typename U = T,
250 bsoncxx::detail::requires_t<int,
251 detail::enable_opt_value_conversion<U&&, T>,
252 std::is_convertible<U&&, T>> = 0>
253 bsoncxx_cxx14_constexpr optional(U&& arg) noexcept(std::is_nothrow_constructible<T, U&&>::value)
254 : optional(in_place, BSONCXX_FWD(arg)) {}
256 template <
typename U,
257 bsoncxx::detail::requires_t<
259 detail::enable_opt_conversion<T, const U&, U>,
260 bsoncxx::detail::negation<std::is_convertible<U const&, T>>> = 0>
261 bsoncxx_cxx14_constexpr
explicit optional(optional<U>
const& other) noexcept(
262 std::is_nothrow_constructible<T, bsoncxx::detail::add_lvalue_reference_t<const U>>::value) {
263 if (other.has_value()) {
264 this->emplace(*other);
268 template <
typename U,
269 bsoncxx::detail::requires_t<int,
270 detail::enable_opt_conversion<T, const U&, U>,
271 std::is_convertible<U const&, T>> = 0>
272 bsoncxx_cxx14_constexpr optional(optional<U>
const& other) noexcept(
273 std::is_nothrow_constructible<T, bsoncxx::detail::add_lvalue_reference_t<const U>>::value) {
274 if (other.has_value()) {
275 this->emplace(*other);
281 bsoncxx::detail::requires_t<int,
282 detail::enable_opt_conversion<T, U&&, U>,
283 bsoncxx::detail::negation<std::is_convertible<U&&, T>>> = 0>
284 bsoncxx_cxx14_constexpr
explicit optional(optional<U>&& other) noexcept(
285 std::is_nothrow_constructible<T, bsoncxx::detail::add_lvalue_reference_t<U&&>>::value) {
286 if (other.has_value()) {
287 this->emplace(*BSONCXX_FWD(other));
291 template <
typename U,
292 bsoncxx::detail::requires_t<int,
293 detail::enable_opt_conversion<T, U&&, U>,
294 std::is_convertible<U&&, T>> = 0>
295 bsoncxx_cxx14_constexpr optional(optional<U>&& other) noexcept(
296 std::is_nothrow_constructible<T, bsoncxx::detail::add_lvalue_reference_t<U&&>>::value) {
297 if (other.has_value()) {
298 this->emplace(*BSONCXX_FWD(other));
302 constexpr
bool has_value() const noexcept {
303 return this->_has_value;
305 constexpr
explicit operator bool() const noexcept {
306 return this->has_value();
310 bsoncxx_cxx14_constexpr reference operator*() & noexcept {
311 _assert_has_value(
"operator*() &");
312 return this->_storage.value;
314 bsoncxx_cxx14_constexpr const_reference operator*() const& noexcept {
315 _assert_has_value(
"operator*() const&");
316 return this->_storage.value;
318 bsoncxx_cxx14_constexpr rvalue_reference operator*() && noexcept {
319 _assert_has_value(
"operator*() &&");
320 return static_cast<rvalue_reference
>(**this);
322 bsoncxx_cxx14_constexpr const_rvalue_reference operator*() const&& noexcept {
323 _assert_has_value(
"operator*() const&&");
324 return static_cast<const_rvalue_reference
>(**this);
328 bsoncxx_cxx14_constexpr pointer operator->() noexcept {
329 _assert_has_value(
"operator->()");
330 return std::addressof(**
this);
332 bsoncxx_cxx14_constexpr const_pointer operator->() const noexcept {
333 _assert_has_value(
"operator->() const");
334 return std::addressof(**
this);
338 bsoncxx_cxx14_constexpr reference value() & {
342 bsoncxx_cxx14_constexpr const_reference value() const& {
346 bsoncxx_cxx14_constexpr rvalue_reference value() && {
348 return static_cast<rvalue_reference
>(**this);
350 bsoncxx_cxx14_constexpr const_rvalue_reference value() const&& {
352 return static_cast<const_rvalue_reference
>(**this);
356 template <
typename U>
357 bsoncxx_cxx14_constexpr value_type value_or(U&& dflt)
const& {
361 return static_cast<value_type
>(BSONCXX_FWD(dflt));
365 template <
typename U>
366 bsoncxx_cxx14_constexpr value_type value_or(U&& dflt) && {
368 return *std::move(*
this);
370 return static_cast<value_type
>(BSONCXX_FWD(dflt));
375 bsoncxx_cxx14_constexpr
void _assert_has_value(
const char* msg)
const noexcept {
376 if (!this->has_value()) {
377 detail::terminate_disengaged_optional(msg);
381 bsoncxx_cxx14_constexpr
void _throw_if_empty()
const {
382 if (!this->has_value()) {
383 detail::throw_bad_optional();
394 template <
typename T>
395 bsoncxx_cxx14_constexpr optional<bsoncxx::detail::decay_t<T>> make_optional(T&& value) noexcept(
396 std::is_nothrow_constructible<bsoncxx::detail::decay_t<T>, T&&>::value) {
397 return optional<bsoncxx::detail::decay_t<T>>(BSONCXX_FWD(value));
407 template <
typename T,
typename... Args>
408 bsoncxx_cxx14_constexpr optional<T> make_optional(Args&&... args) noexcept(
409 std::is_nothrow_constructible<T, Args&&...>::value) {
410 return optional<T>(in_place, BSONCXX_FWD(args)...);
417 template <
typename T,
typename U,
typename... Args>
418 bsoncxx_cxx14_constexpr optional<T>
419 make_optional(std::initializer_list<U> il, Args&&... args) noexcept(
420 std::is_nothrow_constructible<T, std::initializer_list<U>, Args&&...>::value) {
421 return optional<T>(in_place, il, BSONCXX_FWD(args)...);
429 template <typename T, bool = std::is_trivially_destructible<T>::value>
437 storage_for() noexcept : nothing(0) {}
442 storage_for(
const storage_for&) =
delete;
443 storage_for& operator=(
const storage_for&) =
delete;
446 template <
typename T>
447 union storage_for<T, true > {
450 storage_for() noexcept : nothing(0) {}
451 storage_for(
const storage_for&) =
delete;
452 storage_for& operator=(
const storage_for&) =
delete;
456 enum copymove_classification {
463 template <
typename T,
464 bool CanCopy = std::is_copy_constructible<T>::value,
465 bool CanMove = std::is_move_constructible<T>::value>
466 constexpr copymove_classification classify_construct() {
467 return CanCopy ? copyable : CanMove ? movable : immobile;
471 template <
typename T,
472 bool CanCopy = std::is_copy_assignable<T>::value,
473 bool CanMove = std::is_move_assignable<T>::value>
474 constexpr copymove_classification classify_assignment() {
475 return CanCopy ? copyable : CanMove ? movable : immobile;
483 template <
typename T>
484 class optional_common_base;
487 template <typename T, copymove_classification = classify_construct<T>()>
488 struct optional_construct_base;
491 template <typename T, copymove_classification = classify_assignment<T>()>
492 struct optional_assign_base;
494 template <
bool TrivialDestruct>
495 struct optional_destruct_helper;
497 template <
typename T>
498 using optional_destruct_base =
499 typename optional_destruct_helper<std::is_trivially_destructible<T>::value>::template base<T>;
501 template <
typename T>
502 struct optional_assign_base<T, copyable> : optional_construct_base<T> {};
504 template <
typename T>
505 struct optional_assign_base<T, movable> : optional_construct_base<T> {
507 optional_assign_base() =
default;
508 optional_assign_base(optional_assign_base
const&) =
default;
509 optional_assign_base(optional_assign_base&&) =
default;
510 ~optional_assign_base() =
default;
513 bsoncxx_cxx14_constexpr optional_assign_base& operator=(
const optional_assign_base&) =
delete;
515 bsoncxx_cxx14_constexpr optional_assign_base& operator=(optional_assign_base&& other) =
default;
518 template <
typename T>
519 struct optional_assign_base<T, immobile> : optional_construct_base<T> {
520 optional_assign_base() =
default;
521 optional_assign_base(optional_assign_base
const&) =
default;
522 optional_assign_base(optional_assign_base&&) =
default;
523 ~optional_assign_base() =
default;
526 optional_assign_base& operator=(
const optional_assign_base&) =
delete;
527 optional_assign_base& operator=(optional_assign_base&&) =
delete;
530 template <
typename T>
531 struct optional_construct_base<T, copyable> : optional_destruct_base<T> {};
533 template <
typename T>
534 struct optional_construct_base<T, movable> : optional_destruct_base<T> {
535 optional_construct_base() =
default;
537 optional_construct_base(
const optional_construct_base&) =
delete;
538 optional_construct_base(optional_construct_base&& other) =
default;
539 optional_construct_base& operator=(
const optional_construct_base&) =
default;
540 optional_construct_base& operator=(optional_construct_base&&) =
default;
543 template <
typename T>
544 struct optional_construct_base<T, immobile> : optional_destruct_base<T> {
545 optional_construct_base() =
default;
546 optional_construct_base(
const optional_construct_base&) =
delete;
547 optional_construct_base& operator=(
const optional_construct_base&) =
default;
548 optional_construct_base& operator=(optional_construct_base&&) =
default;
552 struct optional_destruct_helper<false > {
553 template <
typename T>
554 struct base : optional_common_base<T> {
557 base(base
const&) =
default;
558 base(base&&) =
default;
559 base& operator=(
const base&) =
default;
560 base& operator=(base&&) =
default;
569 struct optional_destruct_helper<true > {
571 template <
typename T>
572 using base = optional_common_base<T>;
576 struct optional_operators_base {
577 template <
typename T,
typename U>
578 friend bsoncxx_cxx14_constexpr
auto tag_invoke(bsoncxx::detail::equal_to,
579 optional<T>
const& left,
580 optional<U>
const& right) noexcept
581 -> bsoncxx::detail::requires_t<bool, bsoncxx::detail::is_equality_comparable<T, U>> {
582 if (left.has_value() != right.has_value()) {
585 return !left.has_value() || *left == *right;
588 template <
typename T,
typename U>
589 friend constexpr
auto tag_invoke(bsoncxx::detail::equal_to,
590 optional<T>
const& left,
591 U
const& right) noexcept -> bsoncxx::detail::
592 requires_t<bool, not_an_optional<U>, bsoncxx::detail::is_equality_comparable<T, U>> {
593 return left.has_value() && *left == right;
596 template <
typename T>
597 friend constexpr
bool tag_invoke(bsoncxx::detail::equal_to,
598 optional<T>
const& opt,
599 nullopt_t) noexcept {
600 return !opt.has_value();
603 template <
typename T,
typename U>
604 bsoncxx_cxx14_constexpr
friend auto tag_invoke(bsoncxx::detail::compare_three_way compare,
605 optional<T>
const& left,
606 optional<U>
const& right)
607 -> bsoncxx::detail::requires_t<bsoncxx::detail::strong_ordering,
608 bsoncxx::detail::is_totally_ordered_with<T, U>> {
609 if (left.has_value()) {
610 if (right.has_value()) {
611 return compare(*left, *right);
614 return bsoncxx::detail::strong_ordering::greater;
617 if (right.has_value()) {
619 return bsoncxx::detail::strong_ordering::less;
622 return bsoncxx::detail::strong_ordering::equal;
627 template <
typename T,
typename U>
628 bsoncxx_cxx14_constexpr
friend auto tag_invoke(bsoncxx::detail::compare_three_way compare,
629 optional<T>
const& left,
631 -> bsoncxx::detail::requires_t<bsoncxx::detail::strong_ordering,
633 bsoncxx::detail::is_totally_ordered_with<T, U>> {
634 if (left.has_value()) {
635 return compare(*left, right);
638 return bsoncxx::detail::strong_ordering::less;
641 template <
typename T>
642 constexpr
friend bsoncxx::detail::strong_ordering tag_invoke(
643 bsoncxx::detail::compare_three_way compare, optional<T>
const& left, nullopt_t) {
644 return compare(left.has_value(),
false);
649 template <typename T, bool IsSwappable = bsoncxx::detail::is_swappable<T>::value>
650 struct optional_swap_mixin {};
652 template <
typename T>
653 struct optional_swap_mixin<T, true> {
654 bsoncxx_cxx14_constexpr
friend void swap(optional<T>& left, optional<T>& right) noexcept(
655 std::is_nothrow_move_constructible<T>::value&&
656 bsoncxx::detail::is_nothrow_swappable<T>::value) {
662 template <
typename T>
663 class optional_common_base : optional_operators_base, optional_swap_mixin<T> {
664 using storage_type = detail::storage_for<bsoncxx::detail::remove_const_t<T>>;
667 optional_common_base() =
default;
668 ~optional_common_base() =
default;
670 optional_common_base(
const optional_common_base& other) noexcept(
671 std::is_nothrow_copy_constructible<T>::value) {
672 if (other._has_value) {
673 this->emplace(other._storage.value);
677 optional_common_base(optional_common_base&& other) noexcept(
678 std::is_nothrow_move_constructible<T>::value) {
679 if (other._has_value) {
680 this->_emplace_construct_anew(std::move(other)._storage.value);
684 optional_common_base& operator=(
const optional_common_base& other) noexcept(
685 std::is_nothrow_copy_assignable<T>::value) {
686 this->_assign(BSONCXX_FWD(other));
690 optional_common_base& operator=(optional_common_base&& other) noexcept(
691 std::is_nothrow_move_assignable<T>::value) {
692 this->_assign(BSONCXX_FWD(other));
700 void reset() noexcept {
701 if (this->_has_value) {
702 this->_storage.value.~T();
704 this->_has_value =
false;
712 template <
typename... Args>
713 T& emplace(Args&&... args) {
715 this->_emplace_construct_anew(BSONCXX_FWD(args)...);
716 return this->_storage.value;
724 template <
typename U,
typename... Args>
725 T& emplace(std::initializer_list<U> il, Args&&... args) {
727 this->_emplace_construct_anew(il, BSONCXX_FWD(args)...);
728 return this->_storage.value;
735 bsoncxx_cxx14_constexpr
void swap(optional_common_base& other) noexcept(
736 std::is_nothrow_move_constructible<T>::value&&
737 bsoncxx::detail::is_nothrow_swappable<T>::value) {
738 if (other._has_value) {
739 if (this->_has_value) {
742 swap(this->_storage.value, other._storage.value);
745 this->emplace(std::move(other._storage.value));
748 }
else if (this->_has_value) {
749 other.emplace(std::move(this->_storage.value));
758 storage_type _storage;
759 bool _has_value =
false;
766 template <
typename... Args>
767 void _emplace_construct_anew(Args&&... args) noexcept(
768 std::is_nothrow_constructible<T, Args&&...>::value) {
769 new (std::addressof(this->_storage.value)) T(BSONCXX_FWD(args)...);
770 this->_has_value =
true;
777 template <
typename U>
778 void _assign(U&& other_storage) {
779 if (other_storage._has_value) {
781 if (this->_has_value) {
783 this->_storage.value = BSONCXX_FWD(other_storage)._storage.value;
786 this->_emplace_construct_anew(BSONCXX_FWD(other_storage)._storage.value);
795 template <
typename T>
796 struct optional_base_class {
797 using type = optional_assign_base<T>;
800 template <
typename T,
802 std::is_default_constructible<std::hash<bsoncxx::detail::remove_const_t<T>>>::value>
803 struct optional_hash;
807 template <
typename T>
808 struct optional_hash<T, false> {
809 optional_hash() =
delete;
810 optional_hash(
const optional_hash&) =
delete;
813 template <
typename T>
814 struct optional_hash<T, true> {
815 using Td = bsoncxx::detail::remove_const_t<T>;
816 constexpr std::size_t operator()(
const optional<T>& opt)
const
817 noexcept(noexcept(std::hash<Td>()(std::declval<Td const&>()))) {
818 return opt.has_value() ? std::hash<Td>()(*opt)
819 : std::hash<void*>()(
nullptr);
833 template <
typename T>
834 struct hash<
bsoncxx::v_noabi::stdx::optional<T>>
835 : bsoncxx::v_noabi::stdx::detail::optional_hash<T> {};
840 #error "Cannot find a valid polyfill for optional"
843 #include <bsoncxx/config/postlude.hpp>
849 #if !defined(BOOST_VERSION) || BOOST_VERSION >= 106300
850 using ::bsoncxx::v_noabi::stdx::in_place;
851 using ::bsoncxx::v_noabi::stdx::in_place_t;
854 using ::bsoncxx::v_noabi::stdx::make_optional;
855 using ::bsoncxx::v_noabi::stdx::nullopt;
856 using ::bsoncxx::v_noabi::stdx::nullopt_t;
857 using ::bsoncxx::v_noabi::stdx::optional;
type
An enumeration of each BSON type.
Definition: types.hpp:48
The top-level namespace for bsoncxx library entities.
Definition: element-fwd.hpp:19