// shared_ptr atomic access -*- C++ -*- // Copyright (C) 2014-2017 Free Software Foundation, Inc. // // This file is part of the GNU ISO C++ Library. This library is free // software; you can redistribute it and/or modify it under the // terms of the GNU General Public License as published by the // Free Software Foundation; either version 3, or (at your option) // any later version. // This library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // Under Section 7 of GPL version 3, you are granted additional // permissions described in the GCC Runtime Library Exception, version // 3.1, as published by the Free Software Foundation. // You should have received a copy of the GNU General Public License and // a copy of the GCC Runtime Library Exception along with this program; // see the files COPYING3 and COPYING.RUNTIME respectively. If not, see // <http://www.gnu.org/licenses/>. /** @file bits/shared_ptr_atomic.h * This is an internal header file, included by other library headers. * Do not attempt to use it directly. @headername{memory} */ #ifndef _SHARED_PTR_ATOMIC_H #define _SHARED_PTR_ATOMIC_H 1 #include <bits/atomic_base.h> namespace std _GLIBCXX_VISIBILITY(default) { _GLIBCXX_BEGIN_NAMESPACE_VERSION /** * @addtogroup pointer_abstractions * @{ */ struct _Sp_locker { _Sp_locker(const _Sp_locker&) = delete; _Sp_locker& operator=(const _Sp_locker&) = delete; #ifdef __GTHREADS explicit _Sp_locker(const void*) noexcept; _Sp_locker(const void*, const void*) noexcept; ~_Sp_locker(); private: unsigned char _M_key1; unsigned char _M_key2; #else explicit _Sp_locker(const void*, const void* = nullptr) { } #endif }; /** * @brief Report whether shared_ptr atomic operations are lock-free. * @param __p A non-null pointer to a shared_ptr object. * @return True if atomic access to @c *__p is lock-free, false otherwise. * @{ */ template<typename _Tp, _Lock_policy _Lp> inline bool atomic_is_lock_free(const __shared_ptr<_Tp, _Lp>* __p) { #ifdef __GTHREADS return __gthread_active_p() == 0; #else return true; #endif } template<typename _Tp> inline bool atomic_is_lock_free(const shared_ptr<_Tp>* __p) { return std::atomic_is_lock_free<_Tp, __default_lock_policy>(__p); } // @} /** * @brief Atomic load for shared_ptr objects. * @param __p A non-null pointer to a shared_ptr object. * @return @c *__p * * The memory order shall not be @c memory_order_release or * @c memory_order_acq_rel. * @{ */ template<typename _Tp> inline shared_ptr<_Tp> atomic_load_explicit(const shared_ptr<_Tp>* __p, memory_order) { _Sp_locker __lock{__p}; return *__p; } template<typename _Tp> inline shared_ptr<_Tp> atomic_load(const shared_ptr<_Tp>* __p) { return std::atomic_load_explicit(__p, memory_order_seq_cst); } template<typename _Tp, _Lock_policy _Lp> inline __shared_ptr<_Tp, _Lp> atomic_load_explicit(const __shared_ptr<_Tp, _Lp>* __p, memory_order) { _Sp_locker __lock{__p}; return *__p; } template<typename _Tp, _Lock_policy _Lp> inline __shared_ptr<_Tp, _Lp> atomic_load(const __shared_ptr<_Tp, _Lp>* __p) { return std::atomic_load_explicit(__p, memory_order_seq_cst); } // @} /** * @brief Atomic store for shared_ptr objects. * @param __p A non-null pointer to a shared_ptr object. * @param __r The value to store. * * The memory order shall not be @c memory_order_acquire or * @c memory_order_acq_rel. * @{ */ template<typename _Tp> inline void atomic_store_explicit(shared_ptr<_Tp>* __p, shared_ptr<_Tp> __r, memory_order) { _Sp_locker __lock{__p}; __p->swap(__r); // use swap so that **__p not destroyed while lock held } template<typename _Tp> inline void atomic_store(shared_ptr<_Tp>* __p, shared_ptr<_Tp> __r) { std::atomic_store_explicit(__p, std::move(__r), memory_order_seq_cst); } template<typename _Tp, _Lock_policy _Lp> inline void atomic_store_explicit(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp> __r, memory_order) { _Sp_locker __lock{__p}; __p->swap(__r); // use swap so that **__p not destroyed while lock held } template<typename _Tp, _Lock_policy _Lp> inline void atomic_store(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp> __r) { std::atomic_store_explicit(__p, std::move(__r), memory_order_seq_cst); } // @} /** * @brief Atomic exchange for shared_ptr objects. * @param __p A non-null pointer to a shared_ptr object. * @param __r New value to store in @c *__p. * @return The original value of @c *__p * @{ */ template<typename _Tp> inline shared_ptr<_Tp> atomic_exchange_explicit(shared_ptr<_Tp>* __p, shared_ptr<_Tp> __r, memory_order) { _Sp_locker __lock{__p}; __p->swap(__r); return __r; } template<typename _Tp> inline shared_ptr<_Tp> atomic_exchange(shared_ptr<_Tp>* __p, shared_ptr<_Tp> __r) { return std::atomic_exchange_explicit(__p, std::move(__r), memory_order_seq_cst); } template<typename _Tp, _Lock_policy _Lp> inline __shared_ptr<_Tp, _Lp> atomic_exchange_explicit(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp> __r, memory_order) { _Sp_locker __lock{__p}; __p->swap(__r); return __r; } template<typename _Tp, _Lock_policy _Lp> inline __shared_ptr<_Tp, _Lp> atomic_exchange(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp> __r) { return std::atomic_exchange_explicit(__p, std::move(__r), memory_order_seq_cst); } // @} /** * @brief Atomic compare-and-swap for shared_ptr objects. * @param __p A non-null pointer to a shared_ptr object. * @param __v A non-null pointer to a shared_ptr object. * @param __w A non-null pointer to a shared_ptr object. * @return True if @c *__p was equivalent to @c *__v, false otherwise. * * The memory order for failure shall not be @c memory_order_release or * @c memory_order_acq_rel, or stronger than the memory order for success. * @{ */ template<typename _Tp> bool atomic_compare_exchange_strong_explicit(shared_ptr<_Tp>* __p, shared_ptr<_Tp>* __v, shared_ptr<_Tp> __w, memory_order, memory_order) { shared_ptr<_Tp> __x; // goes out of scope after __lock _Sp_locker __lock{__p, __v}; owner_less<shared_ptr<_Tp>> __less; if (*__p == *__v && !__less(*__p, *__v) && !__less(*__v, *__p)) { __x = std::move(*__p); *__p = std::move(__w); return true; } __x = std::move(*__v); *__v = *__p; return false; } template<typename _Tp> inline bool atomic_compare_exchange_strong(shared_ptr<_Tp>* __p, shared_ptr<_Tp>* __v, shared_ptr<_Tp> __w) { return std::atomic_compare_exchange_strong_explicit(__p, __v, std::move(__w), memory_order_seq_cst, memory_order_seq_cst); } template<typename _Tp> inline bool atomic_compare_exchange_weak_explicit(shared_ptr<_Tp>* __p, shared_ptr<_Tp>* __v, shared_ptr<_Tp> __w, memory_order __success, memory_order __failure) { return std::atomic_compare_exchange_strong_explicit(__p, __v, std::move(__w), __success, __failure); } template<typename _Tp> inline bool atomic_compare_exchange_weak(shared_ptr<_Tp>* __p, shared_ptr<_Tp>* __v, shared_ptr<_Tp> __w) { return std::atomic_compare_exchange_weak_explicit(__p, __v, std::move(__w), memory_order_seq_cst, memory_order_seq_cst); } template<typename _Tp, _Lock_policy _Lp> bool atomic_compare_exchange_strong_explicit(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp>* __v, __shared_ptr<_Tp, _Lp> __w, memory_order, memory_order) { __shared_ptr<_Tp, _Lp> __x; // goes out of scope after __lock _Sp_locker __lock{__p, __v}; owner_less<__shared_ptr<_Tp, _Lp>> __less; if (*__p == *__v && !__less(*__p, *__v) && !__less(*__v, *__p)) { __x = std::move(*__p); *__p = std::move(__w); return true; } __x = std::move(*__v); *__v = *__p; return false; } template<typename _Tp, _Lock_policy _Lp> inline bool atomic_compare_exchange_strong(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp>* __v, __shared_ptr<_Tp, _Lp> __w) { return std::atomic_compare_exchange_strong_explicit(__p, __v, std::move(__w), memory_order_seq_cst, memory_order_seq_cst); } template<typename _Tp, _Lock_policy _Lp> inline bool atomic_compare_exchange_weak_explicit(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp>* __v, __shared_ptr<_Tp, _Lp> __w, memory_order __success, memory_order __failure) { return std::atomic_compare_exchange_strong_explicit(__p, __v, std::move(__w), __success, __failure); } template<typename _Tp, _Lock_policy _Lp> inline bool atomic_compare_exchange_weak(__shared_ptr<_Tp, _Lp>* __p, __shared_ptr<_Tp, _Lp>* __v, __shared_ptr<_Tp, _Lp> __w) { return std::atomic_compare_exchange_weak_explicit(__p, __v, std::move(__w), memory_order_seq_cst, memory_order_seq_cst); } // @} // @} group pointer_abstractions _GLIBCXX_END_NAMESPACE_VERSION } // namespace #endif // _SHARED_PTR_ATOMIC_H
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
algorithmfwd.h | File | 21.23 KB | 0644 |
|
alloc_traits.h | File | 19.13 KB | 0644 |
|
allocated_ptr.h | File | 3.43 KB | 0644 |
|
allocator.h | File | 6.73 KB | 0644 |
|
atomic_base.h | File | 23.28 KB | 0644 |
|
atomic_futex.h | File | 9.35 KB | 0644 |
|
atomic_lockfree_defines.h | File | 2.2 KB | 0644 |
|
basic_ios.h | File | 15.7 KB | 0644 |
|
basic_ios.tcc | File | 5.94 KB | 0644 |
|
basic_string.h | File | 235.15 KB | 0644 |
|
basic_string.tcc | File | 52.5 KB | 0644 |
|
boost_concept_check.h | File | 26.41 KB | 0644 |
|
c++0x_warning.h | File | 1.44 KB | 0644 |
|
char_traits.h | File | 20.26 KB | 0644 |
|
codecvt.h | File | 20.8 KB | 0644 |
|
concept_check.h | File | 3.34 KB | 0644 |
|
cpp_type_traits.h | File | 9.37 KB | 0644 |
|
cxxabi_forced.h | File | 1.77 KB | 0644 |
|
cxxabi_init_exception.h | File | 2.17 KB | 0644 |
|
deque.tcc | File | 33.26 KB | 0644 |
|
enable_special_members.h | File | 12.1 KB | 0644 |
|
exception.h | File | 2.23 KB | 0644 |
|
exception_defines.h | File | 1.61 KB | 0644 |
|
exception_ptr.h | File | 6.31 KB | 0644 |
|
forward_list.h | File | 47.5 KB | 0644 |
|
forward_list.tcc | File | 14.68 KB | 0644 |
|
fstream.tcc | File | 32.14 KB | 0644 |
|
functexcept.h | File | 3.18 KB | 0644 |
|
functional_hash.h | File | 8 KB | 0644 |
|
gslice.h | File | 5.39 KB | 0644 |
|
gslice_array.h | File | 7.59 KB | 0644 |
|
hash_bytes.h | File | 2.1 KB | 0644 |
|
hashtable.h | File | 71.56 KB | 0644 |
|
hashtable_policy.h | File | 65.47 KB | 0644 |
|
indirect_array.h | File | 7.68 KB | 0644 |
|
invoke.h | File | 3.57 KB | 0644 |
|
ios_base.h | File | 30.22 KB | 0644 |
|
istream.tcc | File | 30.36 KB | 0644 |
|
list.tcc | File | 15.54 KB | 0644 |
|
locale_classes.h | File | 24.31 KB | 0644 |
|
locale_classes.tcc | File | 8.18 KB | 0644 |
|
locale_conv.h | File | 15.7 KB | 0644 |
|
locale_facets.h | File | 89.98 KB | 0644 |
|
locale_facets.tcc | File | 38.64 KB | 0644 |
|
locale_facets_nonio.h | File | 67.33 KB | 0644 |
|
locale_facets_nonio.tcc | File | 44.46 KB | 0644 |
|
localefwd.h | File | 5.51 KB | 0644 |
|
mask_array.h | File | 7.42 KB | 0644 |
|
memoryfwd.h | File | 2.36 KB | 0644 |
|
move.h | File | 6.5 KB | 0644 |
|
nested_exception.h | File | 4.66 KB | 0644 |
|
node_handle.h | File | 7.97 KB | 0644 |
|
ostream.tcc | File | 12.03 KB | 0644 |
|
ostream_insert.h | File | 3.91 KB | 0644 |
|
parse_numbers.h | File | 7.84 KB | 0644 |
|
postypes.h | File | 8.02 KB | 0644 |
|
predefined_ops.h | File | 8.87 KB | 0644 |
|
ptr_traits.h | File | 4.75 KB | 0644 |
|
quoted_string.h | File | 4.47 KB | 0644 |
|
random.h | File | 171.19 KB | 0644 |
|
random.tcc | File | 103.08 KB | 0644 |
|
range_access.h | File | 9.75 KB | 0644 |
|
refwrap.h | File | 13.48 KB | 0644 |
|
regex.h | File | 95.9 KB | 0644 |
|
regex.tcc | File | 16.1 KB | 0644 |
|
regex_automaton.h | File | 10.47 KB | 0644 |
|
regex_automaton.tcc | File | 7.65 KB | 0644 |
|
regex_compiler.h | File | 16.79 KB | 0644 |
|
regex_compiler.tcc | File | 18.46 KB | 0644 |
|
regex_constants.h | File | 14.36 KB | 0644 |
|
regex_error.h | File | 4.85 KB | 0644 |
|
regex_executor.h | File | 7.31 KB | 0644 |
|
regex_executor.tcc | File | 16.91 KB | 0644 |
|
regex_scanner.h | File | 6.92 KB | 0644 |
|
regex_scanner.tcc | File | 14.66 KB | 0644 |
|
shared_ptr.h | File | 22.37 KB | 0644 |
|
shared_ptr_atomic.h | File | 9.54 KB | 0644 |
|
shared_ptr_base.h | File | 52.54 KB | 0644 |
|
slice_array.h | File | 9.13 KB | 0644 |
|
specfun.h | File | 44.03 KB | 0644 |
|
sstream.tcc | File | 9.88 KB | 0644 |
|
std_abs.h | File | 3.21 KB | 0644 |
|
std_function.h | File | 23.41 KB | 0644 |
|
std_mutex.h | File | 9.08 KB | 0644 |
|
stl_algo.h | File | 208.85 KB | 0644 |
|
stl_algobase.h | File | 49.22 KB | 0644 |
|
stl_bvector.h | File | 33.03 KB | 0644 |
|
stl_construct.h | File | 7.22 KB | 0644 |
|
stl_deque.h | File | 75.95 KB | 0644 |
|
stl_function.h | File | 32.5 KB | 0644 |
|
stl_heap.h | File | 19.73 KB | 0644 |
|
stl_iterator.h | File | 40.77 KB | 0644 |
|
stl_iterator_base_funcs.h | File | 7.88 KB | 0644 |
|
stl_iterator_base_types.h | File | 8.48 KB | 0644 |
|
stl_list.h | File | 63.65 KB | 0644 |
|
stl_map.h | File | 50.02 KB | 0644 |
|
stl_multimap.h | File | 39 KB | 0644 |
|
stl_multiset.h | File | 33.26 KB | 0644 |
|
stl_numeric.h | File | 13.51 KB | 0644 |
|
stl_pair.h | File | 18.21 KB | 0644 |
|
stl_queue.h | File | 21.61 KB | 0644 |
|
stl_raw_storage_iter.h | File | 3.74 KB | 0644 |
|
stl_relops.h | File | 4.49 KB | 0644 |
|
stl_set.h | File | 33.63 KB | 0644 |
|
stl_stack.h | File | 11.18 KB | 0644 |
|
stl_tempbuf.h | File | 8.15 KB | 0644 |
|
stl_tree.h | File | 73.07 KB | 0644 |
|
stl_uninitialized.h | File | 27 KB | 0644 |
|
stl_vector.h | File | 54.33 KB | 0644 |
|
stream_iterator.h | File | 6.5 KB | 0644 |
|
streambuf.tcc | File | 4.81 KB | 0644 |
|
streambuf_iterator.h | File | 12.33 KB | 0644 |
|
string_view.tcc | File | 6.48 KB | 0644 |
|
stringfwd.h | File | 2.55 KB | 0644 |
|
uniform_int_dist.h | File | 9.91 KB | 0644 |
|
unique_ptr.h | File | 24.88 KB | 0644 |
|
unordered_map.h | File | 67.04 KB | 0644 |
|
unordered_set.h | File | 52.56 KB | 0644 |
|
uses_allocator.h | File | 6.23 KB | 0644 |
|
valarray_after.h | File | 22.12 KB | 0644 |
|
valarray_array.h | File | 21.3 KB | 0644 |
|
valarray_array.tcc | File | 7.08 KB | 0644 |
|
valarray_before.h | File | 18.08 KB | 0644 |
|
vector.tcc | File | 26.95 KB | 0644 |
|