2020-09-30 16:53:49 +02:00

1250 lines
35 KiB
C++

#include <windows.h>
#include <stdio.h>
#include <tchar.h>
#include <winver.h>
#include "network.h"
#include "idw_dbg.h"
#include "machines.h"
#include "files.h"
//#include <setupapi.h>
/*++
Filename : Files.c
Description: Contains the network access code.
Created by: Wally Ho
History: Created on 20/02/2000.
09.19.2001 Joe Holman fixes for idwlog bugs 409338, 399178, and 352810
Contains these functions:
1. BOOL GetCurrentSystemBuildInfo ( IN OUT LPINSTALL_DATA pId);
2. BOOL GetCurrentInstallingBuildInfo ( IN OUT LPINSTALL_DATA pId);
3. BOOL GetImageHlpDllInfo ( OUT LPINSTALL_DATA pId,
IN INT iFlag);
4. BOOL MyGetFileVersionInfo(LPTSTR lpszFilename,
LPVOID *lpVersionInfo);
5. BOOL GetSkuFromDosNetInf (IN OUT LPINSTALL_DATA pId);
03.29.2001 Joe Holman Made the code work on Win9x.
--*/
DWORD gdwLength; // global used for Win9x data mining, length of buffer.
BOOL
GetCurrentSystemBuildInfo ( IN OUT LPINSTALL_DATA pId)
/*++
Routine Description:
This loads the three items of
DWORD dwSystemBuild;
DWORD dwSystemBuildDelta;
TCHAR szSystemBuildSourceLocation[100]
From information gleaned from the imagehlp.dll.
*NOTE*
The ImageHlp.dll according to Micheal Lekas is compiled everytime a
build is put out. We are banking on this for this function.
For XP SPs, this will be from the ntoskrnl.exe file.
Arguments:
An Install Data struct for loading the data.
Return Value:
TRUE if success
FALSE if fail
--*/
{
//Initialize all the variables first to expected values.
pId->dwSystemBuild = 0;
pId->dwSystemBuildDelta = 0;
pId->dwSystemSPBuild = 0;
pId->dwSystemMajorVersion = 0;
pId->dwSystemMinorVersion = 0;
_stprintf(pId->szSystemBuildSourceLocation, TEXT("%s"),
TEXT("No_Build_Lab_Information"));
if (DTC == TRUE){
if (FALSE == GetBuildInfoFromOSAndBldFile(pId, F_SYSTEM )){
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile failed. Could not retrieved build, delta, location.\n"));
return FALSE;
}else{
Idwlog(TEXT("Success in retrieving build, delta, location.\n"));
return TRUE;
}
// This is Whistler.
}else{
if (FALSE == GetImageHlpDllInfo(pId, F_SYSTEM_IMAGEHLP_DLL)){
Idwlog(TEXT("GetImageHlpDllInfo failed. Could not retrieved build, delta, location.\n"));
return FALSE;
}else{
Idwlog(TEXT("Success in retrieving build, delta, location.\n"));
return TRUE;
}
}
}
BOOL
GetCurrentInstallingBuildInfo ( IN OUT LPINSTALL_DATA pId)
/*++
Routine Description:
This loads the three items of
DWORD dwInstallingBuild;
DWORD dwInstallingBuildDelta;
TCHAR szCurrentBuildSourceLocation[100];
From information gleaned from the imagehlp.dll.
*NOTE*
The ImageHlp.dll according to Micheal Lekas is compiled everytime a
build is put out. We are banking on this for this function.
Arguments:
An Install Data struct for loading the data.
Return Value:
TRUE if success
FALSE if fail
--*/
{
//Initialize all the variables first to expected values.
pId->dwInstallingBuild = 0;
pId->dwInstallingBuildDelta = 0;
pId->dwInstallingSPBuild = 0;
pId->dwInstallingMajorVersion = 0;
pId->dwInstallingMinorVersion = 0;
_stprintf(pId->szInstallingBuildSourceLocation, TEXT("%s"),
TEXT("No_Build_Lab_Information"));
if (DTC == 1){
if (FALSE == GetBuildInfoFromOSAndBldFile(pId, F_INSTALLING )){
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile failed. Could not retrieved build, delta, location.\n"));
return FALSE;
}else{
Idwlog(TEXT("Success in retrieving build, delta, location.\n"));
return TRUE;
}
}else{
if (FALSE == GetImageHlpDllInfo(pId, F_INSTALLING_IMAGEHLP_DLL )){
Idwlog(TEXT("GetImageHlpDllInfo failed. Could not retrieved build, delta, location.\n"));
return FALSE;
}else{
Idwlog(TEXT("Success in retrieving build, delta, location.\n"));
return TRUE;
}
}
}
BOOL
GetBuildInfoFromOSAndBldFile( OUT LPINSTALL_DATA pId,
IN INT iFlag)
/*++
Routine Description:
This loads the three items of
DWORD dwInstallingBuild;
DWORD dwInstallingBuildDelta;
TCHAR szCurrentBuildSourceLocation[100];
From information gleaned from the OS and the BLD file.
*NOTE*
This is assuming the changes of having the bld files on
the CD in the same dir as the idwlog.exe and in the x86 share in on the server.
Arguments:
These flags are.
F_SYSTEM 0x1
F_INSTALLING 0x2
Return Value:
TRUE if success
FALSE if fail
--*/
{
WIN32_FIND_DATA fd;
HANDLE hFind;
TCHAR szFullPath [ MAX_PATH ];
TCHAR szCurDir [ MAX_PATH ];
LPTSTR ptstr;
OSVERSIONINFO osex;
DWORD dwLength = MAX_PATH;
DWORD dwBuild;
DWORD dwBuildDelta;
TCHAR szDontCare[30];
DWORD dwDontCare;
szCurDir[0] = TEXT('\0');
switch (iFlag){
case F_INSTALLING:
// Get the installing files location of the 2195.0XX.bld file
GetModuleFileName( NULL,szCurDir, dwLength);
// Remove the Idwlog.exe part and get only the directory structure.
ptstr = _tcsrchr(szCurDir,TEXT('\\'));
if (NULL == ptstr) {
Idwlog(TEXT("ERROR GetBuildInfoFromOSAndBldFile could find the file to get bld info from.\n"));
return FALSE;
}
*ptstr = TEXT('\0');
Idwlog(TEXT("Getting Build, Delta, Build lab for the installing files.\n"));
Idwlog(TEXT("Getting Installing file location as %s.\n"), szCurDir);
// Use this to get the location
// idwlog.exe -1 is run from. This tool
// will always assume the 2195.xxx.bld is in its
// current path or two up.
_stprintf (szFullPath, TEXT("%s\\*.bld"),szCurDir);
Idwlog(TEXT("First look for the XXXX.xxx.bld in [CD location] %s.\n"),
szFullPath);
// On a network share the 2195.xxx.bld is two up from where
// idwlog.exe -1 is located.
// On a CD its located in the same directory as where the
// idwlog.exe -1 is located. We will look in both places.
hFind = FindFirstFile (szFullPath, &fd);
if (INVALID_HANDLE_VALUE == hFind){
//
// Now we know the file in not in the
// immediate directory. Move up another 2 levels by
// culling off two more directory.
//
for (INT i = 0; i < 2; i++) {
ptstr = _tcsrchr(szCurDir,TEXT('\\'));
if (NULL == ptstr) {
Idwlog(TEXT("ERROR GetBuildInfoFromOSAndBldFile could not find the file to get bld info from.\n"));
return FALSE;
}
*ptstr = TEXT('\0');
}
_stprintf (szFullPath, TEXT("%s\\*.bld"),szCurDir);
Idwlog(TEXT("Second look for the XXXX.xxx.bld in [NET location] %s.\n"),
szFullPath);
hFind = FindFirstFile (szFullPath,&fd);
if (INVALID_HANDLE_VALUE == hFind){
// In case we cannot find it we will exit.
// Set the currentBuild number to 0;
//_tcscpy (id.szCurrentBuild, TEXT("latest"));
Idwlog(TEXT("Could not find the XXXX.xxx.bld file in %s.\n"),
szFullPath);
Idwlog(TEXT("ERROR - Cannot get build number or delta of the installing build.\n"));
return FALSE;
}
}
_stscanf(fd.cFileName,TEXT("%lu.%lu.%s"),&dwBuild, &dwBuildDelta,szDontCare);
Idwlog(TEXT("Inserting into struct Build, Delta, Build lab for the installing files.\n"));
// Build Location
_tcscpy( pId->szInstallingBuildSourceLocation,TEXT("No_Build_Lab_Information"));
// Major version X.51
pId->dwInstallingMajorVersion = 0;
// Minor version 5.XX
pId->dwInstallingMinorVersion = 0;
// Build
pId->dwInstallingBuild = dwBuild;
// Build Delta.
pId->dwInstallingBuildDelta = dwBuildDelta;
pId->dwInstallingSPBuild = dwBuildDelta;
if ( pId->dwInstallingMajorVersion == 0 || pId->dwInstallingMinorVersion == 0 ) {
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile F_INSTALLING ERROR - need Major or Minor #\n"));
}
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile F_INSTALLING - Build = %ld, Major = %ld, Minor = %ld.\n"),
pId->dwInstallingBuild,
pId->dwInstallingMajorVersion,
pId->dwInstallingMinorVersion
);
break;
case F_SYSTEM:
osex.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
GetVersionEx(&osex);
// The output is like "RC 0.31"
_stscanf(osex.szCSDVersion, TEXT("%s %lu.%lu"),
szDontCare, &dwDontCare, &dwBuildDelta);
// Get the local systems location of the imagehlp.dll
Idwlog(TEXT("Inserting into struct Build, Delta, Build lab for the local system.\n"));
// Build Location
_tcscpy( pId->szSystemBuildSourceLocation,TEXT("No_Build_Lab_Information"));
// Major version X.51
pId->dwSystemMajorVersion = 0;
// Minor version 5.XX
pId->dwSystemMinorVersion = 0;
// Build
pId->dwSystemBuild = osex.dwBuildNumber;
// Build Delta.
pId->dwSystemBuildDelta = dwBuildDelta;
pId->dwSystemSPBuild = dwBuildDelta;
Idwlog(TEXT("Getting Build, Delta, Build lab for local system.\n"));
if ( pId->dwInstallingMajorVersion == 0 || pId->dwInstallingMinorVersion == 0 ) {
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile F_SYSTEM ERROR - need Major or Minor #\n"));
}
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile F_SYSTEM - Build = %ld, Major = %ld, Minor = %ld.\n"),
pId->dwInstallingBuild,
pId->dwInstallingMajorVersion,
pId->dwInstallingMinorVersion
);
break;
}
return TRUE;
}
BOOL
GetImageHlpDllInfo ( OUT LPINSTALL_DATA pId,
IN INT iFlag)
/*++
Routine Description:
This loads the three items of
DWORD dwInstallingBuild;
DWORD dwInstallingBuildDelta;
TCHAR szCurrentBuildSourceLocation[100];
From information gleaned from the imagehlp.dll.
*NOTE*
The ImageHlp.dll according to Micheal Lekas is compiled everytime a
build is put out. We are banking on this for this function.
Arguments:
An Install Data struct and the Imagehelp DLL location.
It so it can be reused for both local probing and
Installation probing.
iFlag for System Probe or Installing Probe.
Also if we want the data from imagehlp.dll or from the
System. These flags are.
F_SYSTEM_IMAGEHLP_DLL 0x1
F_INSTALLING_IMAGEHLP_DLL 0x2
Return Value:
TRUE if success
FALSE if fail
--*/
{
VS_FIXEDFILEINFO* pv;
WIN32_FIND_DATA fd;
HANDLE hFind;
TCHAR szFullPath [ MAX_PATH ] = "\0";
TCHAR szCurDir [ MAX_PATH ] = "\0";
TCHAR szBuildLocation[ MAX_PATH ] = "\0";
TCHAR szTempDir [ MAX_PATH ] = "\0";
TCHAR szTempDirFile [ MAX_PATH ] = "\0";
DWORD dwError = 0;
LPTSTR ptstr;
DWORD dwTemp;
BOOL b;
LPVOID lpData = NULL;
LPVOID lpInfo = NULL;
TCHAR key[80];
DWORD *pdwTranslation;
UINT cch, uLen;
DWORD dwDefLang = 0x40904b0;
DWORD dwLength = MAX_PATH;
CHAR *p = NULL;
DWORD dwHandle;
Idwlog ( TEXT ( "GetImageHlpDllInfo - entered.\n") );
szCurDir[0] = TEXT('\0');
// Configure the path to the where the image file lives.
// This is different for the currently installed system vs. the installing build during -1.
// However, for -3, it's a no-op, since we are installed and logged in.
//
// If we are in phase 3, we switch where the path points to.
//
if ( g_InstallData.bSPUninst || g_InstallData.bSPPatch || g_InstallData.bSPFull || g_InstallData.bSPUpdate ) {
// New code to work with SPs.
//
if (iFlag == F_INSTALLING_IMAGEHLP_DLL && (g_InstallData.iStageSequence == 3 || g_InstallData.iStageSequence == 2) ) {
iFlag = F_SYSTEM_IMAGEHLP_DLL;
}
switch (iFlag){
case F_INSTALLING_IMAGEHLP_DLL:
// Get the image data for the build that we are installing.
p = _tcsstr (GetCommandLine(), TEXT("Path="));
if ( p == NULL ) {
Idwlog ( TEXT ( "GetImageHlpDllInfo ERROR - Could not determine path from command line, you need a Path=<path to the installing build> with trailing slash.\n") );
return(FALSE);
}
p += _tcsclen ( TEXT ("Path=" ) );
// We now have the location of the file to expand.
//
//_stprintf (szCurDir, TEXT("%s\\system32\\idwlog.exe"),p);
_stprintf (szFullPath, TEXT("%s\\imagehlp.dll"),p);
/*****
// Get a temporary location to expand the file.
// Can use this if the file is compressed from the media. This will NOT work on Win9x though (setupapi).
//
//
dwError = GetTempPath ( MAX_PATH, szTempDir );
if ( dwError == 0 ) {
Idwlog(TEXT("GetImageHlpDllInfo - ERROR GetTempPath gle = %ld.\n"), GetLastError () );
}
// Uncompress it, and point to that location.
//
_stprintf ( szTempDirFile, TEXT("%s\\ntoskrnl.exe"), szTempDir);
dwError = SetupDecompressOrCopyFile ( szCurDir, szTempDirFile, NULL );
if ( dwError != ERROR_SUCCESS ) {
Idwlog(TEXT("GetImageHlpDllInfo - ERROR SetupDecompressOrCopyFile gle = %ld.\n"), GetLastError () );
}
// Pass the location of the file.
//
_stprintf (szFullPath, TEXT("%s\\ntoskrnl.exe"),szTempDir );
****/
Idwlog(TEXT("GetImageHlpDllInfo - szFullPath(1) = %s.\n"), szFullPath );
break;
case F_SYSTEM_IMAGEHLP_DLL:
// Get the local systems location for the image file
GetSystemDirectory( szCurDir, MAX_PATH );
_stprintf (szFullPath, TEXT("%s\\imagehlp.dll"),szCurDir);
Idwlog(TEXT("GetImageHlpDllInfo - szFullPath(2) = %s.\n"), szFullPath );
break;
}
// end of new code.
//
}
else {
// This is the old standard code.
//
if (iFlag == F_INSTALLING_IMAGEHLP_DLL && (g_InstallData.iStageSequence == 3 || g_InstallData.iStageSequence == 2)) {
iFlag = F_SYSTEM_IMAGEHLP_DLL;
}
switch (iFlag){
case F_INSTALLING_IMAGEHLP_DLL:
// Get the image data for the build that we are installing.
p = _tcsstr (GetCommandLine(), TEXT("Path="));
if ( p == NULL ) {
Idwlog ( TEXT ( "GetImageHlpDllInfo ERROR - Could not determine path from command line, you need a Path=<path to the installing build> with trailing slash.\n") );
return(FALSE);
}
p += _tcsclen ( TEXT ("Path=" ) );
_stprintf (szCurDir, TEXT("%s"),p);
Idwlog(TEXT("GetImageHlpDllInfo - Getting Installing file location as %s.\n"), szCurDir);
break;
case F_SYSTEM_IMAGEHLP_DLL:
// Get the local systems location for the image file
GetSystemDirectory( szCurDir, MAX_PATH );
Idwlog(TEXT("GetImageHlpDllInfo - Getting Build, Delta, Build lab for local system.\n"));
break;
}
_stprintf (szFullPath, TEXT("%s\\imagehlp.dll"),szCurDir);
Idwlog(TEXT("GetImageHlpDllInfo - szFullPath(3) = %s.\n"), szFullPath );
}
Idwlog(TEXT("GetImageHlpDllInfo - Will look in [%s] for build information.\n"),szFullPath);
/*******
// On a network share the Imagehlp.dll is one up from where
// idwlog.exe -1 is located.
// On a CD its located in the same directory as where the
// idwlog.exe -1 is located. We will look in both places.
hFind = FindFirstFile (szFullPath, &fd);
if (INVALID_HANDLE_VALUE == hFind){
//
// Now we know the file in not in the
// immediate directory. Move up one by
// culling off one more directory.
ptstr = _tcsrchr(szCurDir,TEXT('\\'));
if (NULL == ptstr) {
Idwlog(TEXT("GetBuildInfoFromOSAndBldFile could find the file to get bld info from.\n"));
return FALSE;
}
*ptstr = TEXT('\0');
_stprintf (szFullPath, TEXT("%s\\imagehlp.dll"),szCurDir);
Idwlog(TEXT("First look for imagehlp.dll in %s.\n"),szFullPath);
hFind = FindFirstFile (szFullPath,&fd);
if (INVALID_HANDLE_VALUE == hFind){
// In case we cannot find it we will exit.
// Set the currentBuild number to 0;
//_tcscpy (id.szCurrentBuild, TEXT("latest"));
Idwlog(TEXT("Could not find the imagehlp.dll file in %s.\n"),szFullPath);
Idwlog(TEXT("Cannot get build number of the installing build.\n"));
return FALSE;
}
}
************/
Idwlog ( TEXT ( ">>>>> MyGetFileVersionInfo path = %s\n"), szFullPath );
if (FALSE == MyGetFileVersionInfo(szFullPath,&lpData)){
if(lpData) {
free(lpData);
}
Idwlog(TEXT("ERROR - GetFileVersionInfo failed to retrieve Version Info from: %s.\n"), szFullPath );
return FALSE;
}
else {
Idwlog ( TEXT ( "MyGetFileVersionInfo call OK.\n") );
}
/*****
Idwlog ( TEXT ( ">>>>> GetFileVersionInfoSize path = %s\n"), szFullPath );
uLen = GetFileVersionInfoSize ( szFullPath, &dwHandle );
if ( uLen == 0 ) {
Idwlog ( TEXT ( "GetFileVersionInfoSize ERROR, gle = %ld\n"), GetLastError() );
return FALSE;
}
else {
Idwlog ( TEXT ( "GetFileVersionInfoSize call OK returned uLen = %ld.\n"), uLen );
}
b = GetFileVersionInfo ( szFullPath,
dwHandle,
uLen,
&lpData );
if ( b == 0 ) {
Idwlog ( TEXT ( "GetFileVersionInfo ERROR, gle = %ld\n"), GetLastError() );
return FALSE;
}
else {
Idwlog ( TEXT ( "GetFileVersionInfo call OK.\n") );
}
*****/
b = VerQueryValue(lpData,
TEXT("\\VarFileInfo\\Translation"),
(LPVOID*)&pdwTranslation,
&uLen);
if( b == 0 ){
Idwlog ( TEXT("VerQueryValue ERROR (name does not exist or the specified resource is not valid) \\VarFileInfo\\Translation, assuming default language as: %08lx\n"), dwDefLang);
pdwTranslation = &dwDefLang;
}
else {
Idwlog ( TEXT("VerQueryValue call OK. The specified version-information structure exists, and version information is available.\n" ));
if ( uLen == 0 ) {
Idwlog ( TEXT("However, no value is available...\n") );
}
else {
Idwlog ( TEXT("A value is available, it's size is: %ld.\n"), uLen );
}
}
_stprintf(key, TEXT("\\StringFileInfo\\%04x%04x\\%s"),
LOWORD(*pdwTranslation),
HIWORD(*pdwTranslation),
TEXT("FileVersion"));
Idwlog ( TEXT ( "Just before 2nd VerQueryValue -- key = >>>%s<<< >>%x<<\n"), key, *pdwTranslation );
b = VerQueryValue(lpData, key, &lpInfo, &cch);
if( b ){
Idwlog(TEXT("Retrieved string: >>>%s<<<\n"), lpInfo );
// The output of this should be something like
// 5.0.2195. 30 (LAB01_N.000215-2216)
// we want the Lab part.
//
if (NULL == _tcsstr((LPTSTR)lpInfo,TEXT("(") )){
_stprintf(szBuildLocation,
TEXT("%s"),
TEXT("No_Build_Lab_Information"));
}else{
ptstr = _tcsstr((LPTSTR)lpInfo,TEXT("(") );
ptstr++;
_stprintf(szBuildLocation,TEXT("%s"),ptstr);
// remove the trailing ")"
//
szBuildLocation[_tcslen(szBuildLocation) -1 ] = 0;
}
}else{
DWORD i;
TCHAR * p = (TCHAR*)lpData;
TCHAR szNewString[2048];
DWORD dwAmount;
DWORD dwR;
Idwlog ( TEXT("VerQueryValue ERROR (name does not exist or the specified resource is not valid), gle = %ld\n"), GetLastError());
_stprintf(szBuildLocation,
TEXT("%s"),
TEXT("No_Build_Lab_Information"));
Idwlog(TEXT("Warning - Failed to get the build location in VerQueryValue.\n"));
Idwlog(TEXT("Will manually try to find the data...\n"));
// Let's try to find the data ourselves. We *must* do this on Win9x since Win9x doesn't know what
// to do with Unicode.
//
// We will be careful to stay away from end of the buffer. This should be cleaned up where the 26 values are
// with strlen.
//
for ( i = 0; i < (gdwLength - 26); i++, p++ ) {
//Idwlog ( TEXT("i = %ld, p = %s\n"), i, p );
//Idwlog ( TEXT ( "i = %ld, %x %c" ), i, *p, *p );
if ( *p == 'F' &&
*(p+2) == 'i' &&
*(p+4) == 'l' &&
*(p+6) == 'e' &&
*(p+8) == 'V' &&
*(p+10) == 'e' &&
*(p+12) == 'r' &&
*(p+14) == 's'
) {
Idwlog(TEXT(">>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>We found FileVersion data by2...\n"));
p += 26; // to get past word "fileversion"
break;
}
/***
if ( _tcsstr ( p, TEXT("FileVersion")) ){
Idwlog(TEXT("We found FileVersion data...\n"));
break;
}
//Idwlog ( TEXT ( "i = %ld, %x %c" ), i, *p, *p );
***/
}
Idwlog ( TEXT ( "p = %s\n"), p );
Idwlog ( TEXT ( "%x %c" ), i, *p, *p );
Idwlog ( TEXT ( "%x %c" ), i, *(p+1), *(p+1) );
Idwlog ( TEXT ( "%x %c" ), i, *(p+2), *(p+2) );
dwR = WideCharToMultiByte ( CP_ACP,
WC_NO_BEST_FIT_CHARS,
(const unsigned short *)p,
-1,
szNewString,
2048,
NULL,
NULL );
if ( !dwR ) {
Idwlog ( TEXT("ERROR WideChartoMultiByte: dwR = %ld, gle = %ld\n"), dwR, GetLastError () );
}
else {
Idwlog ( TEXT("amount converted dwR = %ld, %s\n"), dwR, p = szNewString );
}
Idwlog(TEXT("Will print out data manually: %s\n"), p );
// The output of this should be something like
// 5.0.2195. 30 (LAB01_N.000215-2216)
// we want the Lab part.
//
if (NULL == _tcsstr((LPTSTR)p,TEXT("(") )){
_stprintf(szBuildLocation,
TEXT("%s"),
TEXT("No_Build_Lab_Information"));
Idwlog ( TEXT ( "Build lab ERROR - we could not find the build # and lab, even manually.\n") );
}else{
ptstr = _tcsstr((LPTSTR)p,TEXT("(") );
ptstr++;
_stprintf(szBuildLocation,TEXT("%s"),ptstr);
// remove the trailing ")"
//
szBuildLocation[_tcslen(szBuildLocation) -1 ] = 0;
Idwlog ( TEXT ( "Build lab found: %s.\n"), szBuildLocation );
}
}
/*********
if(VerQueryValue(lpData, key, &lpInfo, &cch)){
// The output of this should be something like
// 5.0.2195. 30 (LAB01_N.000215-2216)
// we want the Lab part.
//
if (NULL == _tcsstr((LPTSTR)lpInfo,TEXT("(") )){
_stprintf(szBuildLocation,
TEXT("%s"),
TEXT("No_Build_Lab_Information"));
}else{
ptstr = _tcsstr((LPTSTR)lpInfo,TEXT("(") );
ptstr++;
_stprintf(szBuildLocation,TEXT("%s"),ptstr);
// remove the trailing ")"
szBuildLocation[_tcslen(szBuildLocation) -1 ] = 0;
}
}else{
_stprintf(szBuildLocation,
TEXT("%s"),
TEXT("No_Build_Lab_Information"));
Idwlog(TEXT("ERROR - Failed to get the build location in VerQueryValue.\n"));
}
**************/
// This should retrieve the Major/Minor Version and build and build deltas
//
if (0 == VerQueryValue(lpData, "\\", (PVOID*) &pv, (UINT*) &dwTemp)) {
// We have a problem.
//_tcscpy (szBld, TEXT("latest"));
//
Idwlog(TEXT("ERROR VerQueryValue failed to retrieve Version Info from: %s\n"), szFullPath );
if(lpData)
free (lpData);
return FALSE;
}
Idwlog(TEXT("GetImageHlpDllInfo - pv->dwSignature = %x\n"), pv->dwSignature );
Idwlog(TEXT("GetImageHlpDllInfo - pv->dwFileVersionMS = %x\n"), pv->dwFileVersionMS );
Idwlog(TEXT("GetImageHlpDllInfo - pv->dwFileVersionLS = %x\n"), pv->dwFileVersionLS );
// Cast the pvoid into the correct memory arrangement
//
switch (iFlag){
case F_INSTALLING_IMAGEHLP_DLL:
Idwlog(TEXT("GetImageHlpDllInfo F_INSTALLING_IMAGEHLP_DLL - Inserting into struct Build, Delta, Build lab for the installing files.\n"));
// Build Location
_tcscpy( pId->szInstallingBuildSourceLocation,szBuildLocation);
// Major version X.51
pId->dwInstallingMajorVersion = HIWORD(pv->dwFileVersionMS);
// Minor version 5.XX
pId->dwInstallingMinorVersion = LOWORD(pv->dwFileVersionMS);
// Build
pId->dwInstallingBuild = HIWORD(pv->dwFileVersionLS);
// Build Delta.
pId->dwInstallingBuildDelta = LOWORD(pv->dwFileVersionLS);
pId->dwInstallingSPBuild = LOWORD(pv->dwFileVersionLS);
Idwlog(TEXT("GetImageHlpDllInfo F_INSTALLING_IMAGEHLP_DLL - Build = %ld, Major = %ld, Minor = %ld, Delta = %ld\n"),
pId->dwInstallingBuild,
pId->dwInstallingMajorVersion,
pId->dwInstallingMinorVersion,
pId->dwInstallingBuildDelta
);
break;
case F_SYSTEM_IMAGEHLP_DLL:
// Get the local systems location of the imagehlp.dll
Idwlog(TEXT("GetImageHlpDllInfo F_SYSTEM_IMAGEHLP_DLL - Inserting into struct Build, Delta, Build lab for the local system.\n"));
// Build Location
_tcscpy( pId->szSystemBuildSourceLocation,szBuildLocation);
// Major version X.51
pId->dwSystemMajorVersion = HIWORD(pv->dwFileVersionMS);
// Minor version 5.XX
pId->dwSystemMinorVersion = LOWORD(pv->dwFileVersionMS);
// Build
pId->dwSystemBuild = HIWORD(pv->dwFileVersionLS);
// Build Delta.
pId->dwSystemBuildDelta = LOWORD(pv->dwFileVersionLS);
pId->dwSystemSPBuild = LOWORD(pv->dwFileVersionLS);
Idwlog(TEXT("GetImageHlpDllInfo F_SYSTEM_IMAGEHLP_DLL - Build = %ld, Major = %ld, Minor = %ld, Delta = %ld\n"),
pId->dwSystemBuild,
pId->dwSystemMajorVersion,
pId->dwSystemMinorVersion,
pId->dwSystemBuildDelta
);
break;
}
// The MyGetFileVersion allocates its own memory.
if(lpData)
free (lpData);
return TRUE;
}
typedef struct tagVERHEAD {
WORD wTotLen;
WORD wValLen;
WORD wType; /* always 0 */
WCHAR szKey[(sizeof("VS_VERSION_INFO")+3)&~03];
VS_FIXEDFILEINFO vsf;
} VERHEAD ;
/*
Used from Filever.c wallyho. For some reason regular calls fail easier to just reuse this code
* [alanau]
*
* MyGetFileVersionInfo: Maps a file directly without using LoadLibrary. This ensures
* that the right version of the file is examined without regard to where the loaded image
* is. Since this is a local function, it allocates the memory which is freed by the caller.
* This makes it slightly more efficient than a GetFileVersionInfoSize/GetFileVersionInfo pair.
*/
BOOL
MyGetFileVersionInfo(LPTSTR lpszFilename, LPVOID *lpVersionInfo)
{
VS_FIXEDFILEINFO *pvsFFI = NULL;
UINT uiBytes = 0;
HINSTANCE hinst;
HRSRC hVerRes;
HANDLE FileHandle = NULL;
HANDLE MappingHandle = NULL;
LPVOID DllBase = NULL;
VERHEAD *pVerHead;
BOOL bResult = FALSE;
DWORD dwHandle;
DWORD dwLength;
Idwlog ( TEXT ( "Entering MyGetFileVersionInfo.\n") );
if (!lpVersionInfo) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR lpVersionInfo not.\n") );
return FALSE;
}
*lpVersionInfo = NULL;
FileHandle = CreateFile( lpszFilename,
GENERIC_READ,
FILE_SHARE_READ,
NULL,
OPEN_EXISTING,
0,
NULL
);
if (FileHandle == INVALID_HANDLE_VALUE) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR INVALID_HANDLE.\n") );
goto Cleanup;
}
MappingHandle = CreateFileMapping( FileHandle,
NULL,
PAGE_READONLY,
0,
0,
NULL
);
if (MappingHandle == NULL) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR MappingHandle == NULL.\n") );
goto Cleanup;
}
DllBase = MapViewOfFileEx( MappingHandle,
FILE_MAP_READ,
0,
0,
0,
NULL
);
if (DllBase == NULL) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR DllBase == NULL.\n") );
goto Cleanup;
}
hinst = (HMODULE)((ULONG_PTR)DllBase | 0x00000001);
__try {
hVerRes = FindResource(hinst, MAKEINTRESOURCE(VS_VERSION_INFO), VS_FILE_INFO);
if (hVerRes == NULL)
{
// Probably a 16-bit file. Fall back to system APIs.
if(!(dwLength = GetFileVersionInfoSize(lpszFilename, &dwHandle)))
{
if(!GetLastError()) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR resource not found.\n") );
SetLastError(ERROR_RESOURCE_DATA_NOT_FOUND);
}
__leave;
}
if(!(*lpVersionInfo = malloc( dwLength) ) ) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR could not malloc.\n") );
__leave;
}
else {
Idwlog ( TEXT ( "MyGetFileVersionInfo: malloc size was: %ld.\n"), gdwLength = dwLength );
}
if(!GetFileVersionInfo(lpszFilename, 0, dwLength, *lpVersionInfo)) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR GetFileVersionInfo failed.\n") );
__leave;
}
bResult = TRUE;
Idwlog ( TEXT ( "MyGetFileVersionInfo: result is true.\n") );
__leave;
}
pVerHead = (VERHEAD*)LoadResource(hinst, hVerRes);
if (pVerHead == NULL) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR pVerHead == NULL.\n") );
__leave;
}
*lpVersionInfo = malloc(pVerHead->wTotLen + pVerHead->wTotLen/2);
if (*lpVersionInfo == NULL) {
Idwlog ( TEXT ( "MyGetFileVersionInfo: ERROR *lpVersionInfo == NULL.\n") );
__leave;
}
memcpy(*lpVersionInfo, (PVOID)pVerHead, gdwLength = pVerHead->wTotLen);
bResult = TRUE;
Idwlog ( TEXT ( "MyGetFileVersionInfo: bResult == TRUE.\n") );
} __except (EXCEPTION_EXECUTE_HANDLER) {
Idwlog(TEXT("MyGetFileVersionIfno ERROR - Memory problems failed to retrieve version Info from image file of the installing build.\n"));
}
Cleanup:
if (FileHandle)
CloseHandle(FileHandle);
if (MappingHandle)
CloseHandle(MappingHandle);
if (DllBase)
UnmapViewOfFile(DllBase);
if (*lpVersionInfo && bResult == FALSE)
free(*lpVersionInfo);
Idwlog ( TEXT ( "Exiting MyGetFileVersionInfo: bResult = %ld.\n"), bResult );
return bResult;
}
BOOL
GetSkuFromDosNetInf (IN OUT LPINSTALL_DATA pId)
/*++
Routine Description:
This will retrieve the values from Miscellaneous that tell us
if the build is a personal, enterprise, etc...
This will then load the value into the pId field of dwSku.
Arguments:
A install_data struct which contains all the recording variables necessary.
Return Value:
TRUE if success
FALSE if fail
--*/
{
WIN32_FIND_DATA fd;
HANDLE hFind;
TCHAR szFullPath [ MAX_PATH ];
TCHAR szCurDir [ MAX_PATH ];//
LPTSTR ptstr;
// OSVERSIONINFO osex;
DWORD dwLength = MAX_PATH;
// DWORD dwBuild;
// DWORD dwBuildDelta;
TCHAR szSku[5] = "ERR";
// DWORD dwDontCare;
TCHAR * p = NULL;
TCHAR szTmp[MAX_PATH];
ptstr = szTmp;
// Get the installing files location of the dosnet.inf file
/****
szCurDir[0] = TEXT('\0');
GetModuleFileName( NULL,szCurDir, dwLength);
// Remove the Idwlog.exe part and get only the directory structure.
ptstr = szCurDir + _tcsclen(szCurDir);
while (*ptstr-- != TEXT('\\'));
ptstr++;
*ptstr = TEXT('\0');
Idwlog(TEXT("Getting DosNet.inf file location as %s.\n"), szCurDir);
****/
p = _tcsstr (GetCommandLine(), TEXT("Path="));
if ( p == NULL ) {
Idwlog ( TEXT ( "GetSkuFromDosNetInf ERROR - Could not determine path from command line, you need a Path=<path to the installing build> with trailing slash.\n") );
pId->dwSku = 666;
return(FALSE);
}
p += _tcsclen ( TEXT ("Path=" ) );
Idwlog(TEXT("Source path = %s\n"), p );
// Use this to get the location
// idwlog.exe -1 is run from. This tool
// will always assume the dosnet.inf is in its
// current path or two up.
_stprintf (szFullPath, TEXT("%s\\dosnet.inf"),p);
Idwlog(TEXT("Will attempt to open to get ProductType in szFullPath = %s.\n"), szFullPath);
GetPrivateProfileString(TEXT("Miscellaneous"),
TEXT("ProductType"),
TEXT("0"),
szSku,
sizeof(szSku)/sizeof(TCHAR),
szFullPath);
// Check to see if there was an error in the above call getting the value.
// If so, default to Professional for the value, but return.
//
if ( _tcsstr (szSku, TEXT("ERR")) ) {
pId->dwSku = 0;
Idwlog ( TEXT ( "ERROR - GetSkuFromDosNetInf GetPrivateProfileString FAILed. Will return pid->dwSku = %d\n"), pId->dwSku );
return(FALSE);
}
/*
case 0: // Professional
case 1: // Server
case 2: // Advanced Server
case 3: // DataCenter
case 4: // Personal
case 5: // Blade Server
case 6: // Small Business Server
default: // Professional
*/
*ptstr = TEXT('\0');
pId->dwSku = _tcstoul(szSku, &ptstr, 10);
if(pId->dwSku > 6){
// default to Professional
Idwlog ( TEXT ( "ERROR - found an unknown ProductType in dosnet.inf pid->dwSku = %d\n"), pId->dwSku );
pId->dwSku = 0;
return(FALSE);
}
Idwlog ( TEXT ( "We found the following product type: %d\n"), pId->dwSku );
return TRUE;
}