|
Shinya Kitaoka |
810553 |
#pragma once
|
|
Shinya Kitaoka |
810553 |
|
|
Toshihiro Shimizu |
890ddd |
#ifndef TSMARTPOINTER_INCLUDED
|
|
Toshihiro Shimizu |
890ddd |
#define TSMARTPOINTER_INCLUDED
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#include "tutil.h"
|
|
Toshihiro Shimizu |
890ddd |
#include "tatomicvar.h"
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#undef DVAPI
|
|
Toshihiro Shimizu |
890ddd |
#undef DVVAR
|
|
Toshihiro Shimizu |
890ddd |
#ifdef TNZCORE_EXPORTS
|
|
Toshihiro Shimizu |
890ddd |
#define DVAPI DV_EXPORT_API
|
|
Toshihiro Shimizu |
890ddd |
#define DVVAR DV_EXPORT_VAR
|
|
Toshihiro Shimizu |
890ddd |
#else
|
|
Toshihiro Shimizu |
890ddd |
#define DVAPI DV_IMPORT_API
|
|
Toshihiro Shimizu |
890ddd |
#define DVVAR DV_IMPORT_VAR
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#ifndef NDEBUG
|
|
Toshihiro Shimizu |
890ddd |
#define INSTANCE_COUNT_ENABLED
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
class DVAPI TSmartObject {
|
|
Shinya Kitaoka |
120a6e |
TAtomicVar m_refCount;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
const TINT32 m_classCodeRef;
|
|
Shinya Kitaoka |
120a6e |
static const TINT32 m_unknownClassCode;
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
public:
|
|
Shinya Kitaoka |
120a6e |
typedef short ClassCode;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
TSmartObject(ClassCode
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
classCode
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
)
|
|
Shinya Kitaoka |
120a6e |
: m_refCount()
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
, m_classCodeRef(classCode)
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
{
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
incrementInstanceCount();
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
TSmartObject()
|
|
Shinya Kitaoka |
120a6e |
: m_refCount()
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
, m_classCodeRef(m_unknownClassCode)
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
{
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
incrementInstanceCount();
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
virtual ~TSmartObject() {
|
|
Shinya Kitaoka |
120a6e |
assert(m_refCount == 0);
|
|
Toshihiro Shimizu |
890ddd |
#ifdef INSTANCE_COUNT_ENABLED
|
|
Shinya Kitaoka |
120a6e |
decrementInstanceCount();
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
inline void addRef() { ++m_refCount; }
|
|
Shinya Kitaoka |
120a6e |
inline void release() {
|
|
Shinya Kitaoka |
120a6e |
if ((--m_refCount) <= 0) delete this;
|
|
Shinya Kitaoka |
120a6e |
};
|
|
Shinya Kitaoka |
120a6e |
inline TINT32 getRefCount() const { return m_refCount; }
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
static TINT32 getInstanceCount(ClassCode code);
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
private:
|
|
Shinya Kitaoka |
120a6e |
void incrementInstanceCount();
|
|
Shinya Kitaoka |
120a6e |
void decrementInstanceCount();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
private:
|
|
Shinya Kitaoka |
120a6e |
|
|
Shinya Kitaoka |
120a6e |
TSmartObject(const TSmartObject &);
|
|
Shinya Kitaoka |
120a6e |
TSmartObject &operator=(const TSmartObject &);
|
|
Toshihiro Shimizu |
890ddd |
};
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
#define DECLARE_CLASS_CODE \
|
|
shun-iwasawa |
5b7b9f |
\
|
|
Shinya Kitaoka |
120a6e |
private: \
|
|
Shinya Kitaoka |
120a6e |
static const TSmartObject::ClassCode m_classCode; \
|
|
shun-iwasawa |
5b7b9f |
\
|
|
Shinya Kitaoka |
120a6e |
public: \
|
|
Shinya Kitaoka |
120a6e |
inline static TINT32 getInstanceCount() { \
|
|
Shinya Kitaoka |
120a6e |
return TSmartObject::getInstanceCount(m_classCode); \
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
#define DEFINE_CLASS_CODE(T, ID) \
|
|
Shinya Kitaoka |
120a6e |
const TSmartObject::ClassCode T::m_classCode = ID;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
template <class t=""></class>
|
|
 |
29623f |
class TSmartPointerBaseT {
|
|
 |
29623f |
public:
|
|
 |
29623f |
typedef T Type;
|
|
Toshihiro Shimizu |
890ddd |
protected:
|
|
Shinya Kitaoka |
120a6e |
T *m_pointer;
|
|
 |
29623f |
T& reference() const
|
|
 |
29623f |
{ assert(m_pointer); return *m_pointer; }
|
|
Toshihiro Shimizu |
890ddd |
public:
|
|
 |
29623f |
TSmartPointerBaseT() : m_pointer(0) {}
|
|
 |
29623f |
TSmartPointerBaseT(const TSmartPointerBaseT &src) : m_pointer(src.m_pointer)
|
|
 |
29623f |
{ if (m_pointer) m_pointer->addRef(); }
|
|
 |
29623f |
TSmartPointerBaseT(T *pointer) : m_pointer(pointer)
|
|
 |
29623f |
{ if (m_pointer) m_pointer->addRef(); }
|
|
 |
29623f |
virtual ~TSmartPointerBaseT()
|
|
 |
29623f |
{ if (m_pointer) { m_pointer->release(); m_pointer = 0; } }
|
|
 |
29623f |
TSmartPointerBaseT& operator=(const TSmartPointerBaseT &src)
|
|
 |
29623f |
{ set(src); return *this; }
|
|
 |
29623f |
|
|
 |
29623f |
void set(T *pointer) {
|
|
 |
29623f |
if (m_pointer != pointer) {
|
|
 |
29623f |
// call 'addRef' before 'release'
|
|
 |
29623f |
if (pointer) pointer->addRef();
|
|
 |
29623f |
if (m_pointer) m_pointer->release();
|
|
 |
29623f |
m_pointer = pointer;
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Shinya Kitaoka |
120a6e |
}
|
|
 |
29623f |
void set(const TSmartPointerBaseT &src)
|
|
 |
29623f |
{ set(src.m_pointer); }
|
|
 |
29623f |
bool operator!() const
|
|
 |
29623f |
{ return m_pointer == 0; }
|
|
 |
29623f |
operator bool() const
|
|
 |
29623f |
{ return m_pointer != 0; }
|
|
 |
29623f |
bool operator==(const TSmartPointerBaseT &p) const
|
|
 |
29623f |
{ return m_pointer == p.m_pointer; }
|
|
 |
29623f |
bool operator!=(const TSmartPointerBaseT &p) const
|
|
 |
29623f |
{ return m_pointer != p.m_pointer; }
|
|
 |
29623f |
bool operator<(const TSmartPointerBaseT &p) const
|
|
 |
29623f |
{ return m_pointer < p.m_pointer; }
|
|
 |
29623f |
bool operator>(const TSmartPointerBaseT &p) const
|
|
 |
29623f |
{ return m_pointer > p.m_pointer; }
|
|
 |
29623f |
};
|
|
Shinya Kitaoka |
120a6e |
|
|
 |
29623f |
|
|
Shinya Kitaoka |
120a6e |
|
|
 |
29623f |
template <class t=""></class>
|
|
 |
29623f |
class TSmartPointerT: public TSmartPointerBaseT<t> {</t>
|
|
 |
29623f |
public:
|
|
 |
29623f |
typedef TSmartPointerBaseT<t> Base;</t>
|
|
 |
29623f |
TSmartPointerT() {}
|
|
 |
29623f |
TSmartPointerT(const TSmartPointerT &src): Base(src) {}
|
|
 |
29623f |
TSmartPointerT(T *pointer): Base(pointer) {}
|
|
 |
29623f |
TSmartPointerT& operator=(const TSmartPointerT &src) { Base::set(src); return *this; }
|
|
 |
29623f |
T* operator->() const { return &Base::reference(); }
|
|
 |
29623f |
T& operator*() const { return Base::reference(); }
|
|
 |
29623f |
T* getPointer() const { return Base::m_pointer; }
|
|
 |
29623f |
};
|
|
Shinya Kitaoka |
120a6e |
|
|
 |
29623f |
|
|
Shinya Kitaoka |
120a6e |
|
|
 |
29623f |
//! smart reference returns constant pointer when reference is constant
|
|
 |
29623f |
//! and returns non-constant pointer when reference is non-constant
|
|
 |
29623f |
template <class t=""></class>
|
|
 |
29623f |
class TSmartRefT: public TSmartPointerBaseT<t> {</t>
|
|
 |
29623f |
public:
|
|
 |
29623f |
typedef TSmartPointerBaseT<t> Base;</t>
|
|
 |
29623f |
TSmartRefT() {}
|
|
 |
29623f |
TSmartRefT(const TSmartRefT &src): Base(src) {}
|
|
 |
29623f |
TSmartRefT(T *pointer): Base(pointer) {}
|
|
 |
29623f |
TSmartRefT& operator=(const TSmartRefT &src) { Base::set(src); return *this; }
|
|
 |
29623f |
const T* operator->() const { return &Base::reference(); }
|
|
 |
29623f |
const T& operator*() const { return Base::reference(); }
|
|
 |
29623f |
const T* getPointer() const { return Base::m_pointer; }
|
|
 |
29623f |
T* operator->() { return &Base::reference(); }
|
|
 |
29623f |
T& operator*() { return Base::reference(); }
|
|
 |
29623f |
T* getPointer() { return Base::m_pointer; }
|
|
Toshihiro Shimizu |
890ddd |
};
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
template <class base="" class="" derived,=""></class>
|
|
 |
e280ae |
class TDerivedSmartPointerT : public TSmartPointerT<derived> {</derived>
|
|
Toshihiro Shimizu |
890ddd |
public:
|
|
Shinya Kitaoka |
120a6e |
typedef TDerivedSmartPointerT<derived, base=""> DerivedSmartPointer;</derived,>
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
TDerivedSmartPointerT(){};
|
|
Shinya Kitaoka |
120a6e |
TDerivedSmartPointerT(DERIVED *pointer) : TSmartPointerT<derived>(pointer) {}</derived>
|
|
Toshihiro Shimizu |
890ddd |
|
|
Shinya Kitaoka |
120a6e |
TDerivedSmartPointerT(const TSmartPointerT<base> &p) {
|
|
Shinya Kitaoka |
120a6e |
TSmartPointerT<derived>::m_pointer =</derived>
|
|
Shinya Kitaoka |
120a6e |
dynamic_cast<derived *="">(p.getPointer());</derived>
|
|
Shinya Kitaoka |
120a6e |
if (TSmartPointerT<derived>::m_pointer)</derived>
|
|
Shinya Kitaoka |
120a6e |
TSmartPointerT<derived>::m_pointer->addRef();</derived>
|
|
Shinya Kitaoka |
120a6e |
}
|
|
Toshihiro Shimizu |
890ddd |
};
|
|
Toshihiro Shimizu |
890ddd |
|
|
 |
71b43d |
|
|
 |
71b43d |
|
|
 |
71b43d |
typedef TSmartPointerT<tsmartobject> TSmartObjectP;</tsmartobject>
|
|
 |
71b43d |
|
|
Toshihiro Shimizu |
890ddd |
#endif
|