mirror of https://github.com/tongzx/nt5src
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
6394 lines
211 KiB
6394 lines
211 KiB
/*++
|
|
|
|
Copyright (c) Microsoft Corporation. All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
fileq4.c
|
|
|
|
Abstract:
|
|
|
|
Setup file queue routines for commit (ie, performing enqueued actions).
|
|
|
|
Author:
|
|
|
|
Ted Miller (tedm) 15-Feb-1995
|
|
|
|
Revision History:
|
|
|
|
Jamie Hunter (jamiehun) 28-Jan-1997
|
|
|
|
Added backup queue capabilities
|
|
backup on demand capabilities
|
|
and unwind capabilities
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
|
|
typedef struct _Q_CAB_CB_DATA {
|
|
|
|
PSP_FILE_QUEUE Queue;
|
|
PSOURCE_MEDIA_INFO SourceMedia;
|
|
|
|
PSP_FILE_QUEUE_NODE CurrentFirstNode;
|
|
|
|
PVOID MsgHandler;
|
|
PVOID Context;
|
|
BOOL IsMsgHandlerNativeCharWidth;
|
|
PSETUP_LOG_CONTEXT LogContext;
|
|
|
|
} Q_CAB_CB_DATA, *PQ_CAB_CB_DATA;
|
|
|
|
typedef struct _CERT_PROMPT {
|
|
LPCTSTR lpszDescription;
|
|
LPCTSTR lpszFile;
|
|
SetupapiVerifyProblem ProblemType;
|
|
ULONG DriverSigningPolicy;
|
|
} CERT_PROMPT, *PCERT_PROMPT;
|
|
|
|
typedef struct _DRIVERBLOCK_PROMPT {
|
|
LPCTSTR lpszFile;
|
|
SDBENTRYINFO entryinfo;
|
|
} DRIVERBLOCK_PROMPT, *PDRIVERBLOCK_PROMPT;
|
|
|
|
|
|
DWORD
|
|
pSetupCommitSingleBackup(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PCTSTR FullTargetPath,
|
|
IN LONG TargetRootPath,
|
|
IN LONG TargetSubDir,
|
|
IN LONG TargetFilename,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth,
|
|
IN BOOL RenameExisting,
|
|
OUT PBOOL InUse
|
|
);
|
|
|
|
DWORD
|
|
pCommitCopyQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
DWORD
|
|
pCommitBackupQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
DWORD
|
|
pCommitDeleteQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
DWORD
|
|
pCommitRenameQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
UINT
|
|
pSetupCabinetQueueCallback(
|
|
IN PVOID Context,
|
|
IN UINT Notification,
|
|
IN UINT_PTR Param1,
|
|
IN UINT_PTR Param2
|
|
);
|
|
|
|
|
|
DWORD
|
|
pSetupCopySingleQueuedFile(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN PCTSTR FullSourceName,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
OUT PTSTR NewSourcePath,
|
|
IN BOOL IsMsgHandlerNativeCharWidth,
|
|
IN DWORD CopyStyleFlags
|
|
);
|
|
|
|
DWORD
|
|
pSetupCopySingleQueuedFileCabCase(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN PCTSTR CabinetName,
|
|
IN PCTSTR FullSourceName,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
VOID
|
|
pSetupSetPathOverrides(
|
|
IN PVOID StringTable,
|
|
IN OUT PTSTR RootPath,
|
|
IN OUT PTSTR SubPath,
|
|
IN LONG RootPathId,
|
|
IN LONG SubPathId,
|
|
IN PTSTR NewPath
|
|
);
|
|
|
|
VOID
|
|
pSetupBuildSourceForCopy(
|
|
IN PCTSTR UserRoot,
|
|
IN PCTSTR UserPath,
|
|
IN LONG MediaRoot,
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
OUT PTSTR FullPath
|
|
);
|
|
|
|
INT_PTR
|
|
CALLBACK
|
|
CertifyDlgProc(
|
|
IN HWND hwnd,
|
|
IN UINT msg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam
|
|
);
|
|
|
|
INT_PTR
|
|
CALLBACK
|
|
DriverBlockDlgProc(
|
|
IN HWND hwnd,
|
|
IN UINT msg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam
|
|
);
|
|
|
|
VOID
|
|
RestoreBootReplacedFile(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode
|
|
);
|
|
|
|
VOID
|
|
pSetupExemptFileFromProtection(
|
|
IN PCTSTR FileName,
|
|
IN DWORD FileChangeFlags,
|
|
IN PSETUP_LOG_CONTEXT LogContext, OPTIONAL
|
|
OUT PDWORD QueueNodeFlags OPTIONAL
|
|
);
|
|
|
|
VOID
|
|
pSetupUninstallNewCatalogNodes(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSETUP_LOG_CONTEXT LogContext OPTIONAL
|
|
);
|
|
|
|
|
|
BOOL
|
|
_SetupCommitFileQueue(
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Implementation for SetupCommitFileQueue; handles ANSI and Unicode
|
|
callback routines.
|
|
|
|
Arguments:
|
|
|
|
Same as for SetupCommitFileQueue().
|
|
|
|
IsMsgHandlerNativeCharWidth - indicates whether the MsgHandler callback
|
|
expects native char width args (or ansi ones, in the unicode build
|
|
of this dll).
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, the GetLastError() indicates
|
|
cause of failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSP_FILE_QUEUE Queue;
|
|
DWORD rc;
|
|
BOOL Success = TRUE;
|
|
BOOL ChangedThreadLogContext = FALSE;
|
|
PSETUP_LOG_CONTEXT SavedLogContext = NULL;
|
|
PSETUP_LOG_CONTEXT LogContext = NULL;
|
|
|
|
//
|
|
// Queue handle is actually a pointer to the queue structure.
|
|
//
|
|
Queue = (PSP_FILE_QUEUE)QueueHandle;
|
|
|
|
//
|
|
// do a quick handle validation before anything else
|
|
//
|
|
try {
|
|
Success = ((Queue != NULL) && (Queue != INVALID_HANDLE_VALUE) && (Queue->Signature == SP_FILE_QUEUE_SIG));
|
|
if (Success) {
|
|
LogContext = Queue->LogContext;
|
|
}
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
Success = FALSE;
|
|
}
|
|
if (!Success) {
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
//
|
|
// If there's nothing to do, bail now. This prevents an empty
|
|
// progress dialog from flashing on the screen. Don't return out
|
|
// of the body of the try -- that is bad news performance-wise.
|
|
//
|
|
try {
|
|
Success = (!Queue->DeleteNodeCount && !Queue->RenameNodeCount && !Queue->CopyNodeCount && !Queue->BackupNodeCount);
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return(FALSE);
|
|
}
|
|
if(Success) {
|
|
|
|
//
|
|
// We are successful in that we had no file operations to do. However,
|
|
// we still need to validate the queued catalogs at this time, because
|
|
// we always do validation in the context of file copying. If we don't
|
|
// do this, we have a hole where a device INF that doesn't copy files
|
|
// (e.g., a modem INF) can circumvent driver signing checking.
|
|
//
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_TIME,
|
|
MSG_LOG_BEGIN_VERIFY3_CAT_TIME,
|
|
NULL); // text message
|
|
|
|
rc = _SetupVerifyQueuedCatalogs(Owner,
|
|
Queue,
|
|
VERCAT_INSTALL_INF_AND_CAT,
|
|
NULL,
|
|
NULL
|
|
);
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_TIME,
|
|
MSG_LOG_END_VERIFY3_CAT_TIME,
|
|
NULL); // text message
|
|
|
|
if (rc == NO_ERROR) {
|
|
|
|
//
|
|
// If we performed a backup and this is a device install then call
|
|
// the pSetupCompleteBackup API to create the Reinstall instance
|
|
// subkey and do other device rollback cleanup.
|
|
//
|
|
if (Queue->Flags & FQF_DEVICE_BACKUP) {
|
|
|
|
pSetupCompleteBackup(Queue);
|
|
}
|
|
|
|
Queue->Flags |= FQF_QUEUE_ALREADY_COMMITTED;
|
|
|
|
} else {
|
|
//
|
|
// Go uninstall any newly-copied INFs/PNFs/CATs.
|
|
//
|
|
pSetupUninstallNewCatalogNodes(Queue, LogContext);
|
|
}
|
|
|
|
SetLastError(rc);
|
|
return(rc == NO_ERROR);
|
|
}
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
//
|
|
// make a note of default logging context for duration of queue processing
|
|
// this will catch, eg, INF being opened as part of a callback
|
|
//
|
|
MYASSERT(!ChangedThreadLogContext);
|
|
ChangedThreadLogContext = SetThreadLogContext(LogContext,&SavedLogContext);
|
|
if (ChangedThreadLogContext) {
|
|
//
|
|
// add one more ref to protext log context
|
|
//
|
|
RefLogContext(LogContext);
|
|
}
|
|
|
|
Success = pSetupCallMsgHandler(
|
|
LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTQUEUE,
|
|
(UINT_PTR)Owner,
|
|
0
|
|
);
|
|
if(!Success) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto final;
|
|
}
|
|
|
|
try {
|
|
//
|
|
// Verify catalogs/infs.
|
|
//
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_TIME,
|
|
MSG_LOG_BEGIN_VERIFY2_CAT_TIME,
|
|
NULL); // text message
|
|
|
|
rc = _SetupVerifyQueuedCatalogs(Owner,
|
|
Queue,
|
|
VERCAT_INSTALL_INF_AND_CAT,
|
|
NULL,
|
|
NULL
|
|
);
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_TIME,
|
|
MSG_LOG_END_VERIFY2_CAT_TIME,
|
|
NULL); // text message
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
if(rc != NO_ERROR) {
|
|
goto Bail;
|
|
}
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
//
|
|
// Handle backup first
|
|
// don't commit if there's nothing to do
|
|
//
|
|
|
|
rc = Queue->BackupNodeCount
|
|
? pCommitBackupQueue(Queue,MsgHandler,Context,IsMsgHandlerNativeCharWidth)
|
|
: NO_ERROR;
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
if (!Success) {
|
|
goto Bail;
|
|
}
|
|
|
|
//
|
|
// Handle deletes
|
|
// now done after backups, but may incorporate a per-delete backup
|
|
// don't commit if there's nothing to do
|
|
//
|
|
|
|
rc = Queue->DeleteNodeCount
|
|
? pCommitDeleteQueue(Queue,MsgHandler,Context,IsMsgHandlerNativeCharWidth)
|
|
: NO_ERROR;
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
if (!Success) {
|
|
goto Bail;
|
|
}
|
|
|
|
//
|
|
// Handle renames next.
|
|
// don't commit if there's nothing to do
|
|
//
|
|
|
|
rc = Queue->RenameNodeCount
|
|
? pCommitRenameQueue(Queue,MsgHandler,Context,IsMsgHandlerNativeCharWidth)
|
|
: NO_ERROR;
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
if (!Success) {
|
|
goto Bail;
|
|
}
|
|
|
|
//
|
|
// Handle copies last. Don't bother calling the copy commit routine
|
|
// if there are no files to copy.
|
|
//
|
|
rc = Queue->CopyNodeCount
|
|
? pCommitCopyQueue(Queue,MsgHandler,Context,IsMsgHandlerNativeCharWidth)
|
|
: NO_ERROR;
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
ASSERT_HEAP_IS_VALID();
|
|
|
|
if (!Success) {
|
|
goto Bail;
|
|
}
|
|
|
|
rc = DoAllDelayedMoves(Queue);
|
|
|
|
Success = (rc == NO_ERROR);
|
|
|
|
if(Success) {
|
|
//
|
|
// Set a flag indicating we've committed the file queue (used to keep
|
|
// us from attempting to prune the queue after having committed it).
|
|
//
|
|
Queue->Flags |= FQF_QUEUE_ALREADY_COMMITTED;
|
|
}
|
|
|
|
//
|
|
// If we performed a backup and this is a device install then call
|
|
// the pSetupCompleteBackup API to create the Reinstall instance
|
|
// subkey and do other device rollback cleanup.
|
|
//
|
|
if (Queue->Flags & FQF_DEVICE_BACKUP) {
|
|
|
|
pSetupCompleteBackup(Queue);
|
|
}
|
|
|
|
Bail:
|
|
;
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Success = FALSE;
|
|
rc = ERROR_INVALID_DATA;
|
|
}
|
|
|
|
pSetupCallMsgHandler(
|
|
LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDQUEUE,
|
|
Success,
|
|
0
|
|
);
|
|
|
|
pSetupUnwindAll(Queue, Success);
|
|
|
|
final:
|
|
|
|
//
|
|
// If we didn't succeed, then uninstall any new INFs/PNFs/CATs we may have
|
|
// installed.
|
|
//
|
|
if(!Success) {
|
|
pSetupUninstallNewCatalogNodes(Queue, LogContext);
|
|
}
|
|
|
|
if (ChangedThreadLogContext) {
|
|
//
|
|
// restore thread log context
|
|
//
|
|
SetThreadLogContext(SavedLogContext,NULL);
|
|
DeleteLogContext(LogContext); // counter RefLogContext
|
|
}
|
|
|
|
SetLastError(rc);
|
|
|
|
return(Success);
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version. Also need undecorated (Unicode) version for compatibility
|
|
// with apps that were linked before we had A and W versions.
|
|
//
|
|
BOOL
|
|
SetupCommitFileQueueA(
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PSP_FILE_CALLBACK_A MsgHandler,
|
|
IN PVOID Context
|
|
)
|
|
{
|
|
return(_SetupCommitFileQueue(Owner,QueueHandle,MsgHandler,Context,FALSE));
|
|
}
|
|
|
|
#undef SetupCommitFileQueue
|
|
SetupCommitFileQueue(
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PSP_FILE_CALLBACK_W MsgHandler,
|
|
IN PVOID Context
|
|
)
|
|
{
|
|
return(_SetupCommitFileQueue(Owner,QueueHandle,MsgHandler,Context,TRUE));
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub. Also need undecorated (ANSI) version for compatibility
|
|
// with apps that were linked before we had A and W versions.
|
|
//
|
|
BOOL
|
|
SetupCommitFileQueueW(
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PSP_FILE_CALLBACK_W MsgHandler,
|
|
IN PVOID Context
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(Owner);
|
|
UNREFERENCED_PARAMETER(QueueHandle);
|
|
UNREFERENCED_PARAMETER(MsgHandler);
|
|
UNREFERENCED_PARAMETER(Context);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
|
|
#undef SetupCommitFileQueue
|
|
SetupCommitFileQueue(
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PSP_FILE_CALLBACK_A MsgHandler,
|
|
IN PVOID Context
|
|
)
|
|
{
|
|
return(_SetupCommitFileQueue(Owner,QueueHandle,MsgHandler,Context,TRUE));
|
|
}
|
|
#endif
|
|
|
|
BOOL
|
|
#ifdef UNICODE
|
|
SetupCommitFileQueueW(
|
|
#else
|
|
SetupCommitFileQueueA(
|
|
#endif
|
|
IN HWND Owner, OPTIONAL
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PSP_FILE_CALLBACK MsgHandler,
|
|
IN PVOID Context
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Perform file operations enqueued on a setup file queue.
|
|
|
|
Arguments:
|
|
|
|
OwnerWindow - if specified, supplies the window handle of a window
|
|
that is to be used as the parent of any progress dialogs.
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome.
|
|
|
|
--*/
|
|
|
|
{
|
|
return(_SetupCommitFileQueue(Owner,QueueHandle,MsgHandler,Context,TRUE));
|
|
}
|
|
|
|
|
|
DWORD
|
|
pCommitBackupQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Process the backup Queue
|
|
Backup each file specified in the queue if it exists
|
|
File is marked as backup
|
|
Location of backup is recorded
|
|
Files are not added to unwind queue here
|
|
They get added to unwind queue the first time they are potentially modified
|
|
|
|
See also pCommitDeleteQueue, pCommitRenameQueue and pCommitCopyQueue
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the backup sub-queue
|
|
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
|
|
IsMsgHandlerNativeCharWidth - For Unicode/Ansi support
|
|
|
|
Return Value:
|
|
|
|
DWORD indicating status or success
|
|
|
|
--*/
|
|
{
|
|
PSP_FILE_QUEUE_NODE QueueNode,queueNode;
|
|
UINT u;
|
|
BOOL b;
|
|
DWORD rc;
|
|
PCTSTR FullTargetPath,FullBackupPath;
|
|
FILEPATHS FilePaths;
|
|
BOOL Skipped = FALSE;
|
|
DWORD BackupFlags = SP_BACKUP_BACKUPPASS;
|
|
|
|
MYASSERT(Queue->BackupNodeCount);
|
|
|
|
b = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTSUBQUEUE,
|
|
FILEOP_BACKUP,
|
|
Queue->BackupNodeCount
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
for(QueueNode=Queue->BackupQueue; QueueNode; QueueNode=QueueNode->Next) {
|
|
|
|
//
|
|
// Form the full path of the file to be backed up
|
|
//
|
|
FullBackupPath = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->SourceRootPath,
|
|
QueueNode->SourcePath,
|
|
QueueNode->SourceFilename
|
|
);
|
|
|
|
if(!FullBackupPath) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
FullTargetPath = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->TargetDirectory,
|
|
QueueNode->TargetFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullTargetPath) {
|
|
MyFree(FullBackupPath);
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
FilePaths.Source = FullTargetPath; // copying from
|
|
FilePaths.Target = FullBackupPath; // copying to (backup)
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
Skipped = FALSE;
|
|
|
|
//
|
|
// Inform the callback that we are about to start a backup operation.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTBACKUP,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_BACKUP
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullTargetPath);
|
|
MyFree(FullBackupPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_DOIT) {
|
|
//
|
|
// Attempt the backup. If it fails inform the callback,
|
|
// which may decide to abort, retry. or skip the file.
|
|
//
|
|
//SetFileAttributes(FullTargetPath,FILE_ATTRIBUTE_NORMAL);
|
|
|
|
do {
|
|
rc = pSetupBackupFile((HSPFILEQ)Queue,
|
|
FullTargetPath,
|
|
FullBackupPath,
|
|
-1, // TargetID not known
|
|
QueueNode->TargetDirectory, // what to backup
|
|
-1, // Queue Node's don't maintain this intermediate path
|
|
QueueNode->TargetFilename,
|
|
QueueNode->SourceRootPath, // backup as...
|
|
QueueNode->SourcePath,
|
|
QueueNode->SourceFilename,
|
|
&b
|
|
);
|
|
if (rc == NO_ERROR) {
|
|
if (b) {
|
|
// delayed (in use)
|
|
|
|
QueueNode->InternalFlags |= INUSE_IN_USE;
|
|
//
|
|
// Tell the callback.
|
|
//
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
FilePaths.Flags = FILEOP_BACKUP;
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_FILEOPDELAYED,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
}
|
|
} else {
|
|
FilePaths.Win32Error = rc;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_BACKUPERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullTargetPath);
|
|
MyFree(FullBackupPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_SKIP) {
|
|
// we skipped the backup
|
|
Skipped = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
} else {
|
|
// we skipped the backup
|
|
Skipped = TRUE;
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
FilePaths.Win32Error = rc;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDBACKUP,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
MyFree(FullTargetPath);
|
|
MyFree(FullBackupPath);
|
|
}
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDSUBQUEUE,
|
|
FILEOP_BACKUP,
|
|
0
|
|
);
|
|
|
|
rc = NO_ERROR;
|
|
|
|
clean0:
|
|
|
|
SetLastError(rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
DWORD
|
|
pSetupCommitSingleBackup(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PCTSTR FullTargetPath,
|
|
IN LONG TargetRootPath,
|
|
IN LONG TargetSubDir,
|
|
IN LONG TargetFilename,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth,
|
|
IN BOOL RenameExisting,
|
|
OUT PBOOL InUse
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Check a single file that is potentially about to be modified
|
|
|
|
If the target file doesn't exist, then this routine does nothing
|
|
If the target file hasn't been backed up, back it up
|
|
If the target file has been backed up, but is not on unwind queue,
|
|
add to unwind queue
|
|
|
|
The default target location of the backup is used, which is either
|
|
into a backup directory tree, or a temporary backup location
|
|
Location of backup is recorded
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the backup sub-queue
|
|
FullTargetPath - String giving target path, or NULL if not formed
|
|
TargetRootPath - String ID giving RootPath, or -1 if not specified
|
|
TargetSubDir - String ID giving SubDir (relative to RootPath),
|
|
or -1 if not specified
|
|
TargetFilename - String ID giving Filename, or -1 if not specified
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
IsMsgHandlerNativeCharWidth - For Unicode/Ansi support
|
|
RenameExisting - Should existing file be renamed?
|
|
InUse - if specified, set to indicate if file is in use or not
|
|
This should never be the case
|
|
|
|
Return Value:
|
|
|
|
DWORD indicating status or success
|
|
|
|
--*/
|
|
{
|
|
UINT u;
|
|
BOOL b;
|
|
DWORD rc;
|
|
DWORD rc2;
|
|
FILEPATHS FilePaths;
|
|
LONG TargetID;
|
|
PTSTR TargetPathLocal = NULL;
|
|
PSP_UNWIND_NODE UnwindNode = NULL;
|
|
SP_TARGET_ENT TargetInfo;
|
|
BOOL FileOfSameNameExists;
|
|
BOOL DoBackup = TRUE;
|
|
BOOL NeedUnwind = FALSE;
|
|
BOOL Skipped = FALSE;
|
|
WIN32_FILE_ATTRIBUTE_DATA FileAttribData;
|
|
UINT OldMode;
|
|
BOOL DoRename;
|
|
DWORD BackupFlags = SP_BACKUP_DEMANDPASS;
|
|
|
|
//
|
|
// used in this function to init time field
|
|
//
|
|
static const FILETIME zeroTime = {
|
|
0,0
|
|
};
|
|
|
|
OldMode = SetErrorMode(SEM_FAILCRITICALERRORS); // inhibit unexpected dialog boxes
|
|
|
|
MYASSERT(Queue);
|
|
|
|
if (FullTargetPath == NULL) {
|
|
TargetPathLocal = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
TargetRootPath,
|
|
TargetSubDir,
|
|
TargetFilename);
|
|
|
|
if(!TargetPathLocal) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
FullTargetPath = TargetPathLocal;
|
|
}
|
|
|
|
FileOfSameNameExists = GetFileAttributesEx(FullTargetPath, GetFileExInfoStandard, &FileAttribData);
|
|
|
|
if (!FileOfSameNameExists) {
|
|
// file doesn't exist, so no need to backup
|
|
rc = NO_ERROR;
|
|
goto clean0;
|
|
}
|
|
|
|
rc = pSetupBackupGetTargetByPath((HSPFILEQ)Queue,
|
|
NULL, // use Queue's string table
|
|
FullTargetPath,
|
|
TargetRootPath,
|
|
TargetSubDir,
|
|
TargetFilename,
|
|
&TargetID,
|
|
&TargetInfo
|
|
);
|
|
|
|
if (rc != NO_ERROR) {
|
|
// failed for some strange reason
|
|
goto clean0;
|
|
|
|
}
|
|
|
|
if (TargetInfo.InternalFlags & SP_TEFLG_INUSE) {
|
|
//
|
|
// was "inuse'd" before
|
|
// we mark as still INUSE
|
|
if (InUse != NULL) {
|
|
*InUse = TRUE;
|
|
}
|
|
//
|
|
// Don't consider this an error, unless we were supposed to rename the
|
|
// existing file.
|
|
//
|
|
rc = RenameExisting ? ERROR_SHARING_VIOLATION : NO_ERROR;
|
|
goto clean0;
|
|
}
|
|
|
|
if (TargetInfo.InternalFlags & SP_TEFLG_SKIPPED) {
|
|
//
|
|
// was skipped before
|
|
// we can't rely on it now
|
|
//
|
|
rc = NO_ERROR;
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// If we've been asked to backup the existing file, then make sure the
|
|
// SP_TEFLG_RENAMEEXISTING flag is set in the TargetInfo. Also, figure out
|
|
// if we've already done the rename.
|
|
//
|
|
if(RenameExisting &&
|
|
!(TargetInfo.InternalFlags & SP_TEFLG_RENAMEEXISTING)) {
|
|
//
|
|
// We'd better not think we already renamed this file!
|
|
//
|
|
MYASSERT(!(TargetInfo.InternalFlags & SP_TEFLG_MOVED));
|
|
|
|
TargetInfo.InternalFlags |= SP_TEFLG_RENAMEEXISTING;
|
|
|
|
//
|
|
// update internal info (this call should never fail)
|
|
//
|
|
pSetupBackupSetTargetByID((HSPFILEQ)Queue,
|
|
TargetID,
|
|
&TargetInfo
|
|
);
|
|
}
|
|
|
|
//
|
|
// Figure out whether we've been asked to rename the existing file to a
|
|
// temp name in the same directory, but haven't yet done so.
|
|
//
|
|
DoRename = ((TargetInfo.InternalFlags & (SP_TEFLG_RENAMEEXISTING | SP_TEFLG_MOVED)) == SP_TEFLG_RENAMEEXISTING);
|
|
|
|
if(TargetInfo.InternalFlags & SP_TEFLG_SAVED) {
|
|
//
|
|
// already backed up
|
|
//
|
|
DoBackup = FALSE;
|
|
|
|
if((TargetInfo.InternalFlags & SP_TEFLG_UNWIND) && !DoRename) {
|
|
//
|
|
// already added to unwind queue, and we don't need to do a rename--
|
|
// don't need to do anything at all
|
|
//
|
|
rc = NO_ERROR;
|
|
goto clean0;
|
|
}
|
|
//
|
|
// we don't need to backup
|
|
// but we still need to add to unwind queue, rename the existing file,
|
|
// or both.
|
|
//
|
|
}
|
|
|
|
if(DoBackup) {
|
|
BackupFlags |= SP_BACKUP_DEMANDPASS;
|
|
}
|
|
if(DoRename) {
|
|
BackupFlags |= SP_BACKUP_BOOTFILE | SP_BACKUP_SPECIAL;
|
|
}
|
|
|
|
FilePaths.Source = FullTargetPath; // what we are backing up
|
|
FilePaths.Target = NULL; // indicates an automatic backup
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
if (DoRename) {
|
|
pSetupExemptFileFromProtection(
|
|
FullTargetPath,
|
|
SFC_ACTION_ADDED | SFC_ACTION_REMOVED | SFC_ACTION_MODIFIED
|
|
| SFC_ACTION_RENAMED_OLD_NAME |SFC_ACTION_RENAMED_NEW_NAME,
|
|
Queue->LogContext,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
if (DoBackup && (Queue->Flags & FQF_BACKUP_AWARE)) {
|
|
//
|
|
// Inform the callback that we are about to start a backup operation.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTBACKUP,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_BACKUP
|
|
);
|
|
} else {
|
|
//
|
|
// no backup, or not backup aware, assume a default
|
|
//
|
|
u = FILEOP_DOIT;
|
|
}
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
if((u == FILEOP_DOIT) || (BackupFlags & SP_BACKUP_SPECIAL)) {
|
|
//
|
|
// Attempt the backup. If it fails inform the callback,
|
|
// which may decide to abort, retry. or skip the file.
|
|
//
|
|
//SetFileAttributes(FullTargetPath,FILE_ATTRIBUTE_NORMAL);
|
|
|
|
//
|
|
// Setup an unwind node, unless we already have one.
|
|
//
|
|
if(!(TargetInfo.InternalFlags & SP_TEFLG_UNWIND)) {
|
|
|
|
UnwindNode = MyMalloc(sizeof(SP_UNWIND_NODE));
|
|
if (UnwindNode == NULL) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
UnwindNode->NextNode = Queue->UnwindQueue;
|
|
UnwindNode->TargetID = TargetID;
|
|
if (RetreiveFileSecurity( FullTargetPath, &(UnwindNode->SecurityDesc)) != NO_ERROR) {
|
|
// failed, but not fatal
|
|
UnwindNode->SecurityDesc = NULL;
|
|
}
|
|
if (GetSetFileTimestamp( FullTargetPath, &(UnwindNode->CreateTime),
|
|
&(UnwindNode->AccessTime),
|
|
&(UnwindNode->WriteTime),
|
|
FALSE) != NO_ERROR) {
|
|
// failed, but not fatal
|
|
UnwindNode->CreateTime = zeroTime;
|
|
UnwindNode->AccessTime = zeroTime;
|
|
UnwindNode->WriteTime = zeroTime;
|
|
}
|
|
}
|
|
|
|
if (DoBackup || DoRename) {
|
|
do {
|
|
rc = pSetupBackupFile((HSPFILEQ)Queue,
|
|
FullTargetPath, // since we know this, pass it
|
|
NULL, // automatic destination
|
|
TargetID, // we got this earlier
|
|
TargetRootPath, // since we know this, pass it
|
|
TargetSubDir,
|
|
TargetFilename,
|
|
-1, // use the details from TargetID (or temp)
|
|
-1,
|
|
-1,
|
|
&b // in use (should always return FALSE)
|
|
);
|
|
if (rc == NO_ERROR) {
|
|
if (InUse != NULL) {
|
|
*InUse = b;
|
|
}
|
|
if (b) {
|
|
//
|
|
// if file is in use, callback can decide what to do
|
|
//
|
|
if (Queue->Flags & FQF_BACKUP_AWARE) {
|
|
//
|
|
// Tell the callback.
|
|
//
|
|
FilePaths.Win32Error = ERROR_SHARING_VIOLATION;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
if (Queue->Flags & FQF_BACKUP_AWARE) {
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_BACKUPERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
} else {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
goto clean0;
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// success!!!!!
|
|
// we would have to unwind this if setup fails
|
|
//
|
|
NeedUnwind = TRUE;
|
|
}
|
|
} else {
|
|
FilePaths.Win32Error = rc;
|
|
FilePaths.Flags = BackupFlags;
|
|
|
|
if (Queue->Flags & FQF_BACKUP_AWARE) {
|
|
//
|
|
// inform about error
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_BACKUPERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
} else {
|
|
//
|
|
// if caller is not backup aware, abort
|
|
//
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
goto clean0;
|
|
}
|
|
|
|
if(u == FILEOP_SKIP) {
|
|
//
|
|
// we skipped the backup
|
|
//
|
|
Skipped = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
|
|
} else {
|
|
//
|
|
// didn't need to backup, only need to add to unwind queue
|
|
//
|
|
NeedUnwind = TRUE;
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// we skipped the backup
|
|
//
|
|
Skipped = TRUE;
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
if (DoBackup) {
|
|
|
|
FilePaths.Win32Error = rc;
|
|
|
|
if (Queue->Flags & FQF_BACKUP_AWARE) {
|
|
//
|
|
// report result only if backup aware
|
|
//
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDBACKUP,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
}
|
|
}
|
|
|
|
if (Skipped) {
|
|
//
|
|
// once we return, file may get overwritten or deleted
|
|
// we have to save the fact it has been skipped once
|
|
// so we always skip this file
|
|
//
|
|
if (pSetupBackupGetTargetByID((HSPFILEQ)Queue, TargetID, &TargetInfo) == NO_ERROR) {
|
|
//
|
|
// flag the file should always be skipped
|
|
//
|
|
TargetInfo.InternalFlags|=SP_TEFLG_SKIPPED;
|
|
pSetupBackupSetTargetByID((HSPFILEQ)Queue, TargetID, &TargetInfo);
|
|
}
|
|
}
|
|
else if (NeedUnwind) {
|
|
//
|
|
// We only want to add this to unwind queue
|
|
//
|
|
if (pSetupBackupGetTargetByID((HSPFILEQ)Queue, TargetID, &TargetInfo) == NO_ERROR) {
|
|
if ((TargetInfo.InternalFlags&SP_TEFLG_UNWIND)==FALSE) {
|
|
//
|
|
// node needs to be added to unwind queue
|
|
// we only ever do this once
|
|
//
|
|
Queue->UnwindQueue = UnwindNode;
|
|
//
|
|
// set to NULL so we don't clean it up later
|
|
//
|
|
UnwindNode = NULL;
|
|
|
|
//
|
|
// flag that we've added it to unwind queue
|
|
// so we don't try and do it again later
|
|
//
|
|
TargetInfo.InternalFlags|=SP_TEFLG_UNWIND;
|
|
|
|
pSetupBackupSetTargetByID((HSPFILEQ)Queue, TargetID, &TargetInfo);
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
|
|
rc = NO_ERROR;
|
|
|
|
clean0:
|
|
|
|
if (UnwindNode != NULL) {
|
|
//
|
|
// we allocated, but didn't use this structure
|
|
//
|
|
if (UnwindNode->SecurityDesc != NULL) {
|
|
MyFree(UnwindNode->SecurityDesc);
|
|
}
|
|
MyFree(UnwindNode);
|
|
}
|
|
if (TargetPathLocal != NULL) {
|
|
MyFree(TargetPathLocal);
|
|
}
|
|
|
|
SetErrorMode(OldMode);
|
|
|
|
SetLastError(rc);
|
|
|
|
return rc;
|
|
}
|
|
|
|
DWORD
|
|
pCommitDeleteQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Process the delete Queue
|
|
Delete each file specified in the queue
|
|
Files are backed up before they are deleted (if not already backed up)
|
|
|
|
See also pCommitBackupQueue, pCommitRenameQueue and pCommitCopyQueue
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the delete sub-queue
|
|
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
|
|
IsMsgHandlerNativeCharWidth - For Unicode/Ansi support
|
|
|
|
Return Value:
|
|
|
|
DWORD indicating status or success
|
|
|
|
--*/
|
|
{
|
|
PSP_FILE_QUEUE_NODE QueueNode,queueNode;
|
|
UINT u;
|
|
BOOL b;
|
|
DWORD rc;
|
|
PCTSTR FullTargetPath;
|
|
FILEPATHS FilePaths;
|
|
BOOL BackupInUse = FALSE;
|
|
BOOL TargetIsProtected;
|
|
|
|
MYASSERT(Queue->DeleteNodeCount);
|
|
|
|
b = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTSUBQUEUE,
|
|
FILEOP_DELETE,
|
|
Queue->DeleteNodeCount
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
|
|
for(QueueNode=Queue->DeleteQueue; QueueNode; QueueNode=QueueNode->Next) {
|
|
|
|
//
|
|
// Form the full path of the file to be deleted.
|
|
//
|
|
FullTargetPath = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->TargetDirectory,
|
|
QueueNode->TargetFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullTargetPath) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Backup the file we're about to delete
|
|
//
|
|
if((rc=pSetupDoLastKnownGoodBackup(Queue,
|
|
FullTargetPath,
|
|
LASTGOOD_OPERATION_DELETE,
|
|
NULL)) != NO_ERROR) {
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
rc = pSetupCommitSingleBackup(Queue,
|
|
FullTargetPath,
|
|
QueueNode->TargetDirectory,
|
|
-1,
|
|
QueueNode->TargetFilename,
|
|
MsgHandler,
|
|
Context,
|
|
IsMsgHandlerNativeCharWidth,
|
|
FALSE,
|
|
&BackupInUse
|
|
);
|
|
if (rc != NO_ERROR) {
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
|
|
FilePaths.Source = NULL;
|
|
FilePaths.Target = FullTargetPath;
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
FilePaths.Flags = 0;
|
|
|
|
//
|
|
// Inform the callback that we are about to start a delete operation.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTDELETE,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_DELETE
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_DOIT) {
|
|
//
|
|
// Attempt the delete. If it fails inform the callback,
|
|
// which may decide to abort, retry. or skip the file.
|
|
//
|
|
SetFileAttributes(FullTargetPath,FILE_ATTRIBUTE_NORMAL);
|
|
|
|
do {
|
|
if (BackupInUse) {
|
|
rc = ERROR_SHARING_VIOLATION;
|
|
} else {
|
|
rc = DeleteFile(FullTargetPath) ? NO_ERROR : GetLastError();
|
|
}
|
|
if((rc == ERROR_ACCESS_DENIED)
|
|
|| (rc == ERROR_SHARING_VIOLATION)
|
|
|| (rc == ERROR_USER_MAPPED_FILE)) {
|
|
//
|
|
// The file is probably in use.
|
|
//
|
|
if(QueueNode->InternalFlags & IQF_DELAYED_DELETE_OK) {
|
|
//
|
|
// Inf wanted delete on next reboot. Check to see if
|
|
// we're being asked to delete a protected system file.
|
|
// If so (and all the catalog nodes associated with the
|
|
// queue were OK), then we'll allow this to happen.
|
|
// Otherwise, we'll silently skip the deletion (and log
|
|
// it).
|
|
//
|
|
MYASSERT((Queue->Flags & FQF_DID_CATALOGS_OK) ||
|
|
(Queue->Flags & FQF_DID_CATALOGS_FAILED));
|
|
|
|
if(Queue->Flags & FQF_DID_CATALOGS_OK) {
|
|
|
|
QueueNode->InternalFlags |= INUSE_IN_USE;
|
|
|
|
TargetIsProtected = IsFileProtected(FullTargetPath,
|
|
Queue->LogContext,
|
|
NULL
|
|
);
|
|
|
|
if(b = PostDelayedMove(Queue,
|
|
FullTargetPath,
|
|
NULL,
|
|
-1,
|
|
TargetIsProtected)) {
|
|
//
|
|
// Tell the callback.
|
|
//
|
|
FilePaths.Source = NULL;
|
|
FilePaths.Target = FullTargetPath;
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
FilePaths.Flags = FILEOP_DELETE;
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_FILEOPDELAYED,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
}
|
|
} else {
|
|
//
|
|
// We're installing an unsigned package. Skip the
|
|
// delayed delete operation, and generate a log
|
|
// entry about this.
|
|
//
|
|
WriteLogEntry(Queue->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
MSG_LOG_DELAYED_DELETE_SKIPPED_FOR_SFC,
|
|
NULL,
|
|
FullTargetPath
|
|
);
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// Just skip this file.
|
|
//
|
|
b = TRUE;
|
|
}
|
|
|
|
rc = b ? NO_ERROR : GetLastError();
|
|
|
|
if(rc) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_ERROR | SETUP_LOG_BUFFER,
|
|
MSG_LOG_DELAYDELETE_FILE_ERROR,
|
|
NULL,
|
|
FullTargetPath);
|
|
WriteLogError(Queue->LogContext,SETUP_LOG_ERROR,rc);
|
|
} else {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_DELAYDELETED_FILE,
|
|
NULL,
|
|
FullTargetPath);
|
|
}
|
|
|
|
} else if(rc) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
DEL_ERR_LOG_LEVEL(rc) | SETUP_LOG_BUFFER,
|
|
MSG_LOG_DELETE_FILE_ERROR,
|
|
NULL,
|
|
FullTargetPath);
|
|
WriteLogError(Queue->LogContext,DEL_ERR_LOG_LEVEL(rc),rc);
|
|
} else {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_DELETED_FILE,
|
|
NULL,
|
|
FullTargetPath);
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
if( rc == NO_ERROR )
|
|
{
|
|
rc = pSetupCallSCE(
|
|
ST_SCE_DELETE,
|
|
FullTargetPath,
|
|
NULL,
|
|
NULL,
|
|
-1,
|
|
NULL
|
|
);
|
|
SetLastError( rc );
|
|
}
|
|
#endif
|
|
|
|
if(rc != NO_ERROR) {
|
|
FilePaths.Win32Error = rc;
|
|
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_DELETEERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_SKIP) {
|
|
break;
|
|
}
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
} else {
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
FilePaths.Win32Error = rc;
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDDELETE,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
MyFree(FullTargetPath);
|
|
}
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDSUBQUEUE,
|
|
FILEOP_DELETE,
|
|
0
|
|
);
|
|
|
|
rc = NO_ERROR;
|
|
|
|
clean0:
|
|
SetLastError(rc);
|
|
return rc;
|
|
}
|
|
|
|
DWORD
|
|
pCommitRenameQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Process the rename Queue
|
|
Rename each file specified in the queue
|
|
Files are backed up before they are renamed (if not already backed up)
|
|
If the target exists, it is also backed up (if not already backed up)
|
|
|
|
Performance: this can get optimized by treating the newly named files
|
|
as a backup
|
|
|
|
See also pCommitBackupQueue, pCommitDeleteQueue and pCommitCopyQueue
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the rename sub-queue
|
|
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
|
|
IsMsgHandlerNativeCharWidth - For Unicode/Ansi support
|
|
|
|
Return Value:
|
|
|
|
DWORD indicating status or success
|
|
|
|
--*/
|
|
{
|
|
PSP_FILE_QUEUE_NODE QueueNode,queueNode;
|
|
UINT u;
|
|
BOOL b;
|
|
DWORD rc;
|
|
PCTSTR FullTargetPath;
|
|
PCTSTR FullSourcePath;
|
|
FILEPATHS FilePaths;
|
|
BOOL BackupInUse = FALSE;
|
|
BOOL TargetIsProtected;
|
|
|
|
MYASSERT(Queue->RenameNodeCount);
|
|
|
|
b = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTSUBQUEUE,
|
|
FILEOP_RENAME,
|
|
Queue->RenameNodeCount
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
goto clean0;
|
|
}
|
|
for(QueueNode=Queue->RenameQueue; QueueNode; QueueNode=QueueNode->Next) {
|
|
|
|
//
|
|
// Form the full source path of the file to be renamed.
|
|
//
|
|
FullSourcePath = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->SourcePath,
|
|
QueueNode->SourceFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullSourcePath) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Form the full target path of the file to be renamed.
|
|
//
|
|
FullTargetPath = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->TargetDirectory == -1 ? QueueNode->SourcePath : QueueNode->TargetDirectory,
|
|
QueueNode->TargetFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullTargetPath) {
|
|
MyFree(FullSourcePath);
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Backup the file we may be overwriting
|
|
//
|
|
if((rc=pSetupDoLastKnownGoodBackup(Queue,
|
|
FullTargetPath,
|
|
0,
|
|
NULL)) != NO_ERROR) {
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
rc = pSetupCommitSingleBackup(Queue,
|
|
FullTargetPath,
|
|
QueueNode->TargetDirectory == -1 ? QueueNode->SourcePath : QueueNode->TargetDirectory,
|
|
-1, // we don't use this
|
|
QueueNode->TargetFilename,
|
|
MsgHandler,
|
|
Context,
|
|
IsMsgHandlerNativeCharWidth,
|
|
FALSE,
|
|
&BackupInUse
|
|
);
|
|
if (rc != NO_ERROR) {
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Backup the file we're about to rename
|
|
//
|
|
|
|
if((rc=pSetupDoLastKnownGoodBackup(Queue,
|
|
FullSourcePath,
|
|
LASTGOOD_OPERATION_DELETE,
|
|
NULL)) != NO_ERROR) {
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
rc = pSetupCommitSingleBackup(Queue,
|
|
FullSourcePath,
|
|
QueueNode->SourcePath,
|
|
-1, // we don't use this????
|
|
QueueNode->SourceFilename,
|
|
MsgHandler,
|
|
Context,
|
|
IsMsgHandlerNativeCharWidth,
|
|
FALSE,
|
|
&b
|
|
);
|
|
if (rc != NO_ERROR) {
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
if (b) {
|
|
//
|
|
// BackupInUse is the "OR" of the two backup In-Use flags
|
|
//
|
|
BackupInUse = TRUE;
|
|
}
|
|
|
|
FilePaths.Source = FullSourcePath;
|
|
FilePaths.Target = FullTargetPath;
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
|
|
//
|
|
// Inform the callback that we are about to start a rename operation.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTRENAME,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_RENAME
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_DOIT) {
|
|
//
|
|
// Attempt the rename. If it fails inform the callback,
|
|
// which may decide to abort, retry. or skip the file.
|
|
//
|
|
do {
|
|
if (BackupInUse) {
|
|
//
|
|
// backup is in use, must delay op. Check to see if either
|
|
// the source or target files are protected system files.
|
|
// If so (and all the catalog nodes associated with the
|
|
// queue were OK), then we'll allos this to happen.
|
|
// Otherwise, we'll silently fail the rename (and log it).
|
|
//
|
|
MYASSERT((Queue->Flags & FQF_DID_CATALOGS_OK) ||
|
|
(Queue->Flags & FQF_DID_CATALOGS_FAILED));
|
|
|
|
if(Queue->Flags & FQF_DID_CATALOGS_OK) {
|
|
|
|
TargetIsProtected = IsFileProtected(FullSourcePath,
|
|
Queue->LogContext,
|
|
NULL
|
|
);
|
|
if(!TargetIsProtected) {
|
|
TargetIsProtected = IsFileProtected(FullTargetPath,
|
|
Queue->LogContext,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
if(b = PostDelayedMove(Queue,
|
|
FullSourcePath,
|
|
FullTargetPath,
|
|
-1,
|
|
TargetIsProtected)) {
|
|
rc = NO_ERROR;
|
|
}
|
|
else
|
|
{
|
|
rc = GetLastError();
|
|
}
|
|
if(rc) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
DEL_ERR_LOG_LEVEL(rc) | SETUP_LOG_BUFFER,
|
|
MSG_LOG_DELAYRENAME_FILE_ERROR,
|
|
NULL,
|
|
FullSourcePath,
|
|
FullTargetPath);
|
|
WriteLogError(Queue->LogContext,DEL_ERR_LOG_LEVEL(rc),rc);
|
|
} else {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_DELAYRENAMED_FILE,
|
|
NULL,
|
|
FullSourcePath,
|
|
FullTargetPath);
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// We're installing an unsigned package. Skip the
|
|
// delayed rename operation, and generate a log
|
|
// entry about this.
|
|
//
|
|
WriteLogEntry(Queue->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
MSG_LOG_DELAYED_MOVE_SKIPPED_FOR_SFC,
|
|
NULL,
|
|
FullTargetPath
|
|
);
|
|
//
|
|
// act as if no error occurred.
|
|
//
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
} else {
|
|
rc = MoveFile(FullSourcePath,FullTargetPath) ? NO_ERROR : GetLastError();
|
|
if(rc) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
DEL_ERR_LOG_LEVEL(rc) | SETUP_LOG_BUFFER,
|
|
MSG_LOG_RENAME_FILE_ERROR,
|
|
NULL,
|
|
FullSourcePath,
|
|
FullTargetPath);
|
|
WriteLogError(Queue->LogContext,DEL_ERR_LOG_LEVEL(rc),rc);
|
|
} else {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_RENAMED_FILE,
|
|
NULL,
|
|
FullSourcePath,
|
|
FullTargetPath);
|
|
}
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
|
|
if( rc == NO_ERROR )
|
|
{
|
|
rc = pSetupCallSCE(
|
|
ST_SCE_RENAME,
|
|
FullSourcePath,
|
|
NULL,
|
|
FullTargetPath,
|
|
-1,
|
|
NULL
|
|
);
|
|
SetLastError( rc );
|
|
}
|
|
|
|
#endif
|
|
|
|
if((rc == ERROR_FILE_NOT_FOUND) || (rc == ERROR_PATH_NOT_FOUND)) {
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
if(rc != NO_ERROR) {
|
|
FilePaths.Win32Error = rc;
|
|
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_RENAMEERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
goto clean0;
|
|
}
|
|
if(u == FILEOP_SKIP) {
|
|
break;
|
|
}
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
} else {
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
FilePaths.Win32Error = rc;
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDRENAME,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
MyFree(FullSourcePath);
|
|
MyFree(FullTargetPath);
|
|
}
|
|
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDSUBQUEUE,
|
|
FILEOP_RENAME,
|
|
0
|
|
);
|
|
|
|
rc = NO_ERROR;
|
|
|
|
clean0:
|
|
SetLastError(rc);
|
|
return rc;
|
|
}
|
|
|
|
DWORD
|
|
pCommitCopyQueue(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Process the copy sub-Queues
|
|
Copy each file specified in the sub-queues
|
|
Files are backed up before they are overwritten (if not already backed up)
|
|
See also pCommitBackupQueue, pCommitDeleteQueue and pCommitRenameQueue
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the copy sub-queues
|
|
|
|
MsgHandler - Supplies a callback routine to be notified
|
|
of various significant events in the queue processing.
|
|
|
|
Context - Supplies a value that is passed to the MsgHandler
|
|
callback function.
|
|
|
|
IsMsgHandlerNativeCharWidth - For Unicode/Ansi support
|
|
|
|
Return Value:
|
|
|
|
DWORD indicating status or success
|
|
|
|
--*/
|
|
{
|
|
PSOURCE_MEDIA_INFO SourceMediaInfo;
|
|
SOURCE_MEDIA SourceMedia;
|
|
PTCHAR p, temp;
|
|
UINT SourcePathLen;
|
|
UINT u;
|
|
DWORD rc;
|
|
Q_CAB_CB_DATA QData;
|
|
BOOL b;
|
|
BOOL FirstIteration;
|
|
PSP_FILE_QUEUE_NODE QueueNode,queueNode;
|
|
TCHAR UserSourceRoot[MAX_PATH];
|
|
TCHAR UserSourcePath[MAX_PATH];
|
|
TCHAR FullSourcePath[MAX_PATH];
|
|
TCHAR UserOverride[MAX_PATH];
|
|
LPCTSTR RestorePath = NULL;
|
|
UINT DriveType;
|
|
BOOL IsRemovable, AnyProcessed, AnyNotProcessed, SkipMedia;
|
|
BOOL SpecialMedia = FALSE;
|
|
BOOL LocateCab;
|
|
PCTSTR MediaRoot;
|
|
DWORD MediaLogTag;
|
|
LONG Cabfile;
|
|
LONG Tagfile;
|
|
|
|
//
|
|
// The caller is supposed to skip calling us if there are no files
|
|
// to be copied.
|
|
//
|
|
MYASSERT(Queue->CopyNodeCount);
|
|
|
|
//
|
|
// Inform the callback that we are starting.
|
|
//
|
|
b = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTSUBQUEUE,
|
|
FILEOP_COPY,
|
|
Queue->CopyNodeCount
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
return(rc);
|
|
}
|
|
|
|
if(Queue->RestorePathID != -1) {
|
|
RestorePath = pSetupStringTableStringFromId(Queue->StringTable, Queue->RestorePathID);
|
|
DiskPromptGetDriveType(RestorePath, &DriveType, &IsRemovable);
|
|
if(IsRemovable) {
|
|
//
|
|
// do not allow restore from removable media
|
|
//
|
|
RestorePath = NULL;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Initially, no user-specified override path exists.
|
|
//
|
|
UserSourceRoot[0] = TEXT('\0');
|
|
UserSourcePath[0] = TEXT('\0');
|
|
|
|
//
|
|
// The outermost loop iterates through all the source media descriptors.
|
|
//
|
|
for(SourceMediaInfo=Queue->SourceMediaList; SourceMediaInfo; SourceMediaInfo=SourceMediaInfo->Next) {
|
|
|
|
//
|
|
// If there are no files on this particular media, skip it.
|
|
// Otherwise get pointer to queue node for first file on this media.
|
|
//
|
|
if(!SourceMediaInfo->CopyNodeCount) {
|
|
continue;
|
|
}
|
|
MYASSERT(SourceMediaInfo->CopyQueue);
|
|
|
|
//
|
|
// if last media was special media (see long discussion above),
|
|
// then forget about any user override
|
|
//
|
|
if (SpecialMedia) {
|
|
UserSourceRoot[0] = TEXT('\0');
|
|
UserSourcePath[0] = TEXT('\0');
|
|
SpecialMedia = FALSE;
|
|
}
|
|
|
|
//
|
|
// see if this media is special media
|
|
//
|
|
if (SourceMediaInfo->Flags & ( SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH |
|
|
SMI_FLAG_USE_LOCAL_SOURCE_CAB ) ) {
|
|
SpecialMedia = TRUE;
|
|
}
|
|
|
|
//
|
|
// If we're in restore-mode
|
|
// we've been given a directory to restore from
|
|
// ignore the media root, and use restore-point root
|
|
// restore as many files as we can
|
|
//
|
|
// note, we check for file presence via FileExists
|
|
// rather than trying to determine file name
|
|
// since we'll always backup in uncompressed form
|
|
// with same name as listed in [SourceDisksNames]
|
|
//
|
|
|
|
if(RestorePath) {
|
|
//
|
|
// Restore Symantics - prior to prompting for media, see
|
|
// if we can restore backup
|
|
//
|
|
QueueNode = NULL;
|
|
for(queueNode = SourceMediaInfo->CopyQueue;
|
|
queueNode;
|
|
queueNode=queueNode->Next) {
|
|
|
|
pSetupBuildSourceForCopy(
|
|
RestorePath,
|
|
NULL,
|
|
SourceMediaInfo->SourceRootPath,
|
|
Queue,
|
|
queueNode,
|
|
FullSourcePath
|
|
);
|
|
|
|
//
|
|
// don't allow alternate sourcenames in this case
|
|
//
|
|
if(FileExists(FullSourcePath,NULL)) {
|
|
//
|
|
// backup exists, copy it
|
|
//
|
|
rc = pSetupCopySingleQueuedFile(
|
|
Queue,
|
|
queueNode,
|
|
FullSourcePath,
|
|
MsgHandler,
|
|
Context,
|
|
UserOverride,
|
|
IsMsgHandlerNativeCharWidth,
|
|
SP_COPY_ALREADYDECOMP // backup already decomp'd.
|
|
);
|
|
if(rc == NO_ERROR) {
|
|
//
|
|
// we restored this file through backup
|
|
// carry on to next file
|
|
//
|
|
queueNode->InternalFlags |= IQF_PROCESSED;
|
|
continue;
|
|
}
|
|
//
|
|
// we know backup existed so if this failed
|
|
// consider it major enough to abort restore
|
|
// (eg, file unsigned, user specified abort)
|
|
//
|
|
SetLastError(rc);
|
|
return(rc);
|
|
}
|
|
if(!QueueNode) {
|
|
//
|
|
// first problematic file
|
|
//
|
|
QueueNode = queueNode;
|
|
}
|
|
}
|
|
if(!QueueNode) {
|
|
//
|
|
// we copied all files of this media from backup
|
|
// carry on to next media
|
|
//
|
|
continue;
|
|
}
|
|
} else {
|
|
//
|
|
// not restoring, start at first file
|
|
//
|
|
QueueNode = SourceMediaInfo->CopyQueue;
|
|
}
|
|
|
|
|
|
//
|
|
// We will need to prompt for media, which requires some preparation.
|
|
// We need to get the first file in the queue for this media, because
|
|
// its path is where we will expect to find it or its cabinet or tag
|
|
// file. If there is no tag file, then we will look for the file
|
|
// itself.
|
|
//
|
|
|
|
FirstIteration = TRUE;
|
|
SkipMedia = FALSE;
|
|
LocateCab = FALSE;
|
|
Tagfile = SourceMediaInfo->Tagfile;
|
|
Cabfile = SourceMediaInfo->Cabfile;
|
|
|
|
RepromptMedia:
|
|
//
|
|
// The case where we have non-removeable media and the path was
|
|
// previously overridden must be handled specially. For example, we
|
|
// could have files queued on the same source root but different
|
|
// subdirs. If the user changes the network location, for example,
|
|
// we have to be careful or we'll ignore the change in subdirectories
|
|
// as we move among the media.
|
|
//
|
|
// To work around this, we check on non-removable media to see if the
|
|
// queue node we're presently working with is in a subdirectory. If it
|
|
// is, then we reset our UserSourcePath string.
|
|
//
|
|
// (andrewr)...I don't get this comment above. The current code
|
|
// iterates through each source media info structure, which doesn't include
|
|
// subdirectory information, only source root path information. If it
|
|
// does, then the caller is doing something really wierd, since they
|
|
// should be using the SourcePath to define subdirectories from one master
|
|
// root.
|
|
//
|
|
// It appears that the reasoning behind the code below is as follows:
|
|
//
|
|
// The assumption is that if we have removable media and multiple source
|
|
// paths, then we will have to swap media out of the drive. We don't
|
|
// override source root paths if we are dealing with removable media.
|
|
// If the source root path is non removable, then all of the source media
|
|
// is "tied together." If the user overrides the source root path, then
|
|
// we override subsequent fixed media source root paths.
|
|
//
|
|
// In the case of dealing with service pack source media or a local cab-file
|
|
// drivers cache, the source media info for a queue will not be tied together,
|
|
// even though we're dealing with fixed media.
|
|
//
|
|
// To reconcile the comments above and the reasoning it uses with the
|
|
// contradiction that svc pack media imposes, we have 2 options:
|
|
//
|
|
// 1. If we encounter flags that indicate one of our special cases, then don't
|
|
// use any user override for the new source media. (or, put another way,
|
|
// if we know that the last media was actually one of these special media,
|
|
// then don't allow an override of the normal media.
|
|
//
|
|
// 2. Introduce some sort of hueristic that determines if the prior source media
|
|
// and the current source media are similar. If they are, then go ahead and
|
|
// use any user specified override, otherwise use the proper path.
|
|
//
|
|
//
|
|
// For simplicities sake, I use approach 1 above. This is made a little simpler
|
|
// by following the following rule. When adding source media to the media list,
|
|
// insert special media (ie, has flags identifying the media as svc pack media)
|
|
// at the head of the list, insert normal media after that. By following this
|
|
// approach we know that we can just "zero out" the user overrides for the special
|
|
// media and we'll just do the right thing for the regular media.
|
|
//
|
|
// In the case where there is an explicit cab-file to use
|
|
// then we ask the user to point to cab-file instead of source file (first iteration)
|
|
//
|
|
|
|
MediaRoot = *UserSourceRoot
|
|
? UserSourceRoot
|
|
: pSetupStringTableStringFromId(Queue->StringTable, SourceMediaInfo->SourceRootPath);
|
|
|
|
DiskPromptGetDriveType(MediaRoot, &DriveType, &IsRemovable);
|
|
if(!IsRemovable && (QueueNode->SourcePath != -1)) {
|
|
*UserSourcePath = TEXT('\0');
|
|
}
|
|
|
|
pSetupBuildSourceForCopy(
|
|
UserSourceRoot,
|
|
UserSourcePath,
|
|
SourceMediaInfo->SourceRootPath,
|
|
Queue,
|
|
QueueNode,
|
|
FullSourcePath
|
|
);
|
|
|
|
if (FirstIteration
|
|
&& (Tagfile != Cabfile)
|
|
&& (Cabfile != -1)) {
|
|
|
|
MYASSERT(!SkipMedia);
|
|
MYASSERT(!(SourceMediaInfo->Flags & SMI_FLAG_USE_LOCAL_SOURCE_CAB));
|
|
|
|
//
|
|
// build location of cab file
|
|
//
|
|
temp = _tcsrchr(FullSourcePath,TEXT('\\'));
|
|
MYASSERT( temp );
|
|
if(temp) {
|
|
*(temp+1) = 0;
|
|
} else {
|
|
FullSourcePath[0] = 0;
|
|
}
|
|
|
|
|
|
//
|
|
// obtain path of (potential) cab file
|
|
//
|
|
pSetupConcatenatePaths( FullSourcePath, pSetupStringTableStringFromId(Queue->StringTable,Cabfile), MAX_PATH, NULL );
|
|
LocateCab = TRUE;
|
|
|
|
} else {
|
|
LocateCab = FALSE;
|
|
}
|
|
|
|
if((p = _tcsrchr(FullSourcePath,TEXT('\\')))!=NULL) {
|
|
*p++ = TEXT('\0');
|
|
} else {
|
|
//
|
|
// I'm being pedantic here, this should never happen
|
|
//
|
|
MYASSERT(p);
|
|
p = FullSourcePath;
|
|
}
|
|
|
|
//
|
|
// Now FullSourcePath has the path part and p has the file part
|
|
// for the first file in the queue for this media (or explicit cab file)
|
|
// Get the media in the drive by calling the callback function.
|
|
//
|
|
// Although it would be nice to not have to
|
|
// call this callback if we know that we don't have to (there is media
|
|
// where the caller said there should be (local media, media already in, etc.)
|
|
// we do need to call this so that we afford the caller the luxury of
|
|
// changing their mind one last time.
|
|
//
|
|
// the only exception to this rule is if we are using the local driver
|
|
// cache cab-file. In this case, we don't want the user to ever get
|
|
// prompted for this file, so we skip any media prompting. We know that
|
|
// if we have media added that has this flag set, then the cab already exists
|
|
// and we can just use it (otherwise we wouldn't have initialized it in the
|
|
// first place, we'd just use the os source path!)
|
|
//
|
|
SourceMedia.Tagfile = (Tagfile != -1 && FirstIteration)
|
|
? pSetupStringTableStringFromId(
|
|
Queue->StringTable,
|
|
Tagfile
|
|
)
|
|
: NULL;
|
|
|
|
SourceMedia.Description = (SourceMediaInfo->Description != -1)
|
|
? pSetupStringTableStringFromId(
|
|
Queue->StringTable,
|
|
SourceMediaInfo->DescriptionDisplayName
|
|
)
|
|
: NULL;
|
|
|
|
SourceMedia.SourcePath = FullSourcePath;
|
|
SourceMedia.SourceFile = p;
|
|
SourceMedia.Flags = (QueueNode->StyleFlags & (SP_COPY_NOSKIP | SP_COPY_WARNIFSKIP | SP_COPY_NOBROWSE));
|
|
|
|
MediaLogTag = AllocLogInfoSlotOrLevel(Queue->LogContext,SETUP_LOG_INFO,FALSE);
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
MediaLogTag,
|
|
MSG_LOG_NEEDMEDIA,
|
|
NULL,
|
|
SourceMedia.Tagfile ? SourceMedia.Tagfile : TEXT(""),
|
|
SourceMedia.Description ? SourceMedia.Description : TEXT(""),
|
|
SourceMedia.SourcePath ? SourceMedia.SourcePath : TEXT(""),
|
|
SourceMedia.SourceFile ? SourceMedia.SourceFile : TEXT(""),
|
|
SourceMedia.Flags
|
|
);
|
|
|
|
if( SkipMedia || (FirstIteration && (SourceMediaInfo->Flags & SMI_FLAG_USE_LOCAL_SOURCE_CAB)) ) {
|
|
u = FILEOP_DOIT;
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_VERBOSE,
|
|
MSG_LOG_NEEDMEDIA_AUTOSKIP,
|
|
NULL
|
|
);
|
|
} else {
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_NEEDMEDIA,
|
|
(UINT_PTR)&SourceMedia,
|
|
(UINT_PTR)UserOverride
|
|
);
|
|
}
|
|
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_ERROR|SETUP_LOG_BUFFER,
|
|
MSG_LOG_NEEDMEDIA_ABORT,
|
|
NULL);
|
|
WriteLogError(Queue->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
rc
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,MediaLogTag);
|
|
MediaLogTag = 0;
|
|
SetLastError(rc);
|
|
return(rc);
|
|
}
|
|
if(u == FILEOP_SKIP) {
|
|
//
|
|
// If this file was a bootfile replacement, then we need to restore
|
|
// the original file that was renamed to a temporary filename.
|
|
//
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_WARNING,
|
|
MSG_LOG_NEEDMEDIA_SKIP,
|
|
NULL
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,MediaLogTag);
|
|
MediaLogTag = 0;
|
|
if(QueueNode->StyleFlags & SP_COPY_REPLACE_BOOT_FILE) {
|
|
RestoreBootReplacedFile(Queue, QueueNode);
|
|
}
|
|
|
|
//
|
|
// If there are more files on this media, then try another one.
|
|
// Otherwise we're done with this media.
|
|
//
|
|
QueueNode->InternalFlags |= IQF_PROCESSED;
|
|
for(QueueNode=QueueNode->Next; QueueNode; QueueNode=QueueNode->Next) {
|
|
if(!(QueueNode->InternalFlags & IQF_PROCESSED)) {
|
|
FirstIteration = FALSE;
|
|
goto RepromptMedia;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
if(u == FILEOP_NEWPATH) {
|
|
//
|
|
// User gave us a new source path. See which parts of the new path
|
|
// match the existing path/overrides we are using.
|
|
//
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_NEEDMEDIA_NEWPATH,
|
|
NULL,
|
|
UserOverride
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,MediaLogTag);
|
|
MediaLogTag = 0;
|
|
pSetupSetPathOverrides(
|
|
Queue->StringTable,
|
|
UserSourceRoot,
|
|
UserSourcePath,
|
|
SourceMediaInfo->SourceRootPath,
|
|
QueueNode->SourcePath,
|
|
UserOverride
|
|
);
|
|
}
|
|
//
|
|
// logging specific stuff
|
|
//
|
|
if(MediaLogTag!=0) {
|
|
//
|
|
// we explicitly cleared MediaLogTag for each case we handled
|
|
//
|
|
if (u != FILEOP_DOIT) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_WARNING,
|
|
MSG_LOG_NEEDMEDIA_BADRESULT,
|
|
NULL,
|
|
u);
|
|
}
|
|
ReleaseLogInfoSlot(Queue->LogContext,MediaLogTag);
|
|
MediaLogTag = 0;
|
|
}
|
|
|
|
//
|
|
// If we get here, the media is now accessible.
|
|
// Some or all of the files might be in a cabinet whose name is the tagfile.
|
|
//
|
|
// NOTE: Win95 used the tagfile field to be the cabinet name instead.
|
|
// If present it is used as a tagfile of sorts. The absence of a tagfile
|
|
// means the files are not in cabinets. For NT, we don't bother
|
|
// with all of this but instead try to be a little smarter.
|
|
//
|
|
// Scan the media for all source files we expect to find on it.
|
|
// If we find a file, process it. Later we hit the cabinet and only
|
|
// process the files we didn't already find outside the cabinet.
|
|
//
|
|
// exception to this is "explicit cabinet"
|
|
//
|
|
if(LocateCab) {
|
|
//
|
|
// an explicit cabinet was specified
|
|
// this is first iteration
|
|
// we've gone through NEED_MEDIA to obtain disk for this cabinet
|
|
// don't try to process files outside cabinet
|
|
// we know there is at least one file not processed
|
|
//
|
|
b = TRUE;
|
|
queueNode=QueueNode;
|
|
} else {
|
|
//
|
|
// tagfile may also be a cabfile
|
|
// but process all files outside the cabfile first
|
|
//
|
|
for(queueNode=QueueNode; queueNode; queueNode=queueNode->Next) {
|
|
|
|
if(queueNode->InternalFlags & IQF_PROCESSED) {
|
|
//
|
|
// Already processed. Skip to next file.
|
|
//
|
|
continue;
|
|
}
|
|
|
|
pSetupBuildSourceForCopy(
|
|
UserSourceRoot,
|
|
UserSourcePath,
|
|
SourceMediaInfo->SourceRootPath,
|
|
Queue,
|
|
queueNode,
|
|
FullSourcePath
|
|
);
|
|
|
|
rc = SetupDetermineSourceFileName(FullSourcePath,&b,&p,NULL);
|
|
if(rc == NO_ERROR || SkipMedia) {
|
|
//
|
|
// Found the file outside a cabinet. Process it now.
|
|
//
|
|
if(rc == NO_ERROR) {
|
|
rc = pSetupCopySingleQueuedFile(
|
|
Queue,
|
|
queueNode,
|
|
p,
|
|
MsgHandler,
|
|
Context,
|
|
UserOverride,
|
|
IsMsgHandlerNativeCharWidth,
|
|
0
|
|
);
|
|
MyFree(p);
|
|
} else {
|
|
//
|
|
// We didn't find the source file, but we're going to try
|
|
// to copy it anyway since we've decided not to skip the
|
|
// prompt for media.
|
|
//
|
|
rc = pSetupCopySingleQueuedFile(
|
|
Queue,
|
|
queueNode,
|
|
FullSourcePath,
|
|
MsgHandler,
|
|
Context,
|
|
UserOverride,
|
|
IsMsgHandlerNativeCharWidth,
|
|
0
|
|
);
|
|
}
|
|
|
|
if(rc != NO_ERROR) {
|
|
return(rc);
|
|
}
|
|
|
|
//
|
|
// See if we have a new source path.
|
|
//
|
|
if(UserOverride[0]) {
|
|
pSetupSetPathOverrides(
|
|
Queue->StringTable,
|
|
UserSourceRoot,
|
|
UserSourcePath,
|
|
SourceMediaInfo->SourceRootPath,
|
|
queueNode->SourcePath,
|
|
UserOverride
|
|
);
|
|
}
|
|
}
|
|
}
|
|
//
|
|
// See if any files still need to be processed.
|
|
//
|
|
for(b=FALSE,queueNode=QueueNode; queueNode; queueNode=queueNode->Next) {
|
|
if(!(queueNode->InternalFlags & IQF_PROCESSED)) {
|
|
b = TRUE;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// If any files still need to be processed and we have a potential
|
|
// cabinet file, go try to extract them from a cabinet.
|
|
//
|
|
if(b && (Cabfile != -1) && FirstIteration) {
|
|
|
|
pSetupBuildSourceForCopy(
|
|
UserSourceRoot,
|
|
UserSourcePath,
|
|
SourceMediaInfo->SourceRootPath,
|
|
Queue,
|
|
queueNode,
|
|
FullSourcePath
|
|
);
|
|
|
|
temp = _tcsrchr(FullSourcePath,TEXT('\\'));
|
|
MYASSERT( temp );
|
|
if(temp) {
|
|
*(temp+1) = 0;
|
|
}
|
|
|
|
//
|
|
// obtain path of (potential) cab file
|
|
//
|
|
pSetupConcatenatePaths( FullSourcePath, pSetupStringTableStringFromId(Queue->StringTable,Cabfile), MAX_PATH, NULL );
|
|
|
|
if(DiamondIsCabinet(FullSourcePath)) {
|
|
|
|
QData.Queue = Queue;
|
|
QData.SourceMedia = SourceMediaInfo;
|
|
QData.MsgHandler = MsgHandler;
|
|
QData.IsMsgHandlerNativeCharWidth = IsMsgHandlerNativeCharWidth;
|
|
QData.Context = Context;
|
|
QData.LogContext = Queue->LogContext;
|
|
|
|
rc = DiamondProcessCabinet(
|
|
FullSourcePath,
|
|
0,
|
|
pSetupCabinetQueueCallback,
|
|
&QData,
|
|
TRUE
|
|
);
|
|
|
|
if(rc != NO_ERROR) {
|
|
return(rc);
|
|
}
|
|
|
|
//
|
|
// Now reset the cabfile to indicate that there is no cabinet.
|
|
// If we don't do this and there are still files that have not
|
|
// been processed, we'll end up in an infinite loop -- the prompt
|
|
// will come back successfully, and we'll just keep going around
|
|
// and around looking through the cabinet, etc.
|
|
//
|
|
Cabfile = -1;
|
|
Tagfile = -1; // for compatability
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we get here and files *still* need to be processed,
|
|
// assume the files are in a different directory somewhere
|
|
// and start all over with this media.
|
|
//
|
|
FirstIteration = FALSE;
|
|
DiskPromptGetDriveType(FullSourcePath, &DriveType, &IsRemovable);
|
|
AnyProcessed = FALSE;
|
|
AnyNotProcessed = FALSE;
|
|
|
|
for(QueueNode = SourceMediaInfo->CopyQueue;
|
|
QueueNode;
|
|
QueueNode=QueueNode->Next) {
|
|
|
|
if(IsRemovable) {
|
|
if(!(QueueNode->InternalFlags & IQF_PROCESSED)) {
|
|
if(Tagfile != -1) {
|
|
SkipMedia = TRUE;
|
|
}
|
|
goto RepromptMedia;
|
|
}
|
|
} else { // Fixed media
|
|
if(QueueNode->InternalFlags & IQF_PROCESSED) {
|
|
AnyProcessed = TRUE;
|
|
} else {
|
|
AnyNotProcessed = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(!IsRemovable) {
|
|
if(AnyNotProcessed) {
|
|
|
|
//
|
|
// If some of the files are present on fixed media, we don't
|
|
// want to look elsewhere.
|
|
//
|
|
if(AnyProcessed) {
|
|
SkipMedia = TRUE;
|
|
}
|
|
|
|
//
|
|
// Find the first unprocessed file
|
|
//
|
|
for(QueueNode = SourceMediaInfo->CopyQueue;
|
|
QueueNode;
|
|
QueueNode = QueueNode->Next) {
|
|
|
|
if(!(QueueNode->InternalFlags & IQF_PROCESSED)) {
|
|
break;
|
|
}
|
|
}
|
|
MYASSERT(QueueNode);
|
|
|
|
goto RepromptMedia;
|
|
}
|
|
}
|
|
|
|
} // end for each source media info
|
|
|
|
//
|
|
// Tell handler we're done with the copy queue and return.
|
|
//
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDSUBQUEUE,
|
|
FILEOP_COPY,
|
|
0
|
|
);
|
|
|
|
return(NO_ERROR);
|
|
}
|
|
|
|
|
|
VOID
|
|
pSetupBuildSourceForCopy(
|
|
IN PCTSTR UserRoot,
|
|
IN PCTSTR UserPath,
|
|
IN LONG MediaRoot,
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
OUT PTSTR FullPath
|
|
)
|
|
{
|
|
PCTSTR p;
|
|
|
|
|
|
//
|
|
// If there is a user-specified override root path, use that instead of
|
|
// the root path specified in the source media descriptor.
|
|
//
|
|
MYASSERT(Queue);
|
|
MYASSERT(QueueNode);
|
|
MYASSERT(FullPath);
|
|
|
|
p = (UserRoot && UserRoot[0])
|
|
? UserRoot
|
|
: pSetupStringTableStringFromId(Queue->StringTable,MediaRoot);
|
|
|
|
|
|
lstrcpyn(FullPath,p,MAX_PATH);
|
|
|
|
//
|
|
// If there is a user-specified override path, use that instead of any
|
|
// path specified in the copy node.
|
|
//
|
|
if(UserPath && UserPath[0]) {
|
|
p = UserPath;
|
|
} else {
|
|
if(QueueNode->SourcePath == -1) {
|
|
p = NULL;
|
|
} else {
|
|
p = pSetupStringTableStringFromId(Queue->StringTable,QueueNode->SourcePath);
|
|
}
|
|
}
|
|
|
|
if(p) {
|
|
pSetupConcatenatePaths(FullPath,p,MAX_PATH,NULL);
|
|
}
|
|
|
|
//
|
|
// Fetch the filename and append.
|
|
//
|
|
p = pSetupStringTableStringFromId(Queue->StringTable,QueueNode->SourceFilename),
|
|
pSetupConcatenatePaths(FullPath,p,MAX_PATH,NULL);
|
|
|
|
}
|
|
|
|
VOID
|
|
pSetupSetPathOverrides(
|
|
IN PVOID StringTable,
|
|
IN OUT PTSTR RootPath,
|
|
IN OUT PTSTR SubPath,
|
|
IN LONG RootPathId,
|
|
IN LONG SubPathId,
|
|
IN PTSTR NewPath
|
|
)
|
|
{
|
|
PCTSTR root,path;
|
|
UINT u,l;
|
|
|
|
//
|
|
// See if the existing root override or root path is a prefix
|
|
// of the path the user gave us.
|
|
//
|
|
MYASSERT(RootPath);
|
|
MYASSERT(SubPath);
|
|
root = RootPath[0] ? RootPath : pSetupStringTableStringFromId(StringTable,RootPathId);
|
|
u = lstrlen(root);
|
|
|
|
path = SubPath[0]
|
|
? SubPath
|
|
: ((SubPathId == -1) ? NULL : pSetupStringTableStringFromId(StringTable,SubPathId));
|
|
|
|
if(path && (*path == TEXT('\\'))) {
|
|
path++;
|
|
}
|
|
|
|
if(_tcsnicmp(NewPath,root,u)) {
|
|
//
|
|
// Root path does not match what we're currently using, ie, the user
|
|
// supplied a new path. In this case, we will see if the currently in-use
|
|
// subpath matches the suffix of the new path, and if so, we'll assume
|
|
// that is the override subpath and shorten the override root path.
|
|
//
|
|
lstrcpy(RootPath,NewPath);
|
|
if(path) {
|
|
u = lstrlen(NewPath);
|
|
l = lstrlen(path);
|
|
|
|
if((u > l) && (NewPath[(u-l)-1] == TEXT('\\')) && !lstrcmpi(NewPath+u-l,path)) {
|
|
//
|
|
// Subpath tail matches. Truncate the root override and
|
|
// leave the subpath override alone.
|
|
//
|
|
RootPath[(u-l)-1] = 0;
|
|
} else {
|
|
//
|
|
// In this case, we need to indicate an override subpath of the root,
|
|
// or else all subsequent accesses will still try to append the subpath
|
|
// specified in the copy node, which is not what we want.
|
|
//
|
|
SubPath[0] = TEXT('\\');
|
|
SubPath[1] = 0;
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// Root path matches what we are currently using.
|
|
//
|
|
// See if the tail of the user-specified path matches the existing
|
|
// subpath. If not, then use the rest of the root path as the subpath
|
|
// override. If the tail matches, then extend the user override root.
|
|
//
|
|
// Examples:
|
|
//
|
|
// File was queued with root = f:\, subpath = \mips
|
|
//
|
|
// User override path is f:\alpha
|
|
//
|
|
// The new status will be leave override root alone;
|
|
// override subpath = \alpha
|
|
//
|
|
// File was queued with root = \\foo\bar, subpath = \i386
|
|
//
|
|
// User override path is \\foo\bar\new\i386
|
|
//
|
|
// The new status will be a root override of \\foo\bar\new;
|
|
// no override subpath.
|
|
//
|
|
NewPath += u;
|
|
if(*NewPath == TEXT('\\')) {
|
|
NewPath++;
|
|
}
|
|
|
|
if(path) {
|
|
u = lstrlen(NewPath);
|
|
l = lstrlen(path);
|
|
|
|
if((u >= l) && !lstrcmpi(NewPath+u-l,path)) {
|
|
//
|
|
// Change root override and indicate no override subpath.
|
|
//
|
|
SubPath[0] = TEXT('\0');
|
|
NewPath[u-l] = TEXT('\0');
|
|
lstrcpy(RootPath,root);
|
|
pSetupConcatenatePaths(RootPath,NewPath,MAX_PATH,NULL);
|
|
u = lstrlen(RootPath);
|
|
if(u && (*CharPrev(RootPath,RootPath+u) == TEXT('\\'))) {
|
|
RootPath[u-1] = TEXT('\0'); // valid to do if last char is '\'
|
|
}
|
|
} else {
|
|
//
|
|
// Leave override root alone but change subpath.
|
|
//
|
|
lstrcpy(SubPath,NewPath);
|
|
if(!SubPath[0]) {
|
|
SubPath[0] = TEXT('\\');
|
|
SubPath[1] = TEXT('\0');
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// File was queued without a subpath. If there's a subpath
|
|
// in what the user gave us, use it as the override.
|
|
//
|
|
if(*NewPath) {
|
|
lstrcpy(SubPath,NewPath);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
UINT
|
|
pSetupCabinetQueueCallback(
|
|
IN PVOID Context,
|
|
IN UINT Notification,
|
|
IN UINT_PTR Param1,
|
|
IN UINT_PTR Param2
|
|
)
|
|
{
|
|
UINT rc;
|
|
PCABINET_INFO CabinetInfo;
|
|
PFILE_IN_CABINET_INFO FileInfo;
|
|
TCHAR TempPath[MAX_PATH];
|
|
PTSTR CabinetFile;
|
|
PTSTR QueuedFile;
|
|
PTSTR FilePart1,FilePart2;
|
|
PTSTR FullTargetPath;
|
|
PFILEPATHS FilePaths;
|
|
PSP_FILE_QUEUE_NODE QueueNode,FirstNode,LastNode;
|
|
PQ_CAB_CB_DATA QData;
|
|
UINT h;
|
|
SOURCE_MEDIA SourceMedia;
|
|
DWORD status;
|
|
|
|
QData = (PQ_CAB_CB_DATA)Context;
|
|
|
|
switch(Notification) {
|
|
|
|
case SPFILENOTIFY_CABINETINFO:
|
|
//
|
|
// We don't do anything with this.
|
|
//
|
|
rc = NO_ERROR;
|
|
break;
|
|
|
|
case SPFILENOTIFY_FILEINCABINET:
|
|
//
|
|
// New file within a cabinet.
|
|
//
|
|
// Determine whether we want to copy this file.
|
|
// The context we get has all the stuff we need in it
|
|
// to make this determination.
|
|
//
|
|
// Note that the queue could contain multiple copy operations
|
|
// involving this file, but we only want to extract it once!
|
|
//
|
|
FileInfo = (PFILE_IN_CABINET_INFO)Param1;
|
|
CabinetFile = (PTSTR)Param2;
|
|
|
|
if(FilePart1 = _tcsrchr(FileInfo->NameInCabinet,TEXT('\\'))) {
|
|
FilePart1++;
|
|
} else {
|
|
FilePart1 = (PTSTR)FileInfo->NameInCabinet;
|
|
}
|
|
|
|
rc = FILEOP_SKIP;
|
|
FileInfo->Win32Error = NO_ERROR;
|
|
FirstNode = NULL;
|
|
|
|
//
|
|
// Find ALL instances of this file in the queue and mark them.
|
|
//
|
|
for(QueueNode=QData->SourceMedia->CopyQueue; QueueNode; QueueNode=QueueNode->Next) {
|
|
|
|
if(QueueNode->InternalFlags & IQF_PROCESSED) {
|
|
//
|
|
// This file was already processed. Ignore it.
|
|
//
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// Check the filename in the cabinet against the file
|
|
// in the media's copy queue.
|
|
//
|
|
QueuedFile = pSetupStringTableStringFromId(
|
|
QData->Queue->StringTable,
|
|
QueueNode->SourceFilename
|
|
);
|
|
|
|
if(FilePart2 = _tcsrchr(QueuedFile,TEXT('\\'))) {
|
|
FilePart2++;
|
|
} else {
|
|
FilePart2 = QueuedFile;
|
|
}
|
|
|
|
if(!lstrcmpi(FilePart1,FilePart2)) {
|
|
//
|
|
// We want this file.
|
|
//
|
|
rc = FILEOP_DOIT;
|
|
QueueNode->InternalFlags |= IQF_PROCESSED | IQF_MATCH;
|
|
if(!FirstNode) {
|
|
FirstNode = QueueNode;
|
|
}
|
|
LastNode = QueueNode;
|
|
}
|
|
}
|
|
|
|
if(rc == FILEOP_DOIT) {
|
|
//
|
|
// We want this file. Tell the caller the full target pathname
|
|
// to be used, which is a temporary file in the directory
|
|
// where the first instance of the file will ultimately go.
|
|
// We do this so we can call SetupInstallFile later (perhaps
|
|
// multiple times), which will handle version checks, etc.
|
|
//
|
|
// Before attempting to create a temp file make sure the path exists.
|
|
//
|
|
lstrcpyn(
|
|
TempPath,
|
|
pSetupStringTableStringFromId(QData->Queue->StringTable,FirstNode->TargetDirectory),
|
|
MAX_PATH
|
|
);
|
|
pSetupConcatenatePaths(TempPath,TEXT("x"),MAX_PATH,NULL); // last component ignored
|
|
status = pSetupMakeSurePathExists(TempPath);
|
|
if(status == NO_ERROR) {
|
|
LastNode->InternalFlags |= IQF_LAST_MATCH;
|
|
if(GetTempFileName(
|
|
pSetupStringTableStringFromId(QData->Queue->StringTable,FirstNode->TargetDirectory),
|
|
TEXT("SETP"),
|
|
0,
|
|
FileInfo->FullTargetName
|
|
)) {
|
|
QData->CurrentFirstNode = FirstNode;
|
|
} else {
|
|
status = GetLastError();
|
|
if(status == ERROR_ACCESS_DENIED) {
|
|
FileInfo->Win32Error = ERROR_INVALID_TARGET;
|
|
} else {
|
|
FileInfo->Win32Error = status;
|
|
}
|
|
rc = FILEOP_ABORT;
|
|
SetLastError(FileInfo->Win32Error);
|
|
}
|
|
} else {
|
|
if(status == ERROR_ACCESS_DENIED) {
|
|
FileInfo->Win32Error = ERROR_INVALID_TARGET;
|
|
} else {
|
|
FileInfo->Win32Error = status;
|
|
}
|
|
rc = FILEOP_ABORT;
|
|
SetLastError(FileInfo->Win32Error);
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
case SPFILENOTIFY_FILEEXTRACTED:
|
|
|
|
FilePaths = (PFILEPATHS)Param1;
|
|
//
|
|
// The current file was extracted. If this was successful,
|
|
// then we need to call SetupInstallFile on it to perform version
|
|
// checks and move it into its final location or locations.
|
|
//
|
|
// The .Source member of FilePaths is the cabinet file.
|
|
//
|
|
// The .Target member is the name of the temporary file, which is
|
|
// very useful, as it is the name if the file to use as the source
|
|
// in copy operations.
|
|
//
|
|
// Process each file in the queue that we care about.
|
|
//
|
|
if((rc = FilePaths->Win32Error) == NO_ERROR) {
|
|
|
|
for(QueueNode=QData->CurrentFirstNode; QueueNode && (rc==NO_ERROR); QueueNode=QueueNode->Next) {
|
|
//
|
|
// If we don't care about this file, skip it.
|
|
//
|
|
if(!(QueueNode->InternalFlags & IQF_MATCH)) {
|
|
continue;
|
|
}
|
|
|
|
QueueNode->InternalFlags &= ~IQF_MATCH;
|
|
|
|
|
|
rc = pSetupCopySingleQueuedFileCabCase(
|
|
QData->Queue,
|
|
QueueNode,
|
|
FilePaths->Source,
|
|
FilePaths->Target,
|
|
QData->MsgHandler,
|
|
QData->Context,
|
|
QData->IsMsgHandlerNativeCharWidth
|
|
);
|
|
|
|
//
|
|
// If this was the last file that matched, break out.
|
|
//
|
|
if(QueueNode->InternalFlags & IQF_LAST_MATCH) {
|
|
QueueNode->InternalFlags &= ~IQF_LAST_MATCH;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Delete the temporary file we extracted -- we don't need it any more.
|
|
//
|
|
DeleteFile(FilePaths->Target);
|
|
|
|
break;
|
|
|
|
case SPFILENOTIFY_NEEDNEWCABINET:
|
|
//
|
|
// Need a new cabinet.
|
|
//
|
|
CabinetInfo = (PCABINET_INFO)Param1;
|
|
|
|
SourceMedia.Tagfile = NULL;
|
|
SourceMedia.Description = CabinetInfo->DiskName;
|
|
SourceMedia.SourcePath = CabinetInfo->CabinetPath;
|
|
SourceMedia.SourceFile = CabinetInfo->CabinetFile;
|
|
SourceMedia.Flags = SP_FLAG_CABINETCONTINUATION | SP_COPY_NOSKIP;
|
|
|
|
h = pSetupCallMsgHandler(
|
|
QData->LogContext,
|
|
QData->MsgHandler,
|
|
QData->IsMsgHandlerNativeCharWidth,
|
|
QData->Context,
|
|
SPFILENOTIFY_NEEDMEDIA,
|
|
(UINT_PTR)&SourceMedia,
|
|
Param2
|
|
);
|
|
|
|
switch(h) {
|
|
|
|
case FILEOP_NEWPATH:
|
|
case FILEOP_DOIT:
|
|
rc = NO_ERROR;
|
|
break;
|
|
|
|
case FILEOP_ABORT:
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
break;
|
|
|
|
default:
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
break;
|
|
|
|
}
|
|
//
|
|
// in this case, rc is a status code
|
|
// but also set it as last error
|
|
//
|
|
SetLastError(rc);
|
|
break;
|
|
|
|
default:
|
|
MYASSERT(0);
|
|
rc = 0; // indeterminate
|
|
}
|
|
|
|
return(rc);
|
|
}
|
|
|
|
|
|
DWORD
|
|
pSetupCopySingleQueuedFile(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN PCTSTR FullSourceName,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
OUT PTSTR NewSourcePath,
|
|
IN BOOL IsMsgHandlerNativeCharWidth,
|
|
IN DWORD CopyStyleFlags
|
|
)
|
|
{
|
|
PTSTR FullTargetName;
|
|
FILEPATHS FilePaths;
|
|
UINT u;
|
|
BOOL InUse;
|
|
TCHAR source[MAX_PATH],PathBuffer[MAX_PATH];
|
|
DWORD rc;
|
|
BOOL b;
|
|
BOOL BackupInUse = FALSE;
|
|
BOOL SignatureVerifyFailed;
|
|
|
|
NewSourcePath[0] = 0;
|
|
PathBuffer[0] = 0;
|
|
|
|
QueueNode->InternalFlags |= IQF_PROCESSED;
|
|
|
|
//
|
|
// Form the full target path of the file.
|
|
//
|
|
FullTargetName = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->TargetDirectory,
|
|
QueueNode->TargetFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullTargetName) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
lstrcpyn(source,FullSourceName,MAX_PATH);
|
|
|
|
//
|
|
// check if we need to backup before we copy
|
|
//
|
|
if((rc=pSetupDoLastKnownGoodBackup(Queue,
|
|
FullTargetName,
|
|
0,
|
|
NULL)) != NO_ERROR) {
|
|
MyFree(FullTargetName);
|
|
goto clean0;
|
|
}
|
|
rc = pSetupCommitSingleBackup(Queue,
|
|
FullTargetName,
|
|
QueueNode->TargetDirectory,
|
|
-1,
|
|
QueueNode->TargetFilename,
|
|
MsgHandler,
|
|
Context,
|
|
IsMsgHandlerNativeCharWidth,
|
|
(QueueNode->StyleFlags & SP_COPY_REPLACE_BOOT_FILE),
|
|
&BackupInUse
|
|
);
|
|
if (rc != NO_ERROR) {
|
|
MyFree(FullTargetName);
|
|
goto clean0;
|
|
}
|
|
|
|
if (BackupInUse) {
|
|
//
|
|
// if we couldn't do backup, force the IN_USE flag
|
|
//
|
|
QueueNode->StyleFlags |= SP_COPY_FORCE_IN_USE;
|
|
|
|
}
|
|
|
|
do {
|
|
//
|
|
// Form the full source name.
|
|
//
|
|
FilePaths.Source = source;
|
|
FilePaths.Target = FullTargetName;
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
|
|
//
|
|
// Also, pass the callback routine the CopyStyle flags we're about to
|
|
// use.
|
|
//
|
|
// Callback flags are read-only.
|
|
//
|
|
FilePaths.Flags = QueueNode->StyleFlags;
|
|
|
|
//
|
|
// Notify the callback that the copy is starting.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTCOPY,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_COPY
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_ERROR|SETUP_LOG_BUFFER,
|
|
MSG_LOG_STARTCOPY_ABORT,
|
|
NULL);
|
|
WriteLogError(Queue->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
rc);
|
|
break;
|
|
}
|
|
|
|
if(u == FILEOP_DOIT) {
|
|
|
|
//
|
|
// Attempt the copy.
|
|
//
|
|
//
|
|
|
|
b = _SetupInstallFileEx(
|
|
Queue,
|
|
QueueNode,
|
|
NULL, // no inf handle
|
|
NULL, // no inf context
|
|
source,
|
|
NULL, // source path root is part of FullSourcePath
|
|
FullTargetName,
|
|
QueueNode->StyleFlags | SP_COPY_SOURCE_ABSOLUTE | CopyStyleFlags,
|
|
MsgHandler,
|
|
Context,
|
|
&InUse,
|
|
IsMsgHandlerNativeCharWidth,
|
|
&SignatureVerifyFailed
|
|
);
|
|
|
|
rc = b ? NO_ERROR : GetLastError();
|
|
|
|
#ifdef UNICODE
|
|
|
|
if(b || (rc == NO_ERROR)) {
|
|
if(!InUse && (QueueNode->SecurityDesc != -1)){
|
|
//
|
|
// Set security on the file
|
|
//
|
|
rc = pSetupCallSCE(ST_SCE_SET,
|
|
FullTargetName,
|
|
Queue,
|
|
NULL,
|
|
QueueNode->SecurityDesc,
|
|
NULL
|
|
);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if(rc == NO_ERROR) {
|
|
//
|
|
// File was copied or not copied, but it if was not copied
|
|
// the callback funtcion was already notified about why
|
|
// (version check failed, etc).
|
|
//
|
|
if(QueueNode->StyleFlags & SP_COPY_REPLACE_BOOT_FILE) {
|
|
//
|
|
// _SetupInstallFileEx is responsible for failing the copy
|
|
// when some yahoo comes and copies over a new file (and
|
|
// locks it) before we get a chance to.
|
|
//
|
|
MYASSERT(!InUse);
|
|
|
|
//
|
|
// If the file was copied, we need to set the wants-reboot
|
|
// flag. Otherwise, we need to put back the original file.
|
|
//
|
|
if(b) {
|
|
QueueNode->InternalFlags |= INUSE_INF_WANTS_REBOOT;
|
|
} else {
|
|
RestoreBootReplacedFile(Queue, QueueNode);
|
|
}
|
|
|
|
} else {
|
|
|
|
if(InUse) {
|
|
QueueNode->InternalFlags |= (QueueNode->StyleFlags & SP_COPY_IN_USE_NEEDS_REBOOT)
|
|
? INUSE_INF_WANTS_REBOOT
|
|
: INUSE_IN_USE;
|
|
}
|
|
}
|
|
|
|
} else {
|
|
DWORD LogTag = 0;
|
|
//
|
|
// File was not copied and a real error occurred.
|
|
// Notify the callback (unless the failure was due to a signature
|
|
// verification problem). Disallow skip if that is specified
|
|
// in the node's flags.
|
|
//
|
|
if(SignatureVerifyFailed) {
|
|
break;
|
|
} else {
|
|
LogTag = AllocLogInfoSlotOrLevel(Queue->LogContext,SETUP_LOG_INFO,FALSE);
|
|
|
|
FilePaths.Win32Error = rc;
|
|
FilePaths.Flags = QueueNode->StyleFlags & (SP_COPY_NOSKIP | SP_COPY_WARNIFSKIP | SP_COPY_NOBROWSE);
|
|
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
LogTag,
|
|
MSG_LOG_COPYERROR,
|
|
NULL,
|
|
FilePaths.Source,
|
|
FilePaths.Target,
|
|
FilePaths.Flags,
|
|
FilePaths.Win32Error
|
|
);
|
|
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_COPYERROR,
|
|
(UINT_PTR)&FilePaths,
|
|
(UINT_PTR)PathBuffer
|
|
);
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_ERROR|SETUP_LOG_BUFFER,
|
|
MSG_LOG_COPYERROR_ABORT,
|
|
NULL
|
|
);
|
|
WriteLogError(Queue->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
rc
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,LogTag);
|
|
LogTag = 0;
|
|
|
|
break;
|
|
} else {
|
|
if(u == FILEOP_SKIP) {
|
|
//
|
|
// If this file was a bootfile replacement, then we need
|
|
// to restore the original file that was renamed to a
|
|
// temporary filename.
|
|
//
|
|
if(QueueNode->StyleFlags & SP_COPY_REPLACE_BOOT_FILE) {
|
|
RestoreBootReplacedFile(Queue, QueueNode);
|
|
}
|
|
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_WARNING,
|
|
MSG_LOG_COPYERROR_SKIP,
|
|
NULL
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,LogTag);
|
|
LogTag = 0;
|
|
//
|
|
// Force termination of processing for this file.
|
|
//
|
|
rc = NO_ERROR;
|
|
break;
|
|
|
|
} else {
|
|
if((u == FILEOP_NEWPATH) || ((u == FILEOP_RETRY) && PathBuffer[0])) {
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_WARNING,
|
|
MSG_LOG_COPYERROR_NEWPATH,
|
|
NULL,
|
|
u,
|
|
PathBuffer
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,LogTag);
|
|
LogTag = 0;
|
|
|
|
//
|
|
// Note that rc is already set to something other than
|
|
// NO_ERROR or we wouldn't be here.
|
|
//
|
|
lstrcpyn(NewSourcePath,PathBuffer,MAX_PATH);
|
|
lstrcpyn(source,NewSourcePath,MAX_PATH);
|
|
pSetupConcatenatePaths(
|
|
source,
|
|
pSetupStringTableStringFromId(Queue->StringTable,QueueNode->SourceFilename),
|
|
MAX_PATH,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
//
|
|
// Else we don't have a new path.
|
|
// Just keep using the one we had.
|
|
//
|
|
}
|
|
}
|
|
if (LogTag != 0) {
|
|
//
|
|
// haven't done anything regards logging yet, do it now
|
|
//
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO,
|
|
MSG_LOG_COPYERROR_RETRY,
|
|
NULL,
|
|
u
|
|
);
|
|
ReleaseLogInfoSlot(Queue->LogContext,LogTag);
|
|
LogTag = 0;
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// skip file
|
|
//
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
SETUP_LOG_INFO, // info level as this would be due to override of callback
|
|
MSG_LOG_STARTCOPY_SKIP,
|
|
NULL,
|
|
u
|
|
);
|
|
rc = NO_ERROR;
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
|
|
//
|
|
// Notify the callback that the copy is done.
|
|
//
|
|
FilePaths.Win32Error = rc;
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDCOPY,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
|
|
MyFree(FullTargetName);
|
|
|
|
clean0:
|
|
|
|
return(rc);
|
|
}
|
|
|
|
|
|
DWORD
|
|
pSetupCopySingleQueuedFileCabCase(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN PCTSTR CabinetName,
|
|
IN PCTSTR FullSourceName,
|
|
IN PVOID MsgHandler,
|
|
IN PVOID Context,
|
|
IN BOOL IsMsgHandlerNativeCharWidth
|
|
)
|
|
{
|
|
PTSTR FullTargetName;
|
|
FILEPATHS FilePaths;
|
|
UINT u;
|
|
BOOL InUse;
|
|
TCHAR PathBuffer[MAX_PATH];
|
|
DWORD rc;
|
|
BOOL b;
|
|
BOOL BackupInUse = FALSE;
|
|
BOOL SignatureVerifyFailed;
|
|
DWORD LogTag = 0;
|
|
LPCTSTR SourceName;
|
|
|
|
//
|
|
// Form the full target path of the file.
|
|
//
|
|
SourceName = pSetupStringTableStringFromId(Queue->StringTable,QueueNode->SourceFilename);
|
|
FullTargetName = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
QueueNode->TargetDirectory,
|
|
QueueNode->TargetFilename,
|
|
-1
|
|
);
|
|
|
|
if(!FullTargetName) {
|
|
return(ERROR_NOT_ENOUGH_MEMORY);
|
|
}
|
|
|
|
|
|
LogTag = AllocLogInfoSlotOrLevel(Queue->LogContext,SETUP_LOG_INFO,FALSE);
|
|
WriteLogEntry(
|
|
Queue->LogContext,
|
|
LogTag,
|
|
MSG_LOG_COPY_FROM_CAB,
|
|
NULL,
|
|
CabinetName,
|
|
SourceName,
|
|
FullSourceName,
|
|
FullTargetName
|
|
);
|
|
|
|
//
|
|
// check if we need to backup before we copy
|
|
//
|
|
if((rc=pSetupDoLastKnownGoodBackup(Queue,
|
|
FullTargetName,
|
|
0,
|
|
NULL)) != NO_ERROR) {
|
|
MyFree(FullTargetName);
|
|
goto clean0;
|
|
}
|
|
rc = pSetupCommitSingleBackup(Queue,
|
|
FullTargetName,
|
|
QueueNode->TargetDirectory,
|
|
-1,
|
|
QueueNode->TargetFilename,
|
|
MsgHandler,
|
|
Context,
|
|
IsMsgHandlerNativeCharWidth,
|
|
(QueueNode->StyleFlags & SP_COPY_REPLACE_BOOT_FILE),
|
|
&BackupInUse
|
|
);
|
|
if (rc != NO_ERROR) {
|
|
MyFree(FullTargetName);
|
|
goto clean0;
|
|
}
|
|
|
|
if (BackupInUse) {
|
|
//
|
|
// if we couldn't do backup, force the IN_USE flag
|
|
//
|
|
QueueNode->StyleFlags |= SP_COPY_FORCE_IN_USE;
|
|
|
|
}
|
|
//
|
|
// We use the cabinet name as the source name so the display looks right
|
|
// to the user. Otherwise he sees the name of some temp file in the
|
|
// source field.
|
|
//
|
|
FilePaths.Source = CabinetName;
|
|
FilePaths.Target = FullTargetName;
|
|
FilePaths.Win32Error = NO_ERROR;
|
|
|
|
//
|
|
// Also, pass the callback routine the CopyStyle flags we're about to
|
|
// use.
|
|
//
|
|
// Callback flags are read-only.
|
|
//
|
|
FilePaths.Flags = QueueNode->StyleFlags;
|
|
|
|
do {
|
|
//
|
|
// Notify the callback that the copy is starting.
|
|
//
|
|
u = pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_STARTCOPY,
|
|
(UINT_PTR)&FilePaths,
|
|
FILEOP_COPY
|
|
);
|
|
|
|
if(u == FILEOP_ABORT) {
|
|
rc = GetLastError();
|
|
if(!rc) {
|
|
rc = ERROR_OPERATION_ABORTED;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if(u == FILEOP_DOIT) {
|
|
//
|
|
// Attempt the copy.
|
|
//
|
|
b = _SetupInstallFileEx(
|
|
Queue,
|
|
QueueNode,
|
|
NULL, // no inf handle
|
|
NULL, // no inf context
|
|
FullSourceName,
|
|
NULL, // source path root is part of FullSourcePath
|
|
FullTargetName,
|
|
QueueNode->StyleFlags | SP_COPY_SOURCE_ABSOLUTE,
|
|
MsgHandler,
|
|
Context,
|
|
&InUse,
|
|
IsMsgHandlerNativeCharWidth,
|
|
&SignatureVerifyFailed
|
|
);
|
|
|
|
#ifdef UNICODE
|
|
if(b || ((rc = GetLastError()) == NO_ERROR)) {
|
|
if(!InUse && (QueueNode->SecurityDesc != -1) ){
|
|
// Set security on the file
|
|
|
|
rc = pSetupCallSCE(
|
|
ST_SCE_SET,
|
|
FullTargetName,
|
|
Queue,
|
|
NULL,
|
|
QueueNode->SecurityDesc,
|
|
NULL
|
|
);
|
|
SetLastError( rc );
|
|
}
|
|
|
|
}
|
|
#endif
|
|
|
|
if(b || ((rc = GetLastError()) == NO_ERROR)) {
|
|
//
|
|
// File was copied or not copied, but it if was not copied
|
|
// the callback funtcion was already notified about why
|
|
// (version check failed, etc).
|
|
//
|
|
if(InUse) {
|
|
QueueNode->InternalFlags |= (QueueNode->StyleFlags & SP_COPY_IN_USE_NEEDS_REBOOT)
|
|
? INUSE_INF_WANTS_REBOOT
|
|
: INUSE_IN_USE;
|
|
}
|
|
rc = NO_ERROR;
|
|
} else {
|
|
//
|
|
// File was not copied and a real error occurred.
|
|
// Break out and return the error.
|
|
//
|
|
break;
|
|
}
|
|
} else {
|
|
//
|
|
// skip file
|
|
//
|
|
rc = NO_ERROR;
|
|
}
|
|
} while(rc != NO_ERROR);
|
|
|
|
//
|
|
// Notify the callback that the copy is done.
|
|
//
|
|
FilePaths.Win32Error = rc;
|
|
pSetupCallMsgHandler(
|
|
Queue->LogContext,
|
|
MsgHandler,
|
|
IsMsgHandlerNativeCharWidth,
|
|
Context,
|
|
SPFILENOTIFY_ENDCOPY,
|
|
(UINT_PTR)&FilePaths,
|
|
0
|
|
);
|
|
|
|
MyFree(FullTargetName);
|
|
|
|
clean0:
|
|
if(LogTag) {
|
|
ReleaseLogInfoSlot(Queue->LogContext,LogTag);
|
|
}
|
|
|
|
return(rc);
|
|
}
|
|
|
|
|
|
PTSTR
|
|
pSetupFormFullPath(
|
|
IN PVOID StringTable,
|
|
IN LONG PathPart1,
|
|
IN LONG PathPart2, OPTIONAL
|
|
IN LONG PathPart3 OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Form a full path based on components whose strings are in a string
|
|
table.
|
|
|
|
Arguments:
|
|
|
|
StringTable - supplies handle to string table.
|
|
|
|
PathPart1 - Supplies first part of path
|
|
|
|
PathPart2 - if specified, supplies second part of path
|
|
|
|
PathPart3 - if specified, supplies third part of path
|
|
|
|
Return Value:
|
|
|
|
Pointer to buffer containing full path. Caller can free with MyFree().
|
|
NULL if out of memory.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT RequiredSize;
|
|
PCTSTR p1,p2,p3;
|
|
TCHAR Buffer[MAX_PATH];
|
|
|
|
p1 = pSetupStringTableStringFromId(StringTable,PathPart1);
|
|
if (!p1) {
|
|
return NULL;
|
|
}
|
|
p2 = (PathPart2 == -1) ? NULL : pSetupStringTableStringFromId(StringTable,PathPart2);
|
|
p3 = (PathPart3 == -1) ? NULL : pSetupStringTableStringFromId(StringTable,PathPart3);
|
|
|
|
lstrcpy(Buffer,p1);
|
|
if(!p2 || pSetupConcatenatePaths(Buffer,p2,MAX_PATH,NULL)) {
|
|
if(p3) {
|
|
pSetupConcatenatePaths(Buffer,p3,MAX_PATH,NULL);
|
|
}
|
|
}
|
|
|
|
return(DuplicateString(Buffer));
|
|
}
|
|
|
|
|
|
DWORD
|
|
pSetupVerifyQueuedCatalogs(
|
|
IN HSPFILEQ FileQueue
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Silently verify all catalog nodes in the specified queue.
|
|
|
|
Arguments:
|
|
|
|
FileQueue - supplies a handle to the file queue containing catalog nodes
|
|
to be verified.
|
|
|
|
Return Value:
|
|
|
|
If all catalog nodes are valid, the return value is NO_ERROR. Otherwise,
|
|
it is a Win32 error code indicating the problem.
|
|
|
|
--*/
|
|
{
|
|
return _SetupVerifyQueuedCatalogs(NULL, // No UI, thus no HWND needed
|
|
(PSP_FILE_QUEUE)FileQueue,
|
|
VERCAT_NO_PROMPT_ON_ERROR,
|
|
NULL,
|
|
NULL
|
|
);
|
|
}
|
|
|
|
|
|
DWORD
|
|
_SetupVerifyQueuedCatalogs(
|
|
IN HWND Owner,
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN DWORD Flags,
|
|
OUT PTSTR DeviceInfFinalName, OPTIONAL
|
|
OUT PBOOL DeviceInfNewlyCopied OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine verifies catalogs and infs in a given queue by traversing
|
|
the catalog node list associated with the queue and operating on the
|
|
catalog/inf pair described by each one.
|
|
|
|
If any catalog/inf fails verification, the user is notified via a dialog,
|
|
depending on current policy.
|
|
|
|
** Behavior for native platform verification (w/o catalog override)
|
|
|
|
If an INF is from a system location, we assume that the catalog is
|
|
already installed on the system. Really there is no other option here,
|
|
since we would have no idea where to get the catalog in order to install it
|
|
even if we wanted to try. But the inf might have originally been an
|
|
oem inf which was copied and renamed by the Di stuff at device install
|
|
time. The catalog file knows nothing about the renamed file, so we must
|
|
track mappings from current inf filename to original inf filename.
|
|
|
|
In this case, we calculate the inf's hash value and then using that,
|
|
we ask the system for a catalog file that contains signing data
|
|
for that hash value. We then ask the system for info
|
|
about that catalog file. We keep repeating this process until we get
|
|
at the catalog we want (based on name). Finally we can call WinVerifyTrust
|
|
verify the catalog itself and the inf.
|
|
|
|
If an INF file is instead from an oem location, we copy the oem inf to a
|
|
unique name in the system inf directory (or create a zero-length placeholder
|
|
there, depending on whether or not the VERCAT_INSTALL_INF_AND_CAT flag is
|
|
set), and add the catalog using a filename based on that unique filename.
|
|
|
|
** Behavior for non-native platform verification (w/o catalog override) **
|
|
|
|
We will validate the catalogs and INFs using the alternate platform info
|
|
provided in the file queue. Otherwise, the logic is the same as in the
|
|
native case.
|
|
|
|
** Behavior for verification (w/catalog override) **
|
|
|
|
The actual verification will be done using native or non-native parameters
|
|
as discussed above, but INFs without a CatalogFile= entry will be validated
|
|
against the specified overriding catalog. This means that system INFs won't
|
|
get validated globally, and INF in OEM locations can be validated even if
|
|
they don't have a CatalogFile= entry. The overriding catalog file will be
|
|
installed under its current name, thus blowing away any existing catalog
|
|
having that name.
|
|
|
|
See the documentation on SetupSetFileQueueAlternatePlatform for more
|
|
details.
|
|
|
|
Arguments:
|
|
|
|
Owner - supplies window handle of window to own any ui. This HWND is stored
|
|
away in the queue for use later if any individual files fail verification.
|
|
|
|
Queue - supplies pointer to queue structure.
|
|
|
|
Flags - supplies flags that control behavior of this routine.
|
|
|
|
VERCAT_INSTALL_INF_AND_CAT - if this flag is set, any infs from
|
|
oem locations will be installed on the system, along with
|
|
their catalog files.
|
|
|
|
VERCAT_NO_PROMPT_ON_ERROR - if this flag is set, the user will _not_ be
|
|
notified about verification failures we encounter. If this flag is
|
|
set, then this was only a 'test', and no user prompting should take
|
|
place (nor should any PSS logging take place). If this flag is set,
|
|
then the VERCAT_INSTALL_INF_AND_CAT _should not_ be specified.
|
|
|
|
VERCAT_PRIMARY_DEVICE_INF_FROM_INET - specifies that the primary device
|
|
INF in the queue is from the internet, and should be marked as such
|
|
in the corresponding PNF when installed into the %windir%\Inf
|
|
directory via _SetupCopyOEMInf.
|
|
|
|
DeviceInfFinalName - optionally, supplies the address of a character buffer,
|
|
_at least_ MAX_PATH characters long, that upon success receives the
|
|
final name given to the INF under the %windir%\Inf directory (this will
|
|
be different than the INF's original name if it was an OEM INF).
|
|
|
|
DeviceInfNewlyCopied - optionally, supplies the address of a boolean
|
|
variable that, upon success, is set to indicate whether the INF name
|
|
returned in DeviceInfFinalName was newly-created. If this parameter is
|
|
supplied, then DeviceInfFinalName must also be specified.
|
|
|
|
Return Value:
|
|
|
|
If all catalogs/infs were verified and installed, or the user accepted
|
|
the risk if a verification failed, then the return value is NO_ERROR.
|
|
|
|
If one or more catalogs/infs were not verified, the return value is a Win32
|
|
error code indicating the cause of the failure. NOTE: This error will
|
|
only be returned if the policy is "block", or it it's "warn" and the
|
|
user decided to abort. In this case, the error returned is for the
|
|
catalog/INF where the error was encountered, and any subsequent catalog
|
|
nodes will not have been verified. An exception to this is when the
|
|
VERCAT_NO_PROMPT_ON_ERROR flag is set. In that case, we'll verify all
|
|
catalogs, even if we encounter improperly-signed ones.
|
|
|
|
Remarks:
|
|
|
|
There are some system INFs (for which global verification is required) that
|
|
don't live in %windir%\Inf. The OCM INFs are an example of this. Those
|
|
INFs use layout.inf (which _is_ located in %windir%\Inf) for the source
|
|
media information for any files they copy. There are other INFs that don't
|
|
live in %windir%\Inf which are extracted out of a binary as-needed (into a
|
|
temporary filename), processed in order to do registry munging, and then
|
|
deleted. Such INFs do not do file copying (thus their 'package' consists
|
|
of just the INF). To accommodate such INFs, we allow "OEM" INFs (i.e.,
|
|
those INFs not in %windir%\Inf) to be verified globally, but we remember the
|
|
fact that these INFs didn't contain a CatalogFile= entry, and if any files
|
|
are ever queued for copy using such INFs for source media information, then
|
|
we'll fail digital signature verification for such files, since there's no
|
|
way for us to know what catalog should be used for verification.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSPQ_CATALOG_INFO CatalogNode;
|
|
LPCTSTR InfFullPath;
|
|
LPCTSTR CatName;
|
|
TCHAR PathBuffer[MAX_PATH];
|
|
TCHAR InfNameBuffer[MAX_PATH];
|
|
TCHAR CatalogName[MAX_PATH];
|
|
TCHAR *p;
|
|
DWORD Err, CatalogNodeStatus, ReturnStatus;
|
|
SetupapiVerifyProblem Problem;
|
|
LPCTSTR ProblemFile;
|
|
BOOL DeleteOemInfOnError;
|
|
BOOL OriginalNameDifferent;
|
|
LPCTSTR AltCatalogFile;
|
|
LONG CatStringId;
|
|
ULONG RequiredSize;
|
|
DWORD InfVerifyType;
|
|
DWORD SCOIFlags;
|
|
|
|
//
|
|
// Define values used to indicate how validation should be done on the INFs.
|
|
//
|
|
#define VERIFY_INF_AS_OEM 0 // verify solely against the specific
|
|
// catalog referenced by the INF
|
|
|
|
#define VERIFY_INF_AS_SYSTEM 1 // verify globally (using all catalogs)
|
|
|
|
#define VERIFY_OEM_INF_GLOBALLY 2 // verify OEM INF globally, but remember the
|
|
// original error, in case copy operations
|
|
// are queued using media descriptor info
|
|
// within this INF.
|
|
|
|
|
|
MYASSERT((Flags & (VERCAT_INSTALL_INF_AND_CAT | VERCAT_NO_PROMPT_ON_ERROR))
|
|
!= (VERCAT_INSTALL_INF_AND_CAT | VERCAT_NO_PROMPT_ON_ERROR)
|
|
);
|
|
|
|
MYASSERT(!DeviceInfNewlyCopied || DeviceInfFinalName);
|
|
|
|
if(Queue->Flags & FQF_DID_CATALOGS_OK) {
|
|
//
|
|
// If the caller wants information about the primary device INF, then
|
|
// find the applicable catalog node.
|
|
//
|
|
if(DeviceInfFinalName) {
|
|
for(CatalogNode=Queue->CatalogList; CatalogNode; CatalogNode=CatalogNode->Next) {
|
|
|
|
if(CatalogNode->Flags & CATINFO_FLAG_PRIMARY_DEVICE_INF) {
|
|
MYASSERT(CatalogNode->InfFinalPath != -1);
|
|
InfFullPath = pSetupStringTableStringFromId(Queue->StringTable, CatalogNode->InfFinalPath);
|
|
lstrcpy(DeviceInfFinalName, InfFullPath);
|
|
if(DeviceInfNewlyCopied) {
|
|
*DeviceInfNewlyCopied = (CatalogNode->Flags & CATINFO_FLAG_NEWLY_COPIED);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return NO_ERROR;
|
|
}
|
|
|
|
if(Queue->Flags & FQF_DID_CATALOGS_FAILED) {
|
|
//
|
|
// Scan the catalog nodes until we find the first one that failed
|
|
// verification, and return that failure code.
|
|
//
|
|
for(CatalogNode=Queue->CatalogList; CatalogNode; CatalogNode=CatalogNode->Next) {
|
|
|
|
if(CatalogNode->VerificationFailureError != NO_ERROR) {
|
|
return CatalogNode->VerificationFailureError;
|
|
}
|
|
}
|
|
|
|
//
|
|
// We didn't find a failed catalog node in our catalog list--something's
|
|
// seriously wrong!
|
|
//
|
|
MYASSERT(0);
|
|
return ERROR_INVALID_DATA;
|
|
}
|
|
|
|
//
|
|
// If the queue has an alternate default catalog file associated with it,
|
|
// then retrieve that catalog's name for use later.
|
|
//
|
|
AltCatalogFile = (Queue->AltCatalogFile != -1)
|
|
? pSetupStringTableStringFromId(Queue->StringTable, Queue->AltCatalogFile)
|
|
: NULL;
|
|
|
|
Queue->hWndDriverSigningUi = Owner;
|
|
ReturnStatus = NO_ERROR;
|
|
|
|
for(CatalogNode=Queue->CatalogList; CatalogNode; CatalogNode=CatalogNode->Next) {
|
|
//
|
|
// Assume success for verification of this catalog node.
|
|
//
|
|
CatalogNodeStatus = NO_ERROR;
|
|
|
|
MYASSERT(CatalogNode->InfFullPath != -1);
|
|
InfFullPath = pStringTableStringFromId(Queue->StringTable, CatalogNode->InfFullPath);
|
|
|
|
if(Queue->Flags & FQF_USE_ALT_PLATFORM) {
|
|
//
|
|
// We have an alternate platform override, so use the alternate
|
|
// platform's CatalogFile= entry.
|
|
//
|
|
CatStringId = CatalogNode->AltCatalogFileFromInf;
|
|
} else {
|
|
//
|
|
// We're running native--use the native CatalogFile= entry.
|
|
//
|
|
CatStringId = CatalogNode->CatalogFileFromInf;
|
|
}
|
|
CatName = (CatStringId != -1)
|
|
? pStringTableStringFromId(Queue->StringTable, CatStringId)
|
|
: NULL;
|
|
|
|
InfVerifyType = pSetupInfIsFromOemLocation(InfFullPath, TRUE)
|
|
? VERIFY_INF_AS_OEM
|
|
: VERIFY_INF_AS_SYSTEM;
|
|
|
|
if(InfVerifyType == VERIFY_INF_AS_OEM) {
|
|
//
|
|
// If the caller wants us to, we'll now install the catalog. In
|
|
// addition, if it's a (native platform) device installation, we'll
|
|
// install the INF as well.
|
|
//
|
|
// (Note: we specify the 'no overwrite' switch so that we won't blow
|
|
// away any existing PNF source path information for this INF.
|
|
// We'll only consider an OEM INF to match up with an existing
|
|
// %windir%\Inf\Oem*.INF entry if the catalogs also match up, so
|
|
// we're not going to get into any trouble doing this.
|
|
//
|
|
if(Flags & VERCAT_INSTALL_INF_AND_CAT) {
|
|
//
|
|
// If we're not doing a device install, then we want to suppress
|
|
// popups and error log entries if the INF doesn't reference a
|
|
// catalog. This is because we want to allow such INFs to be
|
|
// validated globally, unless they subsequently try to copy files.
|
|
//
|
|
SCOIFlags = (Queue->Flags & FQF_DEVICE_INSTALL)
|
|
? 0
|
|
: SCOI_NO_ERRLOG_ON_MISSING_CATALOG;
|
|
|
|
//
|
|
// If we're not supposed to generate popups/log entries at all
|
|
// for signature verification failures (e.g., because we've
|
|
// already done so previously), then set that flag as well.
|
|
//
|
|
if(Queue->Flags & FQF_DIGSIG_ERRORS_NOUI) {
|
|
SCOIFlags |= SCOI_NO_UI_ON_SIGFAIL;
|
|
}
|
|
|
|
if(Queue->Flags & FQF_KEEP_INF_AND_CAT_ORIGINAL_NAMES) {
|
|
SCOIFlags |= SCOI_KEEP_INF_AND_CAT_ORIGINAL_NAMES;
|
|
}
|
|
|
|
if(Queue->Flags & FQF_ABORT_IF_UNSIGNED) {
|
|
SCOIFlags |= SCOI_ABORT_IF_UNSIGNED;
|
|
}
|
|
|
|
if(_SetupCopyOEMInf(InfFullPath,
|
|
NULL, // default source location to where INF presently is
|
|
((Flags & VERCAT_PRIMARY_DEVICE_INF_FROM_INET)
|
|
? SPOST_URL
|
|
: SPOST_PATH),
|
|
(((Queue->Flags & (FQF_DEVICE_INSTALL | FQF_USE_ALT_PLATFORM)) == FQF_DEVICE_INSTALL)
|
|
? SP_COPY_NOOVERWRITE
|
|
: SP_COPY_NOOVERWRITE | SP_COPY_OEMINF_CATALOG_ONLY),
|
|
PathBuffer,
|
|
SIZECHARS(PathBuffer),
|
|
NULL,
|
|
&p,
|
|
((CatalogNode->InfOriginalName != -1)
|
|
? pStringTableStringFromId(Queue->StringTable,
|
|
CatalogNode->InfOriginalName)
|
|
: pSetupGetFileTitle(InfFullPath)),
|
|
CatName,
|
|
Owner,
|
|
((Queue->DeviceDescStringId == -1)
|
|
? NULL
|
|
: pStringTableStringFromId(Queue->StringTable,
|
|
Queue->DeviceDescStringId)),
|
|
Queue->DriverSigningPolicy,
|
|
SCOIFlags,
|
|
AltCatalogFile,
|
|
((Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: Queue->ValidationPlatform),
|
|
&Err,
|
|
CatalogNode->CatalogFilenameOnSystem,
|
|
Queue->LogContext,
|
|
&(Queue->hCatAdmin))) {
|
|
//
|
|
// If Err indicates that there was a digital signature
|
|
// problem that the user chose to ignore (or was silently
|
|
// ignored), then set a flag in the queue indicating the
|
|
// user should not be warned about subsequent failures.
|
|
// Don't set this flag if the queue's policy is "Ignore",
|
|
// however, on the chance that the policy might be altered
|
|
// later, and we'd want the user to get informed on any
|
|
// subsequent errors.
|
|
//
|
|
// (Note: if the error was due to the INF not having a
|
|
// CatalogFile= entry, and if we're supposed to ignore such
|
|
// problems, then just set the flag to do global validation
|
|
// later.)
|
|
//
|
|
if((Err == ERROR_NO_CATALOG_FOR_OEM_INF) &&
|
|
(SCOIFlags & SCOI_NO_ERRLOG_ON_MISSING_CATALOG)) {
|
|
|
|
InfVerifyType = VERIFY_OEM_INF_GLOBALLY;
|
|
|
|
} else if((Err != NO_ERROR) && (Queue->DriverSigningPolicy != DRIVERSIGN_NONE)) {
|
|
|
|
Queue->Flags |= FQF_DIGSIG_ERRORS_NOUI;
|
|
}
|
|
|
|
if(*PathBuffer) {
|
|
//
|
|
// Store the INF's final path into our catalog node.
|
|
// This will be under %windir%\Inf unless the INF didn't
|
|
// specify a CatalogFile= entry and we did an alternate
|
|
// catalog installation (i.e., because the file queue had
|
|
// an associated alternate catalog).
|
|
//
|
|
CatalogNode->InfFinalPath = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
PathBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
} else {
|
|
//
|
|
// _SetupCopyOEMInf returned an empty string for the
|
|
// destination INF name, which means that we were doing
|
|
// a catalog-only install, and it didn't find the INF
|
|
// already existing in %windir%\Inf. In this case, just
|
|
// use the INF's original pathname as its final pathname.
|
|
//
|
|
CatalogNode->InfFinalPath = CatalogNode->InfFullPath;
|
|
}
|
|
|
|
if(CatalogNode->InfFinalPath == -1) {
|
|
|
|
CatalogNodeStatus = ERROR_NOT_ENOUGH_MEMORY;
|
|
if(Err == NO_ERROR) {
|
|
Err = CatalogNodeStatus;
|
|
}
|
|
|
|
//
|
|
// Since we couldn't add this filename to the string
|
|
// table, we won't be able to undo this copy later--it
|
|
// must be done here. Delete the INF, PNF, and CAT.
|
|
//
|
|
// NOTE: we should never get here if we did an alternate
|
|
// catalog file-only install, because in that case our
|
|
// new INF name is the same as the INF's original name,
|
|
// thus the string is already in the buffer and there's
|
|
// no way we could run out of memory.
|
|
//
|
|
MYASSERT(lstrcmpi(PathBuffer, InfFullPath));
|
|
|
|
pSetupUninstallOEMInf(PathBuffer,
|
|
Queue->LogContext,
|
|
SUOI_FORCEDELETE,
|
|
NULL
|
|
);
|
|
|
|
} else {
|
|
//
|
|
// Set a flag in the catalog node indicating that this
|
|
// INF was newly-copied into %windir%\Inf. If the
|
|
// string ID for our INF's original name and that of its
|
|
// new name are equal, then we know we did an alternate
|
|
// catalog installation only, and we don't want to set
|
|
// this flag.
|
|
//
|
|
if(CatalogNode->InfFinalPath != CatalogNode->InfFullPath) {
|
|
CatalogNode->Flags |= CATINFO_FLAG_NEWLY_COPIED;
|
|
}
|
|
|
|
//
|
|
// If this is the primary device INF, and the caller
|
|
// requested information about that INF's final
|
|
// pathname, then store that information in the caller-
|
|
// supplied buffer(s) now.
|
|
//
|
|
if(DeviceInfFinalName &&
|
|
(CatalogNode->Flags & CATINFO_FLAG_PRIMARY_DEVICE_INF)) {
|
|
//
|
|
// We'd better not just've done an alternate catalog
|
|
// installation.
|
|
//
|
|
MYASSERT(CatalogNode->InfFinalPath != CatalogNode->InfFullPath);
|
|
|
|
lstrcpy(DeviceInfFinalName, PathBuffer);
|
|
if(DeviceInfNewlyCopied) {
|
|
*DeviceInfNewlyCopied = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
} else {
|
|
|
|
CatalogNodeStatus = GetLastError();
|
|
MYASSERT(CatalogNodeStatus != NO_ERROR);
|
|
|
|
if(CatalogNodeStatus == ERROR_FILE_EXISTS) {
|
|
//
|
|
// INF and CAT already there--this isn't a failure.
|
|
//
|
|
// Store the name under which we found this OEM INF into
|
|
// the catalog node's InfFinalPath field.
|
|
//
|
|
CatalogNode->InfFinalPath = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
PathBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
|
|
if(CatalogNode->InfFinalPath == -1) {
|
|
CatalogNodeStatus = ERROR_NOT_ENOUGH_MEMORY;
|
|
} else {
|
|
CatalogNodeStatus = NO_ERROR;
|
|
//
|
|
// If Err indicates that there was a digital signature
|
|
// problem that the user chose to ignore (or was silently
|
|
// ignored), then set a flag in the queue indicating the
|
|
// user should not be warned about subsequent failures.
|
|
// Don't set this flag if the queue's policy is "Ignore",
|
|
// however, on the chance that the policy might be altered
|
|
// later, and we'd want the user to get informed on any
|
|
// subsequent errors.
|
|
//
|
|
if((Err != NO_ERROR) && Queue->DriverSigningPolicy != DRIVERSIGN_NONE) {
|
|
Queue->Flags |= FQF_DIGSIG_ERRORS_NOUI;
|
|
}
|
|
|
|
//
|
|
// If this is the primary device INF, and the caller
|
|
// requested information about that INF's final
|
|
// pathname, then store that information in the caller-
|
|
// supplied buffer(s) now.
|
|
//
|
|
if(DeviceInfFinalName &&
|
|
(CatalogNode->Flags & CATINFO_FLAG_PRIMARY_DEVICE_INF)) {
|
|
|
|
lstrcpy(DeviceInfFinalName, PathBuffer);
|
|
if(DeviceInfNewlyCopied) {
|
|
*DeviceInfNewlyCopied = FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
} else if(CatalogNodeStatus == ERROR_SET_SYSTEM_RESTORE_POINT) {
|
|
//
|
|
// We should only get this error if the queue flag is
|
|
// set that causes us to abort unsigned installations.
|
|
//
|
|
MYASSERT(Queue->Flags & FQF_ABORT_IF_UNSIGNED);
|
|
|
|
//
|
|
// We don't want the user to see the driver signing
|
|
// UI again when the queue is re-committed...
|
|
//
|
|
if(Queue->DriverSigningPolicy != DRIVERSIGN_NONE) {
|
|
Queue->Flags |= FQF_DIGSIG_ERRORS_NOUI;
|
|
}
|
|
|
|
//
|
|
// Make sure that Err is also set to this same
|
|
// "special" error code...
|
|
//
|
|
Err = CatalogNodeStatus;
|
|
}
|
|
|
|
//
|
|
// If we had a real failure from _SetupCopyOEMInf (or we're
|
|
// out of memory and couldn't add a string to the string
|
|
// table above), then we need to propagate the value of
|
|
// CatalogNodeStatus to Err, if Err doesn't already have a
|
|
// failure code.
|
|
//
|
|
if((CatalogNodeStatus != NO_ERROR) && (Err == NO_ERROR)) {
|
|
Err = CatalogNodeStatus;
|
|
}
|
|
}
|
|
} else {
|
|
//
|
|
// We were told not to copy any files, but we've encountered an
|
|
// OEM INF that needs to be installed. Hence, we have a failure.
|
|
// Note that we _don't_ look to see if this OEM INF (and its
|
|
// corresponding catalog) might happen to already be properly
|
|
// installed. That isn't necessary, because
|
|
// _SetupDiInstallDevice calls _SetupVerifyQueuedCatalogs with
|
|
// the VERCAT_INSTALL_INF_AND_CAT flag _before_ calling
|
|
// SetupScanFileQueue, thus all INFs/CATs should be present when
|
|
// we're called to do simple verification of the catalog nodes.
|
|
//
|
|
Err = CatalogNodeStatus = ERROR_CANNOT_COPY;
|
|
}
|
|
|
|
}
|
|
|
|
if(InfVerifyType != VERIFY_INF_AS_OEM) {
|
|
//
|
|
// Inf is in system location (%windir%\Inf), or we're going to try
|
|
// validating an "OEM" INF globally. Figure out the expected name
|
|
// of the catalog file. If the file was originally copied in by the
|
|
// Di stuff, then we need to use a name based on the name Di gave
|
|
// the inf. Otherwise we use the name from the inf's CatalogFile=
|
|
// entry, if present. Finally, if the INF doesn't specify a
|
|
// CatalogFile= entry, we assume it's a system component and
|
|
// attempt to validate against any catalog that we find a hash
|
|
// match in.
|
|
//
|
|
Err = NO_ERROR; // assume success
|
|
|
|
if(CatalogNode->InfOriginalName != -1) {
|
|
|
|
RequiredSize = SIZECHARS(InfNameBuffer);
|
|
if(pSetupStringTableStringFromIdEx(Queue->StringTable,
|
|
CatalogNode->InfOriginalName,
|
|
InfNameBuffer,
|
|
&RequiredSize)) {
|
|
|
|
OriginalNameDifferent = TRUE;
|
|
} else {
|
|
//
|
|
// This should never fail!
|
|
//
|
|
MYASSERT(0);
|
|
Err = ERROR_INVALID_DATA;
|
|
}
|
|
|
|
} else {
|
|
OriginalNameDifferent = FALSE;
|
|
}
|
|
|
|
if(Err == NO_ERROR) {
|
|
|
|
if(CatName) {
|
|
//
|
|
// If there is a catalog name, then we'd better not be
|
|
// doing our "verify OEM INF globally" trick!
|
|
//
|
|
MYASSERT(InfVerifyType == VERIFY_INF_AS_SYSTEM);
|
|
|
|
if(OriginalNameDifferent) {
|
|
//
|
|
// If the INF specified a catalog file, then we know we
|
|
// would've installed that catalog file using a name based
|
|
// on the unique name we assigned the INF when copying it
|
|
// into the INF directory.
|
|
//
|
|
lstrcpy(CatalogName, pSetupGetFileTitle(InfFullPath));
|
|
p = _tcsrchr(CatalogName, TEXT('.'));
|
|
if(!p) {
|
|
p = CatalogName + lstrlen(CatalogName);
|
|
}
|
|
lstrcpy(p, pszCatSuffix);
|
|
} else {
|
|
lstrcpy(CatalogName, CatName);
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// This system INF didn't specify a CatalogFile= entry. If
|
|
// an alternate catalog is associated with this file queue,
|
|
// then use that catalog for verification.
|
|
//
|
|
if(AltCatalogFile) {
|
|
lstrcpy(CatalogName, AltCatalogFile);
|
|
CatName = pSetupGetFileTitle(CatalogName);
|
|
}
|
|
}
|
|
|
|
//
|
|
// (Note: in the call below, we don't want to store the
|
|
// validating catalog filename in our CatalogFilenameOnSystem
|
|
// field if the INF didn't specify a CatalogFile= entry (and
|
|
// there was no alternate catalog specified), because we want
|
|
// any queue nodes that reference this catalog entry to use
|
|
// global validation as well.)
|
|
//
|
|
if(GlobalSetupFlags & PSPGF_MINIMAL_EMBEDDED) {
|
|
//
|
|
// Don't attempt to call _VerifyFile, because we're
|
|
// asking for the validating catalog's name, and that makes
|
|
// no sense in the "minimal embedded" case.
|
|
//
|
|
*(CatalogNode->CatalogFilenameOnSystem) = TEXT('\0');
|
|
|
|
//
|
|
// (Err is already set to NO_ERROR.)
|
|
//
|
|
|
|
} else {
|
|
|
|
if(!CatName) {
|
|
*(CatalogNode->CatalogFilenameOnSystem) = TEXT('\0');
|
|
}
|
|
|
|
Err = _VerifyFile(
|
|
Queue->LogContext,
|
|
&(Queue->hCatAdmin),
|
|
NULL,
|
|
(CatName ? CatalogName : NULL),
|
|
NULL,
|
|
0,
|
|
(OriginalNameDifferent ? InfNameBuffer : pSetupGetFileTitle(InfFullPath)),
|
|
InfFullPath,
|
|
&Problem,
|
|
PathBuffer,
|
|
FALSE,
|
|
((Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: Queue->ValidationPlatform),
|
|
(VERIFY_FILE_IGNORE_SELFSIGNED
|
|
| VERIFY_FILE_USE_OEM_CATALOGS
|
|
| VERIFY_FILE_NO_DRIVERBLOCKED_CHECK),
|
|
(CatName ? CatalogNode->CatalogFilenameOnSystem : NULL),
|
|
NULL,
|
|
NULL,
|
|
NULL
|
|
);
|
|
}
|
|
}
|
|
|
|
if(Err == NO_ERROR) {
|
|
//
|
|
// INF/CAT was successfully verified--store the INF's final path
|
|
// (which is the same as its current path) into the catalog
|
|
// node.
|
|
//
|
|
CatalogNode->InfFinalPath = CatalogNode->InfFullPath;
|
|
|
|
} else {
|
|
|
|
if(Problem != SetupapiVerifyCatalogProblem) {
|
|
|
|
MYASSERT(Problem != SetupapiVerifyNoProblem);
|
|
//
|
|
// If the problem was not a catalog problem, then it's an
|
|
// INF problem (the _VerifyFile routine doesn't know the
|
|
// file we passed it is an INF).
|
|
//
|
|
Problem = SetupapiVerifyInfProblem;
|
|
}
|
|
ProblemFile = PathBuffer;
|
|
|
|
if((Flags & VERCAT_NO_PROMPT_ON_ERROR)
|
|
|| (Queue->Flags & FQF_QUEUE_FORCE_BLOCK_POLICY)) {
|
|
//
|
|
// Don't notify the caller or log anything--just remember
|
|
// the error.
|
|
//
|
|
CatalogNodeStatus = Err;
|
|
|
|
} else {
|
|
//
|
|
// Notify the caller of the failure (based on policy).
|
|
//
|
|
if(pSetupHandleFailedVerification(
|
|
Owner,
|
|
Problem,
|
|
ProblemFile,
|
|
((Queue->DeviceDescStringId == -1)
|
|
? NULL
|
|
: pStringTableStringFromId(Queue->StringTable, Queue->DeviceDescStringId)),
|
|
Queue->DriverSigningPolicy,
|
|
Queue->Flags & FQF_DIGSIG_ERRORS_NOUI,
|
|
Err,
|
|
Queue->LogContext,
|
|
NULL,
|
|
NULL))
|
|
{
|
|
//
|
|
// If the caller wants a chance to set a system restore
|
|
// point prior to doing any unsigned installations,
|
|
// then we abort now with a "special" error code that
|
|
// tells them what to do...
|
|
//
|
|
if(Queue->Flags & FQF_ABORT_IF_UNSIGNED) {
|
|
//
|
|
// We don't want the user to see the driver signing
|
|
// UI again when the queue is re-committed...
|
|
//
|
|
if(Queue->DriverSigningPolicy != DRIVERSIGN_NONE) {
|
|
Queue->Flags |= FQF_DIGSIG_ERRORS_NOUI;
|
|
}
|
|
|
|
CatalogNodeStatus = Err = ERROR_SET_SYSTEM_RESTORE_POINT;
|
|
|
|
} else {
|
|
//
|
|
// Set a flag in the queue that indicates the user has been
|
|
// informed of a signature problem with this queue, and has
|
|
// elected to go ahead and install anyway. Don't set this
|
|
// flag if the queue's policy is "Ignore", on the chance
|
|
// that the policy might be altered later, and we'd want the
|
|
// user to get informed on any subsequent errors.
|
|
//
|
|
if(Queue->DriverSigningPolicy != DRIVERSIGN_NONE) {
|
|
Queue->Flags |= FQF_DIGSIG_ERRORS_NOUI;
|
|
}
|
|
|
|
//
|
|
// Since we're going to use the INF/CAT anyway, in spite of
|
|
// digital signature problems, then we need to set the INF's
|
|
// final path to be the same as its current path.
|
|
//
|
|
CatalogNode->InfFinalPath = CatalogNode->InfFullPath;
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// The caller doesn't want to proceed.
|
|
//
|
|
CatalogNodeStatus = Err;
|
|
}
|
|
}
|
|
}
|
|
|
|
if(CatalogNodeStatus == NO_ERROR) {
|
|
//
|
|
// If this is the primary device INF, and the caller requested
|
|
// information about that INF's final pathname, then store that
|
|
// information in the caller-supplied buffer(s) now.
|
|
//
|
|
if(DeviceInfFinalName &&
|
|
(CatalogNode->Flags & CATINFO_FLAG_PRIMARY_DEVICE_INF)) {
|
|
|
|
lstrcpy(DeviceInfFinalName, InfFullPath);
|
|
if(DeviceInfNewlyCopied) {
|
|
*DeviceInfNewlyCopied = FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if(Err == NO_ERROR) {
|
|
//
|
|
// If we successfully validated an "OEM" INF globally, then we want
|
|
// to remember this fact. This will allow us to generate a
|
|
// signature verification failure against any file copy nodes
|
|
// associated with this catalog node.
|
|
//
|
|
if(InfVerifyType == VERIFY_OEM_INF_GLOBALLY) {
|
|
CatalogNode->VerificationFailureError = ERROR_NO_CATALOG_FOR_OEM_INF;
|
|
} else {
|
|
CatalogNode->VerificationFailureError = NO_ERROR;
|
|
}
|
|
|
|
} else {
|
|
CatalogNode->VerificationFailureError = Err;
|
|
CatalogNode->CatalogFilenameOnSystem[0] = TEXT('\0');
|
|
}
|
|
|
|
if((ReturnStatus == NO_ERROR) && (CatalogNodeStatus != NO_ERROR)) {
|
|
//
|
|
// First critical error we've encountered--propagate the failure
|
|
// for this catalog to our return status that will be returned to
|
|
// the caller once we've finished looking at all the catalogs.
|
|
//
|
|
ReturnStatus = CatalogNodeStatus;
|
|
|
|
//
|
|
// Unless the VERCAT_NO_PROMPT_ON_ERROR flag has been set, we
|
|
// should abort right now--there's no since in going any further.
|
|
//
|
|
if(!(Flags & VERCAT_NO_PROMPT_ON_ERROR)) {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// If the caller requested no prompting, then we don't want to mark this
|
|
// queue as 'failed', since the user never heard about it. However, if the
|
|
// verification succeeded, then we _do_ want to mark it as successful.
|
|
//
|
|
if(Flags & VERCAT_NO_PROMPT_ON_ERROR) {
|
|
|
|
if(ReturnStatus == NO_ERROR) {
|
|
Queue->Flags |= FQF_DID_CATALOGS_OK;
|
|
}
|
|
|
|
} else {
|
|
|
|
Queue->Flags |= (ReturnStatus == NO_ERROR) ? FQF_DID_CATALOGS_OK
|
|
: FQF_DID_CATALOGS_FAILED;
|
|
}
|
|
|
|
return ReturnStatus;
|
|
}
|
|
|
|
VOID
|
|
LogFailedVerification(
|
|
IN PSETUP_LOG_CONTEXT LogContext, OPTIONAL
|
|
IN DWORD MessageId,
|
|
IN DWORD Error,
|
|
IN LPCTSTR ProblemFile,
|
|
IN LPCTSTR DeviceDesc OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine logs when a verification failed but the file was installed
|
|
anyway.
|
|
|
|
Arguments:
|
|
|
|
LogContext - optionally supplies a pointer to the context for logging.
|
|
If this is not supplied, errors will be logged to the default context.
|
|
|
|
MessageId - Message to display
|
|
|
|
Error - supplies the code the the error that caused the failure.
|
|
|
|
ProblemFile - supplies the file path to the file associated with
|
|
the problem. In some cases this is a full path, in others it's just a
|
|
filename. The caller decides which makes sense in a particular
|
|
scenario. For example, a system catalog is in some funky directory
|
|
and there is no need to tell the user the full path. But in the case
|
|
where a catalog comes from an oem location, there might be some benefit
|
|
to telling the user the full path.
|
|
|
|
DeviceDesc - Optionally, supplies the device description to be used in the
|
|
digital signature verification error dialogs that may be popped up.
|
|
|
|
Return Value:
|
|
|
|
NONE.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSETUP_LOG_CONTEXT lc = NULL;
|
|
|
|
MYASSERT(Error != NO_ERROR);
|
|
MYASSERT(ProblemFile != NULL);
|
|
|
|
if (!LogContext) {
|
|
if (CreateLogContext(NULL, TRUE, &lc) == NO_ERROR) {
|
|
//
|
|
// success
|
|
//
|
|
LogContext = lc;
|
|
} else {
|
|
lc = NULL;
|
|
}
|
|
}
|
|
|
|
//
|
|
// a device install failed
|
|
//
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_ERROR | SETUP_LOG_BUFFER,
|
|
MessageId,
|
|
NULL,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
|
|
WriteLogError(
|
|
LogContext,
|
|
SETUP_LOG_ERROR,
|
|
Error);
|
|
|
|
if (lc) {
|
|
DeleteLogContext(lc);
|
|
}
|
|
}
|
|
|
|
BOOL
|
|
pSetupHandleFailedVerification(
|
|
IN HWND Owner,
|
|
IN SetupapiVerifyProblem Problem,
|
|
IN LPCTSTR ProblemFile,
|
|
IN LPCTSTR DeviceDesc, OPTIONAL
|
|
IN DWORD DriverSigningPolicy,
|
|
IN BOOL NoUI,
|
|
IN DWORD Error,
|
|
IN PVOID LogContext, OPTIONAL
|
|
OUT PDWORD Flags, OPTIONAL
|
|
IN LPCTSTR TargetFile OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine deals with a failed verification.
|
|
|
|
System policy is checked. If the policy is block, UI is displayed telling
|
|
the user that they're hosed. If the policy is ask-user, then ui is
|
|
displayed requesting the user's decision about whether to ignore the
|
|
verification failure and take the risk. If the policy is ignore, nothing
|
|
is done.
|
|
|
|
Arguments:
|
|
|
|
Owner - supplies window to own the dialog.
|
|
|
|
Problem - supplies a constant indicating what caused the failure. There are
|
|
5 cases:
|
|
|
|
Catalog, meaning that the catalog could not be verified
|
|
|
|
CatalogInstall, some other problem occured (e.g., INF directory write-
|
|
protected, thus we couldn't install placeholder INF)
|
|
|
|
Inf, meaning that an inf could not be verified or installed, etc.
|
|
|
|
File, meaning that some random other file failed verification.
|
|
|
|
DriverBlocked, meaning the driver was in the bad driver database.
|
|
|
|
ProblemFile - supplies the file path to the file associated with
|
|
the problem. In some cases this is a full path, in others it's just a
|
|
filename. The caller decides which makes sense in a particular
|
|
scenario. For example, a system catalog is in some funky directory
|
|
and there is no need to tell the user the full path. But in the case
|
|
where a catalog comes from an oem location, there might be some benefit
|
|
to telling the user the full path.
|
|
NOTE: if this API is being called because of a blocked driver then a
|
|
full path should always be passed in.
|
|
|
|
DeviceDesc - Optionally, supplies the device description to be used in the
|
|
digital signature verification error dialogs that may be popped up.
|
|
|
|
DriverSigningPolicy - supplies the driver signing policy currently in
|
|
effect. May be one of the three following values:
|
|
|
|
DRIVERSIGN_NONE - silently succeed installation of unsigned/
|
|
incorrectly-signed files. A PSS log entry will
|
|
be generated, however.
|
|
DRIVERSIGN_WARNING - warn the user, but let them choose whether or not
|
|
they still want to install the problematic file.
|
|
If the user elects to proceed with the
|
|
installation, A PSS log entry will be generated
|
|
noting this fact.
|
|
DRIVERSIGN_BLOCKING - do not allow the file to be installed
|
|
|
|
NoUI - if TRUE, then a dialog box should not be displayed to the user, even
|
|
if policy is warn or block. This will typically be set to TRUE after
|
|
the user has previously been informed of a digital signature problem
|
|
with the package they're attempting to install, but have elected to
|
|
proceed with the installation anyway. The behavior of the "Yes" button,
|
|
then, is really a "yes to all".
|
|
|
|
Error - supplies the code of the error that caused the failure.
|
|
|
|
LogContext - optionally supplies a pointer to the context for logging.
|
|
If this is not supplied, errors will be logged to the default context.
|
|
This is declared as a PVOID so external functions don't need to know
|
|
what a SETUP_LOG_CONTEXT is.
|
|
|
|
Flags - optionally supplies a pointer to a DWORD that receives one or more
|
|
of the following file queue node flags indicating that we made an
|
|
exemption for installing a protected system file:
|
|
|
|
IQF_TARGET_PROTECTED - TargetFile (see below) is a protected system
|
|
file.
|
|
IQF_ALLOW_UNSIGNED - An exception has been granted so that TargetFile
|
|
(see below) may be replaced by an unsigned file.
|
|
|
|
TargetFile - optionally supplies a pointer to a string that specifies a
|
|
destination file if one exists. This is only used if we want to exempt
|
|
a file operation on this file. If this parameter is not specified, then
|
|
it is assumed the file will _not_ be replaced (i.e., it may already be
|
|
on the system in its unsigned state), and no SFP exemption will be
|
|
attempted.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating whether the caller should continue.
|
|
If FALSE, then the current operation should be aborted, as the combination
|
|
of system policy and user input indicated that the risk should not
|
|
be taken.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL b;
|
|
INT_PTR iRes;
|
|
CERT_PROMPT CertPrompt;
|
|
HANDLE hDialogEvent = NULL;
|
|
|
|
//
|
|
// If we're running non-interactive, then we always silently block,
|
|
// regardless of policy.
|
|
//
|
|
if(GlobalSetupFlags & PSPGF_NONINTERACTIVE) {
|
|
//
|
|
// SPLOG -- log a PSS entry recording this event.
|
|
//
|
|
if (Problem == SetupapiVerifyDriverBlocked) {
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_BLOCKED_FOR_DEVICE_ERROR_NONINTERACTIVE : MSG_LOG_DRIVER_BLOCKED_ERROR_NONINTERACTIVE,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
} else {
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_NONINTERACTIVE : MSG_LOG_SIGNING_ERROR_NONINTERACTIVE,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
if (GuiSetupInProgress) {
|
|
hDialogEvent = CreateEvent(NULL,TRUE,FALSE,SETUP_HAS_OPEN_DIALOG_EVENT);
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
if (Problem == SetupapiVerifyDriverBlocked) {
|
|
//
|
|
// Handle a driver block failure.
|
|
// only applicable to UNICODE
|
|
// ANSI won't report this problem code
|
|
//
|
|
HSDB hSDBDrvMain = NULL;
|
|
TAGREF tagref = TAGREF_NULL;
|
|
DRIVERBLOCK_PROMPT DriverBlockPrompt = {0};
|
|
|
|
//
|
|
// Never continue if the driver is in the bad driver database!
|
|
//
|
|
b = FALSE;
|
|
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_BLOCKED_FOR_DEVICE_ERROR : MSG_LOG_DRIVER_BLOCKED_ERROR,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
|
|
if (!(GlobalSetupFlags & PSPGF_UNATTENDED_SETUP)) {
|
|
//
|
|
// Show the driver blocking UI
|
|
//
|
|
DriverBlockPrompt.lpszFile = (TargetFile != NULL)
|
|
? TargetFile
|
|
: ProblemFile;
|
|
|
|
if ((hSDBDrvMain = SdbInitDatabase(SDB_DATABASE_MAIN_DRIVERS, NULL))) {
|
|
HANDLE hFile = INVALID_HANDLE_VALUE;
|
|
|
|
//
|
|
// We are probably dealing with a temp file name at this point,
|
|
// so we need to get a file handle to pass to SdbGetDatabaseMatch
|
|
// along with the final destination file name.
|
|
//
|
|
hFile = CreateFile(ProblemFile,
|
|
GENERIC_READ,
|
|
FILE_SHARE_READ,
|
|
NULL,
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL
|
|
);
|
|
if (hFile != INVALID_HANDLE_VALUE) {
|
|
//
|
|
// Pass the TargetFile (the destination filename) to
|
|
// SdbGetDatabaseMatch because that will be what is
|
|
// in the bad driver database.
|
|
//
|
|
tagref = SdbGetDatabaseMatch(hSDBDrvMain,
|
|
(TargetFile != NULL)
|
|
? pSetupGetFileTitle(TargetFile)
|
|
: ProblemFile,
|
|
hFile,
|
|
NULL,
|
|
0);
|
|
|
|
if (tagref != TAGREF_NULL) {
|
|
SdbReadDriverInformation(hSDBDrvMain,
|
|
tagref,
|
|
&(DriverBlockPrompt.entryinfo));
|
|
}
|
|
|
|
CloseHandle(hFile);
|
|
}
|
|
|
|
SdbReleaseDatabase(hSDBDrvMain);
|
|
}
|
|
|
|
//
|
|
// Always call the dialog code, even if we could access the database.
|
|
//
|
|
iRes = DialogBoxParam(MyDllModuleHandle,
|
|
MAKEINTRESOURCE(IDD_DRIVERBLOCK),
|
|
IsWindow(Owner) ? Owner : NULL,
|
|
DriverBlockDlgProc,
|
|
(LPARAM)&DriverBlockPrompt
|
|
);
|
|
}
|
|
} else
|
|
#endif
|
|
{
|
|
//
|
|
// Handle a digital signature failure.
|
|
//
|
|
// If the policy is block, then the user always gets informed of a problem
|
|
// (i.e., there is no "yes" option, hence no "yes to all" semantics).
|
|
//
|
|
MYASSERT((DriverSigningPolicy != DRIVERSIGN_BLOCKING) || !NoUI);
|
|
|
|
CertPrompt.lpszDescription = DeviceDesc;
|
|
CertPrompt.lpszFile = ProblemFile;
|
|
CertPrompt.ProblemType = Problem;
|
|
CertPrompt.DriverSigningPolicy = DriverSigningPolicy;
|
|
|
|
switch(DriverSigningPolicy) {
|
|
|
|
case DRIVERSIGN_NONE :
|
|
|
|
//
|
|
// SPLOG -- log a PSS entry recording this event.
|
|
//
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_POLICY_NONE : MSG_LOG_SIGNING_ERROR_POLICY_NONE,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
//
|
|
// If requested, find out if the file is protected (we may need to
|
|
// skip it if it's being queued up for delayed copy).
|
|
//
|
|
if(Flags && TargetFile) {
|
|
|
|
if(IsFileProtected(TargetFile,
|
|
(PSETUP_LOG_CONTEXT)LogContext,
|
|
NULL)) {
|
|
|
|
*Flags = IQF_TARGET_PROTECTED;
|
|
}
|
|
}
|
|
|
|
b = TRUE;
|
|
goto exit;
|
|
|
|
case DRIVERSIGN_WARNING :
|
|
if(NoUI) {
|
|
//
|
|
// SPLOG -- log a PSS entry recording this event.
|
|
//
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_AUTO_YES : MSG_LOG_SIGNING_ERROR_AUTO_YES,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
iRes = IDC_VERIFY_WARN_YES;
|
|
} else if(GlobalSetupFlags & PSPGF_UNATTENDED_SETUP) {
|
|
//
|
|
// SPLOG -- log a PSS entry recording this event.
|
|
//
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_AUTO_NO : MSG_LOG_SIGNING_ERROR_AUTO_NO,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
iRes = IDC_VERIFY_WARN_NO;
|
|
} else {
|
|
if (hDialogEvent) {
|
|
SetEvent(hDialogEvent);
|
|
}
|
|
iRes = DialogBoxParam(MyDllModuleHandle,
|
|
CertPrompt.lpszDescription ?
|
|
MAKEINTRESOURCE(IDD_DEVICE_VERIFY_WARNING) :
|
|
MAKEINTRESOURCE(IDD_SOFTWARE_VERIFY_WARNING),
|
|
IsWindow(Owner) ? Owner : NULL,
|
|
CertifyDlgProc,
|
|
(LPARAM)&CertPrompt
|
|
);
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc
|
|
?(iRes == IDC_VERIFY_WARN_YES ? MSG_LOG_DRIVER_SIGNING_ERROR_WARN_YES : MSG_LOG_DRIVER_SIGNING_ERROR_WARN_NO)
|
|
:(iRes == IDC_VERIFY_WARN_YES ? MSG_LOG_SIGNING_ERROR_WARN_YES : MSG_LOG_SIGNING_ERROR_WARN_NO),
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
}
|
|
break;
|
|
|
|
case DRIVERSIGN_BLOCKING :
|
|
|
|
if(GlobalSetupFlags & PSPGF_UNATTENDED_SETUP) {
|
|
//
|
|
// During UNATTENDED, we block silently
|
|
//
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_SILENT_BLOCK : MSG_LOG_SIGNING_ERROR_SILENT_BLOCK,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
iRes = IDC_VERIFY_BLOCK_OK;
|
|
} else {
|
|
LogFailedVerification(
|
|
(PSETUP_LOG_CONTEXT) LogContext,
|
|
DeviceDesc ? MSG_LOG_DRIVER_SIGNING_ERROR_POLICY_BLOCK : MSG_LOG_SIGNING_ERROR_POLICY_BLOCK,
|
|
Error,
|
|
ProblemFile,
|
|
DeviceDesc);
|
|
|
|
if (hDialogEvent) {
|
|
SetEvent(hDialogEvent);
|
|
}
|
|
iRes = DialogBoxParam(MyDllModuleHandle,
|
|
CertPrompt.lpszDescription ?
|
|
MAKEINTRESOURCE(IDD_DEVICE_VERIFY_BLOCK) :
|
|
MAKEINTRESOURCE(IDD_SOFTWARE_VERIFY_BLOCK),
|
|
IsWindow(Owner) ? Owner : NULL,
|
|
CertifyDlgProc,
|
|
(LPARAM)&CertPrompt
|
|
);
|
|
}
|
|
break;
|
|
|
|
default :
|
|
//
|
|
// We don't know about any other policy values!
|
|
//
|
|
MYASSERT(0);
|
|
b = FALSE;
|
|
goto exit;
|
|
}
|
|
|
|
switch(iRes) {
|
|
|
|
case IDC_VERIFY_WARN_NO:
|
|
case IDC_VERIFY_BLOCK_OK:
|
|
b = FALSE;
|
|
break;
|
|
|
|
case IDC_VERIFY_WARN_YES:
|
|
if(TargetFile) {
|
|
pSetupExemptFileFromProtection(TargetFile,
|
|
(DWORD) -1,
|
|
(PSETUP_LOG_CONTEXT)LogContext,
|
|
Flags
|
|
);
|
|
}
|
|
|
|
b = TRUE;
|
|
break;
|
|
|
|
default:
|
|
//
|
|
// Shouldn't get any other values.
|
|
//
|
|
MYASSERT(0);
|
|
b = FALSE;
|
|
}
|
|
}
|
|
|
|
exit:
|
|
if (hDialogEvent) {
|
|
ResetEvent(hDialogEvent);
|
|
CloseHandle(hDialogEvent);
|
|
}
|
|
|
|
|
|
return b;
|
|
}
|
|
|
|
|
|
INT_PTR
|
|
CALLBACK
|
|
CertifyDlgProc(
|
|
IN HWND hwnd,
|
|
IN UINT msg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is the dialog procedure for the driver signing UI that is presented to
|
|
the user when a verification failure is encountered. This dialog handles
|
|
both the 'warn' and 'block' cases.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT MessageLen;
|
|
LOGFONT LogFont;
|
|
HFONT hFontBold = NULL;
|
|
HICON hIcon = NULL;
|
|
|
|
PCERT_PROMPT lpCertPrompt;
|
|
|
|
lpCertPrompt = (PCERT_PROMPT)GetWindowLongPtr(hwnd, DWLP_USER);
|
|
|
|
switch(msg) {
|
|
|
|
case WM_INITDIALOG:
|
|
SetWindowLongPtr(hwnd, DWLP_USER, lParam);
|
|
MessageBeep(MB_ICONASTERISK);
|
|
lpCertPrompt = (PCERT_PROMPT)lParam;
|
|
|
|
//
|
|
// If lpszDescription is not NULL then this is the device verify
|
|
// warning dialog, otherwise it is the software warning dialog.
|
|
//
|
|
if(lpCertPrompt->lpszDescription != NULL) {
|
|
SetDlgItemText(hwnd, IDC_VERIFY_FILENAME, lpCertPrompt->lpszDescription);
|
|
SetDlgText(hwnd, IDC_VERIFY_BOLD, IDS_DEVICE_VERIFY_MSG1, IDS_DEVICE_VERIFY_MSG2);
|
|
} else {
|
|
SetDlgText(hwnd, IDC_VERIFY_BOLD, IDS_SOFTWARE_VERIFY_MSG1, IDS_SOFTWARE_VERIFY_MSG2);
|
|
}
|
|
|
|
//
|
|
// Create the bold font and bold any necessary text.
|
|
//
|
|
hFontBold = (HFONT)SendMessage(GetDlgItem(hwnd, IDC_VERIFY_BOLD),
|
|
WM_GETFONT, 0, 0);
|
|
GetObject(hFontBold, sizeof(LogFont), &LogFont);
|
|
LogFont.lfWeight = FW_BOLD;
|
|
hFontBold = CreateFontIndirect(&LogFont);
|
|
if (hFontBold) {
|
|
SetWindowFont(GetDlgItem(hwnd, IDC_VERIFY_BOLD), hFontBold, TRUE);
|
|
}
|
|
|
|
//
|
|
// Set the appropriate warning or error icon.
|
|
//
|
|
hIcon = LoadIcon(NULL,
|
|
(lpCertPrompt->DriverSigningPolicy == DRIVERSIGN_WARNING) ?
|
|
IDI_WARNING :
|
|
IDI_ERROR
|
|
);
|
|
SendDlgItemMessage(hwnd, IDC_VERIFY_ICON, STM_SETICON, (WPARAM)hIcon, 0L);
|
|
|
|
//
|
|
// The link won't work in GUI mode setup since help center has not yet
|
|
// been installed, so we will just show the static text instead.
|
|
//
|
|
ShowWindow(GetDlgItem(hwnd, IDC_VERIFY_TESTING_LINK), !GuiSetupInProgress);
|
|
ShowWindow(GetDlgItem(hwnd, IDC_VERIFY_TESTING_TEXT), GuiSetupInProgress);
|
|
|
|
//
|
|
// If we are in GUI mode setup then we want to change the text of
|
|
// the buttons to be "Yes" and "No". We also add the following line
|
|
// of text: "Do you want to continue installing the software for
|
|
// this hardware?"
|
|
//
|
|
ShowWindow(GetDlgItem(hwnd, IDC_VERIFY_SETUP_TEXT), GuiSetupInProgress);
|
|
|
|
if (GuiSetupInProgress) {
|
|
TCHAR szButtonText[MAX_PATH];
|
|
|
|
if (LoadString(MyDllModuleHandle, IDS_YES, szButtonText, SIZECHARS(szButtonText))) {
|
|
SetDlgItemText(hwnd, IDC_VERIFY_WARN_YES, szButtonText);
|
|
}
|
|
|
|
if (LoadString(MyDllModuleHandle, IDS_NO, szButtonText, SIZECHARS(szButtonText))) {
|
|
SetDlgItemText(hwnd, IDC_VERIFY_WARN_NO, szButtonText);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Make sure this dialog is in the foreground (at least for this
|
|
// process).
|
|
//
|
|
SetForegroundWindow(hwnd);
|
|
|
|
if (lpCertPrompt->DriverSigningPolicy == DRIVERSIGN_WARNING) {
|
|
SetFocus(GetDlgItem(hwnd, IDC_VERIFY_WARN_NO));
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
case WM_DESTROY:
|
|
if (hFontBold) {
|
|
DeleteObject(hFontBold);
|
|
hFontBold = NULL;
|
|
}
|
|
|
|
if (hIcon) {
|
|
DestroyIcon(hIcon);
|
|
}
|
|
break;
|
|
|
|
case WM_NOTIFY:
|
|
switch (((NMHDR FAR *)lParam)->code) {
|
|
case NM_RETURN:
|
|
case NM_CLICK:
|
|
ShellExecute(hwnd,
|
|
TEXT("open"),
|
|
TEXT("HELPCTR.EXE"),
|
|
TEXT("HELPCTR.EXE -url hcp://services/subsite?node=TopLevelBucket_4/Hardware&topic=MS-ITS%3A%25HELP_LOCATION%25%5Csysdm.chm%3A%3A/logo_testing.htm"),
|
|
NULL,
|
|
SW_SHOWNORMAL
|
|
);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch(wParam) {
|
|
|
|
case IDC_VERIFY_WARN_NO:
|
|
case IDC_VERIFY_WARN_YES:
|
|
case IDC_VERIFY_BLOCK_OK:
|
|
EndDialog(hwnd, (int)wParam);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
INT_PTR
|
|
CALLBACK
|
|
DriverBlockDlgProc(
|
|
IN HWND hwnd,
|
|
IN UINT msg,
|
|
IN WPARAM wParam,
|
|
IN LPARAM lParam
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is the dialog procedure for the driver blocking UI that is presented to
|
|
the user when a a driver that is about to be installed is found in the bad
|
|
driver database.
|
|
|
|
--*/
|
|
|
|
{
|
|
UINT MessageLen;
|
|
HICON hIcon = NULL;
|
|
LPTSTR pBuffer = NULL;
|
|
ULONG BufferSize;
|
|
static HAPPHELPINFOCONTEXT hAppHelpInfoContext = NULL;
|
|
static SDBENTRYINFO SdbEntryInfo;
|
|
|
|
PDRIVERBLOCK_PROMPT lpDriverBlockPrompt;
|
|
|
|
lpDriverBlockPrompt = (PDRIVERBLOCK_PROMPT)GetWindowLongPtr(hwnd, DWLP_USER);
|
|
|
|
switch(msg) {
|
|
|
|
case WM_INITDIALOG:
|
|
SetWindowLongPtr(hwnd, DWLP_USER, lParam);
|
|
MessageBeep(MB_ICONASTERISK);
|
|
lpDriverBlockPrompt = (PDRIVERBLOCK_PROMPT)lParam;
|
|
|
|
hIcon = LoadIcon(MyDllModuleHandle,
|
|
MAKEINTRESOURCE(IDI_DRIVERBLOCK));
|
|
|
|
SendDlgItemMessage(hwnd, IDC_DRIVERBLOCK_ICON, STM_SETICON, (WPARAM)hIcon, 0L);
|
|
|
|
hAppHelpInfoContext = SdbOpenApphelpInformation(&lpDriverBlockPrompt->entryinfo.guidDB,
|
|
&lpDriverBlockPrompt->entryinfo.guidID);
|
|
|
|
if ((hAppHelpInfoContext) &&
|
|
((BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpAppName,
|
|
NULL,
|
|
0)) != 0) &&
|
|
(pBuffer = MyMalloc(BufferSize)) &&
|
|
((BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpAppName,
|
|
pBuffer,
|
|
BufferSize)) != 0)) {
|
|
SetDlgItemText(hwnd, IDC_DRIVERBLOCK_APPNAME, pBuffer);
|
|
MyFree(pBuffer);
|
|
} else if (lpDriverBlockPrompt->lpszFile) {
|
|
SetDlgItemText(hwnd, IDC_DRIVERBLOCK_APPNAME, pSetupGetFileTitle(lpDriverBlockPrompt->lpszFile));
|
|
}
|
|
|
|
if ((hAppHelpInfoContext) &&
|
|
((BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpDetails,
|
|
NULL,
|
|
0)) != 0) &&
|
|
(pBuffer = MyMalloc(BufferSize)) &&
|
|
((BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpDetails,
|
|
pBuffer,
|
|
BufferSize)) != 0)) {
|
|
|
|
SetDlgItemText(hwnd, IDC_DRIVERBLOCK_SUMMARY, pBuffer);
|
|
MyFree(pBuffer);
|
|
}
|
|
|
|
//
|
|
// Make sure this dialog is in the foreground (at least for this
|
|
// process).
|
|
//
|
|
SetForegroundWindow(hwnd);
|
|
return FALSE;
|
|
|
|
case WM_DESTROY:
|
|
if (hIcon) {
|
|
DestroyIcon(hIcon);
|
|
}
|
|
if (hAppHelpInfoContext) {
|
|
SdbCloseApphelpInformation(hAppHelpInfoContext);
|
|
}
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
switch(LOWORD(wParam)) {
|
|
case IDCANCEL:
|
|
EndDialog(hwnd, (int)wParam);
|
|
break;
|
|
|
|
case IDC_DRIVERBLOCK_DETAILS:
|
|
if (hAppHelpInfoContext) {
|
|
|
|
BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpHelpCenterURL,
|
|
NULL,
|
|
0);
|
|
|
|
if (BufferSize && (pBuffer = MyMalloc(BufferSize + (lstrlen(TEXT("HELPCTR.EXE -url ")) * sizeof(TCHAR))))) {
|
|
lstrcpy(pBuffer, TEXT("HELPCTR.EXE -url "));
|
|
|
|
BufferSize = SdbQueryApphelpInformation(hAppHelpInfoContext,
|
|
ApphelpHelpCenterURL,
|
|
(PVOID)&pBuffer[lstrlen(TEXT("HELPCTR.EXE -url "))],
|
|
BufferSize);
|
|
|
|
if (BufferSize) {
|
|
ShellExecute(hwnd,
|
|
TEXT("open"),
|
|
TEXT("HELPCTR.EXE"),
|
|
pBuffer,
|
|
NULL,
|
|
SW_SHOWNORMAL);
|
|
}
|
|
|
|
MyFree(pBuffer);
|
|
}
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
#endif
|
|
|
|
|
|
DWORD
|
|
pGetInfOriginalNameAndCatalogFile(
|
|
IN PLOADED_INF Inf, OPTIONAL
|
|
IN LPCTSTR CurrentName, OPTIONAL
|
|
OUT PBOOL DifferentName, OPTIONAL
|
|
OUT LPTSTR OriginalName, OPTIONAL
|
|
IN DWORD OriginalNameSize,
|
|
OUT LPTSTR OriginalCatalogName, OPTIONAL
|
|
IN DWORD OriginalCatalogNameSize,
|
|
IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine determines whether a specified inf once had a different
|
|
original name, such as in the case where the Di stuff copied and renamed a
|
|
device inf.
|
|
|
|
(Information about an INF's original name comes from the PNF.)
|
|
|
|
This routine can also optionally return the original name of the catalog
|
|
file for this INF.
|
|
|
|
Arguments:
|
|
|
|
Inf - optionally, supplies a pointer to a LOADED_INF whose original name
|
|
and catalog file are to be queried. If this parameter isn't specified,
|
|
then CurrentName must be specified.
|
|
|
|
CurrentName - optionally, supplies the path to the INF whose original name
|
|
is to be queried. If Inf parameter is specified, this parameter is
|
|
ignored.
|
|
|
|
DifferentName - optionally, supplies the address of a boolean variable that,
|
|
upon successful return, is set to TRUE if the INF's current name is
|
|
different than its original name.
|
|
|
|
OriginalName - if this routine returns successfully, and the DifferentName
|
|
boolean was set to TRUE, then this optional buffer receives
|
|
the INF's original name, which _will not_ be the same as the current
|
|
name.
|
|
|
|
OriginalNameSize - supplies size of buffer (bytes for ansi, chars for
|
|
unicode) of OriginalName buffer, or zero if OriginalName is NULL.
|
|
|
|
OriginalCatalogName - optionally, supplies a buffer that receives the
|
|
original name of the catalog specified by this INF. If the catalog
|
|
doesn't specify a catalog file, this buffer will be set to an empty
|
|
string.
|
|
|
|
OriginalCatalogNameSize - supplies size, in characters, of
|
|
OriginalCatalogName buffer (zero if buffer not supplied).
|
|
|
|
AltPlatformInfo - optionally, supplies the address of a structure describing
|
|
the platform parameters that should be used in formulating the decorated
|
|
CatalogFile= entry to be used when searching for the INF's associated
|
|
catalog file.
|
|
|
|
Return Value:
|
|
|
|
If information is successfully retrieved from the INF, the return value is
|
|
NO_ERROR. Otherwise, it is a Win32 error code indicating the cause of
|
|
failure.
|
|
|
|
--*/
|
|
|
|
{
|
|
DWORD d;
|
|
HINF hInf = INVALID_HANDLE_VALUE;
|
|
|
|
MYASSERT((DifferentName && OriginalName && OriginalNameSize) ||
|
|
!(DifferentName || OriginalName || OriginalNameSize));
|
|
|
|
MYASSERT((OriginalCatalogName && OriginalCatalogNameSize) ||
|
|
!(OriginalCatalogName || OriginalCatalogNameSize));
|
|
|
|
MYASSERT(Inf || CurrentName);
|
|
|
|
if(DifferentName) {
|
|
*DifferentName = FALSE;
|
|
}
|
|
|
|
if(!Inf) {
|
|
//
|
|
// Open the INF.
|
|
//
|
|
hInf = SetupOpenInfFile(CurrentName,
|
|
NULL,
|
|
INF_STYLE_OLDNT | INF_STYLE_WIN4,
|
|
NULL
|
|
);
|
|
|
|
if(hInf == INVALID_HANDLE_VALUE) {
|
|
return GetLastError();
|
|
}
|
|
|
|
//
|
|
// We don't need to lock the INF because it'll never be accessible
|
|
// outside of this routine.
|
|
//
|
|
Inf = (PLOADED_INF)hInf;
|
|
}
|
|
|
|
//
|
|
// Enclose in try/except in case we hit an inpage error while using this
|
|
// memory-mapped image.
|
|
//
|
|
d = NO_ERROR;
|
|
try {
|
|
|
|
if(DifferentName) {
|
|
if(Inf->OriginalInfName) {
|
|
lstrcpyn(OriginalName, Inf->OriginalInfName, OriginalNameSize);
|
|
*DifferentName = TRUE;
|
|
}
|
|
}
|
|
|
|
if(OriginalCatalogName) {
|
|
|
|
if(!pSetupGetCatalogFileValue(&(Inf->VersionBlock),
|
|
OriginalCatalogName,
|
|
OriginalCatalogNameSize,
|
|
AltPlatformInfo)) {
|
|
//
|
|
// The INF didn't specify an associated catalog file
|
|
//
|
|
*OriginalCatalogName = TEXT('\0');
|
|
}
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
//
|
|
// If we hit an AV, then use invalid parameter error, otherwise, assume
|
|
// an inpage error when dealing with a mapped-in file.
|
|
//
|
|
d = (GetExceptionCode() == EXCEPTION_ACCESS_VIOLATION) ? ERROR_INVALID_PARAMETER : ERROR_READ_FAULT;
|
|
}
|
|
|
|
if(hInf != INVALID_HANDLE_VALUE) {
|
|
SetupCloseInfFile(hInf);
|
|
}
|
|
|
|
return d;
|
|
}
|
|
|
|
|
|
#ifdef UNICODE
|
|
|
|
PSECURITY_DESCRIPTOR
|
|
pSetupConvertTextToSD(
|
|
IN PCWSTR SDS,
|
|
OUT PULONG SecDescSize
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Helper for cfgmgr.lib
|
|
|
|
Obtains a binary security descriptor from an SDS
|
|
Resulting buffer must be free'd using LocalFree (not MyFree)
|
|
returns NULL if not supported and sets last error
|
|
|
|
Arguments:
|
|
|
|
SDS - string to obtain security descriptor from
|
|
|
|
SecDescSize - filled in with size of security descriptor
|
|
|
|
Return Value:
|
|
|
|
returns security descriptor (use LocalFree to release)
|
|
or NULL with GetLastError indicating error
|
|
|
|
--*/
|
|
{
|
|
SCESTATUS status;
|
|
PSECURITY_DESCRIPTOR pSD = NULL;
|
|
ULONG ulSDSize;
|
|
SECURITY_INFORMATION siSeInfo;
|
|
|
|
//
|
|
// If we're in "Disable SCE" mode on embedded, don't do security stuff...
|
|
//
|
|
if(GlobalSetupFlags & PSPGF_NO_SCE_EMBEDDED) {
|
|
SetLastError(ERROR_SCE_DISABLED);
|
|
return NULL;
|
|
}
|
|
|
|
try {
|
|
status = SceSvcConvertTextToSD((PWSTR)SDS,&pSD,&ulSDSize,&siSeInfo);
|
|
switch (status ) {
|
|
case SCESTATUS_SUCCESS:
|
|
MYASSERT(pSD);
|
|
MYASSERT(ulSDSize);
|
|
if (SecDescSize) {
|
|
*SecDescSize = ulSDSize;
|
|
}
|
|
SetLastError(NO_ERROR);
|
|
break;
|
|
|
|
case SCESTATUS_INVALID_PARAMETER:
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
pSD = NULL;
|
|
break;
|
|
|
|
case SCESTATUS_NOT_ENOUGH_RESOURCE:
|
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|
pSD = NULL;
|
|
break;
|
|
|
|
case SCESTATUS_RECORD_NOT_FOUND:
|
|
default:
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
pSD = NULL;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
//
|
|
// If we hit an AV, then use invalid parameter error, otherwise, assume
|
|
// an inpage error when dealing with a mapped-in file.
|
|
//
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
pSD = NULL;
|
|
}
|
|
return pSD;
|
|
}
|
|
|
|
PWSTR
|
|
pSetupConvertSDToText(
|
|
IN PSECURITY_DESCRIPTOR SD,
|
|
OUT PULONG pSDSSize
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Helper for cfgmgr.lib
|
|
|
|
Obtains an SDS from a binary security descriptor
|
|
Resulting buffer must be free'd using LocalFree (not MyFree)
|
|
returns NULL if not supported and sets last error
|
|
|
|
Arguments:
|
|
|
|
SD - security descriptor to convert to a string
|
|
|
|
pSDSSize - return size of string
|
|
|
|
Return Value:
|
|
|
|
returns security descriptor string (use LocalFree to release)
|
|
or NULL with GetLastError indicating error
|
|
|
|
--*/
|
|
{
|
|
HINSTANCE Dll_Handle;
|
|
FARPROC SceFileProc;
|
|
SCESTATUS status;
|
|
DWORD LoadStatus;
|
|
SECURITY_INFORMATION securityInformation = 0;
|
|
PSID sid;
|
|
PACL acl;
|
|
BOOLEAN tmp,present;
|
|
LPWSTR SDS = NULL;
|
|
ULONG ulSSDSize;
|
|
SECURITY_INFORMATION siSeInfo;
|
|
|
|
//
|
|
// If we're in "Disable SCE" mode on embedded, don't do security stuff...
|
|
//
|
|
if(GlobalSetupFlags & PSPGF_NO_SCE_EMBEDDED) {
|
|
//
|
|
// Report an empty string
|
|
//
|
|
return LocalAlloc(LPTR, sizeof(WCHAR)); // LPTR zeroes out the char
|
|
}
|
|
|
|
try {
|
|
//
|
|
// find out what relevent information is in the descriptor
|
|
// up a securityInformation block to go with it.
|
|
//
|
|
|
|
status = RtlGetOwnerSecurityDescriptor(SD, &sid, &tmp);
|
|
|
|
if(NT_SUCCESS(status) && (sid != NULL)) {
|
|
securityInformation |= OWNER_SECURITY_INFORMATION;
|
|
}
|
|
|
|
status = RtlGetGroupSecurityDescriptor(SD, &sid, &tmp);
|
|
|
|
if(NT_SUCCESS(status) && (sid != NULL)) {
|
|
securityInformation |= GROUP_SECURITY_INFORMATION;
|
|
}
|
|
|
|
status = RtlGetSaclSecurityDescriptor(SD,
|
|
&present,
|
|
&acl,
|
|
&tmp);
|
|
|
|
if(NT_SUCCESS(status) && (present)) {
|
|
securityInformation |= SACL_SECURITY_INFORMATION;
|
|
}
|
|
|
|
status = RtlGetDaclSecurityDescriptor(SD,
|
|
&present,
|
|
&acl,
|
|
&tmp);
|
|
|
|
if(NT_SUCCESS(status) && (present)) {
|
|
securityInformation |= DACL_SECURITY_INFORMATION;
|
|
}
|
|
|
|
//
|
|
// now obtain an SDS
|
|
//
|
|
status = SceSvcConvertSDToText(SD,securityInformation,&SDS,&ulSSDSize);
|
|
switch (status ) {
|
|
case SCESTATUS_SUCCESS:
|
|
MYASSERT(SDS);
|
|
MYASSERT(ulSSDSize);
|
|
if(pSDSSize != NULL) {
|
|
*pSDSSize = ulSSDSize;
|
|
}
|
|
SetLastError(NO_ERROR);
|
|
break;
|
|
|
|
case SCESTATUS_INVALID_PARAMETER:
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
SDS = NULL;
|
|
break;
|
|
|
|
case SCESTATUS_NOT_ENOUGH_RESOURCE:
|
|
SetLastError(ERROR_NOT_ENOUGH_MEMORY);
|
|
SDS = NULL;
|
|
break;
|
|
|
|
case SCESTATUS_RECORD_NOT_FOUND:
|
|
default:
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
SDS = NULL;
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
if (SDS) {
|
|
LocalFree(SDS);
|
|
}
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
SDS = NULL;
|
|
}
|
|
return SDS;
|
|
}
|
|
|
|
DWORD
|
|
pSetupCallSCE(
|
|
IN DWORD Operation,
|
|
IN PCWSTR FullName,
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PCWSTR String1,
|
|
IN DWORD Index1,
|
|
IN PSECURITY_DESCRIPTOR SecDesc OPTIONAL
|
|
)
|
|
/*
|
|
|
|
Operation ST_SCE_SET : - Sets security on a File in File Queue and informs SCE database
|
|
FullName : - Filename (Needed)
|
|
Queue : - Pointer to FileQueue (Needed)
|
|
Index : - Index in String Table of Queue (Needed)
|
|
|
|
Operation ST_SCE_RENAME : - Sets security on a File in File Queue and informs SCE database to
|
|
record it for the filename mentioned in String1
|
|
FullName : - Filename (Needed)
|
|
Queue : - Pointer to FileQueue (Needed)
|
|
String1 ; - Filename to record in Database (Needed)
|
|
Index : - Index in String Table of Queue (Optional - only if it needs to be set otherwise -1)
|
|
|
|
Operation ST_SCE_DELETE : - Removes record of file in SCE database
|
|
FullName : - Filename (Needed)
|
|
|
|
Operation ST_SCE_UNWIND : - Used for Backup Unwinds when we reset the security on a dirty file
|
|
FullName : - Filename (Needed)
|
|
SecDesc : - Pointer to Security Descriptor for the original file that we unwind (Needed)
|
|
|
|
Operation ST_SCE_SERVICES : - Sets security on a Service and informs SCE database
|
|
FullName : - Service Name (Needed)
|
|
Index : - Service Style (Needed)
|
|
String1 ; - Security Descriptor string
|
|
|
|
Operation ST_SCE_SDS_TO_BIN : - Sets security on a Service and informs SCE database
|
|
FullName : - Service Name (Needed)
|
|
Index : - Service Style (Needed)
|
|
String1 ; - Security Descriptor string
|
|
|
|
In each case, return value is error or NO_ERROR
|
|
*/
|
|
{
|
|
|
|
FARPROC SceFileProc;
|
|
PCWSTR SecurityDescriptor;
|
|
HINSTANCE Dll_Handle;
|
|
DWORD ret, LoadStatus;
|
|
|
|
//
|
|
// If we're in "Disable SCE" mode on embedded, don't do security stuff...
|
|
//
|
|
if(GlobalSetupFlags & PSPGF_NO_SCE_EMBEDDED) {
|
|
return NO_ERROR;
|
|
}
|
|
|
|
try {
|
|
switch (Operation) {
|
|
|
|
case ST_SCE_SET:
|
|
|
|
//Get the Security descriptor from the String table of the node
|
|
|
|
if( Index1 != -1 ){
|
|
SecurityDescriptor = pSetupStringTableStringFromId( Queue->StringTable, Index1 );
|
|
|
|
if(!SecurityDescriptor) {
|
|
ret= NO_ERROR;
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
ret = NO_ERROR;
|
|
break;
|
|
}
|
|
|
|
|
|
ret = SceSetupUpdateSecurityFile((PWSTR)FullName, 0, (PWSTR)SecurityDescriptor );
|
|
break;
|
|
|
|
case ST_SCE_RENAME:
|
|
|
|
if( Index1 != -1 ) {
|
|
SecurityDescriptor = pSetupStringTableStringFromId( Queue->StringTable, Index1 );
|
|
} else {
|
|
SecurityDescriptor = NULL;
|
|
}
|
|
|
|
ret = SceSetupMoveSecurityFile( (PWSTR)FullName, (PWSTR)String1, (PWSTR)SecurityDescriptor );
|
|
break;
|
|
|
|
|
|
|
|
case ST_SCE_DELETE:
|
|
|
|
ret = SceSetupMoveSecurityFile( (PWSTR)FullName, NULL, NULL );
|
|
break;
|
|
|
|
|
|
case ST_SCE_UNWIND:
|
|
|
|
ret = SceSetupUnwindSecurityFile( (PWSTR)FullName, SecDesc );
|
|
break;
|
|
|
|
case ST_SCE_SERVICES:
|
|
|
|
if( String1 == NULL ){
|
|
ret = NO_ERROR;
|
|
} else {
|
|
ret = SceSetupUpdateSecurityService( (PWSTR)FullName, Index1, (PWSTR)String1 );
|
|
}
|
|
break;
|
|
|
|
default:
|
|
MYASSERT(0);
|
|
ret = ERROR_INVALID_DATA;
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
ret = ERROR_INVALID_DATA;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
#endif // UNICODE
|
|
|
|
|
|
VOID
|
|
RestoreBootReplacedFile(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSP_FILE_QUEUE_NODE QueueNode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine restores a file that was renamed in preparation for a bootfile
|
|
installation.
|
|
|
|
Arguments:
|
|
|
|
Queue - queue that contains the bootfile copy operation
|
|
|
|
QueueNode - bootfile copy operation being aborted
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD rc;
|
|
LONG TargetID;
|
|
SP_TARGET_ENT TargetInfo;
|
|
PCTSTR TargetFilename, RenamedFilename;
|
|
BOOL UnPostSucceeded;
|
|
|
|
//
|
|
// First, we need to find the corresponding target info node so
|
|
// we can find out what temporary name our file was renamed to.
|
|
//
|
|
rc = pSetupBackupGetTargetByPath((HSPFILEQ)Queue,
|
|
NULL, // use Queue's string table
|
|
NULL,
|
|
QueueNode->TargetDirectory,
|
|
-1,
|
|
QueueNode->TargetFilename,
|
|
&TargetID,
|
|
&TargetInfo
|
|
);
|
|
|
|
if(rc == NO_ERROR) {
|
|
//
|
|
// Has the file previously been renamed (and not yet
|
|
// restored)?
|
|
//
|
|
if((TargetInfo.InternalFlags & (SP_TEFLG_MOVED | SP_TEFLG_RESTORED)) == SP_TEFLG_MOVED) {
|
|
|
|
TargetFilename = pSetupFormFullPath(
|
|
Queue->StringTable,
|
|
TargetInfo.TargetRoot,
|
|
TargetInfo.TargetSubDir,
|
|
TargetInfo.TargetFilename
|
|
);
|
|
MYASSERT(TargetFilename);
|
|
|
|
RenamedFilename = pSetupStringTableStringFromId(
|
|
Queue->StringTable,
|
|
TargetInfo.NewTargetFilename
|
|
);
|
|
MYASSERT(RenamedFilename);
|
|
|
|
//
|
|
// Move the renamed file back to its original name.
|
|
//
|
|
RestoreRenamedOrBackedUpFile(TargetFilename,
|
|
RenamedFilename,
|
|
TRUE,
|
|
Queue->LogContext
|
|
);
|
|
//
|
|
// Set the flag indicating that this file has been
|
|
// restored, and save this info.
|
|
//
|
|
TargetInfo.InternalFlags |= SP_TEFLG_RESTORED;
|
|
pSetupBackupSetTargetByID((HSPFILEQ)Queue, TargetID, &TargetInfo);
|
|
|
|
//
|
|
// Finally, get rid of the delayed-move node that was to
|
|
// delete the renamed file upon reboot.
|
|
//
|
|
UnPostSucceeded = UnPostDelayedMove(Queue,
|
|
RenamedFilename,
|
|
NULL
|
|
);
|
|
MYASSERT(UnPostSucceeded);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
pSetupExemptFileFromProtection(
|
|
IN PCTSTR FileName,
|
|
IN DWORD FileChangeFlags,
|
|
IN PSETUP_LOG_CONTEXT LogContext, OPTIONAL
|
|
OUT PDWORD QueueNodeFlags OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine checks to see if the specified file is a protected system
|
|
file, and if so, it tells SFC to make a replacement exception for this file.
|
|
|
|
Arguments:
|
|
|
|
FileName - Supplies the name of the file for which an exception is being
|
|
requested.
|
|
|
|
FileChangeFlags - Supplies the flags to be passed to SfcFileException, if
|
|
this file is determined to be under the protection of SFP.
|
|
|
|
LogContext - Optionally, supplies the log context to be used when logging
|
|
information resulting from this request.
|
|
|
|
QueueNodeFlags - Optionally, supplies the address of a variable that
|
|
receives one or more of the following queue node flags indicating
|
|
whether the specified file is a protected system file, and whether an
|
|
exception was granted for its replacement:
|
|
|
|
IQF_TARGET_PROTECTED - File is a protected system file.
|
|
IQF_ALLOW_UNSIGNED - An exception has been granted so that the file
|
|
may be replaced by an unsigned file.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
#ifdef UNICODE
|
|
HANDLE hSfp;
|
|
PSETUP_LOG_CONTEXT lc = NULL;
|
|
DWORD Result = NO_ERROR;
|
|
|
|
if(QueueNodeFlags) {
|
|
*QueueNodeFlags = 0;
|
|
}
|
|
|
|
//
|
|
// If the caller didn't supply us with a LogContext, then create our own.
|
|
// We want to do this so that all log entries generated herein will end up
|
|
// in the same section.
|
|
//
|
|
if(!LogContext) {
|
|
if(CreateLogContext(NULL, TRUE, &lc) == NO_ERROR) {
|
|
//
|
|
// success
|
|
//
|
|
LogContext = lc;
|
|
} else {
|
|
lc = NULL;
|
|
}
|
|
}
|
|
|
|
if(IsFileProtected(FileName, LogContext, &hSfp)) {
|
|
|
|
if(QueueNodeFlags) {
|
|
*QueueNodeFlags = IQF_TARGET_PROTECTED;
|
|
}
|
|
|
|
Result = SfcFileException(hSfp,
|
|
(PWSTR)FileName,
|
|
FileChangeFlags
|
|
);
|
|
|
|
if(Result == NO_ERROR) {
|
|
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_ERROR,
|
|
MSG_LOG_SFC_EXEMPT_SUCCESS,
|
|
NULL,
|
|
FileName);
|
|
|
|
if(QueueNodeFlags) {
|
|
*QueueNodeFlags |= IQF_ALLOW_UNSIGNED;
|
|
}
|
|
|
|
} else {
|
|
WriteLogEntry(
|
|
LogContext,
|
|
SETUP_LOG_ERROR|SETUP_LOG_BUFFER,
|
|
MSG_LOG_SFC_EXEMPT_FAIL,
|
|
NULL,
|
|
FileName);
|
|
WriteLogError(
|
|
LogContext,
|
|
SETUP_LOG_ERROR,
|
|
Result);
|
|
}
|
|
|
|
SfcClose(hSfp);
|
|
|
|
//
|
|
// If we created our own local LogContext, we can free it now.
|
|
//
|
|
if(lc) {
|
|
DeleteLogContext(lc);
|
|
}
|
|
}
|
|
#else // no file protection on win9x
|
|
if(QueueNodeFlags) {
|
|
*QueueNodeFlags = 0;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
BOOL
|
|
pSetupProtectedRenamesFlag(
|
|
BOOL bSet
|
|
)
|
|
{
|
|
HKEY hKey;
|
|
long rslt = ERROR_SUCCESS;
|
|
|
|
if (OSVersionInfo.dwPlatformId != VER_PLATFORM_WIN32_NT) {
|
|
return(TRUE);
|
|
}
|
|
|
|
rslt = RegOpenKeyEx(
|
|
HKEY_LOCAL_MACHINE,
|
|
TEXT("System\\CurrentControlSet\\Control\\Session Manager"),
|
|
0,
|
|
KEY_SET_VALUE,
|
|
&hKey);
|
|
|
|
if (rslt == ERROR_SUCCESS) {
|
|
DWORD Value = bSet ? 1 : 0;
|
|
rslt = RegSetValueEx(
|
|
hKey,
|
|
TEXT("AllowProtectedRenames"),
|
|
0,
|
|
REG_DWORD,
|
|
(LPBYTE)&Value,
|
|
sizeof(DWORD));
|
|
|
|
RegCloseKey(hKey);
|
|
|
|
if (rslt != ERROR_SUCCESS) {
|
|
DebugPrintEx( DPFLTR_ERROR_LEVEL, TEXT("couldn't RegSetValueEx, ec = %d\n"), rslt );
|
|
}
|
|
|
|
} else {
|
|
DebugPrintEx( DPFLTR_ERROR_LEVEL, TEXT("couldn't RegOpenKeyEx, ec = %d\n"), rslt );
|
|
}
|
|
|
|
return(rslt == ERROR_SUCCESS);
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
pSetupUninstallNewCatalogNodes(
|
|
IN PSP_FILE_QUEUE Queue,
|
|
IN PSETUP_LOG_CONTEXT LogContext OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This routine uninstalls any newly-copied INFs/PNFs/CATs contained in the
|
|
specified linked list of catalog nodes.
|
|
|
|
Arguments:
|
|
|
|
Queue - Supplies a pointer to the file queue (potentially) containing
|
|
newly-copied catalog nodes to be uninstalled.
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
PSPQ_CATALOG_INFO CatalogNode;
|
|
PTSTR InfToUninstall;
|
|
BOOL Locked = FALSE;
|
|
|
|
try {
|
|
|
|
if(!_pSpUtilsStringTableLock(Queue->StringTable)) {
|
|
leave;
|
|
}
|
|
|
|
Locked = TRUE;
|
|
|
|
for(CatalogNode = Queue->CatalogList;
|
|
CatalogNode;
|
|
CatalogNode = CatalogNode->Next) {
|
|
|
|
if(CatalogNode->Flags & CATINFO_FLAG_NEWLY_COPIED) {
|
|
|
|
InfToUninstall = _pSpUtilsStringTableStringFromId(
|
|
Queue->StringTable,
|
|
CatalogNode->InfFinalPath
|
|
);
|
|
|
|
MYASSERT(InfToUninstall);
|
|
|
|
if(InfToUninstall) {
|
|
pSetupUninstallOEMInf(InfToUninstall, LogContext, SUOI_FORCEDELETE, NULL);
|
|
}
|
|
}
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
//
|
|
// Reference the following variable so the compiler will respect
|
|
// statement ordering w.r.t. its assignment.
|
|
//
|
|
Locked = Locked;
|
|
}
|
|
|
|
if(Locked) {
|
|
_pSpUtilsStringTableUnlock(Queue->StringTable);
|
|
}
|
|
}
|
|
|
|
|
|
BOOL
|
|
WINAPI
|
|
SetupUninstallNewlyCopiedInfs(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN DWORD Flags,
|
|
IN PVOID Reserved
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This API uninstalls any INFs (and their associated PNFs and CATs) that
|
|
were previously installed during committal of the specified file queue.
|
|
|
|
Arguments:
|
|
|
|
QueueHandle - Supplies a handle to a committed file queue (potentially)
|
|
containing newly-copied INFs to be uninstalled.
|
|
|
|
Flags - Supplies flags that alter the behavior of this API. Presently, no
|
|
flags are defined. This parameter must be zero.
|
|
|
|
Reserved - Reserved for future use. This parameter must be NULL.
|
|
|
|
Return Value:
|
|
|
|
If all the parameters were valid, the return value is non-zero (TRUE). Note
|
|
that this does _not_ necessarily mean that any newly-copied INFs were
|
|
uninstalled.
|
|
|
|
If there was a problem with the parameters passed in, the return value is
|
|
FALSE, and GetLastError provides more information on the problem.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSP_FILE_QUEUE Queue;
|
|
BOOL Success;
|
|
PSETUP_LOG_CONTEXT LogContext;
|
|
|
|
if(Flags) {
|
|
SetLastError(ERROR_INVALID_FLAGS);
|
|
return FALSE;
|
|
}
|
|
|
|
if(Reserved) {
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
//
|
|
// Queue handle is actually a pointer to the queue structure.
|
|
//
|
|
Queue = (PSP_FILE_QUEUE)QueueHandle;
|
|
|
|
//
|
|
// do a quick handle validation before anything else
|
|
//
|
|
try {
|
|
Success = ((Queue != NULL) && (Queue != INVALID_HANDLE_VALUE) && (Queue->Signature == SP_FILE_QUEUE_SIG));
|
|
if(Success) {
|
|
LogContext = Queue->LogContext;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
Success = FALSE;
|
|
}
|
|
|
|
if(!Success) {
|
|
SetLastError(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
|
|
pSetupUninstallNewCatalogNodes(Queue, LogContext);
|
|
|
|
return TRUE;
|
|
}
|
|
|