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

918 lines
30 KiB
C++

#include "stdinc.h"
#include "fusioneventlog.h"
#include "search.h"
#include <stdlib.h>
#include "fusionunused.h"
#include "sxsid.h"
#include "smartptr.h"
/*
NTRAID#NTBUG9-591790-2002/03/31-JayKrell
General issues in this file
missing error check on .Win32Format (dbgprint related, under #if DBG)
//
// ISSUE:jonwis:2002-3-29: This version is smarter about rolling back if something bad
// happens, and much better about playing nice with tracing and whatnot. It should be
// put into place at some point, but it's too much of a change to just make offhand.
//
static BOOL
FusionpRegisterEventLog()
Registration of the our event logging should be moved to a setup text file.
FormatMessage with inserts is not "safe". Our code depends on our resources.
Our code picks some maximums that our resources need to stay under.
CEventLogLastError::CEventLogLastError()
and CEventLogLastError::CEventLogLastError(DWORD)
are copy pastes of each other; they should share code
*/
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
const UNICODE_STRING g_strEmptyUnicodeString = { 0, 0, L""};
extern HINSTANCE g_hInstance;
HANDLE g_hEventLog = NULL;
BOOL g_fEventLogOpenAttempted = FALSE;
// a registry key name, and appears in the EventVwr ui.
// should be localized?
// a macro is provided for easy static concatenation
#define EVENT_SOURCE L"SideBySide"
// path we put in the registry to our message file
// we might want to change this to ntdll.dll or kernel32.dll
// whatever file it is, you can't replace it while EventVwr is running, which stinks
#define MESSAGE_FILE L"%SystemRoot%\\System32\\sxs.dll"
// the non macro, string pool formed, to use for other than string concatenation
const WCHAR szEventSource[] = EVENT_SOURCE;
// same thing in another form
const static UNICODE_STRING strEventSource = RTL_CONSTANT_STRING(EVENT_SOURCE);
// machine is assumed to be the local machine
const static UNICODE_STRING strMachine = {0, 0, NULL};
// we only actually log errors, but this is far and away the most common value in the registry
// and there doesn't seem to be a downside to using it
static const DWORD dwEventTypesSupported = (EVENTLOG_ERROR_TYPE | EVENTLOG_WARNING_TYPE | EVENTLOG_INFORMATION_TYPE);
// a registry value name
static const WCHAR szTypesSupportedName[] = L"TypesSupported";
// a registry value name
static const WCHAR szEventMessageFileName[] = L"EventMessageFile";
static const WCHAR szEventMessageFileValue[] = MESSAGE_FILE;
static const HKEY hkeyEventLogRoot = HKEY_LOCAL_MACHINE;
#define EVENT_LOG_SUBKEY_PARENT L"System\\CurrentControlSet\\Services\\EventLog\\System\\"
#define EVENT_LOG_SUBKEY (EVENT_LOG_SUBKEY_PARENT EVENT_SOURCE)
const static PCUNICODE_STRING g_rgpsEmptyStrings[] =
{
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString
};
/*--------------------------------------------------------------------------
call this from DllMain
--------------------------------------------------------------------------*/
BOOL
FusionpEventLogMain(
HINSTANCE,
DWORD dwReason,
PVOID pvReserved
)
{
if ((dwReason == DLL_PROCESS_DETACH) &&
(g_hEventLog != NULL)
)
{
if (pvReserved != NULL)
{
::ElfDeregisterEventSource(g_hEventLog);
}
g_hEventLog = NULL;
}
return TRUE;
}
const static WCHAR Error_Message_is_unavailable[] = L"Error Message is unavailable\n";
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
CEventLogLastError::CEventLogLastError()
{
const DWORD dwLastError = FusionpGetLastWin32Error();
// extra string copy..
WCHAR rgchLastError[NUMBER_OF(m_rgchBuffer)];
rgchLastError[0] = 0;
C_ASSERT(sizeof(Error_Message_is_unavailable) <= sizeof(rgchLastError));
// I expect FormatMessage will truncate, which is acceptable.
const DWORD dwFlags = FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ARGUMENT_ARRAY;
if (::FormatMessageW(dwFlags, NULL, dwLastError, 0, rgchLastError, NUMBER_OF(rgchLastError), NULL) == 0 )
{
CopyMemory(rgchLastError, Error_Message_is_unavailable, sizeof(Error_Message_is_unavailable));
}
// Format will truncate, which is acceptable.
//Format(L"FusionpGetLastWin32Error()=(%ld,%ls)", nLastError, rgchLastError);
Format(L"%ls", rgchLastError);
SetLastError(dwLastError);
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
CEventLogLastError::CEventLogLastError(
DWORD dwLastError
)
{
// extra string copy..
WCHAR rgchLastError[NUMBER_OF(m_rgchBuffer)];
rgchLastError[0] = 0;
C_ASSERT(sizeof(Error_Message_is_unavailable) <= sizeof(rgchLastError));
// I expect FormatMessage will truncate, which is acceptable.
const DWORD dwFlags = FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_ARGUMENT_ARRAY;
if (::FormatMessageW(dwFlags, NULL, dwLastError, 0, rgchLastError, NUMBER_OF(rgchLastError), NULL) == 0)
{
CopyMemory(rgchLastError, Error_Message_is_unavailable, sizeof(Error_Message_is_unavailable));
}
// Format will truncate, which is acceptable.
//Format(L"FusionpGetLastWin32Error()=(%ld,%ls)", nLastError, rgchLastError);
Format(L"%ls", rgchLastError);
SetLastError(dwLastError);
}
/*--------------------------------------------------------------------------
register ourselves in the registry on demand
FUTURE Do this in setup?
HKLM\System\CurrentControlSet\Services\EventLog\System\SideBySide
EventMessageFile = %SystemRoot%\System32\Fusion.dll
TypesSupported = 7
--------------------------------------------------------------------------*/
// NTRAID#NTBUG9 - 566261 - jonwis - 2002/4/25 - We should be doing better in terms of rollback
BOOL
FusionpRegisterEventLog()
{
BOOL fSuccess = FALSE;
FN_TRACE_WIN32(fSuccess);
HKEY hkey = NULL;
BOOL fValidHkey = FALSE;
LONG lRet = ERROR_SUCCESS;
DWORD dwDisposition = 0;
WCHAR szSubKey[] = EVENT_LOG_SUBKEY;
// first see if it's there, in which case we have less to do
lRet = ::RegOpenKeyExW(
hkeyEventLogRoot,
szSubKey,
0, // reserved options
KEY_READ | FUSIONP_KEY_WOW64_64KEY,
&hkey);
if (lRet == ERROR_SUCCESS)
{
fValidHkey = TRUE;
goto Exit;
}
if (lRet != ERROR_FILE_NOT_FOUND && lRet != ERROR_PATH_NOT_FOUND)
{
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpRegisterEventLog/RegOpenKeyExW failed %ld\n", lRet);
goto Exit;
}
lRet = ::RegCreateKeyExW(
hkeyEventLogRoot,
szSubKey,
0, // reserved
NULL, // class
REG_OPTION_NON_VOLATILE,
KEY_ALL_ACCESS | FUSIONP_KEY_WOW64_64KEY,
NULL, // security
&hkey,
&dwDisposition);
if (lRet != ERROR_SUCCESS)
{
goto Exit;
}
fValidHkey = TRUE;
lRet = ::RegSetValueExW(
hkey,
szEventMessageFileName,
0, // reserved
REG_EXPAND_SZ,
reinterpret_cast<const BYTE*>(szEventMessageFileValue),
sizeof(szEventMessageFileValue));
if (lRet != ERROR_SUCCESS)
{
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpRegisterEventLog/RegSetValueExW failed %ld\n", lRet);
goto Exit;
}
lRet = ::RegSetValueExW(
hkey,
szTypesSupportedName,
0, // reserved
REG_DWORD,
reinterpret_cast<const BYTE*>(&dwEventTypesSupported),
sizeof(dwEventTypesSupported));
if (lRet != ERROR_SUCCESS)
{
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpRegisterEventLog/RegSetValueExW failed %ld\n", lRet);
goto Exit;
}
Exit:
if (fValidHkey)
{
if (lRet != ERROR_SUCCESS)
{
if (dwDisposition == REG_CREATED_NEW_KEY)
{
// rollback if there definitely wasn't anything there before
PWSTR szParentKey = szSubKey;
LONG lSubRet = ERROR_SUCCESS;
HKEY hkeyParent = reinterpret_cast<HKEY>(INVALID_HANDLE_VALUE);
ASSERT(szParentKey[NUMBER_OF(szSubKey) - NUMBER_OF(szEventSource)] == L'\\');
szParentKey[NUMBER_OF(szSubKey) - NUMBER_OF(szEventSource)] = 0;
::RegDeleteValueW(hkey, szEventMessageFileName);
::RegDeleteValueW(hkey, szTypesSupportedName);
lSubRet = ::RegOpenKeyExW(
hkeyEventLogRoot,
szParentKey,
0, // reserved options
KEY_WRITE | FUSIONP_KEY_WOW64_64KEY,
&hkeyParent);
if (lSubRet == ERROR_SUCCESS)
{
::RegDeleteKeyW(hkeyParent, szEventSource);
::RegCloseKey(hkeyParent);
}
}
}
::RegCloseKey(hkey);
fValidHkey = FALSE;
}
if (lRet != ERROR_SUCCESS)
{
::SetLastError(lRet);
}
else
fSuccess = TRUE;
return fSuccess;
}
/*--------------------------------------------------------------------------
convert the upper two bits of an event id to the small numbered analogous
parameter to ReportEvent
--------------------------------------------------------------------------*/
WORD
FusionpEventIdToEventType(
DWORD dwEventId
)
{
switch (dwEventId >> 30)
{
case STATUS_SEVERITY_SUCCESS: return EVENTLOG_SUCCESS;
case STATUS_SEVERITY_WARNING: return EVENTLOG_WARNING_TYPE;
case STATUS_SEVERITY_INFORMATIONAL: return EVENTLOG_INFORMATION_TYPE;
case STATUS_SEVERITY_ERROR: return EVENTLOG_ERROR_TYPE;
default: __assume(FALSE);
}
__assume(FALSE);
}
/*--------------------------------------------------------------------------
a Fusion event id and its corresponding Win32 lastError
the mapping is defined in Messages.x
--------------------------------------------------------------------------*/
struct EventIdErrorPair
{
DWORD dwEventId;
LONG nError;
};
/*--------------------------------------------------------------------------
the type of function used with bsearch
--------------------------------------------------------------------------*/
typedef int (__cdecl* PFNBSearchFunction)(const void*, const void*);
/*--------------------------------------------------------------------------
a function appropriate for use with bsearch
--------------------------------------------------------------------------*/
int __cdecl
CompareEventIdErrorPair(
const EventIdErrorPair* x,
const EventIdErrorPair* y
)
{
return
(x->dwEventId < y->dwEventId) ? -1
: (x->dwEventId > y->dwEventId) ? +1
: 0;
}
const static EventIdErrorPair eventIdToErrorMap[] =
{
#include "Messages.hi" // generated from .x file, like .mc
};
/*--------------------------------------------------------------------------
find the Win32 last error corresponding to this Fusion event id
--------------------------------------------------------------------------*/
DWORD
FusionpEventIdToError(
DWORD dwEventId
)
{
DWORD dwFacility = HRESULT_FACILITY(dwEventId);
if (dwFacility < 0x100)
{ // it's actually a system event id
ASSERT2_NTC(FALSE, "system event id in " __FUNCTION__);
return dwEventId;
}
static BOOL fSortVerified = FALSE;
static BOOL fSorted = FALSE;
if (!fSortVerified)
{
ULONG i;
for (i = 0 ; i != NUMBER_OF(eventIdToErrorMap) - 1; ++i)
{
if (eventIdToErrorMap[i+1].dwEventId < eventIdToErrorMap[i].dwEventId)
{
break;
}
}
if (i != NUMBER_OF(eventIdToErrorMap) - 1)
{
ASSERT2_NTC(FALSE, "eventIdToErrorMap is not sorted, reverting to linear search");
fSorted = FALSE;
}
else
{
fSorted = TRUE;
}
fSortVerified = TRUE;
}
const EventIdErrorPair* found = NULL;
const EventIdErrorPair key = { dwEventId };
unsigned numberOf = NUMBER_OF(eventIdToErrorMap);
if (fSorted)
{
found = reinterpret_cast<const EventIdErrorPair*>(
bsearch(
&key,
&eventIdToErrorMap,
numberOf,
sizeof(eventIdToErrorMap[0]),
reinterpret_cast<PFNBSearchFunction>(CompareEventIdErrorPair)));
}
else
{
found = reinterpret_cast<const EventIdErrorPair*>(
_lfind(
&key,
&eventIdToErrorMap,
&numberOf,
sizeof(eventIdToErrorMap[0]),
reinterpret_cast<PFNBSearchFunction>(CompareEventIdErrorPair)));
}
if (found == NULL)
{
#if DBG
CANSIStringBuffer msg;
msg.Win32Format("Event id %lx not found in eventIdToErrorMap", static_cast<ULONG>(dwEventId));
ASSERT2_NTC(found != NULL, const_cast<PSTR>(static_cast<PCSTR>(msg)));
#endif
return ::FusionpGetLastWin32Error();
}
if (found->nError != 0)
{
return found->nError;
}
return ::FusionpGetLastWin32Error();
}
/*--------------------------------------------------------------------------
open the event log on demand
confusingly, this is called "registering" an event source
--------------------------------------------------------------------------*/
BOOL
FusionpOpenEventLog()
{
HANDLE hEventLog;
NTSTATUS status;
if (g_fEventLogOpenAttempted)
{
goto Exit;
}
if (!FusionpRegisterEventLog())
{
goto Exit;
}
status = ::ElfRegisterEventSourceW(
const_cast<PUNICODE_STRING>(&strMachine),
const_cast<PUNICODE_STRING>(&strEventSource),
&hEventLog);
if (!NT_SUCCESS(status))
{
if (status != RPC_NT_SERVER_UNAVAILABLE)
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpOpenEventLog/ElfRegisterEventSourceW failed %lx\n", static_cast<ULONG>(status));
goto Exit;
}
if (InterlockedCompareExchangePointer(
&g_hEventLog,
hEventLog, // exchange value
NULL // compare value
) != NULL) // value returned is value that was there before we called
{
::ElfDeregisterEventSource(hEventLog);
goto Exit;
}
g_hEventLog = hEventLog;
Exit:
g_fEventLogOpenAttempted = TRUE;
return (g_hEventLog != NULL);
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogError(
DWORD dwEventId,
const UNICODE_STRING& s1,
const UNICODE_STRING& s2,
const UNICODE_STRING& s3,
const UNICODE_STRING& s4
)
{
PCUNICODE_STRING rgps[] = { &s1, &s2, &s3, &s4 };
return ::FusionpLogError(dwEventId, NUMBER_OF(rgps), rgps);
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogErrorToDebugger(
DWORD dwEventId,
const UNICODE_STRING& s1,
const UNICODE_STRING& s2,
const UNICODE_STRING& s3,
const UNICODE_STRING& s4
)
{
PCUNICODE_STRING rgps[] = { &s1, &s2, &s3, &s4 };
return FusionpLogErrorToDebugger(dwEventId, NUMBER_OF(rgps), rgps);
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogErrorToEventLog(
DWORD dwEventId,
const UNICODE_STRING& s1,
const UNICODE_STRING& s2,
const UNICODE_STRING& s3,
const UNICODE_STRING& s4
)
{
PCUNICODE_STRING rgps[] = { &s1, &s2, &s3, &s4 };
return FusionpLogErrorToEventLog(dwEventId, NUMBER_OF(rgps), rgps);
}
void
LocalFreeWcharPointer(
WCHAR * p
)
{
LocalFree(p);
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogErrorToDebugger(
DWORD dwEventId,
ULONG nStrings,
const PCUNICODE_STRING * rgps
)
{
const LONG lastError = FusionpEventIdToError(dwEventId);
const HRESULT hr = HRESULT_FROM_WIN32(lastError);
PCUNICODE_STRING rgpsManyStrings[] =
{
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString,
&g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString, &g_strEmptyUnicodeString
};
if (nStrings < NUMBER_OF(rgpsManyStrings))
{
CopyMemory(rgpsManyStrings, rgps, nStrings * sizeof(rgps[0]));
rgps = rgpsManyStrings;
}
DWORD dwFormatMessageFlags = 0;
CSmartPtrWithNamedDestructor<WCHAR, LocalFreeWcharPointer> pszBuffer1;
CSmartPtrWithNamedDestructor<WCHAR, LocalFreeWcharPointer> pszBuffer2;
DWORD dw = 0;
static const WCHAR rgchParseContextPrefix[] = PARSE_CONTEXT_PREFIX;
const SIZE_T cchParseContextPrefixLength = RTL_NUMBER_OF(rgchParseContextPrefix) - 1;
PCWSTR pszSkipFirstLine = NULL;
// load the string from the message table,
// substituting %n with %n!wZ!
// the Rtl limit here is 200, but we don't expect very many in our messages
const static PCWSTR percentZw[] = { L"%1!wZ!", L"%2!wZ!", L"%3!wZ!", L"%4!wZ!", L"%5!wZ!",
L"%6!wZ!", L"%7!wZ!", L"%8!wZ!", L"%9!wZ!", L"%10!wZ!",
L"%11!wZ!", L"%12!wZ!", L"%13!wZ!", L"%14!wZ!", L"%15!wZ!"
L"%16!wZ!", L"%17!wZ!", L"%18!wZ!", L"%19!wZ!", L"%20!wZ!"
};
dwFormatMessageFlags = FORMAT_MESSAGE_ARGUMENT_ARRAY | FORMAT_MESSAGE_FROM_HMODULE;
dwFormatMessageFlags |= FORMAT_MESSAGE_ALLOCATE_BUFFER;
dw = FormatMessageW(
dwFormatMessageFlags,
g_hInstance,
dwEventId,
0, // langid
reinterpret_cast<PWSTR>(static_cast<PWSTR*>(&pszBuffer1)),
300, // minimum allocation
const_cast<va_list*>(reinterpret_cast<const va_list*>(&percentZw)));
if (dw == 0)
{
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpLogError/FormatMessageW failed %ld\n", static_cast<long>(FusionpGetLastWin32Error()));
goto Exit;
}
// do the substitutions
dwFormatMessageFlags = FORMAT_MESSAGE_ARGUMENT_ARRAY | FORMAT_MESSAGE_FROM_STRING;
dwFormatMessageFlags |= FORMAT_MESSAGE_ALLOCATE_BUFFER;
dw = FormatMessageW(
dwFormatMessageFlags,
pszBuffer1,
0, // message id
0, // langid
reinterpret_cast<PWSTR>(static_cast<PWSTR*>(&pszBuffer2)),
1 + StringLength(pszBuffer1), // minimum allocation
reinterpret_cast<va_list*>(const_cast<PUNICODE_STRING*>(rgps)));
if (dw == 0)
{
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpLogError/FormatMessageW failed %ld\n", static_cast<long>(FusionpGetLastWin32Error()));
goto Exit;
}
//
// acceptable hack
//
// The first line of parse errors is a verbose context, see Messages.x.
// For DbgPrint we want instead file(line): on the same line instead.
// We make that transformation here.
//
pszSkipFirstLine = wcschr(pszBuffer2, '\n');
BOOL fAreWeInOSSetupMode = FALSE;
FusionpAreWeInOSSetupMode(&fAreWeInOSSetupMode);
if (
pszSkipFirstLine != NULL
&& nStrings >= PARSE_CONTEXT_INSERTS_END
&& StringLength(pszBuffer2) >= cchParseContextPrefixLength
&& FusionpEqualStringsI(pszBuffer2, cchParseContextPrefixLength, rgchParseContextPrefix, cchParseContextPrefixLength)
)
{
// we might fiddle with the form of the newline, so skip whatever is there
while (wcschr(L"\r\n", *pszSkipFirstLine) != NULL)
pszSkipFirstLine += 1;
::FusionpDbgPrintEx(
FUSION_DBG_LEVEL_ERROR | ( fAreWeInOSSetupMode ? FUSION_DBG_LEVEL_SETUPLOG : 0),
"%wZ(%wZ): %S",
rgps[PARSE_CONTEXT_FILE - 1],
rgps[PARSE_CONTEXT_LINE - 1],
pszSkipFirstLine);
}
else
{
// just print it verbatim
FusionpDbgPrintEx(
FUSION_DBG_LEVEL_ERROR | ( fAreWeInOSSetupMode ? FUSION_DBG_LEVEL_SETUPLOG : 0),
"SXS.DLL: %S",
pszBuffer2);
}
Exit:
::SetLastError(lastError);
return hr;
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogErrorToEventLog(
DWORD dwEventId,
ULONG nStrings,
const PCUNICODE_STRING * rgps
)
{
const LONG lastError = FusionpEventIdToError(dwEventId);
const HRESULT hr = HRESULT_FROM_WIN32(lastError);
const WORD wType = FusionpEventIdToEventType(dwEventId);
// The use of the lower bits of the hresult facility as the event log
// facility is my own invention, but it seems a good one.
// ReportEvent has too many parameters, those three integers instead of one.
const WORD wCategory = 0/*static_cast<WORD>(HRESULT_FACILITY(dwEventId) & 0xff)*/;
const DWORD dwDataSize = 0;
void const* const pvRawData = NULL;
const PSID pSecurityIdentifier = NULL;
if (!::FusionpOpenEventLog())
{
goto Exit;
}
else
{
NTSTATUS status;
status = ::ElfReportEventW(
g_hEventLog,
wType,
wCategory,
dwEventId,
pSecurityIdentifier,
static_cast<USHORT>(nStrings),
dwDataSize,
const_cast<PUNICODE_STRING*>(rgps),
const_cast<void*>(pvRawData),
0,
NULL,
NULL);
//
// the excluded error status is because it is in the early setup time.
//
if (!NT_SUCCESS(status))
{
if (status != RPC_NT_SERVER_UNAVAILABLE)
::FusionpDbgPrintEx(FUSION_DBG_LEVEL_ERROR, "SXS.DLL: FusionpLogError/ElfReportEventW failed %lx\n", static_cast<ULONG>(status));
goto Exit;
}
}
Exit:
::SetLastError(lastError);
return hr;
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
HRESULT
FusionpLogError(
DWORD dwEventId,
ULONG nStrings,
const PCUNICODE_STRING * rgps
)
{
const HRESULT hr = FusionpLogErrorToEventLog(dwEventId, nStrings, rgps);
const HRESULT hr2 = FusionpLogErrorToDebugger(dwEventId, nStrings, rgps);
RETAIL_UNUSED(hr);
RETAIL_UNUSED(hr2);
ASSERT_NTC(hr == hr2);
return hr;
}
HRESULT
FusionpLogParseError(
PCWSTR FilePath,
SIZE_T FilePathCch,
ULONG LineNumber,
DWORD dwLastParseError,
PCUNICODE_STRING p1,
PCUNICODE_STRING p2,
PCUNICODE_STRING p3,
PCUNICODE_STRING p4,
PCUNICODE_STRING p5,
PCUNICODE_STRING p6,
PCUNICODE_STRING p7,
PCUNICODE_STRING p8,
PCUNICODE_STRING p9,
PCUNICODE_STRING p10,
PCUNICODE_STRING p11,
PCUNICODE_STRING p12,
PCUNICODE_STRING p13,
PCUNICODE_STRING p14,
PCUNICODE_STRING p15,
PCUNICODE_STRING p16,
PCUNICODE_STRING p17,
PCUNICODE_STRING p18,
PCUNICODE_STRING p19,
PCUNICODE_STRING p20
)
{
const DWORD lastError = ::FusionpEventIdToError(dwLastParseError);
const HRESULT hr = HRESULT_FROM_WIN32(lastError);
::FusionpDbgPrintEx(
FUSION_DBG_LEVEL_INFO,
"SXS.DLL: %s() entered\n", __FUNCTION__);
//
// FormatMessage (actually sprintf) AVs on NULL UNICODE_STRING*
// and/or when we don't pass enough of them;
// we can't tell it how many strings we are passing,
// and it isn't easy to tell how many it needs,
// so we load it up with a bunch of extra non NULL ones.
// Besides that, we have holes to fill.
//
static const UNICODE_STRING s_strEmptyUnicodeString = { 0, 0, L""};
static const PCUNICODE_STRING s_rgpsEmptyStrings[] =
{
&s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString,
&s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString,
&s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString,
&s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString,
&s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString, &s_strEmptyUnicodeString
};
PCUNICODE_STRING rgpsAll[NUMBER_OF(s_rgpsEmptyStrings)];
::memcpy(rgpsAll, s_rgpsEmptyStrings, sizeof(rgpsAll));
#define HANDLE_STRING(_n) do { if (p ## _n != NULL) rgpsAll[_n - 1] = p ## _n; } while (0)
HANDLE_STRING(1);
HANDLE_STRING(2);
HANDLE_STRING(3);
HANDLE_STRING(4);
HANDLE_STRING(5);
HANDLE_STRING(6);
HANDLE_STRING(7);
HANDLE_STRING(8);
HANDLE_STRING(9);
HANDLE_STRING(10);
HANDLE_STRING(11);
HANDLE_STRING(12);
HANDLE_STRING(13);
HANDLE_STRING(14);
HANDLE_STRING(15);
HANDLE_STRING(16);
HANDLE_STRING(17);
HANDLE_STRING(18);
HANDLE_STRING(19);
HANDLE_STRING(20);
#undef HANDLE_STRING
//
// form up some "context" UNICODE_STRINGs and put them in the array of pointers
// the first two are the ones that we always use, even for DbgPrint
//
CEventLogString file(FilePath, FilePathCch);
CEventLogInteger lineNumber(LineNumber);
rgpsAll[PARSE_CONTEXT_FILE - 1] = &file;
rgpsAll[PARSE_CONTEXT_LINE - 1] = &lineNumber;
::FusionpLogErrorToEventLog(
dwLastParseError,
NUMBER_OF(rgpsAll),
rgpsAll);
// we should tell this function that it was a parse error and to do
// the context munging, but it detects it itself imperfectly
::FusionpLogErrorToDebugger(dwLastParseError, NUMBER_OF(rgpsAll), rgpsAll);
::FusionpDbgPrintEx(
FUSION_DBG_LEVEL_INFO,
"SXS.DLL: %s():%#lx exited\n", __FUNCTION__, hr);
::SetLastError(lastError);
return hr;
}
/*--------------------------------------------------------------------------
--------------------------------------------------------------------------*/
VOID
FusionpLogRequiredAttributeMissingParseError(
PCWSTR SourceFilePath,
SIZE_T SourceFileCch,
ULONG LineNumber,
PCWSTR ElementName,
SIZE_T ElementNameCch,
PCWSTR AttributeName,
SIZE_T AttributeNameCch
)
{
::FusionpLogParseError(
SourceFilePath,
SourceFileCch,
LineNumber,
MSG_SXS_XML_REQUIRED_ATTRIBUTE_MISSING,
CEventLogString(ElementName, ElementNameCch),
CEventLogString(AttributeName, AttributeNameCch));
}
VOID
FusionpLogInvalidAttributeValueParseError(
PCWSTR SourceFilePath,
SIZE_T SourceFileCch,
ULONG LineNumber,
PCWSTR ElementName,
SIZE_T ElementNameCch,
PCWSTR AttributeName,
SIZE_T AttributeNameCch
)
{
::FusionpLogParseError(
SourceFilePath,
SourceFileCch,
LineNumber,
MSG_SXS_XML_INVALID_ATTRIBUTE_VALUE,
CEventLogString(ElementName, ElementNameCch),
CEventLogString(AttributeName, AttributeNameCch));
}
VOID
FusionpLogInvalidAttributeValueParseError(
PCWSTR SourceFilePath,
SIZE_T SourceFileCch,
ULONG LineNumber,
PCWSTR ElementName,
SIZE_T ElementNameCch,
const SXS_ASSEMBLY_IDENTITY_ATTRIBUTE_REFERENCE &rAttribute
)
{
::FusionpLogInvalidAttributeValueParseError(
SourceFilePath,
SourceFileCch,
LineNumber,
ElementName,
ElementNameCch,
rAttribute.Name,
rAttribute.NameCch);
}
VOID
FusionpLogAttributeNotAllowedParseError(
PCWSTR SourceFilePath,
SIZE_T SourceFileCch,
ULONG LineNumber,
PCWSTR ElementName,
SIZE_T ElementNameCch,
PCWSTR AttributeName,
SIZE_T AttributeNameCch
)
{
::FusionpLogParseError(
SourceFilePath,
SourceFileCch,
LineNumber,
MSG_SXS_XML_ATTRIBUTE_NOT_ALLOWED,
CEventLogString(ElementName, ElementNameCch),
CEventLogString(AttributeName, AttributeNameCch));
}
VOID
FusionpLogWin32ErrorToEventLog()
{
DWORD dwLastError = ::FusionpGetLastWin32Error();
if (dwLastError == 0 )
return;
FusionpLogError(MSG_SXS_WIN32_ERROR_MSG, CEventLogLastError(dwLastError));
}