You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2140 lines
56 KiB
2140 lines
56 KiB
/*++
|
|
|
|
Copyright (c) 1997 Microsoft Corporation
|
|
All rights reserved.
|
|
|
|
Module Name:
|
|
|
|
Util.c
|
|
|
|
Abstract:
|
|
|
|
Uitility routines for printer migration from Win9x to NT
|
|
|
|
Author:
|
|
|
|
Muhunthan Sivapragasam (MuhuntS) 02-Jan-1996
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include "precomp.h"
|
|
|
|
extern CHAR szRunOnceRegistryPath[];
|
|
extern CHAR szSpool[];
|
|
extern CHAR szMigDll[];
|
|
|
|
//
|
|
// These are used in the process of creating registry keys where the
|
|
// data necessary the vendor setup to be started will be stored
|
|
//
|
|
CHAR *pszVendorSetupInfoPath = "Software\\Microsoft\\Windows NT\\CurrentVersion\\Print";
|
|
CHAR *pszVendorSetupInfo = "VendorSetupInfo";
|
|
CHAR *pszVendorSetupID = "VendorSetup";
|
|
CHAR *pszVendorSetupEnumerator = "VendorInfoEnumerator";
|
|
CHAR *pszPrinterNameKey = "PrinterName";
|
|
CHAR *pszMigrationVendorSetupCaller = "MigrationVendorSetupCaller";
|
|
CHAR szVendorSetupRunRegistryPath[] = "Software\\Microsoft\\Windows\\CurrentVersion\\Run";
|
|
const CHAR *pszVendorSetupCaller = "CallVendorSetupDlls";
|
|
|
|
const LONG dwMaxVendorSetupIDLength = 12;
|
|
const DWORD dwFourMinutes = 240000;
|
|
BOOL bMigrateDllCopyed = FALSE;
|
|
|
|
|
|
VOID
|
|
DebugMsg(
|
|
LPCSTR pszFormat,
|
|
...
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
On debug builds brings up a message box on severe errors
|
|
|
|
Arguments:
|
|
pszFormat : Format string
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
#if DBG
|
|
LPSTR psz;
|
|
CHAR szMsg[1024];
|
|
va_list vargs;
|
|
|
|
va_start(vargs, pszFormat);
|
|
StringCchVPrintfA(szMsg, SIZECHARS(szMsg), pszFormat, vargs);
|
|
va_end(vargs);
|
|
|
|
#ifdef MYDEBUG
|
|
if ( psz = GetStringFromRcFileA(IDS_TITLE) ) {
|
|
|
|
MessageBoxA(NULL, szMsg, psz, MB_OK);
|
|
FreeMem(psz);
|
|
psz = NULL;
|
|
}
|
|
#else
|
|
OutputDebugStringA("Printing Migration : ");
|
|
OutputDebugStringA(szMsg);
|
|
OutputDebugStringA("\n");
|
|
#endif
|
|
|
|
|
|
#endif
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
LogError(
|
|
IN LogSeverity Severity,
|
|
IN UINT uMessageId,
|
|
...
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Logs an error in the setup error log on NT side when something can not be
|
|
upgraded
|
|
|
|
Arguments:
|
|
uMessageId : Id to string in .rc file
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
LPSTR pszFormat;
|
|
CHAR szMsg[1024];
|
|
|
|
va_list vargs;
|
|
|
|
va_start(vargs, uMessageId);
|
|
|
|
pszFormat = GetStringFromRcFileA(uMessageId);
|
|
|
|
if ( pszFormat ) {
|
|
|
|
StringCchVPrintfA(szMsg, SIZECHARS(szMsg), pszFormat, vargs);
|
|
DebugMsg("%s", szMsg);
|
|
SetupLogErrorA(szMsg, Severity);
|
|
}
|
|
|
|
FreeMem(pszFormat);
|
|
|
|
va_end(vargs);
|
|
}
|
|
|
|
|
|
LPSTR
|
|
ErrorMsg(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Returns the error message string from a Win32 error
|
|
|
|
Arguments:
|
|
None
|
|
|
|
Return Value:
|
|
Pointer to a message string. Caller should free the string
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLastError;
|
|
LPSTR pszStr = NULL;
|
|
|
|
if ( !(dwLastError = GetLastError()) )
|
|
dwLastError = STG_E_UNKNOWN;
|
|
|
|
FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER |
|
|
FORMAT_MESSAGE_IGNORE_INSERTS |
|
|
FORMAT_MESSAGE_FROM_SYSTEM |
|
|
FORMAT_MESSAGE_MAX_WIDTH_MASK,
|
|
NULL,
|
|
dwLastError,
|
|
0,
|
|
(LPSTR)&pszStr,
|
|
0,
|
|
NULL);
|
|
|
|
|
|
|
|
return pszStr;
|
|
}
|
|
|
|
|
|
PVOID
|
|
AllocMem(
|
|
IN UINT cbSize
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Allocate memory from the heap
|
|
|
|
Arguments:
|
|
cbSize : Byte count
|
|
|
|
Return Value:
|
|
Pointer to the allocated memory
|
|
|
|
--*/
|
|
{
|
|
return LocalAlloc(LPTR, cbSize);
|
|
}
|
|
|
|
|
|
VOID
|
|
FreeMem(
|
|
IN PVOID p
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Free memory allocated on the heap
|
|
|
|
Arguments:
|
|
p : Pointer to the memory to be freed
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
LocalFree(p);
|
|
}
|
|
|
|
|
|
LPSTR
|
|
AllocStrA(
|
|
LPCSTR pszStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Allocate memory and make a copy of an ansi string field
|
|
|
|
Arguments:
|
|
pszStr : String to copy
|
|
|
|
Return Value:
|
|
Pointer to the copied string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
LPSTR pszRet = NULL;
|
|
|
|
if ( pszStr && *pszStr ) {
|
|
DWORD dwBufSize = (strlen(pszStr) + 1) * sizeof(CHAR);
|
|
|
|
pszRet = AllocMem(dwBufSize);
|
|
if ( pszRet )
|
|
StringCbCopyA(pszRet, dwBufSize, pszStr);
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
LPWSTR
|
|
AllocStrW(
|
|
LPCWSTR pszStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Allocate memory and make a copy of a unicode string field
|
|
|
|
Arguments:
|
|
pszStr : String to copy
|
|
|
|
Return Value:
|
|
Pointer to the copied string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
LPWSTR pszRet = NULL;
|
|
|
|
if ( pszStr && *pszStr ) {
|
|
DWORD dwBufSize = (wcslen(pszStr) + 1) * sizeof(WCHAR);
|
|
pszRet = AllocMem(dwBufSize);
|
|
if ( pszRet )
|
|
StringCbCopyW(pszRet, dwBufSize, pszStr);
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
LPWSTR
|
|
AllocStrWFromStrA(
|
|
LPCSTR pszStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Returns the unicode string for a give ansi string. Memory is allocated.
|
|
|
|
Arguments:
|
|
pszStr : Gives the ansi string to copy
|
|
|
|
Return Value:
|
|
Pointer to the copied unicode string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLen;
|
|
LPWSTR pszRet = NULL;
|
|
|
|
if ( pszStr &&
|
|
*pszStr &&
|
|
(dwLen = strlen(pszStr)) &&
|
|
(pszRet = AllocMem((dwLen + 1) * sizeof(WCHAR))) ) {
|
|
|
|
if ( MultiByteToWideChar(CP_ACP,
|
|
MB_PRECOMPOSED,
|
|
pszStr,
|
|
dwLen,
|
|
pszRet,
|
|
dwLen) ) {
|
|
|
|
pszRet[dwLen] = 0;
|
|
} else {
|
|
|
|
FreeMem(pszRet);
|
|
pszRet = NULL;
|
|
}
|
|
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
LPSTR
|
|
AllocStrAFromStrW(
|
|
LPCWSTR pszStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Returns the ansi string for a give unicode string. Memory is allocated.
|
|
|
|
Arguments:
|
|
pszStr : Gives the ansi string to copy
|
|
|
|
Return Value:
|
|
Pointer to the copied ansi string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLen;
|
|
LPSTR pszRet = NULL;
|
|
|
|
if ( pszStr &&
|
|
*pszStr &&
|
|
(dwLen = wcslen(pszStr)) &&
|
|
(pszRet = AllocMem((dwLen + 1 ) * sizeof(CHAR))) ) {
|
|
|
|
WideCharToMultiByte(CP_ACP,
|
|
0,
|
|
pszStr,
|
|
dwLen,
|
|
pszRet,
|
|
dwLen,
|
|
NULL,
|
|
NULL );
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
VOID
|
|
WriteToFile(
|
|
HANDLE hFile,
|
|
LPBOOL pbFail,
|
|
LPCSTR pszFormat,
|
|
...
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Format and write a string to the text file. This is used to write the
|
|
printing configuration on Win9x
|
|
|
|
Arguments:
|
|
hFile : File handle
|
|
pbFail : Set on error -- no more processing needed
|
|
pszFormat : Format string for the message
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
CHAR szMsg[1024];
|
|
va_list vargs;
|
|
DWORD dwSize, dwWritten;
|
|
HRESULT hRet;
|
|
|
|
if ( *pbFail )
|
|
return;
|
|
|
|
va_start(vargs, pszFormat);
|
|
hRet = StringCchVPrintfA(szMsg, SIZECHARS(szMsg), pszFormat, vargs);
|
|
va_end(vargs);
|
|
|
|
if(SUCCEEDED(hRet))
|
|
{
|
|
dwSize = strlen(szMsg) * sizeof(CHAR);
|
|
|
|
if ( !WriteFile(hFile, (LPCVOID)szMsg, dwSize, &dwWritten, NULL) ||
|
|
dwSize != dwWritten ) {
|
|
|
|
*pbFail = TRUE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
*pbFail = TRUE;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
WriteString(
|
|
IN HANDLE hFile,
|
|
IN OUT LPBOOL pbFail,
|
|
IN LPCSTR pszStr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Writes a string to the upgrade file on Win9x side. Since spooler strings
|
|
(ex. printer name, driver name) can have space in them we would write
|
|
all strings with []\n. So we can read strings with space on NT.
|
|
|
|
Arguments:
|
|
hFile : File handle
|
|
pszFormat : Format string for the message
|
|
pszStr : String to write
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLen;
|
|
|
|
if ( pszStr ) {
|
|
|
|
dwLen = strlen(pszStr);
|
|
WriteToFile(hFile, pbFail, "%3d [%s]\n", dwLen, pszStr);
|
|
}
|
|
else
|
|
WriteToFile(hFile, pbFail, " 0 []\n");
|
|
|
|
}
|
|
|
|
|
|
VOID
|
|
WriteDevMode(
|
|
IN HANDLE hFile,
|
|
IN OUT LPBOOL pbFail,
|
|
IN LPDEVMODEA pDevMode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Writes a devmode to the upgrade file on Win9x side. We write the size of
|
|
devmode and write this as a binary field
|
|
|
|
Arguments:
|
|
hFile : File handle
|
|
pbFail : On error set to TRUE
|
|
pDevMode : Pointer to devmode
|
|
|
|
Return Value:
|
|
None
|
|
|
|
--*/
|
|
{
|
|
DWORD cbSize, cbWritten;
|
|
|
|
if ( *pbFail )
|
|
return;
|
|
|
|
cbSize = pDevMode ? pDevMode->dmSize + pDevMode->dmDriverExtra : 0;
|
|
|
|
if ( cbSize ) {
|
|
|
|
WriteToFile(hFile, pbFail, "DevMode: %d [", cbSize);
|
|
|
|
if ( !WriteFile(hFile, (LPCVOID)pDevMode, cbSize, &cbWritten, NULL) ||
|
|
cbWritten != cbSize )
|
|
*pbFail = TRUE;
|
|
|
|
WriteToFile(hFile, pbFail, "]\n");
|
|
} else {
|
|
|
|
WriteToFile(hFile, pbFail, "DevMode: 0\n");
|
|
}
|
|
}
|
|
|
|
|
|
LPSTR
|
|
GetStringFromRcFileA(
|
|
UINT uId
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Load a string from the .rc file and make a copy of it by doing AllocStr
|
|
|
|
Arguments:
|
|
uId : Identifier for the string to be loaded
|
|
|
|
Return Value:
|
|
String value loaded, NULL on error. Caller should free the memory
|
|
|
|
--*/
|
|
{
|
|
CHAR buf[MAX_STRING_LEN];
|
|
|
|
if ( LoadStringA(UpgradeData.hInst, uId, buf, SIZECHARS(buf)) )
|
|
return AllocStrA(buf);
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
VOID
|
|
CleanupDriverMapping(
|
|
IN OUT HDEVINFO *phDevInfo,
|
|
IN OUT HINF *phNtInf,
|
|
IN OUT HINF *phUpgInf
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Close INF handles and delete the printer device info list
|
|
|
|
Arguments:
|
|
phDevInfo : Points to printer device info list
|
|
phNtInf : Points to INF handle for ntprint.inf
|
|
phUpgInfo : Points to the handle to upgrade inf
|
|
|
|
Return Value:
|
|
Pointer to the copied unicode string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
if ( phUpgInf && *phUpgInf != INVALID_HANDLE_VALUE ) {
|
|
|
|
SetupCloseInfFile(*phUpgInf);
|
|
*phUpgInf = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
if ( phNtInf && *phNtInf != INVALID_HANDLE_VALUE ) {
|
|
|
|
SetupCloseInfFile(*phNtInf);
|
|
*phNtInf = INVALID_HANDLE_VALUE;
|
|
}
|
|
|
|
if ( phDevInfo && *phDevInfo != INVALID_HANDLE_VALUE ) {
|
|
|
|
SetupDiDestroyDeviceInfoList(*phDevInfo);
|
|
*phDevInfo = INVALID_HANDLE_VALUE;
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
InitDriverMapping(
|
|
OUT HDEVINFO *phDevInfo,
|
|
OUT HINF *phNtInf,
|
|
OUT HINF *phUpgInf,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Opens necessary inf files and create the printer device info list for
|
|
driver upgrade
|
|
|
|
Arguments:
|
|
phDevInfo : Points to printer device info list
|
|
phNtInf : Points to INF handle for ntprint.inf
|
|
phUpgInfo : Points to the handle to upgrade inf
|
|
pbFail : Set on error -- no more processing needed
|
|
|
|
Return Value:
|
|
Pointer to the copied unicode string. Memory is allocated.
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLen;
|
|
CHAR szPath[MAX_PATH];
|
|
SP_DEVINSTALL_PARAMS DevInstallParams;
|
|
|
|
if ( *pbFail )
|
|
{
|
|
return;
|
|
}
|
|
|
|
*phDevInfo = SetupDiCreateDeviceInfoList((LPGUID)&GUID_DEVCLASS_PRINTER,
|
|
NULL);
|
|
|
|
StringCchCopyA(szPath, SIZECHARS(szPath), UpgradeData.pszDir);
|
|
dwLen = strlen(szPath);
|
|
szPath[dwLen++] = '\\';
|
|
StringCchCopyA(szPath+dwLen, SIZECHARS(szPath) - dwLen, "prtupg9x.inf");
|
|
*phUpgInf = SetupOpenInfFileA(szPath, NULL, INF_STYLE_WIN4, NULL);
|
|
|
|
StringCchCopyA(szPath, SIZECHARS(szPath), UpgradeData.pszSourceA);
|
|
dwLen = strlen(szPath);
|
|
szPath[dwLen++] = '\\';
|
|
|
|
StringCchCopyA(szPath+dwLen, SIZECHARS(szPath) - dwLen, "ntprint.inf");
|
|
*phNtInf = SetupOpenInfFileA(szPath, NULL, INF_STYLE_WIN4, NULL);
|
|
|
|
StringCchCopyA(szPath+dwLen, SIZECHARS(szPath) - dwLen, "layout.inf");
|
|
|
|
if ( *phDevInfo == INVALID_HANDLE_VALUE ||
|
|
(phUpgInf && *phUpgInf == INVALID_HANDLE_VALUE) ||
|
|
*phNtInf == INVALID_HANDLE_VALUE ||
|
|
!SetupOpenAppendInfFileA(szPath, *phNtInf, NULL) )
|
|
{
|
|
*pbFail = TRUE;
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Build the list of drivers from ntprint.inf in the working directory
|
|
//
|
|
DevInstallParams.cbSize = sizeof(DevInstallParams);
|
|
if ( !SetupDiGetDeviceInstallParamsA(*phDevInfo,
|
|
NULL,
|
|
&DevInstallParams) )
|
|
{
|
|
*pbFail = TRUE;
|
|
goto Cleanup;
|
|
}
|
|
|
|
DevInstallParams.Flags |= DI_INF_IS_SORTED | DI_ENUMSINGLEINF;
|
|
|
|
StringCchCopyA(szPath+dwLen, SIZECHARS(szPath) - dwLen, "ntprint.inf");
|
|
StringCchCopyA(DevInstallParams.DriverPath, SIZECHARS(DevInstallParams.DriverPath), szPath);
|
|
|
|
if ( !SetupDiSetDeviceInstallParamsA(*phDevInfo, NULL, &DevInstallParams) ||
|
|
!SetupDiBuildDriverInfoList(*phDevInfo, NULL, SPDIT_CLASSDRIVER) )
|
|
{
|
|
*pbFail = TRUE;
|
|
}
|
|
|
|
Cleanup:
|
|
if ( *pbFail )
|
|
CleanupDriverMapping(phDevInfo, phNtInf, phUpgInf);
|
|
}
|
|
|
|
|
|
VOID
|
|
WritePrinterInfo2(
|
|
IN HANDLE hFile,
|
|
IN LPPRINTER_INFO_2A pPrinterInfo2,
|
|
IN LPSTR pszDriver,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
{
|
|
DWORD dwSize;
|
|
LPINT ptr;
|
|
|
|
if ( *pbFail )
|
|
return;
|
|
|
|
WriteToFile(hFile, pbFail, "ServerName: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pServerName);
|
|
|
|
WriteToFile(hFile, pbFail, "PrinterName: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pPrinterName);
|
|
|
|
WriteToFile(hFile, pbFail, "ShareName: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pShareName);
|
|
|
|
WriteToFile(hFile, pbFail, "PortName: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pPortName);
|
|
|
|
//
|
|
// On the Win9x side we could have found a different driver name on NT side
|
|
// if so write it instead of the one returned by spooler
|
|
//
|
|
WriteToFile(hFile, pbFail, "DriverName: ");
|
|
WriteString(hFile, pbFail,
|
|
pszDriver ? pszDriver : pPrinterInfo2->pDriverName);
|
|
|
|
WriteToFile(hFile, pbFail, "Comment: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pComment);
|
|
|
|
WriteToFile(hFile, pbFail, "Location: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pLocation);
|
|
|
|
WriteDevMode(hFile, pbFail, pPrinterInfo2->pDevMode);
|
|
|
|
WriteToFile(hFile, pbFail, "SepFile: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pSepFile);
|
|
|
|
WriteToFile(hFile, pbFail, "PrintProcessor: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pPrintProcessor);
|
|
|
|
WriteToFile(hFile, pbFail, "Datatype: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pDatatype);
|
|
|
|
WriteToFile(hFile, pbFail, "Parameters: ");
|
|
WriteString(hFile, pbFail, pPrinterInfo2->pParameters);
|
|
|
|
// Security descriptor ???
|
|
|
|
WriteToFile(hFile, pbFail, "Attributes: %3d\n", pPrinterInfo2->Attributes);
|
|
|
|
WriteToFile(hFile, pbFail, "Priority: %3d\n", pPrinterInfo2->Priority);
|
|
|
|
WriteToFile(hFile, pbFail, "DefaultPriority: %3d\n", pPrinterInfo2->DefaultPriority);
|
|
|
|
WriteToFile(hFile, pbFail, "StartTime: %3d\n", pPrinterInfo2->StartTime);
|
|
|
|
WriteToFile(hFile, pbFail, "UntilTime: %3d\n", pPrinterInfo2->UntilTime);
|
|
|
|
WriteToFile(hFile, pbFail, "Status: %3d\n", pPrinterInfo2->Status);
|
|
|
|
// cJobs not needed
|
|
// AveragePPM not needed
|
|
WriteToFile(hFile, pbFail, "\n");
|
|
}
|
|
|
|
|
|
VOID
|
|
ReadString(
|
|
IN HANDLE hFile,
|
|
IN LPSTR pszPrefix,
|
|
OUT LPSTR *ppszStr,
|
|
IN BOOL bOptional,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
{
|
|
CHAR c;
|
|
LPSTR psz;
|
|
DWORD dwLen;
|
|
|
|
if ( *pbFail )
|
|
return;
|
|
|
|
//
|
|
// First skip the prefix
|
|
//
|
|
while ( *pszPrefix && (c = (CHAR) My_fgetc(hFile)) == *pszPrefix++ )
|
|
;
|
|
|
|
if ( *pszPrefix )
|
|
goto Fail;
|
|
|
|
//
|
|
// Skip spaces
|
|
//
|
|
while ( (c = (CHAR) My_fgetc(hFile)) == ' ' )
|
|
;
|
|
|
|
//
|
|
// Now is the string length
|
|
//
|
|
if ( !isdigit(c) )
|
|
goto Fail;
|
|
|
|
dwLen = c - '0';
|
|
while ( isdigit(c = (CHAR) My_fgetc(hFile)) )
|
|
dwLen = dwLen * 10 + c - '0';
|
|
|
|
if ( c != ' ' )
|
|
goto Fail;
|
|
|
|
//
|
|
// Now the string is there between []
|
|
//
|
|
if ( *ppszStr = (LPSTR) AllocMem((dwLen + 1) * sizeof(CHAR)) ) {
|
|
|
|
if ( (c = (CHAR) My_fgetc(hFile)) != '[' )
|
|
goto Fail;
|
|
|
|
for ( psz = *ppszStr ;
|
|
dwLen && (*psz = (CHAR) My_fgetc(hFile)) != (CHAR) EOF ;
|
|
++psz, --dwLen )
|
|
;
|
|
|
|
if ( dwLen )
|
|
goto Fail;
|
|
|
|
*psz = '\0';
|
|
|
|
//
|
|
// Make sure line ends with "]\n"
|
|
//
|
|
if ( (CHAR) My_fgetc(hFile) != ']' || (CHAR) My_fgetc(hFile) != '\n' )
|
|
goto Fail;
|
|
|
|
return;
|
|
}
|
|
|
|
Fail:
|
|
*pbFail = TRUE;
|
|
FreeMem(*ppszStr);
|
|
*ppszStr = NULL;
|
|
}
|
|
|
|
|
|
VOID
|
|
ReadDword(
|
|
IN HANDLE hFile,
|
|
IN LPSTR pszLine,
|
|
IN DWORD dwLineSize,
|
|
IN LPSTR pszPrefix,
|
|
OUT LPDWORD pdwValue,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
{
|
|
LPSTR psz;
|
|
|
|
if ( *pbFail || My_fgets(pszLine, dwLineSize, hFile) == NULL ) {
|
|
|
|
*pbFail = TRUE;
|
|
return;
|
|
}
|
|
|
|
//
|
|
// First check the prefix matches to make sure we are in the right line
|
|
//
|
|
for ( psz = (LPSTR)pszLine ;
|
|
*pszPrefix && *psz == *pszPrefix ;
|
|
++psz, ++pszPrefix )
|
|
;
|
|
|
|
if ( *pszPrefix ) {
|
|
|
|
*pbFail = TRUE;
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Skip spaces
|
|
//
|
|
while ( *psz && *psz == ' ' )
|
|
++psz;
|
|
|
|
*pdwValue = atoi(psz);
|
|
}
|
|
|
|
|
|
VOID
|
|
ReadDevMode(
|
|
IN HANDLE hFile,
|
|
OUT LPDEVMODEA *ppDevMode,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
{
|
|
LPSTR pszPrefix = "DevMode:";
|
|
CHAR c;
|
|
DWORD dwLen;
|
|
LPINT ptr;
|
|
|
|
if ( *pbFail )
|
|
return;
|
|
|
|
// First skip the prefix
|
|
//
|
|
while ( *pszPrefix && (c = (CHAR) My_fgetc(hFile)) == *pszPrefix++ )
|
|
;
|
|
|
|
if ( *pszPrefix )
|
|
goto Fail;
|
|
|
|
//
|
|
// Skip spaces
|
|
//
|
|
while ( (c = (CHAR) My_fgetc(hFile)) == ' ' )
|
|
;
|
|
|
|
//
|
|
// Now is the devmode size
|
|
//
|
|
if ( !isdigit(c) )
|
|
goto Fail;
|
|
|
|
dwLen = c - '0';
|
|
while ( isdigit(c = (CHAR) My_fgetc(hFile)) )
|
|
dwLen = dwLen * 10 + c - '0';
|
|
|
|
if ( dwLen == 0 )
|
|
return;
|
|
|
|
if ( c != ' ' )
|
|
goto Fail;
|
|
|
|
//
|
|
// Now the devmode is there between []
|
|
//
|
|
if ( *ppDevMode = (LPDEVMODEA) AllocMem(dwLen) ) {
|
|
|
|
if ( (c = (CHAR) My_fgetc(hFile)) != '[' )
|
|
goto Fail;
|
|
|
|
if ( dwLen != My_fread((LPVOID)*ppDevMode, dwLen, hFile) )
|
|
goto Fail;
|
|
|
|
//
|
|
// Make sure now we have "]\n" to End
|
|
//
|
|
if ( (CHAR) My_fgetc(hFile) != ']' || (CHAR) My_fgetc(hFile) != '\n' ) {
|
|
|
|
DebugMsg("Char check fails");
|
|
goto Fail;
|
|
}
|
|
|
|
return; // Succesful exit
|
|
}
|
|
|
|
Fail:
|
|
*pbFail = TRUE;
|
|
}
|
|
|
|
|
|
VOID
|
|
FreePrinterInfo2Strings(
|
|
PPRINTER_INFO_2A pPrinterInfo2
|
|
)
|
|
{
|
|
if ( pPrinterInfo2 ) {
|
|
|
|
FreeMem(pPrinterInfo2->pServerName);
|
|
FreeMem(pPrinterInfo2->pPrinterName);
|
|
FreeMem(pPrinterInfo2->pShareName);
|
|
FreeMem(pPrinterInfo2->pPortName);
|
|
FreeMem(pPrinterInfo2->pDriverName);
|
|
FreeMem(pPrinterInfo2->pComment);
|
|
FreeMem(pPrinterInfo2->pLocation);
|
|
FreeMem(pPrinterInfo2->pDevMode);
|
|
FreeMem(pPrinterInfo2->pSepFile);
|
|
FreeMem(pPrinterInfo2->pPrintProcessor);
|
|
FreeMem(pPrinterInfo2->pDatatype);
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
ReadPrinterInfo2(
|
|
IN HANDLE hFile,
|
|
IN LPPRINTER_INFO_2A pPrinterInfo2,
|
|
IN OUT LPBOOL pbFail
|
|
)
|
|
{
|
|
CHAR szLine[2*MAX_PATH];
|
|
DWORD dwSize;
|
|
|
|
dwSize = sizeof(szLine)/sizeof(szLine[0]);
|
|
|
|
ReadString(hFile, "ServerName:",
|
|
&pPrinterInfo2->pServerName, TRUE, pbFail);
|
|
|
|
ReadString(hFile, "PrinterName:",
|
|
&pPrinterInfo2->pPrinterName, FALSE, pbFail);
|
|
|
|
ReadString(hFile, "ShareName:",
|
|
&pPrinterInfo2->pShareName, TRUE, pbFail);
|
|
|
|
ReadString(hFile, "PortName:",
|
|
&pPrinterInfo2->pPortName, FALSE, pbFail);
|
|
|
|
ReadString(hFile, "DriverName:",
|
|
&pPrinterInfo2->pDriverName, FALSE, pbFail);
|
|
|
|
ReadString(hFile, "Comment:",
|
|
&pPrinterInfo2->pComment, TRUE, pbFail);
|
|
|
|
ReadString(hFile, "Location:",
|
|
&pPrinterInfo2->pLocation, TRUE, pbFail);
|
|
|
|
ReadDevMode(hFile, &pPrinterInfo2->pDevMode, pbFail);
|
|
|
|
ReadString(hFile, "SepFile:",
|
|
&pPrinterInfo2->pSepFile, TRUE, pbFail);
|
|
|
|
ReadString(hFile, "PrintProcessor:",
|
|
&pPrinterInfo2->pPrintProcessor, FALSE, pbFail);
|
|
|
|
ReadString(hFile, "Datatype:",
|
|
&pPrinterInfo2->pDatatype, TRUE, pbFail);
|
|
|
|
ReadString(hFile, "Parameters:",
|
|
&pPrinterInfo2->pParameters, TRUE, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "Attributes:",
|
|
&pPrinterInfo2->Attributes, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "Priority:",
|
|
&pPrinterInfo2->Priority, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "DefaultPriority:",
|
|
&pPrinterInfo2->DefaultPriority, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "StartTime:",
|
|
&pPrinterInfo2->StartTime, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "UntilTime:",
|
|
&pPrinterInfo2->UntilTime, pbFail);
|
|
|
|
ReadDword(hFile, szLine, dwSize, "Status:",
|
|
&pPrinterInfo2->Status, pbFail);
|
|
|
|
//
|
|
// Skip the blank line
|
|
//
|
|
My_fgets(szLine, dwSize, hFile);
|
|
|
|
if ( *pbFail ) {
|
|
|
|
FreePrinterInfo2Strings(pPrinterInfo2);
|
|
ZeroMemory(pPrinterInfo2, sizeof(*pPrinterInfo2));
|
|
}
|
|
}
|
|
|
|
|
|
LPSTR
|
|
GetDefPrnString(
|
|
IN LPCSTR pszPrinterName
|
|
)
|
|
{
|
|
DWORD dwLen;
|
|
LPSTR pszRet;
|
|
|
|
dwLen = strlen(pszPrinterName) + 1 + strlen("winspool") + 1;
|
|
if ( pszRet = (LPSTR) AllocMem(dwLen * sizeof(CHAR)) ) {
|
|
|
|
StringCchPrintfA(pszRet, dwLen, "%s,%s", pszPrinterName, "winspool");
|
|
}
|
|
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
DWORD
|
|
GetFileNameInSpoolDir(
|
|
IN LPSTR szBuf,
|
|
IN DWORD cchBuf,
|
|
IN LPSTR pszFileName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
Function returns fully qualified path of the given file name in the spool
|
|
directory
|
|
|
|
Arguments:
|
|
szPath : Buffer to put the file name in
|
|
cchBuf : Buffer size in characters
|
|
pszFileName : File name part
|
|
|
|
Return Value:
|
|
Number of chars copied without \0 on success, 0 on failure
|
|
|
|
--*/
|
|
{
|
|
DWORD dwLen, dwLen1;
|
|
|
|
dwLen = GetSystemDirectoryA(szBuf, cchBuf);
|
|
|
|
if ( !dwLen )
|
|
return 0;
|
|
|
|
dwLen += strlen(szSpool) + strlen(pszFileName);
|
|
|
|
if ( dwLen + 1 > cchBuf )
|
|
return 0;
|
|
|
|
StringCchCatA(szBuf, cchBuf, szSpool);
|
|
StringCchCatA(szBuf, cchBuf, pszFileName);
|
|
|
|
return dwLen;
|
|
}
|
|
|
|
|
|
LPSTR
|
|
GetVendorSetupRunOnceValueToSet(
|
|
VOID
|
|
)
|
|
/*++
|
|
--*/
|
|
{
|
|
CHAR szPath[MAX_PATH];
|
|
DWORD dwLen, dwSize;
|
|
LPSTR pszRet = NULL;
|
|
|
|
dwSize = sizeof(szPath)/sizeof(szPath[0]);
|
|
|
|
if ( !(dwLen = GetFileNameInSpoolDir(szPath, dwSize, szMigDll)) )
|
|
goto Done;
|
|
|
|
//
|
|
// Now build up the RunOnce key which will be set for each user
|
|
//
|
|
dwSize = strlen("rundll32.exe") + dwLen +
|
|
+ strlen(pszVendorSetupCaller) + 4;
|
|
|
|
if ( pszRet = AllocMem(dwSize * sizeof(CHAR)) )
|
|
StringCchPrintfA(pszRet, dwSize,
|
|
"rundll32.exe %s,%s",
|
|
szPath, pszVendorSetupCaller);
|
|
Done:
|
|
return pszRet;
|
|
}
|
|
|
|
|
|
LONG
|
|
WriteVendorSetupInfoInRegistry(
|
|
IN CHAR *pszVendorSetup,
|
|
IN CHAR *pszPrinterName
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This routine is called to write the name of the vendor's installer DLL,
|
|
the entry point of that DLL, and the name of the printer
|
|
|
|
The vendor setup information is stored as described below:
|
|
|
|
HKLM
|
|
\Software
|
|
\Microsoft
|
|
\Windows NT
|
|
\CurrentVersion
|
|
\Print
|
|
\VendorSetupInfo
|
|
\VendorInfoEnumerator N
|
|
\VendorSetup1 Vendor1Dll,EntryPoint "Printer1 Name"
|
|
\VendorSetup2 Vendor2Dll,EntryPoint "Printer2 Name"
|
|
.............................................................
|
|
\VendorSetupN VendorNDll,EntryPoint "PrinterN Name"
|
|
|
|
The value N of VendorInfoEnumerator is equal to the number of the printers
|
|
for which vendor setup is provided. That value will be used to enumerate
|
|
the Dll's provided by vendors in the process of calling the entry points
|
|
of those Dlls.
|
|
|
|
The type of VendorInfoEnumerator is REG_DWORD.
|
|
The value of each VendorSetupX key (where 1<= X <= N) is a string containing
|
|
the name of the VendorSetup DLL, the entry point of that DLL and the
|
|
corresponding printer name. WrireVendorSetupInfoInRegistry function
|
|
concatenates its input parameters to produce that value and to write in into
|
|
the registry. The type of every VendorSetupX value is REG_SZ.
|
|
|
|
The information about the function in migrate.dll which to be called after
|
|
the first administrator's logon is stored into the registry as it is shown
|
|
below:
|
|
|
|
HKLM
|
|
\Software
|
|
\Microsoft
|
|
\Windows
|
|
\CurrentVersion
|
|
\Run
|
|
\MigrationVendorSetupCaller
|
|
|
|
The value of MigrationVendorSetupCaller is:
|
|
|
|
rundll32.exe %WinRoot%\system32\spool\migrate.dll,CallVendorSetupDlls
|
|
|
|
The type of the value is REG_SZ.
|
|
|
|
|
|
Arguments:
|
|
pszVendorSetup - null terminated string containing both the name of the
|
|
vendor's DLL and the entry point of that DLL
|
|
pszPrinterName - null terminated string containing the name of the printer
|
|
|
|
Return Value:
|
|
ERROR_SUCCES in the case of success.
|
|
error code in the other case.
|
|
|
|
--*/
|
|
{
|
|
LONG lRet = ERROR_BADKEY;
|
|
HKEY hKeyVendorInfo = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoPath = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoInstaller = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoEnumerator = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorRunOnceValuePath = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorRunOnceCallerValue = INVALID_HANDLE_VALUE;
|
|
CHAR *pszBuffer = NULL;
|
|
CHAR *pszBuffer1 = NULL;
|
|
DWORD dwType = 0;
|
|
DWORD dwSize = 0;
|
|
LONG lEnumerator = 0;
|
|
DWORD dwDisposition = 0;
|
|
UINT cbBufferSize = 0;
|
|
UINT cbBuffer1Size = 0;
|
|
CHAR *pszVendorSetupIDAsStr = NULL;
|
|
CHAR *pszVendorSetupRunOnceValue = NULL;
|
|
|
|
|
|
if (!pszVendorSetup || (strlen(pszVendorSetup) == 0) ||
|
|
!pszPrinterName || (strlen(pszPrinterName) == 0))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// We have to open the HKLM\Software\Microsoft\Windows NT\CurrentVersion\Print\VendorSetupInfo
|
|
// key first.
|
|
//
|
|
lRet = RegCreateKeyEx( HKEY_LOCAL_MACHINE, pszVendorSetupInfoPath, 0,
|
|
NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorInfoPath, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Now we will try to create the VendorSetupInfo key
|
|
//
|
|
lRet = RegCreateKeyEx( hKeyVendorInfoPath, pszVendorSetupInfo, 0,
|
|
NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorInfo, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Here we can create/open the VendorInfoEnumerator key.
|
|
//
|
|
lRet = RegCreateKeyEx( hKeyVendorInfo, pszVendorSetupEnumerator, 0,
|
|
NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorInfoEnumerator, &dwDisposition );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
if (dwDisposition == REG_OPENED_EXISTING_KEY)
|
|
{
|
|
//
|
|
// The VendorInfoEnumerator alredy exists. We opened and existing
|
|
// key. So we have to increment its value with 1 because we intend
|
|
// to create another VendorSetup key and to store there the
|
|
// corresponding information.
|
|
//
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof( lEnumerator );
|
|
if (ERROR_SUCCESS != RegQueryValueEx(hKeyVendorInfoEnumerator,
|
|
pszVendorSetupEnumerator, 0,
|
|
&dwType, (LPBYTE)(&lEnumerator),
|
|
&dwSize ) )
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
lEnumerator++;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// The VendorInfoEnumerator has been created. So this is the first
|
|
// printer for which we have VendorSetup provided.
|
|
//
|
|
lEnumerator = 1;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Below we will convert the value of VendorInfoEnumerator to a string and
|
|
// will concatenate that string to "VendorSetup" to produce the names of
|
|
// the Registry key and value where the data about the vendor provided DLL,
|
|
// its entry point and the printer will be stored.
|
|
//
|
|
pszVendorSetupIDAsStr = AllocMem( dwMaxVendorSetupIDLength * sizeof(CHAR) );
|
|
if (!pszVendorSetupIDAsStr)
|
|
{
|
|
lRet = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
_itoa( lEnumerator, pszVendorSetupIDAsStr, 10 );
|
|
|
|
//
|
|
// Below the memory necessary to build the vendor setup data and the
|
|
// registry key name from the input data and from the value of
|
|
// the VendorInfoEnumerator will be allocated.
|
|
//
|
|
cbBufferSize = (strlen(pszVendorSetup) + strlen(pszPrinterName) + strlen(TEXT(" \"\"")) + 2) * sizeof(CHAR);
|
|
cbBuffer1Size = (strlen(pszVendorSetupID) + strlen(pszVendorSetupIDAsStr) + 2) * sizeof(CHAR);
|
|
pszBuffer1 = AllocMem( cbBuffer1Size );
|
|
pszBuffer = AllocMem( cbBufferSize );
|
|
if (!pszBuffer || !pszBuffer1)
|
|
{
|
|
lRet = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
StringCbCopyA( pszBuffer1, cbBuffer1Size, pszVendorSetupID);
|
|
StringCbCatA( pszBuffer1, cbBuffer1Size, pszVendorSetupIDAsStr );
|
|
|
|
//
|
|
// At this point pszBuffer1 points to the following string:
|
|
// VendorSetupK where K is an integer - the value of VendorInfoEnumerator
|
|
//
|
|
lRet = RegCreateKeyEx( hKeyVendorInfo, pszBuffer1, 0, NULL,
|
|
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorInfoInstaller, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// The Registry Key where to store the vendor setup data was
|
|
// created successfully.
|
|
//
|
|
|
|
StringCbCopyA( pszBuffer, cbBufferSize, pszVendorSetup);
|
|
StringCbCatA( pszBuffer, cbBufferSize, " \"");
|
|
StringCbCatA( pszBuffer, cbBufferSize, pszPrinterName );
|
|
StringCbCatA( pszBuffer, cbBufferSize, "\"");
|
|
|
|
//
|
|
// At this point pszBuffer points to the following string:
|
|
// VendorSetup.DLL,EntryPoint "PrinterName". We will store
|
|
// that string in the Registry Key which we just created.
|
|
//
|
|
lRet = RegSetValueEx(hKeyVendorInfoInstaller, pszBuffer1, 0,
|
|
REG_SZ, (BYTE *)pszBuffer, cbBufferSize );
|
|
if (lRet != ERROR_SUCCESS)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Here we will store the value of VendorInfoEnumerator.
|
|
//
|
|
dwSize = sizeof( lEnumerator );
|
|
lRet = RegSetValueEx(hKeyVendorInfoEnumerator, pszVendorSetupEnumerator,
|
|
0, REG_DWORD, (BYTE*)(&lEnumerator), dwSize );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Now we can try to store into the registry the information how to invoke
|
|
// the migrate.dll after the first log on of an administrator.
|
|
//
|
|
pszVendorSetupRunOnceValue = GetVendorSetupRunOnceValueToSet();
|
|
if (!pszVendorSetupRunOnceValue)
|
|
{
|
|
lRet = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// We will try to open the
|
|
// HKLM\Software\Microsoft\Windows\CurrentVersion\Run
|
|
//
|
|
lRet = RegCreateKeyEx( HKEY_LOCAL_MACHINE,
|
|
szVendorSetupRunRegistryPath, 0, NULL,
|
|
REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorRunOnceValuePath, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// We will try to create the
|
|
// HKLM\Software\Microsoft\Windows\CurrentVersion\Run\MigrationVendorSetupCaller
|
|
//
|
|
lRet = RegCreateKeyEx( hKeyVendorRunOnceValuePath, pszMigrationVendorSetupCaller, 0,
|
|
NULL, REG_OPTION_NON_VOLATILE, KEY_ALL_ACCESS, NULL,
|
|
&hKeyVendorRunOnceCallerValue, &dwDisposition );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
if (dwDisposition == REG_OPENED_EXISTING_KEY)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Here we will store the "rundll.exe %WinRoot%\System32\spool\migrate.dll,CallVendorSetupDlls"
|
|
// string into the registry
|
|
//
|
|
lRet = RegSetValueEx(hKeyVendorRunOnceCallerValue, pszMigrationVendorSetupCaller,
|
|
0, REG_SZ, (BYTE *)pszVendorSetupRunOnceValue,
|
|
strlen(pszVendorSetupRunOnceValue) * sizeof(CHAR) );
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
if (pszVendorSetupRunOnceValue)
|
|
{
|
|
FreeMem(pszVendorSetupRunOnceValue);
|
|
}
|
|
if (pszVendorSetupIDAsStr)
|
|
{
|
|
FreeMem(pszVendorSetupIDAsStr);
|
|
}
|
|
if (pszBuffer)
|
|
{
|
|
FreeMem(pszBuffer);
|
|
}
|
|
if (pszBuffer1)
|
|
{
|
|
FreeMem(pszBuffer1);
|
|
}
|
|
if (hKeyVendorRunOnceValuePath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorRunOnceValuePath );
|
|
}
|
|
if (hKeyVendorRunOnceCallerValue != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorRunOnceCallerValue );
|
|
}
|
|
if (hKeyVendorInfoInstaller != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoInstaller );
|
|
}
|
|
if (hKeyVendorInfoEnumerator != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoEnumerator );
|
|
}
|
|
if (hKeyVendorInfo != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfo );
|
|
}
|
|
if (hKeyVendorInfoPath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
|
|
LONG
|
|
RemoveVendorSetupInfoFromRegistry(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This routine is called to remove the vendor setup information from the
|
|
registry
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
ERROR_SUCCESS in the case of success
|
|
error code in any other case
|
|
|
|
--*/
|
|
{
|
|
LONG lRet = ERROR_SUCCESS;
|
|
HKEY hKeyVendorInfoPath = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfo = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoEnumerator = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorRunOnceValuePath = INVALID_HANDLE_VALUE;
|
|
LONG lVendorSetupKeysNum = 0;
|
|
DWORD dwMaxSubKeyLen = 0;
|
|
DWORD dwMaxClassLen = 0;
|
|
DWORD dwValues = 0;
|
|
DWORD dwMaxValueNameLen = 0;
|
|
DWORD dwMaxValueLen = 0;
|
|
LONG lIndex = 0;
|
|
DWORD dwSize = 0;
|
|
DWORD dwType = 0;
|
|
DWORD dwKeyNameBufferLen = 0;
|
|
CHAR *pszKeyNameBuffer = NULL;
|
|
|
|
|
|
lRet = RegOpenKeyEx( HKEY_LOCAL_MACHINE, szVendorSetupRunRegistryPath, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorRunOnceValuePath );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Delete the registry keys used to store the location and the entry point
|
|
// of migrate.dll
|
|
//
|
|
lRet = RegDeleteKey( hKeyVendorRunOnceValuePath, pszMigrationVendorSetupCaller);
|
|
if (lRet != ERROR_SUCCESS)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Below we have to delete the registry keys used to store the descriptions
|
|
// of the vendor provided setup DLLs.
|
|
//
|
|
lRet = RegOpenKeyEx( HKEY_LOCAL_MACHINE, pszVendorSetupInfoPath, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfoPath );
|
|
if (lRet != ERROR_SUCCESS)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( hKeyVendorInfoPath, pszVendorSetupInfo, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfo );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Here we have to open the VendorInfoEnumerator and to read the
|
|
// number of vendor provided setup DLLs.
|
|
//
|
|
lRet = RegOpenKeyEx( hKeyVendorInfo, pszVendorSetupEnumerator, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfoEnumerator );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof( lVendorSetupKeysNum );
|
|
lRet = RegQueryValueEx(hKeyVendorInfoEnumerator, pszVendorSetupEnumerator, 0,
|
|
&dwType, (LPBYTE)(&lVendorSetupKeysNum), &dwSize );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
RegCloseKey( hKeyVendorInfoEnumerator );
|
|
hKeyVendorInfoEnumerator = INVALID_HANDLE_VALUE;
|
|
lRet = RegDeleteKey( hKeyVendorInfo, pszVendorSetupEnumerator);
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (lVendorSetupKeysNum <= 0)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// We have to add 1 for the the enumerator key itself to calculate the
|
|
// number of registry keys where the vendor setup descriptions are
|
|
// stored.
|
|
//
|
|
lVendorSetupKeysNum += 1;
|
|
|
|
//
|
|
// Below we will find the longest string used in the registry keys where
|
|
// the vendor setup information is stored.
|
|
//
|
|
lRet = RegQueryInfoKey( hKeyVendorInfo, NULL, NULL, NULL, &lVendorSetupKeysNum,
|
|
&dwMaxSubKeyLen, &dwMaxClassLen, &dwValues, &dwMaxValueNameLen,
|
|
&dwMaxValueLen, NULL, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
dwKeyNameBufferLen = __max( __max( dwMaxClassLen, dwMaxSubKeyLen),
|
|
__max( dwMaxValueNameLen, dwMaxValueLen ));
|
|
dwKeyNameBufferLen += 1;
|
|
|
|
//
|
|
// Now we have data enough to allocate a buffer long enough to store
|
|
// the longest string describing a key to delete.
|
|
//
|
|
pszKeyNameBuffer = AllocMem( dwKeyNameBufferLen * sizeof( CHAR ) );
|
|
if (!pszKeyNameBuffer)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Enumerate and delete the keys used to store the VendorSetup
|
|
// descriptions
|
|
//
|
|
lIndex = lVendorSetupKeysNum;
|
|
while (lIndex >= 0)
|
|
{
|
|
if (ERROR_SUCCESS != (lRet = RegEnumKey( hKeyVendorInfo, lIndex, pszKeyNameBuffer, dwKeyNameBufferLen)))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (ERROR_SUCCESS != (lRet = RegDeleteKey( hKeyVendorInfo, pszKeyNameBuffer)))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
lIndex--;
|
|
}
|
|
|
|
Cleanup:
|
|
|
|
if (pszKeyNameBuffer)
|
|
{
|
|
FreeMem( pszKeyNameBuffer );
|
|
}
|
|
if (hKeyVendorInfoEnumerator != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoEnumerator );
|
|
}
|
|
if (hKeyVendorInfo != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfo );
|
|
}
|
|
if (hKeyVendorInfoPath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegDeleteKey( hKeyVendorInfoPath, pszVendorSetupInfo );
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
}
|
|
if (hKeyVendorRunOnceValuePath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorRunOnceValuePath );
|
|
}
|
|
|
|
return lRet;
|
|
}
|
|
|
|
|
|
VOID
|
|
CallVendorSetupDlls(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This is called after the first log on of an administrator. It calls
|
|
vendor setup DLLs using the information we stored in the registry
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
--*/
|
|
{
|
|
LONG lRet = ERROR_SUCCESS;
|
|
HKEY hKeyVendorInfoPath = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfo = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoEnumerator = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorSetup = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorRunOnceValuePath = INVALID_HANDLE_VALUE;
|
|
HWND hwnd = INVALID_HANDLE_VALUE;
|
|
LONG lVendorSetupKeysNum = 0;
|
|
DWORD dwMaxSubKeyLen = 0;
|
|
DWORD dwMaxClassLen = 0;
|
|
DWORD dwValues = 0;
|
|
DWORD dwMaxValueNameLen = 0;
|
|
DWORD dwMaxValueLen = 0;
|
|
LONG lIndex = 0;
|
|
DWORD dwSize = 0;
|
|
DWORD dwType = 0;
|
|
DWORD dwKeyNameBufferLen = 0;
|
|
DWORD dwLastPos = 0;
|
|
CHAR *pszKeyNameBuffer = NULL;
|
|
CHAR *pszVendorSetupRunOnceValue = NULL;
|
|
BYTE *pszVendorSetupPtr = NULL;
|
|
BOOL bLocalAdmin = FALSE;
|
|
|
|
CHAR szParams[2*MAX_PATH+1];
|
|
CHAR szCmd[] = "rundll32.exe";
|
|
SHELLEXECUTEINFO ShellExecInfo;
|
|
|
|
|
|
if (!IsLocalAdmin(&bLocalAdmin))
|
|
{
|
|
lRet = GetLastError();
|
|
goto Cleanup;
|
|
}
|
|
if (!bLocalAdmin)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( HKEY_LOCAL_MACHINE, pszVendorSetupInfoPath, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfoPath );
|
|
if (lRet != ERROR_SUCCESS)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( hKeyVendorInfoPath, pszVendorSetupInfo, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfo );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
//
|
|
// The vendor setup registry keys are missing.
|
|
// So there is nothing to do and we can remove from the registry
|
|
// all the keyes we use to call the vendor setup Dlls
|
|
//
|
|
RemoveVendorSetupInfoFromRegistry();
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( hKeyVendorInfo, pszVendorSetupEnumerator, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfoEnumerator );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
//
|
|
// The vendor setup registry enumerator is missing.
|
|
// So the registry is damaged and the best is to remove from
|
|
// it the other keyes we use to call the vendor setup Dlls
|
|
//
|
|
RemoveVendorSetupInfoFromRegistry();
|
|
goto Cleanup;
|
|
}
|
|
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof( lVendorSetupKeysNum );
|
|
lRet = RegQueryValueEx(hKeyVendorInfoEnumerator, pszVendorSetupEnumerator, 0,
|
|
&dwType, (LPBYTE)(&lVendorSetupKeysNum), &dwSize );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
//
|
|
// We cannot read the vendor setup registry enumerator.
|
|
// So the registry is damaged and the best is to remove from
|
|
// it the other keyes we use to call the vendor setup Dlls
|
|
//
|
|
RemoveVendorSetupInfoFromRegistry();
|
|
goto Cleanup;
|
|
}
|
|
|
|
RegCloseKey( hKeyVendorInfoEnumerator );
|
|
hKeyVendorInfoEnumerator = INVALID_HANDLE_VALUE;
|
|
|
|
if (lVendorSetupKeysNum <= 0)
|
|
{
|
|
//
|
|
// We have only the enumerator and no any vendor setup info key.
|
|
// So there is nothing to do and we can remove from the registry
|
|
// all the keyes we use to call the vendor setup Dlls
|
|
//
|
|
RemoveVendorSetupInfoFromRegistry();
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// We have to add 1 for the enumerator key itself
|
|
//
|
|
lVendorSetupKeysNum += 1;
|
|
|
|
lRet = RegQueryInfoKey( hKeyVendorInfo, NULL, NULL, NULL, &lVendorSetupKeysNum,
|
|
&dwMaxSubKeyLen, &dwMaxClassLen, &dwValues, &dwMaxValueNameLen,
|
|
&dwMaxValueLen, NULL, NULL );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
dwKeyNameBufferLen = __max( __max( dwMaxClassLen, dwMaxSubKeyLen),
|
|
__max( dwMaxValueNameLen, dwMaxValueLen ));
|
|
dwKeyNameBufferLen += 2;
|
|
if ( dwKeyNameBufferLen > SIZECHARS(szParams) )
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
pszKeyNameBuffer = AllocMem( dwKeyNameBufferLen * sizeof( CHAR ) );
|
|
if (!pszKeyNameBuffer)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
dwSize = dwKeyNameBufferLen * sizeof( CHAR );
|
|
dwLastPos = dwKeyNameBufferLen - 1;
|
|
pszVendorSetupPtr = AllocMem( dwSize );
|
|
if (!pszVendorSetupPtr)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
hwnd = GetDesktopWindow();
|
|
for (lIndex = lVendorSetupKeysNum - 1; lIndex >= 0; lIndex--)
|
|
{
|
|
lRet = RegEnumKeyA( hKeyVendorInfo, lIndex, pszKeyNameBuffer, dwKeyNameBufferLen);
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
continue;
|
|
}
|
|
if (strcmp( pszKeyNameBuffer, pszVendorSetupEnumerator))
|
|
{
|
|
lRet = RegOpenKeyEx( hKeyVendorInfo, pszKeyNameBuffer, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorSetup );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
dwType = REG_SZ;
|
|
lRet = RegQueryValueExA(hKeyVendorSetup, pszKeyNameBuffer, 0,
|
|
&dwType, pszVendorSetupPtr, &dwSize );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
if (ERROR_MORE_DATA == lRet)
|
|
{
|
|
FreeMem( pszVendorSetupPtr );
|
|
pszVendorSetupPtr = AllocMem( dwSize );
|
|
dwLastPos = (dwSize/sizeof(CHAR)) - 1;
|
|
if (!pszVendorSetupPtr)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
lRet = RegQueryValueExA(hKeyVendorSetup, pszKeyNameBuffer, 0,
|
|
&dwType, pszVendorSetupPtr, &dwSize );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
}
|
|
|
|
pszVendorSetupPtr[dwLastPos] = '\0';
|
|
|
|
RegCloseKey( hKeyVendorSetup );
|
|
hKeyVendorSetup = INVALID_HANDLE_VALUE;
|
|
|
|
ZeroMemory(&ShellExecInfo, sizeof(ShellExecInfo));
|
|
ShellExecInfo.cbSize = sizeof(ShellExecInfo);
|
|
ShellExecInfo.hwnd = hwnd;
|
|
ShellExecInfo.lpFile = szCmd;
|
|
ShellExecInfo.nShow = SW_SHOWNORMAL;
|
|
ShellExecInfo.fMask = SEE_MASK_NOCLOSEPROCESS;
|
|
ShellExecInfo.lpParameters = pszVendorSetupPtr;
|
|
|
|
//
|
|
// Call run32dll and wait for the vendor dll to return before proceeding
|
|
//
|
|
if ( ShellExecuteEx(&ShellExecInfo) && ShellExecInfo.hProcess )
|
|
{
|
|
WaitForSingleObject(ShellExecInfo.hProcess, dwFourMinutes);
|
|
CloseHandle(ShellExecInfo.hProcess);
|
|
}
|
|
RegDeleteKey( hKeyVendorInfo, pszKeyNameBuffer);
|
|
|
|
//
|
|
// One of the registry keys describing a vendor provided setup DLL
|
|
// was removed. So the value of VendorInfoEnumerator must be
|
|
// decremented by 1.
|
|
//
|
|
DecrementVendorSetupEnumerator();
|
|
}
|
|
}
|
|
|
|
RemoveVendorSetupInfoFromRegistry();
|
|
|
|
Cleanup:
|
|
|
|
if (pszVendorSetupPtr)
|
|
{
|
|
FreeMem(pszVendorSetupPtr);
|
|
pszVendorSetupPtr = NULL;
|
|
}
|
|
if (pszVendorSetupRunOnceValue)
|
|
{
|
|
FreeMem(pszVendorSetupRunOnceValue);
|
|
}
|
|
if (pszKeyNameBuffer)
|
|
{
|
|
FreeMem( pszKeyNameBuffer );
|
|
}
|
|
|
|
if (hKeyVendorRunOnceValuePath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorRunOnceValuePath );
|
|
}
|
|
if (hKeyVendorSetup != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorSetup );
|
|
}
|
|
if (hKeyVendorInfoEnumerator != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoEnumerator );
|
|
}
|
|
if (hKeyVendorInfo != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfo );
|
|
}
|
|
if (hKeyVendorInfoPath != INVALID_HANDLE_VALUE)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
}
|
|
|
|
return;
|
|
}
|
|
|
|
|
|
BOOL
|
|
IsLocalAdmin(
|
|
BOOL *pbAdmin
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This Routine determines if the user is a local admin.
|
|
|
|
Parameters:
|
|
pbAdmin - Return Value, TRUE for local admin.
|
|
|
|
Return Value:
|
|
TRUE - Function succeded (return value is valid).
|
|
|
|
--*/
|
|
{
|
|
HMODULE AdvApi32Dll;
|
|
SID_IDENTIFIER_AUTHORITY SIDAuth = SECURITY_NT_AUTHORITY;
|
|
BOOL bRet = FALSE;
|
|
PSID pSIDAdmin = NULL;
|
|
|
|
AllOCANDINITSID pAllocAndInitID = NULL;
|
|
CHECKTOKENMEMBERSHIP pCheckTokenMemship = NULL;
|
|
FREESID pFreeSid = NULL;
|
|
|
|
|
|
ASSERT( pbAdmin != NULL ); // Called locally
|
|
|
|
*pbAdmin = FALSE;
|
|
|
|
AdvApi32Dll = LoadLibraryUsingFullPathA( "advapi32.dll" );
|
|
if (AdvApi32Dll == NULL)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
pAllocAndInitID = (AllOCANDINITSID)GetProcAddress( AdvApi32Dll, "AllocateAndInitializeSid");
|
|
pCheckTokenMemship = (CHECKTOKENMEMBERSHIP)GetProcAddress( AdvApi32Dll, "CheckTokenMembership");
|
|
pFreeSid = (FREESID)GetProcAddress( AdvApi32Dll, "FreeSid");
|
|
|
|
if (!pAllocAndInitID || !pCheckTokenMemship || !pFreeSid)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
if (!((*pAllocAndInitID)( &SIDAuth, 2,
|
|
SECURITY_BUILTIN_DOMAIN_RID,
|
|
DOMAIN_ALIAS_RID_ADMINS,
|
|
0, 0, 0, 0, 0, 0,
|
|
&pSIDAdmin)))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
if (!((*pCheckTokenMemship)( NULL,
|
|
pSIDAdmin,
|
|
pbAdmin )))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
bRet = TRUE;
|
|
|
|
Cleanup:
|
|
|
|
if (pSIDAdmin != NULL)
|
|
{
|
|
(*pFreeSid)( pSIDAdmin );
|
|
}
|
|
if (AdvApi32Dll)
|
|
{
|
|
FreeLibrary( AdvApi32Dll );
|
|
}
|
|
|
|
return bRet;
|
|
}
|
|
|
|
|
|
LONG
|
|
DecrementVendorSetupEnumerator(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This routine is called to decrement the value of the VendorInfoEnumerator.
|
|
It is called after removing of one of the registry keys containing a description
|
|
of a vendor provided DLL.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
ERROR_SUCCESS in the case of success
|
|
error code in any other case.
|
|
|
|
--*/
|
|
{
|
|
LONG lRet = ERROR_BADKEY;
|
|
HKEY hKeyVendorInfo = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorInfoPath = INVALID_HANDLE_VALUE;
|
|
HKEY hKeyVendorEnumerator = INVALID_HANDLE_VALUE;
|
|
LONG lEnumerator = 0;
|
|
DWORD dwDisposition = 0;
|
|
DWORD dwSize = 0;
|
|
DWORD dwType = 0;
|
|
|
|
|
|
lRet = RegOpenKeyEx( HKEY_LOCAL_MACHINE, pszVendorSetupInfoPath, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfoPath );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( hKeyVendorInfoPath, pszVendorSetupInfo, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorInfo );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
goto Cleanup;
|
|
}
|
|
|
|
lRet = RegOpenKeyEx( hKeyVendorInfo, pszVendorSetupEnumerator, 0,
|
|
KEY_ALL_ACCESS, &hKeyVendorEnumerator );
|
|
if (ERROR_SUCCESS != lRet)
|
|
{
|
|
RegCloseKey( hKeyVendorInfo );
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
goto Cleanup;
|
|
}
|
|
else
|
|
{
|
|
dwType = REG_DWORD;
|
|
dwSize = sizeof( lEnumerator );
|
|
lRet = RegQueryValueEx(hKeyVendorEnumerator, pszVendorSetupEnumerator, 0,
|
|
&dwType, (LPBYTE)(&lEnumerator), &dwSize );
|
|
if (ERROR_SUCCESS == lRet)
|
|
{
|
|
lEnumerator--;
|
|
lRet = RegSetValueEx(hKeyVendorEnumerator, pszVendorSetupEnumerator,
|
|
0, REG_DWORD, (BYTE*)(&lEnumerator), dwSize );
|
|
}
|
|
}
|
|
RegCloseKey( hKeyVendorEnumerator );
|
|
RegCloseKey( hKeyVendorInfo );
|
|
RegCloseKey( hKeyVendorInfoPath );
|
|
|
|
Cleanup:
|
|
|
|
return lRet;
|
|
}
|
|
|
|
BOOL
|
|
MakeACopyOfMigrateDll(
|
|
IN LPCSTR pszWorkingDir
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
This routine is called to copy the Migrate.Dll into the given
|
|
directory.
|
|
|
|
Arguments:
|
|
pszWorkingDir - the path where the Migrate.Dll to be copied.
|
|
|
|
Return Value:
|
|
FALSE - in the case of error
|
|
TRUE - in the case of success
|
|
The bMigrateDllCopyed global variable is set to the corresponding value
|
|
|
|
--*/
|
|
{
|
|
CHAR szSource[MAX_PATH];
|
|
CHAR szTarget[MAX_PATH];
|
|
DWORD dwSize;
|
|
DWORD dwLen;
|
|
|
|
if (bMigrateDllCopyed || !pszWorkingDir || !strlen(pszWorkingDir))
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
//
|
|
// First check if the source paths are ok
|
|
//
|
|
dwLen = strlen(szMigDll);
|
|
|
|
dwSize = sizeof(szTarget)/sizeof(szTarget[0]);
|
|
|
|
if ( strlen(pszWorkingDir) + dwLen + 2 > dwSize )
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
//
|
|
// Need to make a copy of migrate.dll to the %windir%\system32\spool
|
|
// directory
|
|
//
|
|
StringCchPrintfA(szSource, SIZECHARS(szSource), "%s\\%s", pszWorkingDir, szMigDll);
|
|
if ( !(dwLen = GetFileNameInSpoolDir(szTarget, dwSize, szMigDll)) ||
|
|
!CopyFileA(szSource, szTarget, FALSE) )
|
|
{
|
|
goto Cleanup;
|
|
}
|
|
|
|
bMigrateDllCopyed = TRUE;
|
|
|
|
Cleanup:
|
|
return bMigrateDllCopyed;
|
|
}
|
|
|
|
|
|
HMODULE LoadLibraryUsingFullPathA(
|
|
LPCSTR lpFileName
|
|
)
|
|
{
|
|
CHAR szSystemPath[MAX_PATH];
|
|
INT cLength = 0;
|
|
INT cFileNameLength = 0;
|
|
|
|
|
|
if (!lpFileName || ((cFileNameLength = strlen(lpFileName)) == 0))
|
|
{
|
|
return NULL;
|
|
}
|
|
if (GetSystemDirectoryA(szSystemPath, sizeof(szSystemPath)/sizeof(CHAR) ) == 0)
|
|
{
|
|
return NULL;
|
|
}
|
|
cLength = strlen(szSystemPath);
|
|
if (szSystemPath[cLength-1] != '\\')
|
|
{
|
|
if ((cLength + 1) >= MAX_PATH)
|
|
{
|
|
return NULL;
|
|
}
|
|
szSystemPath[cLength] = '\\';
|
|
szSystemPath[cLength + 1] = '\0';
|
|
cLength++;
|
|
}
|
|
if ((cLength + cFileNameLength) >= MAX_PATH)
|
|
{
|
|
return NULL;
|
|
}
|
|
StringCchCatA(szSystemPath, SIZECHARS(szSystemPath), lpFileName);
|
|
|
|
return LoadLibraryA( szSystemPath );
|
|
}
|
|
|