Use GNU make pattern rule in ld Makefile
[binutils-gdb.git] / gdbsupport / gdb_optional.h
blob9b7b7b2f7f4f63a8d2ccbc7c5bb94cb9e8f53a21
1 /* An optional object.
3 Copyright (C) 2017-2023 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"
25 namespace gdb
28 struct in_place_t
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
40 guaranteed.
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. */
50 template<typename T>
51 class optional
53 public:
55 constexpr optional ()
56 : m_dummy ()
59 template<typename... Args>
60 constexpr optional (in_place_t, Args &&... args)
61 : m_item (std::forward<Args> (args)...),
62 m_instantiated (true)
65 ~optional ()
66 { this->reset (); }
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)
84 : m_item (other),
85 m_instantiated (true)
88 constexpr optional (T &&other)
89 noexcept (std::is_nothrow_move_constructible<T> ())
90 : m_item (std::move (other)),
91 m_instantiated (true)
94 /* Assignment operators. */
96 optional &
97 operator= (const optional &other)
99 if (m_instantiated && other.m_instantiated)
100 this->get () = other.get ();
101 else
103 if (other.m_instantiated)
104 this->emplace (other.get ());
105 else
106 this->reset ();
109 return *this;
112 optional &
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 ());
119 else
121 if (other.m_instantiated)
122 this->emplace (std::move (other.get ()));
123 else
124 this->reset ();
126 return *this;
129 optional &
130 operator= (const T &other)
132 if (m_instantiated)
133 this->get () = other;
134 else
135 this->emplace (other);
136 return *this;
139 optional &
140 operator= (T &&other)
141 noexcept (And<std::is_nothrow_move_constructible<T>,
142 std::is_nothrow_move_assignable<T>> ())
144 if (m_instantiated)
145 this->get () = std::move (other);
146 else
147 this->emplace (std::move (other));
148 return *this;
151 template<typename... Args>
152 T &emplace (Args &&... args)
154 this->reset ();
155 new (&m_item) T (std::forward<Args>(args)...);
156 m_instantiated = true;
157 return this->get ();
160 /* Observers. */
161 constexpr const T *operator-> () const
162 { return std::addressof (this->get ()); }
164 T *operator-> ()
165 { return std::addressof (this->get ()); }
167 constexpr const T &operator* () const &
168 { return this->get (); }
170 T &operator* () &
171 { return this->get (); }
173 T &&operator* () &&
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
188 if (m_instantiated)
189 this->destroy ();
192 private:
194 /* Destroy the object. */
195 void destroy ()
197 gdb_assert (m_instantiated);
198 m_instantiated = false;
199 m_item.~T ();
202 /* The get operations have m_instantiated as a precondition. */
203 T &get () noexcept
205 #if defined(_GLIBCXX_DEBUG) && __cplusplus >= 201402L
206 gdb_assert (this->has_value ());
207 #endif
208 return m_item;
210 constexpr const T &get () const noexcept
212 #if defined(_GLIBCXX_DEBUG) && __cplusplus >= 201402L
213 gdb_assert (this->has_value ());
214 #endif
215 return m_item;
218 /* The object. */
219 union
221 struct { } m_dummy;
222 T m_item;
223 volatile char dont_use; /* Silences -Wmaybe-uninitialized warning, see
224 PR gcc/80635. */
227 /* True if the object was ever emplaced. */
228 bool m_instantiated = false;
233 #endif /* COMMON_GDB_OPTIONAL_H */