1a9e65961SEric Fiselier //===----------------------------------------------------------------------===//
2a9e65961SEric Fiselier //
357b08b09SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
457b08b09SChandler Carruth // See https://llvm.org/LICENSE.txt for license information.
557b08b09SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6a9e65961SEric Fiselier //
7a9e65961SEric Fiselier //===----------------------------------------------------------------------===//
8a9e65961SEric Fiselier //
931cbe0f2SLouis Dionne // UNSUPPORTED: c++03, c++11, c++14
10a9e65961SEric Fiselier
112659663eSLouis Dionne // Throwing bad_optional_access is supported starting in macosx10.13
12*c360553cSLouis Dionne // XFAIL: use_system_cxx_lib && target={{.+}}-apple-macosx10.{{9|10|11|12}} && !no-exceptions
13e9c66ad9SMehdi Amini
14a9e65961SEric Fiselier // <optional>
15a9e65961SEric Fiselier
16a9e65961SEric Fiselier // template <class U>
17a9e65961SEric Fiselier // constexpr EXPLICIT optional(U&& u);
18a9e65961SEric Fiselier
19a9e65961SEric Fiselier #include <optional>
20a9e65961SEric Fiselier #include <type_traits>
21a9e65961SEric Fiselier #include <cassert>
22a9e65961SEric Fiselier
23a9e65961SEric Fiselier #include "test_macros.h"
24cc89063bSNico Weber #include "archetypes.h"
25cc89063bSNico Weber #include "test_convertible.h"
26a9e65961SEric Fiselier
27a9e65961SEric Fiselier
28a9e65961SEric Fiselier using std::optional;
29a9e65961SEric Fiselier
30a9e65961SEric Fiselier struct ImplicitThrow
31a9e65961SEric Fiselier {
ImplicitThrowImplicitThrow32a9e65961SEric Fiselier constexpr ImplicitThrow(int x) { if (x != -1) TEST_THROW(6);}
33a9e65961SEric Fiselier };
34a9e65961SEric Fiselier
35a9e65961SEric Fiselier struct ExplicitThrow
36a9e65961SEric Fiselier {
ExplicitThrowExplicitThrow37a9e65961SEric Fiselier constexpr explicit ExplicitThrow(int x) { if (x != -1) TEST_THROW(6);}
38a9e65961SEric Fiselier };
39a9e65961SEric Fiselier
40d3209f93SEric Fiselier struct ImplicitAny {
41d3209f93SEric Fiselier template <class U>
ImplicitAnyImplicitAny42d3209f93SEric Fiselier constexpr ImplicitAny(U&&) {}
43d3209f93SEric Fiselier };
44d3209f93SEric Fiselier
45a9e65961SEric Fiselier
46a9e65961SEric Fiselier template <class To, class From>
implicit_conversion(optional<To> && opt,const From & v)47a9e65961SEric Fiselier constexpr bool implicit_conversion(optional<To>&& opt, const From& v)
48a9e65961SEric Fiselier {
49a9e65961SEric Fiselier using O = optional<To>;
50a9e65961SEric Fiselier static_assert(test_convertible<O, From>(), "");
51a9e65961SEric Fiselier static_assert(!test_convertible<O, void*>(), "");
52a9e65961SEric Fiselier static_assert(!test_convertible<O, From, int>(), "");
53a9e65961SEric Fiselier return opt && *opt == static_cast<To>(v);
54a9e65961SEric Fiselier }
55a9e65961SEric Fiselier
56a9e65961SEric Fiselier template <class To, class Input, class Expect>
explicit_conversion(Input && in,const Expect & v)57a9e65961SEric Fiselier constexpr bool explicit_conversion(Input&& in, const Expect& v)
58a9e65961SEric Fiselier {
59a9e65961SEric Fiselier using O = optional<To>;
60a9e65961SEric Fiselier static_assert(std::is_constructible<O, Input>::value, "");
61a9e65961SEric Fiselier static_assert(!std::is_convertible<Input, O>::value, "");
62a9e65961SEric Fiselier static_assert(!std::is_constructible<O, void*>::value, "");
63a9e65961SEric Fiselier static_assert(!std::is_constructible<O, Input, int>::value, "");
64a9e65961SEric Fiselier optional<To> opt(std::forward<Input>(in));
65a9e65961SEric Fiselier return opt && *opt == static_cast<To>(v);
66a9e65961SEric Fiselier }
67a9e65961SEric Fiselier
test_implicit()68a9e65961SEric Fiselier void test_implicit()
69a9e65961SEric Fiselier {
70a9e65961SEric Fiselier {
71a9e65961SEric Fiselier static_assert(implicit_conversion<long long>(42, 42), "");
72a9e65961SEric Fiselier }
73a9e65961SEric Fiselier {
74a9e65961SEric Fiselier static_assert(implicit_conversion<long double>(3.14, 3.14), "");
75a9e65961SEric Fiselier }
76a9e65961SEric Fiselier {
7764428acfSEric Fiselier int x = 42;
7864428acfSEric Fiselier optional<void* const> o(&x);
7964428acfSEric Fiselier assert(*o == &x);
8064428acfSEric Fiselier }
8164428acfSEric Fiselier {
82a9e65961SEric Fiselier using T = TrivialTestTypes::TestType;
83a9e65961SEric Fiselier static_assert(implicit_conversion<T>(42, 42), "");
84a9e65961SEric Fiselier }
85a9e65961SEric Fiselier {
86a9e65961SEric Fiselier using T = TestTypes::TestType;
87a9e65961SEric Fiselier assert(implicit_conversion<T>(3, T(3)));
88a9e65961SEric Fiselier }
89d3209f93SEric Fiselier {
90d3209f93SEric Fiselier using O = optional<ImplicitAny>;
91d3209f93SEric Fiselier static_assert(!test_convertible<O, std::in_place_t>(), "");
92d3209f93SEric Fiselier static_assert(!test_convertible<O, std::in_place_t&>(), "");
93d3209f93SEric Fiselier static_assert(!test_convertible<O, const std::in_place_t&>(), "");
94d3209f93SEric Fiselier static_assert(!test_convertible<O, std::in_place_t&&>(), "");
95d3209f93SEric Fiselier static_assert(!test_convertible<O, const std::in_place_t&&>(), "");
96d3209f93SEric Fiselier
97d3209f93SEric Fiselier }
98a9e65961SEric Fiselier #ifndef TEST_HAS_NO_EXCEPTIONS
99a9e65961SEric Fiselier {
100a9e65961SEric Fiselier try {
101a9e65961SEric Fiselier using T = ImplicitThrow;
102a9e65961SEric Fiselier optional<T> t = 42;
103a9e65961SEric Fiselier assert(false);
104fd838227SEric Fiselier ((void)t);
105a9e65961SEric Fiselier } catch (int) {
106a9e65961SEric Fiselier }
107a9e65961SEric Fiselier }
108a9e65961SEric Fiselier #endif
109a9e65961SEric Fiselier }
110a9e65961SEric Fiselier
test_explicit()111a9e65961SEric Fiselier void test_explicit() {
112a9e65961SEric Fiselier {
113a9e65961SEric Fiselier using T = ExplicitTrivialTestTypes::TestType;
114a9e65961SEric Fiselier static_assert(explicit_conversion<T>(42, 42), "");
115a9e65961SEric Fiselier }
116a9e65961SEric Fiselier {
117a9e65961SEric Fiselier using T = ExplicitConstexprTestTypes::TestType;
118a9e65961SEric Fiselier static_assert(explicit_conversion<T>(42, 42), "");
119a9e65961SEric Fiselier static_assert(!std::is_convertible<int, T>::value, "");
120a9e65961SEric Fiselier }
121a9e65961SEric Fiselier {
122a9e65961SEric Fiselier using T = ExplicitTestTypes::TestType;
123a9e65961SEric Fiselier T::reset();
124a9e65961SEric Fiselier {
125a9e65961SEric Fiselier assert(explicit_conversion<T>(42, 42));
126a9e65961SEric Fiselier assert(T::alive == 0);
127a9e65961SEric Fiselier }
128a9e65961SEric Fiselier T::reset();
129a9e65961SEric Fiselier {
130a9e65961SEric Fiselier optional<T> t(42);
131a9e65961SEric Fiselier assert(T::alive == 1);
132a9e65961SEric Fiselier assert(T::value_constructed == 1);
133a9e65961SEric Fiselier assert(T::move_constructed == 0);
134a9e65961SEric Fiselier assert(T::copy_constructed == 0);
135a9e65961SEric Fiselier assert(t.value().value == 42);
136a9e65961SEric Fiselier }
137a9e65961SEric Fiselier assert(T::alive == 0);
138a9e65961SEric Fiselier }
139a9e65961SEric Fiselier #ifndef TEST_HAS_NO_EXCEPTIONS
140a9e65961SEric Fiselier {
141a9e65961SEric Fiselier try {
142a9e65961SEric Fiselier using T = ExplicitThrow;
143a9e65961SEric Fiselier optional<T> t(42);
144a9e65961SEric Fiselier assert(false);
145a9e65961SEric Fiselier } catch (int) {
146a9e65961SEric Fiselier }
147a9e65961SEric Fiselier }
148a9e65961SEric Fiselier #endif
149a9e65961SEric Fiselier }
150a9e65961SEric Fiselier
main(int,char **)1512df59c50SJF Bastien int main(int, char**) {
152a9e65961SEric Fiselier test_implicit();
153a9e65961SEric Fiselier test_explicit();
1542df59c50SJF Bastien
1552df59c50SJF Bastien return 0;
156a9e65961SEric Fiselier }
157