atomic_smart_ptr.h
1 /***************************************************************************
2  Copyright (C) 2002-2015 Kentaro Kitagawa
3  kitagawa@phys.s.u-tokyo.ac.jp
4 
5  This program is free software; you can redistribute it and/or
6  modify it under the terms of the GNU Library General Public
7  License as published by the Free Software Foundation; either
8  version 2 of the License, or (at your option) any later version.
9 
10  You should have received a copy of the GNU Library General
11  Public License and a list of authors along with this program;
12  see the files COPYING and AUTHORS.
13  ***************************************************************************/
14 #ifndef ATOMIC_SMART_PTR_H_
15 #define ATOMIC_SMART_PTR_H_
16 
17 #include "atomic_prv_basic.h"
18 #include <functional>
19 
20 //! \brief This is an atomic variant of \a std::unique_ptr.
21 //! An instance of atomic_unique_ptr can be shared among threads by the use of \a swap(\a _shared_target_).\n
22 //! Namely, it is destructive reading.
23 //! Use atomic_shared_ptr when the pointer is required to be shared among scopes and threads.\n
24 //! This implementation relies on an atomic-swap machine code, e.g. lock xchg, or std::atomic.
25 //! \sa atomic_shared_ptr, atomic_unique_ptr_test.cpp
26 template <typename T>
28  typedef T* t_ptr;
29 public:
30  atomic_unique_ptr() noexcept : m_ptr(nullptr) {}
31 
32  explicit atomic_unique_ptr(t_ptr t) noexcept : m_ptr(t) {}
33 
34  ~atomic_unique_ptr() {delete (t_ptr)m_ptr;}
35 
36  void reset(t_ptr t = nullptr) noexcept {
37  t_ptr old = m_ptr.exchange(t);
38  delete old;
39  }
40  //! \param[in,out] x \p x is atomically swapped.
41  //! Nevertheless, this object is not atomically replaced.
42  //! That is, the object pointed by "this" must not be shared among threads.
43  void swap(atomic_unique_ptr &x) noexcept {
44  m_ptr = x.m_ptr.exchange(m_ptr);
45  }
46 
47  bool operator!() const noexcept {return !(t_ptr)m_ptr;}
48  operator bool() const noexcept {return (t_ptr)m_ptr;}
49 
50  //! This function lacks thread-safety.
51  T &operator*() const noexcept { assert((t_ptr)m_ptr); return (T &) *(t_ptr)m_ptr;}
52 
53  //! This function lacks thread-safety.
54  t_ptr operator->() const noexcept { assert((t_ptr)m_ptr); return (t_ptr)m_ptr;}
55 
56  //! This function lacks thread-safety.
57  t_ptr get() const noexcept { return (t_ptr )m_ptr;}
58 
59  atomic_unique_ptr(const atomic_unique_ptr &) = delete;
60  atomic_unique_ptr& operator=(const atomic_unique_ptr &) = delete;
61 
62 private:
63  atomic<t_ptr> m_ptr;
64 };
65 
66 //! This is an internal class holding a global reference counter and a pointer to the object.
67 //! \sa atomic_shared_ptr
68 template <typename T>
70  template <typename D>
71  atomic_shared_ptr_gref_(T *p, D d) noexcept : ptr(p), refcnt(1), deleter(d) {}
72  ~atomic_shared_ptr_gref_() noexcept { assert(refcnt == 0); deleter(); }
73  //! The pointer to the object.
74  T *ptr;
75  typedef uintptr_t Refcnt;
76  //! The global reference counter.
78  std::function<void()> deleter;
79 
81 };
82 
83 template <typename X, typename Y, typename Z, typename E> struct atomic_shared_ptr_base;
84 template <typename X> class atomic_shared_ptr;
85 template <typename X, typename Y> class local_shared_ptr;
86 
87 //! Use subclass of this to be storaged in atomic_shared_ptr with
88 //! intrusive counting to obtain better performance.
90  atomic_countable() noexcept : refcnt(1) {}
91  atomic_countable(const atomic_countable &) noexcept : refcnt(1) {}
92  ~atomic_countable() { assert(refcnt == 0); }
93 
94  atomic_countable& operator=(const atomic_countable &) = delete;
95 private:
96  template <typename X, typename Y, typename Z, typename E> friend struct atomic_shared_ptr_base;
97  template <typename X> friend class atomic_shared_ptr;
98  template <typename X, typename Y> friend class local_shared_ptr;
99  typedef uintptr_t Refcnt;
100  //! Global reference counter.
102 
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;
106 };
107 
108 //! \brief Base class for atomic_shared_ptr without intrusive counting, so-called "simple counted".\n
109 //! A global referece counter (an instance of atomic_shared_ptr_gref_) will be created.
110 template <typename T, typename reflocal_t, typename reflocal_var_t, typename Enable = void>
111 struct atomic_shared_ptr_base {
112 protected:
113  typedef atomic_shared_ptr_gref_<T> Ref;
114  typedef typename Ref::Refcnt Refcnt;
115 
116  static int deleter(Ref *p) noexcept { delete p; return 1; }
117 
118  //! can be used to initialize the internal pointer \a m_ref.
119  //! \sa reset()
120  template<typename Y, typename D> void reset_unsafe(Y *y, D deleter) {
121  m_ref = (reflocal_t)new Ref(y, deleter);
122  }
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; }
125 
126  int _use_count_() const noexcept {return ((const Ref*)(reflocal_t)this->m_ref)->refcnt;}
127 
128  reflocal_var_t m_ref;
129  enum {ATOMIC_SHARED_REF_ALIGNMENT = (sizeof(intptr_t))};
130 };
131 //! \brief Base class for atomic_shared_ptr with intrusive counting.
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 > {
134 protected:
135  typedef T Ref;
136  typedef typename atomic_countable::Refcnt Refcnt;
137 
138  int deleter(T *p) noexcept { auto d = p->get_deleter(); d(); return 1;}
139 
140  //! can be used to initialize the internal pointer \a m_ref.
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);
144  }
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; }
147 
148  int _use_count_() const noexcept {return ((const T*)(reflocal_t)this->m_ref)->refcnt;}
149 
150  reflocal_var_t m_ref;
151  enum {ATOMIC_SHARED_REF_ALIGNMENT = (sizeof(double))};
152 };
153 
154 //! \brief This class provides non-reentrant interfaces for atomic_shared_ptr: operator->(), operator*() and so on.\n
155 //! Use this class in non-reentrant scopes instead of costly atomic_shared_ptr.
156 //! \sa atomic_shared_ptr, atomic_unique_ptr, atomic_shared_ptr_test.cpp.
157 template <typename T, typename reflocal_var_t = uintptr_t>
158 class local_shared_ptr : protected atomic_shared_ptr_base<T, uintptr_t, reflocal_var_t> {
159 public:
160  local_shared_ptr() noexcept { this->m_ref = (RefLocal_)nullptr; }
161 
162  template<typename Y> explicit local_shared_ptr(Y *y) { this->reset_unsafe(y, [y](){delete y;}); }
163  template<typename Y, typename D> local_shared_ptr(Y *y, D deleter) { this->reset_unsafe(y, deleter); }
164 
165  explicit local_shared_ptr(const atomic_shared_ptr<T> &t) noexcept { this->m_ref = reinterpret_cast<RefLocal_>(t.scan_()); }
166  template<typename Y> local_shared_ptr(const atomic_shared_ptr<Y> &y) {
167  static_assert(sizeof(static_cast<const T*>(y.get())), "");
168  this->m_ref = reinterpret_cast<RefLocal_>(y.scan_());
169  }
170  inline local_shared_ptr(const local_shared_ptr<T, reflocal_var_t> &t) noexcept;
171  template<typename Y, typename Z> inline local_shared_ptr(const local_shared_ptr<Y, Z> &y) noexcept;
173  this->m_ref = t.m_ref;
174  t.m_ref = (RefLocal_)nullptr;
175  }
176  template<typename Y, typename Z> local_shared_ptr(local_shared_ptr<Y, Z> &&y) noexcept {
177  this->m_ref = y.m_ref;
178  y.m_ref = (RefLocal_)nullptr;
179  }
180  inline ~local_shared_ptr();
181 
182  local_shared_ptr &operator=(const local_shared_ptr &t) noexcept {
183  local_shared_ptr(t).swap( *this);
184  return *this;
185  }
186  template<typename Y, typename Z> local_shared_ptr &operator=(const local_shared_ptr<Y, Z> &y) noexcept {
187  local_shared_ptr(y).swap( *this);
188  return *this;
189  }
190  local_shared_ptr &operator=(local_shared_ptr &&t) noexcept {
191  t.swap( *this);
192  t.reset();
193  return *this;
194  }
195  template<typename Y, typename Z> local_shared_ptr &operator=(local_shared_ptr<Y, Z> &&y) noexcept {
196  y.swap( *this);
197  y.reset();
198  return *this;
199  }
200  //! \param[in] t The pointer held by this instance is replaced with that of \a t.
202  this->reset();
203  this->m_ref = reinterpret_cast<RefLocal_>(t.scan_());
204  return *this;
205  }
206  //! \param[in] y The pointer held by this instance is replaced with that of \a y.
207  template<typename Y> local_shared_ptr &operator=(const atomic_shared_ptr<Y> &y) noexcept {
208  static_assert(sizeof(static_cast<const T*>(y.get())), "");
209  this->reset();
210  this->m_ref = reinterpret_cast<RefLocal_>(y.scan_());
211  return *this;
212  }
213 
214  //! \param[in,out] x \p The pointer held by \a x is swapped with that of this instance.
215  inline void swap(local_shared_ptr &x) noexcept;
216  //! \param[in,out] x \p The pointer held by \a x is atomically swapped with that of this instance.
217  void swap(atomic_shared_ptr<T> &x) noexcept;
218 
219  //! The pointer held by this instance is reset to null pointer.
220  inline void reset() noexcept;
221  //! The pointer held by this instance is reset with a pointer \a y.
222  template<typename Y> void reset(Y *y) { reset(); this->reset_unsafe(y, [y](){ delete y;}); }
223  template<typename Y, typename D> void reset(Y *y, D deleter) { reset(); this->reset_unsafe(y, deleter); }
224 
225  T *get() noexcept { return atomic_shared_ptr_base<T, uintptr_t, reflocal_var_t>::get(); }
226  const T *get() const noexcept { return atomic_shared_ptr_base<T, uintptr_t, reflocal_var_t>::get(); }
227 
228  T &operator*() noexcept { assert( *this); return *get();}
229  const T &operator*() const noexcept { assert( *this); return *get();}
230 
231  T *operator->() noexcept { assert( *this); return get();}
232  const T *operator->() const noexcept { assert( *this); return get();}
233 
234  bool operator!() const noexcept {return !this->m_ref;}
235  operator bool() const noexcept {return this->m_ref;}
236 
237  template<typename Y, typename Z> bool operator==(const local_shared_ptr<Y, Z> &x) const noexcept {
238  static_assert(sizeof(static_cast<const T*>(x.get())), "");
239  return (this->pref_() == (const Ref *)x.pref_());}
240  template<typename Y> bool operator==(const atomic_shared_ptr<Y> &x) const noexcept {
241  static_assert(sizeof(static_cast<const T*>(x.get())), "");
242  return (this->pref_() == (const Ref *)x.pref_());}
243  template<typename Y, typename Z> bool operator!=(const local_shared_ptr<Y, Z> &x) const noexcept {
244  static_assert(sizeof(static_cast<const T*>(x.get())), "");
245  return (this->pref_() != (const Ref *)x.pref_());}
246  template<typename Y> bool operator!=(const atomic_shared_ptr<Y> &x) const noexcept {
247  static_assert(sizeof(static_cast<const T*>(x.get())), "");
248  return (this->pref_() != (const Ref *)x.pref_());}
249 
250  int use_count() const noexcept { return this->_use_count_();}
251  bool unique() const noexcept {return use_count() == 1;}
252 protected:
253  template <typename Y, typename Z> friend class local_shared_ptr;
254  template <typename Y> friend class atomic_shared_ptr;
257  typedef uintptr_t RefLocal_;
258 
259  //! A pointer to global reference struct.
260  Ref* pref_() const noexcept {return (Ref *)(RefLocal_)(this->m_ref);}
261 };
262 
263 /*! \brief This is an atomic variant of \a std::shared_ptr, and can be shared by atomic and lock-free means.\n
264  *
265 * \a atomic_shared_ptr can be shared among threads by the use of \a operator=(_target_), \a swap(_target_).
266 * An instance of \a atomic_shared_ptr<T> holds:\n
267 * a) a pointer to \a atomic_shared_ptr_gref_<T>, which is a struct. consisting of a pointer to a T-type object and a global reference counter.\n
268 * b) a local (temporary) reference counter, which is embedded in the above pointer by using several LSBs that should be usually zero.\n
269 * The values of a) and b), \a m_ref, are atomically handled with CAS machine codes.
270 * The purpose of b) the local reference counter is to tell the "observation" to the shared target before increasing the global reference counter.
271 * This process is implemented in \a reserve_scan_().\n
272 * A function \a leave_scan_() tries to decrease the local counter first. When it fails, the global counter is decreased.\n
273 * To swap the pointer and local reference counter (which will be reset to zero), the setter must adds the local counting to the global counter before swapping.
274 * \sa atomic_unique_ptr, local_shared_ptr, atomic_shared_ptr_test.cpp.
275  */
276 template <typename T>
277 class atomic_shared_ptr : protected local_shared_ptr<T, atomic<uintptr_t>> {
278 public:
279  atomic_shared_ptr() noexcept : local_shared_ptr<T, atomic<uintptr_t>>() {}
280 
281  template<typename Y> explicit atomic_shared_ptr(Y *y) : local_shared_ptr<T, atomic<uintptr_t>>(y) {}
283  template<typename Y> atomic_shared_ptr(const atomic_shared_ptr<Y> &y) noexcept : local_shared_ptr<T, atomic<uintptr_t>>(y) {}
285  template<typename Y> atomic_shared_ptr(const local_shared_ptr<Y> &y) noexcept : local_shared_ptr<T, atomic<uintptr_t>>(y) {}
287  operator=(std::move(t));
288  }
289  template<typename Y> atomic_shared_ptr(atomic_shared_ptr<Y> &&y) noexcept {
290  operator=(std::move(y));
291  }
292 
293  ~atomic_shared_ptr() {}
294 
295  //! \param[in] t The pointer held by this instance is atomically replaced with that of \a t.
297  local_shared_ptr<T>(t).swap( *this);
298  return *this;
299  }
300  //! \param[in] y The pointer held by this instance is atomically replaced with that of \a y.
301  template<typename Y> atomic_shared_ptr &operator=(const local_shared_ptr<Y> &y) noexcept {
302  local_shared_ptr<T>(y).swap( *this);
303  return *this;
304  }
305  atomic_shared_ptr &operator=(local_shared_ptr<T> &&t) noexcept {
306  t.swap( *this);
307  t.reset();
308  return *this;
309  }
310  template<typename Y> atomic_shared_ptr &operator=(local_shared_ptr<Y> &&y) noexcept {
311  y.swap( *this);
312  y.reset();
313  return *this;
314  }
315  //! The pointer held by this instance is atomically reset to null pointer.
316  void reset() noexcept {
317  local_shared_ptr<T>().swap( *this);
318  }
319  //! The pointer held by this instance is atomically reset with a pointer \a y.
320  template<typename Y> void reset(Y *y) {
321  local_shared_ptr<T>(y).swap( *this);
322  }
323 
324  //! \return true if succeeded.
325  //! \sa compareAndSwap()
326  bool compareAndSet(const local_shared_ptr<T> &oldvalue, const local_shared_ptr<T> &newvalue) noexcept;
327  //! \return true if succeeded.
328  //! \sa compareAndSet()
329  bool compareAndSwap(local_shared_ptr<T> &oldvalue, const local_shared_ptr<T> &newvalue) noexcept;
330 
331  bool operator!() const noexcept {return !this->m_ref;}
332  operator bool() const noexcept {return this->m_ref;}
333 
334  template<typename Y> bool operator==(const local_shared_ptr<Y> &x) const noexcept {
335  static_assert(sizeof(static_cast<const T*>(x.get())), "");
336  return (pref_() == (const Ref*)x.pref_());}
337  template<typename Y> bool operator==(const atomic_shared_ptr<Y> &x) const noexcept {
338  static_assert(sizeof(static_cast<const T*>(x.get())), "");
339  return (pref_() == (const Ref*)x.pref_());}
340  template<typename Y> bool operator!=(const local_shared_ptr<Y> &x) const noexcept {
341  static_assert(sizeof(static_cast<const T*>(x.get())), "");
342  return (pref_() != (const Ref*)x.pref_());}
343  template<typename Y> bool operator!=(const atomic_shared_ptr<Y> &x) const noexcept {
344  static_assert(sizeof(static_cast<const T*>(x.get())), "");
345  return (pref_() != (const Ref*)x.pref_());}
346 protected:
347  template <typename Y, typename Z> friend class local_shared_ptr;
348  template <typename Y> friend class atomic_shared_ptr;
350  typedef typename atomic_shared_ptr_base<T, uintptr_t, atomic<uintptr_t>>::Refcnt Refcnt;
351  typedef atomic<uintptr_t> RefLocal_;
352  //! A pointer to global reference struct.
353  Ref* pref_() const noexcept {return (Ref*)(this->m_ref & (~(uintptr_t)(this->ATOMIC_SHARED_REF_ALIGNMENT - 1)));}
354  //! Local (temporary) reference counter.
355  //! Local reference counter is a trick to tell the observation to other threads.
356  Refcnt refcnt_() const noexcept {return (Refcnt)(this->m_ref & (uintptr_t)(this->ATOMIC_SHARED_REF_ALIGNMENT - 1));}
357 
358  //internal functions below.
359  //! Atomically scans \a m_ref and increases the global reference counter.
360  //! \a scan_() is used for atomically coping the pointer.
361  inline Ref *scan_() const noexcept;
362  //! Atomically scans \a m_ref and increases the local (temporary) reference counter.
363  //! use \a leave_scan_() to release the temporary reference.
364  inline Ref *reserve_scan_(Refcnt *) const noexcept;
365  //! Tries to decrease local (temporary) reference counter.
366  //! In case the reference is lost, \a leave_scan_() releases the global reference counter instead.
367  inline void leave_scan_(Ref *) const noexcept;
368 
369  template <bool NOSWAP>
370  inline bool compareAndSwap_(local_shared_ptr<T> &oldvalue, const local_shared_ptr<T> &newvalue) noexcept;
371 private:
372 };
373 
374 template <typename T, class... Args>
375 local_shared_ptr<T> make_local_shared(Args&&... args) {
376  return local_shared_ptr<T>(new T(std::forward<Args>(args)...));
377 }
378 
379 template <typename T, class Alloc, class... Args>
380 local_shared_ptr<T> allocate_local_shared(Alloc &base_alloc, Args&&... 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 {
385  alloc.destroy(p);
386  alloc.deallocate(p, 1);
387  };
388  return local_shared_ptr<T>(p, deleter);
389 }
390 
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;
395  if(pref_())
396  ++(pref_()->refcnt); //atomic
397 }
398 
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;
404  if(pref_())
405  ++(pref_()->refcnt); //atomic
406 }
407 
408 template <typename T, typename reflocal_var_t>
410  reset();
411 }
412 
413 template <typename T, typename reflocal_var_t>
414 inline void
416  Ref *pref = pref_();
417  if( !pref) return;
418  // decreases global reference counter.
419  if(unique()) {
420  pref->refcnt = 0;
421  this->deleter(pref);
422  }
423  else if(pref->refcnt.decAndTest()) {
424  this->deleter(pref);
425  }
426  this->m_ref = (RefLocal_)nullptr;
427 }
428 template <typename T>
429 inline typename atomic_shared_ptr<T>::Ref *
430 atomic_shared_ptr<T>::reserve_scan_(Refcnt *rcnt) const noexcept {
431  Ref *pref;
432  Refcnt rcnt_new;
433  for(;;) {
434  pref = pref_();
435  Refcnt rcnt_old;
436  rcnt_old = refcnt_();
437  if( !pref) {
438  // target is null.
439  *rcnt = rcnt_old;
440  return (Ref*)nullptr;
441  }
442  rcnt_new = rcnt_old + 1u;
443  /*
444  static int rcnt_max = 0;
445  if(rcnt_new > (int)rcnt_max) {
446  rcnt_max = rcnt_new;
447  fprintf(stderr, "max_rcnt=%d\n", rcnt_max);
448  }
449  */
450  if(rcnt_new >= this->ATOMIC_SHARED_REF_ALIGNMENT) {
451  // This would never happen.
452  pause4spin();
453  continue;
454  }
455  // trying to increase local reference counter w/ same serial.
456  if(const_cast<atomic_shared_ptr<T> *>(this)->m_ref.compare_set_strong(
457  RefLocal_((uintptr_t)pref + rcnt_old),
458  RefLocal_((uintptr_t)pref + rcnt_new)))
459  break;
460  }
461  assert(rcnt_new);
462  *rcnt = rcnt_new;
463  return pref;
464 }
465 template <typename T>
466 inline typename atomic_shared_ptr<T>::Ref *
467 atomic_shared_ptr<T>::scan_() const noexcept {
468  Refcnt rcnt;
469  Ref *pref = reserve_scan_( &rcnt);
470  if( !pref) return (Ref*)nullptr;
471  ++(pref->refcnt); //atomic
472  leave_scan_(pref);
473  return pref;
474 }
475 
476 template <typename T>
477 inline void
478 atomic_shared_ptr<T>::leave_scan_(Ref *pref) const noexcept {
479  for(;;) {
480  Refcnt rcnt_old;
481  rcnt_old = refcnt_();
482  if(rcnt_old) {
483  Refcnt rcnt_new = rcnt_old - 1;
484  // trying to dec. reference counter if stored pointer is unchanged.
485  if(const_cast<atomic_shared_ptr<T> *>(this)->m_ref.compare_set_strong(
486  RefLocal_((uintptr_t)pref + rcnt_old),
487  RefLocal_((uintptr_t)pref + rcnt_new)))
488  break;
489  if(pref == pref_())
490  continue; // try again.
491  }
492  // local reference has released by other processes.
493  if(pref->refcnt.decAndTest()) {
494  const_cast<atomic_shared_ptr*>(this)->deleter(pref);
495  }
496  break;
497  }
498 }
499 
500 template <typename T>
501 template <bool NOSWAP>
502 inline bool
504  Ref *pref;
505  if(newr.pref_()) {
506  ++(newr.pref_()->refcnt); //atomic
507  }
508  for(;;) {
509  Refcnt rcnt_old, rcnt_new;
510  pref = reserve_scan_( &rcnt_old);
511  if(pref != oldr.pref_()) {
512  if(pref) {
513  if( !NOSWAP) {
514  ++(pref->refcnt);//atomic
515  }
516  leave_scan_(pref);
517  }
518  if(newr.pref_())
519  --(newr.pref_()->refcnt); //atomic
520  if( !NOSWAP) {
521  if(oldr.pref_()) {
522  // decreasing global reference counter.
523  if(oldr.pref_()->refcnt.decAndTest()) {
524  this->deleter(oldr.pref_());
525  }
526  }
527  oldr.m_ref = (uintptr_t)pref;
528  }
529  return false;
530  }
531  if(pref && (rcnt_old != 1u)) {
532  pref->refcnt += rcnt_old - 1u; //atomic
533  }
534  rcnt_new = 0;
535  if(this->m_ref.compare_set_strong(
536  RefLocal_((uintptr_t)pref + rcnt_old),
537  RefLocal_((uintptr_t)newr.pref_() + rcnt_new)))
538  break;
539  if(pref) {
540  assert(rcnt_old);
541  if(rcnt_old != 1u)
542  pref->refcnt += (Refcnt)( -(int)(rcnt_old - 1u)); //atomic
543  leave_scan_(pref);
544  }
545  }
546  if(pref) {
547  --(pref->refcnt); //atomic
548  }
549  return true;
550 }
551 template <typename T>
552 bool
554  return compareAndSwap_<true>(const_cast<local_shared_ptr<T> &>(oldr), newr);
555 }
556 template <typename T>
557 bool
559  return compareAndSwap_<false>(oldr, newr);
560 }
561 template <typename T, typename reflocal_var_t>
562 inline void
564  RefLocal_ x = this->m_ref;
565  this->m_ref = (RefLocal_)r.m_ref;
566  r.m_ref = x;
567 }
568 
569 template <typename T, typename reflocal_var_t>
570 void
572  Ref *pref;
573  for(;;) {
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; //atomic
578  }
579  rcnt_new = 0;
580  if(r.m_ref.compare_set_strong(
581  RefLocal_((uintptr_t)pref + rcnt_old),
582  RefLocal_((uintptr_t)this->m_ref + rcnt_new)))
583  break;
584  if(pref) {
585  assert(rcnt_old);
586  if(rcnt_old != 1u)
587  pref->refcnt += (Refcnt)( -(int)(rcnt_old - 1u)); //atomic
588  r.leave_scan_(pref);
589  }
590  }
591  this->m_ref = (RefLocal_)pref;
592 }
593 
594 #endif /*ATOMIC_SMART_PTR_H_*/

Generated for KAME4 by  doxygen 1.8.3