/*++ Copyright (c) Microsoft Corporation. All rights reserved. Module Name: infopenv.c Abstract: Externally exposed INF routines for INF opening, closing, and versioning. Author: Ted Miller (tedm) 20-Jan-1995 Revision History: Jamie Hunter (JamieHun) May-2-2002 Security code review --*/ #include "precomp.h" #pragma hdrstop BOOL pSetupVersionNodeFromInfInformation( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, OUT PINF_VERSION_NODE VersionNode, OUT PTSTR OriginalFilename OPTIONAL ); #ifdef UNICODE // // ANSI version // BOOL SetupGetInfInformationA( IN LPCVOID InfSpec, IN DWORD SearchControl, OUT PSP_INF_INFORMATION ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { PCWSTR infspec; BOOL b; DWORD rc; // // For this API, the return buffer does not have to be translated // from Unicode to ANSI. This makes things much easier since the // required size is the same for the ANSI and Unicode versions. // if((SearchControl == INFINFO_INF_NAME_IS_ABSOLUTE) || (SearchControl == INFINFO_DEFAULT_SEARCH) || (SearchControl == INFINFO_REVERSE_DEFAULT_SEARCH) || (SearchControl == INFINFO_INF_PATH_LIST_SEARCH)) { rc = pSetupCaptureAndConvertAnsiArg(InfSpec,&infspec); if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } } else { // // Not a pointer to a string, just pass it on. // infspec = InfSpec; } // // Note that the data returned from this API is in an // internal format, and thus we don't need any less space // for the ANSI API, and can just use the buffer and sizes // passed in by the caller. // b = SetupGetInfInformationW( infspec, SearchControl, ReturnBuffer, ReturnBufferSize, RequiredSize ); rc = GetLastError(); if(infspec != InfSpec) { MyFree(infspec); } SetLastError(rc); return(b); } #else // // Unicode stub // BOOL SetupGetInfInformationW( IN LPCVOID InfSpec, IN DWORD SearchControl, OUT PSP_INF_INFORMATION ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { UNREFERENCED_PARAMETER(InfSpec); UNREFERENCED_PARAMETER(SearchControl); UNREFERENCED_PARAMETER(ReturnBuffer); UNREFERENCED_PARAMETER(ReturnBufferSize); UNREFERENCED_PARAMETER(RequiredSize); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL SetupGetInfInformation( IN LPCVOID InfSpec, IN DWORD SearchControl, OUT PSP_INF_INFORMATION ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) /*++ Routine Description: Arguments: Return Value: --*/ { BOOL UnloadInf; PLOADED_INF Inf, CurInf; UINT InfCount; PUCHAR Out; DWORD TotalSpaceRequired; DWORD d; DWORD ErrorLineNumber; TCHAR Path[MAX_PATH]; PINF_VERSION_NODE VersionNode; INF_VERSION_BLOCK UNALIGNED *Prev; BOOL TryPnf; WIN32_FIND_DATA FindData; PTSTR DontCare; UINT OriginalFilenameSize; // // Set up some state based on the SearchSpec parameter. // Inf = NULL; switch(SearchControl) { case INFINFO_INF_SPEC_IS_HINF: Inf = (PLOADED_INF)InfSpec; d = NO_ERROR; try { if (!LockInf(Inf)) { d = ERROR_INVALID_HANDLE; } } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_HANDLE; } if (d != NO_ERROR) { SetLastError(d); return FALSE; } break; case INFINFO_INF_NAME_IS_ABSOLUTE: // // Make sure we have a fully-qualified path. // d = GetFullPathName((PCTSTR)InfSpec, SIZECHARS(Path), Path, &DontCare ); if(!d) { // // LastError has already been set // (unless InfSpec was NULL or "") // if (GetLastError()==NO_ERROR) { SetLastError(ERROR_FILE_NOT_FOUND); } return FALSE; } else if(d >= SIZECHARS(Path)) { MYASSERT(0); SetLastError(ERROR_BUFFER_OVERFLOW); return FALSE; } if(FileExists(Path, &FindData)) { InfSourcePathFromFileName(Path, NULL, &TryPnf); break; } else { // // LastError has already been set. // return FALSE; } case INFINFO_DEFAULT_SEARCH: case INFINFO_REVERSE_DEFAULT_SEARCH: case INFINFO_INF_PATH_LIST_SEARCH: try { d = SearchForInfFile((PCTSTR)InfSpec, &FindData, SearchControl, Path, SIZECHARS(Path), NULL ); } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } if(d == NO_ERROR) { TryPnf = TRUE; break; } else { SetLastError(d); return FALSE; } default: SetLastError(ERROR_INVALID_PARAMETER); return(FALSE); } // // Load the inf if necessary. // if(Inf) { UnloadInf = FALSE; } else { d = LoadInfFile(Path, &FindData, INF_STYLE_ALL, TryPnf ? LDINF_FLAG_ALWAYS_TRY_PNF : 0, NULL, NULL, NULL, NULL, NULL, // LogContext &Inf, &ErrorLineNumber, NULL ); if(d != NO_ERROR) { SetLastError(d); return(FALSE); } UnloadInf = TRUE; } // // Determine the number of infs associated with this handle, // and calculate the amount of space that will be needed to // store version information about them. // // For each inf we will need space for the version block, // as well as an offset in the SP_INF_INFORMATION structure // to indicate where that inf's version block is located // in the output buffer. // TotalSpaceRequired = offsetof(SP_INF_INFORMATION, VersionData); for(InfCount = 0, CurInf = Inf; CurInf; InfCount++, CurInf = CurInf->Next) { OriginalFilenameSize = CurInf->OriginalInfName ? (lstrlen(CurInf->OriginalInfName) + 1) * sizeof(TCHAR) : 0; TotalSpaceRequired += (offsetof(INF_VERSION_BLOCK, Filename) + CurInf->VersionBlock.FilenameSize + CurInf->VersionBlock.DataSize + OriginalFilenameSize ); } if(RequiredSize) { *RequiredSize = TotalSpaceRequired; } // // See if we have a large enough output buffer. // If we have a large enough buffer then set up some // initial values in it. // if(ReturnBufferSize < TotalSpaceRequired) { if(UnloadInf) { FreeInfFile(Inf); } else { UnlockInf(Inf); } if(ReturnBuffer) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } else { return TRUE; } } d = NO_ERROR; try { ReturnBuffer->InfStyle = Inf->Style; ReturnBuffer->InfCount = InfCount; } except(EXCEPTION_EXECUTE_HANDLER) { if(UnloadInf) { FreeInfFile(Inf); } else { UnlockInf(Inf); } SetLastError(d = ERROR_INVALID_PARAMETER); } if(d != NO_ERROR) { return FALSE; } Out = (PUCHAR)ReturnBuffer + offsetof(SP_INF_INFORMATION, VersionData); // // Traverse all infs associated with this inf handle and copy // version data into the caller's buffer. Guard with SEH to ensure // that the caller passed a valid buffer. // try { Prev = NULL; for(CurInf = Inf; CurInf; CurInf = CurInf->Next) { // // Store offset into // if(Prev) { Prev->NextOffset = (UINT)((UINT_PTR)Out - (UINT_PTR)ReturnBuffer); } Prev = (PVOID)Out; OriginalFilenameSize = CurInf->OriginalInfName ? (lstrlen(CurInf->OriginalInfName) + 1) * sizeof(TCHAR) : 0; Prev->LastWriteTime = CurInf->VersionBlock.LastWriteTime; Prev->DatumCount = CurInf->VersionBlock.DatumCount; Prev->OffsetToData = CurInf->VersionBlock.FilenameSize + OriginalFilenameSize; Prev->DataSize = CurInf->VersionBlock.DataSize; Prev->TotalSize = offsetof(INF_VERSION_BLOCK, Filename) + CurInf->VersionBlock.FilenameSize + OriginalFilenameSize + CurInf->VersionBlock.DataSize; Out += offsetof(INF_VERSION_BLOCK, Filename); // // Now copy the filename, (optionally) original filename, and // version data into the output buffer. // CopyMemory(Out, CurInf->VersionBlock.Filename, CurInf->VersionBlock.FilenameSize); Out += CurInf->VersionBlock.FilenameSize; if(CurInf->OriginalInfName) { CopyMemory(Out, CurInf->OriginalInfName, OriginalFilenameSize); Out += OriginalFilenameSize; } CopyMemory(Out, CurInf->VersionBlock.DataBlock, CurInf->VersionBlock.DataSize); Out += CurInf->VersionBlock.DataSize; } } except(EXCEPTION_EXECUTE_HANDLER) { if(UnloadInf) { FreeInfFile(Inf); } else { UnlockInf(Inf); } SetLastError(d = ERROR_INVALID_PARAMETER); } if(d != NO_ERROR) { return FALSE; } Prev->NextOffset = 0; // // Unload the inf if necessary // if(UnloadInf) { FreeInfFile(Inf); } else { UnlockInf(Inf); } return TRUE; } #ifdef UNICODE // // ANSI version // BOOL SetupQueryInfFileInformationA( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, OUT PSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { WCHAR returnbuffer[MAX_PATH]; DWORD requiredsize; DWORD rc; PSTR ansi; BOOL b; b = SetupQueryInfFileInformationW( InfInformation, InfIndex, returnbuffer, MAX_PATH, &requiredsize ); rc = GetLastError(); if(b) { if(ansi = pSetupUnicodeToAnsi(returnbuffer)) { rc = NO_ERROR; requiredsize = lstrlenA(ansi)+1; if(RequiredSize) { try { *RequiredSize = requiredsize; } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; b = FALSE; } } if(b) { if(ReturnBuffer) { if(ReturnBufferSize >= requiredsize) { // // lstrcpy returns NULL if it faults // if(!lstrcpyA(ReturnBuffer,ansi)) { rc = ERROR_INVALID_PARAMETER; b = FALSE; } } else { b = FALSE; rc = ERROR_INSUFFICIENT_BUFFER; } } } MyFree(ansi); } else { rc = ERROR_NOT_ENOUGH_MEMORY; b = FALSE; } } SetLastError(rc); return(b); } #else // // Unicode stub // BOOL SetupQueryInfFileInformationW( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, OUT PWSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { UNREFERENCED_PARAMETER(InfInformation); UNREFERENCED_PARAMETER(InfIndex); UNREFERENCED_PARAMETER(ReturnBuffer); UNREFERENCED_PARAMETER(ReturnBufferSize); UNREFERENCED_PARAMETER(RequiredSize); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL SetupQueryInfFileInformation( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, OUT PTSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) /*++ Routine Description: Arguments: Return Value: --*/ { UINT FilenameLength; INF_VERSION_NODE VersionNode; DWORD rc; // // See whether the index is in range and // retrieve the version descriptor for this inf. // rc = NO_ERROR; try { if(!pSetupVersionNodeFromInfInformation(InfInformation,InfIndex,&VersionNode,NULL)) { rc = ERROR_INVALID_PARAMETER; } } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } FilenameLength = VersionNode.FilenameSize / sizeof(TCHAR); if(RequiredSize) { try { *RequiredSize = FilenameLength; } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } } // // Check length of user's buffer. // if(FilenameLength > ReturnBufferSize) { if(ReturnBuffer) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } else { return TRUE; } } // // Copy the data into user's buffer. // try { CopyMemory(ReturnBuffer,VersionNode.Filename,VersionNode.FilenameSize); } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } return TRUE; } #ifdef UNICODE // // ANSI version // BOOL WINAPI SetupQueryInfOriginalFileInformationA( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PSP_ALTPLATFORM_INFO_V2 AlternatePlatformInfo, OPTIONAL OUT PSP_ORIGINAL_FILE_INFO_A OriginalFileInfo ) { SP_ORIGINAL_FILE_INFO_W UnicodeOriginalFileInfo; DWORD rc; int i; BOOL b; rc = NO_ERROR; // // Do an initial check on user-supplied output buffer to see if it seems // to be valid. // try { if(OriginalFileInfo->cbSize != sizeof(SP_ORIGINAL_FILE_INFO_A)) { rc = ERROR_INVALID_USER_BUFFER; } } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } if(rc != NO_ERROR) { SetLastError(rc); return FALSE; } UnicodeOriginalFileInfo.cbSize = sizeof(SP_ORIGINAL_FILE_INFO_W); b = SetupQueryInfOriginalFileInformationW( InfInformation, InfIndex, AlternatePlatformInfo, &UnicodeOriginalFileInfo ); rc = GetLastError(); if(b) { // // Convert the Unicode fields of the original file info structure into // ANSI, and store the information in the caller-supplied ANSI // structure. // try { // // First, translate/store the original INF name... // i = WideCharToMultiByte( CP_ACP, 0, UnicodeOriginalFileInfo.OriginalInfName, -1, OriginalFileInfo->OriginalInfName, SIZECHARS(OriginalFileInfo->OriginalInfName), NULL, NULL ); // // ...and if that succeeded, then translate/store the original // catalog filename. // if(i) { // // Note that the original catalog filename may be the empty // string (i.e., the INF didn't specify an associated catalog // file). We don't need to special-case this, since // WideCharToMultiByte can handle empty strings just fine. // i = WideCharToMultiByte( CP_ACP, 0, UnicodeOriginalFileInfo.OriginalCatalogName, -1, OriginalFileInfo->OriginalCatalogName, SIZECHARS(OriginalFileInfo->OriginalCatalogName), NULL, NULL ); } if(!i) { b = FALSE; rc = GetLastError(); // // If we start seeing cases where our Unicode->ANSI expansion // blows our buffersize, we need to know about it... // MYASSERT((rc != NO_ERROR) && (rc != ERROR_INSUFFICIENT_BUFFER)); } } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; b = FALSE; } } SetLastError(rc); return b; } #else // // Unicode stub // BOOL WINAPI SetupQueryInfOriginalFileInformationW( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PSP_ALTPLATFORM_INFO_V2 AlternatePlatformInfo, OPTIONAL OUT PSP_ORIGINAL_FILE_INFO_W OriginalFileInfo ) { UNREFERENCED_PARAMETER(InfInformation); UNREFERENCED_PARAMETER(InfIndex); UNREFERENCED_PARAMETER(AlternatePlatformInfo); UNREFERENCED_PARAMETER(OriginalFileInfo); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL WINAPI SetupQueryInfOriginalFileInformation( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PSP_ALTPLATFORM_INFO_V2 AlternatePlatformInfo, OPTIONAL OUT PSP_ORIGINAL_FILE_INFO OriginalFileInfo ) /*++ Routine Description: This routine returns an INF's original name (which will be different from its current name if the INF was installed into %windir%\Inf, for example). If the INF's original name is the same as its current name, the current name is returned. It also returns the original filename of the catalog file specified by the INF via a (potentially decorated) CatalogFile= entry in the INF's [version] section. The OS/architecture-specific decoration may be overridden from the default (i.e., current platform) by passing in an optional alternate platform information structure. If the INF doesn't specify any catalog file, then this field in the output OriginalFileInfo structure will be set to an empty string. Both filenames returned in the OriginalFileInfo are simple filenames, i.e., there is no path information included. Arguments: InfInformation - supplies context from which we retrieve information about the INF whose index is specified by InfIndex. InfIndex - supplies the zero-based index of the INF within the InfInformation context buffer that we're retrieving original file information for. AlternatePlatformInfo - optionally, supplies alternate platform information used when searching for the appropriately decorated CatalogFile= entry within the INF's [version] section. (NOTE: caller may actually pass in a V1 struct instead--we detect this case and convert the V1 struct into a V2 one.) OriginalFileInfo - supplies the address of an original file information buffer that upon success receives information about the original (simple) filenames of files associated with this INF. This structure must have its cbSize field set to sizeof(SP_ORIGINAL_FILE_INFO) upon entry to this routine or the call will fail with GetLastError() returning ERROR_INVALID_USER_BUFFER. The fields of this structure are set upon successful return as follows: OriginalInfName - receives the INF's original filename, which may be different than its current filename in the case where the INF was an OEM in that was installed into the %windir%\Inf directory (e.g., via SetupCopyOEMInf). OriginalCatalogName - receives the platform-appropriate CatalogFile= entry in the INF's [version] section (where the platform is the default native one unless AlternatePlatformInfo is supplied). If there is no applicable CatalogFile= entry, this field will be set to the empty string. Return Value: If successful, the return value is non-zero. If unsuccessful, the return value is FALSE, and GetLastError() may be called to determine the cause of failure. --*/ { INF_VERSION_NODE VersionNode; DWORD rc; SP_ALTPLATFORM_INFO_V2 AltPlatformInfoV2; rc = NO_ERROR; // // See whether the index is in range and retrieve the version descriptor // and original filename for this inf. // try { // // Do an initial check on user-supplied output buffer to see if it // seems to be valid. // if(OriginalFileInfo->cbSize != sizeof(SP_ORIGINAL_FILE_INFO)) { rc = ERROR_INVALID_USER_BUFFER; goto clean0; } // // Now validate the AlternatePlatformInfo parameter. // if(AlternatePlatformInfo) { if(AlternatePlatformInfo->cbSize != sizeof(SP_ALTPLATFORM_INFO_V2)) { // // The caller may have passed us in a Version 1 struct, or they // may have passed us in bad data... // if(AlternatePlatformInfo->cbSize == sizeof(SP_ALTPLATFORM_INFO_V1)) { // // Flags/Reserved field is reserved in V1 // if(AlternatePlatformInfo->Reserved) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // Convert the caller-supplied data into Version 2 format. // ZeroMemory(&AltPlatformInfoV2, sizeof(AltPlatformInfoV2)); AltPlatformInfoV2.cbSize = sizeof(SP_ALTPLATFORM_INFO_V2); AltPlatformInfoV2.Platform = ((PSP_ALTPLATFORM_INFO_V1)AlternatePlatformInfo)->Platform; AltPlatformInfoV2.MajorVersion = ((PSP_ALTPLATFORM_INFO_V1)AlternatePlatformInfo)->MajorVersion; AltPlatformInfoV2.MinorVersion = ((PSP_ALTPLATFORM_INFO_V1)AlternatePlatformInfo)->MinorVersion; AltPlatformInfoV2.ProcessorArchitecture = ((PSP_ALTPLATFORM_INFO_V1)AlternatePlatformInfo)->ProcessorArchitecture; AltPlatformInfoV2.Flags = 0; AlternatePlatformInfo = &AltPlatformInfoV2; } else { rc = ERROR_INVALID_USER_BUFFER; goto clean0; } } // // Gotta be either Windows or Windows NT // if((AlternatePlatformInfo->Platform != VER_PLATFORM_WIN32_WINDOWS) && (AlternatePlatformInfo->Platform != VER_PLATFORM_WIN32_NT)) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // Processor had better be either i386, amd64, or ia64 // if((AlternatePlatformInfo->ProcessorArchitecture != PROCESSOR_ARCHITECTURE_INTEL) && (AlternatePlatformInfo->ProcessorArchitecture != PROCESSOR_ARCHITECTURE_IA64) && (AlternatePlatformInfo->ProcessorArchitecture != PROCESSOR_ARCHITECTURE_AMD64)) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // MajorVersion field must be non-zero (MinorVersion field can be // anything) // if(!AlternatePlatformInfo->MajorVersion) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // Validate structure parameter flags (bits indicating what // parts of the structure are valid). // if((AlternatePlatformInfo->Flags & ~ (SP_ALTPLATFORM_FLAGS_VERSION_RANGE)) != 0) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // fill in version validation range if none supplied by caller // if((AlternatePlatformInfo->Flags & SP_ALTPLATFORM_FLAGS_VERSION_RANGE) == 0) { // // If caller does not know about FirstValidate*Version, // version upper and lower bounds are equal. // AlternatePlatformInfo->FirstValidatedMajorVersion = AlternatePlatformInfo->MajorVersion; AlternatePlatformInfo->FirstValidatedMinorVersion = AlternatePlatformInfo->MinorVersion; AlternatePlatformInfo->Flags |= SP_ALTPLATFORM_FLAGS_VERSION_RANGE; } } // // OK, now retrieve the INF's original filename... // if(!pSetupVersionNodeFromInfInformation(InfInformation, InfIndex, &VersionNode, OriginalFileInfo->OriginalInfName)) { rc = ERROR_INVALID_PARAMETER; goto clean0; } // // ...and retrieve the (platform-appropriate) catalog file associated // with this INF (if there is one). // if(!pSetupGetCatalogFileValue(&VersionNode, OriginalFileInfo->OriginalCatalogName, SIZECHARS(OriginalFileInfo->OriginalCatalogName), AlternatePlatformInfo)) { // // No applicable CatalogFile= entry found--set field to empty // string. // *(OriginalFileInfo->OriginalCatalogName) = TEXT('\0'); } clean0: ; // nothing to do } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } SetLastError(rc); return (rc == NO_ERROR); } #ifdef UNICODE // // ANSI version // BOOL SetupQueryInfVersionInformationA( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PCSTR Key, OPTIONAL OUT PSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { INF_VERSION_NODE VersionNode; PCWSTR Data; DWORD rc; PSTR ansidata; UINT ansilength; PCWSTR key; // // See whether the index is in range and // get pointer to version descriptor for this inf. // try { if(pSetupVersionNodeFromInfInformation(InfInformation,InfIndex,&VersionNode,NULL)) { // // See whether we want a specific value. // if(Key) { rc = pSetupCaptureAndConvertAnsiArg(Key,&key); if(rc == NO_ERROR) { if(Data = pSetupGetVersionDatum(&VersionNode,key)) { if(ansidata = pSetupUnicodeToAnsi(Data)) { ansilength = lstrlenA(ansidata) + 1; if(RequiredSize) { *RequiredSize = ansilength; } if(ReturnBuffer) { if(ReturnBufferSize >= ansilength) { CopyMemory(ReturnBuffer,ansidata,ansilength); rc = NO_ERROR; } else { rc = ERROR_INSUFFICIENT_BUFFER; } } else { rc = NO_ERROR; } MyFree(ansidata); } else { rc = ERROR_NOT_ENOUGH_MEMORY; } } else { rc = ERROR_INVALID_DATA; } MyFree(key); } } else { // // Caller wants all values. Copy whole data block to caller's buffer, // plus a terminating NUL character. // // Maximum size the data could be in ansi is the exact same // size it is in unicode, if every char is a double-byte char. // if(ansidata = MyMalloc(VersionNode.DataSize)) { ansilength = WideCharToMultiByte( CP_ACP, 0, (PWSTR)VersionNode.DataBlock, VersionNode.DataSize / sizeof(WCHAR), ansidata, VersionNode.DataSize, NULL, NULL ); if(RequiredSize) { // // account for terminating nul // *RequiredSize = ansilength+1; } if(ReturnBuffer) { if(ReturnBufferSize >= *RequiredSize) { CopyMemory(ReturnBuffer,ansidata,ansilength); ReturnBuffer[ansilength] = 0; rc = NO_ERROR; } else { rc = ERROR_INSUFFICIENT_BUFFER; } } else { rc = NO_ERROR; } MyFree(ansidata); } else { rc = ERROR_NOT_ENOUGH_MEMORY; } } } else { rc = ERROR_INVALID_PARAMETER; } } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } SetLastError(rc); return(rc == NO_ERROR); } #else // // Unicode stub // BOOL SetupQueryInfVersionInformationW( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PCWSTR Key, OPTIONAL OUT PWSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { UNREFERENCED_PARAMETER(InfInformation); UNREFERENCED_PARAMETER(InfIndex); UNREFERENCED_PARAMETER(Key); UNREFERENCED_PARAMETER(ReturnBuffer); UNREFERENCED_PARAMETER(ReturnBufferSize); UNREFERENCED_PARAMETER(RequiredSize); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL SetupQueryInfVersionInformation( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, IN PCTSTR Key, OPTIONAL OUT PTSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) /*++ Routine Description: Arguments: Return Value: --*/ { INF_VERSION_NODE VersionNode; PCTSTR Data; UINT DataLength; DWORD rc; // // See whether the index is in range and // get pointer to version descriptor for this inf. // try { if(pSetupVersionNodeFromInfInformation(InfInformation,InfIndex,&VersionNode,NULL)) { // // See whether we want a specific value. // if(Key) { if(Data = pSetupGetVersionDatum(&VersionNode,Key)) { DataLength = lstrlen(Data) + 1; if(RequiredSize) { *RequiredSize = DataLength; } if(ReturnBuffer) { if(ReturnBufferSize >= DataLength) { CopyMemory(ReturnBuffer,Data,DataLength * sizeof(TCHAR)); rc = NO_ERROR; } else { rc = ERROR_INSUFFICIENT_BUFFER; } } else { rc = NO_ERROR; } } else { rc = ERROR_INVALID_DATA; } } else { // // Caller wants all values. Copy whole data block to caller's buffer, // plus a terminating NUL character. // DataLength = (VersionNode.DataSize / sizeof(TCHAR)) + 1; if(RequiredSize) { *RequiredSize = DataLength; } if(ReturnBuffer) { if(ReturnBufferSize >= DataLength) { CopyMemory(ReturnBuffer,VersionNode.DataBlock,VersionNode.DataSize); ReturnBuffer[VersionNode.DataSize/sizeof(TCHAR)] = 0; rc = NO_ERROR; } else { rc = ERROR_INSUFFICIENT_BUFFER; } } else { rc = NO_ERROR; } } } else { rc = ERROR_INVALID_PARAMETER; } } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; } SetLastError(rc); return(rc == NO_ERROR); } BOOL _SetupGetInfFileList( IN PCTSTR DirectoryPath, OPTIONAL IN DWORD InfStyle, OUT PVOID ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL #ifdef UNICODE IN ,BOOL ConvertToAnsi #endif ) { TCHAR SearchSpec[MAX_PATH]; PTCHAR FilenameStart; WIN32_FIND_DATA FindData; HANDLE FindHandle; DWORD Style; UINT FileNameLength; DWORD RemainingSpaceInBuffer; DWORD CurrentOffsetInBuffer; DWORD TotalSpaceNeededInBuffer; BOOL InsufficientBuffer; DWORD d; PTSTR DontCare; #ifdef UNICODE CHAR ansi[MAX_PATH]; #endif // // Set up the search directory // if(DirectoryPath) { // // Make sure this directory path is fully-qualified. // d = GetFullPathName(DirectoryPath, SIZECHARS(SearchSpec), SearchSpec, &DontCare ); if(!d) { // // LastError has already been set. // return FALSE; } else if(d >= SIZECHARS(SearchSpec)) { MYASSERT(0); SetLastError(ERROR_BUFFER_OVERFLOW); return FALSE; } } else { lstrcpyn(SearchSpec, InfDirectory,SIZECHARS(SearchSpec)); } pSetupConcatenatePaths(SearchSpec, pszInfWildcard, SIZECHARS(SearchSpec), NULL); FilenameStart = (PTSTR)pSetupGetFileTitle(SearchSpec); FindHandle = FindFirstFile(SearchSpec,&FindData); if(FindHandle == INVALID_HANDLE_VALUE) { d = GetLastError(); if((d == ERROR_NO_MORE_FILES) || (d == ERROR_FILE_NOT_FOUND) || (d == ERROR_PATH_NOT_FOUND)) { if(RequiredSize) { d = NO_ERROR; try { *RequiredSize = 1; } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } if(d != NO_ERROR) { SetLastError(d); return(FALSE); } } if(ReturnBuffer) { if(ReturnBufferSize) { d = NO_ERROR; try { #ifdef UNICODE if(ConvertToAnsi) { *(PCHAR)ReturnBuffer = 0; } else #endif *(PTCHAR)ReturnBuffer = 0; } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } SetLastError(d); return(d == NO_ERROR); } else { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } } else { return TRUE; } } SetLastError(d); return(FALSE); } // // Leave space for the extra terminating nul char. // RemainingSpaceInBuffer = ReturnBufferSize; if(RemainingSpaceInBuffer) { RemainingSpaceInBuffer--; } TotalSpaceNeededInBuffer = 1; CurrentOffsetInBuffer = 0; InsufficientBuffer = FALSE; d = NO_ERROR; do { // // Skip directories // if(FindData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) { continue; } // // Form full pathname of file in SearchSpec. // lstrcpy(FilenameStart,FindData.cFileName); // // Determine the inf type and see whether the caller // wants to know about infs of this type. // Style = DetermineInfStyle(SearchSpec, &FindData); if((Style == INF_STYLE_NONE) || !(Style & InfStyle)) { continue; } // // Got a legit inf file. // #ifdef UNICODE if(ConvertToAnsi) { // // The nul is included because it's converted // so no need to add 1 // FileNameLength = WideCharToMultiByte( CP_ACP, 0, FindData.cFileName, -1, ansi, MAX_PATH, NULL, NULL ); } else #endif FileNameLength = lstrlen(FindData.cFileName) + 1; TotalSpaceNeededInBuffer += FileNameLength; if(ReturnBuffer) { if(RemainingSpaceInBuffer >= FileNameLength ) { RemainingSpaceInBuffer -= FileNameLength; // // lstrcpy will return NULL if it faults // #ifdef UNICODE if(ConvertToAnsi) { DontCare = (PVOID)lstrcpyA((PCHAR)ReturnBuffer+CurrentOffsetInBuffer,ansi); } else #endif DontCare = lstrcpy((PTCHAR)ReturnBuffer+CurrentOffsetInBuffer,FindData.cFileName); if(!DontCare) { d = ERROR_INVALID_PARAMETER; } else { CurrentOffsetInBuffer += FileNameLength; try { #ifdef UNICODE if(ConvertToAnsi) { ((PCHAR)ReturnBuffer)[CurrentOffsetInBuffer] = 0; } else #endif ((PTCHAR)ReturnBuffer)[CurrentOffsetInBuffer] = 0; } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } } } else { InsufficientBuffer = TRUE; } } } while((d == NO_ERROR) && FindNextFile(FindHandle,&FindData)); FindClose(FindHandle); if(d != NO_ERROR) { SetLastError(d); } if(GetLastError() == ERROR_NO_MORE_FILES) { d = NO_ERROR; try { if(RequiredSize) { *RequiredSize = TotalSpaceNeededInBuffer; } } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } if(d == NO_ERROR) { if(InsufficientBuffer) { SetLastError(ERROR_INSUFFICIENT_BUFFER); return FALSE; } return(TRUE); } else { SetLastError(d); } } // // Last error already set // return(FALSE); } #ifdef UNICODE // // ANSI version // BOOL SetupGetInfFileListA( IN PCSTR DirectoryPath, OPTIONAL IN DWORD InfStyle, OUT PSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { PWSTR dirpath; DWORD rc; BOOL b; if(DirectoryPath) { rc = pSetupCaptureAndConvertAnsiArg(DirectoryPath,&dirpath); if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } } else { dirpath = NULL; } b = _SetupGetInfFileList(dirpath,InfStyle,ReturnBuffer,ReturnBufferSize,RequiredSize,TRUE); rc = GetLastError(); if(dirpath) { MyFree(dirpath); } SetLastError(rc); return(b); } #else // // Unicode stub // BOOL SetupGetInfFileListW( IN PCWSTR DirectoryPath, OPTIONAL IN DWORD InfStyle, OUT PWSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { UNREFERENCED_PARAMETER(DirectoryPath); UNREFERENCED_PARAMETER(InfStyle); UNREFERENCED_PARAMETER(ReturnBuffer); UNREFERENCED_PARAMETER(ReturnBufferSize); UNREFERENCED_PARAMETER(RequiredSize); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL SetupGetInfFileList( IN PCTSTR DirectoryPath, OPTIONAL IN DWORD InfStyle, OUT PTSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) /*++ Routine Description: Arguments: Return Value: --*/ { PTSTR dirpath; DWORD rc; BOOL b; if(DirectoryPath) { rc = CaptureStringArg(DirectoryPath,&dirpath); if(rc != NO_ERROR) { SetLastError(rc); return(FALSE); } } else { dirpath = NULL; } b = _SetupGetInfFileList( dirpath, InfStyle, ReturnBuffer, ReturnBufferSize, RequiredSize #ifdef UNICODE ,FALSE #endif ); rc = GetLastError(); if(dirpath) { MyFree(dirpath); } SetLastError(rc); return(b); } #ifdef UNICODE // // ANSI version // HINF SetupOpenInfFileA( IN PCSTR FileName, IN PCSTR InfType, OPTIONAL IN DWORD InfStyle, OUT PUINT ErrorLine OPTIONAL ) { PCTSTR fileName,infType; DWORD err; HINF h; err = NO_ERROR; fileName = NULL; infType = NULL; // // Set error line to 0 since ansi arg conversion could fail // and we need to indicate that there's no error in the inf itself. // if(ErrorLine) { try { *ErrorLine = 0; } except(EXCEPTION_EXECUTE_HANDLER) { err = ERROR_INVALID_PARAMETER; } } if(err == NO_ERROR) { err = pSetupCaptureAndConvertAnsiArg(FileName,&fileName); if((err == NO_ERROR) && InfType) { err = pSetupCaptureAndConvertAnsiArg(InfType,&infType); } } if(err == NO_ERROR) { h = SetupOpenInfFileW(fileName,infType,InfStyle,ErrorLine); err = GetLastError(); } else { h = INVALID_HANDLE_VALUE; } if(fileName) { MyFree(fileName); } if(infType) { MyFree(infType); } SetLastError(err); return(h); } #else // // Unicode stub // HINF SetupOpenInfFileW( IN PCWSTR FileName, IN PCWSTR InfType, OPTIONAL IN DWORD InfStyle, OUT PUINT ErrorLine OPTIONAL ) { UNREFERENCED_PARAMETER(FileName); UNREFERENCED_PARAMETER(InfType); UNREFERENCED_PARAMETER(InfStyle); UNREFERENCED_PARAMETER(ErrorLine); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(INVALID_HANDLE_VALUE); } #endif HINF SetupOpenInfFile( IN PCTSTR FileName, IN PCTSTR InfClass, OPTIONAL IN DWORD InfStyle, OUT PUINT ErrorLine OPTIONAL ) /*++ Routine Description: This routine opens an INF file and returns a handle to it. Arguments: FileName - Supplies the address of a NULL-terminated string containing the name (and optionally, the path) of the INF file to be opened. If the filename contains no path separator characters, it is searched for first in the %windir%\inf directory, and then in the %windir%\system32 directory. Otherwise, the name is assumed to be a full path specification, and no is opened as-is. InfClass - Optionally, supplies the address of a NULL-terminated string containing the class of the INF file desired. For old-style (i.e., Windows NT 3.x script-base) INF files, this string must match the type specified in the OptionType value of the [Identification] section of the INF (e.g., OptionType=NetAdapter). For Windows 95-compatibile INF files, this string must match the class of the specified INF. If the INF has a Class value in its [version] section, then this value is used for the comparison. If no Class value is present, but a ClassGUID value is present in the [version] section, then the corresponding class name for that GUID is retrieved, and comparison is done based on that name. InfStyle - Specifies the style of the INF to open. May be a combination of the following flags: INF_STYLE_OLDNT - Windows NT 3.x script-based INF files. INF_STYLE_WIN4 - Windows 95-compatible INF files. INF_STYLE_CACHE_ENABLE - always cache INF, even outside of %windir%\Inf. INF_STYLE_CACHE_DISABLE - delete INF from cache, if outside of %windir%\Inf. INF_STYLE_CACHE_IGNORE - access INF only, do not touch or look at other files. ErrorLine - If an error occurs loading the file, this parameter receives the (1-based) line number where the error occurred. This value is generally reliable only if GetLastError does not return ERROR_NOT_ENOUGH_MEMORY. If out-of-memory does occur, the ErrorLine may be 0. Return Value: If the function succeeds, the return value is a handle to the opened INF file. If the function fails, the return value is INVALID_HANDLE_VALUE. To get extended error information, call GetLastError. Remarks: If the load fails because the INF class does not match InfClass, the function returns FALSE, and GetLastError returns ERROR_CLASS_MISMATCH. The SetupCloseInfFile function is used to close a handle returned by SetupOpenInfFile. If multiple INF styles are specified, the style of the INF file opened may be ascertained by calling SetupGetInfInformation. Since there may be multiple class GUIDs all having the same class name, callers interested in only INFs of a particular class (i.e., a particular class GUID) should retrieve the ClassGuid value from the INF to verify that the class matches exactly. If both the INF_STYLE_CACHE_ENABLE and INF_STYLE_CACHE_DISABLE InfStyle flags are specified, then the existing cached information that is maintained about the INF (e.g., original source location) is discarded, and the INF is re-added to the cache without this old information. (Internal: Presently, the INF_STYLE_CACHE_ENABLE and INF_STYLE_CACHE_DISABLE flags cause us to create or delete PNFs outside of %windir%\Inf. Their rather vague-sounding names were chosen to reflect the possibility of modifying the caching/indexing scheme in the future.) --*/ { UINT errorLine; DWORD d; PLOADED_INF Inf; PCTSTR Class; TCHAR TempString[MAX_PATH]; GUID ClassGuid; HRESULT hr; BOOL TryPnf = FALSE; BOOL IgnorePnf = FALSE; WIN32_FIND_DATA FindData; PTSTR DontCare; PTSTR TempCharPtr = NULL; // // Determine whether just the filename (no path) was specified. If so, // look for it in the DevicePath directory search path. Otherwise, // use the path as-is. // try { if(FileName == pSetupGetFileTitle(FileName)) { // // The specified INF name is a simple filename. Search for it in // the INF directories using the default search order. // d = SearchForInfFile( FileName, &FindData, INFINFO_DEFAULT_SEARCH, TempString, SIZECHARS(TempString), NULL ); if(d == NO_ERROR) { TryPnf = TRUE; } } else { // // The specified INF filename contains more than just a filename. // Assume it's an absolute path. (We need to make sure it's // fully-qualified, because that's what LoadInfFile expects.) // d = GetFullPathName(FileName, SIZECHARS(TempString), TempString, &DontCare ); if(!d) { d = GetLastError(); } else if(d >= SIZECHARS(TempString)) { MYASSERT(0); d = ERROR_BUFFER_OVERFLOW; } else { // // We successfully retrieved the full pathname, now see if the // file exists. // if(FileExists(TempString, &FindData)) { // // We have everything we need to load this INF. // InfSourcePathFromFileName(TempString, &TempCharPtr, &TryPnf); d = NO_ERROR; } else { d = GetLastError(); } } } } except(EXCEPTION_EXECUTE_HANDLER) { // // Assume FileName was invalid and thus pSetupGetFileTitle fell over. // d = ERROR_INVALID_PARAMETER; TempCharPtr = TempCharPtr; } if(d != NO_ERROR) { goto PrepareForReturn; } if(InfStyle & INF_STYLE_CACHE_DISABLE) { // // Delete the existing PNF for this INF (if any). // TCHAR PnfFullPath[MAX_PATH]; PTSTR PnfFileName, PnfFileExt; lstrcpy(PnfFullPath, TempString); // // Find the start of the filename component of the path, and then find // the last period (if one exists) in that filename. // PnfFileName = (PTSTR)pSetupGetFileTitle(PnfFullPath); if(!(PnfFileExt = _tcsrchr(PnfFileName, TEXT('.')))) { PnfFileExt = PnfFullPath + lstrlen(PnfFullPath); } // // Now create a corresponding filename with the extension '.PNF' // lstrcpyn(PnfFileExt, pszPnfSuffix, SIZECHARS(PnfFullPath) - (int)(PnfFileExt - PnfFullPath)); SetFileAttributes(PnfFullPath, FILE_ATTRIBUTE_NORMAL); DeleteFile(PnfFullPath); } if(InfStyle & INF_STYLE_CACHE_ENABLE) { // // The caller has requested that this INF be cached (even though it may // be outside of our INF search path). // TryPnf = TRUE; } if(InfStyle & INF_STYLE_CACHE_IGNORE) { // // The caller has requested that we don't trust the PNF // overrides TryPnf // TryPnf = FALSE; IgnorePnf = TRUE; } try { d = LoadInfFile( TempString, &FindData, InfStyle, (TryPnf ? LDINF_FLAG_ALWAYS_TRY_PNF : 0) | (IgnorePnf ? LDINF_FLAG_ALWAYS_IGNORE_PNF : 0), NULL, TempCharPtr, NULL, NULL, NULL, // LogContext &Inf, &errorLine, NULL ); } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } if(d == NO_ERROR) { if(InfClass) { d = ERROR_CLASS_MISMATCH; // assume mismatch // // Match based on class of inf. The following check works for // both new and old style infs, because old-style infs use // [Identification].OptionType as the class (see oldinf.c // function ProcessOldInfVersionBlock()). // if(Class = pSetupGetVersionDatum(&(Inf->VersionBlock), pszClass)) { try { if(!lstrcmpi(Class,InfClass)) { d = NO_ERROR; } } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } } else { // // No Class entry--check for ClassGUID entry. // if(Class = pSetupGetVersionDatum(&(Inf->VersionBlock), pszClassGuid)) { // // Get the class name associated with this GUID, and see if it // matches the caller-supplied class name. // // (We have to cast away the CONST-ness of the Class string, because // the prototype for CLSIDFromString doesn't specify this parameter // as constant.) // if((hr = pSetupGuidFromString((PTSTR)Class, &ClassGuid)) == S_OK) { if(SetupDiClassNameFromGuid(&ClassGuid, TempString, SIZECHARS(TempString), NULL)) { try { if(!lstrcmpi(TempString,InfClass)) { d = NO_ERROR; } } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } } } else { if(hr == E_OUTOFMEMORY) { d = ERROR_NOT_ENOUGH_MEMORY; } } } } if(d != NO_ERROR) { FreeInfFile(Inf); } } } else { if(ErrorLine) { try { *ErrorLine = errorLine; } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } } } PrepareForReturn: if(TempCharPtr) { MyFree(TempCharPtr); } SetLastError(d); return((d == NO_ERROR) ? (HINF)Inf : (HINF)INVALID_HANDLE_VALUE); } HINF SetupOpenMasterInf( VOID ) /*++ Routine Description: Arguments: Return Value: --*/ { TCHAR FileName[MAX_PATH]; lstrcpyn(FileName,InfDirectory,SIZECHARS(FileName)-11); lstrcat(FileName,TEXT("\\LAYOUT.INF")); return(SetupOpenInfFile(FileName,NULL,INF_STYLE_WIN4,NULL)); } #ifdef UNICODE // // ANSI version // BOOL SetupOpenAppendInfFileA( IN PCSTR FileName, OPTIONAL IN HINF InfHandle, OUT PUINT ErrorLine OPTIONAL ) { PCWSTR fileName = NULL; DWORD d; BOOL b; // // Set error line to 0 since ansi arg conversion could fail // and we need to indicate that there's no error in the inf itself. // d = NO_ERROR; if(ErrorLine) { try { *ErrorLine = 0; } except(EXCEPTION_EXECUTE_HANDLER) { d = ERROR_INVALID_PARAMETER; } } if(d == NO_ERROR) { if(FileName) { d = pSetupCaptureAndConvertAnsiArg(FileName,&fileName); } else { fileName = NULL; } } if(d == NO_ERROR) { b = SetupOpenAppendInfFileW(fileName,InfHandle,ErrorLine); d = GetLastError(); } else { b = FALSE; } if(fileName) { MyFree(fileName); } SetLastError(d); return(b); } #else // // Unicode stub // BOOL SetupOpenAppendInfFileW( IN PCWSTR FileName, OPTIONAL IN HINF InfHandle, OUT PUINT ErrorLine OPTIONAL ) { UNREFERENCED_PARAMETER(FileName); UNREFERENCED_PARAMETER(InfHandle); UNREFERENCED_PARAMETER(ErrorLine); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif BOOL SetupOpenAppendInfFile( IN PCTSTR FileName, OPTIONAL IN HINF InfHandle, OUT PUINT ErrorLine OPTIONAL ) /*++ Routine Description: Arguments: Return Value: --*/ { PLOADED_INF ExistingInf = NULL, CurInf = NULL; DWORD d = NO_ERROR; TCHAR Filename[2][MAX_PATH]; UINT FilenameCount, i, Field; UINT errorLine = 0; BOOL LookInInfDirAlso; BOOL TryPnf; WIN32_FIND_DATA FindData; PTSTR TempCharPtr = NULL; PTSTR DontCare; PINF_SECTION InfSection; UINT LineNumber; PINF_LINE InfLine = NULL; try { if(LockInf((PLOADED_INF)InfHandle)) { ExistingInf = (PLOADED_INF)InfHandle; } else { d = ERROR_INVALID_HANDLE; goto clean0; } // // Check INF Signature field as a further validation on the InfHandle. // if(ExistingInf->Signature != LOADED_INF_SIG) { d = ERROR_INVALID_HANDLE; goto clean0; } // // Only allow this for win95-style infs. // if(ExistingInf->Style != INF_STYLE_WIN4) { d = ERROR_INVALID_PARAMETER; goto clean0; } // // If there is no filename, search through the list of existing INFs, looking // for a layout entry in their version blocks. We begin at the end of the list, // and search backward, using the first layout file we encounter. This allows // for the possibility of append-loading several INFs together (e.g., as done by the // optional components dialog), and calling SetupOpenAppendInfFile with no filename // for each. Each INF could specify its own layout file, and everything works great. // (NOTE: In the above example, if all the INFs specified the same layout file, it // would only get loaded once, as expected.) // // We also can now handle 'LayoutFile' entries that specify multiple layout files. E.g., // // LayoutFile = pluslay.inf, layout.inf // // In the above example, we would append-load 'pluslay.inf', followed by 'layout.inf'. // Because of the way we store INFs, any duplicate entries in both INFs would resolve in // favor of pluslay.inf, since it was loaded first (unless, of course, layout.inf was // already in our list of loaded INFs). // if(!FileName) { // // First, find the end of the list. // for(CurInf = ExistingInf; CurInf->Next; CurInf = CurInf->Next); // // Now, search the list, back-to-front, looking for a layout file in each INF's // [version] section. // for(; CurInf; CurInf = CurInf->Prev) { // // Locate the [Version] section. // if(InfSection = InfLocateSection(CurInf, pszVersion, NULL)) { // // Now look for a LayoutFile line. // LineNumber = 0; if(InfLocateLine(CurInf, InfSection, pszLayoutFile, &LineNumber, &InfLine)) { // // We've found the line containing the INFs to be append- // loaded. Get the first field on the line to start off // our loop below. // FileName = InfGetField(CurInf, InfLine, 1, NULL); break; } else { // // Make sure InfLine is still NULL, so we won't try to use it. // InfLine = NULL; } } } if(!FileName) { // // Then we didn't find any INFs that specify a layout file. // d = ERROR_INVALID_DATA; goto clean0; } } // // Now append-load the INF (or the possibly multiple INFs, if we're // using a LayoutFile line). // for(Field = 1; FileName; FileName = InfLine ? InfGetField(CurInf, InfLine, ++Field, NULL) : NULL) { FilenameCount = 0; LookInInfDirAlso = TRUE; TryPnf = FALSE; // // Determine whether just the filename (no path) was specified. // if(FileName == pSetupGetFileTitle(FileName)) { // // If we retrieved this filename from an INF's [version] section, // then we first attempt to open up the layout file from the // directory where we found the INF. If we don't find it in that // directory, and that directory wasn't the Inf directory, then // we try to open it up in %windir%\Inf as well. // if(CurInf) { // // Copy the path without the ending backslash character, // because that's how the 'InfDirectory' string is formatted. // lstrcpyn(Filename[0], CurInf->VersionBlock.Filename, (int)(pSetupGetFileTitle(CurInf->VersionBlock.Filename) - CurInf->VersionBlock.Filename) ); // // Compare this path against the InfDirectory path, to see // if they're the same. // if(!lstrcmpi(Filename[0], InfDirectory)) { TryPnf = TRUE; LookInInfDirAlso = FALSE; } // // Now concatenate the layout filename onto the path. // pSetupConcatenatePaths(Filename[0], FileName, MAX_PATH, NULL); FilenameCount = 1; // // If 'TryPnf' is still FALSE, then that means that the INF // wasn't in the INF directory. Now find out if it's in a // location that requires a non-NULL SourcePath (i.e., // something other than the default). // if(!TryPnf) { InfSourcePathFromFileName(Filename[0], &TempCharPtr, &TryPnf); } } if(LookInInfDirAlso) { lstrcpy(Filename[FilenameCount], InfDirectory); pSetupConcatenatePaths(Filename[FilenameCount], FileName, MAX_PATH, NULL); if(!FilenameCount) { TryPnf = TRUE; } FilenameCount++; } } else { // // The INF filename contains more than just a filename. Assume // it's an absolute path. (We need to make sure it's fully- // qualified, because that's what LoadInfFile expects.) // d = GetFullPathName(FileName, SIZECHARS(Filename[0]), Filename[0], &DontCare ); if(!d) { d = GetLastError(); goto clean0; } else if(d >= SIZECHARS(Filename[0])) { MYASSERT(0); d = ERROR_BUFFER_OVERFLOW; goto clean0; } InfSourcePathFromFileName(Filename[0], &TempCharPtr, &TryPnf); FilenameCount = 1; // // (Since we're setting FilenameCount to 1, we know we'll go // through the loop below at least once, thus d will get set to // the proper error, so we don't have to re-initialize it to // NO_ERROR here.) // } for(i = 0; i < FilenameCount; i++) { // // Load the inf // if(FileExists(Filename[i], &FindData)) { if((d = LoadInfFile(Filename[i], &FindData, INF_STYLE_WIN4, (i | TryPnf) ? LDINF_FLAG_ALWAYS_TRY_PNF : 0, NULL, (i | TryPnf) ? NULL : TempCharPtr, NULL, ExistingInf, ExistingInf->LogContext, &ExistingInf, &errorLine, NULL)) == NO_ERROR) { break; } } else { d = GetLastError(); } } // // We no longer need the INF source path--free it if necessary. // if(TempCharPtr) { MyFree(TempCharPtr); TempCharPtr = NULL; } if(d != NO_ERROR) { break; } } clean0: // // If the caller requested it, give them the line number at which any error occurred. // (This may be zero on non-parse errors.) // if(ErrorLine) { *ErrorLine = errorLine; } } 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(TempCharPtr) { MyFree(TempCharPtr); } // // Access the 'ExistingInf' variable, so that the compiler will respect our statement // ordering w.r.t. this variable. Otherwise, we may not always know whether or not // we should be unlocking this INF. // ExistingInf = ExistingInf; } if(ExistingInf) { UnlockInf(ExistingInf); } SetLastError(d); return(d == NO_ERROR); } VOID SetupCloseInfFile( IN HINF InfHandle ) /*++ Routine Description: Arguments: Return Value: --*/ { PLOADED_INF CurInf, NextInf; try { // // Make sure we can lock the head of the INF list before // we start deleting! // if(LockInf((PLOADED_INF)InfHandle)) { // // Also check INF Signature field as a further validation. // if(((PLOADED_INF)InfHandle)->Signature == LOADED_INF_SIG) { CurInf = ((PLOADED_INF)InfHandle)->Next; DestroySynchronizedAccess(&(((PLOADED_INF)InfHandle)->Lock)); FreeLoadedInfDescriptor((PLOADED_INF)InfHandle); for(; CurInf; CurInf = NextInf) { NextInf = CurInf->Next; FreeInfFile(CurInf); } } else { UnlockInf((PLOADED_INF)InfHandle); } } } except(EXCEPTION_EXECUTE_HANDLER) { ; } } // // ANSI version // BOOL WINAPI SetupVerifyInfFileA( IN PCSTR InfName, IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo, OPTIONAL OUT PSP_INF_SIGNER_INFO_A InfSignerInfo ) { DWORD Err = NO_ERROR; SP_INF_SIGNER_INFO_W InfSignerInfoW; int i; PWSTR InfNameUnicode = NULL; try { if (!InfName) { SetLastError(ERROR_INVALID_PARAMETER); leave; } if (!InfSignerInfo || (InfSignerInfo->cbSize != sizeof(SP_INF_SIGNER_INFO_A))) { SetLastError(ERROR_INVALID_PARAMETER); leave; // exit try block } Err = pSetupCaptureAndConvertAnsiArg(InfName, &InfNameUnicode); if (Err != NO_ERROR) { leave; // exit try block } InfSignerInfoW.cbSize = sizeof(InfSignerInfoW); Err = GLE_FN_CALL(FALSE, SetupVerifyInfFile(InfNameUnicode, AltPlatformInfo, &InfSignerInfoW) ); if((Err == NO_ERROR) || (Err == ERROR_AUTHENTICODE_TRUSTED_PUBLISHER) || (Err == ERROR_AUTHENTICODE_TRUST_NOT_ESTABLISHED)) { i = WideCharToMultiByte( CP_ACP, 0, InfSignerInfoW.CatalogFile, -1, InfSignerInfo->CatalogFile, SIZECHARS(InfSignerInfo->CatalogFile), NULL, NULL ); if (i==0) { // // error occurred (LastError set to error) // Err = GetLastError(); leave; // exit try block } i = WideCharToMultiByte( CP_ACP, 0, InfSignerInfoW.DigitalSigner, -1, InfSignerInfo->DigitalSigner, SIZECHARS(InfSignerInfo->DigitalSigner), NULL, NULL ); if (i==0) { // // error occurred (LastError set to error) // Err = GetLastError(); leave; // exit try block } i = WideCharToMultiByte( CP_ACP, 0, InfSignerInfoW.DigitalSignerVersion, -1, InfSignerInfo->DigitalSignerVersion, SIZECHARS(InfSignerInfo->DigitalSignerVersion), NULL, NULL ); if (i==0) { // // error occurred (LastError set to error) // Err = GetLastError(); leave; // exit try block } } } except(EXCEPTION_EXECUTE_HANDLER) { Err = ERROR_INVALID_PARAMETER; } if (InfNameUnicode) { MyFree(InfNameUnicode); } SetLastError(Err); return (Err == NO_ERROR); } BOOL WINAPI SetupVerifyInfFile( IN LPCTSTR InfName, IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo, OPTIONAL OUT PSP_INF_SIGNER_INFO InfSignerInfo ) /*++ Routine Description: This routine verifies the digital signature of the specified INF, using its corresponding catalog. The verification can optionally be performed against a non-native platform. Arguments: InfName - Supplies the name of the INF file to be verified. This name may include a path. AltPlatformInfo - optionally, supplies the address of a structure containing information regarding the alternate platform that is to be used when validating the INF file. InfSignerInfo - Supplies the address of a structure that receives information about the INF's digital signature (if it is signed). Return Value: If the INF was successfully validated using driver signing policy, the return value is TRUE. If the INF was successfully validated using Authenticode policy, and the publisher was in the TrustedPublisher store, the return value is FALSE and GetLastError returns ERROR_AUTHENTICODE_TRUSTED_PUBLISHER. If the INF was successfully validated using Authenticode policy, and the publisher was *not* in the TrustedPublisher store, the return value is FALSE and GetLastError returns ERROR_AUTHENTICODE_TRUST_NOT_ESTABLISHED. If the function fails, the return value is FALSE. To get extended error information, call GetLastError. --*/ { DWORD Err = NO_ERROR; DWORD AuthenticodeError; TCHAR PathBuffer[MAX_PATH]; PLOADED_INF Inf = NULL; BOOL PnfWasUsed; UINT ErrorLineNumber; BOOL TryPnf; WIN32_FIND_DATA FindData; DWORD TempRequiredSize; PTSTR DontCare; HANDLE hWVTStateData; PCRYPT_PROVIDER_DATA ProviderData; PCRYPT_PROVIDER_SGNR ProviderSigner; PCRYPT_PROVIDER_CERT ProviderCert; try { if (!InfName) { Err = ERROR_INVALID_PARAMETER; leave; } if (!InfSignerInfo || (InfSignerInfo->cbSize != sizeof(SP_INF_SIGNER_INFO))) { Err = ERROR_INVALID_PARAMETER; leave; } if(InfName == pSetupGetFileTitle(InfName)) { // // The specified INF name is a simple filename. Search for it in // the DevicePath search path list. // Err = SearchForInfFile(InfName, &FindData, INFINFO_INF_PATH_LIST_SEARCH, PathBuffer, SIZECHARS(PathBuffer), NULL ); if(Err == NO_ERROR) { TryPnf = TRUE; } else { leave; } } else { // // The specified INF filename contains more than just a filename. // Assume it's an absolute path. (We need to make sure it's // fully-qualified, because that's what LoadInfFile expects.) // TempRequiredSize = GetFullPathName(InfName, SIZECHARS(PathBuffer), PathBuffer, &DontCare ); if(!TempRequiredSize) { Err = GetLastError(); leave; } else if(TempRequiredSize >= SIZECHARS(PathBuffer)) { MYASSERT(0); Err = ERROR_BUFFER_OVERFLOW; leave; } if(FileExists(PathBuffer, &FindData)) { // // We have a valid file path, and we're ready to load this INF. // InfSourcePathFromFileName(PathBuffer, NULL, &TryPnf); } else { Err = GetLastError(); leave; } } // // Load the INF. // Err = LoadInfFile(PathBuffer, &FindData, INF_STYLE_WIN4, LDINF_FLAG_IGNORE_VOLATILE_DIRIDS | (TryPnf ? LDINF_FLAG_ALWAYS_TRY_PNF : 0), NULL, NULL, NULL, NULL, NULL, // LogContext &Inf, &ErrorLineNumber, NULL ); if(Err != NO_ERROR) { leave; } InfSignerInfo->CatalogFile[0] = TEXT('\0'); InfSignerInfo->DigitalSigner[0] = TEXT('\0'); InfSignerInfo->DigitalSignerVersion[0] = TEXT('\0'); if(GlobalSetupFlags & PSPGF_MINIMAL_EMBEDDED) { // // We can't call the VerifyDeviceInfFile internal routine, because // it doesn't expect to be asked for the signer info (it doesn't // make any sense to talk about the signer info, because we have // no idea who signed the INF, or even if it was signed). // Err = NO_ERROR; } else { Err = VerifyDeviceInfFile(NULL, NULL, PathBuffer, Inf, AltPlatformInfo, InfSignerInfo->CatalogFile, InfSignerInfo->DigitalSigner, InfSignerInfo->DigitalSignerVersion, 0, NULL ); if((Err != NO_ERROR) && (Err != ERROR_SIGNATURE_OSATTRIBUTE_MISMATCH)) { // // We failed to verify via driver signing policy (and it wasn't // simply because a valid driver signing catalog didn't have an // applicable osattribute). Fall back to validating via // Authenticode policy. NOTE: we don't have to worry about // whether Authenticode verification is acceptable for this // class--the VerifyDeviceInfFile routine will check for that. // // NTRAID#NTBUG9-719853-2002/10/11-LonnyM We may be using Authenticode policy when we shouldn't! // It's possible that we have a valid catalog (per driver // signing policy), and the failure was due to the INF having // been tampered with. In that case, we really shouldn't be // doing Authenticode validation. However, we know that // WinVerifyTrust should fail in either case, and Authenticode // policy actually gives us a better error (TRUST_E_NOSIGNATURE // instead of ERROR_INVALID_PARAMETER). // AuthenticodeError = VerifyDeviceInfFile( NULL, NULL, PathBuffer, Inf, AltPlatformInfo, InfSignerInfo->CatalogFile, NULL, NULL, VERIFY_INF_USE_AUTHENTICODE_CATALOG, &hWVTStateData ); if((AuthenticodeError == ERROR_AUTHENTICODE_TRUSTED_PUBLISHER) || (AuthenticodeError == ERROR_AUTHENTICODE_TRUST_NOT_ESTABLISHED)) { // // Update error to indicate that the INF was Authenticode // signed. // Err = AuthenticodeError; ProviderData = WTHelperProvDataFromStateData(hWVTStateData); MYASSERT(ProviderData); if (ProviderData) { ProviderSigner = WTHelperGetProvSignerFromChain(ProviderData, 0, FALSE, 0); MYASSERT(ProviderSigner); if (ProviderSigner) { ProviderCert = WTHelperGetProvCertFromChain(ProviderSigner, 0); MYASSERT(ProviderCert); if (ProviderCert) { // // Get the publisher and add this // as the DigitalSigner. // CertGetNameString(ProviderCert->pCert, CERT_NAME_SIMPLE_DISPLAY_TYPE, 0, NULL, InfSignerInfo->DigitalSigner, SIZECHARS(InfSignerInfo->DigitalSigner)); } } } if(hWVTStateData) { pSetupCloseWVTStateData(hWVTStateData); } } else if(AuthenticodeError != ERROR_AUTHENTICODE_DISALLOWED) { // // It was acceptable to validate using Authenticode // policy (it just didn't work). Use this error, // instead of the one generated based on driver signing // verification policy. // Err = AuthenticodeError; } } } } except(EXCEPTION_EXECUTE_HANDLER) { Err = ERROR_INVALID_DATA; } if (Inf) { FreeInfFile(Inf); Inf = NULL; } SetLastError(Err); return (Err == NO_ERROR); } ///////////////////////////////////////////////////////////////// // // Internal routines // ///////////////////////////////////////////////////////////////// BOOL pSetupVersionNodeFromInfInformation( IN PSP_INF_INFORMATION InfInformation, IN UINT InfIndex, OUT PINF_VERSION_NODE VersionNode, OUT PTSTR OriginalFilename OPTIONAL ) /*++ Routine Description: Fills in a caller-supplied INF_VERSION_NODE buffer for an INF file from the SP_INF_INFORMATION structure. Arguments: InfInformation - supplies the inf information descriptor InfIndex - supplies the 0-based index of the inf whose version block is requested. If this value is not inrange an error is returned. VersionNode - supplies the address of a buffer that receives the version node structure. OriginalFilename - optionally, supplies the address of a character buffer (that must be at least MAX_PATH characters large) that receives the INF's original filename (which may be the same as its current filename if the INF isn't an OEM INF. Return Value: If successful, the return value is TRUE, otherwise, it is FALSE. --*/ { PINF_VERSION_BLOCK First; INF_VERSION_BLOCK UNALIGNED *Ver; PUCHAR Base; UINT ord; INF_VERSION_BLOCK TempVersionBlock; UINT FilenameSize; // // Get pointer to first version block. // Base = (PUCHAR)InfInformation; First = (PINF_VERSION_BLOCK)(Base+offsetof(SP_INF_INFORMATION,VersionData)); // // Find relevant version block // ord = 0; for(Ver=First; Ver; Ver=(INF_VERSION_BLOCK UNALIGNED *)(Base+Ver->NextOffset)) { if(ord++ == InfIndex) { break; } } if(!Ver) { SetLastError(ERROR_NO_MORE_ITEMS); return FALSE; } // // Now fill in the version node based on the information contained in the version block. // VersionNode->LastWriteTime = Ver->LastWriteTime; VersionNode->DataBlock = (CONST TCHAR *)((PBYTE)(Ver->Filename) + Ver->OffsetToData); VersionNode->DataSize = Ver->DataSize; VersionNode->DatumCount = Ver->DatumCount; // // The 'filename' character buffer may actually contain two strings--the // first being the INF's current filename (with path), and the second being // the INF's original filename (this won't be present if the INF's name // hasn't changed from its original name). // // Copy the first MAX_PATH characters of this buffer (or the entire buffer, // whichever is smaller) into the VersionNode's Filename buffer, then after // we've computed the string length of that string, we can ascertain whether // or not there's another string following it containing the INF's original // name. // FilenameSize = (Ver->OffsetToData < SIZECHARS(VersionNode->Filename)) ? Ver->OffsetToData : SIZECHARS(VersionNode->Filename); CopyMemory(VersionNode->Filename, Ver->Filename, FilenameSize); VersionNode->FilenameSize = (lstrlen(VersionNode->Filename) + 1) * sizeof(TCHAR); MYASSERT(Ver->OffsetToData >= VersionNode->FilenameSize); if(OriginalFilename) { if(Ver->OffsetToData > VersionNode->FilenameSize) { // // Then there's more data in the Filename buffer, namely the INF's // original name--fill this filename into the caller-supplied buffer. // FilenameSize = Ver->OffsetToData - VersionNode->FilenameSize; MYASSERT(((UINT)(FilenameSize / sizeof(TCHAR)) * sizeof(TCHAR)) == FilenameSize); MYASSERT(FilenameSize > sizeof(TCHAR)); CopyMemory(OriginalFilename, (PBYTE)Ver->Filename + VersionNode->FilenameSize, FilenameSize ); MYASSERT(((lstrlen(OriginalFilename) + 1) * sizeof(TCHAR)) == FilenameSize); } else { // // No original name info stored--must be same as current name. // if(FAILED(StringCchCopy(OriginalFilename,MAX_PATH, pSetupGetFileTitle(VersionNode->Filename)))) { SetLastError(ERROR_BUFFER_OVERFLOW); return FALSE; } } } return TRUE; } PCTSTR pSetupGetVersionDatum( IN PINF_VERSION_NODE VersionNode, IN PCTSTR DatumName ) /*++ Routine Description: Look up a piece of version data in an version data node. Arguments: VersionNode - supplies a pointer to the version node to be searched for the datum. DatumName - supplies the name of the datum to be retreived. Return Value: NULL if the datum does not exist in the data block. Otherwise a pointer to the datum value is returned. The caller must not free or write into this memory. --*/ { WORD Datum; UINT StringLength; PCTSTR Data = VersionNode->DataBlock; for(Datum=0; Datum < VersionNode->DatumCount; Datum++) { StringLength = lstrlen(Data) + 1; // // Go through the version block looking for a matching datum name. // if(lstrcmpi(Data, DatumName)) { // // Point to the next one. // Data += StringLength; Data += lstrlen(Data) + 1; } else { // // Found it. Return datum value to caller. // return (Data + StringLength); } } return(NULL); } BOOL pSetupGetCatalogFileValue( IN PINF_VERSION_NODE InfVersionNode, OUT LPTSTR Buffer, IN DWORD BufferSize, IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo OPTIONAL ) /*++ Routine Description: This routine fetches the (potentially decorated) CatalogFile= value from the specified inf version section. Arguments: InfVersionNode - points to the INF version node from which we're attempting to retrieve the associated catalog file. Buffer - if the routine returns TRUE, receives the value for CatalogFile= in the [Version] section of the inf. BufferSize - supplies the size in bytes (ansi) or chars (unicode) of the buffer pointed to by Buffer. 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: Boolean value indicating whether a value was found and copied to the caller-supplied Buffer. --*/ { TCHAR CatFileWithExt[64]; LPCTSTR p, NtPlatformSuffixToUse; DWORD PlatformId; MYASSERT(BufferSize >= MAX_PATH); p = NULL; CopyMemory(CatFileWithExt, pszCatalogFile, sizeof(pszCatalogFile) - sizeof(TCHAR)); // // Set up some variables based on the native platform or upon the non-native // platform specified in the AltPlatformInfo parameter. // if(AltPlatformInfo) { PlatformId = AltPlatformInfo->Platform; switch(AltPlatformInfo->ProcessorArchitecture) { case PROCESSOR_ARCHITECTURE_INTEL: NtPlatformSuffixToUse = pszNtX86Suffix; break; case PROCESSOR_ARCHITECTURE_IA64: NtPlatformSuffixToUse = pszNtIA64Suffix; break; case PROCESSOR_ARCHITECTURE_AMD64: NtPlatformSuffixToUse = pszNtAMD64Suffix; break; default: return FALSE; } } else { PlatformId = OSVersionInfo.dwPlatformId; NtPlatformSuffixToUse = pszNtPlatformSuffix; } if(PlatformId == VER_PLATFORM_WIN32_NT) { // // We're running on NT, so first try the NT architecture-specific // extension, then the generic NT extension. // lstrcpyn((PTSTR)((PBYTE)CatFileWithExt + (sizeof(pszCatalogFile) - sizeof(TCHAR))), NtPlatformSuffixToUse, SIZECHARS(CatFileWithExt) - (sizeof(pszCatalogFile) - sizeof(TCHAR)) ); p = pSetupGetVersionDatum(InfVersionNode, CatFileWithExt); if(!p) { // // We didn't find an NT architecture-specific CatalogFile= entry, so // fall back to looking for just an NT-specific one. // CopyMemory((PBYTE)CatFileWithExt + (sizeof(pszCatalogFile) - sizeof(TCHAR)), pszNtSuffix, sizeof(pszNtSuffix) ); p = pSetupGetVersionDatum(InfVersionNode, CatFileWithExt); } } else { // // We're running on Windows 95, so try the Windows-specific extension // CopyMemory((PBYTE)CatFileWithExt + (sizeof(pszCatalogFile) - sizeof(TCHAR)), pszWinSuffix, sizeof(pszWinSuffix) ); p = pSetupGetVersionDatum(InfVersionNode, CatFileWithExt); } // // If we didn't find an OS/architecture-specific CatalogFile= entry above, // then look for an undecorated entry. // if(!p) { p = pSetupGetVersionDatum(InfVersionNode, pszCatalogFile); } // // If we got back an empty string, then treat this as if there was no // CatalogFile= entry (this might be used, for example, so that a system- // supplied INF that supports both NT and Win98 could specify an undecorated // CatalogFile= entry for Win98, yet supply an NT-specific CatalogFile= // entry that's an empty string, so that we'd do global verification on NT). // if(p && lstrlen(p)) { lstrcpyn(Buffer, p, BufferSize); return TRUE; } else { return FALSE; } } VOID pSetupGetPhysicalInfFilepath( IN PINFCONTEXT LineContext, OUT LPTSTR Buffer, IN DWORD BufferSize ) { lstrcpyn( Buffer, ((PLOADED_INF)LineContext->CurrentInf)->VersionBlock.Filename, BufferSize ); }