2020-09-30 16:53:55 +02:00

2497 lines
67 KiB
C++

/*****************************************************************************
*
* Copyright (c) 2000 Microsoft Corporation
*
* Module Name:
* datastormgr.cpp
*
* Abstract:
* CDataStoreMgr class functions
*
* Revision History:
* Brijesh Krishnaswami (brijeshk) 03/28/2000
* created
*
*****************************************************************************/
#include "datastormgr.h"
#include "srapi.h"
#include "srconfig.h"
#include "evthandler.h"
#include "ntservice.h"
#include "ntservmsg.h"
#include <coguid.h>
#include <rpc.h>
#include <stdio.h>
#include <shlwapi.h>
#ifdef THIS_FILE
#undef THIS_FILE
#endif
static char __szTraceSourceFile[] = __FILE__;
#define THIS_FILE __szTraceSourceFile
CDataStoreMgr * g_pDataStoreMgr = NULL; // the global instance
//
// we can't use %s for the volume label because it can contain spaces
// so we look for all characters until the end-of-line
//
static WCHAR gs_wcsScanFormat[] = L"%[^/]/%s %x %i %i %[^\r]\n";
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::CDriveTable ()
//
// Synopsis:
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDriveTable::CDriveTable ()
{
_pdtNext = NULL;
_nLastDrive = 0;
_fDirty = FALSE;
_fLockInit = FALSE;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::~CDriveTable
//
// Synopsis: delete all drive table entries
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDriveTable::~CDriveTable ()
{
for (int i = 0; i < _nLastDrive; i++)
{
if (_rgDriveTable[i] != NULL)
{
delete _rgDriveTable[i];
_rgDriveTable[i] = NULL;
}
}
_nLastDrive = 0;
if (_pdtNext != NULL)
{
delete _pdtNext;
_pdtNext = NULL;
}
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::CreateNewEntry
//
// Synopsis: populate the table with this datastore object
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::CreateNewEntry (CDataStore *pds)
{
if (_nLastDrive < DRIVE_TABLE_SIZE)
{
_rgDriveTable[_nLastDrive] = pds;
_nLastDrive++;
return ERROR_SUCCESS;
}
else
{
// this table is full, allocate a new one if needed
if (_pdtNext == NULL)
{
_pdtNext = new CDriveTable();
if (_pdtNext == NULL)
{
return ERROR_NOT_ENOUGH_MEMORY;
}
}
// Now add the entry to the new table
return _pdtNext->CreateNewEntry (pds);
}
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::FindDriveInTable
//
// Synopsis: return datastore object matching this drive
//
// Arguments: can pass in dos drive letter, mount point path, or volume guid
//
// Returns: pointer to corresponding datastore object
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStore * CDriveTable::FindDriveInTable (WCHAR *pwszDrive) const
{
if (NULL == pwszDrive) // not a valid drive
return NULL;
if (0 == wcsncmp(pwszDrive, L"\\\\?\\Volume", 10))
return FindGuidInTable(pwszDrive);
for (const CDriveTable *pdt = this; pdt != NULL; pdt = pdt->_pdtNext)
{
for (int i = 0; i < pdt->_nLastDrive; i++)
{
if ((pdt->_rgDriveTable[i] != NULL) &&
lstrcmpi (pwszDrive, pdt->_rgDriveTable[i]->GetDrive()) == 0)
{
return pdt->_rgDriveTable[i];
}
}
}
return NULL;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::RemoveDrivesFromTable
//
// Synopsis: remove inactive drive table entries
//
// Arguments:
//
// Returns: Win32 error code
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::RemoveDrivesFromTable ()
{
DWORD dwErr = ERROR_SUCCESS;
tenter("removedrivesfromtable");
for (CDriveTable *pdt = this; pdt != NULL; pdt = pdt->_pdtNext)
{
for (int i = 0; i < pdt->_nLastDrive; i++)
{
if (pdt->_rgDriveTable[i] != NULL &&
pdt->_rgDriveTable[i]->IsVolumeDeleted())
{
trace(0, "removing %S from drivetable", pdt->_rgDriveTable[i]->GetDrive());
delete pdt->_rgDriveTable[i];
pdt->_rgDriveTable[i] = NULL;
if (i == pdt->_nLastDrive - 1)
--(pdt->_nLastDrive);
_fDirty = TRUE;
}
}
}
tleave();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::FindGuidInTable
//
// Synopsis: get the drive table entry matching the volume GUID
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStore * CDriveTable::FindGuidInTable (WCHAR *pwszGuid) const
{
if (NULL == pwszGuid) // not a valid string
return NULL;
for (const CDriveTable *pdt = this; pdt != NULL; pdt = pdt->_pdtNext)
{
for (int i = 0; i < pdt->_nLastDrive; i++)
{
if (pdt->_rgDriveTable[i] != NULL &&
lstrcmp (pwszGuid, pdt->_rgDriveTable[i]->GetGuid()) == 0)
{
return pdt->_rgDriveTable[i];
}
}
}
return NULL;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::FindSystemDrive
//
// Synopsis: get the drive table entry for the system drive
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStore * CDriveTable::FindSystemDrive () const
{
for (const CDriveTable *pdt = this; pdt != NULL; pdt = pdt->_pdtNext)
{
for (int i = 0; i < pdt->_nLastDrive; i++)
{
if (pdt->_rgDriveTable[i] != NULL &&
pdt->_rgDriveTable[i]->GetFlags() & SR_DRIVE_SYSTEM)
{
return pdt->_rgDriveTable[i];
}
}
}
return NULL;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::ForAllDrives
//
// Synopsis: Execute this CDataStore method for all drives
//
// Arguments: [pMethod] -- CDataStore method to call
// [lParam] -- parameter to that method
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::ForAllDrives (PDATASTOREMETHOD pMethod, LONG_PTR lParam)
{
TENTER ("ForAllDrives");
DWORD dwErr = ERROR_SUCCESS;
for (CDriveTable *pdt = this; pdt != NULL; pdt = pdt->_pdtNext)
{
for (int i = 0; i < pdt->_nLastDrive; i++)
{
if (pdt->_rgDriveTable[i] != NULL)
{
dwErr = (pdt->_rgDriveTable[i]->*pMethod) (lParam);
if (dwErr != ERROR_SUCCESS)
{
TRACE(0, "%S ForAllDrives failed %x", pdt->_rgDriveTable[i]->GetDrive(), dwErr);
dwErr = ERROR_SUCCESS;
}
}
}
}
if (dwErr == ERROR_SUCCESS && _fDirty)
dwErr = SaveDriveTable ((CRestorePoint *) NULL);
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::ForOneOrAllDrives
//
// Synopsis: Execute this CDataStore method for one or all drives
//
// Arguments: [pwszDrive] -- drive to execute method
// [pMethod] -- CDataStore method to call
// [lParam] -- parameter to that method
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::ForOneOrAllDrives (WCHAR *pwszDrive,
PDATASTOREMETHOD pMethod,
LONG_PTR lParam)
{
DWORD dwErr = ERROR_SUCCESS;
if (pwszDrive == NULL)
{
dwErr = ForAllDrives (pMethod, lParam);
}
else
{
CDataStore *pds = FindDriveInTable (pwszDrive);
dwErr = (pds != NULL) ? (pds->*pMethod)(lParam) : ERROR_INVALID_DRIVE;
if (dwErr == ERROR_SUCCESS && _fDirty)
dwErr = SaveDriveTable ((CRestorePoint *) NULL);
}
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::FindMountPoint
//
// Synopsis: Given a volume GUID, find a mount point that points to it
//
// Arguments: [pwszGuid] -- input volume GUID
// [pwszPath] -- output path to mount point
//
// Returns: Win32 error
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::FindMountPoint (WCHAR *pwszGuid, WCHAR *pwszPath) const
{
TENTER ("CDriveTable::FindMountPoint");
DWORD dwErr = ERROR_MORE_DATA; // initialize for loop
WCHAR * pwszMount = NULL; // MultiSz string
DWORD dwMountLen = MAX_PATH; // initial buffer size
DWORD dwChars = 0;
pwszPath[0] = L'\0';
while (dwErr == ERROR_MORE_DATA)
{
dwErr = ERROR_SUCCESS;
pwszMount = new WCHAR [dwMountLen];
if (pwszMount == NULL)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto Err;
}
if (FALSE == GetVolumePathNamesForVolumeNameW (pwszGuid,
pwszMount,
dwMountLen,
&dwChars ))
{
dwErr = GetLastError();
delete [] pwszMount; // free the existing buffer
pwszMount = NULL;
dwMountLen *= 2; // double the length
}
}
if (ERROR_SUCCESS == dwErr && pwszMount != NULL)
{
if (L'\0' == pwszMount[0]) // empty string
{
dwErr = ERROR_NOT_DOS_DISK; // no drive letter or mount point
}
else if (lstrlenW (pwszMount) < MAX_MOUNTPOINT_PATH)
{
lstrcpyW (pwszPath, pwszMount); // copy the first string
}
else
{
dwErr = ERROR_BAD_PATHNAME; // 1st path too long
}
}
Err:
if (pwszMount != NULL)
delete [] pwszMount;
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::AddDriveToTable
//
// Synopsis: add the volume to the drive table
//
// Arguments: [pwszGuid] -- the volume GUID
//
// Returns: Win32 error code
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::AddDriveToTable(WCHAR *pwszDrive, WCHAR *pwszGuid)
{
DWORD dwErr = ERROR_SUCCESS;
if (NULL == pwszDrive)
return ERROR_INVALID_DRIVE;
// Eventually, this routine will require pwszGuid to be non-NULL
CDataStore *pds = pwszGuid != NULL ? FindGuidInTable (pwszGuid) :
FindDriveInTable (pwszDrive);
if (pds != NULL) // found the drive already
{
if (lstrcmpiW (pwszDrive, pds->GetDrive()) != 0) // drive rename
pds->SetDrive (pwszDrive);
return dwErr;
}
pds = new CDataStore(this);
if (pds == NULL)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
return dwErr;
}
dwErr = pds->Initialize (pwszDrive, pwszGuid);
if (dwErr == ERROR_SUCCESS)
{
dwErr = CreateNewEntry (pds);
_fDirty = TRUE;
}
if (dwErr != ERROR_SUCCESS) // clean up on error
{
delete pds;
}
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::FindFirstDrive
// CDriveTable::FindNextDrive
//
// Synopsis: loop through drive table entries
//
// Arguments: [dtec] -- enumeration context
//
// Returns: CDataStore object pointer
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStore * CDriveTable::FindFirstDrive (SDriveTableEnumContext & dtec) const
{
for (dtec._pdt = this; dtec._pdt != NULL; dtec._pdt = dtec._pdt->_pdtNext)
{
for (dtec._iIndex = 0; dtec._iIndex < dtec._pdt->_nLastDrive; dtec._iIndex++)
{
CDataStore *pds = dtec._pdt->_rgDriveTable[dtec._iIndex];
if (pds != NULL)
{
return pds;
}
}
}
return NULL;
}
CDataStore * CDriveTable::FindNextDrive (SDriveTableEnumContext & dtec) const
{
for (; dtec._pdt != NULL; dtec._pdt = dtec._pdt->_pdtNext)
{
dtec._iIndex++;
for (; dtec._iIndex < dtec._pdt->_nLastDrive; dtec._iIndex++)
{
CDataStore *pds = dtec._pdt->_rgDriveTable[dtec._iIndex];
if (pds != NULL)
{
return pds;
}
}
dtec._iIndex = 0;
}
return NULL;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::Merge
//
// Synopsis: loop through drive table entries and merge
//
// Arguments: [dt] -- drive table read from disk
//
// Returns: Win32 error
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::Merge (CDriveTable &dt)
{
DWORD dwErr = ERROR_SUCCESS;
BOOL fApplyDefaults = FALSE;
SDriveTableEnumContext dtec = {NULL, 0};
HKEY hKeyGP = NULL;
if (ERROR_SUCCESS == RegOpenKeyEx (HKEY_LOCAL_MACHINE,
s_cszGroupPolicy,
0,
KEY_READ,
&hKeyGP))
{
DWORD dwRet = 0;
if (ERROR_SUCCESS == RegReadDWORD(hKeyGP, s_cszDisableConfig, &dwRet))
{
if (dwRet == 0)
fApplyDefaults = TRUE;
}
RegCloseKey (hKeyGP);
}
CDataStore *pds = dt.FindFirstDrive (dtec);
while (pds != NULL)
{
CDataStore *pdsFound = FindGuidInTable (pds->GetGuid());
if (pdsFound != NULL)
{
pds->GetVolumeInfo(); // refresh the volume flags
if (fApplyDefaults)
{
pds->MonitorDrive(TRUE); // make sure drive is monitored
pds->SetSizeLimit(0); // set max datastore size to default
}
// don't overwrite the newer drive letter and label
dwErr = pdsFound->LoadDataStore (NULL, pds->GetGuid(), NULL,
pds->GetFlags() | SR_DRIVE_ACTIVE, pds->GetNumChangeLogs(), pds->GetSizeLimit());
}
else
{
CDataStore *pdsNew = new CDataStore ( this);
if (pdsNew == NULL)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
return dwErr;
}
dwErr = pdsNew->LoadDataStore (pds->GetDrive(), pds->GetGuid(),
pds->GetLabel(),
pds->GetFlags() & ~SR_DRIVE_ACTIVE,
pds->GetNumChangeLogs(), pds->GetSizeLimit());
if (dwErr != ERROR_SUCCESS)
{
delete pdsNew;
pdsNew = NULL;
goto Err;
}
dwErr = CreateNewEntry (pdsNew);
}
if (dwErr != ERROR_SUCCESS)
goto Err;
pds = dt.FindNextDrive (dtec);
}
if (fApplyDefaults && g_pEventHandler != NULL)
{
dwErr = g_pEventHandler->SRUpdateMonitoredListS(NULL);
}
Err:
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::IsAdvancedRp()
//
// Synopsis: method to determine if a given restore point is an
// advanced restore point
//
// Arguments: restore point, pointer to flags
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD
CDriveTable::IsAdvancedRp(CRestorePoint *prp, PDWORD pdwFlags)
{
TraceFunctEnter("CDriveTable::IsAdvancedRp");
WCHAR szPath[MAX_PATH];
WCHAR szSysDrive[MAX_PATH];
SDriveTableEnumContext dtec = {NULL, 0};
DWORD dwErr = ERROR_SUCCESS;
CDataStore *pds = NULL;
CRestorePoint rp;
CDriveTable dt;
//
// read the drivetable file for this restore point
//
if (FALSE == GetSystemDrive(szSysDrive))
{
dwErr = ERROR_INVALID_DRIVE;
trace(0, "! GetSystemDrive : %ld", dwErr);
goto Err;
}
dwErr = GetCurrentRestorePoint(rp);
if ( dwErr != ERROR_SUCCESS )
goto Err;
//
// if no restore point specified, assume current
//
if (! prp)
prp = &rp;
if (prp->GetNum() == rp.GetNum())
{
MakeRestorePath(szPath, szSysDrive, s_cszDriveTable);
}
else
{
MakeRestorePath(szPath, szSysDrive, prp->GetDir());
PathAppend(szPath, s_cszDriveTable);
}
dwErr = dt.LoadDriveTable(szPath);
if (dwErr != ERROR_SUCCESS)
goto Err;
//
// check if the rp directory exists on all drives
// if it does not, then it is an advanced restore point
//
*pdwFlags = RP_NORMAL;
pds = dt.FindFirstDrive(dtec);
while (pds)
{
//
// is the rp dir supposed to exist?
//
if ((pds->GetFlags() & SR_DRIVE_ACTIVE) &&
(pds->GetFlags() & SR_DRIVE_MONITORED) &&
(pds->GetFlags() & SR_DRIVE_PARTICIPATE) &&
!(pds->GetFlags() & SR_DRIVE_FROZEN))
{
MakeRestorePath(szPath, pds->GetDrive(), rp.GetDir());
if (0xFFFFFFFF == GetFileAttributes(szPath))
{
*pdwFlags = RP_ADVANCED;
break;
}
}
pds = dt.FindNextDrive(dtec);
}
Err:
TraceFunctLeave();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::AnyMountedDrives()
//
// Synopsis: check if there are any mounted drives
//
// Arguments:
//
// Returns:
//
// History: 25-Oct-2000 Brijeshk Created
//
//--------------------------------------------------------------------------
BOOL CDriveTable::AnyMountedDrives()
{
SDriveTableEnumContext dtec = {NULL, 0};
CDataStore *pds = FindFirstDrive(dtec);
while (pds)
{
//
// get the first '\' in the drive path
// if this is not the last character in the path
// then this is a mount point
//
LPWSTR pszfirst = wcschr(pds->GetDrive(), L'\\');
if (pszfirst &&
pszfirst != pds->GetDrive() + (lstrlen(pds->GetDrive()) - 1))
{
return TRUE;
}
pds = FindNextDrive(dtec);
}
return FALSE;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::CDataStoreMgr()
//
// Synopsis:
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStoreMgr::CDataStoreMgr()
{
_fStop = FALSE;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::~CDataStoreMgr()
//
// Synopsis:
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
CDataStoreMgr::~CDataStoreMgr()
{
}
// helper functions for Fifo
BOOL
IsCurrentRp(CRestorePoint& rp, CRestorePoint& rpCur)
{
return rp.GetNum() == rpCur.GetNum();
}
BOOL
IsTargetPercentMet(int nUsagePercent, int nTargetPercent)
{
return nUsagePercent <= nTargetPercent;
}
BOOL
IsTargetRpMet(DWORD dwRPNum, DWORD dwTargetRPNum)
{
return dwRPNum > dwTargetRPNum;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::Fifo
//
// Synopsis: fifo restore points upto a given percentage
//
// Arguments: drive to fifo, target RP dir, target percentage to stop fifo
// specify only one of both (dwTargetRPNum or nTargetPercent)
// fIncludeCurrentRp = TRUE : fifo current rp if necessary (i.e. freeze)
// fIncludeCurrentRp = FALSE : don't fifo current rp
//
// Returns: Win32 error code
//
// History: 27-Apr-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::Fifo(
WCHAR *pwszDrive,
DWORD dwTargetRPNum,
int nTargetPercent,
BOOL fIncludeCurrentRp,
BOOL fFifoAtLeastOneRp
)
{
TENTER("CDataStoreMgr::Fifo");
DWORD dwErr = ERROR_SUCCESS;
CDataStore *pds = _dt.FindDriveInTable(pwszDrive);
BOOL fFifoed = FALSE;
DWORD dwLastFifoedRp;
CDataStore *pdsLead = NULL;
BOOL fFirstIteration;
SDriveTableEnumContext dtec = {NULL, 0};
CRestorePointEnum *prpe = NULL;
CRestorePoint *prp = new CRestorePoint;
// can't specify many target criteria
if (dwTargetRPNum != 0 && nTargetPercent != 0)
{
dwErr = ERROR_INVALID_PARAMETER;
goto Err;
}
// can't specify no target criteria
if (fIncludeCurrentRp == TRUE && dwTargetRPNum == 0 && nTargetPercent == 0)
{
dwErr = ERROR_INVALID_PARAMETER;
goto Err;
}
// can't specify bad target criteria
if (dwTargetRPNum > g_pEventHandler->m_CurRp.GetNum() ||
nTargetPercent < 0 ||
nTargetPercent > 100)
{
dwErr = ERROR_INVALID_PARAMETER;
goto Err;
}
if (!prp)
{
trace(0, "cannot allocate memory for restore point");
dwErr = ERROR_OUTOFMEMORY;
goto Err;
}
if (! pds)
{
TRACE(0, "! Drive %S not in drivetable", pwszDrive);
dwErr = ERROR_INVALID_DRIVE;
goto Err;
}
if (! (pds->GetFlags() & SR_DRIVE_MONITORED) ||
(pds->GetFlags() & SR_DRIVE_FROZEN) )
{
trace(0, "Drive %S already frozen/disabled", pwszDrive);
goto Err;
}
if (g_pSRConfig->m_dwTestBroadcast)
PostTestMessage(g_pSRConfig->m_uiTMFifoStart, (WPARAM) NULL, (LPARAM) NULL);
pdsLead = NULL;
fFirstIteration = TRUE;
while (pds)
{
fFifoed = FALSE;
//
// skip the drive we fifoed first
//
if (pds != pdsLead)
{
//
// enum forward, don't skip last
//
prpe = new CRestorePointEnum( pds->GetDrive(), TRUE, FALSE );
if (!prpe)
{
trace(0, "cannot allocate memory for restore point enum");
dwErr = ERROR_OUTOFMEMORY;
goto Err;
}
{
WCHAR szFifoedRpPath[MAX_PATH];
//
// blow away any obsolete "Fifoed" directories
//
MakeRestorePath(szFifoedRpPath, pwszDrive, s_cszFifoedRpDir);
CHECKERR(Delnode_Recurse(szFifoedRpPath, TRUE, &_fStop),
"Denode_Recurse Fifoed");
//
// blow away any obsolete "RP0" directories
//
MakeRestorePath(szFifoedRpPath, pwszDrive, L"RP0");
dwErr = Delnode_Recurse(szFifoedRpPath, TRUE, &_fStop);
if (ERROR_SUCCESS != dwErr)
{
trace (0, "Cannot FIFO RP0 error %d, ignoring", dwErr);
dwErr = ERROR_SUCCESS;
}
}
//
// loop through restore points on this drive
//
dwErr = prpe->FindFirstRestorePoint (*prp);
//
// enumeration can return ERROR_FILE_NOT_FOUND for restorepoints
// that are missing rp.log
// we will just continue in this case
//
while (dwErr == ERROR_SUCCESS || dwErr == ERROR_FILE_NOT_FOUND)
{
//
// check for the stop event
//
ASSERT(g_pSRConfig);
if (IsStopSignalled(g_pSRConfig->m_hSRStopEvent))
{
TRACE(0, "Stop signalled - aborting fifo");
dwErr = ERROR_OPERATION_ABORTED;
goto Err;
}
//
// check if fifo is disabled from this restore point
//
if (g_pSRConfig->GetFifoDisabledNum() != 0 &&
prp->GetNum() >= g_pSRConfig->GetFifoDisabledNum())
{
TRACE(0, "Fifo disabled from %S", prp->GetDir());
break;
}
//
// check if we've reached target restore point or percentage
//
if (dwTargetRPNum)
{
if (IsTargetRpMet(prp->GetNum(), dwTargetRPNum))
{
TRACE(0, "Target restore point reached");
break;
}
}
else if (nTargetPercent && FALSE == fFifoAtLeastOneRp)
{
int nUsagePercent = 0;
if (ERROR_SUCCESS == pds->GetUsagePercent(&nUsagePercent) &&
IsTargetPercentMet(nUsagePercent, nTargetPercent))
{
TRACE(0, "Target percentage reached");
break;
}
}
//
// check if we've reached the current rp
//
if (IsCurrentRp(*prp, g_pEventHandler->m_CurRp))
{
if (fIncludeCurrentRp)
{
//
// need to fifo this one too
// this is same as freezing the drive
// so freeze
//
dwErr = FreezeDrive(pwszDrive);
goto Err;
}
else
{
//
// don't fifo current rp
// (usually called from Disk Cleanup)
//
trace(0, "No more rps to fifo");
break;
}
}
//
// throw away this restore point on this drive
//
dwErr = pds->FifoRestorePoint (*prp);
if ( ERROR_SUCCESS != dwErr )
{
TRACE(0, "! FifoRestorePoint on %S on drive %S : %ld",
prp->GetDir(), pwszDrive, dwErr);
goto Err;
}
//
// record in the fifo log
//
WriteFifoLog (prp->GetDir(), pds->GetDrive());
dwLastFifoedRp = prp->GetNum();
fFifoed = TRUE;
fFifoAtLeastOneRp = FALSE;
dwErr = prpe->FindNextRestorePoint(*prp);
}
if (prpe)
{
delete prpe;
prpe = NULL;
}
}
//
// go to next drive
//
if (fFirstIteration)
{
if (! fFifoed) // we did not fifo anything
{
break;
}
pdsLead = pds;
pds = _dt.FindFirstDrive(dtec);
fFirstIteration = FALSE;
dwTargetRPNum = dwLastFifoedRp; // fifo till what we fifoed just now
nTargetPercent = 0;
fIncludeCurrentRp = TRUE;
fFifoAtLeastOneRp = FALSE;
}
else
{
pds = _dt.FindNextDrive(dtec);
}
}
if (dwErr == ERROR_NO_MORE_ITEMS)
dwErr = ERROR_SUCCESS;
if (g_pSRConfig->m_dwTestBroadcast)
PostTestMessage(g_pSRConfig->m_uiTMFifoStop, (WPARAM) dwLastFifoedRp, (LPARAM) NULL);
Err:
if (prpe)
delete prpe;
if (prp)
delete prp;
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::WriteFifoLog
//
// Synopsis: appends to the fifo log
//
// Arguments: dir name of restore point fifoed, drive
//
// Returns:
//
// History: 27-Apr-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::WriteFifoLog(LPWSTR pwszDir, LPWSTR pwszDrive)
{
FILE *f = NULL;
WCHAR szLog[MAX_PATH];
DWORD dwRc = ERROR_INTERNAL_ERROR;
WCHAR wszTime[MAX_PATH] = L"";
WCHAR wszDate[MAX_PATH] = L"";
CDataStore *pds = NULL;
TENTER("CDataStoreMgr::WriteFifoLog");
TRACE(0, "Fifoed %S on drive %S", pwszDir, pwszDrive);
if (pds = _dt.FindSystemDrive())
{
MakeRestorePath(szLog, pds->GetDrive(), s_cszFifoLog);
f = (FILE *) _wfopen(szLog, L"a");
if (f)
{
_wstrdate(wszDate);
_wstrtime(wszTime);
fwprintf(f, L"%s-%s : Fifoed %s on drive %s\n", wszDate, wszTime, pwszDir, pwszDrive);
fclose(f);
dwRc = ERROR_SUCCESS;
}
else
{
TRACE(0, "_wfopen failed on %s", szLog);
}
}
TLEAVE();
return dwRc;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::EnumAllVolumes
//
// Synopsis: enumerates all local volumes and updates the drive table
//
// Arguments:
//
// Returns: Win32 error code
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::EnumAllVolumes ()
{
TENTER("CDriveTable::EnumAllVolumes");
DWORD dwErr = ERROR_SUCCESS;
WCHAR wcsVolumeName[MAX_PATH];
WCHAR wcsDosName[MAX_PATH];
//
// Let's first get all the local volumes
//
HANDLE hVolume = FindFirstVolume (wcsVolumeName, MAX_PATH);
// If we can't even find one volume, return an error
if (hVolume == INVALID_HANDLE_VALUE)
{
dwErr = GetLastError();
return dwErr;
}
do
{
//
// We have to find a mount point that points to this volume
// If there is no such mount point, then the volume is not
// accessible, and we ignore it
//
dwErr = FindMountPoint (wcsVolumeName, wcsDosName);
if (dwErr == ERROR_SUCCESS)
{
dwErr = AddDriveToTable (wcsDosName, wcsVolumeName);
if (dwErr == ERROR_BAD_DEV_TYPE || //add only fixed drives
dwErr == ERROR_UNRECOGNIZED_VOLUME) //unformatted
dwErr = ERROR_SUCCESS;
if (dwErr != ERROR_SUCCESS)
{
goto Err;
}
}
}
while (FindNextVolume (hVolume, wcsVolumeName, MAX_PATH));
dwErr = ERROR_SUCCESS;
Err:
FindVolumeClose (hVolume);
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::LoadDriveTable
//
// Synopsis: loads a drive table from a restore point directory
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::LoadDriveTable (WCHAR *pwszPath)
{
TENTER ("CDriveTable::LoadDriveTable");
DWORD dwErr = ERROR_SUCCESS;
if (FALSE == _fLockInit)
{
dwErr = _lock.Init();
if (dwErr != ERROR_SUCCESS)
return dwErr;
_fLockInit = TRUE;
}
BOOL fLocked = _lock.Lock(CLock::TIMEOUT);
if (!fLocked)
{
return WAIT_TIMEOUT;
}
CDataStore *pds = NULL;
WCHAR *pwcBuffer = NULL;
WCHAR *pwszLine = NULL;
HANDLE hFile = INVALID_HANDLE_VALUE;
DWORD dwFlags = 0;
DWORD dwFileSize = 0;
DWORD cbRead = 0;
int iChangeLogs = 0;
WCHAR wcsDrive[MAX_PATH] = L"";
WCHAR wcsGuid[GUID_STRLEN] = L"";
WCHAR wcsLabel[CDataStore::LABEL_STRLEN];
DWORD dwSizeLimit = 0;
hFile = CreateFileW ( pwszPath, // file name
GENERIC_READ, // file access
FILE_SHARE_READ, // share mode
NULL, // SD
OPEN_EXISTING, // how to create
0, // file attributes
NULL); // handle to template file
if (INVALID_HANDLE_VALUE == hFile)
{
dwErr = GetLastError();
goto Err;
}
dwFileSize = GetFileSize (hFile, NULL);
if (dwFileSize > SR_DEFAULT_DSMAX * MEGABYTE)
{
dwErr = ERROR_FILE_CORRUPT;
goto Err;
}
pwcBuffer = (WCHAR *) SRMemAlloc (dwFileSize);
if (pwcBuffer == NULL)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
goto Err;
}
if (FALSE == ReadFile (hFile, (BYTE*)pwcBuffer, dwFileSize, &cbRead, NULL))
{
dwErr = GetLastError();
goto Err;
}
pwszLine = pwcBuffer;
for (UINT i = 0; i < dwFileSize / sizeof(WCHAR); i++)
{
if (pwcBuffer[i] == L'\n')
{
pwcBuffer[i] = L'\0'; // convert all newlines to terminators
wcsLabel[0] = L'\0'; // initialize in case scanf terminates early
if (EOF != swscanf(pwszLine, gs_wcsScanFormat, wcsDrive,
wcsGuid, &dwFlags, &iChangeLogs,
&dwSizeLimit, wcsLabel))
{
pds = new CDataStore ( this);
if (pds == NULL)
{
dwErr = ERROR_NOT_ENOUGH_MEMORY;
return dwErr;
}
dwErr = pds->LoadDataStore (wcsDrive, wcsGuid,
wcsLabel, dwFlags, iChangeLogs,
(INT64) dwSizeLimit * MEGABYTE);
if (dwErr != ERROR_SUCCESS)
goto Err;
dwErr = CreateNewEntry (pds);
if (dwErr != ERROR_SUCCESS)
goto Err;
}
pwszLine = &pwcBuffer[i+1]; // skip to next line
}
}
Err:
if (hFile != INVALID_HANDLE_VALUE)
CloseHandle (hFile);
if (pwcBuffer != NULL)
SRMemFree (pwcBuffer);
if (fLocked)
_lock.Unlock();
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDriveTable::SaveDriveTable
//
// Synopsis: saves a drive table into a restore point directory
//
// Arguments: [prp] -- restore point to save into
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDriveTable::SaveDriveTable (CRestorePoint *prp)
{
DWORD dwErr = ERROR_SUCCESS;
WCHAR wcsPath[MAX_PATH];
CDataStore *pds = FindSystemDrive();
if (NULL == pds)
{
dwErr = ERROR_INVALID_DRIVE;
goto Err;
}
if (prp == NULL) // no restore point, so save to the datastore directory
{
MakeRestorePath(wcsPath, pds->GetDrive(), L"");
}
else
{
MakeRestorePath(wcsPath, pds->GetDrive(), prp->GetDir());
}
lstrcatW (wcsPath, L"\\");
lstrcatW (wcsPath, s_cszDriveTable);
dwErr = SaveDriveTable (wcsPath);
Err:
return dwErr;
}
DWORD CDriveTable::SaveDriveTable (WCHAR *pwszPath)
{
TENTER ("CDriveTable::SaveDriveTable");
DWORD dwErr = ERROR_SUCCESS;
if (FALSE == _fLockInit)
{
dwErr = _lock.Init();
if (dwErr != ERROR_SUCCESS)
return dwErr;
_fLockInit = TRUE;
}
BOOL fLocked = _lock.Lock(CLock::TIMEOUT);
if (!fLocked)
{
return WAIT_TIMEOUT;
}
BOOL fDirtySave = _fDirty; // save the dirty bit
HANDLE hFile = CreateFileW ( pwszPath, // file name
GENERIC_WRITE, // file access
0, // share mode
NULL, // SD
CREATE_ALWAYS, // how to create
FILE_FLAG_WRITE_THROUGH, // file attributes
NULL); // handle to template file
if (INVALID_HANDLE_VALUE == hFile)
{
dwErr = GetLastError();
goto Err;
}
_fDirty = FALSE; // avoid calling back into SaveDriveTable again
dwErr = ForAllDrives (CDataStore::SaveDataStore, (LONG_PTR) hFile);
if (dwErr != ERROR_SUCCESS)
goto Err;
if (lstrcmp (pwszPath, L"CONOUT$") != 0)
{
if (FALSE == FlushFileBuffers (hFile)) // make sure it's on disk
dwErr = GetLastError();
}
Err:
if (hFile != INVALID_HANDLE_VALUE)
CloseHandle (hFile);
if (ERROR_SUCCESS == dwErr)
_fDirty = FALSE;
else
_fDirty = fDirtySave;
if (fLocked)
_lock.Unlock();
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::Initialize
//
// Synopsis:
//
// Arguments: fFirstRun -- true if run on first boot
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::Initialize (BOOL fFirstRun)
{
TENTER("CDataStoreMgr::Initialize");
DWORD dwErr = ERROR_SUCCESS;
dwErr = _dt.EnumAllVolumes();
if (dwErr == ERROR_SUCCESS)
{
CDataStore *pds = _dt.FindSystemDrive();
// now create system datastore on firstrun
if (pds != NULL)
{
WCHAR wcsPath[MAX_PATH];
if (fFirstRun)
{
// destroy datastores on all drives and
// create new on system drive
SDriveTableEnumContext dtec = {NULL, 0};
CDataStore *pdsDel = _dt.FindFirstDrive (dtec);
while (pdsDel != NULL)
{
if (pdsDel->GetFlags() & (SR_DRIVE_ACTIVE))
{
dwErr = pdsDel->DestroyDataStore (TRUE);
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! DestroyDataStore : %ld", dwErr);
goto Err;
}
}
pdsDel = _dt.FindNextDrive (dtec);
}
dwErr = pds->CreateDataStore (NULL);
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! CreateDataStore : %ld", dwErr);
goto Err;
}
(void) WriteMachineGuid();
}
else // verify that the system datastore exists
{
CDriveTable dt;
MakeRestorePath (wcsPath, pds->GetDrive(), L"");
DWORD dwAttr = GetFileAttributes (wcsPath);
if (0xFFFFFFFF==dwAttr || !(FILE_ATTRIBUTE_DIRECTORY & dwAttr))
{
dwErr = pds->CreateDataStore( NULL );
}
MakeRestorePath(wcsPath, pds->GetDrive(), s_cszDriveTable);
if (ERROR_SUCCESS == dt.LoadDriveTable (wcsPath))
{
dwErr = _dt.Merge(dt);
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! CDriveTable::Merge : %ld", dwErr);
goto Err;
}
}
}
// update the disk free space variable and
// set datastore size for each datastore if not already done
dwErr = UpdateDiskFree(NULL);
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! UpdateDiskFree : %ld", dwErr);
goto Err;
}
// freeze system drive if this is firstrun and disk free is < 200MB
if (fFirstRun && g_pSRConfig)
{
if (pds->GetDiskFree() < (g_pSRConfig->m_dwDSMin * MEGABYTE))
{
dwErr = FreezeDrive(pds->GetGuid());
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! FreezeDrive : %ld", dwErr);
goto Err;
}
}
}
MakeRestorePath(wcsPath, pds->GetDrive(), s_cszDriveTable);
dwErr = _dt.SaveDriveTable (wcsPath);
if (dwErr != ERROR_SUCCESS)
{
trace(0, "! SaveDriveTable : %ld", dwErr);
goto Err;
}
}
else dwErr = ERROR_INVALID_DRIVE;
}
Err:
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::TriggerFreezeOrFifo
//
// Synopsis: checks freedisk space & datastore size,
// triggering freeze or fifo as required
// Arguments:
//
// Returns:
//
// History: 27-Apr-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::TriggerFreezeOrFifo()
{
TENTER("CDataStoreMgr::TriggerFreezeOrFifo");
SDriveTableEnumContext dtec = {NULL, 0};
DWORD dwRc = ERROR_SUCCESS;
CDataStore *pds = _dt.FindFirstDrive (dtec);
// update datastore capacities
// and get free space on each drive
dwRc = UpdateDiskFree(NULL);
if (dwRc != ERROR_SUCCESS)
{
trace(0, "! UpdateDiskFree : %ld", dwRc);
goto done;
}
// check the free space and datastore usage
while (pds != NULL && dwRc == ERROR_SUCCESS)
{
// we care only if the drive is not already frozen
// and if it is monitored
if (!(pds->GetFlags() & SR_DRIVE_FROZEN) &&
(pds->GetFlags() & SR_DRIVE_MONITORED))
{
//
// if there is no rp on this drive,
// then we don't really care
//
CRestorePointEnum rpe((LPWSTR) pds->GetDrive(), FALSE, FALSE); // backward, include current
CRestorePoint rp;
int nUsagePercent = 0;
DWORD dwErr = rpe.FindFirstRestorePoint(rp) ;
if (dwErr == ERROR_SUCCESS || dwErr == ERROR_FILE_NOT_FOUND)
{
if (pds->GetDiskFree() <= THRESHOLD_FREEZE_DISKSPACE * MEGABYTE)
{
dwRc = FreezeDrive(pds->GetGuid());
}
else if (pds->GetDiskFree() <= THRESHOLD_FIFO_DISKSPACE * MEGABYTE)
{
dwRc = Fifo(pds->GetGuid(), 0, TARGET_FIFO_PERCENT, TRUE, TRUE);
}
else if (ERROR_SUCCESS == pds->GetUsagePercent(&nUsagePercent)
&& nUsagePercent >= THRESHOLD_FIFO_PERCENT)
{
dwRc = Fifo(pds->GetGuid(), 0, TARGET_FIFO_PERCENT, TRUE, FALSE);
}
}
rpe.FindClose();
}
pds = _dt.FindNextDrive (dtec);
}
done:
TLEAVE();
return dwRc;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::FindFrozenDrive
//
// Synopsis: returns ERROR_SUCCESS if any drives are frozen
// ERROR_NO_MORE_ITEMS otherwise
// Arguments:
//
// Returns:
//
// History: 27-Apr-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::FindFrozenDrive()
{
CDataStore * pds = NULL;
SDriveTableEnumContext dtec = {NULL, 0};
pds = _dt.FindFirstDrive (dtec);
while (pds != NULL)
{
if ((pds->GetFlags() & SR_DRIVE_MONITORED) &&
(pds->GetFlags() & SR_DRIVE_FROZEN))
return ERROR_SUCCESS;
pds = _dt.FindNextDrive (dtec);
}
return ERROR_NO_MORE_ITEMS;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::FifoOldRps
//
// Synopsis: fifoes out restore points older than a given time period
//
// Arguments: [llTimeInSeconds] -- FIFO interval time
//
// Returns:
//
// History: 14-Jul-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::FifoOldRps(
INT64 llTimeInSeconds)
{
TENTER("CDataStoreMgr::FifoOldRps");
CDataStore *pds = _dt.FindSystemDrive();
DWORD dwRc = ERROR_SUCCESS;
CRestorePointEnum *prpe = NULL;
CRestorePoint *prp = NULL;
if (pds)
{
DWORD dwRpFifo = 0;
LARGE_INTEGER *pllRp = NULL, *pllNow = NULL;
FILETIME ftNow, *pftRp = NULL;
// enumerate RPs forward on the system drive
// skip the current one
// even if the current rp is older than a month, we won't fifo it
prpe = new CRestorePointEnum(pds->GetDrive(), TRUE, TRUE);
prp = new CRestorePoint;
if (!prpe || !prp)
{
trace(0, "Cannot allocate memory for restore point enum");
dwRc = ERROR_OUTOFMEMORY;
goto done;
}
// get the current time
GetSystemTimeAsFileTime(&ftNow);
pllNow = (LARGE_INTEGER *) &ftNow;
dwRc = prpe->FindFirstRestorePoint(*prp);
while (dwRc == ERROR_SUCCESS || dwRc == ERROR_FILE_NOT_FOUND)
{
// first check if this is not a cancelled restore point
if (dwRc != ERROR_FILE_NOT_FOUND && ! prp->IsDefunct())
{
// get the restore point creation time
pftRp = prp->GetTime();
pllRp = (LARGE_INTEGER *) pftRp;
if (!pllRp || !pllNow)
{
trace(0, "! pulRp or pulNow = NULL");
dwRc = ERROR_INTERNAL_ERROR;
goto done;
}
// check if it is newer than a month
// if so, stop looking
// else, try the next restore point
if (pllNow->QuadPart - pllRp->QuadPart < llTimeInSeconds * 10 * 1000 * 1000)
{
trace(0, "%S newer than a month", prp->GetDir());
break;
}
else
{
dwRpFifo = prp->GetNum();
}
}
dwRc = prpe->FindNextRestorePoint(*prp);
}
// at this point, if dwRpFifo != 0,
// it contains the latest RP that's older than a month
// call fifo on this to fifo out all previous RPs including this one
if (dwRpFifo != 0)
dwRc = Fifo(pds->GetGuid(), dwRpFifo, 0, FALSE, FALSE);
else
dwRc = ERROR_SUCCESS;
}
else
{
trace(0, "! FindSystemDrive");
dwRc = ERROR_INVALID_DRIVE;
}
done:
if (prpe)
delete prpe;
if (prp)
delete prp;
TLEAVE();
return dwRc;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::UpdateDataStoreUsage
//
// Synopsis: update the size of a datastore
//
// Arguments:
//
// Returns:
//
// History: 27-Apr-2000 brijeshk Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::UpdateDataStoreUsage(WCHAR *pwszDrive, INT64 llDelta)
{
TENTER ("CDataStoreMgr::UpdateDataStoreUsage");
DWORD dwErr = ERROR_SUCCESS;
CDataStore *pds = _dt.FindDriveInTable(pwszDrive);
if (pds)
{
if ((pds->GetFlags() & SR_DRIVE_FROZEN) ||
! (pds->GetFlags() & SR_DRIVE_MONITORED))
{
TRACE(0, "Size update on frozen/unmonitored drive!");
}
else dwErr = pds->UpdateDataStoreUsage(llDelta, TRUE);
}
else
dwErr = ERROR_INVALID_DRIVE;
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::GetFlags
//
// Synopsis: get the participation bit from a drive
//
// Arguments: [pwszDrive] -- drive letter
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::GetFlags(WCHAR *pwszDrive, PDWORD pdwFlags)
{
CDataStore *pds = _dt.FindDriveInTable (pwszDrive);
DWORD dwErr = ERROR_SUCCESS;
if (NULL != pds)
{
*pdwFlags = pds->GetFlags();
}
else dwErr = ERROR_INVALID_DRIVE;
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::GetUsagePercent
//
// Synopsis: get the datastore usage for a drive
//
// Arguments: [pwszDrive] -- drive letter
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::GetUsagePercent(WCHAR *pwszDrive, int *pnPercent)
{
CDataStore *pds = _dt.FindDriveInTable (pwszDrive);
DWORD dwErr = ERROR_SUCCESS;
if (NULL != pds)
{
dwErr = pds->GetUsagePercent(pnPercent);
}
else dwErr = ERROR_INVALID_DRIVE;
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::SwitchRestorePoint
//
// Synopsis: change the drive table when switching restore points
//
// Arguments: [prp] -- old restore point
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::SwitchRestorePoint (CRestorePoint *prp)
{
DWORD dwErr = ERROR_SUCCESS;
dwErr = _dt.ForAllDrives (&CDataStore::CountChangeLogs, (LONG_PTR) prp);
if (dwErr != ERROR_SUCCESS)
goto Err;
// persist old restore point dt
if (prp)
{
dwErr = _dt.SaveDriveTable (prp);
if (dwErr != ERROR_SUCCESS)
goto Err;
}
// remove old volumes
dwErr = _dt.RemoveDrivesFromTable ();
if (dwErr != ERROR_SUCCESS)
goto Err;
// reset per-rp flags
dwErr = _dt.ForAllDrives (&CDataStore::ResetFlags, NULL);
if (dwErr != ERROR_SUCCESS)
goto Err;
// persist current restore point dt
dwErr = _dt.SaveDriveTable((CRestorePoint *) NULL);
Err:
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::FreezeDrive
//
// Synopsis: freeze a drive
//
// Arguments: [pwszDrive] -- drive
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::FreezeDrive(WCHAR *pwszDrive)
{
DWORD dwErr = ERROR_SUCCESS;
WCHAR szThawSize[GUID_STRLEN], szSystemDrive[MAX_SYS_DRIVE] = L" ";
TENTER("CDataStoreMgr::FreezeDrive");
if (pwszDrive != NULL)
{
CDataStore *pds = _dt.FindDriveInTable (pwszDrive);
if (! pds)
{
dwErr = ERROR_INVALID_DRIVE;
TRACE (0, "FindDriveInTable failed in CDataStoreMgr::FreezeDrive %ld\n", dwErr);
goto Err;
}
//
// freeze all drives
//
}
dwErr = _dt.ForAllDrives (CDataStore::FreezeDrive, NULL);
if (dwErr == ERROR_SUCCESS)
{
//
// rebuild _filelst.cfg and pass to filter
//
ASSERT(g_pEventHandler);
dwErr = g_pEventHandler->SRUpdateMonitoredListS(NULL);
if (g_pSRService != NULL)
{
if (g_pSRConfig && g_pSRConfig->m_dwFreezeThawLogCount < MAX_FREEZETHAW_LOG_MESSAGES)
{
TRACE (0, "Freezing the SR service due to low disk space.");
wsprintf(szThawSize, L"%d",THRESHOLD_THAW_DISKSPACE);
if (pwszDrive == NULL)
{
if(GetSystemDrive(szSystemDrive) == FALSE)
TRACE (0, "GetSystemDrive failed in CDataStoreMgr::FreezeDrive.");
pwszDrive = szSystemDrive;
}
g_pSRService->LogEvent(EVENTLOG_INFORMATION_TYPE, EVMSG_FROZEN, NULL, 0, szThawSize, pwszDrive);
g_pSRConfig->m_dwFreezeThawLogCount++;
}
}
}
//
// request for idle time
// so that we can thaw later
//
ASSERT(g_pSRConfig);
SetEvent(g_pSRConfig->m_hIdleRequestEvent);
if (g_pEventHandler)
g_pEventHandler->RefreshCurrentRp(FALSE);
if (g_pSRConfig->m_dwTestBroadcast)
PostTestMessage(g_pSRConfig->m_uiTMFreeze, NULL, NULL);
Err:
TLEAVE();
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::IsDriveFrozen
//
// Synopsis: check if given drive is frozen
// if NULL, check if any drive is frozen
//
// Arguments:
//
// Returns:
//
// History: 21-Jul-2000 Brijeshk Created
//
//--------------------------------------------------------------------------
BOOL CDataStoreMgr::IsDriveFrozen(LPWSTR pszDrive)
{
CDataStore *pds = NULL;
SDriveTableEnumContext dtec = {NULL, 0};
if (!pszDrive)
{
pds = _dt.FindFirstDrive(dtec);
while (pds)
{
if (pds->GetFlags() & SR_DRIVE_FROZEN)
return TRUE;
pds = _dt.FindNextDrive(dtec);
}
}
else
{
CDataStore *pds = _dt.FindDriveInTable(pszDrive);
if (pds)
{
if (pds->GetFlags() & SR_DRIVE_FROZEN)
return TRUE;
}
}
return FALSE;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::ThawDrives
//
// Synopsis: thaw one or more drives
//
// Arguments: [fCheckOnly] -- if TRUE do not actually thaw
//
// Returns: if any drive is thawed, returns ERROR_SUCCESS
// else returns ERROR_NO_MORE_ITEMS
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD
CDataStoreMgr::ThawDrives(BOOL fCheckOnly)
{
TENTER("CDataStoreMgr::ThawDrives");
CDataStore *pds = NULL, *pdsSys = _dt.FindSystemDrive();
DWORD dwRc = ERROR_NO_MORE_ITEMS;
SDriveTableEnumContext dtec;
DWORD dwTemp;
if (! pdsSys)
{
dwRc = ERROR_INVALID_DRIVE;
TRACE (0, "Cannot find system drive %ld\n", dwRc);
goto done;
}
// if system drive is frozen, then check if 200mb is free
// if yes, then thaw all drives
// if no, thaw none
ASSERT(pdsSys->GetFlags() & SR_DRIVE_MONITORED);
if ((pdsSys->GetFlags() & SR_DRIVE_FROZEN))
{
dwRc = pdsSys->UpdateDiskFree(NULL);
if (dwRc != ERROR_SUCCESS)
{
trace(0, "! UpdateDiskFree : %ld", dwRc);
goto done;
}
if (pdsSys->GetDiskFree() >= THRESHOLD_THAW_DISKSPACE * MEGABYTE)
{
if (fCheckOnly)
{
dwRc = ERROR_SUCCESS;
goto done;
}
pds = _dt.FindFirstDrive (dtec);
while (pds != NULL)
{
dwTemp = pds->ThawDrive(NULL);
if (dwTemp != ERROR_SUCCESS) // remember the error and
{
dwRc = dwTemp; // keep on going
TRACE (0, "ThawDrive failed with %ld\n", dwRc);
}
pds = _dt.FindNextDrive (dtec);
}
}
else // cannot thaw now
{
dwRc = ERROR_NO_MORE_ITEMS;
TRACE (0, "No drives to thaw %ld\n", dwRc);
}
}
else // make sure all the other drives are thawed too for consistency
{
pds = _dt.FindFirstDrive (dtec);
while (pds != NULL)
{
if (pds->GetFlags() & SR_DRIVE_FROZEN)
pds->ThawDrive(NULL);
pds = _dt.FindNextDrive (dtec);
}
dwRc = ERROR_SUCCESS;
}
if (_dt.GetDirty())
{
dwRc = _dt.SaveDriveTable ((CRestorePoint *) NULL);
if (dwRc != ERROR_SUCCESS)
TRACE (0, "SaveDriveTable failed with %ld\n", dwRc);
}
if (g_pSRService != NULL && ERROR_SUCCESS == dwRc && FALSE == fCheckOnly)
{
if (g_pSRConfig && g_pSRConfig->m_dwFreezeThawLogCount <= MAX_FREEZETHAW_LOG_MESSAGES)
{
g_pSRService->LogEvent(EVENTLOG_INFORMATION_TYPE, EVMSG_THAWED);
g_pSRConfig->m_dwFreezeThawLogCount++;
}
}
done:
TLEAVE();
return dwRc;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::MonitorDrive
//
// Synopsis: enable/disable a drive
//
// Arguments: [pwszDrive] -- drive, [fSet] -- enable/disable
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::MonitorDrive(WCHAR *pwszDrive, BOOL fSet)
{
CDataStore *pds = pwszDrive ? _dt.FindDriveInTable(pwszDrive) : NULL;
DWORD dwErr = ERROR_SUCCESS;
BOOL fTellFilter = TRUE;
if (! pwszDrive || ! pds || pds == _dt.FindSystemDrive()) // system drive
{
// something wrong
// cannot enable/disable whole of SR this way
dwErr = ERROR_INVALID_DRIVE;
}
else
{
// enable/disable only this drive
dwErr = pds->MonitorDrive(fSet);
if (ERROR_SUCCESS == dwErr && (pds->GetFlags() & SR_DRIVE_FROZEN))
fTellFilter = FALSE;
}
if (dwErr == ERROR_SUCCESS)
{
// update drivetable on disk
if (_dt.GetDirty())
{
dwErr = _dt.SaveDriveTable ((CRestorePoint *) NULL);
// rebuild _filelst.cfg and pass to filter
if (fTellFilter)
{
ASSERT(g_pEventHandler);
dwErr = g_pEventHandler->SRUpdateMonitoredListS(NULL);
}
}
}
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::WriteMachineGuid
//
// Synopsis: write machine guid for disk cleanup utility
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::WriteMachineGuid ()
{
DWORD dwErr = ERROR_SUCCESS;
WCHAR wcsPath [MAX_PATH];
if (0 == GetSystemDirectoryW (wcsPath, MAX_PATH))
{
dwErr = GetLastError();
}
else
{
lstrcatW (wcsPath, L"\\Restore\\MachineGuid.txt");
HANDLE hFile = CreateFileW ( wcsPath, // file name
GENERIC_WRITE, // file access
0, // share mode
NULL, // SD
CREATE_ALWAYS, // how to create
0, // file attributes
NULL); // handle to template file
if (INVALID_HANDLE_VALUE == hFile)
{
dwErr = GetLastError();
}
else
{
WCHAR *pwszGuid = GetMachineGuid();
ULONG cbWritten;
if (FALSE == WriteFile (hFile, (BYTE *) pwszGuid,
(lstrlenW(pwszGuid)+1) * sizeof(WCHAR),
&cbWritten, NULL))
{
dwErr = GetLastError();
}
CloseHandle (hFile);
}
}
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::DeleteMachineGuidFile
//
// Synopsis: write machine guid for disk cleanup utility
//
// Arguments:
//
// Returns:
//
// History: 13-Apr-2000 HenryLee Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::DeleteMachineGuidFile ()
{
DWORD dwErr = ERROR_SUCCESS;
WCHAR wcsPath [MAX_PATH];
if (0 == GetSystemDirectoryW (wcsPath, MAX_PATH))
{
dwErr = GetLastError();
}
else
{
lstrcatW (wcsPath, L"\\Restore\\MachineGuid.txt");
if (FALSE == DeleteFileW (wcsPath))
dwErr = GetLastError();
}
return dwErr;
}
//+-------------------------------------------------------------------------
//
// Function: CDataStoreMgr::Compress
//
// Synopsis: compress files in each datastore
//
// Arguments: lDuration - duration to compress
//
// Returns:
//
// History: 26-Feb-01 Brijeshk Created
//
//--------------------------------------------------------------------------
DWORD CDataStoreMgr::Compress (
LPWSTR pszDrive,
LONG lDuration)
{
TENTER("CDataStoreMgr::Compress");
DWORD dwErr = ERROR_SUCCESS;
CDataStore *pds = NULL, *pdsSys = NULL;
INT64 llAllocated = 0, llUsed = 0;
SDriveTableEnumContext dtec = {NULL, 0};
BOOL fFirstTime;
llAllocated = lDuration * 1000 * 1000 * 10; // 100's of nanoseconds
//
// if drive specified, compress only that
//
if (pszDrive)
{
pds = _dt.FindDriveInTable(pszDrive);
if (pds)
{
dwErr = pds->Compress(llAllocated, &llUsed);
}
else
{
dwErr = ERROR_INVALID_DRIVE;
}
goto Err;
}
//
// else, compress all drives if time permits
// starting with system drive
//
pdsSys = _dt.FindSystemDrive();
pds = pdsSys;
fFirstTime = TRUE;
while (pds)
{
if (fFirstTime || pds != pdsSys)
{
trace(0, "Allocated time for %S is %I64d", pds->GetDrive(), llAllocated);
llUsed = 0;
dwErr = pds->Compress(llAllocated, &llUsed);
if (dwErr != ERROR_SUCCESS && dwErr != ERROR_OPERATION_ABORTED)
{
trace(0, "! Compress : %ld", dwErr);
goto Err;
}
llAllocated -= llUsed;
if (llAllocated <= 0)
{
//
// used up all time
//
dwErr = ERROR_OPERATION_ABORTED;
break;
}
}
//
// go to next drive
//
if (fFirstTime)
{
pds = _dt.FindFirstDrive(dtec);
}
else
{
pds = _dt.FindNextDrive(dtec);
}
fFirstTime = FALSE;
}
Err:
TLEAVE();
return dwErr;
}