libstdc++
experimental/memory
Go to the documentation of this file.
1// <experimental/memory> -*- C++ -*-
2
3// Copyright (C) 2015-2021 Free Software Foundation, Inc.
4//
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)
9// any later version.
10
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.
15
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.
19
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/>.
24
25/** @file experimental/memory
26 * This is a TS C++ Library header.
27 * @ingroup libfund-ts
28 */
29
30//
31// N4336 Working Draft, C++ Extensions for Library Fundamentals, Version 2
32//
33
34#ifndef _GLIBCXX_EXPERIMENTAL_MEMORY
35#define _GLIBCXX_EXPERIMENTAL_MEMORY 1
36
37#pragma GCC system_header
38
39#if __cplusplus >= 201402L
40
41#include <memory>
42#include <type_traits>
43#include <utility>
44#include <experimental/bits/shared_ptr.h>
45#include <bits/functional_hash.h>
46
47namespace std _GLIBCXX_VISIBILITY(default)
48{
49_GLIBCXX_BEGIN_NAMESPACE_VERSION
50
51namespace experimental
52{
53inline namespace fundamentals_v2
54{
55#define __cpp_lib_experimental_observer_ptr 201411
56
57 template <typename _Tp>
58 class observer_ptr
59 {
60 public:
61 // publish our template parameter and variations thereof
62 using element_type = _Tp;
63 using __pointer = add_pointer_t<_Tp>; // exposition-only
64 using __reference = add_lvalue_reference_t<_Tp>; // exposition-only
65
66 // 3.2.2, observer_ptr constructors
67 // default c'tor
68 constexpr observer_ptr() noexcept
69 : __t()
70 { }
71
72 // pointer-accepting c'tors
73 constexpr observer_ptr(nullptr_t) noexcept
74 : __t()
75 { }
76
77 constexpr explicit observer_ptr(__pointer __p) noexcept
78 : __t(__p)
79 { }
80
81 // copying c'tors (in addition to compiler-generated copy c'tor)
82 template <typename _Up,
83 typename = typename enable_if<
84 is_convertible<typename add_pointer<_Up>::type, __pointer
85 >::value
86 >::type>
87 constexpr observer_ptr(observer_ptr<_Up> __p) noexcept
88 : __t(__p.get())
89 {
90 }
91
92 // 3.2.3, observer_ptr observers
93 constexpr __pointer
94 get() const noexcept
95 {
96 return __t;
97 }
98
99 constexpr __reference
100 operator*() const
101 {
102 return *get();
103 }
104
105 constexpr __pointer
106 operator->() const noexcept
107 {
108 return get();
109 }
110
111 constexpr explicit operator bool() const noexcept
112 {
113 return get() != nullptr;
114 }
115
116 // 3.2.4, observer_ptr conversions
117 constexpr explicit operator __pointer() const noexcept
118 {
119 return get();
120 }
121
122 // 3.2.5, observer_ptr modifiers
123 constexpr __pointer
124 release() noexcept
125 {
126 __pointer __tmp = get();
127 reset();
128 return __tmp;
129 }
130
131 constexpr void
132 reset(__pointer __p = nullptr) noexcept
133 {
134 __t = __p;
135 }
136
137 constexpr void
138 swap(observer_ptr& __p) noexcept
139 {
140 std::swap(__t, __p.__t);
141 }
142
143 private:
144 __pointer __t;
145 }; // observer_ptr<>
146
147 template<typename _Tp>
148 void
149 swap(observer_ptr<_Tp>& __p1, observer_ptr<_Tp>& __p2) noexcept
150 {
151 __p1.swap(__p2);
152 }
153
154 template<typename _Tp>
155 observer_ptr<_Tp>
156 make_observer(_Tp* __p) noexcept
157 {
158 return observer_ptr<_Tp>(__p);
159 }
160
161 template<typename _Tp, typename _Up>
162 bool
163 operator==(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
164 {
165 return __p1.get() == __p2.get();
166 }
167
168 template<typename _Tp, typename _Up>
169 bool
170 operator!=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
171 {
172 return !(__p1 == __p2);
173 }
174
175 template<typename _Tp>
176 bool
177 operator==(observer_ptr<_Tp> __p, nullptr_t) noexcept
178 {
179 return !__p;
180 }
181
182 template<typename _Tp>
183 bool
184 operator==(nullptr_t, observer_ptr<_Tp> __p) noexcept
185 {
186 return !__p;
187 }
188
189 template<typename _Tp>
190 bool
191 operator!=(observer_ptr<_Tp> __p, nullptr_t) noexcept
192 {
193 return bool(__p);
194 }
195
196 template<typename _Tp>
197 bool
198 operator!=(nullptr_t, observer_ptr<_Tp> __p) noexcept
199 {
200 return bool(__p);
201 }
202
203 template<typename _Tp, typename _Up>
204 bool
205 operator<(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
206 {
207 return std::less<typename common_type<typename add_pointer<_Tp>::type,
208 typename add_pointer<_Up>::type
209 >::type
210 >{}(__p1.get(), __p2.get());
211 }
212
213 template<typename _Tp, typename _Up>
214 bool
215 operator>(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
216 {
217 return __p2 < __p1;
218 }
219
220 template<typename _Tp, typename _Up>
221 bool
222 operator<=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
223 {
224 return !(__p2 < __p1);
225 }
226
227 template<typename _Tp, typename _Up>
228 bool
229 operator>=(observer_ptr<_Tp> __p1, observer_ptr<_Up> __p2)
230 {
231 return !(__p1 < __p2);
232 }
233} // namespace fundamentals_v2
234} // namespace experimental
235
236template <typename _Tp>
237 struct hash<experimental::observer_ptr<_Tp>>
238 {
239 using result_type = size_t;
240 using argument_type = experimental::observer_ptr<_Tp>;
241
242 size_t
243 operator()(const experimental::observer_ptr<_Tp>& __t) const
244 noexcept(noexcept(hash<typename add_pointer<_Tp>::type> {}(__t.get())))
245 {
246 return hash<typename add_pointer<_Tp>::type> {}(__t.get());
247 }
248 };
249
250
251_GLIBCXX_END_NAMESPACE_VERSION
252} // namespace std
253
254#endif // __cplusplus <= 201103L
255
256#endif // _GLIBCXX_EXPERIMENTAL_MEMORY