1// <experimental/memory> -*- C++ -*-
3// Copyright (C) 2015-2023 Free Software Foundation, Inc.
5// This file is part of the GNU ISO C++ Library. This library is free
6// software; you can redistribute it and/or modify it under the
7// terms of the GNU General Public License as published by the
8// Free Software Foundation; either version 3, or (at your option)
11// This library is distributed in the hope that it will be useful,
12// but WITHOUT ANY WARRANTY; without even the implied warranty of
13// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14// GNU General Public License for more details.
16// Under Section 7 of GPL version 3, you are granted additional
17// permissions described in the GCC Runtime Library Exception, version
18// 3.1, as published by the Free Software Foundation.
20// You should have received a copy of the GNU General Public License and
21// a copy of the GCC Runtime Library Exception along with this program;
22// see the files COPYING3 and COPYING.RUNTIME respectively. If not, see
23// <http://www.gnu.org/licenses/>.
25/** @file experimental/memory
26 * This is a TS C++ Library header.
31// N4336 Working Draft, C++ Extensions for Library Fundamentals, Version 2
34#ifndef _GLIBCXX_EXPERIMENTAL_MEMORY
35#define _GLIBCXX_EXPERIMENTAL_MEMORY 1
37#pragma GCC system_header
39#include <bits/requires_hosted.h> // experimental is currently omitted
41#if __cplusplus >= 201402L
45#include <experimental/bits/shared_ptr.h>
46#include <bits/functional_hash.h>
48namespace std _GLIBCXX_VISIBILITY(default)
50_GLIBCXX_BEGIN_NAMESPACE_VERSION
54inline namespace fundamentals_v2
56#define __cpp_lib_experimental_observer_ptr 201411
58 template <typename _Tp>
62 // publish our template parameter and variations thereof
63 using element_type = _Tp;
64 using __pointer = add_pointer_t<_Tp>; // exposition-only
65 using __reference = add_lvalue_reference_t<_Tp>; // exposition-only
67 // 3.2.2, observer_ptr constructors
69 constexpr observer_ptr() noexcept
73 // pointer-accepting c'tors
74 constexpr observer_ptr(nullptr_t) noexcept
78 constexpr explicit observer_ptr(__pointer __p) noexcept
82 // copying c'tors (in addition to compiler-generated copy c'tor)
83 template <typename _Up,
84 typename = typename enable_if<
85 is_convertible<typename add_pointer<_Up>::type, __pointer
88 constexpr observer_ptr(observer_ptr<_Up> __p) noexcept
93 // 3.2.3, observer_ptr observers
100 constexpr __reference
107 operator->() const noexcept
112 constexpr explicit operator bool() const noexcept
114 return get() != nullptr;
117 // 3.2.4, observer_ptr conversions
118 constexpr explicit operator __pointer() const noexcept
123 // 3.2.5, observer_ptr modifiers
127 __pointer __tmp = get();
133 reset(__pointer __p = nullptr) noexcept
139 swap(observer_ptr& __p) noexcept
141 std::swap(__t, __p.__t);
148 template<typename _Tp>
150 swap(observer_ptr<_Tp>& __p1, observer_ptr<_Tp>& __p2) noexcept
155 template<typename _Tp>
157 make_observer(_Tp* __p) noexcept
159 return observer_ptr<_Tp>(__p);
162 template<typename _Tp, typename _Up>
164 operator==(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
166 return __p1.get() == __p2.get();
169 template<typename _Tp, typename _Up>
171 operator!=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
173 return !(__p1 == __p2);
176 template<typename _Tp>
178 operator==(observer_ptr<_Tp> __p, nullptr_t) noexcept
183 template<typename _Tp>
185 operator==(nullptr_t, observer_ptr<_Tp> __p) noexcept
190 template<typename _Tp>
192 operator!=(observer_ptr<_Tp> __p, nullptr_t) noexcept
197 template<typename _Tp>
199 operator!=(nullptr_t, observer_ptr<_Tp> __p) noexcept
204 template<typename _Tp, typename _Up>
206 operator<(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
208 return std::less<typename common_type<typename add_pointer<_Tp>::type,
209 typename add_pointer<_Up>::type
211 >{}(__p1.get(), __p2.get());
214 template<typename _Tp, typename _Up>
216 operator>(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
221 template<typename _Tp, typename _Up>
223 operator<=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
225 return !(__p2 < __p1);
228 template<typename _Tp, typename _Up>
230 operator>=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
232 return !(__p1 < __p2);
234} // namespace fundamentals_v2
235} // namespace experimental
237template <typename _Tp>
238 struct hash<experimental::observer_ptr<_Tp>>
240 using result_type = size_t;
241 using argument_type = experimental::observer_ptr<_Tp>;
244 operator()(const experimental::observer_ptr<_Tp>& __t) const
245 noexcept(noexcept(hash<typename add_pointer<_Tp>::type> {}(__t.get())))
247 return hash<typename add_pointer<_Tp>::type> {}(__t.get());
252_GLIBCXX_END_NAMESPACE_VERSION
255#endif // __cplusplus <= 201103L
257#endif // _GLIBCXX_EXPERIMENTAL_MEMORY