| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 | /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at https://mozilla.org/MPL/2.0/. * * (c) ZeroTier, Inc. * https://www.zerotier.com/ */#ifndef ZT_SHAREDPTR_HPP#define ZT_SHAREDPTR_HPP#include "AtomicCounter.hpp"#include "Mutex.hpp"namespace ZeroTier {/** * Simple zero-overhead introspective reference counted pointer * * This is an introspective shared pointer. Classes that need to be reference * counted must list this as a 'friend' and must have a private instance of * AtomicCounter called __refCount. */template <typename T> class SharedPtr {  public:	SharedPtr() : _ptr((T*)0)	{	}	SharedPtr(T* obj) : _ptr(obj)	{		++obj->__refCount;	}	SharedPtr(const SharedPtr& sp) : _ptr(sp._getAndInc())	{	}	~SharedPtr()	{		if (_ptr) {			if (--_ptr->__refCount <= 0) {				delete _ptr;			}		}	}	inline SharedPtr& operator=(const SharedPtr& sp)	{		if (_ptr != sp._ptr) {			T* p = sp._getAndInc();			if (_ptr) {				if (--_ptr->__refCount <= 0) {					delete _ptr;				}			}			_ptr = p;		}		return *this;	}	/**	 * Set to a naked pointer and increment its reference count	 *	 * This assumes this SharedPtr is NULL and that ptr is not a 'zombie.' No	 * checks are performed.	 *	 * @param ptr Naked pointer to assign	 */	inline void set(T* ptr)	{		zero();		++ptr->__refCount;		_ptr = ptr;	}	/**	 * Swap with another pointer 'for free' without ref count overhead	 *	 * @param with Pointer to swap with	 */	inline void swap(SharedPtr& with)	{		T* tmp = _ptr;		_ptr = with._ptr;		with._ptr = tmp;	}	inline operator bool() const	{		return (_ptr != (T*)0);	}	inline T& operator*() const	{		return *_ptr;	}	inline T* operator->() const	{		return _ptr;	}	/**	 * @return Raw pointer to held object	 */	inline T* ptr() const	{		return _ptr;	}	/**	 * Set this pointer to NULL	 */	inline void zero()	{		if (_ptr) {			if (--_ptr->__refCount <= 0) {				delete _ptr;			}			_ptr = (T*)0;		}	}	/**	 * @return Number of references according to this object's ref count or 0 if NULL	 */	inline int references()	{		if (_ptr) {			return _ptr->__refCount.load();		}		return 0;	}	inline bool operator==(const SharedPtr& sp) const	{		return (_ptr == sp._ptr);	}	inline bool operator!=(const SharedPtr& sp) const	{		return (_ptr != sp._ptr);	}	inline bool operator>(const SharedPtr& sp) const	{		return (_ptr > sp._ptr);	}	inline bool operator<(const SharedPtr& sp) const	{		return (_ptr < sp._ptr);	}	inline bool operator>=(const SharedPtr& sp) const	{		return (_ptr >= sp._ptr);	}	inline bool operator<=(const SharedPtr& sp) const	{		return (_ptr <= sp._ptr);	}  private:	inline T* _getAndInc() const	{		if (_ptr) {			++_ptr->__refCount;		}		return _ptr;	}	T* _ptr;};}	// namespace ZeroTier#endif
 |