///////////////////////////////////////////////////////////////////////////// // // Copyright (c) 1999 Microsoft Corporation // // Module Name: // ClusDisk.cpp // // Description: // Implementation of the cluster disk class for the MSCLUS // automation classes. // // Author: // Galen Barbee (galenb) 11-Feb-1999 // // Revision History: // // Notes: // ///////////////////////////////////////////////////////////////////////////// #include "stdafx.h" #if CLUSAPI_VERSION >= 0x0500 #include #else #include "PropList.h" #endif // CLUSAPI_VERSION >= 0x0500 #include "ClusDisk.h" ///////////////////////////////////////////////////////////////////////////// // Global variables ///////////////////////////////////////////////////////////////////////////// static const IID * iidCClusDisk[] = { &IID_ISClusDisk }; static const IID * iidCClusDisks[] = { &IID_ISClusDisks }; static const IID * iidCClusScsiAddress[] = { &IID_ISClusScsiAddress }; //*************************************************************************// ///////////////////////////////////////////////////////////////////////////// // CClusDisk class ///////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::CClusDisk // // Description: // Constructor. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// CClusDisk::CClusDisk( void ) { m_pPartitions = NULL; m_dwSignature = 0; m_dwDiskNumber = 0; m_piids = (const IID *) iidCClusDisk; m_piidsSize = ARRAYSIZE( iidCClusDisk ); } //*** CClusDisk::CClusDisk() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::~CClusDisk // // Description: // Destructor. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// CClusDisk::~CClusDisk( void ) { if ( m_pPartitions != NULL ) { m_pPartitions->Release(); } // if: } //*** CClusDisk::~CClusDisk() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::Create // // Description: // Finish creating this object. This method get the value list from // the passed in physical disk resource handle. // // Arguments: // hResource [IN] - Handle to the physical disk resource. // // Return Value: // S_OK if successful, or Win32 error wrapped in HRESULT. // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisk::Create( IN HRESOURCE hResource ) { HRESULT _hr = E_POINTER; DWORD _sc = ERROR_SUCCESS; CClusPropValueList _cpvl; _sc = _cpvl.ScGetResourceValueList( hResource, CLUSCTL_RESOURCE_STORAGE_GET_DISK_INFO ); _hr = HRESULT_FROM_WIN32( _sc ); if ( SUCCEEDED( _hr ) ) { _sc = _cpvl.ScMoveToFirstValue(); _hr = HRESULT_FROM_WIN32( _sc ); if ( SUCCEEDED( _hr ) ) { CLUSPROP_BUFFER_HELPER _cbhValue = { NULL }; CComObject< CClusPartitions > * pPartitions = NULL; if ( m_pPartitions != NULL ) { m_pPartitions->Release(); m_pPartitions = NULL; } // if: clean up any old partitions collection _hr = CComObject< CClusPartitions >::CreateInstance( &pPartitions ); if ( SUCCEEDED( _hr ) ) { CSmartPtr< CComObject< CClusPartitions > > ptrPartitions( pPartitions ); m_pPartitions = ptrPartitions; ptrPartitions->AddRef(); do { _cbhValue = _cpvl; switch ( _cbhValue.pSyntax->dw ) { case CLUSPROP_SYNTAX_PARTITION_INFO : { _hr = ptrPartitions->HrCreateItem( _cbhValue.pPartitionInfoValue ); break; } // case: CLUSPROP_SYNTAX_PARTITION_INFO case CLUSPROP_SYNTAX_DISK_SIGNATURE : { m_dwSignature = _cbhValue.pDiskSignatureValue->dw; break; } // case: CLUSPROP_SYNTAX_DISK_SIGNATURE case CLUSPROP_SYNTAX_SCSI_ADDRESS : { m_csaScsiAddress.dw = _cbhValue.pScsiAddressValue->dw; break; } // case: CLUSPROP_SYNTAX_SCSI_ADDRESS case CLUSPROP_SYNTAX_DISK_NUMBER : { m_dwDiskNumber = _cbhValue.pDiskNumberValue->dw; break; } // case: CLUSPROP_SYNTAX_DISK_NUMBER } // switch: // // Move to the next value. // _sc = _cpvl.ScMoveToNextValue(); if ( _sc == ERROR_NO_MORE_ITEMS ) { _hr = S_OK; break; } // if: error occurred moving to the next value _hr = HRESULT_FROM_WIN32( _sc ); } while ( SUCCEEDED( _hr ) ); // do-while: there are no errors } // if: created the partition collection } // if: move to first value ok } // if: get the value list ok return _hr; } //*** CClusDisk::Create() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::HrCreate // // Description: // Finish creating this object. This method parses a passed in value // list to get the values for the physical disk object. // // Arguments: // rcpvl [IN OUT] - Value list to parse. // pbEndFound [OUT] - Did find the end of the value list? // // Return Value: // S_OK, or Win32 error code wrapped in an HRESULT. // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisk::HrCreate( IN OUT CClusPropValueList & rcpvl, OUT BOOL * pbEndFound ) { DWORD _sc = ERROR_SUCCESS; CLUSPROP_BUFFER_HELPER _cbhValue = { NULL }; CComObject< CClusPartitions > * pPartitions = NULL; HRESULT _hr = S_FALSE; if ( m_pPartitions != NULL ) { m_pPartitions->Release(); m_pPartitions = NULL; } // if: clean up any old partitions collection _hr = CComObject< CClusPartitions >::CreateInstance( &pPartitions ); if ( SUCCEEDED( _hr ) ) { CSmartPtr< CComObject< CClusPartitions > > ptrPartitions( pPartitions ); m_pPartitions = ptrPartitions; ptrPartitions->AddRef(); _cbhValue = rcpvl; do { switch ( _cbhValue.pSyntax->dw ) { case CLUSPROP_SYNTAX_DISK_SIGNATURE : { m_dwSignature = _cbhValue.pDiskSignatureValue->dw; break; } // case: CLUSPROP_SYNTAX_DISK_SIGNATURE case CLUSPROP_SYNTAX_PARTITION_INFO : { _hr = ptrPartitions->HrCreateItem( _cbhValue.pPartitionInfoValue ); break; } // case: CLUSPROP_SYNTAX_PARTITION_INFO case CLUSPROP_SYNTAX_SCSI_ADDRESS : { m_csaScsiAddress.dw = _cbhValue.pScsiAddressValue->dw; break; } // case: CLUSPROP_SYNTAX_SCSI_ADDRESS case CLUSPROP_SYNTAX_DISK_NUMBER : { m_dwDiskNumber = _cbhValue.pDiskNumberValue->dw; break; } // case: CLUSPROP_SYNTAX_DISK_NUMBER } // switch: // // Move to the next value. // _sc = rcpvl.ScMoveToNextValue(); if ( _sc == ERROR_NO_MORE_ITEMS ) { _hr = S_OK; *pbEndFound = TRUE; break; } // if: error occurred moving to the next value _cbhValue = rcpvl; if ( _cbhValue.pSyntax->dw == CLUSPROP_SYNTAX_DISK_SIGNATURE ) { _hr = HRESULT_FROM_WIN32( _sc ); break; } // if: exit if another signature is found before the end of the list is seen _hr = HRESULT_FROM_WIN32( _sc ); } while ( SUCCEEDED( _hr ) ); // do-while: there are no errors } // if: the patitions collection can be created return _hr; } //*** CClusDisk::HrCreate() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::get_Signature // // Description: // Get the disk signature. // // Arguments: // plSignature [OUT] - catches the signature. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisk::get_Signature( OUT long * plSignature ) { //ASSERT( plSignature != NULL ); HRESULT _hr = E_POINTER; if ( plSignature != NULL ) { *plSignature = static_cast< long >( m_dwSignature ); _hr = S_OK; } return _hr; } //*** CClusDisk::get_Signature() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::get_ScsiAddress // // Description: // Get the disk's SCSI address. // // Arguments: // ppScsiAddress [OUT] - catches the SCSI address.. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisk::get_ScsiAddress( OUT ISClusScsiAddress ** ppScsiAddress ) { //ASSERT( ppScsiAddress != NULL ); HRESULT _hr = E_POINTER; if ( ppScsiAddress != NULL ) { CComObject< CClusScsiAddress > * _pScsiAddress = NULL; _hr = CComObject< CClusScsiAddress >::CreateInstance( &_pScsiAddress ); if ( SUCCEEDED( _hr ) ) { CSmartPtr< CComObject< CClusScsiAddress > > _ptrScsiAddress( _pScsiAddress ); _hr = _ptrScsiAddress->Create( m_csaScsiAddress ); if ( SUCCEEDED( _hr ) ) { _hr = _ptrScsiAddress->QueryInterface( IID_ISClusScsiAddress, (void **) ppScsiAddress ); } // if: } // if: } // if: return _hr; } //*** CClusDisk::get_ScsiAddress() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::get_DiskNumber // // Description: // Get the disk number. // // Arguments: // plDiskNumber [OUT] - catches the disk number. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisk::get_DiskNumber( OUT long * plDiskNumber ) { //ASSERT( plDiskNumber != NULL ); HRESULT _hr = E_POINTER; if ( plDiskNumber != NULL ) { *plDiskNumber = static_cast< long >( m_dwDiskNumber ); _hr = S_OK; } return _hr; } //*** CClusDisk::get_DiskNumber() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisk::get_Partitions // // Description: // Get the disk partitions. // // Arguments: // ppPartitions [OUT] - catches the partitions collection. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisk::get_Partitions( OUT ISClusPartitions ** ppPartitions ) { //ASSERT( ppPartitions != NULL ); ASSERT( m_pPartitions != NULL ); HRESULT _hr = E_POINTER; if ( ppPartitions != NULL ) { if ( ppPartitions != NULL ) { _hr = m_pPartitions->QueryInterface( IID_ISClusPartitions, (void **) ppPartitions ); } // if: } // if: return _hr; } //*** CClusDisk::get_Partitions() //*************************************************************************// ///////////////////////////////////////////////////////////////////////////// // CClusDisks class ///////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::CClusDisks // // Description: // Constructor. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// CClusDisks::CClusDisks( void ) { m_pClusRefObject = NULL; m_piids = (const IID *) iidCClusDisks; m_piidsSize = ARRAYSIZE( iidCClusDisks ); } //*** CClusDisks::CClusDisks() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::~CClusDisks // // Description: // Destructor. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// CClusDisks::~CClusDisks( void ) { Clear(); if ( m_pClusRefObject != NULL ) { m_pClusRefObject->Release(); m_pClusRefObject = NULL; } // if: do we have a pointer to the cluster handle wrapper? } //*** CClusDisks::~CClusDisks() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::Create // // Description: // Complete the heavy weight construction, // // Arguments: // rpvl [IN] - Property value list. // // Return Value: // E_NOTIMPL // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisks::Create( IN const CClusPropValueList &rpvl ) { HRESULT _hr = E_NOTIMPL; return _hr; } //*** CClusDisks::Create() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::GetIndex // // Description: // Convert the passed in 1 based index into a 0 based index. // // Arguments: // varIndex [IN] - holds the 1 based index. // pnIndex [OUT] - catches the 0 based index. // // Return Value: // S_OK if successful, E_POINTER, or E_INVALIDARG if out of range. // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisks::GetIndex( IN VARIANT varIndex, OUT UINT * pnIndex ) { //ASSERT( pnIndex != NULL ); HRESULT _hr = E_POINTER; if ( pnIndex != NULL ) { CComVariant v; UINT nIndex = 0; *pnIndex = 0; v.Copy( &varIndex ); // Check to see if the index is a number. _hr = v.ChangeType( VT_I4 ); if ( SUCCEEDED( _hr ) ) { nIndex = v.lVal; nIndex--; // Adjust index to be 0 relative instead of 1 relative if ( nIndex < m_dvDisks.size() ) { *pnIndex = nIndex; } else { _hr = E_INVALIDARG; } } } return _hr; } //*** CClusDisks::GetIndex() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::get_Count // // Description: // Get the count of objects in the collection. // // Arguments: // plCount [OUT] - Catches the count. // // Return Value: // S_OK if successful or E_POINTER. // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisks::get_Count( OUT long * plCount ) { //ASSERT( plCount != NULL ); HRESULT _hr = E_POINTER; if ( plCount != NULL ) { *plCount = m_dvDisks.size(); _hr = S_OK; } return _hr; } //*** CClusDisks::get_Count() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::Clear // // Description: // Empty the vector of disks. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// void CClusDisks::Clear( void ) { ::ReleaseAndEmptyCollection< DiskVector, CComObject< CClusDisk > >( m_dvDisks ); } //*** CClusDisks::Clear() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::get_Item // // Description: // Get the item (disk) at the passed in index. // // Arguments: // varIndex [IN] - Contains the index requested. // ppbstrRegistryKey [OUT] - Catches the key. // // Return Value: // S_OK if successful, E_POINTER, or other HRESULT error. // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisks::get_Item( IN VARIANT varIndex, OUT ISClusDisk ** ppDisk ) { //ASSERT( ppDisk != NULL ); HRESULT _hr = E_POINTER; if ( ppDisk != NULL ) { CComObject< CClusDisk > * pDisk = NULL; // Zero the out param *ppDisk = NULL; UINT nIndex = 0; _hr = GetIndex( varIndex, &nIndex ); if ( SUCCEEDED( _hr ) ) { pDisk = m_dvDisks[ nIndex ]; _hr = pDisk->QueryInterface( IID_ISClusDisk, (void **) ppDisk ); } } return _hr; } //*** CClusDisks::get_Item() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::get__NewEnum // // Description: // Create and return a new enumeration for this collection. // // Arguments: // ppunk [OUT] - Catches the new enumeration. // // Return Value: // S_OK if successful, E_POINTER, or other HRESULT error. // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisks::get__NewEnum( OUT IUnknown ** ppunk ) { return ::HrNewIDispatchEnum< DiskVector, CComObject< CClusDisk > >( ppunk, m_dvDisks ); } //*** CClusDisks::get__NewEnum() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::Create // // Description: // Finish creating the object by doing things that cannot be done in // a light weight constructor. // // Arguments: // pClusRefObject [IN] - Wraps the cluster handle. // bstrResTypeName [IN] - The resource type this collection is for. // // Return Value: // S_OK if successful, or E_POINTER if not. // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisks::Create( IN ISClusRefObject * pClusRefObject, IN BSTR bstrResTypeName ) { ASSERT( pClusRefObject != NULL ); HRESULT _hr = E_POINTER; if ( pClusRefObject != NULL ) { m_pClusRefObject = pClusRefObject; m_pClusRefObject->AddRef(); m_bstrResTypeName = bstrResTypeName; _hr = S_OK; } return _hr; } //*** CClusDisks::Create() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::Refresh // // Description: // Load the collection from the cluster database. // // Arguments: // None. // // Return Value: // S_OK if successful, or Win32 error as HRESULT if not. // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusDisks::Refresh( void ) { HRESULT _hr = S_OK; DWORD _sc = ERROR_SUCCESS; HCLUSTER _hCluster = NULL; BOOL _bEndFound = FALSE; _hr = m_pClusRefObject->get_Handle( (ULONG_PTR *) &_hCluster ); if ( SUCCEEDED( _hr ) ) { CClusPropValueList _cpvl; DWORD _sc = ERROR_SUCCESS; _sc = _cpvl.ScGetResourceTypeValueList( _hCluster, m_bstrResTypeName, CLUSCTL_RESOURCE_TYPE_STORAGE_GET_AVAILABLE_DISKS ); _hr = HRESULT_FROM_WIN32( _sc ); if ( SUCCEEDED( _hr ) ) { Clear(); _sc = _cpvl.ScMoveToFirstValue(); _hr = HRESULT_FROM_WIN32( _sc ); if ( SUCCEEDED( _hr ) ) { CLUSPROP_BUFFER_HELPER _cbhValue = { NULL }; do { _cbhValue = _cpvl; if ( _cbhValue.pSyntax->dw == CLUSPROP_SYNTAX_DISK_SIGNATURE ) { _hr = HrCreateDisk( _cpvl, &_bEndFound ); } // if: value list MUST start with signature! } while ( ! _bEndFound ); // do-while: there are values in the list } // if: we moved to the first value } // if: the value list of available disks was retrieved } // if: we have a cluster handle return _hr; } //*** CClusDisks::Refresh() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusDisks::HrCreateDisk // // Description: // Create a CClusDisk object from the passed in value list and add it // to the collection. This method assumes that the value list's curent // value is the disk signature. // // Arguments: // rcpvl [IN OUT] - The value list to parse. // pbEndFound [IN] - Catches the end of list state. // // Return Value: // S_OK, if successful, Win32 error code wrapped in an HRESULT. // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusDisks::HrCreateDisk( IN OUT CClusPropValueList & rcpvl, OUT BOOL * pbEndFound ) { CComObject< CClusDisk > * _pDisk = NULL; HRESULT _hr = S_FALSE; _hr = CComObject< CClusDisk >::CreateInstance( &_pDisk ); if ( SUCCEEDED( _hr ) ) { CSmartPtr< CComObject< CClusDisk > > _ptrDisk( _pDisk ); _hr = _ptrDisk->HrCreate( rcpvl, pbEndFound ); if ( SUCCEEDED( _hr ) ) { m_dvDisks.insert( m_dvDisks.end(), _pDisk ); _ptrDisk->AddRef(); } // if: } // if: return _hr; } //*** CClusDisks::HrCreateDisk() //*************************************************************************// ///////////////////////////////////////////////////////////////////////////// // CClusScsiAddress class ///////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::CClusScsiAddress // // Description: // Constructor. // // Arguments: // None. // // Return Value: // None. // //-- ///////////////////////////////////////////////////////////////////////////// CClusScsiAddress::CClusScsiAddress( void ) { m_piids = (const IID *) iidCClusScsiAddress; m_piidsSize = ARRAYSIZE( iidCClusScsiAddress ); } //*** CClusScsiAddress::CClusScsiAddress() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::Create // // Description: // Finish creating this object. // // Arguments: // pcpi [IN] - points to the CLUS_PARTITION_INFO struct. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// HRESULT CClusScsiAddress::Create( IN const CLUS_SCSI_ADDRESS & rcsa ) { m_csa = rcsa; return S_OK; } //*** CClusScsiAddress::Create() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::get_PortNumber // // Description: // Get the disk's port number. // // Arguments: // pvarPortNumber [OUT] - catches the port number. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusScsiAddress::get_PortNumber( OUT VARIANT * pvarPortNumber ) { //ASSERT( pvarPortNumber != NULL ); HRESULT _hr = E_POINTER; if ( pvarPortNumber != NULL ) { pvarPortNumber->bVal = m_csa.PortNumber; pvarPortNumber->vt = VT_UI1; _hr = S_OK; } return _hr; } //*** CClusScsiAddress::get_PortNumber() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::get_PathId // // Description: // Get the disk's path id. // // Arguments: // pvarPathId [OUT] - catches the path id. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusScsiAddress::get_PathId( OUT VARIANT * pvarPathId ) { //ASSERT( pvarPathId != NULL ); HRESULT _hr = E_POINTER; if ( pvarPathId != NULL ) { pvarPathId->bVal = m_csa.PathId; pvarPathId->vt = VT_UI1; _hr = S_OK; } return _hr; } //*** CClusScsiAddress::get_PathId() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::get_TargetId // // Description: // Get the disk's target id. // // Arguments: // pvarTargetId [OUT] - catches the target id. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusScsiAddress::get_TargetId( OUT VARIANT * pvarTargetId ) { //ASSERT( pvarTargetId != NULL ); HRESULT _hr = E_POINTER; if ( pvarTargetId != NULL ) { pvarTargetId->bVal = m_csa.TargetId; pvarTargetId->vt = VT_UI1; _hr = S_OK; } return _hr; } //*** CClusScsiAddress::get_TargetId() ///////////////////////////////////////////////////////////////////////////// //++ // // CClusScsiAddress::get_Lun // // Description: // Get the disk's Lun. // // Arguments: // pvarLun [OUT] - catches the Lun. // // Return Value: // S_OK if successful, or E_POINTER // //-- ///////////////////////////////////////////////////////////////////////////// STDMETHODIMP CClusScsiAddress::get_Lun( OUT VARIANT * pvarLun ) { //ASSERT( pvarLun != NULL ); HRESULT _hr = E_POINTER; if ( pvarLun != NULL ) { pvarLun->bVal = m_csa.Lun; pvarLun->vt = VT_UI1; _hr = S_OK; } return _hr; } //*** CClusScsiAddress::get_Lun()