8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-02-01 23:20:39 +01:00
firebird-mirror/src/common/classes/RefCounted.h

212 lines
3.3 KiB
C
Raw Normal View History

/*
* The contents of this file are subject to the Initial
* Developer's Public License Version 1.0 (the "License");
* you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
* See the License for the specific language governing rights
* and limitations under the License.
*
* The Original Code was created by Vlad Horsun
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2008 Vlad Horsun <hvlad@users.sf.net>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
* Dmitry Yemanov <dimitr@users.sf.net>
*/
#ifndef COMMON_REF_COUNTED_H
#define COMMON_REF_COUNTED_H
#include "../common/classes/fb_atomic.h"
#include "../common/gdsassert.h"
namespace Firebird
{
class RefCounted
{
public:
virtual int addRef()
{
return ++m_refCnt;
}
virtual int release()
{
fb_assert(m_refCnt.value() > 0);
const int refCnt = --m_refCnt;
if (!refCnt)
delete this;
return refCnt;
}
protected:
RefCounted() : m_refCnt(0) {}
2008-03-24 11:36:11 +01:00
2008-03-24 16:18:26 +01:00
virtual ~RefCounted()
2008-03-24 11:36:11 +01:00
{
fb_assert(m_refCnt.value() == 0);
2008-03-24 11:36:11 +01:00
}
private:
2008-03-21 03:42:24 +01:00
AtomicCounter m_refCnt;
};
// reference counted object guard
class Reference
{
public:
2008-03-28 14:25:47 +01:00
explicit Reference(RefCounted& refCounted) :
2008-04-06 23:58:08 +02:00
r(refCounted)
{
2008-04-06 23:58:08 +02:00
r.addRef();
}
~Reference()
{
try {
2008-04-06 23:58:08 +02:00
r.release();
}
catch (const Exception&)
{
DtorException::devHalt();
}
}
private:
2008-04-06 23:58:08 +02:00
RefCounted& r;
};
// controls reference counter of the object where points
template <typename T>
class RefPtr
{
public:
RefPtr() : ptr(NULL)
{ }
2008-03-28 14:25:47 +01:00
explicit RefPtr(T* p) : ptr(p)
{
if (ptr)
{
ptr->addRef();
}
}
RefPtr(const RefPtr& r) : ptr(r.ptr)
2008-04-15 10:53:54 +02:00
{
if (ptr)
{
ptr->addRef();
}
}
~RefPtr()
{
if (ptr)
{
ptr->release();
}
}
T* operator=(T* p)
{
return assign(p);
}
T* operator=(const RefPtr& r)
{
2008-04-15 10:53:54 +02:00
return assign(r.ptr);
}
2008-12-05 01:56:15 +01:00
operator T*()
{
return ptr;
}
2008-12-05 01:56:15 +01:00
T* operator->()
2008-04-06 23:58:08 +02:00
{
return ptr;
}
operator const T*() const
{
return ptr;
}
const T* operator->() const
{
return ptr;
}
2009-09-03 03:28:54 +02:00
/* NS: you cannot have operator bool here. It creates ambiguity with
operator T* with some of the compilers (at least VS2003)
2008-04-15 10:53:54 +02:00
operator bool() const
{
return ptr ? true : false;
}*/
2008-04-15 10:53:54 +02:00
bool hasData() const
{
return ptr ? true : false;
}
2008-04-15 10:53:54 +02:00
bool operator !() const
{
2008-04-16 11:25:36 +02:00
return !ptr;
2008-12-05 01:56:15 +01:00
}
2008-04-15 10:53:54 +02:00
bool operator ==(const RefPtr& r) const
{
return ptr == r.ptr;
2008-12-05 01:56:15 +01:00
}
2008-04-15 10:53:54 +02:00
bool operator !=(const RefPtr& r) const
{
return ptr != r.ptr;
2008-12-05 01:56:15 +01:00
}
2008-04-15 10:53:54 +02:00
private:
2008-04-06 23:58:08 +02:00
T* assign(T* const p)
{
if (ptr != p)
{
if (p)
{
p->addRef();
}
if (ptr)
{
ptr->release();
}
2008-03-30 22:43:36 +02:00
ptr = p;
}
2008-03-30 22:43:36 +02:00
return ptr;
}
T* ptr;
};
2009-02-01 23:03:10 +01:00
template <typename T>
class AnyRef : public T, public RefCounted
{
public:
inline AnyRef() : T() {}
inline AnyRef(const T& v) : T(v) {}
inline explicit AnyRef(MemoryPool& p) : T(p) {}
inline AnyRef(MemoryPool& p, const T& v) : T(p, v) {}
};
2008-03-21 03:42:24 +01:00
} // namespace
#endif // COMMON_REF_COUNTED_H