WindowsXP-SP1/base/ntsetup/opktools/buildhive/buildhive.cpp
2020-09-30 16:53:49 +02:00

368 lines
9.8 KiB
C++

/*++
Copyright (c) 2000 Microsoft Corporation
Module Name:
buildhive.cpp
Abstract:
Builds the hive from the specified inf files.
The inf files follow the same syntax as used
by setup.
Author:
Mike Cirello
Vijay Jayaseelan (vijayj)
Revision History:
03 March 2001 :
Rewamp the whole source to make it more maintainable
(particularly readable)
--*/
#include <new.h>
#include "buildhive.h"
#include "File.h"
#include "Data.h"
//
// Global variables used to get formatted message for this program.
//
HMODULE ThisModule = NULL;
WCHAR Message[4096];
//
// Define a function to be called if new fails to allocate memory.
//
int __cdecl MyNewHandler( size_t size )
{
_putws(GetFormattedMessage( ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_MEMORY_ALLOC_FAILED) );
// Exit program
//
ExitProcess(errOUT_OF_MEMORY);
}
//
// main() entry point
//
int
_cdecl
wmain(
int Argc,
wchar_t *Argv[]
)
{
DWORD ErrorCode = 0;
HANDLE hToken;
ThisModule = GetModuleHandle(NULL);
_set_new_handler( MyNewHandler );
try {
if (Argc < 2) {
return ShowProgramUsage();
}
std::wstring InputFile = Argv[1];
if ((InputFile == L"/?") ||
(InputFile == L"?") ||
(InputFile == L"-?") ||
(InputFile == L"-h")) {
return ShowProgramUsage();
}
RegUnLoadKey(HKEY_USERS, L"dummy");
//
// Set privileges needed to load and save registry keys.
//
OpenProcessToken(GetCurrentProcess(), TOKEN_ADJUST_PRIVILEGES, &hToken);
SetPrivilege(hToken,SE_BACKUP_NAME,TRUE);
SetPrivilege(hToken,SE_RESTORE_NAME,TRUE);
ErrorCode = GetLastError();
if (ErrorCode != ERROR_SUCCESS) {
throw new W32Error(ErrorCode);
}
std::cout << InputFile << std::endl;
//
// load the configuration file
//
File ConfigFile(InputFile.c_str(), false);
//
// look for the sections defining the target files and .inf files
//
//
// get the target directory
//
ConfigFile.AddInfSection(InputFile.c_str(),
L"Directory",
L"SetDirectory");
//
// set the directory
//
ConfigFile.ProcessSections();
//
// NOTE: The sections are processed in the order of addition
//
ConfigFile.AddInfSection(InputFile.c_str(),
L"Add Registry New",
L"AddRegNew");
//
// do the actual conversion from .inf to hive files since
// we may need them for adding existing entries
//
ConfigFile.ProcessSections();
//
// process the localization specific registry sections
//
ConfigFile.ProcessNlsRegistryEntries();
//
// process modify/delete entries
//
ConfigFile.AddInfSection(InputFile.c_str(),
L"Add Registry Existing",
L"AddRegExisting");
ConfigFile.AddInfSection(InputFile.c_str(),
L"Delete Registry Existing",
L"DelRegExisting");
//
// do the actual conversion from .inf to hive file
//
ConfigFile.ProcessSections();
//
// save the hive files and clean out the registry
//
ConfigFile.Cleanup();
} catch (DWORD x) {
ErrorCode = x;
std::cout << GetFormattedMessage( ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_ABNORMAL_PGM_TERMINATION);
switch (x) {
case errFILE_LOCKED:
std::cout << GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_FILE_LOCKED);
break;
case errBAD_FLAGS:
std::cout << GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_BAD_FLAGS);
break;
case errFILE_NOT_FOUND:
std::cout << GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_FILE_NOT_FOUND);
break;
case errGENERAL_ERROR:
std::cout << GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_GENERAL_ERROR);
break;
default:
std::cout << GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_ERROR_CODE,
x);
}
}
catch(W32Error *Error) {
if (Error) {
Error->Dump(std::cout);
ErrorCode = Error->ErrorCode;
delete Error;
} else {
ErrorCode = 1;
}
}
catch(...) {
ErrorCode = 1; // unknown error
_putws( GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_ERROR_ABNORMAL_PGM_TERMINATION) );
}
RegUnLoadKey(HKEY_USERS, L"dummy");
_putws( GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_COMPLETED) );
return ErrorCode;
}
BOOL SetPrivilege(
IN HANDLE hToken,
IN LPCTSTR lpszPrivilege,
IN BOOL bEnablePrivilege
)
/*++
Routine Description:
Sets privileges for the current process. Used to get permission
to save and loadregistry keys
Arguments :
hToken : Handle to the token whose priviledge has to be modified
lpszPrivilege : Priviledge name
bEnablePrivilege : Enable or disable the priviledge
Return Value :
TRUE if successful, otherwise FALSE.
--*/
{
TOKEN_PRIVILEGES tp;
LUID luid;
if (!LookupPrivilegeValue(NULL, lpszPrivilege, &luid)){
return FALSE;
}
tp.PrivilegeCount = 1;
tp.Privileges[0].Luid = luid;
if (bEnablePrivilege) {
tp.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;
} else {
tp.Privileges[0].Attributes = 0;
}
//
// Enable the privilege or disable all privileges.
//
AdjustTokenPrivileges(hToken,
FALSE,
&tp,
sizeof(TOKEN_PRIVILEGES),
(PTOKEN_PRIVILEGES) NULL,
(PDWORD) NULL);
//
// Call GetLastError to determine whether the function succeeded.
//
return (GetLastError() != ERROR_SUCCESS) ? FALSE : TRUE;
}
INT
ShowProgramUsage(
VOID
)
/*++
Routine Description:
Shows show help message on how to use the program.
Arguments:
None.
Return Value:
0 if successful other non-zero value
--*/
{
//
// TBD : Need to localize this message in future
// based on the need for localized WinPE build
// tools
//
_putws( GetFormattedMessage(ThisModule,
FALSE,
Message,
sizeof(Message)/sizeof(Message[0]),
MSG_PGM_USAGE) );
return 0;
}
//
// Returns a TCHAR string explaining the last win32 error code
//
PCTSTR
Error(
VOID
)
{
static TCHAR MessageBuffer[4096];
MessageBuffer[0] = UNICODE_NULL;
FormatMessage(
FORMAT_MESSAGE_FROM_SYSTEM |
FORMAT_MESSAGE_IGNORE_INSERTS,
NULL,
GetLastError(),
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
MessageBuffer,
sizeof(MessageBuffer)/sizeof(TCHAR),
NULL);
return MessageBuffer;
}