2016-03-31 12:54:05 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2011 André Hentschel
|
|
|
|
* Copyright 2013 Mislav Blažević
|
|
|
|
* Copyright 2015 Mark Jansen
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define WIN32_NO_STATUS
|
|
|
|
#include "windows.h"
|
|
|
|
#include "ntndk.h"
|
|
|
|
#include "strsafe.h"
|
|
|
|
#include "apphelp.h"
|
|
|
|
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
|
|
|
|
|
2016-05-01 21:43:30 +00:00
|
|
|
static const GUID GUID_DATABASE_MSI = {0xd8ff6d16,0x6a3a,0x468a, {0x8b,0x44,0x01,0x71,0x4d,0xdc,0x49,0xea}};
|
|
|
|
static const GUID GUID_DATABASE_SHIM = {0x11111111,0x1111,0x1111, {0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11}};
|
|
|
|
static const GUID GUID_DATABASE_DRIVERS = {0xf9ab2228,0x3312,0x4a73, {0xb6,0xf9,0x93,0x6d,0x70,0xe1,0x12,0xef}};
|
|
|
|
|
2016-03-31 12:54:05 +00:00
|
|
|
static HANDLE SdbpHeap(void);
|
|
|
|
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
|
|
|
|
typedef struct SHIM_ALLOC_ENTRY
|
|
|
|
{
|
|
|
|
PVOID Address;
|
|
|
|
SIZE_T Size;
|
|
|
|
int Line;
|
|
|
|
const char* File;
|
|
|
|
PVOID Next;
|
|
|
|
PVOID Prev;
|
|
|
|
} SHIM_ALLOC_ENTRY, *PSHIM_ALLOC_ENTRY;
|
|
|
|
|
|
|
|
|
|
|
|
static RTL_AVL_TABLE g_SdbpAllocationTable;
|
|
|
|
|
|
|
|
|
|
|
|
static RTL_GENERIC_COMPARE_RESULTS
|
|
|
|
NTAPI ShimAllocCompareRoutine(_In_ PRTL_AVL_TABLE Table, _In_ PVOID FirstStruct, _In_ PVOID SecondStruct)
|
|
|
|
{
|
|
|
|
PVOID First = ((PSHIM_ALLOC_ENTRY)FirstStruct)->Address;
|
|
|
|
PVOID Second = ((PSHIM_ALLOC_ENTRY)SecondStruct)->Address;
|
|
|
|
|
|
|
|
if (First < Second)
|
|
|
|
return GenericLessThan;
|
|
|
|
else if (First == Second)
|
|
|
|
return GenericEqual;
|
|
|
|
return GenericGreaterThan;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PVOID NTAPI ShimAllocAllocateRoutine(_In_ PRTL_AVL_TABLE Table, _In_ CLONG ByteSize)
|
|
|
|
{
|
|
|
|
return HeapAlloc(SdbpHeap(), HEAP_ZERO_MEMORY, ByteSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
static VOID NTAPI ShimAllocFreeRoutine(_In_ PRTL_AVL_TABLE Table, _In_ PVOID Buffer)
|
|
|
|
{
|
|
|
|
HeapFree(SdbpHeap(), 0, Buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SdbpInsertAllocation(PVOID address, SIZE_T size, int line, const char* file)
|
|
|
|
{
|
|
|
|
SHIM_ALLOC_ENTRY Entry = {0};
|
|
|
|
|
|
|
|
Entry.Address = address;
|
|
|
|
Entry.Size = size;
|
|
|
|
Entry.Line = line;
|
|
|
|
Entry.File = file;
|
|
|
|
RtlInsertElementGenericTableAvl(&g_SdbpAllocationTable, &Entry, sizeof(Entry), NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SdbpUpdateAllocation(PVOID address, PVOID newaddress, SIZE_T size, int line, const char* file)
|
|
|
|
{
|
|
|
|
SHIM_ALLOC_ENTRY Lookup = {0};
|
|
|
|
PSHIM_ALLOC_ENTRY Entry;
|
|
|
|
Lookup.Address = address;
|
|
|
|
Entry = RtlLookupElementGenericTableAvl(&g_SdbpAllocationTable, &Lookup);
|
|
|
|
|
|
|
|
if (address == newaddress)
|
|
|
|
{
|
|
|
|
Entry->Size = size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Lookup.Address = newaddress;
|
|
|
|
Lookup.Size = size;
|
|
|
|
Lookup.Line = line;
|
|
|
|
Lookup.File = file;
|
|
|
|
Lookup.Prev = address;
|
|
|
|
RtlInsertElementGenericTableAvl(&g_SdbpAllocationTable, &Lookup, sizeof(Lookup), NULL);
|
|
|
|
Entry->Next = newaddress;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void SdbpRemoveAllocation(PVOID address, int line, const char* file)
|
|
|
|
{
|
|
|
|
char buf[512];
|
|
|
|
SHIM_ALLOC_ENTRY Lookup = {0};
|
|
|
|
PSHIM_ALLOC_ENTRY Entry;
|
|
|
|
|
|
|
|
sprintf(buf, "\r\n===============\r\n%s(%d): SdbpFree called, tracing alloc:\r\n", file, line);
|
|
|
|
OutputDebugStringA(buf);
|
|
|
|
|
|
|
|
Lookup.Address = address;
|
|
|
|
while (Lookup.Address)
|
|
|
|
{
|
|
|
|
Entry = RtlLookupElementGenericTableAvl(&g_SdbpAllocationTable, &Lookup);
|
|
|
|
if (Entry)
|
|
|
|
{
|
|
|
|
Lookup = *Entry;
|
|
|
|
RtlDeleteElementGenericTableAvl(&g_SdbpAllocationTable, Entry);
|
|
|
|
|
|
|
|
sprintf(buf, " > %s(%d): %s%sAlloc( %d ) ==> %p\r\n", Lookup.File, Lookup.Line,
|
|
|
|
Lookup.Next ? "Invalidated " : "", Lookup.Prev ? "Re" : "", Lookup.Size, Lookup.Address);
|
|
|
|
OutputDebugStringA(buf);
|
|
|
|
Lookup.Address = Lookup.Prev;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Lookup.Address = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sprintf(buf, "\r\n===============\r\n");
|
|
|
|
OutputDebugStringA(buf);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static HANDLE g_Heap;
|
|
|
|
void SdbpHeapInit(void)
|
|
|
|
{
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
RtlInitializeGenericTableAvl(&g_SdbpAllocationTable, ShimAllocCompareRoutine,
|
|
|
|
ShimAllocAllocateRoutine, ShimAllocFreeRoutine, NULL);
|
|
|
|
#endif
|
|
|
|
g_Heap = HeapCreate(0, 0x10000, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SdbpHeapDeinit(void)
|
|
|
|
{
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
if (g_SdbpAllocationTable.NumberGenericTableElements != 0)
|
|
|
|
__debugbreak();
|
|
|
|
#endif
|
|
|
|
HeapDestroy(g_Heap);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HANDLE SdbpHeap(void)
|
|
|
|
{
|
|
|
|
return g_Heap;
|
|
|
|
}
|
|
|
|
|
|
|
|
LPVOID SdbpAlloc(SIZE_T size
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
, int line, const char* file
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
LPVOID mem = HeapAlloc(SdbpHeap(), HEAP_ZERO_MEMORY, size);
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
SdbpInsertAllocation(mem, size, line, file);
|
|
|
|
#endif
|
|
|
|
return mem;
|
|
|
|
}
|
|
|
|
|
|
|
|
LPVOID SdbpReAlloc(LPVOID mem, SIZE_T size
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
, int line, const char* file
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
LPVOID newmem = HeapReAlloc(SdbpHeap(), HEAP_ZERO_MEMORY, mem, size);
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
SdbpUpdateAllocation(mem, newmem, size, line, file);
|
|
|
|
#endif
|
|
|
|
return newmem;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SdbpFree(LPVOID mem
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
, int line, const char* file
|
|
|
|
#endif
|
|
|
|
)
|
|
|
|
{
|
|
|
|
#if SDBAPI_DEBUG_ALLOC
|
|
|
|
SdbpRemoveAllocation(mem, line, file);
|
|
|
|
#endif
|
|
|
|
HeapFree(SdbpHeap(), 0, mem);
|
|
|
|
}
|
2016-04-24 13:05:00 +00:00
|
|
|
|
2016-05-01 19:12:06 +00:00
|
|
|
DWORD SdbpStrlen(PCWSTR string)
|
|
|
|
{
|
|
|
|
return (lstrlenW(string) + 1) * sizeof(WCHAR);
|
|
|
|
}
|
|
|
|
|
|
|
|
PWSTR SdbpStrDup(LPCWSTR string)
|
|
|
|
{
|
|
|
|
PWSTR ret = SdbpAlloc(SdbpStrlen(string));
|
|
|
|
lstrcpyW(ret, string);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOL WINAPI SdbpOpenMemMappedFile(LPCWSTR path, PMEMMAPPED mapping)
|
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
|
|
IO_STATUS_BLOCK IoStatusBlock;
|
|
|
|
FILE_STANDARD_INFORMATION FileStandard;
|
|
|
|
UNICODE_STRING FileName;
|
|
|
|
|
|
|
|
RtlZeroMemory(mapping, sizeof(*mapping));
|
|
|
|
|
|
|
|
if(!RtlDosPathNameToNtPathName_U(path, &FileName, NULL, NULL))
|
|
|
|
{
|
|
|
|
RtlFreeUnicodeString(&FileName);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
InitializeObjectAttributes(&ObjectAttributes, &FileName, OBJ_CASE_INSENSITIVE, NULL, NULL);
|
|
|
|
Status = NtOpenFile(&mapping->file, GENERIC_READ | SYNCHRONIZE, &ObjectAttributes, &IoStatusBlock, FILE_SHARE_READ, FILE_SYNCHRONOUS_IO_NONALERT);
|
|
|
|
RtlFreeUnicodeString(&FileName);
|
|
|
|
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SHIM_ERR("Failed to open file %S: 0x%lx\n", path, Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = NtCreateSection(&mapping->section, STANDARD_RIGHTS_REQUIRED | SECTION_QUERY | SECTION_MAP_READ, 0, 0, PAGE_READONLY, SEC_COMMIT, mapping->file);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Special case */
|
|
|
|
if (Status == STATUS_MAPPED_FILE_SIZE_ZERO)
|
|
|
|
{
|
|
|
|
NtClose(mapping->file);
|
|
|
|
mapping->file = mapping->section = NULL;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
SHIM_ERR("Failed to create mapping for file: 0x%lx\n", Status);
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = NtQueryInformationFile(mapping->file, &IoStatusBlock, &FileStandard, sizeof(FileStandard), FileStandardInformation);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SHIM_ERR("Failed to read file info for file: 0x%lx\n", Status);
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
mapping->mapped_size = mapping->size = FileStandard.EndOfFile.LowPart;
|
|
|
|
Status = NtMapViewOfSection(mapping->section, NtCurrentProcess(), (PVOID*)&mapping->view, 0, 0, 0, &mapping->mapped_size, ViewUnmap, 0, PAGE_READONLY);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
SHIM_ERR("Failed to map view of file: 0x%lx\n", Status);
|
|
|
|
goto err_out;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
err_out:
|
|
|
|
if (!mapping->view)
|
|
|
|
{
|
|
|
|
if (mapping->section)
|
|
|
|
NtClose(mapping->section);
|
|
|
|
NtClose(mapping->file);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
void WINAPI SdbpCloseMemMappedFile(PMEMMAPPED mapping)
|
|
|
|
{
|
|
|
|
NtUnmapViewOfSection(NtCurrentProcess(), mapping->view);
|
|
|
|
NtClose(mapping->section);
|
|
|
|
NtClose(mapping->file);
|
|
|
|
RtlZeroMemory(mapping, sizeof(*mapping));
|
|
|
|
}
|
|
|
|
|
2016-05-01 21:43:30 +00:00
|
|
|
/**
|
|
|
|
* Parses a string to retrieve a GUID.
|
|
|
|
*
|
|
|
|
* @param [in] GuidString The string to parse.
|
|
|
|
* @param [out] Guid The resulting GUID.
|
|
|
|
*
|
|
|
|
* @return TRUE if it succeeds, FALSE if it fails.
|
|
|
|
*/
|
|
|
|
BOOL WINAPI SdbGUIDFromString(PCWSTR GuidString, GUID *Guid)
|
|
|
|
{
|
|
|
|
UNICODE_STRING GuidString_u;
|
|
|
|
RtlInitUnicodeString(&GuidString_u, GuidString);
|
|
|
|
return NT_SUCCESS(RtlGUIDFromString(&GuidString_u, Guid));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Converts a GUID to a string.
|
|
|
|
*
|
|
|
|
* @param [in] Guid The GUID to convert.
|
|
|
|
* @param [out] GuidString The resulting string representation of Guid.
|
|
|
|
* @param [in] Length The length of GuidString.
|
|
|
|
*
|
|
|
|
* @return TRUE if it succeeds, FALSE if it fails.
|
|
|
|
*/
|
|
|
|
BOOL WINAPI SdbGUIDToString(CONST GUID *Guid, PWSTR GuidString, SIZE_T Length)
|
|
|
|
{
|
|
|
|
UNICODE_STRING GuidString_u;
|
|
|
|
if(NT_SUCCESS(RtlStringFromGUID(Guid, &GuidString_u)))
|
|
|
|
{
|
|
|
|
HRESULT hr = StringCchCopyNW(GuidString, Length, GuidString_u.Buffer, GuidString_u.Length / 2);
|
|
|
|
RtlFreeUnicodeString(&GuidString_u);
|
|
|
|
return SUCCEEDED(hr);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the specified GUID is a NULL GUID
|
|
|
|
*
|
|
|
|
* @param [in] Guid The GUID to check.
|
|
|
|
*
|
|
|
|
* @return TRUE if it is a NULL GUID.
|
|
|
|
*/
|
|
|
|
BOOL WINAPI SdbIsNullGUID(CONST GUID *Guid)
|
|
|
|
{
|
|
|
|
static GUID NullGuid = { 0 };
|
|
|
|
return !Guid || IsEqualGUID(&NullGuid, Guid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the GUID from one of the standard databases.
|
|
|
|
*
|
|
|
|
* @param [in] Flags The ID to retrieve the guid from. (See SDB_DATABASE_MAIN_[xxx])
|
|
|
|
* @param [out] Guid The resulting GUID.
|
|
|
|
*
|
|
|
|
* @return TRUE if a known database ID.
|
|
|
|
*/
|
|
|
|
BOOL WINAPI SdbGetStandardDatabaseGUID(DWORD Flags, GUID* Guid)
|
|
|
|
{
|
|
|
|
const GUID* copy_from = NULL;
|
|
|
|
switch(Flags & HID_DATABASE_TYPE_MASK)
|
|
|
|
{
|
|
|
|
case SDB_DATABASE_MAIN_MSI:
|
|
|
|
copy_from = &GUID_DATABASE_MSI;
|
|
|
|
break;
|
|
|
|
case SDB_DATABASE_MAIN_SHIM:
|
|
|
|
copy_from = &GUID_DATABASE_SHIM;
|
|
|
|
break;
|
|
|
|
case SDB_DATABASE_MAIN_DRIVERS:
|
|
|
|
copy_from = &GUID_DATABASE_DRIVERS;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SHIM_ERR("Cannot obtain database guid for databases other than main\n");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if(Guid)
|
|
|
|
{
|
|
|
|
memcpy(Guid, copy_from, sizeof(GUID));
|
|
|
|
}
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2016-04-24 13:05:00 +00:00
|
|
|
/**
|
|
|
|
* Converts specified tag into a string.
|
|
|
|
*
|
|
|
|
* @param [in] tag The tag which will be converted to a string.
|
|
|
|
*
|
|
|
|
* @return Success: Pointer to the string matching specified tag, or L"InvalidTag" on failure.
|
|
|
|
*/
|
|
|
|
LPCWSTR WINAPI SdbTagToString(TAG tag)
|
|
|
|
{
|
|
|
|
/* lookup tables for tags in range 0x1 -> 0xFF | TYPE */
|
2016-04-24 17:19:09 +00:00
|
|
|
static const WCHAR table[9][0x32][25] = {
|
2016-04-24 13:05:00 +00:00
|
|
|
{ /* TAG_TYPE_NULL */
|
|
|
|
{'I','N','C','L','U','D','E',0},
|
|
|
|
{'G','E','N','E','R','A','L',0},
|
|
|
|
{'M','A','T','C','H','_','L','O','G','I','C','_','N','O','T',0},
|
|
|
|
{'A','P','P','L','Y','_','A','L','L','_','S','H','I','M','S',0},
|
|
|
|
{'U','S','E','_','S','E','R','V','I','C','E','_','P','A','C','K','_','F','I','L','E','S',0},
|
|
|
|
{'M','I','T','I','G','A','T','I','O','N','_','O','S',0},
|
|
|
|
{'B','L','O','C','K','_','U','P','G','R','A','D','E',0},
|
|
|
|
{'I','N','C','L','U','D','E','E','X','C','L','U','D','E','D','L','L',0},
|
|
|
|
{'R','A','C','_','E','V','E','N','T','_','O','F','F',0},
|
|
|
|
{'T','E','L','E','M','E','T','R','Y','_','O','F','F',0},
|
|
|
|
{'S','H','I','M','_','E','N','G','I','N','E','_','O','F','F',0},
|
|
|
|
{'L','A','Y','E','R','_','P','R','O','P','A','G','A','T','I','O','N','_','O','F','F',0},
|
|
|
|
{'R','E','I','N','S','T','A','L','L','_','U','P','G','R','A','D','E',0}
|
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_BYTE */
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0}
|
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_WORD */
|
|
|
|
{'M','A','T','C','H','_','M','O','D','E',0}
|
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_DWORD */
|
|
|
|
{'S','I','Z','E',0},
|
|
|
|
{'O','F','F','S','E','T',0},
|
|
|
|
{'C','H','E','C','K','S','U','M',0},
|
|
|
|
{'S','H','I','M','_','T','A','G','I','D',0},
|
|
|
|
{'P','A','T','C','H','_','T','A','G','I','D',0},
|
|
|
|
{'M','O','D','U','L','E','_','T','Y','P','E',0},
|
|
|
|
{'V','E','R','D','A','T','E','H','I',0},
|
|
|
|
{'V','E','R','D','A','T','E','L','O',0},
|
|
|
|
{'V','E','R','F','I','L','E','O','S',0},
|
|
|
|
{'V','E','R','F','I','L','E','T','Y','P','E',0},
|
|
|
|
{'P','E','_','C','H','E','C','K','S','U','M',0},
|
|
|
|
{'P','R','E','V','O','S','M','A','J','O','R','V','E','R',0},
|
|
|
|
{'P','R','E','V','O','S','M','I','N','O','R','V','E','R',0},
|
|
|
|
{'P','R','E','V','O','S','P','L','A','T','F','O','R','M','I','D',0},
|
|
|
|
{'P','R','E','V','O','S','B','U','I','L','D','N','O',0},
|
|
|
|
{'P','R','O','B','L','E','M','S','E','V','E','R','I','T','Y',0},
|
|
|
|
{'L','A','N','G','I','D',0},
|
|
|
|
{'V','E','R','_','L','A','N','G','U','A','G','E',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'E','N','G','I','N','E',0},
|
|
|
|
{'H','T','M','L','H','E','L','P','I','D',0},
|
|
|
|
{'I','N','D','E','X','_','F','L','A','G','S',0},
|
|
|
|
{'F','L','A','G','S',0},
|
|
|
|
{'D','A','T','A','_','V','A','L','U','E','T','Y','P','E',0},
|
|
|
|
{'D','A','T','A','_','D','W','O','R','D',0},
|
|
|
|
{'L','A','Y','E','R','_','T','A','G','I','D',0},
|
|
|
|
{'M','S','I','_','T','R','A','N','S','F','O','R','M','_','T','A','G','I','D',0},
|
|
|
|
{'L','I','N','K','E','R','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'L','I','N','K','_','D','A','T','E',0},
|
|
|
|
{'U','P','T','O','_','L','I','N','K','_','D','A','T','E',0},
|
|
|
|
{'O','S','_','S','E','R','V','I','C','E','_','P','A','C','K',0},
|
|
|
|
{'F','L','A','G','_','T','A','G','I','D',0},
|
|
|
|
{'R','U','N','T','I','M','E','_','P','L','A','T','F','O','R','M',0},
|
|
|
|
{'O','S','_','S','K','U',0},
|
|
|
|
{'O','S','_','P','L','A','T','F','O','R','M',0},
|
|
|
|
{'A','P','P','_','N','A','M','E','_','R','C','_','I','D',0},
|
|
|
|
{'V','E','N','D','O','R','_','N','A','M','E','_','R','C','_','I','D',0},
|
|
|
|
{'S','U','M','M','A','R','Y','_','M','S','G','_','R','C','_','I','D',0},
|
|
|
|
{'V','I','S','T','A','_','S','K','U',0},
|
|
|
|
{'D','E','S','C','R','I','P','T','I','O','N','_','R','C','_','I','D',0},
|
|
|
|
{'P','A','R','A','M','E','T','E','R','1','_','R','C','_','I','D',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'C','O','N','T','E','X','T','_','T','A','G','I','D',0},
|
2016-04-24 17:19:09 +00:00
|
|
|
{'E','X','E','_','W','R','A','P','P','E','R',0},
|
|
|
|
{'U','R','L','_','I','D',0}
|
2016-04-24 13:05:00 +00:00
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_QWORD */
|
|
|
|
{'T','I','M','E',0},
|
|
|
|
{'B','I','N','_','F','I','L','E','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'B','I','N','_','P','R','O','D','U','C','T','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'M','O','D','T','I','M','E',0},
|
|
|
|
{'F','L','A','G','_','M','A','S','K','_','K','E','R','N','E','L',0},
|
|
|
|
{'U','P','T','O','_','B','I','N','_','P','R','O','D','U','C','T','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'D','A','T','A','_','Q','W','O','R','D',0},
|
|
|
|
{'F','L','A','G','_','M','A','S','K','_','U','S','E','R',0},
|
|
|
|
{'F','L','A','G','S','_','N','T','V','D','M','1',0},
|
|
|
|
{'F','L','A','G','S','_','N','T','V','D','M','2',0},
|
|
|
|
{'F','L','A','G','S','_','N','T','V','D','M','3',0},
|
|
|
|
{'F','L','A','G','_','M','A','S','K','_','S','H','E','L','L',0},
|
|
|
|
{'U','P','T','O','_','B','I','N','_','F','I','L','E','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'F','L','A','G','_','M','A','S','K','_','F','U','S','I','O','N',0},
|
|
|
|
{'F','L','A','G','_','P','R','O','C','E','S','S','P','A','R','A','M',0},
|
|
|
|
{'F','L','A','G','_','L','U','A',0},
|
|
|
|
{'F','L','A','G','_','I','N','S','T','A','L','L',0}
|
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_STRINGREF */
|
|
|
|
{'N','A','M','E',0},
|
|
|
|
{'D','E','S','C','R','I','P','T','I','O','N',0},
|
|
|
|
{'M','O','D','U','L','E',0},
|
|
|
|
{'A','P','I',0},
|
|
|
|
{'V','E','N','D','O','R',0},
|
|
|
|
{'A','P','P','_','N','A','M','E',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'C','O','M','M','A','N','D','_','L','I','N','E',0},
|
|
|
|
{'C','O','M','P','A','N','Y','_','N','A','M','E',0},
|
|
|
|
{'D','L','L','F','I','L','E',0},
|
|
|
|
{'W','I','L','D','C','A','R','D','_','N','A','M','E',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'P','R','O','D','U','C','T','_','N','A','M','E',0},
|
|
|
|
{'P','R','O','D','U','C','T','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'F','I','L','E','_','D','E','S','C','R','I','P','T','I','O','N',0},
|
|
|
|
{'F','I','L','E','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'O','R','I','G','I','N','A','L','_','F','I','L','E','N','A','M','E',0},
|
|
|
|
{'I','N','T','E','R','N','A','L','_','N','A','M','E',0},
|
|
|
|
{'L','E','G','A','L','_','C','O','P','Y','R','I','G','H','T',0},
|
|
|
|
{'1','6','B','I','T','_','D','E','S','C','R','I','P','T','I','O','N',0},
|
|
|
|
{'A','P','P','H','E','L','P','_','D','E','T','A','I','L','S',0},
|
|
|
|
{'L','I','N','K','_','U','R','L',0},
|
|
|
|
{'L','I','N','K','_','T','E','X','T',0},
|
|
|
|
{'A','P','P','H','E','L','P','_','T','I','T','L','E',0},
|
|
|
|
{'A','P','P','H','E','L','P','_','C','O','N','T','A','C','T',0},
|
|
|
|
{'S','X','S','_','M','A','N','I','F','E','S','T',0},
|
|
|
|
{'D','A','T','A','_','S','T','R','I','N','G',0},
|
|
|
|
{'M','S','I','_','T','R','A','N','S','F','O','R','M','_','F','I','L','E',0},
|
|
|
|
{'1','6','B','I','T','_','M','O','D','U','L','E','_','N','A','M','E',0},
|
|
|
|
{'L','A','Y','E','R','_','D','I','S','P','L','A','Y','N','A','M','E',0},
|
|
|
|
{'C','O','M','P','I','L','E','R','_','V','E','R','S','I','O','N',0},
|
|
|
|
{'A','C','T','I','O','N','_','T','Y','P','E',0},
|
2016-04-24 17:19:09 +00:00
|
|
|
{'E','X','P','O','R','T','_','N','A','M','E',0},
|
|
|
|
{'U','R','L',0}
|
2016-04-24 13:05:00 +00:00
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_LIST */
|
|
|
|
{'D','A','T','A','B','A','S','E',0},
|
|
|
|
{'L','I','B','R','A','R','Y',0},
|
|
|
|
{'I','N','E','X','C','L','U','D','E',0},
|
|
|
|
{'S','H','I','M',0},
|
|
|
|
{'P','A','T','C','H',0},
|
|
|
|
{'A','P','P',0},
|
|
|
|
{'E','X','E',0},
|
|
|
|
{'M','A','T','C','H','I','N','G','_','F','I','L','E',0},
|
|
|
|
{'S','H','I','M','_','R','E','F',0},
|
|
|
|
{'P','A','T','C','H','_','R','E','F',0},
|
|
|
|
{'L','A','Y','E','R',0},
|
|
|
|
{'F','I','L','E',0},
|
|
|
|
{'A','P','P','H','E','L','P',0},
|
|
|
|
{'L','I','N','K',0},
|
|
|
|
{'D','A','T','A',0},
|
|
|
|
{'M','S','I','_','T','R','A','N','S','F','O','R','M',0},
|
|
|
|
{'M','S','I','_','T','R','A','N','S','F','O','R','M','_','R','E','F',0},
|
|
|
|
{'M','S','I','_','P','A','C','K','A','G','E',0},
|
|
|
|
{'F','L','A','G',0},
|
|
|
|
{'M','S','I','_','C','U','S','T','O','M','_','A','C','T','I','O','N',0},
|
|
|
|
{'F','L','A','G','_','R','E','F',0},
|
|
|
|
{'A','C','T','I','O','N',0},
|
|
|
|
{'L','O','O','K','U','P',0},
|
|
|
|
{'C','O','N','T','E','X','T',0},
|
2016-04-24 17:19:09 +00:00
|
|
|
{'C','O','N','T','E','X','T','_','R','E','F',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'S','P','C',0}
|
2016-04-24 13:05:00 +00:00
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_STRING */
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0}
|
|
|
|
},
|
|
|
|
{ /* TAG_TYPE_BINARY */
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'P','A','T','C','H','_','B','I','T','S',0},
|
|
|
|
{'F','I','L','E','_','B','I','T','S',0},
|
|
|
|
{'E','X','E','_','I','D',0},
|
|
|
|
{'D','A','T','A','_','B','I','T','S',0},
|
|
|
|
{'M','S','I','_','P','A','C','K','A','G','E','_','I','D',0},
|
|
|
|
{'D','A','T','A','B','A','S','E','_','I','D',0},
|
|
|
|
{'C','O','N','T','E','X','T','_','P','L','A','T','F','O','R','M','_','I','D',0},
|
|
|
|
{'C','O','N','T','E','X','T','_','B','R','A','N','C','H','_','I','D',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'I','n','v','a','l','i','d','T','a','g',0},
|
|
|
|
{'F','I','X','_','I','D',0},
|
|
|
|
{'A','P','P','_','I','D',0}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/* sizes of tables in above array (# strings per type) */
|
|
|
|
static const WORD limits[9] = {
|
|
|
|
/* switch off TYPE_* nibble of last tag for each type */
|
|
|
|
TAG_REINSTALL_UPGRADE & 0xFF,
|
|
|
|
1,
|
|
|
|
TAG_MATCH_MODE & 0xFF,
|
2016-04-24 17:19:09 +00:00
|
|
|
TAG_URL_ID & 0xFF,
|
2016-04-24 13:05:00 +00:00
|
|
|
TAG_FLAG_INSTALL & 0xFF,
|
2016-04-24 17:19:09 +00:00
|
|
|
TAG_URL & 0xFF,
|
|
|
|
TAG_SPC & 0xFF,
|
2016-04-24 13:05:00 +00:00
|
|
|
1,
|
|
|
|
TAG_APP_ID & 0xFF
|
|
|
|
};
|
|
|
|
|
|
|
|
/* lookup tables for tags in range 0x800 + (0x1 -> 0xFF) | TYPE */
|
|
|
|
static const WCHAR table2[9][3][17] = {
|
|
|
|
{ {'I','n','v','a','l','i','d','T','a','g',0} }, /* TAG_TYPE_NULL */
|
|
|
|
{ {'I','n','v','a','l','i','d','T','a','g',0} }, /* TAG_TYPE_BYTE */
|
|
|
|
{
|
|
|
|
{'T','A','G',0}, /* TAG_TYPE_WORD */
|
|
|
|
{'I','N','D','E','X','_','T','A','G',0},
|
|
|
|
{'I','N','D','E','X','_','K','E','Y',0}
|
|
|
|
},
|
|
|
|
{ {'T','A','G','I','D',0} }, /* TAG_TYPE_DWORD */
|
|
|
|
{ {'I','n','v','a','l','i','d','T','a','g',0} }, /* TAG_TYPE_QWORD */
|
|
|
|
{ {'I','n','v','a','l','i','d','T','a','g',0} }, /* TAG_TYPE_STRINGREF */
|
|
|
|
{
|
|
|
|
{'S','T','R','I','N','G','T','A','B','L','E',0}, /* TAG_TYPE_LIST */
|
|
|
|
{'I','N','D','E','X','E','S',0},
|
|
|
|
{'I','N','D','E','X',0}
|
|
|
|
},
|
|
|
|
{ {'S','T','R','I','N','G','T','A','B','L','E','_','I','T','E','M',0}, }, /* TAG_TYPE_STRING */
|
|
|
|
{ {'I','N','D','E','X','_','B','I','T','S',0} } /* TAG_TYPE_BINARY */
|
|
|
|
};
|
|
|
|
|
|
|
|
/* sizes of tables in above array, hardcoded for simplicity */
|
|
|
|
static const WORD limits2[9] = { 0, 0, 3, 1, 0, 0, 3, 1, 1 };
|
|
|
|
|
|
|
|
static const WCHAR null[] = {'N','U','L','L',0};
|
|
|
|
static const WCHAR invalid[] = {'I','n','v','a','l','i','d','T','a','g',0};
|
|
|
|
|
|
|
|
BOOL switch_table; /* should we use table2 and limits2? */
|
|
|
|
WORD index, type_index;
|
|
|
|
|
|
|
|
/* special case: null tag */
|
|
|
|
if (tag == TAG_NULL)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
/* tags with only type mask or no type mask are invalid */
|
|
|
|
if ((tag & ~TAG_TYPE_MASK) == 0 || (tag & TAG_TYPE_MASK) == 0)
|
|
|
|
return invalid;
|
|
|
|
|
|
|
|
/* some valid tags are in range 0x800 + (0x1 -> 0xF) | TYPE */
|
|
|
|
if ((tag & 0xF00) == 0x800)
|
|
|
|
switch_table = TRUE;
|
|
|
|
else if ((tag & 0xF00) == 0)
|
|
|
|
switch_table = FALSE;
|
|
|
|
else return invalid;
|
|
|
|
|
|
|
|
/* index of table in array is type nibble */
|
|
|
|
type_index = (tag >> 12) - 1;
|
|
|
|
|
|
|
|
/* index of string in table is low byte */
|
|
|
|
index = (tag & 0xFF) - 1;
|
|
|
|
|
|
|
|
/* bound check */
|
|
|
|
if (type_index >= 9 || index >= (switch_table ? limits2[type_index] : limits[type_index]))
|
|
|
|
return invalid;
|
|
|
|
|
|
|
|
/* tag is valid */
|
|
|
|
return switch_table ? table2[type_index][index] : table[type_index][index];
|
|
|
|
}
|