123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232 |
- #if !defined(AFX_REFERENCEOBJECT_H__BAEBCE9D_CD68_40AF_8A54_B23A0D14E807__INCLUDED_)
- #define AFX_REFERENCEOBJECT_H__BAEBCE9D_CD68_40AF_8A54_B23A0D14E807__INCLUDED_
- #if _MSC_VER > 1000
- #pragma once
- #endif
- #ifdef WIN32
- #define WIN32_LEAN_AND_MEAN
- #include "windows.h"
- #endif
- #include "assert.h"
- #include "limits.h"
- #include "ReferenceCounter.h"
- template<class HEADER_T, class DATA_T>
- class ReferenceObjectBlock
- {
- public:
- DATA_T* pData;
- mutable ReferenceCounter ReferenceCount;
- unsigned int AllocLength;
- HEADER_T Header;
- };
- template<class HEADER_T, class DATA_T>
- class ReferenceObject
- {
- public:
- ReferenceObject();
- ReferenceObject(unsigned int size);
- ReferenceObject(const ReferenceObject<HEADER_T,DATA_T>& other);
- virtual ~ReferenceObject();
- public:
- HEADER_T& getHeader();
- DATA_T* getData();
- ReferenceObject<HEADER_T,DATA_T>& operator=(const ReferenceObject<HEADER_T,DATA_T>& other);
- protected:
- void AllocBlock(unsigned int size);
- void AllocHeader();
- void AllocData(unsigned int size);
- virtual void Release();
- virtual void InitHeader();
- virtual void FreeHeader();
- virtual void CopyForWrite();
- virtual void CopyForWrite(unsigned int size);
- ReferenceObjectBlock<HEADER_T,DATA_T>* m_pData;
- };
- template<class HEADER_T, class DATA_T>
- ReferenceObject<HEADER_T,DATA_T>::ReferenceObject()
- {
- m_pData = NULL;
- AllocHeader();
- InitHeader();
- }
- template<class HEADER_T, class DATA_T>
- ReferenceObject<HEADER_T,DATA_T>::ReferenceObject(unsigned int size)
- {
- m_pData = NULL;
- AllocBlock(size);
- }
- template<class HEADER_T, class DATA_T>
- ReferenceObject<HEADER_T,DATA_T>::ReferenceObject(const ReferenceObject<HEADER_T,DATA_T>& other)
- {
- m_pData = other.m_pData;
- m_pData->ReferenceCount++;
- }
- template<class HEADER_T, class DATA_T>
- ReferenceObject<HEADER_T,DATA_T>::~ReferenceObject()
- {
- Release();
- }
- template<class HEADER_T, class DATA_T>
- ReferenceObject<HEADER_T,DATA_T>& ReferenceObject<HEADER_T,DATA_T>::operator=(const ReferenceObject<HEADER_T,DATA_T>& other)
- {
- if (m_pData != other.m_pData)
- {
- Release();
- m_pData = other.m_pData;
- m_pData->ReferenceCount++;
- }
- return *this;
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::Release()
- {
- assert(m_pData != NULL);
- if (m_pData->ReferenceCount.dec() <= 0)
- {
- FreeHeader();
- delete[] m_pData->pData;
- delete m_pData;
- m_pData = NULL;
- }
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::AllocBlock(unsigned int size)
- {
- AllocHeader();
- AllocData(size);
- InitHeader();
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::AllocHeader()
- {
- m_pData = new ReferenceObjectBlock<HEADER_T,DATA_T>;
- m_pData->ReferenceCount = 1;
- m_pData->pData = NULL;
- m_pData->AllocLength = 0;
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::AllocData(unsigned int size)
- {
- assert(m_pData != NULL);
- assert((size * sizeof(DATA_T)) < INT_MAX);
- if (size)
- {
- m_pData->pData = new DATA_T[size];
- }
- else
- {
- m_pData->pData = NULL;
- }
- m_pData->AllocLength = size;
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::InitHeader()
- {
-
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::FreeHeader()
- {
-
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::CopyForWrite()
- {
- CopyForWrite(m_pData->AllocLength);
- }
- template<class HEADER_T, class DATA_T>
- void ReferenceObject<HEADER_T,DATA_T>::CopyForWrite(unsigned int size)
- {
- unsigned int oldsize = m_pData->AllocLength;
- if (size)
- {
- if ((m_pData->ReferenceCount > 1) || (size != oldsize))
- {
- ReferenceObjectBlock<HEADER_T,DATA_T>* pTmp = m_pData;
- AllocBlock(size);
- m_pData->Header = pTmp->Header;
- unsigned int sizetocopy = min(size,oldsize);
-
- for (unsigned int x=0;x<sizetocopy;x++)
- {
- m_pData->pData[x] = pTmp->pData[x];
- }
- if (pTmp->ReferenceCount.dec() <= 0)
- {
- delete[] pTmp->pData;
- delete pTmp;
- }
- }
- }
- else
- {
- Release();
- AllocHeader();
- InitHeader();
- }
- }
- template<class HEADER_T, class DATA_T>
- HEADER_T& ReferenceObject<HEADER_T,DATA_T>::getHeader()
- {
- CopyForWrite();
- return m_pData->Header;
- }
- template<class HEADER_T, class DATA_T>
- DATA_T* ReferenceObject<HEADER_T,DATA_T>::getData()
- {
- CopyForWrite();
- return m_pData->pData;
- }
- #endif
|