You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdesvn/src/svnqt/smart_pointer.h

147 lines
4.5 KiB

/***************************************************************************
* Copyright (C) 2006-2007 by Rajko Albrecht *
* ral@alwins-world.de *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 2 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program; if not, write to the *
* Free Software Foundation, Inc., *
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *
***************************************************************************/
#ifndef _smart_pointer_hpp
#define _smart_pointer_hpp
#if defined TQT_THREAD_SUPPORT
#include "tqmutex.h"
#endif
#include "svnqt/svnqt_defines.h"
/*!
* \file smart_pointer.h
* \brief smart pointer and reference counter
* \author Rajko Albrecht
*
*/
namespace svn
{
//! simple reference counter class
class ref_count {
protected:
//! reference count member
long m_RefCount;
#ifdef TQT_THREAD_SUPPORT
TQMutex m_RefcountMutex;
#endif
public:
//! first reference must be added after "new" via Pointer()
ref_count() : m_RefCount(0)
#ifdef TQT_THREAD_SUPPORT
,m_RefcountMutex()
#endif
{}
virtual ~ref_count() {}
//! add a reference
void Incr() {
#ifdef TQT_THREAD_SUPPORT
TQMutexLocker a(&m_RefcountMutex);
#endif
++m_RefCount;
}
//! delete a reference
bool Decr() {
#ifdef TQT_THREAD_SUPPORT
TQMutexLocker a(&m_RefcountMutex);
#endif
--m_RefCount;
return Shared();
}
//! is it referenced
bool Shared() { return (m_RefCount > 0); }
};
//! reference counting wrapper class
template<class T> class smart_pointer {
//! pointer to object
/*!
* this object must contain Incr(), Decr() and Shared()
* methode as public members. The best way is, that it will be a child
* class of RefCount
*/
T *ptr;
public:
//! standart constructor
smart_pointer() { ptr = 0; }
//! standart destructor
/*!
* release the reference, if it were the last reference, destroys
* ptr
*/
~smart_pointer()
{
if (ptr && !ptr->Decr()) {
delete ptr;
}
}
//! construction
smart_pointer(T* t) { if ( (ptr = t) ) ptr->Incr(); }
//! Pointer copy
smart_pointer(const smart_pointer<T>& p)
{ if ( (ptr = p.ptr) ) ptr->Incr(); }
//! pointer copy by assignment
smart_pointer<T>& operator= (const smart_pointer<T>& p)
{
// already same: nothing to do
if (ptr == p.ptr) return *this;
// decouple reference
if ( ptr && !ptr->Decr()) delete ptr;
// establish new reference
if ( (ptr = p.ptr) ) ptr->Incr();
return *this;
}
smart_pointer<T>& operator= (T*p)
{
if (ptr==p)return *this;
if (ptr && !ptr->Decr()) delete ptr;
if ( (ptr=p) ) ptr->Incr();
return *this;
}
//! cast to conventional pointer
operator T* () const { return ptr; }
//! deref: fails for 0 pointer
T& operator* () {return *ptr; }
//! deref: fails for 0 pointer
const T& operator* ()const {return *ptr; }
//! deref with method call
T* operator-> () {return ptr; }
//! deref with const method call
const T* operator-> ()const {return ptr; }
//! supports "if (pointer)"
operator bool () const { return (ptr != 0); }
//! "if (pointer)" as non const
operator bool () { return ptr != 0;}
//! support if (!pointer)"
bool operator! () const { return (ptr == 0); }
//! support if (!pointer)" as non const
bool operator! () { return (ptr == 0); }
};
} // namespace svn
#endif