// RefCount.hh for FbTk - Fluxbox Toolkit // Copyright (c) 2002 Henrik Kinnunen (fluxgen at fluxbox dot org) // // Permission is hereby granted, free of charge, to any person obtaining a // copy of this software and associated documentation files (the "Software"), // to deal in the Software without restriction, including without limitation // the rights to use, copy, modify, merge, publish, distribute, sublicense, // and/or sell copies of the Software, and to permit persons to whom the // Software is furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL // THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER // DEALINGS IN THE SOFTWARE. #ifndef FBTK_REFCOUNT_HH #define FBTK_REFCOUNT_HH namespace FbTk { /// holds a pointer with reference counting, similar to std:auto_ptr template class RefCount { typedef Pointer* RefCount::*bool_type; public: RefCount(); explicit RefCount(Pointer *p); RefCount(const RefCount ©); template RefCount(const RefCount ©); ~RefCount(); RefCount &operator = (const RefCount ©); Pointer &operator * () const { return *get(); } Pointer *operator -> () const { return get(); } Pointer *get() const { return m_data; } void reset(Pointer *p = 0); /// conversion to "bool" operator bool_type() const { return m_data ? &RefCount::m_data : 0; } private: /// increase reference count void incRefCount(); /// decrease reference count void decRefCount(); Pointer *m_data; ///< data holder unsigned int *m_refcount; ///< holds reference counting // we need this for the template copy constructor template friend class RefCount; }; // implementation template RefCount::RefCount():m_data(0), m_refcount(new unsigned int(0)) { incRefCount(); // it really counts how many things are storing m_refcount } template template RefCount::RefCount(const RefCount ©): m_data(copy.m_data), m_refcount(copy.m_refcount) { incRefCount(); } template RefCount::RefCount(Pointer *p):m_data(p), m_refcount(new unsigned int(0)) { incRefCount(); } template RefCount::RefCount(const RefCount ©): m_data(copy.m_data), m_refcount(copy.m_refcount) { incRefCount(); } template RefCount::~RefCount() { decRefCount(); } template RefCount &RefCount::operator = (const RefCount ©) { decRefCount(); // dec current ref count m_refcount = copy.m_refcount; // set new ref count m_data = copy.m_data; // set new data pointer incRefCount(); // inc new ref count return *this; } template void RefCount::reset(Pointer *p) { decRefCount(); m_data = p; // set data pointer m_refcount = new unsigned int(0); // create new counter incRefCount(); } template void RefCount::decRefCount() { if (m_refcount == 0) return; if (*m_refcount == 0) { // already zero, then delete refcount delete m_refcount; m_refcount = 0; return; } (*m_refcount)--; if (*m_refcount == 0) { // destroy m_data and m_refcount if nobody else is using this if (m_data != 0) delete m_data; m_data = 0; delete m_refcount; m_refcount = 0; } } template void RefCount::incRefCount() { if (m_refcount == 0) return; (*m_refcount)++; } template inline RefCount makeRef() { return RefCount(new Pointer); } template inline RefCount makeRef(const Arg1 &arg1) { return RefCount(new Pointer(arg1)); } template inline RefCount makeRef(const Arg1 &arg1, const Arg2 &arg2) { return RefCount(new Pointer(arg1, arg2)); } template inline RefCount makeRef(const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3) { return RefCount(new Pointer(arg1, arg2, arg3)); } template inline bool operator == (const RefCount &a, const RefCount &b) { return a.get() == b.get(); } template inline bool operator != (const RefCount &a, const RefCount &b) { return a.get() != b.get(); } template inline bool operator < (const RefCount &a, const RefCount &b) { return a.get() < b.get(); } template inline bool operator > (const RefCount &a, const RefCount &b) { return a.get() > b.get(); } template inline bool operator <= (const RefCount &a, const RefCount &b) { return a.get() <= b.get(); } template inline bool operator >= (const RefCount &a, const RefCount &b) { return a.get() >= b.get(); } } // end namespace FbTk #endif // FBTK_REFCOUNT_HH