14 #ifndef ATOMIC_SMART_PTR_H_
15 #define ATOMIC_SMART_PTR_H_
17 #include "atomic_prv_basic.h"
36 void reset(t_ptr t =
nullptr) noexcept {
37 t_ptr old = m_ptr.exchange(t);
44 m_ptr = x.m_ptr.exchange(m_ptr);
47 bool operator!() const noexcept {
return !(t_ptr)m_ptr;}
48 operator bool() const noexcept {
return (t_ptr)m_ptr;}
75 typedef uintptr_t Refcnt;
78 std::function<void()> deleter;
99 typedef uintptr_t Refcnt;
103 void set_deleter(
const std::function<
void()> &d) {m_deleter = d;}
104 std::function<void()> get_deleter()
const {
return m_deleter;}
105 std::function<void()> m_deleter;
110 template <
typename T,
typename reflocal_t,
typename reflocal_var_t,
typename Enable =
void>
114 typedef typename Ref::Refcnt Refcnt;
116 static int deleter(Ref *p) noexcept {
delete p;
return 1; }
121 m_ref = (reflocal_t)
new Ref(y, deleter);
123 T *
get() noexcept {
return this->m_ref ? ((Ref*)(reflocal_t)this->m_ref)->ptr : NULL; }
124 const T *get() const noexcept {
return this->m_ref ? ((
const Ref*)(reflocal_t)this->m_ref)->ptr : NULL; }
126 int _use_count_() const noexcept {
return ((
const Ref*)(reflocal_t)this->m_ref)->refcnt;}
128 reflocal_var_t m_ref;
129 enum {ATOMIC_SHARED_REF_ALIGNMENT = (
sizeof(intptr_t))};
132 template <
typename T,
typename reflocal_t,
typename reflocal_var_t>
133 struct atomic_shared_ptr_base<T, reflocal_t, reflocal_var_t, typename std::enable_if<std::is_base_of<atomic_countable, T>::value>::type > {
136 typedef typename atomic_countable::Refcnt Refcnt;
138 int deleter(T *p) noexcept {
auto d = p->get_deleter(); d();
return 1;}
141 template<
typename Y,
typename D>
void reset_unsafe(Y *y, D d) noexcept {
142 m_ref = (reflocal_t)static_cast<T*>(y);
143 get()->set_deleter(d);
145 T *
get() noexcept {
return (T*)(reflocal_t)this->m_ref; }
146 const T *
get()
const noexcept {
return (
const T*)(reflocal_t)this->m_ref; }
148 int _use_count_() const noexcept {
return ((
const T*)(reflocal_t)this->m_ref)->refcnt;}
150 reflocal_var_t m_ref;
151 enum {ATOMIC_SHARED_REF_ALIGNMENT = (
sizeof(double))};
157 template <
typename T,
typename reflocal_var_t = u
intptr_t>
167 static_assert(
sizeof(static_cast<const T*>(y.get())),
"");
168 this->m_ref =
reinterpret_cast<RefLocal_
>(y.
scan_());
173 this->m_ref = t.m_ref;
174 t.m_ref = (RefLocal_)
nullptr;
177 this->m_ref = y.m_ref;
178 y.m_ref = (RefLocal_)
nullptr;
203 this->m_ref =
reinterpret_cast<RefLocal_
>(t.scan_());
208 static_assert(
sizeof(static_cast<const T*>(y.get())),
"");
210 this->m_ref =
reinterpret_cast<RefLocal_
>(y.
scan_());
220 inline void reset() noexcept;
223 template<
typename Y,
typename D>
void reset(Y *y, D deleter) { reset(); this->
reset_unsafe(y, deleter); }
228 T &operator*() noexcept { assert( *
this);
return *
get();}
229 const T &operator*() const noexcept { assert( *
this);
return *
get();}
231 T *operator->() noexcept { assert( *
this);
return get();}
232 const T *operator->() const noexcept { assert( *
this);
return get();}
234 bool operator!() const noexcept {
return !this->m_ref;}
235 operator bool() const noexcept {
return this->m_ref;}
238 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
239 return (this->pref_() == (
const Ref *)x.pref_());}
241 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
242 return (this->pref_() == (
const Ref *)x.pref_());}
244 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
245 return (this->pref_() != (
const Ref *)x.pref_());}
247 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
248 return (this->pref_() != (
const Ref *)x.pref_());}
250 int use_count() const noexcept {
return this->_use_count_();}
251 bool unique() const noexcept {
return use_count() == 1;}
257 typedef uintptr_t RefLocal_;
260 Ref*
pref_() const noexcept {
return (
Ref *)(RefLocal_)(this->m_ref);}
276 template <
typename T>
287 operator=(std::move(t));
290 operator=(std::move(y));
320 template<
typename Y>
void reset(Y *y) {
331 bool operator!() const noexcept {
return !this->m_ref;}
332 operator bool() const noexcept {
return this->m_ref;}
335 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
336 return (pref_() == (
const Ref*)x.pref_());}
338 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
339 return (pref_() == (
const Ref*)x.pref_());}
341 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
342 return (pref_() != (
const Ref*)x.pref_());}
344 static_assert(
sizeof(static_cast<const T*>(x.get())),
"");
345 return (pref_() != (
const Ref*)x.pref_());}
353 Ref*
pref_() const noexcept {
return (
Ref*)(this->m_ref & (~(uintptr_t)(this->ATOMIC_SHARED_REF_ALIGNMENT - 1)));}
356 Refcnt
refcnt_() const noexcept {
return (Refcnt)(this->m_ref & (uintptr_t)(this->ATOMIC_SHARED_REF_ALIGNMENT - 1));}
361 inline Ref *
scan_() const noexcept;
369 template <
bool NOSWAP>
374 template <typename T, class... Args>
379 template <
typename T,
class Alloc,
class... Args>
381 typename Alloc::template rebind<T>::other alloc(base_alloc);
382 auto p = alloc.allocate(1);
383 alloc.construct(p, std::forward<Args>(args)...);
384 auto deleter = [alloc, p]()
mutable {
386 alloc.deallocate(p, 1);
391 template <
typename T,
typename reflocal_var_t>
393 static_assert(
sizeof(static_cast<const T*>(y.get())),
"");
394 this->m_ref = (RefLocal_)y.m_ref;
399 template <
typename T,
typename reflocal_var_t>
400 template<
typename Y,
typename Z>
402 static_assert(
sizeof(static_cast<const T*>(y.get())),
"");
403 this->m_ref = (RefLocal_)y.m_ref;
408 template <
typename T,
typename reflocal_var_t>
413 template <
typename T,
typename reflocal_var_t>
423 else if(pref->refcnt.decAndTest()) {
426 this->m_ref = (RefLocal_)
nullptr;
428 template <
typename T>
440 return (
Ref*)
nullptr;
442 rcnt_new = rcnt_old + 1u;
450 if(rcnt_new >= this->ATOMIC_SHARED_REF_ALIGNMENT) {
465 template <
typename T>
470 if( !pref)
return (
Ref*)
nullptr;
476 template <
typename T>
483 Refcnt rcnt_new = rcnt_old - 1;
493 if(pref->refcnt.decAndTest()) {
500 template <
typename T>
501 template <
bool NOSWAP>
506 ++(newr.pref_()->refcnt);
509 Refcnt rcnt_old, rcnt_new;
511 if(pref != oldr.pref_()) {
519 --(newr.pref_()->refcnt);
523 if(oldr.pref_()->refcnt.decAndTest()) {
524 this->deleter(oldr.pref_());
527 oldr.m_ref = (uintptr_t)pref;
531 if(pref && (rcnt_old != 1u)) {
532 pref->refcnt += rcnt_old - 1u;
535 if(this->m_ref.compare_set_strong(
536 RefLocal_((uintptr_t)pref + rcnt_old),
537 RefLocal_((uintptr_t)newr.pref_() + rcnt_new)))
542 pref->refcnt += (Refcnt)( -(
int)(rcnt_old - 1u));
551 template <
typename T>
556 template <
typename T>
559 return compareAndSwap_<false>(oldr, newr);
561 template <
typename T,
typename reflocal_var_t>
564 RefLocal_ x = this->m_ref;
565 this->m_ref = (RefLocal_)r.m_ref;
569 template <
typename T,
typename reflocal_var_t>
574 Refcnt rcnt_old, rcnt_new;
575 pref = r.reserve_scan_( &rcnt_old);
576 if(pref && (rcnt_old != 1u)) {
577 pref->refcnt += rcnt_old - 1u;
580 if(r.m_ref.compare_set_strong(
581 RefLocal_((uintptr_t)pref + rcnt_old),
582 RefLocal_((uintptr_t)this->m_ref + rcnt_new)))
587 pref->refcnt += (Refcnt)( -(
int)(rcnt_old - 1u));
591 this->m_ref = (RefLocal_)pref;