Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1682 lines
44 KiB

  1. /////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright (c) 1997-2000 Microsoft Corporation
  4. //
  5. // Module Name:
  6. // Cluster.cpp
  7. //
  8. // Description:
  9. // Implementation of the cluster and application classes and other
  10. // support classes for the MSCLUS automation classes.
  11. //
  12. // Author:
  13. // Charles Stacy Harris (stacyh) 28-Feb-1997
  14. // Galen Barbee (galenb) July 1998
  15. //
  16. // Revision History:
  17. // July 1998 GalenB Maaaaaajjjjjjjjjoooooorrrr clean up
  18. //
  19. // Notes:
  20. //
  21. /////////////////////////////////////////////////////////////////////////////
  22. #include "stdafx.h"
  23. #include "ClusterObject.h"
  24. #include "property.h"
  25. #include "ClusRes.h"
  26. #include "ClusNeti.h"
  27. #include "ClusResg.h"
  28. #include "ClusRest.h"
  29. #include "ClusNode.h"
  30. #include "ClusNetw.h"
  31. #include "ClusApp.h"
  32. #include "version.h"
  33. #include "cluster.h"
  34. /////////////////////////////////////////////////////////////////////////////
  35. // Global variables
  36. /////////////////////////////////////////////////////////////////////////////
  37. static const IID * iidCClusRefObject[] =
  38. {
  39. &IID_ISClusRefObject
  40. };
  41. static const IID * iidCCluster[] =
  42. {
  43. &IID_ISCluster
  44. };
  45. //*************************************************************************//
  46. /////////////////////////////////////////////////////////////////////////////
  47. // CClusRefObject class
  48. /////////////////////////////////////////////////////////////////////////////
  49. /////////////////////////////////////////////////////////////////////////////
  50. //++
  51. //
  52. // CClusRefObject::CClusRefObject
  53. //
  54. // Description:
  55. // Constructor.
  56. //
  57. // Arguments:
  58. // None.
  59. //
  60. // Return Value:
  61. // None.
  62. //
  63. //--
  64. /////////////////////////////////////////////////////////////////////////////
  65. CClusRefObject::CClusRefObject( void )
  66. {
  67. m_hCluster = NULL;
  68. m_piids = (const IID *) iidCClusRefObject;
  69. m_piidsSize = ARRAYSIZE( iidCClusRefObject );
  70. } //*** CClusRefObject::CClusRefObject( void )
  71. /////////////////////////////////////////////////////////////////////////////
  72. //++
  73. //
  74. // CClusRefObject::~CClusRefObject
  75. //
  76. // Description:
  77. // Destructor.
  78. //
  79. // Arguments:
  80. // None.
  81. //
  82. // Return Value:
  83. // None.
  84. //
  85. //--
  86. /////////////////////////////////////////////////////////////////////////////
  87. CClusRefObject::~CClusRefObject( void )
  88. {
  89. if ( m_hCluster != NULL )
  90. {
  91. ::CloseCluster( m_hCluster );
  92. m_hCluster = NULL;
  93. }
  94. } //*** CClusRefObject::~CClusRefObject()
  95. /////////////////////////////////////////////////////////////////////////////
  96. //++
  97. //
  98. // CCluster::ClusRefObject
  99. //
  100. // Description:
  101. // Copy constructor -- sort of.
  102. //
  103. // Arguments:
  104. // pClusRefObject [IN] - Cluster handle wrapper to hold copy.
  105. //
  106. // Return Value:
  107. // None.
  108. //
  109. //--
  110. /////////////////////////////////////////////////////////////////////////////
  111. void CCluster::ClusRefObject( IN ISClusRefObject * pClusRefObject )
  112. {
  113. ASSERT( pClusRefObject != NULL );
  114. if ( pClusRefObject != NULL )
  115. {
  116. if ( m_pClusRefObject != NULL )
  117. {
  118. m_pClusRefObject->Release();
  119. m_pClusRefObject = NULL;
  120. } // if:
  121. m_pClusRefObject = pClusRefObject;
  122. m_pClusRefObject->AddRef();
  123. } // if: args are not NULL
  124. } //*** CCluster::ClusRefObject( pClusRefObject )
  125. /////////////////////////////////////////////////////////////////////////////
  126. //++
  127. //
  128. // CCluster::Hcluster
  129. //
  130. // Description:
  131. // Changes the raw cluster handle that this class holds onto.
  132. //
  133. // Arguments:
  134. // hCluster [IN] - The new cluster handle.
  135. //
  136. // Return Value:
  137. // None.
  138. //
  139. //--
  140. /////////////////////////////////////////////////////////////////////////////
  141. void CCluster::Hcluster( IN HCLUSTER hCluster )
  142. {
  143. ASSERT( hCluster != NULL );
  144. if ( hCluster != NULL )
  145. {
  146. if ( m_hCluster != NULL )
  147. {
  148. ::CloseCluster( m_hCluster );
  149. m_hCluster = NULL;
  150. } // if:
  151. m_hCluster = hCluster;
  152. } // if:
  153. } //*** CCluster::Hcluster()
  154. /////////////////////////////////////////////////////////////////////////////
  155. //++
  156. //
  157. // CClusRefObject::get_Handle
  158. //
  159. // Description:
  160. // Returns the raw cluster handle.
  161. //
  162. // Arguments:
  163. // phandle [OUT] - Catches the cluster handle.
  164. //
  165. // Return Value:
  166. // S_OK if successful, or E_POINTER.
  167. //
  168. //--
  169. /////////////////////////////////////////////////////////////////////////////
  170. STDMETHODIMP CClusRefObject::get_Handle( OUT ULONG_PTR * phandle )
  171. {
  172. //ASSERT( phandle != NULL );
  173. HRESULT _hr = E_POINTER;
  174. if ( phandle != NULL )
  175. {
  176. _hr = E_HANDLE;
  177. if ( m_hCluster != NULL )
  178. {
  179. *phandle = (ULONG_PTR) m_hCluster;
  180. _hr = S_OK;
  181. } // if: cluster handle is not NULL
  182. } // if: args are not NULL
  183. return _hr;
  184. } //*** CClusRefObject::get_Handle()
  185. //*************************************************************************//
  186. /////////////////////////////////////////////////////////////////////////////
  187. // CCluster class
  188. /////////////////////////////////////////////////////////////////////////////
  189. /////////////////////////////////////////////////////////////////////////////
  190. //++
  191. //
  192. // CCluster::CCluster
  193. //
  194. // Description:
  195. // Constructor.
  196. //
  197. // Arguments:
  198. // None.
  199. //
  200. // Return Value:
  201. // None.
  202. //
  203. //--
  204. /////////////////////////////////////////////////////////////////////////////
  205. CCluster::CCluster( void )
  206. {
  207. // Initializing all data members.
  208. m_hCluster = NULL;
  209. m_pClusterNodes = NULL;
  210. m_pClusterResourceGroups = NULL;
  211. m_pClusterResources = NULL;
  212. m_pResourceTypes = NULL;
  213. m_pNetworks = NULL;
  214. m_pNetInterfaces = NULL;
  215. m_pClusRefObject = NULL;
  216. m_nQuorumLogSize = -1;
  217. m_pCommonProperties = NULL;
  218. m_pPrivateProperties = NULL;
  219. m_pCommonROProperties = NULL;
  220. m_pPrivateROProperties = NULL;
  221. m_pParentApplication = NULL;
  222. m_piids = (const IID *) iidCCluster;
  223. m_piidsSize = ARRAYSIZE( iidCCluster );
  224. } //*** CCluster::CCluster()
  225. /////////////////////////////////////////////////////////////////////////////
  226. //++
  227. //
  228. // CCluster::~CCluster
  229. //
  230. // Description:
  231. // Destructor.
  232. //
  233. // Arguments:
  234. // None.
  235. //
  236. // Return Value:
  237. // None.
  238. //
  239. //--
  240. /////////////////////////////////////////////////////////////////////////////
  241. CCluster::~CCluster( void )
  242. {
  243. Clear();
  244. } //*** CCluster::~CCluster()
  245. /////////////////////////////////////////////////////////////////////////////
  246. //++
  247. //
  248. // CCluster::Clear
  249. //
  250. // Description:
  251. // Clean out all of the collections we are hanging onto.
  252. //
  253. // Arguments:
  254. // None.
  255. //
  256. // Return Value:
  257. // None.
  258. //
  259. //--
  260. /////////////////////////////////////////////////////////////////////////////
  261. void CCluster::Clear( void )
  262. {
  263. if ( m_pParentApplication != NULL )
  264. {
  265. m_pParentApplication->Release();
  266. m_pParentApplication = NULL;
  267. }
  268. if ( m_pClusterNodes != NULL )
  269. {
  270. m_pClusterNodes->Release();
  271. m_pClusterNodes = NULL;
  272. }
  273. if ( m_pClusterResourceGroups != NULL )
  274. {
  275. m_pClusterResourceGroups->Release();
  276. m_pClusterResourceGroups = NULL;
  277. }
  278. if ( m_pClusterResources != NULL )
  279. {
  280. m_pClusterResources->Release();
  281. m_pClusterResources = NULL;
  282. }
  283. if ( m_pResourceTypes != NULL )
  284. {
  285. m_pResourceTypes->Release();
  286. m_pResourceTypes = NULL;
  287. }
  288. if ( m_pNetworks != NULL )
  289. {
  290. m_pNetworks->Release();
  291. m_pNetworks = NULL;
  292. }
  293. if ( m_pNetInterfaces != NULL )
  294. {
  295. m_pNetInterfaces->Release();
  296. m_pNetInterfaces = NULL;
  297. }
  298. if ( m_pCommonProperties != NULL )
  299. {
  300. m_pCommonProperties->Release();
  301. m_pCommonProperties = NULL;
  302. } // if: release the property collection
  303. if ( m_pPrivateProperties != NULL )
  304. {
  305. m_pPrivateProperties->Release();
  306. m_pPrivateProperties = NULL;
  307. } // if: release the property collection
  308. if ( m_pCommonROProperties != NULL )
  309. {
  310. m_pCommonROProperties->Release();
  311. m_pCommonROProperties = NULL;
  312. } // if: release the property collection
  313. if ( m_pPrivateROProperties != NULL )
  314. {
  315. m_pPrivateROProperties->Release();
  316. m_pPrivateROProperties = NULL;
  317. } // if: release the property collection
  318. if ( m_pClusRefObject != NULL )
  319. {
  320. m_pClusRefObject->Release();
  321. m_pClusRefObject = NULL;
  322. }
  323. m_hCluster = NULL;
  324. } //*** CCluster::Clear()
  325. /////////////////////////////////////////////////////////////////////////////
  326. //++
  327. //
  328. // CCluster::Create
  329. //
  330. // Description:
  331. // Complete heavy weight construction.
  332. //
  333. // Arguments:
  334. // pParentApplication [IN] - The parent ClusApplication object.
  335. //
  336. // Return Value:
  337. // S_OK if successful, E_POINTER, or other HRESULT error.
  338. //
  339. //--
  340. /////////////////////////////////////////////////////////////////////////////
  341. STDMETHODIMP CCluster::Create( IN CClusApplication * pParentApplication )
  342. {
  343. //ASSERT( pParentApplication != NULL );
  344. HRESULT _hr = E_POINTER;
  345. if ( pParentApplication != NULL )
  346. {
  347. _hr = pParentApplication->_InternalQueryInterface( IID_ISClusApplication, (void **) &m_pParentApplication );
  348. } // if: args are not NULL
  349. return _hr;
  350. } //*** CCluster::Create()
  351. /////////////////////////////////////////////////////////////////////////////
  352. //++
  353. //
  354. // CCluster::Open
  355. //
  356. // Description:
  357. // Open the cluster whose name is in bstrClusterName.
  358. //
  359. // Arguments:
  360. // bstrCluserName [IN] - Cluster name.
  361. //
  362. // Return Value:
  363. // S_OK if successful, or E_POINTER if the cluster is already open.
  364. // Win32 errors passed back as HRESULT.
  365. //
  366. //--
  367. /////////////////////////////////////////////////////////////////////////////
  368. STDMETHODIMP CCluster::Open( IN BSTR bstrClusterName )
  369. {
  370. //ASSERT( bstrClusterName != NULL );
  371. //ASSERT( m_hCluster == NULL );
  372. HRESULT _hr = E_POINTER;
  373. if ( bstrClusterName != NULL )
  374. {
  375. _hr = E_HANDLE;
  376. if ( m_hCluster == NULL )
  377. {
  378. _hr = S_OK;
  379. m_hCluster = ::OpenCluster( bstrClusterName );
  380. if ( m_hCluster == NULL )
  381. {
  382. DWORD _sc = GetLastError();
  383. _hr = HRESULT_FROM_WIN32( _sc );
  384. } // if: was the cluster opened?
  385. else
  386. {
  387. CComObject< CClusRefObject > * pCClusRefObject = NULL;
  388. _hr = CComObject< CClusRefObject >::CreateInstance( &pCClusRefObject );
  389. if ( SUCCEEDED( _hr ) )
  390. {
  391. CSmartPtr< CComObject< CClusRefObject > > ptrRefObject( pCClusRefObject );
  392. ptrRefObject->SetClusHandle( m_hCluster );
  393. _hr = pCClusRefObject->QueryInterface( IID_ISClusRefObject, (void **) &m_pClusRefObject );
  394. } // if: CreateInstance OK.
  395. } // else: the cluster was opened
  396. } // if: is there already a cluster open?
  397. } // if: bstrClusterName != NULL
  398. return _hr;
  399. } //*** CCluster::Open()
  400. /////////////////////////////////////////////////////////////////////////////
  401. //++
  402. //
  403. // CCluster::get_Handle
  404. //
  405. // Description:
  406. // Return the cluster handle.
  407. //
  408. // Arguments:
  409. // phandle [OUT] - Catches the cluster handle.
  410. //
  411. // Return Value:
  412. // S_OK if successful, or E_POINTER if not.
  413. //
  414. //--
  415. /////////////////////////////////////////////////////////////////////////////
  416. STDMETHODIMP CCluster::get_Handle( OUT ULONG_PTR * phandle )
  417. {
  418. //ASSERT( phandle != NULL );
  419. HRESULT _hr = E_POINTER;
  420. if ( phandle != NULL )
  421. {
  422. _hr = E_HANDLE;
  423. if ( m_hCluster != NULL )
  424. {
  425. *phandle = (ULONG_PTR) m_hCluster;
  426. _hr = S_OK;
  427. } // if: cluster handle is not NULL
  428. } // if: args are not NULL
  429. return _hr;
  430. } //*** CCluster::get_Handle()
  431. /////////////////////////////////////////////////////////////////////////////
  432. //++
  433. //
  434. // CCluster::Close
  435. //
  436. // Description:
  437. // Close the cluster.
  438. //
  439. // Arguments:
  440. // None.
  441. //
  442. // Return Value:
  443. // S_OK.
  444. //
  445. //--
  446. /////////////////////////////////////////////////////////////////////////////
  447. STDMETHODIMP CCluster::Close( void )
  448. {
  449. if ( m_hCluster != NULL )
  450. {
  451. //
  452. // If the Cluster Handle will be closed only when the
  453. // reference count on the RefObj becomes 0. But the
  454. // Cluster Object will be initialized and is reusable.
  455. //
  456. Clear();
  457. }
  458. return S_OK;
  459. } //*** CCluster::Close()
  460. /////////////////////////////////////////////////////////////////////////////
  461. //++
  462. //
  463. // CCluster::put_Name
  464. //
  465. // Description:
  466. // Change the name of this object (Cluster).
  467. //
  468. // Arguments:
  469. // bstrClusterName [IN] - The new name.
  470. //
  471. // Return Value:
  472. // S_OK if successful, E_POINTER, or other Win32 error as HRESULT.
  473. //
  474. //--
  475. /////////////////////////////////////////////////////////////////////////////
  476. STDMETHODIMP CCluster::put_Name( IN BSTR bstrClusterName )
  477. {
  478. //ASSERT( bstrClusterName != NULL );
  479. //ASSERT( pvarStatusCode != NULL );
  480. //ASSERT( bstrClusterName[ 0 ] != '\0' );
  481. ASSERT( m_hCluster != NULL );
  482. HRESULT _hr = E_POINTER;
  483. if ( ( bstrClusterName != NULL ) && ( bstrClusterName[ 0 ] != '\0' ) )
  484. {
  485. _hr = E_HANDLE;
  486. if ( m_hCluster != NULL )
  487. {
  488. DWORD _sc = ::SetClusterName( m_hCluster, bstrClusterName );
  489. //
  490. // Convert status, it's not an error, into error success since we
  491. // don't want an exception to be thrown when the client is a scripting
  492. // client.
  493. //
  494. if ( _sc == ERROR_RESOURCE_PROPERTIES_STORED )
  495. {
  496. _sc = ERROR_SUCCESS;
  497. }
  498. _hr = HRESULT_FROM_WIN32( _sc );
  499. }
  500. } // if: args are not NULL and the new name is not empty
  501. return _hr;
  502. } //*** CCluster::put_Name()
  503. /////////////////////////////////////////////////////////////////////////////
  504. //++
  505. //
  506. // CCluster::get_Name
  507. //
  508. // Description:
  509. // Return the name of this object (Cluster).
  510. //
  511. // Arguments:
  512. // pbstrClusterName [OUT] - Catches the name of this object.
  513. //
  514. // Return Value:
  515. // S_OK if successful, E_POINTER, or other HRESULT error.
  516. //
  517. //--
  518. /////////////////////////////////////////////////////////////////////////////
  519. STDMETHODIMP CCluster::get_Name( OUT BSTR * pbstrClusterName )
  520. {
  521. //ASSERT( pbstrClusterName != NULL );
  522. HRESULT _hr = E_POINTER;
  523. if ( pbstrClusterName != NULL )
  524. {
  525. if ( m_hCluster != NULL )
  526. {
  527. CLUSTERVERSIONINFO clusinfo;
  528. LPWSTR pwszName = NULL;
  529. DWORD _sc;
  530. clusinfo.dwVersionInfoSize = sizeof( clusinfo );
  531. _sc = WrapGetClusterInformation( m_hCluster, &pwszName, &clusinfo );
  532. if ( _sc == ERROR_SUCCESS )
  533. {
  534. *pbstrClusterName = SysAllocString( pwszName );
  535. if ( *pbstrClusterName == NULL )
  536. {
  537. _hr = E_OUTOFMEMORY;
  538. }
  539. ::LocalFree( pwszName );
  540. pwszName = NULL;
  541. }
  542. _hr = HRESULT_FROM_WIN32( _sc );
  543. }
  544. }
  545. return _hr;
  546. } //*** CCluster::get_Name()
  547. /////////////////////////////////////////////////////////////////////////////
  548. //++
  549. //
  550. // CCluster::get_Version
  551. //
  552. // Description:
  553. // Return the version info for this cluster.
  554. //
  555. // Arguments:
  556. // ppClusVersion [OUT] - Catches the ClusVersion object.
  557. //
  558. // Return Value:
  559. // S_OK if successful, E_POINTER, or Win32 error as HRESULT.
  560. //
  561. //--
  562. /////////////////////////////////////////////////////////////////////////////
  563. STDMETHODIMP CCluster::get_Version( OUT ISClusVersion ** ppClusVersion )
  564. {
  565. //ASSERT( ppClusVersion != NULL );
  566. ASSERT( m_hCluster != NULL );
  567. HRESULT _hr = E_POINTER;
  568. if ( ppClusVersion != NULL )
  569. {
  570. _hr = E_HANDLE;
  571. if ( m_hCluster != NULL )
  572. {
  573. CComObject< CClusVersion > * pClusVersion = NULL;
  574. *ppClusVersion = NULL;
  575. _hr = CComObject< CClusVersion >::CreateInstance( &pClusVersion );
  576. if ( SUCCEEDED( _hr ) )
  577. {
  578. CSmartPtr< ISClusRefObject > ptrRefObject( m_pClusRefObject );
  579. CSmartPtr< CComObject< CClusVersion > > ptrClusVersion( pClusVersion );
  580. _hr = ptrClusVersion->Create( ptrRefObject );
  581. if ( SUCCEEDED( _hr ) )
  582. {
  583. _hr = ptrClusVersion->QueryInterface( IID_ISClusVersion, (void **) ppClusVersion );
  584. } // if: ClusVersion object created
  585. } // if: ClusVersion object allocated
  586. } // if: cluster handle is not NULL
  587. } // if: args are not NULL
  588. return _hr;
  589. } //*** CCluster::GetVersion()
  590. /////////////////////////////////////////////////////////////////////////////
  591. //++
  592. //
  593. // CCluster::put_QuorumResource
  594. //
  595. // Description:
  596. // Change the quorum resource.
  597. //
  598. // Arguments:
  599. // pResource [IN] - The new quorum resource.
  600. //
  601. // Return Value:
  602. // S_OK if successful, E_POINTER, or Win32 error as HRESULT.
  603. //
  604. //--
  605. /////////////////////////////////////////////////////////////////////////////
  606. STDMETHODIMP CCluster::put_QuorumResource( IN ISClusResource * pResource )
  607. {
  608. //ASSERT( pResource != NULL );
  609. HRESULT _hr = E_POINTER;
  610. if ( pResource != NULL )
  611. {
  612. _hr = E_HANDLE;
  613. if ( m_hCluster != NULL )
  614. {
  615. _hr = pResource->BecomeQuorumResource( m_bstrQuorumPath, m_nQuorumLogSize );
  616. } // if: the cluster handle is not NULL
  617. } // if: args are not NULL
  618. return _hr;
  619. } //*** CCluster::put_QuorumResource()
  620. /////////////////////////////////////////////////////////////////////////////
  621. //++
  622. //
  623. // CCluster::get_QuorumResource
  624. //
  625. // Description:
  626. // Returns the quorum resource.
  627. //
  628. // Arguments:
  629. // ppResource [IN] - Catches the quorum resource.
  630. //
  631. // Return Value:
  632. // S_OK if successful, E_POINTER, or Win32 error as HRESULT.
  633. //
  634. //--
  635. /////////////////////////////////////////////////////////////////////////////
  636. STDMETHODIMP CCluster::get_QuorumResource( ISClusResource ** ppResource )
  637. {
  638. //ASSERT( ppResource != NULL );
  639. HRESULT _hr = E_POINTER;
  640. if ( ppResource != NULL )
  641. {
  642. _hr = E_HANDLE;
  643. if ( m_hCluster != NULL )
  644. {
  645. LPWSTR lpszResourceName = NULL;
  646. LPWSTR lpszDeviceName = NULL;
  647. DWORD dwLogSize = 0;
  648. DWORD _sc;
  649. _sc = ::WrapGetClusterQuorumResource( m_hCluster, &lpszResourceName, &lpszDeviceName, &dwLogSize );
  650. if ( _sc == ERROR_SUCCESS )
  651. {
  652. _hr = OpenResource( lpszResourceName, ppResource );
  653. if ( SUCCEEDED( _hr ) )
  654. {
  655. if ( lpszResourceName != NULL )
  656. {
  657. ::LocalFree( lpszResourceName );
  658. }
  659. if ( lpszDeviceName != NULL )
  660. {
  661. m_bstrQuorumPath = lpszDeviceName;
  662. ::LocalFree( lpszDeviceName );
  663. }
  664. m_nQuorumLogSize = dwLogSize;
  665. }
  666. }
  667. else
  668. {
  669. _hr = HRESULT_FROM_WIN32( _sc );
  670. }
  671. }
  672. }
  673. return _hr;
  674. } //*** CCluster::get_QuorumResource()
  675. /////////////////////////////////////////////////////////////////////////////
  676. //++
  677. //
  678. // CCluster::HrGetQuorumInfo
  679. //
  680. // Description:
  681. // Retrieves the current quorum info and stores it in member vars.
  682. //
  683. // Arguments:
  684. // None.
  685. //
  686. // Return Value:
  687. // S_OK if successful, or Win32 error as HRESULT if not.
  688. //
  689. //--
  690. /////////////////////////////////////////////////////////////////////////////
  691. STDMETHODIMP CCluster::HrGetQuorumInfo( void )
  692. {
  693. LPWSTR lpszResourceName = NULL;
  694. LPWSTR lpszDeviceName = NULL;
  695. DWORD dwLogSize = 0;
  696. DWORD _sc = NO_ERROR;
  697. HRESULT _hr = E_HANDLE;
  698. if ( m_hCluster != NULL )
  699. {
  700. _sc = ::WrapGetClusterQuorumResource( m_hCluster, &lpszResourceName, &lpszDeviceName, &dwLogSize );
  701. _hr = HRESULT_FROM_WIN32( _sc );
  702. if ( SUCCEEDED( _hr ) )
  703. {
  704. if ( lpszResourceName != NULL )
  705. {
  706. m_bstrQuorumResourceName = lpszResourceName;
  707. ::LocalFree( lpszResourceName );
  708. }
  709. if ( lpszDeviceName != NULL )
  710. {
  711. m_bstrQuorumPath = lpszDeviceName;
  712. ::LocalFree( lpszDeviceName );
  713. }
  714. m_nQuorumLogSize = dwLogSize;
  715. } // if: WrapGetClusterQuorumResource() succeeded
  716. } // if: cluster handle is not NULL
  717. return _hr;
  718. } //*** CCluster::HrGetQuorumInfo()
  719. /////////////////////////////////////////////////////////////////////////////
  720. //++
  721. //
  722. // CCluster::get_QuorumLogSize
  723. //
  724. // Description:
  725. // Returns the current quorum log size.
  726. //
  727. // Arguments:
  728. // pnQuorumLogSize [OUT] - Catches the log file size.
  729. //
  730. // Return Value:
  731. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  732. //
  733. //--
  734. /////////////////////////////////////////////////////////////////////////////
  735. STDMETHODIMP CCluster::get_QuorumLogSize( OUT long * pnQuorumLogSize )
  736. {
  737. //ASSERT( pnQuorumLogSize != NULL );
  738. HRESULT _hr = E_POINTER;
  739. if ( pnQuorumLogSize != NULL )
  740. {
  741. _hr = E_HANDLE;
  742. if ( m_hCluster != NULL )
  743. {
  744. _hr = HrGetQuorumInfo();
  745. if ( SUCCEEDED( _hr ) )
  746. {
  747. *pnQuorumLogSize = m_nQuorumLogSize;
  748. }
  749. }
  750. }
  751. return _hr;
  752. } //*** CCluster::get_QuorumLogSize()
  753. /////////////////////////////////////////////////////////////////////////////
  754. //++
  755. //
  756. // CCluster::put_QuorumLogSize
  757. //
  758. // Description:
  759. // Set the current quorum log size.
  760. //
  761. // Arguments:
  762. // nQuorumLogSize [IN] - The new log file size.
  763. //
  764. // Return Value:
  765. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  766. //
  767. //--
  768. /////////////////////////////////////////////////////////////////////////////
  769. STDMETHODIMP CCluster::put_QuorumLogSize( IN long nQuoromLogSize )
  770. {
  771. //ASSERT( nQuoromLogSize > 0 );
  772. HRESULT _hr = E_INVALIDARG;
  773. if ( nQuoromLogSize > 0 )
  774. {
  775. _hr = E_HANDLE;
  776. if ( m_hCluster != NULL )
  777. {
  778. _hr = HrGetQuorumInfo();
  779. if ( SUCCEEDED( _hr ) )
  780. {
  781. DWORD _sc = NO_ERROR;
  782. HRESOURCE hResource = NULL;
  783. hResource = ::OpenClusterResource( m_hCluster, m_bstrQuorumResourceName );
  784. if ( hResource != NULL )
  785. {
  786. m_nQuorumLogSize = nQuoromLogSize;
  787. _sc = ::SetClusterQuorumResource( hResource, m_bstrQuorumPath, m_nQuorumLogSize );
  788. _hr = HRESULT_FROM_WIN32( _sc );
  789. ::CloseClusterResource( hResource );
  790. }
  791. else
  792. {
  793. _sc = GetLastError();
  794. _hr = HRESULT_FROM_WIN32( _sc );
  795. }
  796. }
  797. }
  798. }
  799. return _hr;
  800. } //*** CCluster::put_QuorumLogSize()
  801. /////////////////////////////////////////////////////////////////////////////
  802. //++
  803. //
  804. // CCluster::get_QuorumPath
  805. //
  806. // Description:
  807. // Returns the current quorum log path.
  808. //
  809. // Arguments:
  810. // ppPath [OUT] - Catches the device path.
  811. //
  812. // Return Value:
  813. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  814. //
  815. //--
  816. /////////////////////////////////////////////////////////////////////////////
  817. STDMETHODIMP CCluster::get_QuorumPath( OUT BSTR * ppPath )
  818. {
  819. //ASSERT( ppPath != NULL );
  820. HRESULT _hr = E_POINTER;
  821. if ( ppPath != NULL )
  822. {
  823. _hr = E_HANDLE;
  824. if ( m_hCluster != NULL )
  825. {
  826. _hr = HrGetQuorumInfo();
  827. if ( SUCCEEDED( _hr ) )
  828. {
  829. *ppPath = m_bstrQuorumPath.Copy();
  830. }
  831. }
  832. }
  833. return _hr;
  834. } //*** CCluster::get_QuorumPath()
  835. /////////////////////////////////////////////////////////////////////////////
  836. //++
  837. //
  838. // CCluster::put_QuorumPath
  839. //
  840. // Description:
  841. // Change the current quorum log path.
  842. //
  843. // Arguments:
  844. // pPath [IN] - The new device path.
  845. //
  846. // Return Value:
  847. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  848. //
  849. //--
  850. /////////////////////////////////////////////////////////////////////////////
  851. STDMETHODIMP CCluster::put_QuorumPath( IN BSTR pPath )
  852. {
  853. //ASSERT( pPath != NULL );
  854. HRESULT _hr = E_POINTER;
  855. if ( pPath != NULL )
  856. {
  857. _hr = E_HANDLE;
  858. if ( m_hCluster != NULL )
  859. {
  860. _hr = HrGetQuorumInfo();
  861. if ( SUCCEEDED( _hr ) )
  862. {
  863. DWORD _sc = NO_ERROR;
  864. HRESOURCE hResource = NULL;
  865. hResource = ::OpenClusterResource( m_hCluster, m_bstrQuorumResourceName );
  866. if ( hResource != NULL )
  867. {
  868. m_bstrQuorumPath = pPath;
  869. _sc = ::SetClusterQuorumResource( hResource, m_bstrQuorumPath, m_nQuorumLogSize );
  870. _hr = HRESULT_FROM_WIN32( _sc );
  871. ::CloseClusterResource( hResource );
  872. }
  873. else
  874. {
  875. _sc = GetLastError();
  876. _hr = HRESULT_FROM_WIN32( _sc );
  877. }
  878. }
  879. }
  880. }
  881. return _hr;
  882. } //*** CCluster::put_QuorumPath()
  883. /////////////////////////////////////////////////////////////////////////////
  884. //++
  885. //
  886. // CCluster::get_Nodes
  887. //
  888. // Description:
  889. // Returns the collection of nodes for this cluster.
  890. //
  891. // Arguments:
  892. // ppClusterNodes [OUT] - Catches the collection.
  893. //
  894. // Return Value:
  895. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  896. //
  897. //--
  898. /////////////////////////////////////////////////////////////////////////////
  899. STDMETHODIMP CCluster::get_Nodes( OUT ISClusNodes ** ppClusterNodes )
  900. {
  901. return ::HrCreateResourceCollection< CClusNodes, ISClusNodes, HNODE >(
  902. ppClusterNodes,
  903. IID_ISClusNodes,
  904. m_pClusRefObject
  905. );
  906. } //*** CCluster::get_Nodes()
  907. /////////////////////////////////////////////////////////////////////////////
  908. //++
  909. //
  910. // CCluster::get_ResourceGroups
  911. //
  912. // Description:
  913. // Returns the collection of resource groups for this cluster.
  914. //
  915. // Arguments:
  916. // ppClusterResourceGroups [OUT] - Catches the collection.
  917. //
  918. // Return Value:
  919. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  920. //
  921. //--
  922. /////////////////////////////////////////////////////////////////////////////
  923. STDMETHODIMP CCluster::get_ResourceGroups(
  924. OUT ISClusResGroups ** ppClusterResourceGroups
  925. )
  926. {
  927. return ::HrCreateResourceCollection< CClusResGroups, ISClusResGroups, HRESOURCE >(
  928. ppClusterResourceGroups,
  929. IID_ISClusResGroups,
  930. m_pClusRefObject
  931. );
  932. } //*** CCluster::get_ResourceGroups()
  933. /////////////////////////////////////////////////////////////////////////////
  934. //++
  935. //
  936. // CCluster::get_Resources
  937. //
  938. // Description:
  939. // Returns the collection of resources for this cluster.
  940. //
  941. // Arguments:
  942. // ppClusterResources [OUT] - Catches the collection.
  943. //
  944. // Return Value:
  945. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  946. //
  947. //--
  948. /////////////////////////////////////////////////////////////////////////////
  949. STDMETHODIMP CCluster::get_Resources(
  950. OUT ISClusResources ** ppClusterResources
  951. )
  952. {
  953. return ::HrCreateResourceCollection< CClusResources, ISClusResources, HRESOURCE >(
  954. &m_pClusterResources,
  955. ppClusterResources,
  956. IID_ISClusResources,
  957. m_pClusRefObject
  958. );
  959. } //*** CCluster::get_Resources()
  960. /////////////////////////////////////////////////////////////////////////////
  961. //++
  962. //
  963. // CCluster::OpenResource
  964. //
  965. // Description:
  966. // Create and open a new resource.
  967. //
  968. // Arguments:
  969. // bstrResourceName [IN] - The name of the resource to open.
  970. // ppClusterResource [OUT] - Catches the new resource.
  971. //
  972. // Return Value:
  973. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  974. //
  975. //--
  976. /////////////////////////////////////////////////////////////////////////////
  977. STDMETHODIMP CCluster::OpenResource(
  978. IN BSTR bstrResourceName,
  979. OUT ISClusResource ** ppClusterResource
  980. )
  981. {
  982. //ASSERT( bstrResourceName != NULL );
  983. //ASSERT( ppClusterResource != NULL );
  984. ASSERT( m_hCluster != NULL );
  985. HRESULT _hr = E_POINTER;
  986. if ( ( bstrResourceName != NULL ) && ( ppClusterResource != NULL ) )
  987. {
  988. _hr = E_HANDLE;
  989. if ( m_hCluster != NULL )
  990. {
  991. CComObject< CClusResource > * pClusterResource = NULL;
  992. *ppClusterResource = NULL;
  993. _hr = CComObject< CClusResource >::CreateInstance( &pClusterResource );
  994. if ( SUCCEEDED( _hr ) )
  995. {
  996. CSmartPtr< ISClusRefObject > ptrRefObject( m_pClusRefObject );
  997. CSmartPtr< CComObject< CClusResource > > ptrClusterResource( pClusterResource );
  998. _hr = ptrClusterResource->Open( ptrRefObject, bstrResourceName );
  999. if ( SUCCEEDED( _hr ) )
  1000. {
  1001. _hr = ptrClusterResource->QueryInterface( IID_ISClusResource, (void **) ppClusterResource );
  1002. }
  1003. }
  1004. }
  1005. }
  1006. return _hr;
  1007. } //*** CCluster::OpenResource()
  1008. /////////////////////////////////////////////////////////////////////////////
  1009. //++
  1010. //
  1011. // CCluster::get_ResourceTypes
  1012. //
  1013. // Description:
  1014. // Returns the collection of resource types for this cluster.
  1015. //
  1016. // Arguments:
  1017. // ppResourceTypes [OUT] - Catches the collection.
  1018. //
  1019. // Return Value:
  1020. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  1021. //
  1022. //--
  1023. /////////////////////////////////////////////////////////////////////////////
  1024. STDMETHODIMP CCluster::get_ResourceTypes(
  1025. OUT ISClusResTypes ** ppResourceTypes
  1026. )
  1027. {
  1028. return ::HrCreateResourceCollection< CClusResTypes, ISClusResTypes, CComBSTR >(
  1029. &m_pResourceTypes,
  1030. ppResourceTypes,
  1031. IID_ISClusResTypes,
  1032. m_pClusRefObject
  1033. );
  1034. } //*** CCluster::get_ResourceTypes()
  1035. /////////////////////////////////////////////////////////////////////////////
  1036. //++
  1037. //
  1038. // CCluster::get_Networks
  1039. //
  1040. // Description:
  1041. // Returns the collection of networks for this cluster.
  1042. //
  1043. // Arguments:
  1044. // ppNetworks [OUT] - Catches the collection.
  1045. //
  1046. // Return Value:
  1047. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  1048. //
  1049. //--
  1050. /////////////////////////////////////////////////////////////////////////////
  1051. STDMETHODIMP CCluster::get_Networks( OUT ISClusNetworks ** ppNetworks )
  1052. {
  1053. return ::HrCreateResourceCollection< CClusNetworks, ISClusNetworks, HNETWORK >(
  1054. &m_pNetworks,
  1055. ppNetworks,
  1056. IID_ISClusNetworks,
  1057. m_pClusRefObject
  1058. );
  1059. } //*** CCluster::get_Networks()
  1060. /////////////////////////////////////////////////////////////////////////////
  1061. //++
  1062. //
  1063. // CCluster::get_NetInterfaces
  1064. //
  1065. // Description:
  1066. // Returns the collection of netinterfaces for this cluster.
  1067. //
  1068. // Arguments:
  1069. // ppNetInterfaces [OUT] - Catches the collection.
  1070. //
  1071. // Return Value:
  1072. // S_OK if successful, E_POINTER, or Win32 error as HRESULT if not.
  1073. //
  1074. //--
  1075. /////////////////////////////////////////////////////////////////////////////
  1076. STDMETHODIMP CCluster::get_NetInterfaces(
  1077. OUT ISClusNetInterfaces ** ppNetInterfaces
  1078. )
  1079. {
  1080. return ::HrCreateResourceCollection< CClusNetInterfaces, ISClusNetInterfaces, HNETINTERFACE >(
  1081. &m_pNetInterfaces,
  1082. ppNetInterfaces,
  1083. IID_ISClusNetInterfaces,
  1084. m_pClusRefObject
  1085. );
  1086. } //*** CCluster::get_NetInterfaces()
  1087. /////////////////////////////////////////////////////////////////////////////
  1088. //++
  1089. //
  1090. // CCluster::GetProperties
  1091. //
  1092. // Description:
  1093. // Creates a property collection for this object type (Cluster).
  1094. //
  1095. // Arguments:
  1096. // ppProperties [OUT] - Catches the newly created collection.
  1097. // bPrivate [IN] - Are these private properties? Or Common?
  1098. // bReadOnly [IN] - Are these read only properties?
  1099. //
  1100. // Return Value:
  1101. // S_OK if successful, or other HRESULT error.
  1102. //
  1103. //--
  1104. /////////////////////////////////////////////////////////////////////////////
  1105. HRESULT CCluster::GetProperties(
  1106. ISClusProperties ** ppProperties,
  1107. BOOL bPrivate,
  1108. BOOL bReadOnly
  1109. )
  1110. {
  1111. //ASSERT( ppProperties != NULL );
  1112. HRESULT _hr = E_POINTER;
  1113. if ( ppProperties != NULL )
  1114. {
  1115. *ppProperties = NULL;
  1116. CComObject< CClusProperties > * pProperties = NULL;
  1117. _hr = CComObject< CClusProperties >::CreateInstance( &pProperties );
  1118. if ( SUCCEEDED( _hr ) )
  1119. {
  1120. CSmartPtr< CComObject< CClusProperties > > ptrProperties( pProperties );
  1121. _hr = ptrProperties->Create( this, bPrivate, bReadOnly );
  1122. if ( SUCCEEDED( _hr ) )
  1123. {
  1124. _hr = ptrProperties->Refresh();
  1125. if ( SUCCEEDED( _hr ) )
  1126. {
  1127. _hr = ptrProperties->QueryInterface( IID_ISClusProperties, (void **) ppProperties );
  1128. if ( SUCCEEDED( _hr ) )
  1129. {
  1130. ptrProperties->AddRef();
  1131. if ( bPrivate )
  1132. {
  1133. if ( bReadOnly )
  1134. {
  1135. m_pPrivateROProperties = pProperties;
  1136. }
  1137. else
  1138. {
  1139. m_pPrivateProperties = pProperties;
  1140. }
  1141. }
  1142. else
  1143. {
  1144. if ( bReadOnly )
  1145. {
  1146. m_pCommonROProperties = pProperties;
  1147. }
  1148. else
  1149. {
  1150. m_pCommonProperties = pProperties;
  1151. }
  1152. }
  1153. }
  1154. }
  1155. }
  1156. }
  1157. }
  1158. return _hr;
  1159. } //*** CCluster::GetProperties()
  1160. /////////////////////////////////////////////////////////////////////////////
  1161. //++
  1162. //
  1163. // CCluster::HrLoadProperties
  1164. //
  1165. // Description:
  1166. // This virtual function does the actual load of the property list from
  1167. // the cluster.
  1168. //
  1169. // Arguments:
  1170. // rcplPropList [IN OUT] - The property list to load.
  1171. // bReadOnly [IN] - Load the read only properties?
  1172. // bPrivate [IN] - Load the common or the private properties?
  1173. //
  1174. // Return Value:
  1175. // S_OK if successful, or other HRESULT error.
  1176. //
  1177. //--
  1178. /////////////////////////////////////////////////////////////////////////////
  1179. HRESULT CCluster::HrLoadProperties(
  1180. IN OUT CClusPropList & rcplPropList,
  1181. IN BOOL bReadOnly,
  1182. IN BOOL bPrivate
  1183. )
  1184. {
  1185. HRESULT _hr = E_INVALIDARG;
  1186. #if CLUSAPI_VERSION >= 0x0500
  1187. DWORD _dwControlCode = 0;
  1188. DWORD _sc = NO_ERROR;
  1189. _hr = E_HANDLE;
  1190. if ( m_hCluster != NULL )
  1191. {
  1192. if ( bReadOnly )
  1193. {
  1194. _dwControlCode = bPrivate
  1195. ? CLUSCTL_CLUSTER_GET_RO_PRIVATE_PROPERTIES
  1196. : CLUSCTL_CLUSTER_GET_RO_COMMON_PROPERTIES;
  1197. }
  1198. else
  1199. {
  1200. _dwControlCode = bPrivate
  1201. ? CLUSCTL_CLUSTER_GET_PRIVATE_PROPERTIES
  1202. : CLUSCTL_CLUSTER_GET_COMMON_PROPERTIES;
  1203. }
  1204. _sc = rcplPropList.ScGetClusterProperties( m_hCluster, _dwControlCode );
  1205. _hr = HRESULT_FROM_WIN32( _sc );
  1206. } // if: cluster handle is not NULL
  1207. #else
  1208. _hr = E_NOTIMPL;
  1209. #endif // CLUSAPI_VERSION >= 0x0500
  1210. return _hr;
  1211. } //*** CCluster::HrLoadProperties()
  1212. /////////////////////////////////////////////////////////////////////////////
  1213. //++
  1214. //
  1215. // CCluster::ScWriteProperties
  1216. //
  1217. // Description:
  1218. // This virtual function does the actual saving of the property list to
  1219. // the cluster.
  1220. //
  1221. // Arguments:
  1222. // rcplPropList [IN] - The property list to save.
  1223. // bPrivate [IN] - Save the common or the private properties?
  1224. //
  1225. // Return Value:
  1226. // ERROR_SUCCESS if successful, or other Win32 error if not.
  1227. //
  1228. //--
  1229. /////////////////////////////////////////////////////////////////////////////
  1230. DWORD CCluster::ScWriteProperties(
  1231. const CClusPropList & rcplPropList,
  1232. BOOL bPrivate
  1233. )
  1234. {
  1235. //ASSERT( bPrivate == FALSE );
  1236. DWORD _sc = ERROR_INVALID_HANDLE;
  1237. #if CLUSAPI_VERSION >= 0x0500
  1238. if ( m_hCluster != NULL )
  1239. {
  1240. DWORD nBytesReturned = 0;
  1241. DWORD _dwControlCode = bPrivate
  1242. ? CLUSCTL_CLUSTER_SET_PRIVATE_PROPERTIES
  1243. : CLUSCTL_CLUSTER_SET_COMMON_PROPERTIES;
  1244. _sc = ClusterControl(
  1245. m_hCluster,
  1246. NULL,
  1247. _dwControlCode,
  1248. rcplPropList,
  1249. rcplPropList.CbBufferSize(),
  1250. 0,
  1251. 0,
  1252. &nBytesReturned
  1253. );
  1254. } // if: cluster handle is not NULL
  1255. #else
  1256. _sc = ERROR_CALL_NOT_IMPLEMENTED;
  1257. #endif // CLUSAPI_VERSION >= 0x0500
  1258. return _sc;
  1259. } //*** CCluster::ScWriteProperties()
  1260. /////////////////////////////////////////////////////////////////////////////
  1261. //++
  1262. //
  1263. // CCluster::get_CommonProperties
  1264. //
  1265. // Description:
  1266. // Get this object's (Cluster) common properties collection.
  1267. //
  1268. // Arguments:
  1269. // ppProperties [OUT] - Catches the properties collection.
  1270. //
  1271. // Return Value:
  1272. // S_OK if successful, or other HRESULT error.
  1273. //
  1274. //--
  1275. /////////////////////////////////////////////////////////////////////////////
  1276. STDMETHODIMP CCluster::get_CommonProperties(
  1277. OUT ISClusProperties ** ppProperties
  1278. )
  1279. {
  1280. //ASSERT( ppProperties != NULL );
  1281. HRESULT _hr = E_POINTER;
  1282. if ( ppProperties != NULL )
  1283. {
  1284. if ( m_pCommonProperties != NULL )
  1285. {
  1286. _hr = m_pCommonProperties->QueryInterface( IID_ISClusProperties, (void **) ppProperties );
  1287. }
  1288. else
  1289. {
  1290. _hr = GetProperties( ppProperties, FALSE, FALSE );
  1291. }
  1292. }
  1293. return _hr;
  1294. } //*** CCluster::get_CommonProperties()
  1295. /////////////////////////////////////////////////////////////////////////////
  1296. //++
  1297. //
  1298. // CCluster::get_PrivateProperties
  1299. //
  1300. // Description:
  1301. // Get this object's (Cluster) private properties collection.
  1302. //
  1303. // Arguments:
  1304. // ppProperties [OUT] - Catches the properties collection.
  1305. //
  1306. // Return Value:
  1307. // S_OK if successful, or other HRESULT error.
  1308. //
  1309. //--
  1310. /////////////////////////////////////////////////////////////////////////////
  1311. STDMETHODIMP CCluster::get_PrivateProperties(
  1312. OUT ISClusProperties ** ppProperties
  1313. )
  1314. {
  1315. //ASSERT( ppProperties != NULL );
  1316. HRESULT _hr = E_POINTER;
  1317. if ( ppProperties != NULL )
  1318. {
  1319. if ( m_pPrivateProperties != NULL )
  1320. {
  1321. _hr = m_pPrivateProperties->QueryInterface( IID_ISClusProperties, (void **) ppProperties );
  1322. }
  1323. else
  1324. {
  1325. _hr = GetProperties( ppProperties, TRUE, FALSE );
  1326. }
  1327. }
  1328. return _hr;
  1329. } //*** CCluster::get_PrivateProperties()
  1330. /////////////////////////////////////////////////////////////////////////////
  1331. //++
  1332. //
  1333. // CCluster::get_CommonROProperties
  1334. //
  1335. // Description:
  1336. // Get this object's (Cluster) common read only properties collection.
  1337. //
  1338. // Arguments:
  1339. // ppProperties [OUT] - Catches the properties collection.
  1340. //
  1341. // Return Value:
  1342. // S_OK if successful, or other HRESULT error.
  1343. //
  1344. /////////////////////////////////////////////////////////////////////////////
  1345. STDMETHODIMP CCluster::get_CommonROProperties(
  1346. OUT ISClusProperties ** ppProperties
  1347. )
  1348. {
  1349. //ASSERT( ppProperties != NULL );
  1350. HRESULT _hr = E_POINTER;
  1351. if ( ppProperties != NULL )
  1352. {
  1353. if ( m_pCommonROProperties != NULL )
  1354. {
  1355. _hr = m_pCommonROProperties->QueryInterface( IID_ISClusProperties, (void **) ppProperties );
  1356. }
  1357. else
  1358. {
  1359. _hr = GetProperties( ppProperties, FALSE, TRUE );
  1360. }
  1361. }
  1362. return _hr;
  1363. } //*** CCluster::get_CommonROProperties()
  1364. /////////////////////////////////////////////////////////////////////////////
  1365. //++
  1366. //
  1367. // CCluster::get_PrivateROProperties
  1368. //
  1369. // Description:
  1370. // Get this object's (Cluster) private read only properties collection.
  1371. //
  1372. // Arguments:
  1373. // ppProperties [OUT] - Catches the properties collection.
  1374. //
  1375. // Return Value:
  1376. // S_OK if successful, or other HRESULT error.
  1377. //
  1378. //--
  1379. /////////////////////////////////////////////////////////////////////////////
  1380. STDMETHODIMP CCluster::get_PrivateROProperties(
  1381. OUT ISClusProperties ** ppProperties
  1382. )
  1383. {
  1384. //ASSERT( ppProperties != NULL );
  1385. HRESULT _hr = E_POINTER;
  1386. if ( ppProperties != NULL )
  1387. {
  1388. if ( m_pPrivateROProperties != NULL )
  1389. {
  1390. _hr = m_pPrivateROProperties->QueryInterface( IID_ISClusProperties, (void **) ppProperties );
  1391. }
  1392. else
  1393. {
  1394. _hr = GetProperties( ppProperties, TRUE, TRUE );
  1395. }
  1396. }
  1397. return _hr;
  1398. } //*** CCluster::get_PrivateROProperties()
  1399. /*
  1400. /////////////////////////////////////////////////////////////////////////////
  1401. //++
  1402. //
  1403. // CCluster::get_Parent
  1404. //
  1405. // Description:
  1406. // Returns the parent of the cluster object. This is an automation
  1407. // thing and the parent could be NULL.
  1408. //
  1409. // Arguments:
  1410. // ppParent [OUT] - Catches the parent.
  1411. //
  1412. // Return Value:
  1413. // S_OK if successful, or other HRESULT error.
  1414. //
  1415. //--
  1416. /////////////////////////////////////////////////////////////////////////////
  1417. STDMETHODIMP CCluster::get_Parent( OUT IDispatch ** ppParent )
  1418. {
  1419. //ASSERT( ppParent != NULL );
  1420. HRESULT _hr = E_POINTER;
  1421. if ( ppParent != NULL )
  1422. {
  1423. if ( m_pParentApplication != NULL )
  1424. {
  1425. _hr = m_pParentApplication->QueryInterface( IID_IDispatch, (void **) ppParent );
  1426. }
  1427. else
  1428. {
  1429. _hr = _InternalQueryInterface( IID_IDispatch, (void **) ppParent );
  1430. }
  1431. }
  1432. return _hr;
  1433. } //*** CCluster::get_Parent()
  1434. /////////////////////////////////////////////////////////////////////////////
  1435. //++
  1436. //
  1437. // CCluster::get_Application
  1438. //
  1439. // Description:
  1440. // Get the parent application for this cluster object. This is an
  1441. // automation thing and it could be NULL.
  1442. //
  1443. // Arguments:
  1444. // ppParentApplication [OUT] - Catches the parent app object.
  1445. //
  1446. // Return Value:
  1447. // S_OK if successful, or other HRESULT error.
  1448. //
  1449. //--
  1450. /////////////////////////////////////////////////////////////////////////////
  1451. STDMETHODIMP CCluster::get_Application(
  1452. OUT ISClusApplication ** ppParentApplication
  1453. )
  1454. {
  1455. //ASSERT( ppParentApplication != NULL );
  1456. HRESULT _hr = E_POINTER;
  1457. if ( ppParentApplication != NULL )
  1458. {
  1459. if ( m_pParentApplication != NULL )
  1460. {
  1461. _hr = m_pParentApplication->QueryInterface( IID_IDispatch, (void **) ppParentApplication );
  1462. }
  1463. }
  1464. return _hr;
  1465. } //*** CCluster::get_Application()
  1466. */