|
|
/*++
Copyright (c) 1995-2000 Microsoft Corporation
Module Name:
infsdisk.c
Abstract:
Externally exposed INF routines for source disk descriptor manipulation.
Author:
Ted Miller (tedm) 9-Feb-1995
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
//
// Locations of various fields in the [SourceDisksNames] section
// of an inf
//
#define DISKNAMESECT_DESCRIPTION 1
#define DISKNAMESECT_TAGFILE 2 // cabinet name in win95
#define DISKNAMESECT_OEM 3 // unused, indicates oem disk in win95
#define DISKNAMESECT_PATH 4
#define DISKNAMESECT_FLAGS 5 // indicates extra tags
#define DISKNAMESECT_TAGFILE2 6 // real tagfile if DISKNAMESECT_TAGFILE is really a cabfile
#ifdef UNICODE
//
// ANSI version
//
BOOL SetupGetSourceInfoA( IN HINF InfHandle, IN UINT SourceId, IN UINT InfoDesired, OUT PSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { DWORD rc; BOOL b; PWCHAR buffer; DWORD requiredsize; PCSTR ansi;
buffer = MyMalloc(MAX_INF_STRING_LENGTH); if (buffer) {
b = pSetupGetSourceInfo( InfHandle, NULL, SourceId, NULL, InfoDesired, buffer, MAX_INF_STRING_LENGTH, &requiredsize );
rc = GetLastError();
if(b) {
rc = NO_ERROR;
if(ansi = pSetupUnicodeToAnsi(buffer)) {
requiredsize = lstrlenA(ansi)+1;
if(RequiredSize) { try { *RequiredSize = requiredsize; } except(EXCEPTION_EXECUTE_HANDLER) { rc = ERROR_INVALID_PARAMETER; b = FALSE; } }
if((rc == NO_ERROR) && ReturnBuffer) {
if(!lstrcpynA(ReturnBuffer,ansi,ReturnBufferSize)) { //
// ReturnBuffer invalid
//
rc = ERROR_INVALID_PARAMETER; b = FALSE; } }
MyFree(ansi); } else { rc = ERROR_NOT_ENOUGH_MEMORY; b = FALSE; } }
MyFree(buffer); } else { rc = ERROR_NOT_ENOUGH_MEMORY; b = FALSE; }
SetLastError(rc); return(b); } #else
//
// Unicode stub
//
BOOL SetupGetSourceInfoW( IN HINF InfHandle, IN UINT SourceId, IN UINT InfoDesired, OUT PWSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) { UNREFERENCED_PARAMETER(InfHandle); UNREFERENCED_PARAMETER(SourceId); UNREFERENCED_PARAMETER(InfoDesired); UNREFERENCED_PARAMETER(ReturnBuffer); UNREFERENCED_PARAMETER(ReturnBufferSize); UNREFERENCED_PARAMETER(RequiredSize); SetLastError(ERROR_CALL_NOT_IMPLEMENTED); return(FALSE); } #endif
BOOL SetupGetSourceInfo( IN HINF InfHandle, IN UINT SourceId, IN UINT InfoDesired, OUT PTSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) //
// Native version
//
{ return pSetupGetSourceInfo(InfHandle, NULL, SourceId, NULL, InfoDesired, ReturnBuffer, ReturnBufferSize, RequiredSize ); }
BOOL pSetupGetSourceInfo( IN HINF InfHandle, OPTIONAL IN PINFCONTEXT LayoutLineContext, OPTIONAL IN UINT SourceId, IN PSP_ALTPLATFORM_INFO_V2 AltPlatformInfo, OPTIONAL IN UINT InfoDesired, OUT PTSTR ReturnBuffer, OPTIONAL IN DWORD ReturnBufferSize, OUT PDWORD RequiredSize OPTIONAL ) /*++
Routine Description:
Get information from SourceDisksNames
If InfHandle specified instead of LayoutLineContext and the ID is specified in more than one INF then the wrong information *MAY* be returned. This effects callers of SetupGetSourceInfo we need a SetupGetSourceInfoEx post 5.0
Arguments:
InfHandle - required if LayoutLineContext is not provided, else specifies a layout inf
SourceId - numerical source ID, used as search key in SourceDisksNames section
AltPlatformInfo - optionally, supplies alternate platform information used in decorating the [SourceDisksNames] section.
InfoDesired - SRCINFO_PATH SRCINFO_TAGFILE SRCINFO_DESCRIPTION SRCINFO_FLAGS
ReturnBuffer - buffer for returned string ReturnBufferSize - size of buffer RequiredSize - size buffer needs to be if ReturnBufferSize too small LayoutLineContext - if specified, used to determine correct INF to use if SourceID's conflict
Return Value:
Boolean value indicating outcome. If FALSE, GetLastError() returns extended error information. ReturnBuffer filled out with string RequiredSize filled out with required size of buffer to hold string
--*/ { PCTSTR PlatformName; UINT ValueIndex; BOOL Mandatory; BOOL IsPath; INFCONTEXT InfContext; INFCONTEXT SelectedInfContext; int SelectedRank; TCHAR SourceIdString[24]; PCTSTR Value; BOOL b; UINT Length; TCHAR MediaListSectionName[64]; HINF hInfPreferred = (HINF)(-1);
try { if ((LayoutLineContext != NULL) && (LayoutLineContext != (PINFCONTEXT)(-1))) { hInfPreferred = (HINF)LayoutLineContext->CurrentInf; } } except(EXCEPTION_EXECUTE_HANDLER) { hInfPreferred = (HINF)(-1); }
//
// Determine the index of the value that gives the caller the info he wants.
//
switch(InfoDesired) {
case SRCINFO_PATH: ValueIndex = DISKNAMESECT_PATH; Mandatory = FALSE; IsPath = TRUE; break;
case SRCINFO_TAGFILE: ValueIndex = DISKNAMESECT_TAGFILE; Mandatory = FALSE; IsPath = TRUE; break;
case SRCINFO_DESCRIPTION: ValueIndex = DISKNAMESECT_DESCRIPTION; Mandatory = TRUE; IsPath = FALSE; break;
case SRCINFO_FLAGS: ValueIndex = DISKNAMESECT_FLAGS; Mandatory = FALSE; IsPath = FALSE; break;
case SRCINFO_TAGFILE2: ValueIndex = DISKNAMESECT_TAGFILE2; Mandatory = FALSE; IsPath = TRUE; break;
default: SetLastError(ERROR_INVALID_PARAMETER); return(FALSE); }
wsprintf(SourceIdString,TEXT("%d"),SourceId);
if(AltPlatformInfo) {
switch(AltPlatformInfo->ProcessorArchitecture) {
case PROCESSOR_ARCHITECTURE_INTEL : PlatformName = pszX86SrcDiskSuffix; break;
case PROCESSOR_ARCHITECTURE_IA64 : PlatformName = pszIa64SrcDiskSuffix; break;
case PROCESSOR_ARCHITECTURE_AMD64 : PlatformName = pszAmd64SrcDiskSuffix; break;
default : //
// unknown/unsupported processor architecture.
//
MYASSERT((AltPlatformInfo->ProcessorArchitecture == PROCESSOR_ARCHITECTURE_INTEL) || (AltPlatformInfo->ProcessorArchitecture == PROCESSOR_ARCHITECTURE_IA64) || (AltPlatformInfo->ProcessorArchitecture == PROCESSOR_ARCHITECTURE_AMD64) );
SetLastError(ERROR_INVALID_PARAMETER); return(FALSE); }
} else { PlatformName = pszPlatformSrcDiskSuffix; }
wnsprintf( MediaListSectionName, sizeof(MediaListSectionName)/sizeof(MediaListSectionName[0]), TEXT("%s.%s"), pszSourceDisksNames, PlatformName );
//
// we will prefer
// (1) an entry in hInfPreferred (Rank 11/12 decorated over undecorated)
// (2) an entry linked to hInfPreferred (Rank 21/22 decorated over undecorated)
// (3) an entry in hInfHandle (Rank 31/32 decorated over undecorated)
// (4) an entry linked to InfHandle (Rank 41/42 decorated over undecorated)
//
SelectedRank = 100; // 11-42 as above
if ((hInfPreferred != NULL) && (hInfPreferred != (HINF)(-1))) { //
// see if we can find the SourceIdString in the INF that we found the section in
//
// rank 11 or 21 (decorated) - always try
//
if(SetupFindFirstLine(hInfPreferred,MediaListSectionName,SourceIdString,&InfContext)) { if (InfContext.Inf == InfContext.CurrentInf) { SelectedRank = 11; SelectedInfContext = InfContext; } else { SelectedRank = 21; SelectedInfContext = InfContext; } } if (SelectedRank > 12) { //
// rank 12 or 22 (undecorated) only try if we haven't got anything better than 12
//
if(SetupFindFirstLine(hInfPreferred,pszSourceDisksNames,SourceIdString,&InfContext)) { if (InfContext.Inf == InfContext.CurrentInf) { SelectedRank = 12; SelectedInfContext = InfContext; } else if (SelectedRank > 22) { SelectedRank = 22; SelectedInfContext = InfContext; } } } } if ((InfHandle != NULL) && (InfHandle != (HINF)(-1)) && (SelectedRank > 31)) { //
// see if we can find the SourceIdString in the supplied INF
//
// rank 31 or 41 (decorated) - only try if we haven't got anything better than 31
//
if(SetupFindFirstLine(InfHandle,MediaListSectionName,SourceIdString,&InfContext)) { if (InfContext.Inf == InfContext.CurrentInf) { SelectedRank = 31; SelectedInfContext = InfContext; } else if (SelectedRank > 41) { SelectedRank = 41; SelectedInfContext = InfContext; } } if (SelectedRank > 32) { //
// rank 32 or 42 (undecorated) - only try if we haven't got anything better than 32
//
if(SetupFindFirstLine(InfHandle,pszSourceDisksNames,SourceIdString,&InfContext)) { if (InfContext.Inf == InfContext.CurrentInf) { SelectedRank = 32; SelectedInfContext = InfContext; } else if (SelectedRank > 42) { SelectedRank = 42; SelectedInfContext = InfContext; } } } } if(SelectedRank == 100 || (Value = pSetupGetField(&InfContext,ValueIndex))==NULL) { if(Mandatory) { SetLastError(ERROR_LINE_NOT_FOUND); return(FALSE); } else { Value = TEXT(""); } }
//
// Figure out how many characters are in the output.
// If the value is a path type value we want to remove
// the trailing backslash if there is one.
//
Length = lstrlen(Value); if(IsPath && Length && (*CharPrev(Value,Value+Length) == TEXT('\\'))) { Length--; }
//
// Need to leave space for the trailing nul.
//
Length++; if(RequiredSize) { b = TRUE; try { *RequiredSize = Length; } except(EXCEPTION_EXECUTE_HANDLER) { b = FALSE; } if(!b) { SetLastError(ERROR_INVALID_PARAMETER); return(FALSE); } }
b = TRUE; if(ReturnBuffer) { if(Length <= ReturnBufferSize) { //
// lstrcpyn is a strange API but the below is correct --
// the size parameter is actually the capacity of the
// target buffer. So to get it to put the nul in the
// right place we pass one larger than the number of chars
// we want copied.
//
if(!lstrcpyn(ReturnBuffer,Value,Length)) { //
// ReturnBuffer invalid
//
b = FALSE; SetLastError(ERROR_INVALID_PARAMETER); } } else { b = FALSE; SetLastError(ERROR_INSUFFICIENT_BUFFER); } }
return(b); }
|