861 lines
24 KiB
C++
861 lines
24 KiB
C++
/******************************************************************************
|
|
|
|
Copyright (c) 2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
api.cpp
|
|
|
|
Abstract:
|
|
This file contains the top level APIs, InitiateRestore and ResumeRestore.
|
|
|
|
Revision History:
|
|
Seong Kook Khang (SKKhang) 06/20/00
|
|
created
|
|
|
|
******************************************************************************/
|
|
|
|
#include "stdwin.h"
|
|
#include "rstrcore.h"
|
|
#include "resource.h"
|
|
extern CSRClientLoader g_CSRClientLoader;
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// EnsureTrace
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
//static BOOL s_fTraceEnabled = FALSE;
|
|
static DWORD s_dwTraceCount = 0;
|
|
|
|
void EnsureTrace()
|
|
{
|
|
if ( s_dwTraceCount++ == 0 )
|
|
{
|
|
::InitAsyncTrace();
|
|
}
|
|
}
|
|
|
|
void ReleaseTrace()
|
|
{
|
|
if ( --s_dwTraceCount == 0 )
|
|
{
|
|
::TermAsyncTrace();
|
|
}
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CRestoreContext
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
class CRestoreContext : public IRestoreContext
|
|
{
|
|
public:
|
|
CRestoreContext();
|
|
|
|
protected:
|
|
~CRestoreContext();
|
|
|
|
// operations - IRestoreContext methods
|
|
public:
|
|
BOOL IsAnyDriveOfflineOrDisabled( LPWSTR szOffline );
|
|
void SetSilent();
|
|
void SetUndo();
|
|
BOOL Release();
|
|
|
|
// attributes
|
|
public:
|
|
int m_nRP;
|
|
CRDIArray m_aryDrv;
|
|
BOOL m_fSilent;
|
|
BOOL m_fUndo;
|
|
};
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CRestoreContext - construction / destruction
|
|
|
|
CRestoreContext::CRestoreContext()
|
|
{
|
|
m_nRP = -1;
|
|
m_fSilent = FALSE;
|
|
m_fUndo = FALSE;
|
|
}
|
|
|
|
CRestoreContext::~CRestoreContext()
|
|
{
|
|
m_aryDrv.DeleteAll();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
// CRestoreContext - IRestoreContext methods
|
|
|
|
BOOL
|
|
CRestoreContext::IsAnyDriveOfflineOrDisabled( LPWSTR szOffline )
|
|
{
|
|
TraceFunctEnter("CRestoreContext::IsAnyDriveOffline");
|
|
BOOL fRet = FALSE;
|
|
|
|
szOffline[0] = L'\0';
|
|
|
|
for ( int i = m_aryDrv.GetUpperBound(); i >= 0; i-- )
|
|
{
|
|
if ( m_aryDrv[i]->IsOffline() || m_aryDrv[i]->IsExcluded())
|
|
{
|
|
::lstrcat( szOffline, L" " );
|
|
::lstrcat( szOffline, m_aryDrv[i]->GetMount() );
|
|
fRet = TRUE;
|
|
}
|
|
}
|
|
|
|
TraceFunctLeave();
|
|
return( fRet );
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
void
|
|
CRestoreContext::SetSilent()
|
|
{
|
|
TraceFunctEnter("CRestoreContext::SetSilent");
|
|
m_fSilent = TRUE;
|
|
TraceFunctLeave();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
void
|
|
CRestoreContext::SetUndo()
|
|
{
|
|
TraceFunctEnter("CRestoreContext::SetUndo");
|
|
m_fUndo = TRUE;
|
|
TraceFunctLeave();
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL
|
|
CRestoreContext::Release()
|
|
{
|
|
TraceFunctEnter("CRestoreContext::Release");
|
|
delete this;
|
|
TraceFunctLeave();
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Helper Functions
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
BOOL
|
|
IsAdminUser()
|
|
{
|
|
TraceFunctEnter("IsAdminUser");
|
|
BOOL fRet = FALSE;
|
|
LPCWSTR cszErr;
|
|
PSID pSidAdmin = NULL;
|
|
SID_IDENTIFIER_AUTHORITY cSIA = SECURITY_NT_AUTHORITY;
|
|
BOOL fRes;
|
|
|
|
if ( !::AllocateAndInitializeSid( &cSIA, 2,
|
|
SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS,
|
|
0, 0, 0, 0, 0, 0, &pSidAdmin ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::AllocateAndInitializeSid failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
if ( !::CheckTokenMembership( NULL, pSidAdmin, &fRes ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::CheckMembership failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
DebugTrace(0, "IsAdminUser = %d", fRes);
|
|
|
|
fRet = fRes;
|
|
Exit:
|
|
if ( pSidAdmin != NULL )
|
|
::FreeSid( pSidAdmin );
|
|
TraceFunctLeave();
|
|
return( fRet );
|
|
}
|
|
|
|
//
|
|
// NOTE: 7/28/00 - skkhang
|
|
// Behavior of AdjustTokenPrivilege is a little bit confusing.
|
|
// It returns TRUE if given privilege does not exist at all, so you need to
|
|
// call GetLastError to see if it's ERROR_SUCCESS or ERROR_NOT_ALL_ASSIGNED
|
|
// (meaning the privilege does not exist.)
|
|
// Also, if the privilege was already enabled, tpOld will be empty. You
|
|
// don't need to restore the privilege in that case.
|
|
//
|
|
BOOL
|
|
CheckPrivilege( LPCWSTR szPriv, BOOL fCheckOnly )
|
|
{
|
|
TraceFunctEnter("CheckPrivilege");
|
|
BOOL fRet = FALSE;
|
|
LPCWSTR cszErr;
|
|
HANDLE hToken = NULL;
|
|
LUID luid;
|
|
TOKEN_PRIVILEGES tpNew;
|
|
TOKEN_PRIVILEGES tpOld;
|
|
DWORD dwRes;
|
|
|
|
// Prepare Process Token
|
|
if ( !::OpenProcessToken( ::GetCurrentProcess(),
|
|
TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY,
|
|
&hToken ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::OpenProcessToken failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
// Get Luid
|
|
if ( !::LookupPrivilegeValue( NULL, szPriv, &luid ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::LookupPrivilegeValue failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
// Try to enable the privilege
|
|
tpNew.PrivilegeCount = 1;
|
|
tpNew.Privileges[0].Luid = luid;
|
|
tpNew.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
|
|
if ( !::AdjustTokenPrivileges( hToken, FALSE, &tpNew, sizeof(tpNew), &tpOld, &dwRes ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::AdjustTokenPrivileges(ENABLE) failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
if ( ::GetLastError() == ERROR_NOT_ALL_ASSIGNED )
|
|
{
|
|
// This means process does not even have the privilege so
|
|
// AdjustTokenPrivilege simply ignored the request.
|
|
ErrorTrace(0, "Privilege '%ls' does not exist, probably user is not an admin.", szPriv);
|
|
goto Exit;
|
|
}
|
|
|
|
if ( fCheckOnly )
|
|
{
|
|
// Restore the privilege if it was not enabled
|
|
if ( tpOld.PrivilegeCount > 0 )
|
|
{
|
|
if ( !::AdjustTokenPrivileges( hToken, FALSE, &tpOld, sizeof(tpOld), NULL, NULL ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::AdjustTokenPrivileges(RESTORE) failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
}
|
|
}
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
if ( hToken != NULL )
|
|
::CloseHandle( hToken );
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// IsSRFrozen
|
|
//
|
|
// This routine checks if SR is frozen. If any error happens during Drive
|
|
// Table creation or System Drive does not exist (broken drive table???),
|
|
// return value is FALSE.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
BOOL APIENTRY
|
|
IsSRFrozen()
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("IsSRFrozen");
|
|
BOOL fRet = FALSE;
|
|
CRDIArray aryDrv;
|
|
int i;
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
if ( !::CreateDriveList( 0, aryDrv, TRUE ) )
|
|
goto Exit;
|
|
|
|
for ( i = aryDrv.GetUpperBound(); i >= 0; i-- )
|
|
{
|
|
if ( aryDrv[i]->IsSystem() )
|
|
{
|
|
fRet = aryDrv[i]->IsFrozen();
|
|
goto Exit;
|
|
}
|
|
}
|
|
|
|
Exit:
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// CheckPrivilegesForRestore
|
|
//
|
|
// This routine checks if necessary privileges can be set, to verify if
|
|
// logon user has necessary credential (Administrators or Backup Operators.)
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
BOOL APIENTRY
|
|
CheckPrivilegesForRestore()
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("CheckPrivilegesForRestore");
|
|
BOOL fRet = FALSE;
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
// NOTE: 8/17/00 - skkhang
|
|
//
|
|
// Backup operator does not have below two privileges... SE_SECURITY_NAME
|
|
// is enabled by default for SYSTEM so probably can simply removed, but
|
|
// SE_TAKE_OWNERSHIP is off for SYSTEM and needs to be turned on. To solve
|
|
// the problem, this routine should accept parameter to distinguish
|
|
// "Check"(from UI) and "Set"(from ResumeRestore.)
|
|
//
|
|
if ( !::CheckPrivilege( SE_SECURITY_NAME, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Cannot enable SE_SECURITY_NAME privilege...");
|
|
goto Exit;
|
|
}
|
|
if ( !::CheckPrivilege( SE_TAKE_OWNERSHIP_NAME, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Cannot enable SE_SHUTDOWN_NAME privilege...");
|
|
goto Exit;
|
|
}
|
|
|
|
if ( !::CheckPrivilege( SE_BACKUP_NAME, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Cannot enable SE_BACKUP_NAME privilege...");
|
|
goto Exit;
|
|
}
|
|
if ( !::CheckPrivilege( SE_RESTORE_NAME, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Cannot enable SE_RESTORE_NAME privilege...");
|
|
goto Exit;
|
|
}
|
|
if ( !::CheckPrivilege( SE_SHUTDOWN_NAME, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Cannot enable SE_SHUTDOWN_NAME privilege...");
|
|
goto Exit;
|
|
}
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// InvokeDiskCleanup
|
|
//
|
|
// This routine invokes Disk Cleanup Utility. A specific drive can be
|
|
// provided.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
static LPCWSTR s_cszDCUPath = L"%windir%\\system32\\cleanmgr.exe";
|
|
static LPCWSTR s_cszDCUName = L"cleanmgr.exe";
|
|
static LPCWSTR s_cszDCUOptDrv = L" /d ";
|
|
|
|
BOOL APIENTRY
|
|
InvokeDiskCleanup( LPCWSTR cszDrive )
|
|
{
|
|
TraceFunctEnter("InvokeDiskCleanup");
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
BOOL fRet = FALSE;
|
|
LPCWSTR cszErr;
|
|
WCHAR szCmdLine[MAX_PATH];
|
|
STARTUPINFO sSI;
|
|
PROCESS_INFORMATION sPI;
|
|
|
|
if ( ::ExpandEnvironmentStrings( s_cszDCUPath, szCmdLine, MAX_PATH ) == 0 )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::GetFullPathName failed - %ls", cszErr);
|
|
::lstrcpy( szCmdLine, s_cszDCUName );
|
|
}
|
|
|
|
if ( cszDrive != NULL && cszDrive[0] != L'\0' )
|
|
{
|
|
::lstrcat( szCmdLine, s_cszDCUOptDrv );
|
|
::lstrcat( szCmdLine, cszDrive );
|
|
}
|
|
|
|
DebugTrace(0, "szCmdLine='%s'", szCmdLine);
|
|
::ZeroMemory( &sSI, sizeof(sSI ) );
|
|
sSI.cb = sizeof(sSI);
|
|
if ( !::CreateProcess( NULL, szCmdLine, NULL, NULL, FALSE, 0, NULL, NULL, &sSI, &sPI ) )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::CreateProcess failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
::CloseHandle( sPI.hThread );
|
|
::CloseHandle( sPI.hProcess );
|
|
|
|
// Should I wait for DCU to finish???
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
TraceFunctLeave();
|
|
return( TRUE );
|
|
}
|
|
|
|
|
|
#ifdef DBG
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// TestRestore
|
|
//
|
|
// This routine performs core restoration functionality, without reboot or
|
|
// snapshot restoration.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
extern "C" __declspec(dllexport)
|
|
BOOL APIENTRY
|
|
TestRestore( int nRP )
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("TestRestore");
|
|
BOOL fRet = FALSE;
|
|
CRDIArray aryDrv;
|
|
RESTOREPOINTINFO sRPI;
|
|
STATEMGRSTATUS sStatus;
|
|
SRstrLogHdrV3 sLogHdr;
|
|
CRestoreOperationManager *pROMgr = NULL;
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
if ( !::CheckPrivilegesForRestore() )
|
|
goto Exit;
|
|
|
|
// Create Drive Table
|
|
if ( !::CreateDriveList( nRP, aryDrv, FALSE ) )
|
|
goto Exit;
|
|
|
|
// Create Restore Point
|
|
sRPI.dwEventType = BEGIN_SYSTEM_CHANGE;
|
|
sRPI.dwRestorePtType = RESTORE;
|
|
sRPI.llSequenceNumber = 0;
|
|
::LoadString( g_hInst, IDS_RESTORE_POINT_TEXT, sRPI.szDescription, MAX_DESC );
|
|
if ( !::SRSetRestorePoint( &sRPI, &sStatus ) )
|
|
{
|
|
ErrorTrace(0, "::SRSetRestorePoint failed, nStatus=%d", sStatus.nStatus);
|
|
goto Exit;
|
|
}
|
|
|
|
// Create the log file
|
|
sLogHdr.dwRPNum = nRP;
|
|
sLogHdr.dwRPNew = sStatus.llSequenceNumber;
|
|
sLogHdr.dwDrives = aryDrv.GetSize();
|
|
if ( !::CreateRestoreLogFile( &sLogHdr, aryDrv ) )
|
|
goto Exit;
|
|
|
|
|
|
// also call TS folks to get them to preserve RA keys on restore
|
|
_VERIFY(TRUE==RemoteAssistancePrepareSystemRestore(SERVERNAME_CURRENT));
|
|
|
|
// Create CRestoreOperationManager object
|
|
if ( !::CreateRestoreOperationManager( &pROMgr ) )
|
|
goto Exit;
|
|
|
|
// Perform the Restore Operation.
|
|
if ( !pROMgr->Run( FALSE ) )
|
|
goto Exit;
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
SAFE_RELEASE(pROMgr);
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
#endif
|
|
|
|
#ifdef DBG
|
|
#define TIMEOUT_PROGRESSTHREAD 5000
|
|
|
|
#define TESTPROG_COUNT_CHGLOG 300
|
|
#define TESTPROG_TIME_PREPARE 1
|
|
#define TESTPROG_COUNT_RESTORE 100
|
|
#define TESTPROG_TIME_RESTORE 1
|
|
#define TESTPROG_TIME_SNAPSHOT 2000
|
|
|
|
DWORD WINAPI
|
|
TestProgressWindowThreadProc( LPVOID lpParam )
|
|
{
|
|
CRestoreProgressWindow *pProgress = (CRestoreProgressWindow*)lpParam;
|
|
int i, j;
|
|
|
|
// Stage 1. Prepare (change log enumeration)
|
|
pProgress->SetStage( RPS_PREPARE, 0 );
|
|
for ( i = 0; i < TESTPROG_COUNT_CHGLOG; i++ )
|
|
{
|
|
::Sleep( TESTPROG_TIME_PREPARE );
|
|
for ( j = 0; j < 10; j++ )
|
|
pProgress->Increment();
|
|
}
|
|
|
|
// Stage 2. Restore
|
|
pProgress->SetStage( RPS_RESTORE, TESTPROG_COUNT_RESTORE );
|
|
for ( i = 0; i < TESTPROG_COUNT_RESTORE; i++ )
|
|
{
|
|
::Sleep( TESTPROG_TIME_RESTORE );
|
|
pProgress->Increment();
|
|
}
|
|
|
|
// Stage 3. Snapshot
|
|
pProgress->SetStage( RPS_SNAPSHOT, 0 );
|
|
::Sleep( TESTPROG_TIME_SNAPSHOT );
|
|
|
|
pProgress->Close();
|
|
|
|
return( 0 );
|
|
}
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// TestProgressWindow
|
|
//
|
|
// This routine invokes Progress Window and simulates progress change
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
extern "C" __declspec(dllexport)
|
|
BOOL APIENTRY
|
|
TestProgressWindow()
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("TestProgressWindow");
|
|
BOOL fRet = FALSE;
|
|
CRestoreProgressWindow *pProgress = NULL;
|
|
HANDLE hThread = NULL;
|
|
DWORD dwRet;
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
// Create progress window object
|
|
if ( !::CreateRestoreProgressWindow( &pProgress ) )
|
|
goto Exit;
|
|
|
|
// Create progress window
|
|
if ( !pProgress->Create() )
|
|
goto Exit;
|
|
|
|
// Create secondary thread for main restore operation
|
|
hThread = ::CreateThread( NULL, 0, TestProgressWindowThreadProc, pProgress, 0, NULL );
|
|
if ( hThread == NULL )
|
|
{
|
|
LPCWSTR cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::CreateThread failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
|
|
// Message loop, wait until restore thread closes progress window
|
|
if ( !pProgress->Run() )
|
|
goto Exit;
|
|
|
|
// Double check if thread has been terminated
|
|
dwRet = ::WaitForSingleObject( hThread, TIMEOUT_PROGRESSTHREAD );
|
|
if ( dwRet == WAIT_FAILED )
|
|
{
|
|
LPCWSTR cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::WaitForSingleObject failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
else if ( dwRet == WAIT_TIMEOUT )
|
|
{
|
|
ErrorTrace(0, "Timeout while waiting for the restore thread finishes...");
|
|
goto Exit;
|
|
}
|
|
|
|
pProgress->Close();
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
if ( hThread != NULL )
|
|
::CloseHandle( hThread );
|
|
SAFE_RELEASE(pProgress);
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
#endif
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// PrepareRestore
|
|
//
|
|
// This routine creates a IRestoreContext for use by InitiateRestore.
|
|
// IRestoreContext contains chosen restore point ID, drive list, etc.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
BOOL APIENTRY
|
|
PrepareRestore( int nRP, IRestoreContext **ppCtx )
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("PrepareRestore");
|
|
BOOL fRet = FALSE;
|
|
CRestoreContext *pRC = NULL;
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
if ( ppCtx == NULL )
|
|
{
|
|
ErrorTrace(0, "Invalid parameter, ppCtx is NULL.");
|
|
goto Exit;
|
|
}
|
|
*ppCtx = NULL;
|
|
|
|
if ( !::IsAdminUser() )
|
|
{
|
|
ErrorTrace(0, "Not an admin user");
|
|
goto Exit;
|
|
}
|
|
|
|
pRC = new CRestoreContext;
|
|
if ( pRC == NULL )
|
|
{
|
|
ErrorTrace(0, "Insufficient memory...");
|
|
goto Exit;
|
|
}
|
|
|
|
pRC->m_nRP = nRP;
|
|
if ( !::CreateDriveList( nRP, pRC->m_aryDrv, FALSE ) )
|
|
{
|
|
ErrorTrace(0, "Creating drive list failed");
|
|
goto Exit;
|
|
}
|
|
|
|
*ppCtx = pRC;
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
if ( !fRet )
|
|
SAFE_RELEASE(pRC);
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// InitiateRestore
|
|
//
|
|
// This routine creates a temporary persistent storage with informations
|
|
// like restore point ID. The storage will be used by ResumeRestore later.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
|
|
static LPCWSTR s_cszRunOnceValueName = L"*Restore";
|
|
static LPCWSTR s_cszRestoreUIPath = L"%SystemRoot%\\system32\\restore\\rstrui.exe";
|
|
static LPCWSTR s_cszRunOnceOptInterrupted = L" -i";
|
|
|
|
BOOL APIENTRY
|
|
InitiateRestore( IRestoreContext *pCtx, DWORD *pdwNewRP )
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("InitiateRestore");
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
BOOL fRet = FALSE;
|
|
HCURSOR hCursor = NULL;
|
|
RESTOREPOINTINFO sRPI;
|
|
STATEMGRSTATUS sStatus;
|
|
SRstrLogHdrV3 sLogHdr;
|
|
CRestoreContext *pRC;
|
|
DWORD dwVal;
|
|
WCHAR szUIPath[MAX_PATH];
|
|
BOOL fCreatedRp = FALSE;
|
|
|
|
if ( !::IsAdminUser() )
|
|
goto Exit;
|
|
|
|
// Set RunOnce key for interrupted case...
|
|
// Doing this here before anything else so result screen would appear.
|
|
::ExpandEnvironmentStrings( s_cszRestoreUIPath, szUIPath, MAX_PATH );
|
|
::lstrcat( szUIPath, s_cszRunOnceOptInterrupted );
|
|
if ( !::SRSetRegStr( HKEY_LOCAL_MACHINE, REGSTR_PATH_RUNONCE, s_cszRunOnceValueName, szUIPath ) )
|
|
goto Exit;
|
|
|
|
// similarly, set RestoreStatus key in SystemRestore
|
|
// so that test tools can know status of silent restores
|
|
// set this to indicate interrrupted status
|
|
// if restore succeeds or reverts, this value will be updated
|
|
if ( !::SRSetRegDword( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszRestoreStatus, 2 ) )
|
|
goto Exit;
|
|
|
|
|
|
// Create Restore Point
|
|
hCursor = ::SetCursor( ::LoadCursor( NULL, IDC_WAIT ) );
|
|
|
|
// make this a nested restore point so that
|
|
// no other app can create a restore point between here and a reboot
|
|
sRPI.dwEventType = BEGIN_NESTED_SYSTEM_CHANGE;
|
|
|
|
if (0 != GetSystemMetrics(SM_CLEANBOOT)) // safe mode
|
|
{
|
|
sRPI.dwRestorePtType = CANCELLED_OPERATION;
|
|
}
|
|
else // normal mode
|
|
{
|
|
sRPI.dwRestorePtType = RESTORE;
|
|
}
|
|
|
|
sRPI.llSequenceNumber = 0;
|
|
::LoadString( g_hInst, IDS_RESTORE_POINT_TEXT, sRPI.szDescription, MAX_DESC );
|
|
if ( !::SRSetRestorePoint( &sRPI, &sStatus ) )
|
|
{
|
|
ErrorTrace(0, "::SRSetRestorePoint failed, nStatus=%d", sStatus.nStatus);
|
|
goto Exit;
|
|
}
|
|
if ( pdwNewRP != NULL )
|
|
*pdwNewRP = sStatus.llSequenceNumber;
|
|
|
|
fCreatedRp = TRUE;
|
|
|
|
// Create the log file
|
|
pRC = (CRestoreContext*)pCtx;
|
|
sLogHdr.dwFlags = pRC->m_fSilent ? RLHF_SILENT : 0;
|
|
sLogHdr.dwFlags |= pRC->m_fUndo ? RLHF_UNDO : 0;
|
|
sLogHdr.dwRPNum = pRC->m_nRP;
|
|
sLogHdr.dwRPNew = sStatus.llSequenceNumber;
|
|
sLogHdr.dwDrives = pRC->m_aryDrv.GetSize();
|
|
if ( !::CreateRestoreLogFile( &sLogHdr, pRC->m_aryDrv ) )
|
|
goto Exit;
|
|
|
|
// also call TS folks to get them to preserve RA keys on restore
|
|
_VERIFY(TRUE==RemoteAssistancePrepareSystemRestore(SERVERNAME_CURRENT));
|
|
|
|
// Set RestoreInProgress registry key so winlogon would invoke us
|
|
if ( !::SRSetRegDword( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszRestoreInProgress, 1 ) )
|
|
goto Exit;
|
|
|
|
fRet = TRUE;
|
|
|
|
Exit:
|
|
if (fRet == FALSE)
|
|
{
|
|
// if something failed and we had set a nested restore point,
|
|
// end the nesting now
|
|
|
|
if (fCreatedRp == TRUE)
|
|
{
|
|
sRPI.dwRestorePtType = RESTORE;
|
|
sRPI.dwEventType = END_NESTED_SYSTEM_CHANGE;
|
|
SRSetRestorePoint( &sRPI, &sStatus );
|
|
}
|
|
|
|
// delete the runonce key
|
|
SHDeleteValue(HKEY_LOCAL_MACHINE, REGSTR_PATH_RUNONCE, s_cszRunOnceValueName);
|
|
}
|
|
|
|
if ( hCursor != NULL )
|
|
::SetCursor( hCursor );
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// ResumeRestore
|
|
//
|
|
// This routine is the main routine to run the restore operation.
|
|
//
|
|
/////////////////////////////////////////////////////////////////////////////
|
|
BOOL APIENTRY
|
|
ResumeRestore()
|
|
{
|
|
EnsureTrace();
|
|
TraceFunctEnter("ResumeRestore");
|
|
|
|
// Load SRClient
|
|
g_CSRClientLoader.LoadSrClient();
|
|
|
|
BOOL fRet = FALSE;
|
|
LPCWSTR cszErr;
|
|
DWORD dwInRestore, dwType, dwSize, dwRes;
|
|
CRestoreOperationManager *pROMgr = NULL;
|
|
|
|
if ( !::CheckPrivilegesForRestore() )
|
|
goto Exit;
|
|
|
|
// 1. Even though winlogon would check the registry before calling this
|
|
// API, double check the registry key and then delete it.
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof(DWORD);
|
|
dwRes = ::SHGetValue( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszRestoreInProgress, &dwType, &dwInRestore, &dwSize );
|
|
if ( dwRes != ERROR_SUCCESS )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
DebugTrace(0, "::SHGetValue failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
dwRes = ::SHDeleteValue( HKEY_LOCAL_MACHINE, s_cszSRRegKey, s_cszRestoreInProgress );
|
|
if ( dwRes != ERROR_SUCCESS )
|
|
{
|
|
cszErr = ::GetSysErrStr();
|
|
ErrorTrace(0, "::SHDeleteValue failed - %ls", cszErr);
|
|
goto Exit;
|
|
}
|
|
if ( dwInRestore == 0 )
|
|
{
|
|
DebugTrace(0, "RestoreInProgress is 0");
|
|
goto Exit;
|
|
}
|
|
|
|
// 1. Create CRestoreOperationManager object.
|
|
if ( !::CreateRestoreOperationManager( &pROMgr ) )
|
|
goto Exit;
|
|
|
|
// 2. Perform the Restore Operation.
|
|
if ( !pROMgr->Run( TRUE ) )
|
|
goto Exit;
|
|
|
|
fRet = TRUE;
|
|
Exit:
|
|
SAFE_RELEASE(pROMgr);
|
|
TraceFunctLeave();
|
|
ReleaseTrace();
|
|
return( fRet );
|
|
}
|
|
|
|
|
|
// end of file
|