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.
3221 lines
103 KiB
3221 lines
103 KiB
/*++
|
|
|
|
Copyright (c) 1995-2000 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
fileq2.c
|
|
|
|
Abstract:
|
|
|
|
Setup file queue routines for enqueing copy operations.
|
|
|
|
Author:
|
|
|
|
Ted Miller (tedm) 15-Feb-1995
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
|
|
#define STR_DRIVERCACHEINF TEXT("drvindex.inf")
|
|
|
|
|
|
//
|
|
// Structure used with _SetupQueueCopy
|
|
//
|
|
typedef struct _SP_FILE_COPY_PARAMS_AEX {
|
|
DWORD cbSize;
|
|
HSPFILEQ QueueHandle;
|
|
PCSTR SourceRootPath; OPTIONAL
|
|
PCSTR SourcePath; OPTIONAL
|
|
PCSTR SourceFilename;
|
|
PCSTR SourceDescription; OPTIONAL
|
|
PCSTR SourceTagfile; OPTIONAL
|
|
PCSTR TargetDirectory;
|
|
PCSTR TargetFilename; OPTIONAL
|
|
DWORD CopyStyle;
|
|
HINF LayoutInf; OPTIONAL
|
|
PCSTR SecurityDescriptor; OPTIONAL
|
|
DWORD SourceFlags; OPTIONAL
|
|
BOOL SourceFlagsSet; OPTIONAL // we need this flag since SourceFlags may be zero
|
|
PCSTR CacheName;
|
|
} SP_FILE_COPY_PARAMS_AEX, *PSP_FILE_COPY_PARAMS_AEX;
|
|
|
|
typedef struct _SP_FILE_COPY_PARAMS_WEX {
|
|
DWORD cbSize;
|
|
HSPFILEQ QueueHandle;
|
|
PCWSTR SourceRootPath; OPTIONAL
|
|
PCWSTR SourcePath; OPTIONAL
|
|
PCWSTR SourceFilename;
|
|
PCWSTR SourceDescription; OPTIONAL
|
|
PCWSTR SourceTagfile; OPTIONAL
|
|
PCWSTR TargetDirectory;
|
|
PCWSTR TargetFilename; OPTIONAL
|
|
DWORD CopyStyle;
|
|
HINF LayoutInf; OPTIONAL
|
|
PCWSTR SecurityDescriptor; OPTIONAL
|
|
DWORD SourceFlags; OPTIONAL
|
|
BOOL SourceFlagsSet; OPTIONAL // we need this flag since SourceFlags may be zero
|
|
PCWSTR CacheName;
|
|
} SP_FILE_COPY_PARAMS_WEX, *PSP_FILE_COPY_PARAMS_WEX;
|
|
|
|
#ifdef UNICODE
|
|
typedef SP_FILE_COPY_PARAMS_WEX SP_FILE_COPY_PARAMSEX;
|
|
typedef PSP_FILE_COPY_PARAMS_WEX PSP_FILE_COPY_PARAMSEX;
|
|
#else
|
|
typedef SP_FILE_COPY_PARAMS_AEX SP_FILE_COPY_PARAMSEX;
|
|
typedef PSP_FILE_COPY_PARAMS_AEX PSP_FILE_COPY_PARAMSEX;
|
|
#endif
|
|
|
|
|
|
|
|
|
|
BOOL
|
|
_SetupQueueCopy(
|
|
IN PSP_FILE_COPY_PARAMSEX CopyParams,
|
|
IN PINFCONTEXT LayoutLineContext, OPTIONAL
|
|
IN HINF AdditionalInfs OPTIONAL
|
|
);
|
|
|
|
PSOURCE_MEDIA_INFO
|
|
pSetupQueueSourceMedia(
|
|
IN OUT PSP_FILE_QUEUE Queue,
|
|
IN OUT PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN LONG SourceRootStringId,
|
|
IN PCTSTR SourceDescription, OPTIONAL
|
|
IN PCTSTR SourceTagfile, OPTIONAL
|
|
IN PCTSTR SourceCabfile, OPTIONAL
|
|
IN DWORD MediaFlags
|
|
);
|
|
|
|
BOOL
|
|
pSetupQueueSingleCopy(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN HINF InfHandle,
|
|
IN HINF ListInfHandle, OPTIONAL
|
|
IN PCTSTR SectionName, OPTIONAL
|
|
IN PCTSTR SourceRootPath,
|
|
IN PCTSTR SourceFilename,
|
|
IN PCTSTR TargetFilename,
|
|
IN DWORD CopyStyle,
|
|
IN PCTSTR SecurityDescriptor,
|
|
IN PCTSTR CacheName
|
|
);
|
|
|
|
BOOL
|
|
pSetupGetSourceAllInfo(
|
|
IN HINF InfHandle,
|
|
IN PINFCONTEXT LayoutLineContext, OPTIONAL
|
|
IN UINT SourceId,
|
|
IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo, OPTIONAL
|
|
OUT PCTSTR *Description,
|
|
OUT PCTSTR *Tagfile,
|
|
OUT PCTSTR *RelativePath,
|
|
OUT PUINT SourceFlags
|
|
);
|
|
|
|
BOOL
|
|
pIsDriverCachePresent(
|
|
IN PCTSTR DriverName,
|
|
IN PCTSTR SubDirectory,
|
|
OUT PTSTR DriverBuffer
|
|
);
|
|
|
|
BOOL
|
|
pIsFileInDriverCache(
|
|
IN HINF CabInf,
|
|
IN PCTSTR TargetFilename,
|
|
IN PCTSTR SubDirectory,
|
|
OUT PCTSTR *CacheName
|
|
);
|
|
|
|
|
|
|
|
//
|
|
// HACK ALERT!!! HACK HACK HACK!!!!
|
|
//
|
|
// There might be an override platform specified. If this is so,
|
|
// we will look for \i386, \mips, etc as the final component of the
|
|
// specified path when queuing files, and replace it with the
|
|
// override path. This is a TOTAL HACK.
|
|
//
|
|
PCTSTR PlatformPathOverride;
|
|
|
|
VOID
|
|
pSetupInitPlatformPathOverrideSupport(
|
|
IN BOOL Init
|
|
)
|
|
{
|
|
if(Init) {
|
|
PlatformPathOverride = NULL;
|
|
} else {
|
|
if( PlatformPathOverride ) {
|
|
MyFree(PlatformPathOverride);
|
|
PlatformPathOverride = NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version
|
|
//
|
|
BOOL
|
|
SetupSetPlatformPathOverrideA(
|
|
IN PCSTR Override OPTIONAL
|
|
)
|
|
{
|
|
BOOL b;
|
|
DWORD rc;
|
|
PCWSTR p;
|
|
|
|
if(Override) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(Override,&p);
|
|
} else {
|
|
p = NULL;
|
|
rc = NO_ERROR;
|
|
}
|
|
|
|
if(rc == NO_ERROR) {
|
|
b = SetupSetPlatformPathOverrideW(p);
|
|
rc = GetLastError();
|
|
} else {
|
|
b = FALSE;
|
|
}
|
|
|
|
if(p) {
|
|
MyFree(p);
|
|
}
|
|
|
|
SetLastError(rc);
|
|
return(b);
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub
|
|
//
|
|
BOOL
|
|
SetupSetPlatformPathOverrideW(
|
|
IN PCWSTR Override OPTIONAL
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(Override);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
BOOL
|
|
SetupSetPlatformPathOverride(
|
|
IN PCTSTR Override OPTIONAL
|
|
)
|
|
{
|
|
BOOL b = FALSE;
|
|
DWORD rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
BOOL locked = FALSE;
|
|
|
|
try {
|
|
EnterCriticalSection(&PlatformPathOverrideCritSect);
|
|
locked = TRUE;
|
|
rc = ERROR_INVALID_DATA;
|
|
|
|
if(Override) {
|
|
if(PlatformPathOverride) {
|
|
MyFree(PlatformPathOverride);
|
|
PlatformPathOverride = NULL;
|
|
}
|
|
|
|
try {
|
|
b = ((PlatformPathOverride = DuplicateString(Override)) != NULL);
|
|
if(!b) {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
b = FALSE;
|
|
rc = ERROR_INVALID_PARAMETER;
|
|
}
|
|
} else {
|
|
if(PlatformPathOverride) {
|
|
MyFree(PlatformPathOverride);
|
|
PlatformPathOverride = NULL;
|
|
}
|
|
b = TRUE;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
}
|
|
|
|
if(locked) {
|
|
LeaveCriticalSection(&PlatformPathOverrideCritSect);
|
|
} else {
|
|
b = FALSE;
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
|
|
if(!b) {
|
|
SetLastError(rc);
|
|
}
|
|
return(b);
|
|
}
|
|
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version
|
|
//
|
|
BOOL
|
|
SetupQueueCopyA(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCSTR SourceRootPath, OPTIONAL
|
|
IN PCSTR SourcePath, OPTIONAL
|
|
IN PCSTR SourceFilename,
|
|
IN PCSTR SourceDescription, OPTIONAL
|
|
IN PCSTR SourceTagfile, OPTIONAL
|
|
IN PCSTR TargetDirectory,
|
|
IN PCSTR TargetFilename, OPTIONAL
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
PCWSTR sourceRootPath;
|
|
PCWSTR sourcePath;
|
|
PCWSTR sourceFilename;
|
|
PCWSTR sourceDescription;
|
|
PCWSTR sourceTagfile;
|
|
PCWSTR targetDirectory;
|
|
PCWSTR targetFilename;
|
|
BOOL b;
|
|
DWORD rc;
|
|
SP_FILE_COPY_PARAMS_WEX CopyParams = {0};
|
|
|
|
sourceRootPath = NULL;
|
|
sourcePath = NULL;
|
|
sourceFilename = NULL;
|
|
sourceDescription = NULL;
|
|
sourceTagfile = NULL;
|
|
targetDirectory = NULL;
|
|
targetFilename = NULL;
|
|
rc = NO_ERROR;
|
|
b = FALSE;
|
|
|
|
if(SourceRootPath) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceRootPath,&sourceRootPath);
|
|
}
|
|
if((rc == NO_ERROR) && SourcePath) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourcePath,&sourcePath);
|
|
}
|
|
if((rc == NO_ERROR) && SourceFilename) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceFilename,&sourceFilename);
|
|
}
|
|
if((rc == NO_ERROR) && SourceDescription) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceDescription,&sourceDescription);
|
|
}
|
|
if((rc == NO_ERROR) && SourceTagfile) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceTagfile,&sourceTagfile);
|
|
}
|
|
if((rc == NO_ERROR) && TargetDirectory) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(TargetDirectory,&targetDirectory);
|
|
}
|
|
if((rc == NO_ERROR) && TargetFilename) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(TargetFilename,&targetFilename);
|
|
}
|
|
|
|
if(rc == NO_ERROR) {
|
|
|
|
CopyParams.cbSize = sizeof(SP_FILE_COPY_PARAMS_WEX);
|
|
CopyParams.QueueHandle = QueueHandle;
|
|
CopyParams.SourceRootPath = sourceRootPath;
|
|
CopyParams.SourcePath = sourcePath;
|
|
CopyParams.SourceFilename = sourceFilename;
|
|
CopyParams.SourceDescription = sourceDescription;
|
|
CopyParams.SourceTagfile = sourceTagfile;
|
|
CopyParams.TargetDirectory = targetDirectory;
|
|
CopyParams.TargetFilename = targetFilename;
|
|
CopyParams.CopyStyle = CopyStyle;
|
|
CopyParams.LayoutInf = NULL;
|
|
CopyParams.SecurityDescriptor= NULL;
|
|
|
|
b = _SetupQueueCopy(&CopyParams, NULL, NULL);
|
|
rc = GetLastError();
|
|
}
|
|
|
|
if(sourceRootPath) {
|
|
MyFree(sourceRootPath);
|
|
}
|
|
if(sourcePath) {
|
|
MyFree(sourcePath);
|
|
}
|
|
if(sourceFilename) {
|
|
MyFree(sourceFilename);
|
|
}
|
|
if(sourceDescription) {
|
|
MyFree(sourceDescription);
|
|
}
|
|
if(sourceTagfile) {
|
|
MyFree(sourceTagfile);
|
|
}
|
|
if(targetDirectory) {
|
|
MyFree(targetDirectory);
|
|
}
|
|
if(targetFilename) {
|
|
MyFree(targetFilename);
|
|
}
|
|
|
|
SetLastError(rc);
|
|
return(b);
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub
|
|
//
|
|
BOOL
|
|
SetupQueueCopyW(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCWSTR SourceRootPath, OPTIONAL
|
|
IN PCWSTR SourcePath, OPTIONAL
|
|
IN PCWSTR SourceFilename,
|
|
IN PCWSTR SourceDescription, OPTIONAL
|
|
IN PCWSTR SourceTagfile, OPTIONAL
|
|
IN PCWSTR TargetDirectory,
|
|
IN PCWSTR TargetFilename, OPTIONAL
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(QueueHandle);
|
|
UNREFERENCED_PARAMETER(SourceRootPath);
|
|
UNREFERENCED_PARAMETER(SourcePath);
|
|
UNREFERENCED_PARAMETER(SourceFilename);
|
|
UNREFERENCED_PARAMETER(SourceDescription);
|
|
UNREFERENCED_PARAMETER(SourceTagfile);
|
|
UNREFERENCED_PARAMETER(TargetDirectory);
|
|
UNREFERENCED_PARAMETER(TargetFilename);
|
|
UNREFERENCED_PARAMETER(CopyStyle);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
BOOL
|
|
SetupQueueCopy(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCTSTR SourceRootPath, OPTIONAL
|
|
IN PCTSTR SourcePath, OPTIONAL
|
|
IN PCTSTR SourceFilename,
|
|
IN PCTSTR SourceDescription, OPTIONAL
|
|
IN PCTSTR SourceTagfile, OPTIONAL
|
|
IN PCTSTR TargetDirectory,
|
|
IN PCTSTR TargetFilename, OPTIONAL
|
|
IN DWORD CopyStyle
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Place a copy operation on a setup file queue.
|
|
|
|
Arguments:
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
SourceRootPath - Supplies the root of the source for this copy,
|
|
such as A:\ or \\FOO\BAR\BAZ. If this parameter isn't supplied, then
|
|
this queue node will be added to a media descriptor's queue that
|
|
matches on SourceDescription and SourceTagfile. (This merge will take
|
|
place regardless of whether or not the media descriptor entry was
|
|
already in the queue prior to calling SetupQueueCopy.)
|
|
|
|
If there is no matching media descriptor that contains SourceRootPath
|
|
information, the path will be set to the directory where the system was
|
|
installed from.
|
|
|
|
SourcePath - if specified, supplies the path relative to SourceRootPath
|
|
where the file can be found.
|
|
|
|
SourceFilename - supplies the filename part of the file to be copied.
|
|
|
|
SourceDescription - if specified, supplies a description of the source
|
|
media, to be used during disk prompts.
|
|
|
|
SourceTagfile - if specified, supplies a tag file whose presence at
|
|
SourceRootPath indicates the presence of the source media.
|
|
If not specified, the file itself will be used as the tag file
|
|
if required (tagfiles are used only for removable media).
|
|
|
|
TargetDirectory - supplies the directory where the file is to be copied.
|
|
|
|
TargetFilename - if specified, supplies the name of the target file.
|
|
If not specified, the target file will have the same name as the source.
|
|
|
|
CopyStyle - supplies flags that control the behavior of the copy operation
|
|
for this file.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, GetLastError() returns
|
|
extended error information.
|
|
|
|
--*/
|
|
|
|
{
|
|
SP_FILE_COPY_PARAMSEX CopyParams = {0};
|
|
|
|
//
|
|
// Work is done by common worker routine
|
|
//
|
|
CopyParams.cbSize = sizeof(SP_FILE_COPY_PARAMSEX);
|
|
CopyParams.QueueHandle = QueueHandle;
|
|
CopyParams.SourceRootPath = SourceRootPath;
|
|
CopyParams.SourcePath = SourcePath;
|
|
CopyParams.SourceFilename = SourceFilename;
|
|
CopyParams.SourceDescription = SourceDescription;
|
|
CopyParams.SourceTagfile = SourceTagfile;
|
|
CopyParams.TargetDirectory = TargetDirectory;
|
|
CopyParams.TargetFilename = TargetFilename;
|
|
CopyParams.CopyStyle = CopyStyle;
|
|
CopyParams.LayoutInf = NULL;
|
|
CopyParams.SecurityDescriptor= NULL;
|
|
CopyParams.CacheName = NULL;
|
|
|
|
return(_SetupQueueCopy(&CopyParams, NULL, NULL));
|
|
}
|
|
|
|
|
|
BOOL
|
|
_SetupQueueCopy(
|
|
IN PSP_FILE_COPY_PARAMSEX CopyParams,
|
|
IN PINFCONTEXT LayoutLineContext, OPTIONAL
|
|
IN HINF AdditionalInfs OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Worker routine for SetupQueueCopy and friends.
|
|
|
|
Arguments:
|
|
|
|
CopyParams - supplies a structure with information about the file
|
|
to be queued. Fields are used as follows.
|
|
|
|
cbSize - must be sizeof(SP_FILE_COPY_PARAMS). The caller should
|
|
have verified this before calling this routine.
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
SourceRootPath - Supplies the root of the source for this copy,
|
|
such as A:\ or \\FOO\BAR\BAZ. If this field is NULL, then this
|
|
queue node will be added to a media descriptor's queue that matches
|
|
on SourceDescription and SourceTagfile. (This merge will take
|
|
place regardless of whether or not the media descriptor entry was
|
|
already in the queue prior to calling SetupQueueCopy.)
|
|
|
|
If there is no matching media descriptor that contains
|
|
SourceRootPath information, the path will be set to the directory
|
|
where the system was installed from.
|
|
|
|
SourcePath - if specified, supplies the path relative to SourceRootPath
|
|
where the file can be found.
|
|
|
|
SourceFilename - supplies the filename part of the file to be copied.
|
|
|
|
SourceDescription - if specified, supplies a description of the source
|
|
media, to be used during disk prompts.
|
|
|
|
SourceTagfile - if specified, supplies a tag file whose presence at
|
|
SourceRootPath indicates the presence of the source media.
|
|
If not specified, the file itself will be used as the tag file
|
|
if required (tagfiles are used only for removable media).
|
|
|
|
TargetDirectory - supplies the directory where the file is to be copied.
|
|
|
|
TargetFilename - if specified, supplies the name of the target file. If
|
|
not specified, the target file will have the same name as the source.
|
|
|
|
CopyStyle - supplies flags that control the behavior of the copy
|
|
operation for this file.
|
|
|
|
LayoutInf - supplies the handle to the inf which contains the source
|
|
layout info for this file, if any.
|
|
|
|
LayoutLineContext - if specified, this argument provides the INF context
|
|
for the [SourceDisksFiles] entry pertaining to the file to be copied.
|
|
If not specified, the relevant [SourceDisksFiles] entry will be searched
|
|
for in the LayoutInf handle specified in the CopyParams structure. This
|
|
context must be contained within either the CopyParams->LayoutInf or
|
|
AdditionalInfs loaded INF handles (because those are the two INFs we're
|
|
gonna lock). The argument is used to prevent us from having to search
|
|
for the file to be copied in a [SourceDisksFiles] section. The caller
|
|
has already done that, and is either handing us the context to that INF
|
|
entry, or has passed -1 to indicate that there is no [SourceDisksFiles]
|
|
entry.
|
|
|
|
AdditionalInfs - if specified, supplies an additional HINF (potentially
|
|
containing multiple append-loaded INFs) that need to be added to our
|
|
SPQ_CATALOG_INFO list for later validation. Do not supply this parameter
|
|
if it is identical to the value of the LayoutInf field in the CopyParams
|
|
structure.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, GetLastError() returns
|
|
extended error information.
|
|
|
|
--*/
|
|
|
|
{
|
|
PSP_FILE_QUEUE Queue;
|
|
PSP_FILE_QUEUE_NODE QueueNode, TempNode, PrevQueueNode;
|
|
PSOURCE_MEDIA_INFO Source;
|
|
TCHAR TempBuffer[MAX_PATH];
|
|
TCHAR TempSubDir[MAX_PATH];
|
|
TCHAR SourceCabfileBuffer[MAX_PATH];
|
|
TCHAR SourceTagfile2Buffer[MAX_PATH];
|
|
TCHAR DriverCache[MAX_PATH] = {0};
|
|
PCTSTR LastPathPart;
|
|
PCTSTR p;
|
|
int Size;
|
|
DWORD d;
|
|
HINF LayoutInfHandle;
|
|
INFCONTEXT LineContext;
|
|
BOOL b;
|
|
PSPQ_CATALOG_INFO CatalogNode, PrevCatalogNode, LastOldCatalogNode;
|
|
LONG l1,l2, l3, l4;
|
|
PLOADED_INF pLoadedInfs[2];
|
|
DWORD LoadedInfCount, i;
|
|
PLOADED_INF pCurLoadedInf;
|
|
DWORD MediaFlags;
|
|
PCTSTR SourcePath, SourceRootPath;
|
|
PCTSTR SourceTagfile = NULL;
|
|
PCTSTR SourceCabfile = NULL;
|
|
#if defined(_X86_)
|
|
BOOL ForcePlatform = FALSE;
|
|
#endif
|
|
UINT SourceFlags = 0;
|
|
PINFCONTEXT pContext = LayoutLineContext;
|
|
INFCONTEXT tmpContext,tmpContext2;
|
|
UINT SourceId = 0;
|
|
BOOL locked = FALSE;
|
|
|
|
d = NO_ERROR;
|
|
LoadedInfCount = 0;
|
|
MediaFlags = 0;
|
|
|
|
try {
|
|
MYASSERT(CopyParams->cbSize == sizeof(SP_FILE_COPY_PARAMSEX));
|
|
Queue = (PSP_FILE_QUEUE)CopyParams->QueueHandle;
|
|
if (Queue->Signature != SP_FILE_QUEUE_SIG) {
|
|
d = ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
LayoutInfHandle = CopyParams->LayoutInf;
|
|
|
|
//
|
|
// Maintain local pointers to the SourceRootPath and SourcePath strings,
|
|
// since we may be modifying them, and we don't want to muck with the
|
|
// caller-supplied buffer.
|
|
//
|
|
SourcePath = CopyParams->SourcePath;
|
|
if(CopyParams->SourceRootPath) {
|
|
SourceRootPath = CopyParams->SourceRootPath;
|
|
} else {
|
|
SourceRootPath = SystemSourcePath;
|
|
MediaFlags |= SMI_FLAG_NO_SOURCE_ROOT_PATH;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
d = ERROR_INVALID_PARAMETER;
|
|
}
|
|
if(d != NO_ERROR) {
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Make sure that we weren't passed the same HINF in both CopyParams->LayoutInf
|
|
// and AdditionalInfs (just adds redundant work to process the same
|
|
// LOADED_INF list twice).
|
|
//
|
|
MYASSERT(!LayoutInfHandle || (LayoutInfHandle != AdditionalInfs));
|
|
|
|
//
|
|
// Lock inf(s). We do a whole bunch of operations on the inf later,
|
|
// and we don't want anything changing out from under us.
|
|
//
|
|
if(LayoutInfHandle) {
|
|
if(LockInf((PLOADED_INF)LayoutInfHandle)) {
|
|
pLoadedInfs[LoadedInfCount++] = (PLOADED_INF)LayoutInfHandle;
|
|
} else {
|
|
d = ERROR_INVALID_HANDLE;
|
|
goto clean0;
|
|
}
|
|
}
|
|
|
|
if(AdditionalInfs) {
|
|
if(LockInf((PLOADED_INF)AdditionalInfs)) {
|
|
pLoadedInfs[LoadedInfCount++] = (PLOADED_INF)AdditionalInfs;
|
|
} else {
|
|
d = ERROR_INVALID_HANDLE;
|
|
goto clean0;
|
|
}
|
|
}
|
|
|
|
if(!(Queue->Flags & FQF_DEVICE_INSTALL)) {
|
|
//
|
|
// Look through all the INFs to see if any of them are device INFs.
|
|
//
|
|
for(i = 0; i < LoadedInfCount; i++) {
|
|
|
|
if(IsInfForDeviceInstall(Queue->LogContext,
|
|
NULL,
|
|
pLoadedInfs[i],
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL)) {
|
|
//
|
|
// There be device INFs here! Mark the queue accordingly.
|
|
//
|
|
d = MarkQueueForDeviceInstall(CopyParams->QueueHandle,
|
|
(HINF)(pLoadedInfs[i]),
|
|
NULL
|
|
);
|
|
if(d == NO_ERROR) {
|
|
break;
|
|
} else {
|
|
goto clean0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// check if we already have a line context for the file we're adding
|
|
// and if we don't, then try to fetch it
|
|
//
|
|
if (!LayoutLineContext || LayoutLineContext == (PINFCONTEXT) -1) {
|
|
if ((LayoutInfHandle == (PINFCONTEXT) -1) || (LayoutInfHandle == NULL)) {
|
|
pContext = NULL;
|
|
} else {
|
|
//
|
|
// find the sourcerootpaths section
|
|
//
|
|
b = _SetupGetSourceFileLocation(
|
|
LayoutInfHandle,
|
|
NULL,
|
|
CopyParams->SourceFilename,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&tmpContext // location in SourceDisksFiles
|
|
);
|
|
|
|
pContext = b ? &tmpContext : NULL;
|
|
}
|
|
}
|
|
if(pContext) {
|
|
//
|
|
// now obtain the source id (file,*disk*,...)
|
|
//
|
|
SetupGetIntField(pContext,1,&SourceId);
|
|
}
|
|
|
|
//
|
|
// if we have a NULL source path, then we should check 2 things:
|
|
// 1) source flag information, which indicates if we have a service
|
|
// pack or CDM source location.
|
|
// 2) if the relative source path is null, we'll go looking for
|
|
// a relative path in the inf file, just in case the caller didn't
|
|
// supply it
|
|
//
|
|
// note that we use the SourceFlagsSet item in the COPY_FILE structure
|
|
// to optimize this path -- the first item contains the source flags,
|
|
// and the second item indicates that we shouldn't bother looking for
|
|
// any information, we've already done a search
|
|
//
|
|
if (CopyParams->SourceFlagsSet) {
|
|
SourceFlags = CopyParams->SourceFlags;
|
|
} else if (pContext && LayoutInfHandle) {
|
|
TCHAR data[32];
|
|
|
|
if(pSetupGetSourceInfo(LayoutInfHandle,
|
|
pContext,
|
|
SourceId,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
SRCINFO_FLAGS,
|
|
data,
|
|
SIZECHARS(data),
|
|
NULL)) {
|
|
|
|
pAToI(data,&SourceFlags);
|
|
}
|
|
}
|
|
|
|
if (MediaFlags & SMI_FLAG_NO_SOURCE_ROOT_PATH) {
|
|
if(pContext
|
|
&& LayoutInfHandle
|
|
&& !CopyParams->SourceFlagsSet
|
|
&& !SourcePath
|
|
&& pSetupGetSourceInfo(LayoutInfHandle,
|
|
pContext,
|
|
SourceId,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
SRCINFO_PATH,
|
|
TempSubDir,
|
|
SIZECHARS(TempSubDir),
|
|
NULL)) {
|
|
|
|
SourcePath = TempSubDir;
|
|
}
|
|
//
|
|
// override the system source path with the servicepack source path
|
|
// if the flags are set
|
|
//
|
|
if (SourceFlags & SRC_FLAGS_SVCPACK_SOURCE) {
|
|
MediaFlags |= SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH;
|
|
SourceRootPath = ServicePackSourcePath;
|
|
}
|
|
|
|
}
|
|
//
|
|
// now determine tag file vs cab file
|
|
//
|
|
SourceTagfile = CopyParams->SourceTagfile;
|
|
if (LayoutInfHandle && pContext && (SourceFlags & SRC_FLAGS_CABFILE)) {
|
|
//
|
|
// the given tagfile is really a cabfile, we may have optionally a real tagfile
|
|
//
|
|
SourceCabfile = CopyParams->SourceTagfile;
|
|
if(SourceCabfile == NULL || SourceCabfile[0]==TEXT('\0')) {
|
|
//
|
|
// cab name hasn't been determined yet
|
|
//
|
|
if(pSetupGetSourceInfo(LayoutInfHandle,
|
|
pContext,
|
|
SourceId,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
SRCINFO_TAGFILE,
|
|
SourceCabfileBuffer,
|
|
SIZECHARS(SourceCabfileBuffer),
|
|
NULL
|
|
)) {
|
|
SourceCabfile = SourceCabfileBuffer;
|
|
}
|
|
}
|
|
|
|
if(SourceCabfile == NULL || SourceCabfile[0]==TEXT('\0')) {
|
|
//
|
|
// cabfilename is erroneous
|
|
//
|
|
SourceCabfile = SourceTagfile = NULL;
|
|
|
|
} else if(pSetupGetSourceInfo(LayoutInfHandle,
|
|
pContext,
|
|
SourceId,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
SRCINFO_TAGFILE2,
|
|
SourceTagfile2Buffer,
|
|
SIZECHARS(SourceTagfile2Buffer),
|
|
NULL
|
|
)) {
|
|
SourceTagfile = SourceTagfile2Buffer;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Allocate a queue structure.
|
|
//
|
|
QueueNode = MyMalloc(sizeof(SP_FILE_QUEUE_NODE));
|
|
if(!QueueNode) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
|
|
//
|
|
// Operation is copy.
|
|
//
|
|
QueueNode->Operation = FILEOP_COPY;
|
|
QueueNode->InternalFlags = 0;
|
|
|
|
//
|
|
// HACK ALERT!!! HACK HACK HACK!!!!
|
|
//
|
|
// There might be an override platform specified. If this is so,
|
|
// we will look for \i386, \mips, etc as the final component of the
|
|
// specified path, and replace it with the override path.
|
|
// This is a TOTAL HACK.
|
|
//
|
|
try {
|
|
EnterCriticalSection(&PlatformPathOverrideCritSect);
|
|
locked = TRUE;
|
|
if(PlatformPathOverride) {
|
|
p = SourcePath ? SourcePath : SourceRootPath;
|
|
if(LastPathPart = _tcsrchr(p,L'\\')) {
|
|
LastPathPart++;
|
|
} else {
|
|
LastPathPart = p;
|
|
}
|
|
#if defined(_AXP64_)
|
|
if(!lstrcmpi(LastPathPart,TEXT("axp64"))) {
|
|
#elif defined(_ALPHA_)
|
|
if(!lstrcmpi(LastPathPart,TEXT("alpha"))) {
|
|
#elif defined(_AMD64_)
|
|
if(!lstrcmpi(LastPathPart,TEXT("amd64"))) {
|
|
#elif defined(_X86_)
|
|
//
|
|
// NEC98
|
|
//
|
|
// During GUI setup, source path on local disk must be "nec98",
|
|
// so we don't override "i386".
|
|
//
|
|
if (IsNEC98()) {
|
|
HKEY hKey;
|
|
DWORD DataType, DataSize;
|
|
PTSTR ForceOverride;
|
|
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, TEXT("SYSTEM\\Setup"), 0, KEY_READ, &hKey) == ERROR_SUCCESS) {
|
|
if (QueryRegistryValue(hKey, TEXT("ForcePlatform"), &ForceOverride, &DataType, &DataSize)
|
|
== NO_ERROR) {
|
|
ForcePlatform = TRUE;
|
|
MyFree(ForceOverride);
|
|
}
|
|
RegCloseKey(hKey);
|
|
}
|
|
// If use driver cache, There is not i386 driver cache on NEC98.
|
|
if ((CopyParams->CopyStyle & PSP_COPY_USE_DRIVERCACHE) && !lstrcmpi(PlatformPathOverride,TEXT("i386"))) {
|
|
ForcePlatform = TRUE;
|
|
}
|
|
}
|
|
if((!IsNEC98() && (!lstrcmpi(LastPathPart,TEXT("x86")) || !lstrcmpi(LastPathPart,TEXT("i386"))))
|
|
|| (IsNEC98() && (!lstrcmpi(LastPathPart,TEXT("nec98")) && !ForcePlatform))) {
|
|
#elif defined(_IA64_)
|
|
if(!lstrcmpi(LastPathPart,TEXT("ia64"))) {
|
|
#endif
|
|
Size = (int)(LastPathPart - p);
|
|
Size = min(Size,MAX_PATH);
|
|
Size *= sizeof(TCHAR);
|
|
|
|
CopyMemory(TempBuffer,p,Size);
|
|
TempBuffer[Size/sizeof(TCHAR)] = 0;
|
|
|
|
//
|
|
// If the path was something like "mips" then TempBuffer
|
|
// will be empty and we don't want to introduce any extra
|
|
// backslashes.
|
|
//
|
|
if(*TempBuffer) {
|
|
pSetupConcatenatePaths(TempBuffer,PlatformPathOverride,MAX_PATH,NULL);
|
|
} else {
|
|
lstrcpyn(TempBuffer,PlatformPathOverride,MAX_PATH);
|
|
}
|
|
|
|
if(SourcePath) {
|
|
SourcePath = TempBuffer;
|
|
} else {
|
|
SourceRootPath = TempBuffer;
|
|
}
|
|
}
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
d = ERROR_INVALID_PARAMETER;
|
|
}
|
|
if(locked) {
|
|
LeaveCriticalSection(&PlatformPathOverrideCritSect);
|
|
} else {
|
|
//
|
|
// if lock not grabbed, this is the cause
|
|
//
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
if(d != NO_ERROR) {
|
|
goto clean1;
|
|
}
|
|
|
|
//
|
|
// check here if the cab-file is present on the disk.
|
|
// if it isn't, then we fall back on the current
|
|
// sourcerootpath
|
|
//
|
|
if (CopyParams->CopyStyle & PSP_COPY_USE_DRIVERCACHE) {
|
|
if (pIsDriverCachePresent(CopyParams->CacheName,
|
|
SourcePath,
|
|
DriverCache)) {
|
|
SourceRootPath = DriverCache;
|
|
MediaFlags |= SMI_FLAG_USE_LOCAL_SOURCE_CAB;
|
|
}
|
|
|
|
SourceTagfile = CopyParams->CacheName;
|
|
SourceCabfile = NULL;
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// NOTE: When adding the following strings to the string table, we cast away
|
|
// their CONST-ness to avoid a compiler warning. Since we are adding them
|
|
// case-sensitively, we are guaranteed they will not be modified.
|
|
//
|
|
try {
|
|
//
|
|
// Set up the source root path.
|
|
//
|
|
QueueNode->SourceRootPath = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)SourceRootPath,
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->SourceRootPath == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
|
|
//
|
|
// Set up the source path.
|
|
//
|
|
if(d == NO_ERROR) {
|
|
if(SourcePath) {
|
|
QueueNode->SourcePath = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)SourcePath,
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->SourcePath == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
} else {
|
|
QueueNode->SourcePath = -1;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set up the source filename.
|
|
//
|
|
if(d == NO_ERROR) {
|
|
QueueNode->SourceFilename = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)CopyParams->SourceFilename,
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->SourceFilename == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set up the target directory.
|
|
//
|
|
if(d == NO_ERROR) {
|
|
QueueNode->TargetDirectory = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)CopyParams->TargetDirectory,
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->TargetDirectory == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set up the target filename.
|
|
//
|
|
if(d == NO_ERROR) {
|
|
QueueNode->TargetFilename = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)(CopyParams->TargetFilename ? CopyParams->TargetFilename
|
|
: CopyParams->SourceFilename),
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->TargetFilename == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Set up the Security Descriptor
|
|
//
|
|
if(d == NO_ERROR) {
|
|
if( CopyParams->SecurityDescriptor){
|
|
|
|
QueueNode->SecurityDesc = pSetupStringTableAddString(
|
|
Queue->StringTable,
|
|
(PTSTR)(CopyParams->SecurityDescriptor),
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if(QueueNode->SecurityDesc == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
} else {
|
|
QueueNode->SecurityDesc = -1;
|
|
}
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
d = ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
if(d != NO_ERROR) {
|
|
goto clean1;
|
|
}
|
|
|
|
//
|
|
// Initialize a pointer to the end of the queue's current catalog info list.
|
|
// We do this so that later, we can easily back-out our changes by truncating
|
|
// the list after this node, and freeing all subsequent elements.
|
|
//
|
|
LastOldCatalogNode = Queue->CatalogList;
|
|
if(LastOldCatalogNode) {
|
|
while(LastOldCatalogNode->Next) {
|
|
LastOldCatalogNode = LastOldCatalogNode->Next;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Now process all members of our pLoadedInfs lists, adding each one to the
|
|
// SPQ_CATALOG_INFO list (avoiding duplicates entries, of course).
|
|
//
|
|
for(i = 0; i < LoadedInfCount; i++) {
|
|
|
|
for(pCurLoadedInf = pLoadedInfs[i]; pCurLoadedInf; pCurLoadedInf = pCurLoadedInf->Next) {
|
|
//
|
|
// First, get the (native) CatalogFile= entry from the version block
|
|
// of this INF member.
|
|
//
|
|
if(pSetupGetCatalogFileValue(&(pCurLoadedInf->VersionBlock),
|
|
TempBuffer,
|
|
SIZECHARS(TempBuffer),
|
|
NULL)) {
|
|
|
|
l1 = pSetupStringTableAddString(Queue->StringTable,
|
|
TempBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
if(l1 == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
} else {
|
|
//
|
|
// This INF doesn't have a CatalogFile= entry.
|
|
//
|
|
l1 = -1;
|
|
}
|
|
|
|
//
|
|
// If this file queue is currently setup for a platform override,
|
|
// then retrieve that CatalogFile= entry as well.
|
|
//
|
|
if(Queue->Flags & FQF_USE_ALT_PLATFORM) {
|
|
|
|
if(pSetupGetCatalogFileValue(&(pCurLoadedInf->VersionBlock),
|
|
TempBuffer,
|
|
SIZECHARS(TempBuffer),
|
|
&(Queue->AltPlatformInfo))) {
|
|
|
|
l3 = pSetupStringTableAddString(Queue->StringTable,
|
|
TempBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
if(l3 == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
} else {
|
|
//
|
|
// This INF doesn't have a CatalogFile= entry.
|
|
//
|
|
l3 = -1;
|
|
}
|
|
} else {
|
|
//
|
|
// We're not in a platform override scenario.
|
|
//
|
|
l3 = -1;
|
|
}
|
|
|
|
//
|
|
// Now, get the INF's full path.
|
|
//
|
|
lstrcpyn(TempBuffer, pCurLoadedInf->VersionBlock.Filename, SIZECHARS(TempBuffer));
|
|
l2 = pSetupStringTableAddString(Queue->StringTable,
|
|
TempBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
if(l2 == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
|
|
//
|
|
// Finally, retrieve the INF's original name, if different than the
|
|
// current name.
|
|
//
|
|
if(pCurLoadedInf->OriginalInfName) {
|
|
lstrcpyn(TempBuffer, pCurLoadedInf->OriginalInfName, SIZECHARS(TempBuffer));
|
|
l4 = pSetupStringTableAddString(Queue->StringTable,
|
|
TempBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
if(l4 == -1) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
|
|
} else {
|
|
//
|
|
// The INF's original name is the same as its current name.
|
|
//
|
|
l4 = -1;
|
|
}
|
|
|
|
b = TRUE;
|
|
for(PrevCatalogNode=NULL, CatalogNode=Queue->CatalogList;
|
|
CatalogNode;
|
|
CatalogNode=CatalogNode->Next) {
|
|
|
|
if(CatalogNode->InfFullPath == l2) {
|
|
//
|
|
// Already in there. No need to create a new node.
|
|
// Break out here, with CatalogNode pointing at the
|
|
// proper node for this catalog file.
|
|
//
|
|
// In this case, PrevCatalogNode should not be used later,
|
|
// but it shouldn't need to be used, since we won't be
|
|
// adding anything new onto the list of catalog nodes.
|
|
//
|
|
MYASSERT(CatalogNode->CatalogFileFromInf == l1);
|
|
MYASSERT(CatalogNode->InfOriginalName == l4);
|
|
b = FALSE;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// PrevCatalogNode will end up pointing to the final node
|
|
// currently in the linked list, in the case where we need
|
|
// to allocate a new node. This is useful so we don't have to
|
|
// traverse the list again later when we add the new catalog
|
|
// node to the list for this queue.
|
|
//
|
|
PrevCatalogNode = CatalogNode;
|
|
}
|
|
|
|
if(b) {
|
|
//
|
|
// Need to create a new catalog node.
|
|
//
|
|
CatalogNode = MyMalloc(sizeof(SPQ_CATALOG_INFO));
|
|
if(!CatalogNode) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
ZeroMemory(CatalogNode, sizeof(SPQ_CATALOG_INFO));
|
|
CatalogNode->CatalogFileFromInf = l1;
|
|
CatalogNode->InfFullPath = l2;
|
|
CatalogNode->AltCatalogFileFromInf = l3;
|
|
CatalogNode->InfOriginalName = l4;
|
|
CatalogNode->AltCatalogFileFromInfPending = -1;
|
|
CatalogNode->InfFinalPath = -1;
|
|
//
|
|
// Go ahead and link the new node into the list. If we
|
|
// encounter a failure later, we can easily back out of this by
|
|
// truncating the list, since we know we always append, and we
|
|
// remembered the original list tail.
|
|
//
|
|
if(Queue->CatalogList) {
|
|
PrevCatalogNode->Next = CatalogNode;
|
|
} else {
|
|
Queue->CatalogList = CatalogNode;
|
|
}
|
|
|
|
//
|
|
// We've successfully added a new, as yet unvalidated, catalog
|
|
// node. We must therefore reset the "catalog verifications
|
|
// done" flags so that we'll redo them later.
|
|
//
|
|
Queue->Flags &= ~(FQF_DID_CATALOGS_OK | FQF_DID_CATALOGS_FAILED);
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// At this point, all the INFs involved in this installation (i.e., all INFs
|
|
// in the HINFs we were passed in) have been added to our catalog info list,
|
|
// if they weren't already present. Now we need to figure out which one of
|
|
// them should be associated with the file to be copied.
|
|
//
|
|
// Note that we want to get the CatalogFile= line from the actual inf
|
|
// that contains source layout information for the file being queued.
|
|
// If the layout inf handle references multiple append-loaded infs,
|
|
// the simple mechanism of just looking up CatalogFile= in the [Version]
|
|
// section using the given handle might give us the value from the
|
|
// wrong inf.
|
|
//
|
|
// To deal with this, we attempt to locate the file in a [SourceDisksFiles]
|
|
// section using the given inf handle, which gives us back a line context.
|
|
// From the line context we can easily get at the [Version] section of the
|
|
// actual inf where the file's layout info is contained.
|
|
//
|
|
// This handles all cases properly. For example, a file that is shipped by
|
|
// a vendor that replaces one of our files. If the OEM's inf has a
|
|
// SourceDisksFiles section with the file in it, it will be found first
|
|
// when we look the file up using the given inf handle because of the way
|
|
// inf append-loading works.
|
|
//
|
|
// If we cannot find the file in a [SourceDisksFiles] section (such as
|
|
// if there is no such section), then we can't associate the file to be
|
|
// copied with any INF/CAT. If we do find a [SourceDisksFiles] entry, but
|
|
// the containing INF doesn't specify a CatalogFile= entry, then we'll go
|
|
// ahead and associate that with a SPQ_CATALOG_INFO node for that INF, but
|
|
// that catalog info node will have a CatalogFileFromInf field of -1.
|
|
// That's OK for system-provided INFs, but it will fail validation if it's
|
|
// an OEM INF (this check is done later in _SetupVerifyQueuedCatalogs).
|
|
//
|
|
if(LayoutInfHandle || LayoutLineContext) {
|
|
//
|
|
// If we already have a valid layout line context, we don't need to go
|
|
// looking for the file in [SourceDisksFiles] again (the caller is
|
|
// assumed to have done that already). The caller might also have told
|
|
// us that he *knows* that there is no [SourceDisksFiles] by passing us
|
|
// a LayoutLineContext of -1.
|
|
//
|
|
if(LayoutLineContext == (PINFCONTEXT)(-1)) {
|
|
//
|
|
// For driver signing purposes, this may be an invalid file copy,
|
|
// because it's being copied by an INF that contains no source
|
|
// media information, nor does it use a layout file to supply such
|
|
// information.
|
|
//
|
|
// Since we don't have a LayoutLineContext, we don't know exactly
|
|
// which INF contained the CopyFile directive that initiated this
|
|
// copy. However, since the context is -1, that means that it was
|
|
// INF based (i.e., as opposed to being manually queued up via
|
|
// SetupQueueCopy). Therefore, we scan all the INFs passed into
|
|
// this routine (i.e., all the INFs in the pLoadedInfs lists), and
|
|
// check to see if they're all located in %windir%\Inf. If any of
|
|
// them aren't, then we mark this copynode such that later it will
|
|
// result in a signature verification failure of
|
|
// ERROR_NO_CATALOG_FOR_OEM_INF.
|
|
//
|
|
for(i = 0; i < LoadedInfCount; i++) {
|
|
|
|
for(pCurLoadedInf = pLoadedInfs[i]; pCurLoadedInf; pCurLoadedInf = pCurLoadedInf->Next) {
|
|
|
|
if(pSetupInfIsFromOemLocation(pCurLoadedInf->VersionBlock.Filename,
|
|
TRUE)) {
|
|
//
|
|
// INF doesn't exist in %windir%\Inf--mark the copynode
|
|
// for codesigning verification failure.
|
|
//
|
|
QueueNode->InternalFlags |= IQF_FROM_BAD_OEM_INF;
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Even if the INF does exist in %windir%\Inf, it might have
|
|
// originally been an OEM INF that was installed here--check
|
|
// its original filename to be sure...
|
|
//
|
|
if(pCurLoadedInf->OriginalInfName &&
|
|
pSetupInfIsFromOemLocation(pCurLoadedInf->OriginalInfName, TRUE)) {
|
|
//
|
|
// INF was an OEM INF--in this case, too, we need to
|
|
// mark the copynode for codesigning verification failure.
|
|
//
|
|
QueueNode->InternalFlags |= IQF_FROM_BAD_OEM_INF;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(QueueNode->InternalFlags & IQF_FROM_BAD_OEM_INF) {
|
|
//
|
|
// We found an OEM INF--no need to look any further.
|
|
//
|
|
break;
|
|
}
|
|
}
|
|
|
|
LayoutLineContext = NULL;
|
|
|
|
} else {
|
|
if(!LayoutLineContext) {
|
|
b = _SetupGetSourceFileLocation(
|
|
LayoutInfHandle,
|
|
NULL,
|
|
CopyParams->SourceFilename,
|
|
(Queue->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(Queue->AltPlatformInfo)
|
|
: NULL,
|
|
NULL,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&LineContext
|
|
);
|
|
|
|
LayoutLineContext = b ? &LineContext : NULL;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// At this point, a non-NULL LayoutLineContext indicates that we found an
|
|
// INF to associate with the file to be copied (via a [SourceDisksFiles]
|
|
// entry).
|
|
//
|
|
if(LayoutLineContext) {
|
|
|
|
pSetupGetPhysicalInfFilepath(LayoutLineContext,
|
|
TempBuffer,
|
|
SIZECHARS(TempBuffer)
|
|
);
|
|
|
|
l2 = pSetupStringTableAddString(Queue->StringTable,
|
|
TempBuffer,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
//
|
|
// This INF path should already be in the string table, and since we're
|
|
// supplying a writeable buffer, there's no need for memory allocation.
|
|
// Thus the addition of this string can't fail.
|
|
//
|
|
MYASSERT(l2 != -1);
|
|
|
|
for(CatalogNode=Queue->CatalogList; CatalogNode; CatalogNode=CatalogNode->Next) {
|
|
if(CatalogNode->InfFullPath == l2) {
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// This node had better already be in the list!
|
|
//
|
|
MYASSERT(CatalogNode);
|
|
|
|
QueueNode->CatalogInfo = CatalogNode;
|
|
|
|
} else {
|
|
//
|
|
// There really is no catalog info.
|
|
//
|
|
QueueNode->CatalogInfo = NULL;
|
|
}
|
|
|
|
//
|
|
// Unlock the INF(s) here, since the code below potentially returns without
|
|
// hitting the final clean-up code at the bottom of the routine.
|
|
//
|
|
for(i = 0; i < LoadedInfCount; i++) {
|
|
UnlockInf(pLoadedInfs[i]);
|
|
}
|
|
LoadedInfCount = 0;
|
|
|
|
//
|
|
// Set up the copy style flags
|
|
//
|
|
QueueNode->StyleFlags = CopyParams->CopyStyle;
|
|
QueueNode->Next = NULL;
|
|
|
|
//
|
|
// Set up the source media.
|
|
//
|
|
try {
|
|
Source = pSetupQueueSourceMedia(
|
|
Queue,
|
|
QueueNode,
|
|
QueueNode->SourceRootPath,
|
|
CopyParams->SourceDescription,
|
|
SourceTagfile,
|
|
SourceCabfile,
|
|
MediaFlags
|
|
);
|
|
if(!Source) {
|
|
d = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
d = ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
if(d != NO_ERROR) {
|
|
goto clean2;
|
|
}
|
|
|
|
//
|
|
// Link the node onto the end of the copy queue for this source media.
|
|
//
|
|
if(Source->CopyQueue) {
|
|
//
|
|
// Check to see if this same copy operation has already been enqueued
|
|
// for this source media, and if so, get rid of the new one, to avoid
|
|
// duplicates. NOTE: We don't check the "InternalFlags" field, since
|
|
// if the node already exists in the queue (based on all the other
|
|
// fields comparing successfully), then any internal flags that were set
|
|
// on the previously-existing node should be preserved. (I.e., our new
|
|
// node always is created with InternalFlags set to zero, except for
|
|
// possibly IQF_FROM_BAD_OEM_INF, which we'll OR into the original
|
|
// queue node's InternalFlags, if necessary.)
|
|
//
|
|
for(TempNode=Source->CopyQueue, PrevQueueNode = NULL;
|
|
TempNode;
|
|
PrevQueueNode = TempNode, TempNode=TempNode->Next) {
|
|
|
|
if((TempNode->SourceRootPath == QueueNode->SourceRootPath) &&
|
|
(TempNode->SourcePath == QueueNode->SourcePath) &&
|
|
(TempNode->SourceFilename == QueueNode->SourceFilename) &&
|
|
(TempNode->TargetDirectory == QueueNode->TargetDirectory) &&
|
|
(TempNode->TargetFilename == QueueNode->TargetFilename) &&
|
|
(TempNode->StyleFlags == QueueNode->StyleFlags) &&
|
|
(TempNode->CatalogInfo == QueueNode->CatalogInfo)) {
|
|
//
|
|
// We have a duplicate. OR in the IQF_FROM_BAD_OEM_INF flag
|
|
// from our present queue node, if necessary, into the existing
|
|
// queue node's InternalFlags.
|
|
//
|
|
if(QueueNode->InternalFlags & IQF_FROM_BAD_OEM_INF) {
|
|
TempNode->InternalFlags |= IQF_FROM_BAD_OEM_INF;
|
|
}
|
|
|
|
//
|
|
// Now kill the newly-created queue node and return success.
|
|
//
|
|
MyFree(QueueNode);
|
|
return TRUE;
|
|
}
|
|
}
|
|
MYASSERT(PrevQueueNode);
|
|
PrevQueueNode->Next = QueueNode;
|
|
} else {
|
|
Source->CopyQueue = QueueNode;
|
|
}
|
|
|
|
Queue->CopyNodeCount++;
|
|
Source->CopyNodeCount++;
|
|
|
|
return TRUE;
|
|
|
|
clean2:
|
|
//
|
|
// Truncate the catalog info node list at its original tail, and free all
|
|
// subsequent (newly-added) nodes.
|
|
//
|
|
if(LastOldCatalogNode) {
|
|
while(LastOldCatalogNode->Next) {
|
|
CatalogNode = LastOldCatalogNode->Next;
|
|
LastOldCatalogNode->Next = CatalogNode->Next;
|
|
MyFree(CatalogNode);
|
|
}
|
|
}
|
|
|
|
clean1:
|
|
MyFree(QueueNode);
|
|
|
|
clean0:
|
|
for(i = 0; i < LoadedInfCount; i++) {
|
|
UnlockInf(pLoadedInfs[i]);
|
|
}
|
|
|
|
SetLastError(d);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version
|
|
//
|
|
BOOL
|
|
SetupQueueCopyIndirectA(
|
|
IN PSP_FILE_COPY_PARAMS_A CopyParams
|
|
)
|
|
{
|
|
SP_FILE_COPY_PARAMS_WEX copyParams;
|
|
DWORD rc;
|
|
BOOL b;
|
|
|
|
ZeroMemory(©Params,sizeof(SP_FILE_COPY_PARAMS_W));
|
|
rc = NO_ERROR;
|
|
b = FALSE;
|
|
|
|
try {
|
|
if(CopyParams->cbSize == sizeof(SP_FILE_COPY_PARAMS_W)) {
|
|
copyParams.QueueHandle = CopyParams->QueueHandle;
|
|
copyParams.CopyStyle = CopyParams->CopyStyle;
|
|
copyParams.LayoutInf = CopyParams->LayoutInf;
|
|
copyParams.SecurityDescriptor = NULL;
|
|
} else {
|
|
rc = ERROR_INVALID_PARAMETER;
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->SourceRootPath) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->SourceRootPath,©Params.SourceRootPath);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->SourcePath) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->SourcePath,©Params.SourcePath);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->SourceFilename) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->SourceFilename,©Params.SourceFilename);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->SourceDescription) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->SourceDescription,©Params.SourceDescription);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->SourceTagfile) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->SourceTagfile,©Params.SourceTagfile);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->TargetDirectory) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->TargetDirectory,©Params.TargetDirectory);
|
|
}
|
|
if((rc == NO_ERROR) && CopyParams->TargetFilename) {
|
|
rc = pSetupCaptureAndConvertAnsiArg(CopyParams->TargetFilename,©Params.TargetFilename);
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
//
|
|
// This is to catch the case where the CopyParams pointer goes bad.
|
|
//
|
|
rc = ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
if(rc == NO_ERROR) {
|
|
|
|
copyParams.cbSize = sizeof(SP_FILE_COPY_PARAMS_WEX);
|
|
|
|
b = _SetupQueueCopy(©Params, NULL, NULL);
|
|
rc = GetLastError();
|
|
}
|
|
|
|
if(copyParams.SourceRootPath) {
|
|
MyFree(copyParams.SourceRootPath);
|
|
}
|
|
if(copyParams.SourcePath) {
|
|
MyFree(copyParams.SourcePath);
|
|
}
|
|
if(copyParams.SourceFilename) {
|
|
MyFree(copyParams.SourceFilename);
|
|
}
|
|
if(copyParams.SourceDescription) {
|
|
MyFree(copyParams.SourceDescription);
|
|
}
|
|
if(copyParams.SourceTagfile) {
|
|
MyFree(copyParams.SourceTagfile);
|
|
}
|
|
if(copyParams.TargetDirectory) {
|
|
MyFree(copyParams.TargetDirectory);
|
|
}
|
|
if(copyParams.TargetFilename) {
|
|
MyFree(copyParams.TargetFilename);
|
|
}
|
|
|
|
SetLastError(rc);
|
|
return(b);
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub
|
|
//
|
|
BOOL
|
|
SetupQueueCopyIndirectW(
|
|
IN PSP_FILE_COPY_PARAMS_W CopyParams
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(CopyParams);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
BOOL
|
|
SetupQueueCopyIndirect(
|
|
IN PSP_FILE_COPY_PARAMS CopyParams
|
|
)
|
|
{
|
|
BOOL b;
|
|
SP_FILE_COPY_PARAMSEX copyParamsEx = {0};
|
|
|
|
//
|
|
// All work is done by an internal subroutine.
|
|
// The only thing we need to do here is validate the size
|
|
// of the structure we've been given by the caller.
|
|
//
|
|
try {
|
|
b = (CopyParams->cbSize == sizeof(SP_FILE_COPY_PARAMS));
|
|
if (b) {
|
|
CopyMemory(©ParamsEx,CopyParams,sizeof(SP_FILE_COPY_PARAMS));
|
|
copyParamsEx.cbSize = sizeof(SP_FILE_COPY_PARAMSEX);
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
b = FALSE;
|
|
}
|
|
|
|
if(b) {
|
|
b = _SetupQueueCopy(©ParamsEx, NULL, NULL);
|
|
} else {
|
|
SetLastError(ERROR_INVALID_PARAMETER);
|
|
}
|
|
|
|
return(b);
|
|
}
|
|
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version
|
|
//
|
|
BOOL
|
|
SetupQueueCopySectionA(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCSTR SourceRootPath,
|
|
IN HINF InfHandle,
|
|
IN HINF ListInfHandle, OPTIONAL
|
|
IN PCSTR Section,
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
PWSTR sourcerootpath;
|
|
PWSTR section;
|
|
DWORD rc;
|
|
BOOL b;
|
|
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceRootPath,&sourcerootpath);
|
|
if(rc != NO_ERROR) {
|
|
SetLastError(rc);
|
|
return(FALSE);
|
|
}
|
|
rc = pSetupCaptureAndConvertAnsiArg(Section,§ion);
|
|
if(rc != NO_ERROR) {
|
|
MyFree(sourcerootpath);
|
|
SetLastError(rc);
|
|
return(FALSE);
|
|
}
|
|
|
|
b = SetupQueueCopySectionW(
|
|
QueueHandle,
|
|
sourcerootpath,
|
|
InfHandle,
|
|
ListInfHandle,
|
|
section,
|
|
CopyStyle
|
|
);
|
|
|
|
rc = GetLastError();
|
|
|
|
MyFree(sourcerootpath);
|
|
MyFree(section);
|
|
|
|
SetLastError(rc);
|
|
return(b);
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub
|
|
//
|
|
BOOL
|
|
SetupQueueCopySectionW(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCWSTR SourceRootPath,
|
|
IN HINF InfHandle,
|
|
IN HINF ListInfHandle, OPTIONAL
|
|
IN PCWSTR Section,
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(QueueHandle);
|
|
UNREFERENCED_PARAMETER(SourceRootPath);
|
|
UNREFERENCED_PARAMETER(InfHandle);
|
|
UNREFERENCED_PARAMETER(ListInfHandle);
|
|
UNREFERENCED_PARAMETER(Section);
|
|
UNREFERENCED_PARAMETER(CopyStyle);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
BOOL
|
|
SetupQueueCopySection(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN PCTSTR SourceRootPath,
|
|
IN HINF InfHandle,
|
|
IN HINF ListInfHandle, OPTIONAL
|
|
IN PCTSTR Section,
|
|
IN DWORD CopyStyle
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Queue an entire section in an inf file for copy. The section must be
|
|
in copy-section format and the inf file must contain [SourceDisksFiles]
|
|
and [SourceDisksNames] sections.
|
|
|
|
Arguments:
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
SourceRootPath - supplies the root directory for the intended source.
|
|
This should be a sharepoint or a device root such as a:\ or g:\.
|
|
|
|
InfHandle - supplies a handle to an open inf file, that contains the
|
|
[SourceDisksFiles] and [SourceDisksNames] sections, and, if
|
|
ListInfHandle is not specified, contains the section names by Section.
|
|
This handle must be for a win95-style inf.
|
|
|
|
ListInfHandle - if specified, supplies a handle to an open inf file
|
|
containing the section to be queued for copy. Otherwise InfHandle
|
|
is assumed to contain the section.
|
|
|
|
Section - supplies the name of the section to be queued for copy.
|
|
|
|
CopyStyle - supplies flags that control the behavior of the copy operation
|
|
for this file.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, GetLastError() returns
|
|
extended error information. Some of the files may have been queued.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL b;
|
|
INFCONTEXT LineContext;
|
|
PCTSTR SourceFilename;
|
|
PCTSTR TargetFilename;
|
|
PCTSTR SecurityDescriptor = NULL;
|
|
PCTSTR CacheName = NULL;
|
|
UINT Flags;
|
|
DWORD CopyStyleLocal;
|
|
LONG LineCount;
|
|
HINF CabInf = INVALID_HANDLE_VALUE;
|
|
DWORD rc;
|
|
|
|
//
|
|
// Note that there are no potential faults here so no try/excepts
|
|
// are necessary. pSetupQueueSingleCopy does all validation.
|
|
//
|
|
|
|
if(!ListInfHandle || (ListInfHandle == INVALID_HANDLE_VALUE)) {
|
|
ListInfHandle = InfHandle;
|
|
}
|
|
|
|
//
|
|
// Check for missing section
|
|
//
|
|
LineCount = SetupGetLineCount (ListInfHandle, Section);
|
|
if(LineCount == -1) {
|
|
rc = GetLastError();
|
|
pSetupLogSectionError(ListInfHandle,NULL,NULL,QueueHandle,Section,MSG_LOG_NOSECTION_COPY,rc,NULL);
|
|
SetLastError(ERROR_SECTION_NOT_FOUND); // maintain existing error code, log contains correct error code
|
|
return(FALSE);
|
|
}
|
|
|
|
//
|
|
// if section is empty, do nothing.
|
|
//
|
|
if(LineCount == 0) {
|
|
return(TRUE);
|
|
}
|
|
|
|
//
|
|
// The section has to exist and there has to be at least one line in it.
|
|
//
|
|
b = SetupFindFirstLine(ListInfHandle,Section,NULL,&LineContext);
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
pSetupLogSectionError(ListInfHandle,NULL,NULL,QueueHandle,Section,MSG_LOG_NOSECTION_COPY,rc,NULL);
|
|
SetLastError(ERROR_SECTION_NOT_FOUND); // maintain existing error code, log contains correct error code
|
|
return(FALSE);
|
|
}
|
|
|
|
//
|
|
//Get the Security descriptor
|
|
//
|
|
|
|
if( !pSetupGetSecurityInfo( ListInfHandle, Section, &SecurityDescriptor ) )
|
|
SecurityDescriptor = NULL;
|
|
|
|
|
|
//
|
|
// load driver cache inf
|
|
//
|
|
CabInf = SetupOpenInfFile( STR_DRIVERCACHEINF , NULL, INF_STYLE_WIN4, NULL );
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
CopyStyle |= PSP_COPY_USE_DRIVERCACHE;
|
|
}
|
|
|
|
//
|
|
// Iterate every line in the section.
|
|
//
|
|
do {
|
|
CopyStyleLocal = CopyStyle;
|
|
//
|
|
// Get the target filename out of the line.
|
|
// Field 1 is the target so there must be one for the line to be valid.
|
|
//
|
|
TargetFilename = pSetupFilenameFromLine(&LineContext,FALSE);
|
|
if(!TargetFilename) {
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
SetupCloseInfFile(CabInf);
|
|
}
|
|
try {
|
|
if (QueueHandle != NULL
|
|
&& QueueHandle != (HSPFILEQ)INVALID_HANDLE_VALUE
|
|
&& ((PSP_FILE_QUEUE)QueueHandle)->Signature == SP_FILE_QUEUE_SIG) {
|
|
|
|
WriteLogEntry(
|
|
((PSP_FILE_QUEUE)QueueHandle)->LogContext,
|
|
SETUP_LOG_ERROR,
|
|
MSG_LOG_COPY_TARGET,
|
|
NULL,
|
|
Section);
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
}
|
|
SetLastError(ERROR_INVALID_DATA);
|
|
return(FALSE);
|
|
}
|
|
|
|
//
|
|
// Get source filename out of the line. If there is none, use
|
|
// the target name as the source name.
|
|
//
|
|
SourceFilename = pSetupFilenameFromLine(&LineContext,TRUE);
|
|
if(!SourceFilename || (*SourceFilename == 0)) {
|
|
SourceFilename = TargetFilename;
|
|
}
|
|
|
|
//
|
|
// if we were asked to use the driver cache, then check if the file
|
|
// is in the associated INF for the cab.
|
|
//
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
if (!pIsFileInDriverCache(CabInf, SourceFilename, NULL, &CacheName)) {
|
|
CopyStyleLocal &= ~PSP_COPY_USE_DRIVERCACHE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If present, flags are field 3.
|
|
//
|
|
if(SetupGetIntField(&LineContext,4,(PINT)&Flags)) {
|
|
|
|
if(Flags & COPYFLG_WARN_IF_SKIP) {
|
|
CopyStyleLocal |= SP_COPY_WARNIFSKIP;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NOSKIP) {
|
|
CopyStyleLocal |= SP_COPY_NOSKIP;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NOVERSIONCHECK) {
|
|
CopyStyleLocal &= ~SP_COPY_NEWER;
|
|
}
|
|
|
|
if(Flags & COPYFLG_FORCE_FILE_IN_USE) {
|
|
CopyStyleLocal |= SP_COPY_FORCE_IN_USE;
|
|
CopyStyleLocal |= SP_COPY_IN_USE_NEEDS_REBOOT;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NO_OVERWRITE) {
|
|
CopyStyleLocal |= SP_COPY_FORCE_NOOVERWRITE;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NO_VERSION_DIALOG) {
|
|
CopyStyleLocal |= SP_COPY_FORCE_NEWER;
|
|
}
|
|
|
|
if(Flags & COPYFLG_OVERWRITE_OLDER_ONLY) {
|
|
CopyStyleLocal |= SP_COPY_NEWER_ONLY;
|
|
}
|
|
|
|
if(Flags & COPYFLG_REPLACEONLY) {
|
|
CopyStyleLocal |= SP_COPY_REPLACEONLY;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NODECOMP) {
|
|
CopyStyleLocal |= SP_COPY_NODECOMP;
|
|
}
|
|
|
|
if(Flags & COPYFLG_REPLACE_BOOT_FILE) {
|
|
CopyStyleLocal |= SP_COPY_REPLACE_BOOT_FILE;
|
|
}
|
|
|
|
if(Flags & COPYFLG_NOPRUNE) {
|
|
CopyStyleLocal |= SP_COPY_NOPRUNE;
|
|
}
|
|
|
|
}
|
|
|
|
b = pSetupQueueSingleCopy(
|
|
QueueHandle,
|
|
InfHandle,
|
|
ListInfHandle,
|
|
Section,
|
|
SourceRootPath,
|
|
SourceFilename,
|
|
TargetFilename,
|
|
CopyStyleLocal,
|
|
SecurityDescriptor,
|
|
CacheName
|
|
);
|
|
|
|
if (CacheName) {
|
|
MyFree( CacheName );
|
|
CacheName = NULL;
|
|
}
|
|
|
|
if(!b) {
|
|
DWORD LastError = GetLastError();
|
|
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
SetupCloseInfFile(CabInf);
|
|
}
|
|
|
|
SetLastError( LastError );
|
|
|
|
return(FALSE);
|
|
}
|
|
} while(SetupFindNextLine(&LineContext,&LineContext));
|
|
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
SetupCloseInfFile(CabInf);
|
|
}
|
|
return(TRUE);
|
|
}
|
|
|
|
|
|
BOOL
|
|
pSetupQueueSingleCopy(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN HINF InfHandle,
|
|
IN HINF ListInfHandle, OPTIONAL
|
|
IN PCTSTR SectionName, OPTIONAL
|
|
IN PCTSTR SourceRootPath,
|
|
IN PCTSTR SourceFilename,
|
|
IN PCTSTR TargetFilename,
|
|
IN DWORD CopyStyle,
|
|
IN PCTSTR SecurityDescriptor,
|
|
IN PCTSTR CacheName
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Add a single file to the copy queue, using the default source media
|
|
and destination as specified in an inf file.
|
|
|
|
Arguments:
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
InfHandle - supplies a handle to an open inf file, that contains the
|
|
[SourceDisksFiles] and [SourceDisksNames] sections.
|
|
This handle must be for a win95-style inf.
|
|
|
|
ListInfHandle - if specified, supplies handle to the inf in which
|
|
the file being copied appears (such as in a file copy list section).
|
|
If not specified, this is assumed to be the same inf as InfHandle.
|
|
|
|
SourceRootPath - supplies the root directory for the intended source.
|
|
This should be a sharepoint or a device root such as a:\ or g:\.
|
|
|
|
SourceFilename - supplies the filename of the source file. Filename part
|
|
only.
|
|
|
|
TargetFilename - supplies the filename of the target file. Filename part
|
|
only.
|
|
|
|
CopyStyle - supplies flags that control the behavior of the copy operation
|
|
for this file.
|
|
|
|
SecurityDescriptor - describes the permissions for the target file
|
|
|
|
CacheName - if supplied this is the name of the driver cache we should
|
|
use to copy the file out of instead of the specified source path
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, GetLastError() returns
|
|
extended error information.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL b;
|
|
UINT SourceId;
|
|
DWORD SizeRequired;
|
|
PTSTR TargetDirectory;
|
|
PCTSTR SourceDescription,SourceTagfile,SourceRelativePath;
|
|
PCTSTR TmpCacheName = CacheName;
|
|
UINT SourceFlags;
|
|
DWORD rc;
|
|
TCHAR FileSubdir[MAX_PATH];
|
|
TCHAR RelativePath[MAX_PATH];
|
|
INFCONTEXT LineContext;
|
|
PINFCONTEXT pLineContext;
|
|
SP_FILE_COPY_PARAMSEX CopyParams;
|
|
HINF CabInf = INVALID_HANDLE_VALUE;
|
|
PSETUP_LOG_CONTEXT lc = NULL;
|
|
BOOL AlreadyLoggedError = FALSE;
|
|
|
|
if(!ListInfHandle || (ListInfHandle == INVALID_HANDLE_VALUE)) {
|
|
ListInfHandle = InfHandle;
|
|
}
|
|
|
|
//
|
|
// Determine the source disk id and subdir where the file is located.
|
|
//
|
|
try {
|
|
|
|
if((QueueHandle != NULL) &&
|
|
(QueueHandle != INVALID_HANDLE_VALUE) &&
|
|
(((PSP_FILE_QUEUE)QueueHandle)->Signature == SP_FILE_QUEUE_SIG)) {
|
|
|
|
lc = ((PSP_FILE_QUEUE)QueueHandle)->LogContext;
|
|
|
|
b = _SetupGetSourceFileLocation(
|
|
InfHandle,
|
|
NULL,
|
|
SourceFilename,
|
|
(((PSP_FILE_QUEUE)QueueHandle)->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(((PSP_FILE_QUEUE)QueueHandle)->AltPlatformInfo)
|
|
: NULL,
|
|
&SourceId,
|
|
FileSubdir,
|
|
MAX_PATH,
|
|
&rc,
|
|
&LineContext
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
}
|
|
|
|
} else {
|
|
b = FALSE;
|
|
rc = ERROR_INVALID_HANDLE;
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
b = FALSE;
|
|
rc = ERROR_INVALID_HANDLE;
|
|
lc = NULL;
|
|
}
|
|
|
|
if(!b) {
|
|
|
|
if((rc == ERROR_INVALID_PARAMETER) || (rc == ERROR_INVALID_HANDLE)) {
|
|
//
|
|
// if we failed due to a bad parameter, bail now
|
|
//
|
|
goto clean1;
|
|
}
|
|
|
|
//
|
|
// Try to fetch just the id and assume there is no subdir.
|
|
//
|
|
try {
|
|
|
|
b = _SetupGetSourceFileLocation(
|
|
InfHandle,
|
|
NULL,
|
|
SourceFilename,
|
|
(((PSP_FILE_QUEUE)QueueHandle)->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(((PSP_FILE_QUEUE)QueueHandle)->AltPlatformInfo)
|
|
: NULL,
|
|
&SourceId,
|
|
NULL,
|
|
0,
|
|
NULL,
|
|
&LineContext
|
|
);
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
b = FALSE;
|
|
}
|
|
|
|
if(b) {
|
|
FileSubdir[0] = 0;
|
|
}
|
|
}
|
|
|
|
if(b) {
|
|
//
|
|
// Get information about the source. Need the tag file,
|
|
// description, and relative source path.
|
|
//
|
|
try {
|
|
|
|
b = pSetupGetSourceAllInfo(
|
|
InfHandle,
|
|
&LineContext,
|
|
SourceId,
|
|
(((PSP_FILE_QUEUE)QueueHandle)->Flags & FQF_USE_ALT_PLATFORM)
|
|
? &(((PSP_FILE_QUEUE)QueueHandle)->AltPlatformInfo)
|
|
: NULL,
|
|
&SourceDescription,
|
|
&SourceTagfile,
|
|
&SourceRelativePath,
|
|
&SourceFlags
|
|
);
|
|
|
|
if(!b) {
|
|
rc = GetLastError();
|
|
if((rc == ERROR_LINE_NOT_FOUND) || (rc == ERROR_SECTION_NOT_FOUND)) {
|
|
WriteLogEntry(
|
|
lc,
|
|
SETUP_LOG_ERROR,
|
|
MSG_LOG_NO_SOURCE,
|
|
NULL,
|
|
SourceId
|
|
);
|
|
AlreadyLoggedError = TRUE;
|
|
}
|
|
}
|
|
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
b = FALSE;
|
|
rc = ERROR_INVALID_PARAMETER;
|
|
}
|
|
|
|
if(!b) {
|
|
goto clean1;
|
|
}
|
|
|
|
//
|
|
// Set a value that causes _SetupQueueCopy to skip looking for the
|
|
// [SourceDisksFiles] section -- we just found it, so we just pass
|
|
// the info along!
|
|
//
|
|
pLineContext = &LineContext;
|
|
|
|
} else {
|
|
//
|
|
// Assume there is no SourceDisksFiles section and fake it as best we can.
|
|
// Assume the media has a description of "Unknown," set the source path to
|
|
// the source root if there is one, and assume no tag file.
|
|
//
|
|
// We also set a special value that tells _SetupQueueCopy not to bother trying
|
|
// to look for the [SourceDisksFiles] section itself, since there isn't one.
|
|
//
|
|
FileSubdir[0] = 0;
|
|
SourceDescription = NULL;
|
|
SourceTagfile = NULL;
|
|
SourceRelativePath = NULL;
|
|
pLineContext = (PINFCONTEXT)(-1);
|
|
}
|
|
|
|
if ( CopyStyle & PSP_COPY_CHK_DRIVERCACHE) {
|
|
CabInf = SetupOpenInfFile( STR_DRIVERCACHEINF, NULL, INF_STYLE_WIN4, NULL );
|
|
if (CabInf != INVALID_HANDLE_VALUE) {
|
|
if (pIsFileInDriverCache(CabInf, SourceFilename, SourceRelativePath, &TmpCacheName)) {
|
|
CopyStyle |= PSP_COPY_USE_DRIVERCACHE;
|
|
CopyStyle &= ~PSP_COPY_CHK_DRIVERCACHE;
|
|
}
|
|
|
|
SetupCloseInfFile(CabInf);
|
|
|
|
}
|
|
}
|
|
|
|
if (CopyStyle & PSP_COPY_USE_DRIVERCACHE) {
|
|
//
|
|
// check if the inf we want to copy from is an OEM inf
|
|
//
|
|
if (!pLineContext || pLineContext==(PINFCONTEXT)-1) {
|
|
CopyStyle &= ~PSP_COPY_USE_DRIVERCACHE;
|
|
} else if (pSetupInfIsFromOemLocation( ((PLOADED_INF)pLineContext->CurrentInf)->VersionBlock.Filename,TRUE )) {
|
|
CopyStyle &= ~PSP_COPY_USE_DRIVERCACHE;
|
|
} else if ( ((PLOADED_INF)pLineContext->CurrentInf)->OriginalInfName
|
|
&& pSetupInfIsFromOemLocation( ((PLOADED_INF)pLineContext->CurrentInf)->OriginalInfName, TRUE) ) {
|
|
CopyStyle &= ~PSP_COPY_USE_DRIVERCACHE;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Determine the target path for the file.
|
|
//
|
|
if(b = SetupGetTargetPath(ListInfHandle,NULL,SectionName,NULL,0,&SizeRequired)) {
|
|
|
|
if(TargetDirectory = MyMalloc(SizeRequired*sizeof(TCHAR))) {
|
|
|
|
if(b = SetupGetTargetPath(ListInfHandle,NULL,SectionName,TargetDirectory,SizeRequired,NULL)) {
|
|
|
|
try {
|
|
WriteLogEntry(
|
|
lc,
|
|
SETUP_LOG_VVERBOSE,
|
|
SectionName ? MSG_LOG_COPY_QUEUE : MSG_LOG_DEFCOPY_QUEUE,
|
|
NULL,
|
|
SectionName ? SectionName : TEXT(""),
|
|
((PLOADED_INF)ListInfHandle)->VersionBlock.Filename,
|
|
TargetFilename ? TargetFilename : TEXT(""),
|
|
SourceFilename ? SourceFilename : TEXT(""),
|
|
CopyStyle,
|
|
TargetDirectory ? TargetDirectory : TEXT(""));
|
|
if (pLineContext && (pLineContext != (PINFCONTEXT)(-1))) {
|
|
LPCTSTR SrcSecName = NULL;
|
|
LPCTSTR SrcInfName = NULL;
|
|
PLOADED_INF pInf = (PLOADED_INF)(pLineContext->CurrentInf);
|
|
|
|
MYASSERT(pInf);
|
|
SrcSecName = pStringTableStringFromId(
|
|
pInf->StringTable,
|
|
pInf->SectionBlock[pLineContext->Section].SectionName);
|
|
SrcInfName = pInf->VersionBlock.Filename;
|
|
WriteLogEntry(
|
|
lc,
|
|
SETUP_LOG_VVERBOSE,
|
|
(CopyStyle & PSP_COPY_USE_DRIVERCACHE) ? MSG_LOG_COPY_QUEUE_DRIVERCACHE : MSG_LOG_COPY_QUEUE_SOURCE,
|
|
NULL,
|
|
SrcSecName ? SrcSecName : TEXT(""),
|
|
SrcInfName ? SrcInfName : TEXT(""),
|
|
SourceId ? SourceId : TEXT('\0'),
|
|
SourceDescription ? SourceDescription : TEXT(""),
|
|
SourceTagfile ? SourceTagfile : TEXT(""),
|
|
SourceRelativePath ? SourceRelativePath : TEXT(""));
|
|
|
|
} else {
|
|
WriteLogEntry(
|
|
lc,
|
|
SETUP_LOG_VVERBOSE,
|
|
MSG_LOG_COPY_QUEUE_DEFAULT,
|
|
NULL);
|
|
}
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
}
|
|
//
|
|
// Append the source relative path and the file subdir.
|
|
//
|
|
if(SourceRelativePath) {
|
|
lstrcpyn(RelativePath,SourceRelativePath,MAX_PATH);
|
|
if(FileSubdir[0]) {
|
|
pSetupConcatenatePaths(RelativePath,FileSubdir,MAX_PATH,NULL);
|
|
}
|
|
} else {
|
|
RelativePath[0] = 0;
|
|
}
|
|
|
|
//
|
|
// Add to queue.
|
|
//
|
|
CopyParams.cbSize = sizeof(SP_FILE_COPY_PARAMSEX);
|
|
CopyParams.QueueHandle = QueueHandle;
|
|
CopyParams.SourceRootPath = SourceRootPath;
|
|
CopyParams.SourcePath = RelativePath[0] ? RelativePath : NULL ;
|
|
CopyParams.SourceFilename = SourceFilename;
|
|
CopyParams.SourceDescription = SourceDescription;
|
|
CopyParams.SourceTagfile = SourceTagfile;
|
|
CopyParams.TargetDirectory = TargetDirectory;
|
|
CopyParams.TargetFilename = TargetFilename;
|
|
CopyParams.CopyStyle = CopyStyle;
|
|
CopyParams.LayoutInf = InfHandle;
|
|
CopyParams.SecurityDescriptor= SecurityDescriptor;
|
|
CopyParams.CacheName = TmpCacheName;
|
|
//
|
|
// first item indicates source flag information
|
|
// second item indicates that we've already retrieved
|
|
// this information, so even if the SourceFlags are zero,
|
|
// we won't go looking for it again
|
|
//
|
|
CopyParams.SourceFlags = SourceFlags;
|
|
CopyParams.SourceFlagsSet = TRUE;
|
|
|
|
b = _SetupQueueCopy(&CopyParams,
|
|
pLineContext,
|
|
((InfHandle == ListInfHandle) ? NULL : ListInfHandle)
|
|
);
|
|
|
|
rc = GetLastError();
|
|
}
|
|
|
|
MyFree(TargetDirectory);
|
|
|
|
} else {
|
|
rc = ERROR_NOT_ENOUGH_MEMORY;
|
|
}
|
|
} else {
|
|
rc = GetLastError();
|
|
}
|
|
|
|
if(SourceDescription) {
|
|
MyFree(SourceDescription);
|
|
}
|
|
if(SourceTagfile) {
|
|
MyFree(SourceTagfile);
|
|
}
|
|
if(SourceRelativePath) {
|
|
MyFree(SourceRelativePath);
|
|
}
|
|
if(TmpCacheName && TmpCacheName != CacheName) {
|
|
MyFree(TmpCacheName);
|
|
}
|
|
|
|
clean1:
|
|
if(!b) {
|
|
|
|
BOOL FreeLC = FALSE;
|
|
|
|
if(!lc) {
|
|
if(CreateLogContext(NULL, TRUE, &lc) == NO_ERROR) {
|
|
//
|
|
// success
|
|
//
|
|
FreeLC = TRUE;
|
|
} else {
|
|
lc = NULL;
|
|
}
|
|
}
|
|
|
|
//
|
|
// If we couldn't create a log context (i.e., due to out-of-memory),
|
|
// don't bother calling WriteLogEntry, because it's not going to have
|
|
// any better luck...
|
|
//
|
|
if(lc) {
|
|
|
|
if(!AlreadyLoggedError) {
|
|
try {
|
|
WriteLogEntry(
|
|
lc,
|
|
SETUP_LOG_ERROR|SETUP_LOG_BUFFER,
|
|
MSG_LOG_COPY_QUEUE_ERROR,
|
|
NULL,
|
|
SectionName ? SectionName : TEXT(""),
|
|
((PLOADED_INF)ListInfHandle)->VersionBlock.Filename,
|
|
TargetFilename ? TargetFilename : TEXT(""),
|
|
SourceFilename ? SourceFilename : TEXT(""));
|
|
WriteLogError(
|
|
lc,
|
|
SETUP_LOG_ERROR,
|
|
rc
|
|
);
|
|
} except(EXCEPTION_EXECUTE_HANDLER) {
|
|
}
|
|
}
|
|
|
|
if(FreeLC) {
|
|
DeleteLogContext(lc);
|
|
}
|
|
}
|
|
|
|
SetLastError(rc);
|
|
}
|
|
|
|
return(b);
|
|
}
|
|
|
|
|
|
#ifdef UNICODE
|
|
//
|
|
// ANSI version
|
|
//
|
|
BOOL
|
|
SetupQueueDefaultCopyA(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN HINF InfHandle,
|
|
IN PCSTR SourceRootPath,
|
|
IN PCSTR SourceFilename,
|
|
IN PCSTR TargetFilename,
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
PWSTR sourcerootpath;
|
|
PWSTR sourcefilename;
|
|
PWSTR targetfilename;
|
|
DWORD rc;
|
|
BOOL b;
|
|
|
|
b = FALSE;
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceRootPath,&sourcerootpath);
|
|
if(rc == NO_ERROR) {
|
|
|
|
rc = pSetupCaptureAndConvertAnsiArg(SourceFilename,&sourcefilename);
|
|
if(rc == NO_ERROR) {
|
|
|
|
rc = pSetupCaptureAndConvertAnsiArg(TargetFilename,&targetfilename);
|
|
if(rc == NO_ERROR) {
|
|
|
|
b = SetupQueueDefaultCopyW(
|
|
QueueHandle,
|
|
InfHandle,
|
|
sourcerootpath,
|
|
sourcefilename,
|
|
targetfilename,
|
|
CopyStyle
|
|
);
|
|
|
|
rc = GetLastError();
|
|
|
|
MyFree(targetfilename);
|
|
}
|
|
|
|
MyFree(sourcefilename);
|
|
}
|
|
|
|
MyFree(sourcerootpath);
|
|
}
|
|
|
|
SetLastError(rc);
|
|
return(b);
|
|
}
|
|
#else
|
|
//
|
|
// Unicode stub
|
|
//
|
|
BOOL
|
|
SetupQueueDefaultCopyW(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN HINF InfHandle,
|
|
IN PCWSTR SourceRootPath,
|
|
IN PCWSTR SourceFilename,
|
|
IN PCWSTR TargetFilename,
|
|
IN DWORD CopyStyle
|
|
)
|
|
{
|
|
UNREFERENCED_PARAMETER(QueueHandle);
|
|
UNREFERENCED_PARAMETER(InfHandle);
|
|
UNREFERENCED_PARAMETER(SourceRootPath);
|
|
UNREFERENCED_PARAMETER(SourceFilename);
|
|
UNREFERENCED_PARAMETER(TargetFilename);
|
|
UNREFERENCED_PARAMETER(CopyStyle);
|
|
SetLastError(ERROR_CALL_NOT_IMPLEMENTED);
|
|
return(FALSE);
|
|
}
|
|
#endif
|
|
|
|
|
|
BOOL
|
|
SetupQueueDefaultCopy(
|
|
IN HSPFILEQ QueueHandle,
|
|
IN HINF InfHandle,
|
|
IN PCTSTR SourceRootPath,
|
|
IN PCTSTR SourceFilename,
|
|
IN PCTSTR TargetFilename,
|
|
IN DWORD CopyStyle
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Add a single file to the copy queue, using the default source media
|
|
and destination as specified in an inf file.
|
|
|
|
Arguments:
|
|
|
|
QueueHandle - supplies a handle to a setup file queue, as returned
|
|
by SetupOpenFileQueue.
|
|
|
|
InfHandle - supplies a handle to an open inf file, that contains the
|
|
[SourceDisksFiles] and [SourceDisksNames] sections.
|
|
This handle must be for a win95-style inf.
|
|
|
|
SourceRootPath - supplies the root directory for the intended source.
|
|
This should be a sharepoint or a device root such as a:\ or g:\.
|
|
|
|
SourceFilename - supplies the filename of the source file. Filename part
|
|
only.
|
|
|
|
TargetFilename - supplies the filename of the target file. Filename part
|
|
only.
|
|
|
|
CopyStyle - supplies flags that control the behavior of the copy operation
|
|
for this file.
|
|
|
|
Return Value:
|
|
|
|
Boolean value indicating outcome. If FALSE, GetLastError() returns
|
|
extended error information.
|
|
|
|
--*/
|
|
|
|
{
|
|
BOOL b;
|
|
|
|
b = pSetupQueueSingleCopy(
|
|
QueueHandle,
|
|
InfHandle,
|
|
NULL,
|
|
NULL,
|
|
SourceRootPath,
|
|
SourceFilename,
|
|
TargetFilename,
|
|
CopyStyle | PSP_COPY_CHK_DRIVERCACHE,
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
return(b);
|
|
}
|
|
|
|
|
|
PSOURCE_MEDIA_INFO
|
|
pSetupQueueSourceMedia(
|
|
IN OUT PSP_FILE_QUEUE Queue,
|
|
IN OUT PSP_FILE_QUEUE_NODE QueueNode,
|
|
IN LONG SourceRootStringId,
|
|
IN PCTSTR SourceDescription, OPTIONAL
|
|
IN PCTSTR SourceTagfile, OPTIONAL
|
|
IN PCTSTR SourceCabfile, OPTIONAL
|
|
IN DWORD MediaFlags
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Set up a file queue node's source media descriptor pointer, creating a new
|
|
source media descriptor if necessary.
|
|
|
|
Arguments:
|
|
|
|
Queue - supplies pointer to file queue with which the queue node
|
|
is associated.
|
|
|
|
QueueNode - supplies file queue node whose source media descriptor pointer
|
|
is to be set.
|
|
|
|
SourceRootStringId - supplies string id of root to source (something like a:\).
|
|
|
|
SourceDescription - if specified, supplies a description for the media.
|
|
|
|
SourceTagfile - if specified, supplies a tag file for the media.
|
|
|
|
SourceCabfile - if specified, supplies a cabfile for the media different to the tagfile.
|
|
|
|
MediaFlags - specifies additional information used in searching for an
|
|
existing source media descriptor in the specified queue, and in adding
|
|
new source media descriptors to that queue. May be a combination of
|
|
the following values:
|
|
|
|
SMI_FLAG_NO_SOURCE_ROOT_PATH : The caller didn't supply a SourceRootPath
|
|
for this copy action, so we're using a default path. This flag
|
|
causes us to not include the SourceRootStringId as part of our
|
|
match criteria when searching to see if the specified source media
|
|
information is already present in an existing media descriptor. If
|
|
we don't find a match (i.e., we have to create a new descriptor),
|
|
we'll store this flag away in the SOURCE_MEDIA_INFO.Flags field so
|
|
that if we come along later to add source media descriptors where
|
|
the caller did specify SourceRootPath, then we'll re-use this
|
|
descriptor and overwrite the existing (default) source root path
|
|
with the caller-specified one.
|
|
|
|
SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH : The caller didn't supply a SourceRootPath
|
|
for this copy action, and it's a tagged source media, so we're using a
|
|
service pack path. This flag causes us to not include the SourceRootStringId
|
|
as part of our match criteria when searching to see if the specified source media
|
|
information is already present in an existing media descriptor. If
|
|
we don't find a match (i.e., we have to create a new descriptor),
|
|
we'll store this flag away in the SOURCE_MEDIA_INFO.Flags field so
|
|
that if we come along later to add source media descriptors where
|
|
the caller did specify SourceRootPath, then we'll re-use this
|
|
descriptor and overwrite the existing (default) source root path
|
|
with the caller-specified one.
|
|
|
|
SMI_FLAG_USE_LOCAL_SOURCE_CAB : The caller wants to use the local source cab containing
|
|
driver files, etc. In this case, we supply the source description and tagfile,
|
|
ignoring what the caller passes in. At this point we know the media is present, as
|
|
the caller provided this check. If it wasnt't, we default to the OS Source path location.
|
|
|
|
|
|
Return Value:
|
|
|
|
Pointer to source media info structure, or NULL if out of memory.
|
|
|
|
--*/
|
|
|
|
{
|
|
LONG DescriptionStringId;
|
|
LONG TagfileStringId;
|
|
LONG CabfileStringId;
|
|
PSOURCE_MEDIA_INFO Source,LastSource, TempSource;
|
|
BOOL b1,b2,b3;
|
|
TCHAR TempTagfileString[MAX_PATH];
|
|
TCHAR TempCabfileString[MAX_PATH];
|
|
TCHAR TempSrcDescString[LINE_LEN];
|
|
|
|
|
|
if (MediaFlags & SMI_FLAG_USE_LOCAL_SOURCE_CAB) {
|
|
LoadString( MyDllModuleHandle, IDS_DRIVERCACHE_DESC, TempSrcDescString, sizeof(TempSrcDescString)/sizeof(TCHAR) );
|
|
SourceDescription = TempSrcDescString;
|
|
} else {
|
|
//
|
|
// For the optional SourceDescription and SourceTagfile parameters, treat
|
|
// empty strings as if the parameter had not been specified.
|
|
//
|
|
if(SourceDescription && !(*SourceDescription)) {
|
|
SourceDescription = NULL;
|
|
}
|
|
if(SourceTagfile && !(*SourceTagfile)) {
|
|
SourceTagfile = NULL;
|
|
}
|
|
|
|
//
|
|
// If no description is specified, force the tagfile to none.
|
|
//
|
|
if(!SourceDescription) {
|
|
SourceTagfile = NULL;
|
|
}
|
|
}
|
|
|
|
if(SourceDescription) {
|
|
//
|
|
// Description specified. See if it's in the table. If not,
|
|
// no need to search the list of media descriptors because we know
|
|
// we can't find a match.
|
|
//
|
|
// (We must first copy this string to a writeable buffer, to speed up the
|
|
// case-insensitive lookup.
|
|
//
|
|
lstrcpyn(TempSrcDescString, SourceDescription, SIZECHARS(TempSrcDescString));
|
|
DescriptionStringId = pSetupStringTableLookUpString(Queue->StringTable,
|
|
TempSrcDescString,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
b1 = (DescriptionStringId != -1);
|
|
} else {
|
|
//
|
|
// No description specified, look for a source media with -1 as the
|
|
// description string id
|
|
//
|
|
DescriptionStringId = -1;
|
|
b1 = TRUE;
|
|
}
|
|
|
|
if(SourceTagfile) {
|
|
//
|
|
// Tagfile specified. See if it's in the table. If not,
|
|
// no need to search the list of media descriptors because we know
|
|
// we can't find a match.
|
|
//
|
|
// (Again, we must first copy the string to a writeable buffer.
|
|
//
|
|
lstrcpyn(TempTagfileString, SourceTagfile, SIZECHARS(TempTagfileString));
|
|
TagfileStringId = pSetupStringTableLookUpString(Queue->StringTable,
|
|
TempTagfileString,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
b2 = (TagfileStringId != -1);
|
|
} else {
|
|
//
|
|
// No tagfile specified, look for a source media with -1 as the
|
|
// tagfile string id
|
|
//
|
|
TagfileStringId = -1;
|
|
b2 = TRUE;
|
|
}
|
|
|
|
if(SourceCabfile) {
|
|
//
|
|
// Cabfile specified. See if it's in the table. If not,
|
|
// no need to search the list of media descriptors because we know
|
|
// we can't find a match.
|
|
//
|
|
// (Again, we must first copy the string to a writeable buffer.
|
|
//
|
|
lstrcpyn(TempCabfileString, SourceCabfile, SIZECHARS(TempCabfileString));
|
|
CabfileStringId = pSetupStringTableLookUpString(Queue->StringTable,
|
|
TempCabfileString,
|
|
STRTAB_CASE_INSENSITIVE | STRTAB_BUFFER_WRITEABLE
|
|
);
|
|
b3 = (CabfileStringId != -1);
|
|
} else {
|
|
//
|
|
// No cabfile specified, merge Cabfile&Tagfile together
|
|
// since b2==b3, then we have the identities b2|b3 == b2 and b2&b3 == b2
|
|
// ie, old behavior
|
|
//
|
|
CabfileStringId = TagfileStringId;
|
|
b3 = b2;
|
|
}
|
|
|
|
//
|
|
// If we think there's a possibility of finding an existing source that
|
|
// matches the caller's parameters, scan the source media list looking
|
|
// for a match.
|
|
//
|
|
if(b1 && b2 && b3) {
|
|
|
|
for(Source=Queue->SourceMediaList; Source; Source=Source->Next) {
|
|
|
|
if((Source->Description == DescriptionStringId)
|
|
&& (Source->Tagfile == TagfileStringId)
|
|
&& (Source->Cabfile == CabfileStringId)) {
|
|
//
|
|
// We only consider the SourceRootPath when both existing
|
|
// media descriptor and new media descriptor have actual
|
|
// caller-supplied paths (as opposed to something we made up).
|
|
//
|
|
if((Source->Flags & SMI_FLAG_NO_SOURCE_ROOT_PATH) ||
|
|
(MediaFlags & SMI_FLAG_NO_SOURCE_ROOT_PATH) ||
|
|
(Source->SourceRootPath == SourceRootStringId)) {
|
|
//
|
|
// Got a match. Point the queue node at this source and return.
|
|
//
|
|
QueueNode->SourceMediaInfo = Source;
|
|
//
|
|
// If the existing media descriptor had a made-up source
|
|
// root path, but the new media information had an actual
|
|
// caller-supplied one, then replace the made-up one with
|
|
// the real one and clear the no-source-root-path flag.
|
|
//
|
|
if((Source->Flags & SMI_FLAG_NO_SOURCE_ROOT_PATH) &&
|
|
!(MediaFlags & SMI_FLAG_NO_SOURCE_ROOT_PATH)) {
|
|
|
|
Source->SourceRootPath = SourceRootStringId;
|
|
Source->Flags &= ~SMI_FLAG_NO_SOURCE_ROOT_PATH;
|
|
}
|
|
|
|
return(Source);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Need to add a new source media descriptor.
|
|
// Allocate the structure and fill it in.
|
|
//
|
|
Source = MyMalloc(sizeof(SOURCE_MEDIA_INFO));
|
|
if(!Source) {
|
|
return(NULL);
|
|
}
|
|
|
|
Source->Next = NULL;
|
|
Source->CopyQueue = NULL;
|
|
Source->CopyNodeCount = 0;
|
|
Source->Flags = MediaFlags;
|
|
|
|
if(SourceDescription) {
|
|
//
|
|
// Since we already passed this in for a case-insensitive lookup with a writeable
|
|
// buffer, we can add it case-sensitively, because it's already lower-cased.
|
|
//
|
|
Source->Description = pSetupStringTableAddString(Queue->StringTable,
|
|
TempSrcDescString,
|
|
STRTAB_CASE_SENSITIVE | STRTAB_ALREADY_LOWERCASE
|
|
);
|
|
//
|
|
// We also must add the description in its original case, since this is a displayable string.
|
|
// (We're safe in casting away the CONST-ness of this string, since it won't be modified.)
|
|
//
|
|
Source->DescriptionDisplayName = pSetupStringTableAddString(Queue->StringTable,
|
|
(PTSTR)SourceDescription,
|
|
STRTAB_CASE_SENSITIVE
|
|
);
|
|
|
|
if((Source->Description == -1) || (Source->DescriptionDisplayName == -1)) {
|
|
MyFree(Source);
|
|
return(NULL);
|
|
}
|
|
} else {
|
|
Source->Description = Source->DescriptionDisplayName = -1;
|
|
}
|
|
|
|
if(SourceTagfile) {
|
|
//
|
|
// Again, we already lower-cased this in a writeable buffer above.
|
|
//
|
|
Source->Tagfile = pSetupStringTableAddString(Queue->StringTable,
|
|
TempTagfileString,
|
|
STRTAB_CASE_SENSITIVE | STRTAB_ALREADY_LOWERCASE
|
|
);
|
|
if(Source->Tagfile == -1) {
|
|
MyFree(Source);
|
|
return(NULL);
|
|
}
|
|
} else {
|
|
Source->Tagfile = -1;
|
|
}
|
|
|
|
if(SourceCabfile) {
|
|
//
|
|
// Again, we already lower-cased this in a writeable buffer above.
|
|
//
|
|
Source->Cabfile = pSetupStringTableAddString(Queue->StringTable,
|
|
TempCabfileString,
|
|
STRTAB_CASE_SENSITIVE | STRTAB_ALREADY_LOWERCASE
|
|
);
|
|
if(Source->Cabfile == -1) {
|
|
MyFree(Source);
|
|
return(NULL);
|
|
}
|
|
} else {
|
|
Source->Cabfile = Source->Tagfile;
|
|
}
|
|
|
|
Source->SourceRootPath = SourceRootStringId;
|
|
|
|
//
|
|
// insert our media descriptor into the list of descriptors
|
|
// Note: if the new descriptor has the "service pack" or
|
|
// "local cab driver cache" tag set, then we insert it into
|
|
// the head of the list, otherwise we put it into the end
|
|
// of the list. This ensures that if the user get's a
|
|
// need media complaint for os binaries, and overrides
|
|
// the source path, then the user will first be prompted for service
|
|
// pack media, then the os media. This saves us from adding lots of
|
|
// code to handle need media overrides in this case, since we would
|
|
// potentially have the os source files first in the media list, which
|
|
// would cause us to install the os media files instead of the service
|
|
// pack media files
|
|
//
|
|
// another potential service pack issue is if we get Tag==Cab entries mixed with Tag!=Cab
|
|
// for exactly the same cab
|
|
// nothing much we can do here, other than ensure that any change where Tag!=Cab
|
|
// is done across the board
|
|
//
|
|
LastSource = NULL;
|
|
for(TempSource=Queue->SourceMediaList; TempSource; LastSource=TempSource,TempSource=LastSource->Next) {
|
|
if ((Source->Flags ^ TempSource->Flags) & (SMI_FLAG_USE_LOCAL_SOURCE_CAB | SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH)) {
|
|
//
|
|
// one is either normal, local source cab, or source root path, and the other is different
|
|
//
|
|
// SMI_FLAG_USE_LOCAL_SOURCE_CAB inserted before anything else
|
|
//
|
|
if(TempSource->Flags & SMI_FLAG_USE_LOCAL_SOURCE_CAB) {
|
|
//
|
|
// ISSUE-2002/04/19-JamieHun Source->Flags can have this bit set
|
|
// ordering needs to be performed with this in mind
|
|
//
|
|
//MYASSERT(!(Source->Flags & SMI_FLAG_USE_LOCAL_SOURCE_CAB));
|
|
continue;
|
|
}
|
|
if(Source->Flags & SMI_FLAG_USE_LOCAL_SOURCE_CAB) {
|
|
break;
|
|
}
|
|
//
|
|
// SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH comes next
|
|
//
|
|
if(TempSource->Flags & SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH) {
|
|
//
|
|
// ISSUE-2002/04/19-JamieHun Source->Flags can have this bit set
|
|
// ordering needs to be performed with this in mind
|
|
//
|
|
//MYASSERT(!(Source->Flags & SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH));
|
|
continue;
|
|
}
|
|
if(Source->Flags & SMI_FLAG_USE_SVCPACK_SOURCE_ROOT_PATH) {
|
|
break;
|
|
}
|
|
}
|
|
//
|
|
// group same tagfiles together (needed because of tag+cab combinations)
|
|
//
|
|
if( LastSource && (Source->Tagfile == LastSource->Tagfile)
|
|
&& (Source->Tagfile != TempSource->Tagfile)) {
|
|
break;
|
|
}
|
|
}
|
|
if (LastSource) {
|
|
//
|
|
// insert after this one
|
|
//
|
|
Source->Next = LastSource->Next;
|
|
LastSource->Next = Source;
|
|
} else {
|
|
//
|
|
// TempSource will either be NULL (no media) or first media (insert before first)
|
|
//
|
|
Source->Next = TempSource;
|
|
Queue->SourceMediaList = Source;
|
|
}
|
|
|
|
Queue->SourceMediaCount++;
|
|
|
|
QueueNode->SourceMediaInfo = Source;
|
|
return(Source);
|
|
}
|
|
|
|
|
|
BOOL
|
|
pSetupGetSourceAllInfo(
|
|
IN HINF InfHandle,
|
|
IN PINFCONTEXT LayoutLineContext, OPTIONAL
|
|
IN UINT SourceId,
|
|
IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo, OPTIONAL
|
|
OUT PCTSTR *Description,
|
|
OUT PCTSTR *Tagfile,
|
|
OUT PCTSTR *RelativePath,
|
|
OUT PUINT SourceFlags
|
|
)
|
|
{
|
|
BOOL b;
|
|
DWORD RequiredSize;
|
|
PTSTR p;
|
|
DWORD ec;
|
|
TCHAR Buffer[MAX_PATH];
|
|
|
|
//
|
|
// Get path relative to the source.
|
|
//
|
|
b = pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_PATH,
|
|
NULL,
|
|
0,
|
|
&RequiredSize
|
|
);
|
|
if(!b) {
|
|
ec = GetLastError();
|
|
goto clean0;
|
|
}
|
|
|
|
p = MyMalloc(RequiredSize*sizeof(TCHAR));
|
|
if(!p) {
|
|
ec = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean0;
|
|
}
|
|
pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_PATH,
|
|
p,
|
|
RequiredSize,
|
|
NULL
|
|
);
|
|
*RelativePath = p;
|
|
|
|
//
|
|
// Get description.
|
|
//
|
|
b = pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_DESCRIPTION,
|
|
NULL,
|
|
0,
|
|
&RequiredSize
|
|
);
|
|
if(!b) {
|
|
ec = GetLastError();
|
|
goto clean1;
|
|
}
|
|
|
|
p = MyMalloc(RequiredSize*sizeof(TCHAR));
|
|
if(!p) {
|
|
ec = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean1;
|
|
}
|
|
pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_DESCRIPTION,
|
|
p,
|
|
RequiredSize,
|
|
NULL
|
|
);
|
|
*Description = p;
|
|
|
|
//
|
|
// Get tagfile, if any.
|
|
//
|
|
b = pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_TAGFILE,
|
|
NULL,
|
|
0,
|
|
&RequiredSize
|
|
);
|
|
if(!b) {
|
|
ec = GetLastError();
|
|
goto clean2;
|
|
}
|
|
|
|
p = MyMalloc(RequiredSize*sizeof(TCHAR));
|
|
if(!p) {
|
|
ec = ERROR_NOT_ENOUGH_MEMORY;
|
|
goto clean2;
|
|
}
|
|
pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_TAGFILE,
|
|
p,
|
|
RequiredSize,
|
|
NULL
|
|
);
|
|
if(*p) {
|
|
*Tagfile = p;
|
|
} else {
|
|
MyFree(p);
|
|
*Tagfile = NULL;
|
|
}
|
|
|
|
//
|
|
// Get flags, if any.
|
|
//
|
|
b = pSetupGetSourceInfo(InfHandle,
|
|
LayoutLineContext,
|
|
SourceId,
|
|
AltPlatformInfo,
|
|
SRCINFO_FLAGS,
|
|
Buffer,
|
|
SIZECHARS(Buffer),
|
|
NULL
|
|
);
|
|
if(!b) {
|
|
ec = GetLastError();
|
|
goto clean3;
|
|
}
|
|
|
|
pAToI( Buffer, SourceFlags );
|
|
|
|
|
|
return(TRUE);
|
|
|
|
clean3:
|
|
MyFree(*Tagfile);
|
|
clean2:
|
|
MyFree(*Description);
|
|
clean1:
|
|
MyFree(*RelativePath);
|
|
clean0:
|
|
SetLastError(ec);
|
|
return(FALSE);
|
|
}
|
|
|
|
|
|
BOOL
|
|
pIsFileInDriverCache(
|
|
IN HINF CabInf,
|
|
IN PCTSTR TargetFilename,
|
|
IN PCTSTR SubDirectory,
|
|
OUT PCTSTR *CacheName
|
|
)
|
|
{
|
|
INFCONTEXT Context,SectionContext;
|
|
PCTSTR SectionName,CabName;
|
|
TCHAR TempBuffer[MAX_PATH];
|
|
|
|
UINT Field, FieldCount;
|
|
|
|
MYASSERT(CabInf != INVALID_HANDLE_VALUE);
|
|
MYASSERT(TargetFilename);
|
|
MYASSERT(CacheName);
|
|
|
|
if (!SetupFindFirstLine(CabInf, TEXT("Version"), TEXT("CabFiles"), &SectionContext)) {
|
|
return(FALSE);
|
|
}
|
|
|
|
do {
|
|
|
|
|
|
FieldCount = SetupGetFieldCount(&SectionContext);
|
|
for(Field=1; Field<=FieldCount; Field++) {
|
|
|
|
SectionName = pSetupGetField(&SectionContext,Field);
|
|
|
|
if (SetupFindFirstLine(CabInf,SectionName,TargetFilename,&Context)) {
|
|
//
|
|
// we found a match
|
|
//
|
|
if (SetupFindFirstLine(CabInf,TEXT("Cabs"),SectionName,&Context)) {
|
|
CabName= pSetupGetField(&Context,1);
|
|
//if (pIsDriverCachePresent(CabName,SubDirectory,TempBuffer)) {
|
|
*CacheName = DuplicateString( CabName );
|
|
if (*CacheName) {
|
|
return(TRUE);
|
|
}
|
|
//}
|
|
}
|
|
}
|
|
} // end for
|
|
|
|
} while (SetupFindNextMatchLine(&SectionContext,TEXT("CabFiles"),&SectionContext));
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
BOOL
|
|
pIsDriverCachePresent(
|
|
IN PCTSTR DriverName,
|
|
IN PCTSTR Subdirectory,
|
|
IN OUT PTSTR DriverBuffer
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Looks at the proper location for the driver cache cab-file, and if it's
|
|
present, return TRUE. If present, it returns the partial path to the
|
|
cab file
|
|
|
|
Arguments:
|
|
|
|
DriveName - the cab file we're looking for
|
|
|
|
Subdirectory - if specified, use this as the subdirectory from the root of the driver
|
|
cache, otherwise use the specified architecture's subdirectory
|
|
|
|
DriverBuffer - if the cab file is present, return the source root to the cab file
|
|
|
|
|
|
Return Value:
|
|
|
|
TRUE if the cab file is present
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
TCHAR TempBuffer[MAX_PATH];
|
|
|
|
if (!DriverCacheSourcePath || !DriverName) {
|
|
return FALSE;
|
|
}
|
|
|
|
if (!Subdirectory) {
|
|
Subdirectory =
|
|
#if defined(_AXP64_)
|
|
TEXT("axp64");
|
|
#elif defined(_ALPHA_)
|
|
TEXT("alpha");
|
|
#elif defined(_X86_)
|
|
IsNEC98() ? TEXT("nec98") : TEXT("i386");
|
|
#elif defined(_IA64_)
|
|
TEXT("ia64");
|
|
#elif defined(_AMD64_)
|
|
TEXT("amd64");
|
|
#endif
|
|
}
|
|
|
|
lstrcpy(TempBuffer, DriverCacheSourcePath);
|
|
pSetupConcatenatePaths(TempBuffer, Subdirectory , MAX_PATH, NULL);
|
|
pSetupConcatenatePaths(TempBuffer, DriverName, MAX_PATH, NULL);
|
|
|
|
if (FileExists(TempBuffer,NULL)) {
|
|
lstrcpy(DriverBuffer,DriverCacheSourcePath);
|
|
return(TRUE);
|
|
}
|
|
|
|
return(FALSE);
|
|
|
|
}
|