smartptr.h
上传用户:dangjiwu
上传日期:2013-07-19
资源大小:42019k
文件大小:12k
源码类别:

Symbian

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK *****
  2.  * Source last modified: $Id: smartptr.h,v 1.3.36.3 2004/07/09 01:48:00 hubbe Exp $
  3.  * 
  4.  * Portions Copyright (c) 1995-2004 RealNetworks, Inc. All Rights Reserved.
  5.  * 
  6.  * The contents of this file, and the files included with this file,
  7.  * are subject to the current version of the RealNetworks Public
  8.  * Source License (the "RPSL") available at
  9.  * http://www.helixcommunity.org/content/rpsl unless you have licensed
  10.  * the file under the current version of the RealNetworks Community
  11.  * Source License (the "RCSL") available at
  12.  * http://www.helixcommunity.org/content/rcsl, in which case the RCSL
  13.  * will apply. You may also obtain the license terms directly from
  14.  * RealNetworks.  You may not use this file except in compliance with
  15.  * the RPSL or, if you have a valid RCSL with RealNetworks applicable
  16.  * to this file, the RCSL.  Please see the applicable RPSL or RCSL for
  17.  * the rights, obligations and limitations governing use of the
  18.  * contents of the file.
  19.  * 
  20.  * Alternatively, the contents of this file may be used under the
  21.  * terms of the GNU General Public License Version 2 or later (the
  22.  * "GPL") in which case the provisions of the GPL are applicable
  23.  * instead of those above. If you wish to allow use of your version of
  24.  * this file only under the terms of the GPL, and not to allow others
  25.  * to use your version of this file under the terms of either the RPSL
  26.  * or RCSL, indicate your decision by deleting the provisions above
  27.  * and replace them with the notice and other provisions required by
  28.  * the GPL. If you do not delete the provisions above, a recipient may
  29.  * use your version of this file under the terms of any one of the
  30.  * RPSL, the RCSL or the GPL.
  31.  * 
  32.  * This file is part of the Helix DNA Technology. RealNetworks is the
  33.  * developer of the Original Code and owns the copyrights in the
  34.  * portions it created.
  35.  * 
  36.  * This file, and the files included with this file, is distributed
  37.  * and made available on an 'AS IS' basis, WITHOUT WARRANTY OF ANY
  38.  * KIND, EITHER EXPRESS OR IMPLIED, AND REALNETWORKS HEREBY DISCLAIMS
  39.  * ALL SUCH WARRANTIES, INCLUDING WITHOUT LIMITATION, ANY WARRANTIES
  40.  * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, QUIET
  41.  * ENJOYMENT OR NON-INFRINGEMENT.
  42.  * 
  43.  * Technology Compatibility Kit Test Suite(s) Location:
  44.  *    http://www.helixcommunity.org/content/tck
  45.  * 
  46.  * Contributor(s):
  47.  * 
  48.  * ***** END LICENSE BLOCK ***** */
  49. //
  50. //
  51. // Macro implementation of "Smart Pointers" to COM interfaces
  52. //
  53. //  Description:
  54. //
  55. // A SmartPointer is a class that contains a pointer, but acts like it 
  56. // *is* the contained pointer.  When used it helps enforce correct usage 
  57. // of the pointer it contains.  In this case it provides the following 
  58. // safeguards:
  59. //     HX_ASSERT if the contained pointer is NULL when you attempt to 
  60. // use it.
  61. //     HX_ASSERT if you attempt to replace a valid pointer without 
  62. // releasing it.
  63. //     Automatically AddRef() a pointer during assignment.
  64. //     Automatically QI() an incoming pointer to the correct type 
  65. // during assignment.
  66. //     Automatically Release() an existing pointer during assignment.
  67. //     Automatically Release() an existing pointer during destruction.
  68. //
  69. // SmartPointers also simplify usage. Example:
  70. //
  71. //     ReadDone(HX_RESULT status, IUnknown* pbufData)
  72. //     {
  73. // DECLARE_SMART_POINTER(IHXBuffer) spbufData;
  74. //
  75. // spbufData = pbufData;
  76. //
  77. // if(spbufData.IsValid())
  78. // {
  79. //     // Use the Pointer
  80. //     cout << spbufData->GetBuffer() << endl;
  81. // }
  82. //     }
  83. // This example has no memory leaks.  In the Assignment the IUnknown 
  84. // pointer is QI'd for the IHXBuffer.  If the QI had failed, then 
  85. // the IsValid test would fail too.  Even if the IsValid Test was 
  86. // forgotten, spbufData->GetBuffer() would cause an assertion if 
  87. // the QI had failed.
  88. //
  89. //  Usage Tips:
  90. //
  91. // Should Not use SmartPointers as function parameters. ie:
  92. //     ReadDone(HX_RESULT status, DECLARE_SMART_POINTER_UNKNOWN spbufData)
  93. // Cannot use SmartPointers as List elements.
  94. //
  95. //  Macros Defined in this file:
  96. //
  97. // DECLARE_SMART_POINTER_UNKNOWN
  98. //     This Macro is used to create a SmartPointer to an IUnknown 
  99. //     interface.
  100. // Example:    DECLARE_SMART_POINTER_UNKNOWN  m_spunkContext;
  101. //
  102. // DEFINE_SMART_POINTER(IACTUAL)
  103. //     This Macro is used to define a SmartPointer class for any 
  104. //     interface other than IUnknown.  It should be used in the header 
  105. //     file where the interface is defined.
  106. // Example:    DEFINE_SMART_POINTER(IHXBuffer);
  107. //
  108. // IMPLEMENT_SMART_POINTER(IACTUAL)
  109. //     This Macro is used to Implement a SmartPointer class for any 
  110. //     interface other than IUnknown.  It should be used in a cpp 
  111. //     file that is linked into each binary.
  112. // Example:    IMPLEMENT_SMART_POINTER(IHXBuffer)
  113. //
  114. // DECLARE_SMART_POINTER(IACTUAL)
  115. //     This Macro is used to create a SmartPointer to any interface other 
  116. //     than IUnknown.  The header file that contains the call to
  117. //     IMPLEMENT_SMART_POINTER(IACTUAL) must be included before this macro
  118. //     can be used.
  119. // Example:    DECLARE_SMART_POINTER(IHXBuffer)  m_spbufData;
  120. //
  121. //
  122. #ifndef _SMARTPTR_H_
  123. #define _SMARTPTR_H_
  124. #include "hxassert.h"
  125. class _CIUnknown_SP
  126. {
  127. public:
  128.     _CIUnknown_SP() : m_pUnk(NULL)
  129.     {}
  130.     _CIUnknown_SP(_CIUnknown_SP& rspunk) : m_pUnk(NULL)
  131.     {*this = rspunk.m_pUnk;}
  132.     _CIUnknown_SP(IUnknown* punk) : m_pUnk(NULL)
  133.     {*this = punk;}
  134.     ~_CIUnknown_SP()
  135.     {Release();}
  136.     operator IUnknown* ()
  137.     {return m_pUnk;}
  138.     operator const IUnknown* () const
  139.     {return m_pUnk;}
  140.     IUnknown* operator -> () const
  141.     {HX_ASSERT(m_pUnk);return m_pUnk;}
  142.     IUnknown& operator * () const
  143.     {HX_ASSERT(m_pUnk);return *m_pUnk;}
  144.     IUnknown** operator & ()
  145.     {HX_ASSERT(!m_pUnk);return &m_pUnk;}
  146.     _CIUnknown_SP& operator = (IUnknown* punkNew)
  147.     {
  148. Release();
  149. if(punkNew)
  150. {
  151.     m_pUnk = punkNew;
  152.     m_pUnk->AddRef();
  153. }
  154. return *this;
  155.     }
  156.     BOOL operator ! () const
  157.     {return !IsValid();}
  158.     BOOL IsValid() const
  159.     {return (m_pUnk!=NULL);}
  160.     IUnknown*& ptr_reference()
  161.     {HX_ASSERT(!m_pUnk);return m_pUnk;}
  162.     IUnknown* AsUnknown() const
  163.     {
  164. HX_ASSERT(m_pUnk);
  165. IUnknown* punkOut = m_pUnk;
  166. punkOut->AddRef();
  167. return punkOut;
  168.     }
  169.     void Release()
  170.     {HX_RELEASE(m_pUnk);}
  171. private:
  172.     IUnknown*     m_pUnk;
  173. };
  174. #define DECLARE_SMART_POINTER_UNKNOWN 
  175.     _CIUnknown_SP
  176. #define DEFINE_SMART_POINTER(IACTUAL)
  177. class _C##IACTUAL##_SP
  178. {
  179. public:
  180.     _C##IACTUAL##_SP();
  181.     _C##IACTUAL##_SP(_C##IACTUAL##_SP& rspact);
  182.     _C##IACTUAL##_SP(IACTUAL* pact);
  183.     _C##IACTUAL##_SP(IUnknown* punk);
  184.     ~_C##IACTUAL##_SP();
  185.     operator IACTUAL* ();
  186.     operator const IACTUAL* () const;
  187.     IACTUAL* operator -> () const;
  188.     IACTUAL& operator * () const;
  189.     IACTUAL** operator & ();
  190.     _C##IACTUAL##_SP& operator = (IACTUAL* pNew);
  191.     _C##IACTUAL##_SP& operator = (IUnknown* punkNew);
  192.     BOOL operator ! () const;
  193.     BOOL IsValid() const;
  194.     IACTUAL*& ptr_reference();
  195.     IUnknown* AsUnknown() const;
  196.     void Release();
  197. private:
  198.     IACTUAL*     pActual;
  199. };
  200. #define IMPLEMENT_SMART_POINTER(IACTUAL)
  201.     DEFINE_SMART_POINTER(IACTUAL)
  202.     _C##IACTUAL##_SP::_C##IACTUAL##_SP()
  203. : pActual(NULL)
  204.     {}
  205.     _C##IACTUAL##_SP::_C##IACTUAL##_SP
  206.     (
  207. _C##IACTUAL##_SP& rspact
  208.     )
  209. : pActual(NULL)
  210.     {*this = rspact.pActual;}
  211.     _C##IACTUAL##_SP::_C##IACTUAL##_SP(IACTUAL* pact)
  212. : pActual(NULL)
  213.     {*this = pact;}
  214.     _C##IACTUAL##_SP::_C##IACTUAL##_SP(IUnknown* punk)
  215. : pActual(NULL)
  216.     {*this = punk;}
  217.     _C##IACTUAL##_SP::~_C##IACTUAL##_SP()
  218.     {Release();}
  219.     _C##IACTUAL##_SP::operator IACTUAL* ()
  220.     {return pActual;}
  221.     _C##IACTUAL##_SP::operator const IACTUAL* () const
  222.     {return pActual;}
  223.     IACTUAL* _C##IACTUAL##_SP::operator -> () const
  224.     {HX_ASSERT(pActual);return pActual;}
  225.     IACTUAL& _C##IACTUAL##_SP::operator * () const
  226.     {HX_ASSERT(pActual);return *pActual;}
  227.     IACTUAL** _C##IACTUAL##_SP::operator & ()
  228.     {HX_ASSERT(!pActual);return &pActual;}
  229.     _C##IACTUAL##_SP& _C##IACTUAL##_SP::operator =
  230.     (
  231. IACTUAL* pNew
  232.     )
  233.     {
  234. Release();
  235. if(pNew)
  236. {
  237.     pActual = pNew;
  238.     pActual->AddRef();
  239. }
  240. return *this;
  241.     }
  242.     _C##IACTUAL##_SP& _C##IACTUAL##_SP::operator = 
  243.     (
  244. IUnknown* punkNew
  245.     )
  246.     {
  247. Release();
  248. if(punkNew)
  249. {
  250.     punkNew->QueryInterface
  251.     (
  252. IID_##IACTUAL,
  253. (void**)&pActual
  254.     );
  255. }
  256. return *this;
  257.     }
  258.     BOOL _C##IACTUAL##_SP::operator ! () const
  259.     {return !IsValid();}
  260.     BOOL _C##IACTUAL##_SP::IsValid() const
  261.     {return pActual!=NULL;}
  262.     IACTUAL*& _C##IACTUAL##_SP::ptr_reference()
  263.     {HX_ASSERT(!pActual);return pActual;}
  264.     IUnknown* _C##IACTUAL##_SP::AsUnknown() const
  265.     {
  266. HX_ASSERT(pActual);
  267. IUnknown* punkOut;
  268. pActual->QueryInterface
  269. (
  270. IID_IUnknown,
  271. (void**)&punkOut
  272. );
  273. return punkOut;
  274.     }
  275.     void _C##IACTUAL##_SP::Release()
  276.     {HX_RELEASE(pActual);}
  277. #define DECLARE_SMART_POINTER(IACTUAL) 
  278.     _C##IACTUAL##_SP
  279. #define DEFINE_WRAPPED_POINTER(CACTUAL)
  280. class _C##CACTUAL##_WP
  281. {
  282. public:
  283.     _C##CACTUAL##_WP();
  284.     _C##CACTUAL##_WP(const _C##CACTUAL##_WP& rspact);
  285.     _C##CACTUAL##_WP(const _C##CACTUAL##_SP& rspact);
  286.     _C##CACTUAL##_WP(const CACTUAL* pActual);
  287.     ~_C##CACTUAL##_WP();
  288.     _C##CACTUAL##_WP& operator=
  289.     (
  290. const _C##CACTUAL##_WP& rspact
  291.     );
  292.     void wrapped_ptr(const CACTUAL* pNew);
  293.     const CACTUAL* wrapped_ptr() const;
  294.     CACTUAL* wrapped_ptr();
  295.     BOOL IsValid() const;
  296.     void Empty();
  297. private:
  298.     CACTUAL*     pActual;
  299. };
  300. #define IMPLEMENT_WRAPPED_POINTER(CACTUAL)
  301.     DEFINE_WRAPPED_POINTER(CACTUAL)
  302.     _C##CACTUAL##_WP::_C##CACTUAL##_WP()
  303. : pActual(NULL)
  304.     {}
  305.     _C##CACTUAL##_WP::_C##CACTUAL##_WP
  306.     (
  307. const _C##CACTUAL##_WP& rspact
  308.     )
  309. : pActual(NULL)
  310.     {wrapped_ptr(rspact.pActual);}
  311.     _C##CACTUAL##_WP::_C##CACTUAL##_WP
  312.     (
  313. const _C##CACTUAL##_SP& rspact
  314.     )
  315. : pActual(NULL)
  316.     {wrapped_ptr(rspact);}
  317.     _C##CACTUAL##_WP::_C##CACTUAL##_WP
  318.     (
  319. const CACTUAL* pActual
  320.     )
  321. : pActual(NULL)
  322.     {wrapped_ptr(pActual);}
  323.     _C##CACTUAL##_WP::~_C##CACTUAL##_WP()
  324.     {Empty();}
  325.     _C##CACTUAL##_WP& 
  326.     _C##CACTUAL##_WP::operator=
  327.     (
  328. const _C##CACTUAL##_WP& rspact
  329.     )
  330.     {wrapped_ptr(rspact.pActual); return *this;}
  331.     void _C##CACTUAL##_WP::wrapped_ptr
  332.     (
  333. const CACTUAL* pNew
  334.     )
  335.     {
  336. Empty();
  337. if(pNew)
  338. {
  339.     pActual = (CACTUAL*)pNew;
  340.     pActual->AddRef();
  341. }
  342. return;
  343.     }
  344.     const CACTUAL* _C##CACTUAL##_WP::wrapped_ptr() const
  345.     {
  346. HX_ASSERT(pActual);
  347. return pActual;
  348.     }
  349.     CACTUAL* _C##CACTUAL##_WP::wrapped_ptr()
  350.     {
  351. HX_ASSERT(pActual);
  352. return pActual;
  353.     }
  354.     BOOL _C##CACTUAL##_WP::IsValid() const
  355.     {return (pActual != NULL);}
  356.     void _C##CACTUAL##_WP::Empty()
  357.     {HX_RELEASE(pActual);}
  358. #define WRAPPED_POINTER(CACTUAL) 
  359.     _C##CACTUAL##_WP
  360. #endif // _SMARTPTR_H_