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

Symbian

开发平台:

Visual C++

  1. /* ***** BEGIN LICENSE BLOCK *****
  2.  * Source last modified: $Id: hxsmartptr.h,v 1.3.18.3 2004/07/09 01:45:13 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.   @header hxsmartptr.h
  51.   
  52.   @abstract Macro implementation of smart pointers to COM interfaces
  53.             Based on pnmiscpubsmartptr.h
  54.  
  55.   @discussion
  56.   A SmartPointer is a class that contains a pointer, but acts like it 
  57.   <em>is</em> the contained pointer.  When used it helps enforce correct usage 
  58.   of the pointer it contains.  In this case it provides the following 
  59.   safeguards:<ul>
  60.       <li>ASSERT if the contained pointer is NULL when you attempt to 
  61.   use it.
  62.       <li>ASSERT if you attempt to replace a valid pointer without 
  63.   releasing it.
  64.       <li>Automatically AddRef() a pointer during assignment.
  65.       <li>Automatically QI() an incoming pointer to the correct type 
  66.   during assignment.
  67.       <li>Automatically Release() an existing pointer during assignment.
  68.       <li>Automatically Release() an existing pointer during destruction.
  69.        </ul>
  70.  
  71.   SmartPointers also simplify usage. Example:
  72.  
  73.     <pre><code>
  74.     HX_SMART_POINTER_INLINE( SPIHXBuffer, IHXBuffer );
  75.     ReadDone(HX_RESULT status, IUnknown* pbufData)
  76.     {
  77.         SPIHXBufferPtr spBufData = pbufData;
  78.         if(spBufData.IsValid())
  79.         {
  80.             cout << spBufData->GetBuffer() << endl;
  81.         }
  82.     }
  83.     </code></pre>
  84.   This example has no memory leaks.  In the Assignment the IUnknown 
  85.   pointer is QI'd for the IHXBuffer.  If the QI had failed, then 
  86.   the IsValid test would fail too.  Even if the IsValid Test was 
  87.   omitted, spbufData->GetBuffer() would cause an assertion if 
  88.   the QI had failed.
  89.  
  90.    Usage Tips:
  91.  
  92.   Cannot use SmartPointers as List elements.
  93.  
  94.    Note that there are specific macros for creating smart pointers to 
  95.    IUnknowns - plugging IUnknown into the standard smart pointer 
  96.    generation macros will cause compile errors.
  97.  
  98.    The interface has been cut down to a minimum. In particular, the 
  99.    automatic conversion operators have been removed to avoid the 
  100.    possibility of unexpected conversions happening automatically. It 
  101.    is still possible to get to the raw pointer if it is necessary.
  102.  
  103.  */
  104. /*!
  105.     @class SPIUnknown
  106.     @description A smart pointer to an IUnknown. This will always contain the unique IUnknown 
  107.       - i.e. it will QI any pointer passed to it (including IUnknown pointers) to make sure it gets 
  108.       the unique IUnknown.
  109. */
  110. /*!
  111.     @class SPCIUnknown
  112.     @description A const smart pointer to an IUnknown. This will always contain the unique IUnknown 
  113.       - i.e. it will QI any pointer passed to it (including IUnknown pointers) to make sure it gets 
  114.       the unique IUnknown.
  115. */
  116. /*!
  117.     @function CLASS_NAME()
  118. */
  119. /*!
  120.     @function CLASS_NAME( const CLASS_NAME& rspact )
  121. */
  122. /*!
  123.     @function CLASS_NAME(INTERFACE* pact)
  124. */
  125. /*!
  126.     @function ~CLASS_NAME()
  127. */
  128. /*!
  129.     @function CLASS_NAME(IHXCommonClassFactory* pIObjectSource, REFCLSID clsid, HX_RESULT* pResult = NULL )
  130.     @description Create an object from a factory and attempt to assign it 
  131.       to this smart pointer. If the created object supports the interface 
  132.       this smart pointer refers, the smart pointer will point to that object. 
  133.       If the created object does not support the interface, the smart pointer 
  134.       will end up as NULL and the created object will be immediately destroyed.
  135.     @param pIObjectSource The class factory to use
  136.     @param clsid The clsid of the object to create
  137.     @param pResult The result of the CreateInstance call - if this parameter 
  138.       is NULL the result will not be returned.
  139. */
  140. /*!
  141.     @function INTERFACE* operator -> () const
  142. */
  143. /*!
  144.     @function INTERFACE& operator * () const
  145. */
  146. /*!
  147.     @function CLASS_NAME& operator =( const CLASS_NAME& rspact )
  148. */
  149. /*!
  150.     @function CLASS_NAME& operator =( INTERFACE* pNew )
  151. */
  152. /*!
  153.     @function BOOL IsValid() const
  154. */
  155. /*!
  156.     @function INTERFACE* Ptr() const
  157.     @description Returns a non-addrefed version of the internal 
  158.       pointer. This is designed to be used as an easy way of 
  159.       passing the pointer to functions that take an unadorned 
  160.       pointer. If the pointer is assigned to a local variable, 
  161.       please bear in mind that it is not addrefed and should 
  162.       be treated accordingly. The best way to get an AddReffed 
  163.       pointer assigned to a local variable is to use the AsPtr 
  164.       method.
  165. */
  166. /*!
  167.     @function AsPtr( INTERFACE*  ) const
  168.     @description Returns an addrefed version of the internal 
  169.       pointer. Onlu available on non-const smart pointers (if
  170.       this was available for a const smart pointer it would 
  171.       have to return a const pointer which then couldn't be 
  172.       released ).
  173. */
  174. /*!
  175.     @function void AsUnknown( IUnknown** ppIUnknown ) const
  176.     @description Returns an addrefed version of the pointer as an 
  177.       IUnknown*.
  178. */
  179. /*!
  180.      @function Query( IUnknown* pIUnk )
  181.      @description If possible, set the value of this smart pointer to the
  182. given pointer. The value of this smart pointer will only be changed if the
  183. supplied pointer is non-NULL and it supports the interface corresponding to
  184. this smart pointer. If either of these conditions is not true the value of
  185. the smart pointer will be unchanged.
  186.      @param pIUnk The pointer to try and assign to this.
  187.      @result There are three possible results:
  188.          1. If the pointer passed in was non-NULL and the object supported the
  189.     appropriate interface, a SUCCEEDED result will be returned (corresponding
  190.     to the result of the internal QI).
  191.          2. If the pointer passed in was non_NULL but the object did not 
  192.     support the appropriate interface, a FAILED result will be returned 
  193.     (corresponding to the result of the internal QI).
  194.          3. If the pointer passed in was NULL, HXR_INVALID_PARAMETER will be returned.
  195. */
  196. /*!
  197.      @function Clear()
  198.      @description Releases the smart pointers reference to the underlying pointer and 
  199.        set the smart pointer to NULL.
  200. */
  201. /*!
  202.      @function AsInOutParam()
  203.      @description Clears the smart pointer, then returns a pointer to the underlying 
  204.        pointer. This is suitable for passing to functions which pass back an AddReffed 
  205.        pointer as output. E.g.
  206.       void SomeFunction( IHXSomeInterface** ppOutput );
  207.       SPIHXSomeInterface spI;
  208.       SomeFunction( spi>AsInOutParam() );
  209. */
  210. #ifndef _HXSMARTPTR_H_
  211. #define _HXSMARTPTR_H_
  212. #include "hxassert.h"
  213. #include "hxcom.h"
  214. #include "hxccf.h"
  215. /*!
  216.     @defined HX_SMART_POINTER_INLINE
  217.     @abstract 
  218. Declare a smart pointer of name CLASS_NAME which points to an 
  219. interface INTERFACE. Inline definitions of all functions.
  220. */
  221. #define HX_SMART_POINTER_INLINE( CLASS_NAME, INTERFACE )
  222. HX_PRIVATE_SMART_POINTER_INLINE( CLASS_NAME, INTERFACE, HX_PRIVATE_BLANK, HX_PRIVATE_NON_IUNKNOWN_FUNCTIONS( CLASS_NAME, INTERFACE, HX_PRIVATE_BLANK ) )
  223. /*!
  224.     @defined HX_CONST_SMART_POINTER_INLINE
  225.     @abstract 
  226. Declare a const smart pointer of name CLASS_NAME which points to an 
  227. interface INTERFACE. Inline definitions of all functions.
  228. */
  229. #define HX_CONST_SMART_POINTER_INLINE( CLASS_NAME, INTERFACE )
  230. HX_PRIVATE_SMART_POINTER_INLINE( CLASS_NAME, INTERFACE, const, HX_PRIVATE_NON_IUNKNOWN_FUNCTIONS( CLASS_NAME, INTERFACE, const ) )
  231. /*!
  232.     @defined HX_IUNKNOWN_SMART_POINTER_INLINE
  233.     @abstract 
  234. Declare a smart pointer of name CLASS_NAME which points to an 
  235. IUnknown interface. Inline definitions of all functions.
  236. */
  237. #define HX_IUNKNOWN_SMART_POINTER_INLINE( CLASS_NAME )
  238. HX_PRIVATE_SMART_POINTER_INLINE( CLASS_NAME, IUnknown, HX_PRIVATE_BLANK, HX_PRIVATE_BLANK )
  239. /*!
  240.     @defined HX_IUNKNOWN_CONST_SMART_POINTER_INLINE
  241.     @abstract 
  242. Declare a const smart pointer of name CLASS_NAME which points to an 
  243. IUnknown interface. Inline definitions of all functions.
  244. */
  245. #define HX_IUNKNOWN_CONST_SMART_POINTER_INLINE( CLASS_NAME )
  246. HX_PRIVATE_SMART_POINTER_INLINE( CLASS_NAME, IUnknown, const, HX_PRIVATE_BLANK )
  247. // ---------------------------------------------------------------------
  248. // The macros below here should not be used directly. They might 
  249. // change in future implementations
  250. // ---------------------------------------------------------------------
  251. // The IUnknown functions macros have been included 
  252. // to reduce code duplication. If we just attempt to use the normal 
  253. // macros for creating an IUnknown smart pointer we end up with compile 
  254. // errors because the copy constructor and assignment ops are duplicated.
  255. // The HX_PRIVATE_NON_IUNKNOWN_FUNCTIONS macro separates out a copy 
  256. // constructor / assignment op that are needed in the non-IUnknown versions 
  257. // but would cause errors in the IUnknown versions.
  258. #define HX_PRIVATE_NON_IUNKNOWN_FUNCTIONS( CLASS_NAME, INTERFACE, QUALIFIERS )
  259.     CLASS_NAME(INTERFACE QUALIFIERS* pact)
  260.     : pActual(NULL)
  261.     {
  262. if( pact )
  263. {
  264.     pActual = (INTERFACE*) pact;
  265.     pActual->AddRef();
  266. }
  267.     }
  268.     CLASS_NAME& operator =( INTERFACE QUALIFIERS* pNew )
  269.     {
  270. if( pNew != pActual )
  271. {
  272.     INTERFACE* pTemp = pActual;
  273.     pActual = (INTERFACE*) pNew;
  274.     if( pActual )
  275.     {
  276. pActual->AddRef();
  277.     }
  278.     HX_RELEASE( pTemp );
  279. }
  280. return *this;
  281.     }
  282. #define HX_PRIVATE_SMART_POINTER_INLINE( CLASS_NAME, INTERFACE, QUALIFIERS, NON_IUNKNOWN_DEFINITIONS )
  283. class CLASS_NAME
  284. {
  285. public:
  286.     CLASS_NAME()
  287. : pActual(NULL)
  288.     {}
  289.     CLASS_NAME( const CLASS_NAME& rspact )
  290.     : pActual(NULL)
  291.     {
  292. if( rspact.pActual )
  293. {
  294.     pActual = rspact.pActual;
  295.     pActual->AddRef();
  296. }
  297.     }
  298.     CLASS_NAME(IUnknown QUALIFIERS* punk)
  299.     : pActual(NULL)
  300.     {
  301.         if(punk)
  302.     ((IUnknown*) punk)->QueryInterface( IID_##INTERFACE, ( void** )( &pActual ) );
  303.     }
  304.     CLASS_NAME& operator =( IUnknown QUALIFIERS* punkNew )
  305.     {
  306. if( punkNew != pActual )
  307. {
  308.     INTERFACE* pTemp = pActual;
  309.     if(punkNew)
  310.     {
  311. ((IUnknown*) punkNew)->QueryInterface( IID_##INTERFACE, ( void** )( &pActual ) );
  312.     }
  313.     else
  314.     {
  315. pActual = NULL;
  316.     }
  317.     HX_RELEASE( pTemp );
  318. }
  319. return *this;
  320.     }
  321.     CLASS_NAME( IHXCommonClassFactory* pIObjectSource, REFCLSID clsid, HX_RESULT* pResult = NULL ) 
  322.     : pActual(NULL)
  323.     {
  324. HX_ASSERT(pIObjectSource);
  325. IUnknown* pIUnk = NULL;
  326. HX_RESULT result = pIObjectSource->CreateInstance( clsid, (void**)&pIUnk );
  327. if (SUCCEEDED(result))
  328. {
  329.     HX_ASSERT(pIUnk);
  330.     result = pIUnk->QueryInterface( IID_##INTERFACE, ( void** )( &pActual ) );
  331.     pIUnk->Release();
  332. }
  333. if( pResult )
  334. {
  335.     *pResult = result;
  336. }
  337.     }
  338.     NON_IUNKNOWN_DEFINITIONS
  339.     ~CLASS_NAME()
  340.     { HX_RELEASE (pActual); }
  341.     class INTERFACE##_InternalSP : public INTERFACE
  342.     {
  343.     private:
  344. virtual ULONG32 STDMETHODCALLTYPE AddRef () PURE;
  345. virtual ULONG32 STDMETHODCALLTYPE Release () PURE;
  346.     };
  347.     INTERFACE##_InternalSP QUALIFIERS* operator -> () const
  348.     {HX_ASSERT(pActual);return (INTERFACE##_InternalSP*) pActual;}
  349.     INTERFACE##_InternalSP QUALIFIERS& operator * () const
  350.     {HX_ASSERT(pActual);return *((INTERFACE##_InternalSP*) pActual);}
  351.     CLASS_NAME& operator =( const CLASS_NAME& rspact )
  352.     {*this = rspact.pActual; return *this;}
  353.     HX_RESULT Query(IUnknown QUALIFIERS* punkNew)
  354.     {
  355. HX_RESULT result = HXR_INVALID_PARAMETER;
  356. if( punkNew )
  357. {
  358.     result = HXR_OK;
  359.     if( punkNew != pActual )
  360.     {
  361. INTERFACE* pTemp = NULL;
  362. result = ((IUnknown*) punkNew)->QueryInterface( IID_##INTERFACE, ( void** )( &pTemp ) );
  363. if( SUCCEEDED( result ) )
  364. {
  365.     HX_RELEASE( pActual );
  366.     pActual = pTemp;
  367. }
  368.     }
  369. }
  370. return result;
  371.     }
  372.     BOOL IsValid() const
  373.     {return pActual!=NULL;}
  374.     INTERFACE** AsInOutParam () 
  375.     { HX_RELEASE (pActual); return &pActual; }
  376.     INTERFACE QUALIFIERS* Ptr() const
  377.     {return pActual;}
  378.     void AsPtr( INTERFACE QUALIFIERS** ppActual ) const
  379.     {
  380. HX_ASSERT(pActual);
  381. pActual->AddRef ();
  382. *ppActual = pActual;
  383.     }
  384.     void AsUnknown( IUnknown QUALIFIERS** ppIUnknown ) const
  385.     {
  386. HX_ASSERT(pActual);
  387. pActual->QueryInterface
  388. (
  389.     IID_IUnknown,
  390.     ( void** )ppIUnknown
  391. );
  392.     }
  393.     void Clear ()
  394.     { HX_RELEASE (pActual); }
  395. private:
  396.     INTERFACE*     pActual;
  397. }
  398. #define HX_PRIVATE_BLANK
  399. HX_IUNKNOWN_SMART_POINTER_INLINE( SPIUnknown );
  400. HX_IUNKNOWN_CONST_SMART_POINTER_INLINE( SPCIUnknown );
  401. // Since we are guaranteed that SPIUnknown and SPCIUnknown will be storing 
  402. // the unique IUnknown pointer we can just compare the results of the Ptr function.
  403. inline BOOL operator ==( const SPIUnknown& sp1, const SPIUnknown& sp2 )
  404. {
  405.     return sp1.Ptr() == sp2.Ptr();
  406. }
  407. inline BOOL operator ==( const SPCIUnknown& spc1, const SPCIUnknown& spc2 )
  408. {
  409.     return spc1.Ptr() == spc2.Ptr();
  410. }
  411. /* These cannot be activated without letting everybody know. Some code might
  412.  fail to compile: 
  413.   IRCAClickable* p = 0;
  414.   if (p == SPIUnknown(p))  //is ambigous
  415.   { ... }
  416.  
  417.  
  418. inline BOOL operator ==( const SPIUnknown& sp1, const SPCIUnknown& spc2 )
  419. {
  420.     return sp1.Ptr() == spc2.Ptr();
  421. }
  422. inline BOOL operator ==( const SPCIUnknown& spc1, const SPIUnknown& sp2 )
  423. {
  424.     return spc1.Ptr() == sp2.Ptr();
  425. }
  426. */
  427. inline BOOL operator !=( const SPIUnknown& sp1, const SPIUnknown& sp2 )
  428. {
  429.     return ! ( sp1 == sp2 );
  430. }
  431. inline BOOL operator !=( const SPCIUnknown& spc1, const SPCIUnknown& spc2 )
  432. {
  433.     return ! ( spc1 == spc2 );
  434. }
  435. /*
  436. inline BOOL operator !=( const SPIUnknown& sp1, const SPCIUnknown& spc2 )
  437. {
  438.     return ! ( sp1 == spc2 );
  439. }
  440. inline BOOL operator !=( const SPCIUnknown& spc1, const SPIUnknown& sp2 )
  441. {
  442.     return ! ( spc1 == sp2 );
  443. }
  444. */
  445. #endif // _HXSMARTPTR_H_