Windows2003-3790/admin/pchealth/sr/rstrcore/drvtable.cpp
2020-09-30 16:53:55 +02:00

1209 lines
34 KiB
C++

/******************************************************************************
Copyright (c) 2000 Microsoft Corporation
Module Name:
drvtable.cpp
Abstract:
This file contains CRstrDriveInfo class and CreateDriveList function.
Revision History:
Seong Kook Khang (SKKhang) 07/20/00
created
******************************************************************************/
#include "stdwin.h"
#include "rstrcore.h"
#include "resource.h"
#include "..\shell\resource.h"
static LPCWSTR s_cszEmpty = L"";
WCHAR s_szSysDrv[MAX_PATH];
/////////////////////////////////////////////////////////////////////////////
//
// CRstrDriveInfo class
//
/////////////////////////////////////////////////////////////////////////////
//
// NOTE - 7/26/00 - skkhang
// CSRStr has one issue -- NULL return in case of memory failure. Even though
// the behavior is just same with regular C language pointer, many codes are
// blindly passing it to some external functions (e.g. strcmp) which does not
// gracefully handle NULL pointer. Ideally and eventually all of code should
// prevent any possible NULL pointers from getting passed to such functions,
// but for now, I'm using an alternative workaround -- GetID, GetMount, and
// GetLabel returns a static empty string instead of NULL pointer.
//
/////////////////////////////////////////////////////////////////////////////
// CRstrDriveInfo construction / destruction
CRstrDriveInfo::CRstrDriveInfo()
{
m_dwFlags = 0;
m_hIcon[0] = NULL;
m_hIcon[1] = NULL;
m_llDSMin = SR_DEFAULT_DSMIN * MEGABYTE;
m_llDSMax = SR_DEFAULT_DSMAX * MEGABYTE;
m_uDSUsage = 0;
m_fCfgExcluded = FALSE;
m_uCfgDSUsage = 0;
m_ulTotalBytes.QuadPart = 0;
}
CRstrDriveInfo::~CRstrDriveInfo()
{
if ( m_hIcon[0] != NULL )
::DestroyIcon( m_hIcon[0] );
if ( m_hIcon[1] != NULL )
::DestroyIcon( m_hIcon[1] );
}
BOOL CRstrDriveInfo::InitUsage (LPCWSTR cszID, INT64 llDSUsage)
{
TraceFunctEnter("CRstrDriveInfo::InitUsage");
//
// calculate max datastore size - max (12% of disk, 400mb)
//
// read % from registry
HKEY hKey = NULL;
DWORD dwPercent = SR_DEFAULT_DISK_PERCENT;
DWORD dwDSMax = SR_DEFAULT_DSMAX;
DWORD dwDSMin = IsSystem() ? SR_DEFAULT_DSMIN : SR_DEFAULT_DSMIN_NONSYSTEM;
ULARGE_INTEGER ulDummy;
DWORD dwRes = RegOpenKeyEx(HKEY_LOCAL_MACHINE,
s_cszSRRegKey, 0, KEY_READ, &hKey);
if (ERROR_SUCCESS == dwRes)
{
RegReadDWORD(hKey, s_cszDiskPercent, &dwPercent);
RegReadDWORD(hKey, s_cszDSMax, &dwDSMax);
if (IsSystem())
RegReadDWORD(hKey, s_cszDSMin, &dwDSMin);
RegCloseKey(hKey);
}
else
{
ErrorTrace(0, "! RegOpenKeyEx : %ld", dwRes);
}
// BUGBUG - this call may not always give total disk space (per-user quota)
ulDummy.QuadPart = 0;
if (FALSE == GetDiskFreeSpaceEx (cszID, &ulDummy, &m_ulTotalBytes, NULL))
{
ErrorTrace(0, "! GetDiskFreeSpaceEx : %ld", GetLastError());
goto done;
}
m_llDSMin = min(m_ulTotalBytes.QuadPart, (INT64) dwDSMin * MEGABYTE);
m_llDSMax = min(m_ulTotalBytes.QuadPart,
max( (INT64) dwDSMax * MEGABYTE,
(INT64) dwPercent * m_ulTotalBytes.QuadPart / 100 ));
if (m_llDSMax < m_llDSMin)
m_llDSMax = m_llDSMin;
//
// take floor of this value
//
m_llDSMax = ((INT64) (m_llDSMax / (INT64) MEGABYTE)) * (INT64) MEGABYTE;
DebugTrace(0, "m_llDSMax: %I64d, Size: %I64d", m_llDSMax, llDSUsage);
if ( ( llDSUsage == 0) || (llDSUsage > m_llDSMax) )
// not initialized, assume maximum
{
llDSUsage = m_llDSMax;
}
if ( ( llDSUsage - m_llDSMin > 0) && ( m_llDSMax - m_llDSMin > 0))
{
// + ((llDSUsage - m_llDSMin)/2) is to ensure that correct
// rounding off happens here
m_uDSUsage =( ((llDSUsage - m_llDSMin) * DSUSAGE_SLIDER_FREQ)
+ ((m_llDSMax - m_llDSMin)/2))/( m_llDSMax - m_llDSMin);
}
else
m_uDSUsage = 0;
m_uCfgDSUsage = m_uDSUsage;
done:
TraceFunctLeave();
return TRUE;
}
/////////////////////////////////////////////////////////////////////////////
// CRstrDriveInfo operations
BOOL
CRstrDriveInfo::Init( LPCWSTR cszID, DWORD dwFlags, INT64 llDSUsage, LPCWSTR cszMount, LPCWSTR cszLabel )
{
TraceFunctEnter("CRstrDriveInfo::Init");
BOOL fRet = FALSE;
LPCWSTR cszErr;
DWORD dwRes;
WCHAR szMount[MAX_PATH];
WCHAR szLabel[MAX_PATH];
m_dwFlags = dwFlags;
m_strID = cszID;
if ( !IsOffline() )
{
// Get Mount Point (drive letter or root directory path) from Unique Volume ID
//
if ( !::GetVolumePathNamesForVolumeName( cszID, szMount, MAX_PATH, &dwRes ) && GetLastError() != ERROR_MORE_DATA)
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::GetVolumePathNamesForVolumeName failed - %ls", cszErr);
// Instead of fail, use cszMount even if it may not accurate
::lstrcpy( szMount, cszMount );
}
else
{
szMount[MAX_PATH-1] = L'\0';
if (lstrlenW (szMount) > MAX_MOUNTPOINT_PATH)
{
// Instead of fail, use cszMount even if it may not accurate
::lstrcpy( szMount, cszMount );
}
}
// Get Volume Label from Mount Point
//
if ( !::GetVolumeInformation( cszID, szLabel, MAX_PATH, NULL, NULL, NULL, NULL, 0 ) )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::GetVolumeInformation failed - %ls", cszErr);
// Instead of fail, use cszLabel even if it may not accurate
::lstrcpy( szLabel, cszLabel );
}
}
if ( ( szMount[1] == L':' ) && ( szMount[2] == L'\\' ) && ( szMount[3] == L'\0' ) )
szMount[2] = L'\0';
m_strMount = szMount;
m_strLabel = szLabel;
InitUsage (cszID, llDSUsage);
m_fCfgExcluded = IsExcluded();
fRet = TRUE;
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::Init( LPCWSTR cszID, CDataStore *pDS, BOOL fOffline )
{
TraceFunctEnter("CRstrDriveInfo::Init");
BOOL fRet = FALSE;
LPCWSTR cszErr;
DWORD dwRes;
WCHAR szMount[MAX_PATH];
WCHAR szLabel[MAX_PATH];
m_strID = cszID;
UpdateStatus( pDS->GetFlags(), fOffline );
if ( !fOffline )
{
// Get Mount Point (drive letter or root directory path) from Unique Volume ID
//
if ( !::GetVolumePathNamesForVolumeName( cszID, szMount, MAX_PATH, &dwRes ) && GetLastError() != ERROR_MORE_DATA )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::GetVolumePathNamesForVolumeName failed - %ls", cszErr);
goto Exit;
}
else
{
szMount[MAX_PATH-1] = L'\0';
if (lstrlenW (szMount) > MAX_MOUNTPOINT_PATH)
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "mount point too long %ls", cszErr);
goto Exit;
}
}
// Get Volume Label from Mount Point
//
if ( !::GetVolumeInformation( cszID, szLabel, MAX_PATH, NULL, NULL, NULL, NULL, 0 ) )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::GetVolumeInformation failed - %ls", cszErr);
// this is not a fatal error - this can happen if the
// volume is being formatted for example. assume that the
// label is empty
szLabel[0]= L'\0';
}
}
else
{
::lstrcpyW (szMount, pDS->GetDrive());
::lstrcpyW (szLabel, pDS->GetLabel());
}
if ( ( szMount[1] == L':' ) && ( szMount[2] == L'\\' ) && ( szMount[3] == L'\0' ) )
szMount[2] = L'\0';
m_strMount = szMount;
m_strLabel = szLabel;
InitUsage (cszID, pDS->GetSizeLimit());
m_fCfgExcluded = IsExcluded();
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::LoadFromLog( HANDLE hfLog )
{
TraceFunctEnter("CRstrDriveInfo::LoadFromLog");
BOOL fRet = FALSE;
DWORD dwRes;
WCHAR szBuf[MAX_PATH];
// Read m_dwFlags
READFILE_AND_VALIDATE( hfLog, &m_dwFlags, sizeof(DWORD), dwRes, Exit );
// Read m_strID
if ( !::ReadStrAlign4( hfLog, szBuf ) )
{
ErrorTrace(0, "Cannot read drive ID...");
goto Exit;
}
if ( szBuf[0] == L'\0' )
{
ErrorTrace(0, "Drive Guid is empty...");
goto Exit;
}
m_strID = szBuf;
// Read m_strMount
if ( !::ReadStrAlign4( hfLog, szBuf ) )
{
ErrorTrace(0, "Cannot read drive mount point...");
goto Exit;
}
m_strMount = szBuf;
// Read m_strLabel
if ( !::ReadStrAlign4( hfLog, szBuf ) )
{
ErrorTrace(0, "Cannot read drive mount point...");
goto Exit;
}
m_strLabel = szBuf;
m_fCfgExcluded = IsExcluded();
// m_nCfgMaxSize = ...
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
void
CRstrDriveInfo::UpdateStatus( DWORD dwFlags, BOOL fOffline )
{
TraceFunctEnter("CRstrDriveInfo::UpdateStatus");
m_dwFlags = 0;
if ( fOffline )
{
m_dwFlags |= RDIF_OFFLINE;
}
else
{
// check if frozen
if ( ( dwFlags & SR_DRIVE_FROZEN ) != 0 )
m_dwFlags |= RDIF_FROZEN;
// check if system drive
if ( ( dwFlags & SR_DRIVE_SYSTEM ) != 0 )
{
m_dwFlags |= RDIF_SYSTEM;
}
else
{
// if not system drive, simply use MONITORED flag of drive table
if ( ( dwFlags & SR_DRIVE_MONITORED ) == 0 )
m_dwFlags |= RDIF_EXCLUDED;
}
}
DebugTrace(0, "Status has been updated, m_dwFlags=%08X", m_dwFlags);
TraceFunctLeave();
}
/////////////////////////////////////////////////////////////////////////////
// CRstrDriveInfo - methods
DWORD
CRstrDriveInfo::GetFlags()
{
return( m_dwFlags );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::IsExcluded()
{
return( ( m_dwFlags & RDIF_EXCLUDED ) != 0 );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::IsFrozen()
{
return( ( m_dwFlags & RDIF_FROZEN ) != 0 );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::IsOffline()
{
return( ( m_dwFlags & RDIF_OFFLINE ) != 0 );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::IsSystem()
{
return( ( m_dwFlags & RDIF_SYSTEM ) != 0 );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::RefreshStatus()
{
TraceFunctEnter("CRstrDriveInfo::RefreshStatus");
BOOL fRet = FALSE;
LPCWSTR cszErr;
WCHAR szDTFile[MAX_PATH];
DWORD dwRes;
CDriveTable cDrvTable;
CDataStore *pDS;
::MakeRestorePath( szDTFile, s_szSysDrv, NULL );
::PathAppend( szDTFile, s_cszDriveTable );
DebugTrace(0, "Loading drive table - %ls", szDTFile);
dwRes = cDrvTable.LoadDriveTable( szDTFile );
if ( dwRes != ERROR_SUCCESS )
{
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "Cannot load a drive table - %ls", cszErr);
ErrorTrace(0, " szDTFile: '%ls'", szDTFile);
goto Exit;
}
dwRes = cDrvTable.RemoveDrivesFromTable();
if ( dwRes != ERROR_SUCCESS )
{
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "CDriveTable::RemoveDrivesFromTable failed - %ls", cszErr);
// ignore error
}
pDS = cDrvTable.FindGuidInTable( (LPWSTR)GetID() );
if ( pDS == NULL )
UpdateStatus( 0, TRUE );
else
UpdateStatus( pDS->GetFlags(), FALSE );
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
LPCWSTR
CRstrDriveInfo::GetID()
{
return( ( m_strID.Length() > 0 ) ? m_strID : s_cszEmpty );
}
/////////////////////////////////////////////////////////////////////////////
LPCWSTR
CRstrDriveInfo::GetMount()
{
return( ( m_strMount.Length() > 0 ) ? m_strMount : s_cszEmpty );
}
/////////////////////////////////////////////////////////////////////////////
LPCWSTR
CRstrDriveInfo::GetLabel()
{
return( ( m_strLabel.Length() > 0 ) ? m_strLabel : s_cszEmpty );
}
/////////////////////////////////////////////////////////////////////////////
void
CRstrDriveInfo::SetMountAndLabel( LPCWSTR cszMount, LPCWSTR cszLabel )
{
TraceFunctEnter("CRstrDriveInfo::SetMountAndLabel");
m_strMount = cszMount;
m_strLabel = cszLabel;
TraceFunctLeave();
}
/////////////////////////////////////////////////////////////////////////////
HICON
CRstrDriveInfo::GetIcon( BOOL fSmall )
{
TraceFunctEnter("CRstrDriveInfo::GetIcon");
LPCWSTR cszErr;
int nIdx = fSmall ? 0 : 1;
int cxIcon, cyIcon;
HICON hIcon;
if ( m_hIcon[nIdx] != NULL )
goto Exit;
cxIcon = ::GetSystemMetrics( fSmall ? SM_CXSMICON : SM_CXICON );
cyIcon = ::GetSystemMetrics( fSmall ? SM_CYSMICON : SM_CYICON );
hIcon = (HICON)::LoadImage( g_hInst, MAKEINTRESOURCE(IDI_DRIVE_FIXED),
IMAGE_ICON, cxIcon, cyIcon, LR_DEFAULTCOLOR );
if ( hIcon == NULL )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::LoadImage failed - %ls", cszErr);
goto Exit;
}
m_hIcon[nIdx] = hIcon;
Exit:
TraceFunctLeave();
return( m_hIcon[nIdx] );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::SaveToLog( HANDLE hfLog )
{
TraceFunctEnter("CRstrDriveInfo::SaveToLog");
BOOL fRet = FALSE;
BYTE pbBuf[7*MAX_PATH];
DWORD dwSize = 0;
DWORD dwRes;
*((DWORD*)pbBuf) = m_dwFlags;
dwSize += sizeof(DWORD);
dwSize += ::StrCpyAlign4( pbBuf+dwSize, m_strID );
dwSize += ::StrCpyAlign4( pbBuf+dwSize, m_strMount );
dwSize += ::StrCpyAlign4( pbBuf+dwSize, m_strLabel );
WRITEFILE_AND_VALIDATE( hfLog, pbBuf, dwSize, dwRes, Exit );
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
UINT
CRstrDriveInfo::GetDSUsage()
{
TraceFunctEnter("CRstrDriveInfo::GetDSUsage");
TraceFunctLeave();
return( m_uDSUsage );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::GetUsageText( LPWSTR szUsage )
{
TraceFunctEnter("CRstrDriveInfo::GetUsageText");
INT64 llUsage;
int nPercent;
int nUsage;
if (m_llDSMax - m_llDSMin > 0)
llUsage = m_llDSMin + ( m_llDSMax - m_llDSMin ) * m_uCfgDSUsage / DSUSAGE_SLIDER_FREQ;
else
llUsage = m_llDSMin;
if (m_ulTotalBytes.QuadPart != 0)
{
// the m_ulTotalBytes.QuadPart/200 addition is to ensure that
// the correct round off happens
nPercent = (llUsage + (m_ulTotalBytes.QuadPart/200)) * 100/
m_ulTotalBytes.QuadPart;
}
else nPercent = 0;
nUsage = llUsage / ( 1024 * 1024 );
::wsprintf( szUsage, L"%d%% (%d MB)", nPercent, nUsage );
TraceFunctLeave();
return( TRUE );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::GetCfgExcluded( BOOL *pfExcluded )
{
TraceFunctEnter("CRstrDriveInfo::GetCfgExcluded");
BOOL fRet = FALSE;
if ( m_fCfgExcluded != IsExcluded() )
{
*pfExcluded = m_fCfgExcluded;
fRet = TRUE;
}
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
void
CRstrDriveInfo::SetCfgExcluded( BOOL fExcluded )
{
TraceFunctEnter("CRstrDriveInfo::SetCfgExcluded");
m_fCfgExcluded = fExcluded;
TraceFunctLeave();
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::GetCfgDSUsage( UINT *puPos )
{
TraceFunctEnter("CRstrDriveInfo::GetCfgDSUsage");
BOOL fRet = FALSE;
if ( m_uCfgDSUsage != m_uDSUsage )
{
*puPos = m_uCfgDSUsage;
fRet = TRUE;
}
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
void
CRstrDriveInfo::SetCfgDSUsage( UINT uPos )
{
TraceFunctEnter("CRstrDriveInfo::SetCfgDSUsage");
m_uCfgDSUsage = uPos;
TraceFunctLeave();
}
/////////////////////////////////////////////////////////////////////////////
void
CloseRestoreUI()
{
WCHAR szPath[MAX_PATH], szTitle[MAX_PATH] = L"";
if (ExpandEnvironmentStrings(L"%windir%\\system32\\restore\\rstrui.exe", szPath, MAX_PATH))
{
if (ERROR_SUCCESS == SRLoadString(szPath, IDS_RESTOREUI_TITLE, szTitle, MAX_PATH))
{
HWND hWnd = FindWindow(CLSNAME_RSTRSHELL, szTitle);
if (hWnd != NULL)
PostMessage(hWnd, WM_CLOSE, 0, 0);
}
}
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::ApplyConfig( HWND hWnd )
{
TraceFunctEnter("CRstrDriveInfo::ApplyConfig");
BOOL fRet = FALSE;
LPCWSTR cszErr;
INT64 llUsage;
DWORD dwRes;
if ( m_fCfgExcluded != IsExcluded() )
{
if ( m_fCfgExcluded )
{
WCHAR szTitle[MAX_STR];
WCHAR szMsg[MAX_STR+2*MAX_PATH];
// Confirm if it's ok to turn drive or SR off.
::LoadString( g_hInst, IDS_SYSTEMRESTORE, szTitle,
sizeof(szTitle)/sizeof(WCHAR) );
if ( IsSystem() )
::LoadString( g_hInst, IDS_CONFIRM_TURN_SR_OFF, szMsg,
sizeof(szMsg)/sizeof(WCHAR) );
else
{
::SRFormatMessage( szMsg, IDS_CONFIRM_TURN_DRV_OFF, GetLabel() ? GetLabel() : L"", GetMount() );
}
if ( ::MessageBox( hWnd, szMsg, szTitle, MB_YESNO ) == IDNO )
{
m_fCfgExcluded = IsExcluded();
goto Exit;
}
//
// if disabling all of SR, close the wizard if open
//
if (IsSystem())
{
CloseRestoreUI();
}
dwRes = ::DisableSR( m_strID );
if ( dwRes != ERROR_SUCCESS )
{
ShowSRErrDlg (IDS_ERR_SR_ON_OFF);
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "::DisableSR failed - %ls", cszErr);
goto Exit;
}
m_dwFlags |= RDIF_EXCLUDED;
}
else
{
//
// make a synchronous call to enable sr
// this will block till the firstrun checkpoint is created
// and the service is fully initialized
//
dwRes = ::EnableSREx( m_strID, TRUE );
if ( dwRes != ERROR_SUCCESS )
{
ShowSRErrDlg (IDS_ERR_SR_ON_OFF);
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "::EnableSR failed - %ls", cszErr);
goto Exit;
}
m_dwFlags &= ~RDIF_EXCLUDED;
}
}
if ( m_uCfgDSUsage != m_uDSUsage )
{
if (m_llDSMax - m_llDSMin > 0)
llUsage = m_llDSMin + (m_llDSMax - m_llDSMin)* m_uCfgDSUsage /DSUSAGE_SLIDER_FREQ;
else
llUsage = m_llDSMin;
dwRes = ::SRUpdateDSSize( m_strID, llUsage );
if ( dwRes != ERROR_SUCCESS )
{
LPCWSTR cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "::SRUpdateDriveTable failed - %ls", cszErr);
goto Exit;
}
m_uDSUsage = m_uCfgDSUsage;
}
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
CRstrDriveInfo::Release()
{
TraceFunctEnter("CRstrDriveInfo::Release");
delete this;
TraceFunctLeave();
return( TRUE );
}
/////////////////////////////////////////////////////////////////////////////
//
// Helper Function
//
/////////////////////////////////////////////////////////////////////////////
//
// Enumerate Volumes without Drive Table if SR is disabled and DS not exists.
//
BOOL
EnumVolumes( CRDIArray &aryDrv )
{
TraceFunctEnter("EnumVolumes");
BOOL fRet = FALSE;
LPCWSTR cszErr;
HANDLE hEnumVol = INVALID_HANDLE_VALUE;
WCHAR szVolume[MAX_PATH];
WCHAR szMount[MAX_PATH];
DWORD cbMount;
CRstrDriveInfo *pDrv = NULL;
DWORD dwFlags;
hEnumVol = ::FindFirstVolume( szVolume, MAX_PATH );
if ( hEnumVol == INVALID_HANDLE_VALUE )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::FindFirstVolume failed - %ls", cszErr);
goto Exit;
}
// dummy space for system drive
if ( !aryDrv.AddItem( NULL ) )
goto Exit;
do
{
HANDLE hfDrv;
DebugTrace(0, "Guid=%ls", szVolume);
if ( !::GetVolumePathNamesForVolumeName( szVolume, szMount, MAX_PATH, &cbMount ) && GetLastError() != ERROR_MORE_DATA)
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::GetVolumePathNamesForVolumeName failed - %ls", cszErr);
continue;
}
else
{
szMount[MAX_PATH-1] = L'\0';
if (lstrlenW (szMount) > MAX_MOUNTPOINT_PATH)
continue;
}
DebugTrace(0, " Mount=%ls", szMount);
if ( ::GetDriveType( szMount ) != DRIVE_FIXED )
{
DebugTrace(0, "Non-fixed drive");
// includes only the fixed drives.
continue;
}
hfDrv = ::CreateFile( szVolume, GENERIC_READ, FILE_SHARE_READ | FILE_SHARE_WRITE,
NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS, NULL );
if ( hfDrv == INVALID_HANDLE_VALUE )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "::CreateFile(volume) failed - %ls", cszErr);
// probably an unformatted drive.
continue;
}
::CloseHandle( hfDrv );
pDrv = new CRstrDriveInfo;
if ( pDrv == NULL )
{
FatalTrace(0, "Insufficient memory...");
goto Exit;
}
dwFlags = RDIF_EXCLUDED;
if ( ::IsSystemDrive( szVolume ) )
{
dwFlags |= RDIF_SYSTEM;
if ( !aryDrv.SetItem( 0, pDrv ) )
goto Exit;
}
else
{
if ( !aryDrv.AddItem( pDrv ) )
goto Exit;
}
if ( !pDrv->Init( szVolume, dwFlags, 0, szMount, NULL ) )
goto Exit;
pDrv = NULL;
}
while ( ::FindNextVolume( hEnumVol, szVolume, MAX_PATH ) );
fRet = TRUE;
Exit:
if ( pDrv != NULL )
if ( hEnumVol != INVALID_HANDLE_VALUE )
::FindVolumeClose( hEnumVol );
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
LoadDriveTable( LPCWSTR cszRPDir, CRDIArray &aryDrv, BOOL fRemoveDrives)
{
TraceFunctEnter("LoadDriveTable");
BOOL fRet = FALSE;
LPCWSTR cszErr;
WCHAR szDTFile[MAX_PATH];
DWORD dwRes;
CDriveTable cDrvTable;
SDriveTableEnumContext sDTEnum = { NULL, 0 };
CDataStore *pDS;
CRstrDriveInfo *pDrv = NULL;
BOOL fOffline;
::MakeRestorePath( szDTFile, s_szSysDrv, cszRPDir );
::PathAppend( szDTFile, s_cszDriveTable );
DebugTrace(0, "Loading drive table - %ls", szDTFile);
dwRes = cDrvTable.LoadDriveTable( szDTFile );
if ( dwRes != ERROR_SUCCESS )
{
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "Cannot load a drive table - %ls", cszErr);
ErrorTrace(0, " szDTFile: '%ls'", szDTFile);
goto Exit;
}
// If this is for the current drive table, try to update information
// about removed volumes.
if ( cszRPDir == NULL )
{
if (fRemoveDrives)
cDrvTable.RemoveDrivesFromTable();
else
{
sDTEnum.Reset();
pDS = cDrvTable.FindFirstDrive (sDTEnum);
while (pDS != NULL)
{
pDS->IsVolumeDeleted(); // mark deleted volumes as inactive
pDS = cDrvTable.FindNextDrive( sDTEnum );
}
}
}
sDTEnum.Reset();
pDS = cDrvTable.FindFirstDrive( sDTEnum );
while ( pDS != NULL )
{
int i;
LPCWSTR cszGuid = pDS->GetGuid();
DebugTrace(0, "Drive: %ls %ls", pDS->GetDrive(), cszGuid);
if ( cszRPDir != NULL ) // not the current restore point
{
for ( i = aryDrv.GetUpperBound(); i >= 0; i-- )
{
CRstrDriveInfo *pExist = aryDrv.GetItem( i );
if ( ::lstrcmpi( cszGuid, pExist->GetID() ) == 0 )
{
// Match has been found. Check if it's offline, in which
// case mount point and volume label should be updated to the
// latest ones.
if ( pExist->IsOffline() )
pExist->SetMountAndLabel( pDS->GetDrive(), pDS->GetLabel() );
break;
}
pDrv = NULL;
}
if ( i >= 0 )
goto NextDrv;
}
pDrv = new CRstrDriveInfo;
if ( pDrv == NULL )
{
FatalTrace(0, "Insufficient memory...");
goto Exit;
}
//
// mark a drive as offline if it's not in the current restore point
// or it's inactive in the current restore point
//
fOffline = (cszRPDir != NULL) || !(pDS->GetFlags() & SR_DRIVE_ACTIVE);
if ( !pDrv->Init( cszGuid, pDS, fOffline ) )
goto Exit;
if (( pDrv->GetMount() == NULL ) || ( (pDrv->GetMount())[0] == L'\0' ))
{
pDrv->Release();
goto NextDrv;
}
if ( pDrv->IsSystem() )
{
if ( !aryDrv.SetItem( 0, pDrv ) )
goto Exit;
}
else
{
if ( !aryDrv.AddItem( pDrv ) )
goto Exit;
}
pDrv = NULL;
NextDrv:
pDS = cDrvTable.FindNextDrive( sDTEnum );
}
fRet = TRUE;
Exit:
if ( !fRet )
SAFE_RELEASE(pDrv);
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
BOOL
UpdateDriveList( CRDIArray &aryDrv )
{
TraceFunctEnter("UpdateDriveTable");
BOOL fRet = FALSE;
LPCWSTR cszErr;
DWORD dwDisable = 0;
WCHAR szDTFile[MAX_PATH];
DWORD dwRes;
CDriveTable cDrvTable;
CDataStore *pDS;
CRstrDriveInfo *pDrv;
int i;
// Check if SR is disabled
if ( ::SRGetRegDword( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszDisableSR, &dwDisable ) )
if ( dwDisable != 0 )
{
for ( i = aryDrv.GetUpperBound(); i >= 0; i-- )
{
pDrv = (CRstrDriveInfo*)aryDrv[i];
pDrv->UpdateStatus( SR_DRIVE_FROZEN, FALSE );
}
goto Done;
}
::MakeRestorePath( szDTFile, s_szSysDrv, NULL );
::PathAppend( szDTFile, s_cszDriveTable );
DebugTrace(0, "Loading drive table - %ls", szDTFile);
dwRes = cDrvTable.LoadDriveTable( szDTFile );
if ( dwRes != ERROR_SUCCESS )
{
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "Cannot load a drive table - %ls", cszErr);
ErrorTrace(0, " szDTFile: '%ls'", szDTFile);
goto Exit;
}
dwRes = cDrvTable.RemoveDrivesFromTable();
if ( dwRes != ERROR_SUCCESS )
{
cszErr = ::GetSysErrStr( dwRes );
ErrorTrace(0, "CDriveTable::RemoveDrivesFromTable failed - %ls", cszErr);
// ignore error
}
for ( i = aryDrv.GetUpperBound(); i >= 0; i-- )
{
pDrv = (CRstrDriveInfo*)aryDrv[i];
pDS = cDrvTable.FindGuidInTable( (LPWSTR)pDrv->GetID() );
if ( ( pDS == NULL ) || ( pDS->GetDrive() == NULL ) || ( (pDS->GetDrive())[0] == L'\0' ) )
pDrv->UpdateStatus( 0, TRUE );
else
pDrv->UpdateStatus( pDS->GetFlags(), FALSE );
}
Done:
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
//
// CreateAndLoadDriveInfoInstance
//
// This routine creates a CRstrDriveInfo class instance and load the content
// from a log file.
//
/////////////////////////////////////////////////////////////////////////////
BOOL
CreateAndLoadDriveInfoInstance( HANDLE hfLog, CRstrDriveInfo **ppRDI )
{
TraceFunctEnter("CreateAndLoadDriveInfoInstance");
BOOL fRet = FALSE;
CRstrDriveInfo *pRDI=NULL;
if ( ppRDI == NULL )
{
ErrorTrace(0, "Invalid parameter, ppRDI is NULL.");
goto Exit;
}
*ppRDI = NULL;
pRDI = new CRstrDriveInfo;
if ( pRDI == NULL )
{
ErrorTrace(0, "Insufficient memory...");
goto Exit;
}
if ( !pRDI->LoadFromLog( hfLog ) )
goto Exit;
*ppRDI = pRDI;
fRet = TRUE;
Exit:
if ( !fRet )
SAFE_RELEASE(pRDI);
TraceFunctLeave();
return( fRet );
}
/////////////////////////////////////////////////////////////////////////////
//
// CreateDriveList
//
// This routine creates a drive list consists of CDriveInfo class instances.
//
/////////////////////////////////////////////////////////////////////////////
BOOL
CreateDriveList( int nRP, CRDIArray &aryDrv, BOOL fRemoveDrives )
{
TraceFunctEnter("CreateDriveList");
BOOL fRet = FALSE;
LPCWSTR cszErr;
DWORD fDisable;
if ( !::GetSystemDrive( s_szSysDrv ) )
{
cszErr = ::GetSysErrStr();
ErrorTrace(0, "Cannot get system drive - %ls", cszErr);
goto Exit;
}
DebugTrace(0, "SystemDrive=%ls", s_szSysDrv);
// Check if SR is disabled
if ( !::SRGetRegDword( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszDisableSR, &fDisable ) )
{
DebugTrace(0, "Cannot get disable reg key");
goto Exit;
}
if ( fDisable )
{
DebugTrace(0, "SR is DISABLED!!!");
// Enumerate instead of reading drive table...
if ( !EnumVolumes( aryDrv ) )
goto Exit;
}
else
{
// dummy space for system drive
if ( !aryDrv.AddItem( NULL ) )
goto Exit;
// process the current drive table...
if ( !LoadDriveTable( NULL, aryDrv, fRemoveDrives ) )
{
DebugTrace(0, "Loading current drive table failed");
goto Exit;
}
if ( nRP > 0 )
{
CRestorePointEnum cEnum( s_szSysDrv, FALSE, FALSE );
CRestorePoint cRP;
DWORD dwRes;
dwRes = cEnum.FindFirstRestorePoint( cRP );
if ( dwRes != ERROR_SUCCESS && dwRes != ERROR_FILE_NOT_FOUND )
{
cszErr = ::GetSysErrStr(dwRes);
ErrorTrace(0, "CRestorePointEnum::FindFirstRestorePoint failed - %ls", cszErr);
goto Exit;
}
while ( (dwRes == ERROR_SUCCESS || dwRes == ERROR_FILE_NOT_FOUND) && ( cRP.GetNum() >= nRP ))
{
dwRes = cEnum.FindNextRestorePoint( cRP );
if ( dwRes == ERROR_NO_MORE_ITEMS )
break;
if ( dwRes != ERROR_SUCCESS && dwRes != ERROR_FILE_NOT_FOUND )
{
cszErr = ::GetSysErrStr(dwRes);
ErrorTrace(0, "CRestorePointEnum::FindNextRestorePoint failed - %ls", cszErr);
goto Exit;
}
DebugTrace(0, "RPNum=%d", cRP.GetNum());
if ( cRP.GetNum() >= nRP )
{
// process drive table of each RP...
if ( !LoadDriveTable( cRP.GetDir(), aryDrv, fRemoveDrives))
{
// The last restore point does not have drive table...
// simply ignore it.
}
}
}
}
}
fRet = TRUE;
Exit:
TraceFunctLeave();
return( fRet );
}
// end of file