// 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 { public: RefCount(); explicit RefCount(Pointer *p); explicit RefCount(RefCount ©); RefCount(const RefCount ©); ~RefCount(); RefCount &operator = (const RefCount ©); RefCount &operator = (Pointer *p); Pointer *operator * () const { return get(); } Pointer *operator -> () const { return get(); } Pointer *get() const { return m_data; } #ifdef NOT_USED /// @return number of referenses unsigned int usedBy() const { return (m_refcount != 0 ? *m_refcount : 0); } #endif private: /// increase reference count void incRefCount(); /// decrease reference count void decRefCount(); /// decrease refcount count void decRefCountCount(); Pointer *m_data; ///< data holder mutable unsigned int *m_refcount; ///< holds reference counting // This one counts the number of active references pointing to the m_refcount data! // when it reaches zero, *then* we can delete it, otherwise someone else might check it. mutable unsigned int *m_refcount_count; ///< holds reference counting }; // implementation template RefCount::RefCount():m_data(0), m_refcount(new unsigned int(0)), m_refcount_count(new unsigned int(1)) { } template RefCount::RefCount(RefCount ©): m_data(copy.m_data), m_refcount(copy.m_refcount), m_refcount_count(copy.m_refcount_count) { (*m_refcount_count)++; incRefCount(); } template RefCount::RefCount(Pointer *p):m_data(p), m_refcount(new unsigned int(0)), m_refcount_count(new unsigned int(1)) { incRefCount(); } template RefCount::RefCount(const RefCount ©): m_data(copy.m_data), m_refcount(copy.m_refcount), m_refcount_count(copy.m_refcount_count) { (*m_refcount_count)++; 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_refcount_count = copy.m_refcount_count; (*m_refcount_count)++; m_data = copy.m_data; // set new data pointer incRefCount(); // inc new ref count return *this; } template RefCount &RefCount::operator = (Pointer *p) { decRefCount(); m_data = p; // set data pointer m_refcount = new unsigned int(0); // create new counter m_refcount_count = new unsigned int(1); incRefCount(); return *this; } template void RefCount::decRefCount() { if (m_refcount != 0) { (*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; } } decRefCountCount(); } template void RefCount::decRefCountCount() { if (*m_refcount_count == 0) return; // shouldnt happen (*m_refcount_count)--; if (*m_refcount_count == 0) { delete m_refcount; delete m_refcount_count; } m_refcount = 0; m_refcount_count = 0; } template void RefCount::incRefCount() { if (m_refcount == 0) return; (*m_refcount)++; } } // end namespace FbTk #endif // FBTK_REFCOUNT_HH