3 Copyright (C) 2017-2020 Free Software Foundation, Inc.
5 This file is part of GDB.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20 #ifndef COMMON_GDB_OPTIONAL_H
21 #define COMMON_GDB_OPTIONAL_H
23 #include "gdbsupport/traits.h"
30 explicit in_place_t () = default;
33 constexpr gdb::in_place_t in_place
{};
35 /* This class attempts to be a compatible subset of std::optional,
36 which is slated to be available in C++17. This class optionally
37 holds an object of some type -- by default it is constructed not
38 holding an object, but later the object can be "emplaced". This is
39 similar to using std::unique_ptr, but in-object allocation is
42 Unlike std::optional, we currently only support copy/move
43 construction/assignment of an optional<T> from either exactly
44 optional<T> or T. I.e., we don't support copy/move
45 construction/assignment from optional<U> or U, when U is a type
46 convertible to T. Making that work depending on the definitions of
47 T and U is somewhat complicated, and currently the users of this
48 class don't need it. */
59 template<typename
... Args
>
60 constexpr optional (in_place_t
, Args
&&... args
)
61 : m_item (std::forward
<Args
> (args
)...),
68 /* Copy and move constructors. */
70 optional (const optional
&other
)
72 if (other
.m_instantiated
)
73 this->emplace (other
.get ());
76 optional (optional
&&other
)
77 noexcept(std::is_nothrow_move_constructible
<T
> ())
79 if (other
.m_instantiated
)
80 this->emplace (std::move (other
.get ()));
83 constexpr optional (const T
&other
)
88 constexpr optional (T
&&other
)
89 noexcept (std::is_nothrow_move_constructible
<T
> ())
90 : m_item (std::move (other
)),
94 /* Assignment operators. */
97 operator= (const optional
&other
)
99 if (m_instantiated
&& other
.m_instantiated
)
100 this->get () = other
.get ();
103 if (other
.m_instantiated
)
104 this->emplace (other
.get ());
113 operator= (optional
&&other
)
114 noexcept (And
<std::is_nothrow_move_constructible
<T
>,
115 std::is_nothrow_move_assignable
<T
>> ())
117 if (m_instantiated
&& other
.m_instantiated
)
118 this->get () = std::move (other
.get ());
121 if (other
.m_instantiated
)
122 this->emplace (std::move (other
.get ()));
130 operator= (const T
&other
)
133 this->get () = other
;
135 this->emplace (other
);
140 operator= (T
&&other
)
141 noexcept (And
<std::is_nothrow_move_constructible
<T
>,
142 std::is_nothrow_move_assignable
<T
>> ())
145 this->get () = std::move (other
);
147 this->emplace (std::move (other
));
151 template<typename
... Args
>
152 T
&emplace (Args
&&... args
)
155 new (&m_item
) T (std::forward
<Args
>(args
)...);
156 m_instantiated
= true;
161 constexpr const T
*operator-> () const
162 { return std::addressof (this->get ()); }
165 { return std::addressof (this->get ()); }
167 constexpr const T
&operator* () const &
168 { return this->get (); }
171 { return this->get (); }
174 { return std::move (this->get ()); }
176 constexpr const T
&&operator* () const &&
177 { return std::move (this->get ()); }
179 constexpr explicit operator bool () const noexcept
180 { return m_instantiated
; }
182 constexpr bool has_value () const noexcept
183 { return m_instantiated
; }
185 /* 'reset' is a 'safe' operation with no precondition. */
186 void reset () noexcept
194 /* Destroy the object. */
197 gdb_assert (m_instantiated
);
198 m_instantiated
= false;
202 /* The get operations have m_instantiated as a precondition. */
203 T
&get () noexcept
{ return m_item
; }
204 constexpr const T
&get () const noexcept
{ return m_item
; }
213 /* True if the object was ever emplaced. */
214 bool m_instantiated
= false;
219 #endif /* COMMON_GDB_OPTIONAL_H */