reactos/modules/rosapps/applications/cmdutils/gflags/imagefile.c
2018-10-19 23:13:51 +02:00

263 lines
8.2 KiB
C

/*
* PROJECT: Global Flags utility
* LICENSE: GPL-2.0 (https://spdx.org/licenses/GPL-2.0)
* PURPOSE: Global Flags utility image file options
* COPYRIGHT: Copyright 2018 Mark Jansen (mark.jansen@reactos.org)
*/
#include "gflags.h"
static PWSTR ImageFile = NULL;
static DWORD OptionsAdd = 0;
static DWORD OptionsRemove = 0;
static BOOL OptionsSet = FALSE;
typedef struct FlagInfo
{
DWORD dwFlag;
const wchar_t* szAbbr;
WORD wDest;
const wchar_t* szDesc;
} FlagInfo;
#define FLG_DISABLE_DBGPRINT 0x8000000
#define FLG_CRITSEC_EVENT_CREATION 0x10000000
#define FLG_STOP_ON_UNHANDLED_EXCEPTION 0x20000000
#define FLG_ENABLE_HANDLE_EXCEPTIONS 0x40000000
#define FLG_DISABLE_PROTDLLS 0x80000000
static const FlagInfo g_Flags[] =
{
{FLG_STOP_ON_EXCEPTION, L"soe", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Stop on exception"},
{FLG_SHOW_LDR_SNAPS, L"sls", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Show loader snaps"},
{FLG_DEBUG_INITIAL_COMMAND, L"dic", (DEST_REGISTRY), L"Debug initial command"},
{FLG_STOP_ON_HUNG_GUI, L"shg", (DEST_KERNEL), L"Stop on hung GUI"},
{FLG_HEAP_ENABLE_TAIL_CHECK, L"htc", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap tail checking"},
{FLG_HEAP_ENABLE_FREE_CHECK, L"hfc", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap free checking"},
{FLG_HEAP_VALIDATE_PARAMETERS, L"hpc", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap parameter checking"},
{FLG_HEAP_VALIDATE_ALL, L"hvc", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap validation on call"},
{FLG_APPLICATION_VERIFIER, L"vrf", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable application verifier"},
// FLG_MONITOR_SILENT_PROCESS_EXIT
{FLG_POOL_ENABLE_TAGGING, L"ptg", (DEST_REGISTRY), L"Enable pool tagging"},
{FLG_HEAP_ENABLE_TAGGING, L"htg", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap tagging"},
{FLG_USER_STACK_TRACE_DB, L"ust", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Create user mode stack trace database"},
{FLG_KERNEL_STACK_TRACE_DB, L"kst", (DEST_REGISTRY), L"Create kernel mode stack trace database"},
{FLG_MAINTAIN_OBJECT_TYPELIST, L"otl", (DEST_REGISTRY), L"Maintain a list of objects for each type"},
{FLG_HEAP_ENABLE_TAG_BY_DLL, L"htd", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable heap tagging by DLL"},
{FLG_DISABLE_STACK_EXTENSION, L"dse", (DEST_IMAGE), L"Disable stack extension"},
{FLG_ENABLE_CSRDEBUG, L"d32", (DEST_REGISTRY), L"Enable debugging of Win32 subsystem"},
{FLG_ENABLE_KDEBUG_SYMBOL_LOAD, L"ksl", (DEST_REGISTRY | DEST_KERNEL), L"Enable loading of kernel debugger symbols"},
{FLG_DISABLE_PAGE_KERNEL_STACKS, L"dps", (DEST_REGISTRY), L"Disable paging of kernel stacks"},
{FLG_ENABLE_SYSTEM_CRIT_BREAKS, L"scb", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable system critical breaks"},
{FLG_HEAP_DISABLE_COALESCING, L"dhc", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Disable heap coalesce on free"},
{FLG_ENABLE_CLOSE_EXCEPTIONS, L"ece", (DEST_REGISTRY | DEST_KERNEL), L"Enable close exception"},
{FLG_ENABLE_EXCEPTION_LOGGING, L"eel", (DEST_REGISTRY | DEST_KERNEL), L"Enable exception logging"},
{FLG_ENABLE_HANDLE_TYPE_TAGGING, L"eot", (DEST_REGISTRY | DEST_KERNEL), L"Enable object handle type tagging"},
{FLG_HEAP_PAGE_ALLOCS, L"hpa", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Enable page heap"},
{FLG_DEBUG_INITIAL_COMMAND_EX, L"dwl", (DEST_REGISTRY), L"Debug WinLogon"},
{FLG_DISABLE_DBGPRINT, L"ddp", (DEST_REGISTRY | DEST_KERNEL), L"Buffer DbgPrint Output"},
{FLG_CRITSEC_EVENT_CREATION, L"cse", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Early critical section event creation"},
{FLG_STOP_ON_UNHANDLED_EXCEPTION, L"sue", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Stop on unhandled user-mode exception"},
{FLG_ENABLE_HANDLE_EXCEPTIONS, L"bhd", (DEST_REGISTRY | DEST_KERNEL), L"Enable bad handles detection"},
{FLG_DISABLE_PROTDLLS, L"dpd", (DEST_REGISTRY | DEST_KERNEL | DEST_IMAGE), L"Disable protected DLL verification"},
};
void PrintFlags(IN DWORD GlobalFlags, IN OPTIONAL WORD Dest)
{
DWORD n;
for (n = 0; n < ARRAYSIZE(g_Flags); ++n)
{
if (!Dest || (g_Flags[n].wDest & Dest))
{
if (g_Flags[n].dwFlag & GlobalFlags)
{
wprintf(L" %s - %s\n", g_Flags[n].szAbbr, g_Flags[n].szDesc);
}
}
}
}
static void ShowStatus(DWORD GlobalFlags, DWORD Ignored)
{
if (GlobalFlags)
{
wprintf(L"Current Registry Settings for %s executable are: %08x\n", ImageFile, GlobalFlags);
PrintFlags(GlobalFlags, 0);
}
else
{
wprintf(L"No Registry Settings for %s executable\n", ImageFile);
}
if (Ignored)
{
wprintf(L"The following settings were ignored: %08x\n", Ignored);
PrintFlags(Ignored, 0);
}
}
static DWORD ValidateFlags(DWORD GlobalFlags, WORD Dest)
{
DWORD n;
DWORD Valid = 0;
for (n = 0; n < ARRAYSIZE(g_Flags); ++n)
{
if (g_Flags[n].wDest & Dest)
{
Valid |= g_Flags[n].dwFlag;
}
}
return GlobalFlags & Valid;
}
static DWORD FindFlag(PCWSTR Name, WORD Dest)
{
DWORD n;
for (n = 0; n < ARRAYSIZE(g_Flags); ++n)
{
if (g_Flags[n].wDest & Dest)
{
if (!wcsicmp(Name, g_Flags[n].szAbbr))
{
return g_Flags[n].dwFlag;
}
}
}
return 0;
}
static VOID ModifyStatus(VOID)
{
LONG Ret;
DWORD GlobalFlags, Requested, Ignored;
HKEY IFEOKey;
WCHAR Buffer[11];
if (!OpenImageFileExecOptions(KEY_WRITE | KEY_READ, ImageFile, &IFEOKey))
{
return;
}
if (OptionsSet)
{
Requested = OptionsAdd;
}
else
{
Requested = ReadSZFlagsFromRegistry(IFEOKey, L"GlobalFlag");
Requested &= ~OptionsRemove;
Requested |= OptionsAdd;
}
GlobalFlags = ValidateFlags(Requested, DEST_IMAGE);
Ignored = GlobalFlags ^ Requested;
if (GlobalFlags)
{
wsprintf(Buffer, L"0x%08x", GlobalFlags);
Ret = RegSetValueExW(IFEOKey, L"GlobalFlag", 0, REG_SZ, (BYTE*)Buffer, (wcslen(Buffer) + 1) * sizeof(WCHAR));
if (Ret != ERROR_SUCCESS)
{
wprintf(L"MS: RegSetValueEx failed (%d)\n", Ret);
}
else
{
ShowStatus(GlobalFlags, Ignored);
}
}
else
{
Ret = RegDeleteValueW(IFEOKey, L"GlobalFlag");
if (Ret != ERROR_SUCCESS)
{
wprintf(L"MS: RegDeleteValue failed (%d)\n", Ret);
}
else
{
ShowStatus(GlobalFlags, Ignored);
}
}
CloseHandle(IFEOKey);
}
static VOID DisplayStatus(VOID)
{
HKEY IFEOKey;
DWORD GlobalFlags;
if (!OpenImageFileExecOptions(KEY_READ, ImageFile, &IFEOKey))
{
return;
}
GlobalFlags = ReadSZFlagsFromRegistry(IFEOKey, L"GlobalFlag");
ShowStatus(GlobalFlags, 0);
CloseHandle(IFEOKey);
}
BOOL ImageFile_ParseCmdline(INT i, int argc, LPWSTR argv[])
{
for (; i < argc; i++)
{
if (ImageFile == NULL)
{
ImageFile = argv[i];
}
else if (argv[i][0] == '+')
{
if (OptionsSet)
{
wprintf(L"Unexpected argument - '%s'\n", argv[i]);
return FALSE;
}
OptionsAdd |= FindFlag(argv[i] + 1, DEST_IMAGE);
}
else if (argv[i][0] == '-')
{
if (OptionsSet)
{
wprintf(L"Unexpected argument - '%s'\n", argv[i]);
return FALSE;
}
OptionsRemove |= FindFlag(argv[i] + 1, DEST_IMAGE);
}
else
{
OptionsSet = TRUE;
OptionsAdd = wcstoul(argv[i], NULL, 16);
if (OptionsAdd == ~0)
OptionsAdd = 0;
}
}
if (ImageFile == NULL)
{
wprintf(L"No Image specified\n");
return FALSE;
}
return TRUE;
}
INT ImageFile_Execute()
{
if (!OptionsAdd && !OptionsRemove && !OptionsSet)
{
DisplayStatus();
}
else
{
ModifyStatus();
}
return 0;
}