Source code of Windows XP (NT5)
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.

2550 lines
63 KiB

  1. /*++
  2. 1998 Seagate Software, Inc. All rights reserved
  3. Module Name:
  4. WsbTrace.cpp
  5. Abstract:
  6. These functions are used to provide an ability to trace the flow
  7. of the application for debugging purposes.
  8. Author:
  9. Chuck Bardeen [cbardeen] 29-Oct-1996
  10. Revision History:
  11. Brian Dodd [brian] 09-May-1996 - Added event logging
  12. --*/
  13. #include "stdafx.h"
  14. #include "stdio.h"
  15. #undef WsbThrow
  16. #define WsbThrow(hr) throw(hr)
  17. #define WSB_INDENT_STRING OLESTR(" ")
  18. #define WSB_APP_EVENT_LOG OLESTR("\\System32\\config\\AppEvent.evt")
  19. #define WSB_APP_EVENT_LOG_BKUP OLESTR("\\System32\\config\\AppEvent.bkp")
  20. #define WSB_APP_EVENT_LOG_NAME OLESTR("\\AppEvent.evt")
  21. #define WSB_SYS_EVENT_LOG OLESTR("\\System32\\config\\SysEvent.evt")
  22. #define WSB_SYS_EVENT_LOG_BKUP OLESTR("\\System32\\config\\SysEvent.bkp")
  23. #define WSB_SYS_EVENT_LOG_NAME OLESTR("\\SysEvent.evt")
  24. #define WSB_RS_TRACE_FILES OLESTR("Trace\\*.*")
  25. #define WSB_RS_TRACE_PATH OLESTR("Trace\\")
  26. #define BOGUS_TLS_INDEX 0xFFFFFFFF
  27. // Per-thread data:
  28. typedef struct {
  29. ULONG TraceOffCount; // Trace only if this is zero
  30. LONG IndentLevel;
  31. char *LogModule;
  32. DWORD LogModuleLine;
  33. DWORD LogNTBuild;
  34. DWORD LogRSBuild;
  35. } THREAD_DATA;
  36. static DWORD TlsIndex = BOGUS_TLS_INDEX; // Per-thread data index
  37. // The globals that control the tracing
  38. LONGLONG g_WsbTraceModules = WSB_TRACE_BIT_NONE;
  39. IWsbTrace *g_pWsbTrace = 0;
  40. BOOL g_WsbTraceEntryExit = TRUE;
  41. // The globals that control the event logging and printing
  42. WORD g_WsbLogLevel = WSB_LOG_LEVEL_DEFAULT;
  43. BOOL g_WsbLogSnapShotOn = FALSE;
  44. WORD g_WsbLogSnapShotLevel = 0;
  45. OLECHAR g_pWsbLogSnapShotPath[250];
  46. BOOL g_WsbLogSnapShotResetTrace = FALSE;
  47. WORD g_WsbPrintLevel = WSB_LOG_LEVEL_DEFAULT;
  48. //
  49. // WsbTraceCount is a running count of the trace output count: normally we
  50. // use the shared count among the processes, but if we can't get access to
  51. // the shared var., we use this
  52. //
  53. LONG g_WsbTraceCount = 0;
  54. // Helper function
  55. static HRESULT OutputTraceString(ULONG indentLevel, OLECHAR* introString,
  56. OLECHAR* format, va_list vaList);
  57. static HRESULT GetThreadDataPointer(THREAD_DATA** ppTD);
  58. static void SnapShotTraceAndEvent( SYSTEMTIME stime );
  59. void
  60. WsbTraceInit(
  61. void
  62. )
  63. /*++
  64. Routine Description:
  65. Initialize this trace module
  66. Arguments:
  67. None.
  68. Return Value:
  69. None.
  70. --*/
  71. {
  72. // Get an index for the thread local storage
  73. TlsIndex = TlsAlloc();
  74. }
  75. void
  76. WsbTraceCleanupThread(
  77. void
  78. )
  79. /*++
  80. Routine Description:
  81. Cleanup information for this thread (which is going away)
  82. Arguments:
  83. None.
  84. Return Value:
  85. None.
  86. --*/
  87. {
  88. THREAD_DATA* pThreadData = NULL;
  89. if (BOGUS_TLS_INDEX != TlsIndex) {
  90. pThreadData = static_cast<THREAD_DATA*>(TlsGetValue(TlsIndex));
  91. if (pThreadData) {
  92. WsbFree(pThreadData);
  93. TlsSetValue(TlsIndex, NULL);
  94. }
  95. }
  96. }
  97. void
  98. WsbTraceEnter(
  99. OLECHAR* methodName,
  100. OLECHAR* argString,
  101. ...
  102. )
  103. /*++
  104. Routine Description:
  105. This routine prints out trace information indicating that the
  106. method specified has been entered, and the values of its arguements
  107. (if supplied).
  108. Arguments:
  109. methodName - The name of the method that was entered.
  110. argString - A printf style string indicating the number of
  111. arguments and how they should be formatted.
  112. Return Value:
  113. None.
  114. --*/
  115. {
  116. HRESULT hr = S_OK;
  117. OLECHAR tmpString[WSB_TRACE_BUFF_SIZE];
  118. va_list vaList;
  119. try {
  120. THREAD_DATA* pThreadData = NULL;
  121. WsbAffirmHr(GetThreadDataPointer(&pThreadData));
  122. // Make sure we are supposed to trace
  123. WsbAffirm( 0 != g_pWsbTrace, S_OK);
  124. WsbAffirm(0 == pThreadData->TraceOffCount, S_OK);
  125. // Identify the function.
  126. swprintf(tmpString, OLESTR("Enter <%ls> : "), methodName);
  127. // Format & print out
  128. va_start(vaList, argString);
  129. WsbAffirmHr(OutputTraceString(pThreadData->IndentLevel, tmpString,
  130. argString, vaList));
  131. va_end(vaList);
  132. // Increment the indentation level
  133. pThreadData->IndentLevel++;
  134. } WsbCatch (hr);
  135. }
  136. void
  137. WsbTraceExit(
  138. OLECHAR* methodName,
  139. OLECHAR* argString,
  140. ...
  141. )
  142. /*++
  143. Routine Description:
  144. This routine prints out trace information indicating that the
  145. method specified has been exitted, and the values it is returning
  146. (if supplied).
  147. Arguments:
  148. methodName - The name of the method that was exitted.
  149. argString - A printf style string indicating the number of
  150. arguments and how they should be formatted.
  151. Return Value:
  152. None.
  153. --*/
  154. {
  155. HRESULT hr = S_OK;
  156. OLECHAR tmpString[WSB_TRACE_BUFF_SIZE];
  157. va_list vaList;
  158. try {
  159. THREAD_DATA* pThreadData = NULL;
  160. WsbAffirmHr(GetThreadDataPointer(&pThreadData));
  161. // Make sure we are supposed to trace
  162. WsbAffirm( 0 != g_pWsbTrace, S_OK);
  163. WsbAffirm(0 == pThreadData->TraceOffCount, S_OK);
  164. // Decrement the indentation level.
  165. if (pThreadData->IndentLevel > 0) {
  166. pThreadData->IndentLevel--;
  167. } else {
  168. g_pWsbTrace->Print(OLESTR("WARNING: Badly matched TraceIn/TraceOut\r\n"));
  169. }
  170. // Identify the function.
  171. swprintf(tmpString, OLESTR("Exit <%ls> : "), methodName);
  172. // Format & print out
  173. va_start(vaList, argString);
  174. WsbAffirmHr(OutputTraceString(pThreadData->IndentLevel, tmpString,
  175. argString, vaList));
  176. va_end(vaList);
  177. } WsbCatch( hr );
  178. }
  179. void
  180. WsbTracef(
  181. OLECHAR* argString,
  182. ...
  183. )
  184. /*++
  185. Routine Description:
  186. This routine prints out trace information from a printf style string.
  187. A carriage return should be add to the format string if desired.
  188. Arguments:
  189. argString - A printf style string indicating the number of
  190. arguments and how they should be formatted.
  191. Return Value:
  192. None.
  193. --*/
  194. {
  195. HRESULT hr = S_OK;
  196. va_list vaList;
  197. try {
  198. THREAD_DATA* pThreadData = NULL;
  199. WsbAffirmHr(GetThreadDataPointer(&pThreadData));
  200. // Make sure we are supposed to trace
  201. WsbAffirm( 0 != g_pWsbTrace, S_OK);
  202. WsbAffirm(0 == pThreadData->TraceOffCount, S_OK);
  203. // Format & print out
  204. va_start(vaList, argString);
  205. WsbAffirmHr(OutputTraceString(pThreadData->IndentLevel, NULL,
  206. argString, vaList));
  207. va_end(vaList);
  208. } WsbCatch (hr);
  209. }
  210. void
  211. WsbSetEventInfo(
  212. char *fileName,
  213. DWORD lineNo,
  214. DWORD ntBuild,
  215. DWORD rsBuild
  216. )
  217. /*++
  218. Routine Description:
  219. This routine sets information used in logging events.
  220. Arguments:
  221. fileName - The name of the module that logged the event.
  222. lineNo - The source line number of the statement that logged the event
  223. ntBuild - The NT Build version
  224. rsBuild - The RS Build version
  225. Return Value:
  226. None.
  227. Notes:
  228. ntBuild, and rsBuild are passed in with each call to get the build version for
  229. the modules actually logging the event.
  230. --*/
  231. {
  232. THREAD_DATA* pThreadData = NULL;
  233. if (S_OK == GetThreadDataPointer(&pThreadData)) {
  234. pThreadData->LogModule = fileName;
  235. pThreadData->LogModuleLine = lineNo;
  236. pThreadData->LogNTBuild = ntBuild;
  237. pThreadData->LogRSBuild = rsBuild;
  238. }
  239. }
  240. void
  241. WsbTraceAndLogEvent(
  242. DWORD eventId,
  243. DWORD dataSize,
  244. LPVOID data,
  245. ...
  246. )
  247. /*++
  248. Routine Description:
  249. This routine writes a message into the system event log. The message
  250. is also written to the application trace file.
  251. Arguments:
  252. eventId - The message Id to log.
  253. dataSize - Size of arbitrary data.
  254. data - Arbitrary data buffer to display with the message.
  255. Inserts - Message inserts that are merged with the message description specified by
  256. eventId. The number of inserts must match the number specified by the
  257. message description. The last insert must be NULL to indicate the
  258. end of the insert list.
  259. Return Value:
  260. None.
  261. --*/
  262. {
  263. HRESULT hr = S_OK;
  264. try {
  265. va_list vaList;
  266. va_start(vaList, data);
  267. WsbTraceAndLogEventV( eventId, dataSize, data, &vaList );
  268. va_end(vaList);
  269. }
  270. WsbCatch( hr );
  271. }
  272. void
  273. WsbTraceAndLogEventV(
  274. DWORD eventId,
  275. DWORD dataSize,
  276. LPVOID data,
  277. va_list * inserts
  278. )
  279. /*++
  280. Routine Description:
  281. This routine writes a message into the system event log. The message
  282. is also written to the application trace file. The file name and line number is appended
  283. to the log data, if any.
  284. Arguments:
  285. eventId - The message Id to log.
  286. dataSize - Size of arbitrary data.
  287. data - Arbitrary data buffer to display with the message.
  288. inserts - An array of message inserts that are merged with the message description
  289. specified by eventId. The number of inserts must match the number
  290. specified by the message description. The last insert must be NULL,
  291. to indicate the end of the insert list.
  292. Return Value:
  293. None.
  294. --*/
  295. {
  296. HRESULT hr = S_OK;
  297. char *newData = NULL, *fileName;
  298. DWORD newDataSize=0;
  299. OLECHAR ** logString=0;
  300. WORD count=0;
  301. SYSTEMTIME stime;
  302. try {
  303. WsbAssertPointer( inserts );
  304. WORD logType;
  305. const OLECHAR * facilityName = 0;
  306. WORD category = 0;
  307. va_list vaList;
  308. BOOL bLog;
  309. BOOL bSnapShot;
  310. THREAD_DATA* pThreadData = NULL;
  311. // Get space for the passed in data plus the file and line number. If we fail to allocate
  312. // memory for this we just log the data they passed in (without file and line)
  313. GetThreadDataPointer(&pThreadData);
  314. if (pThreadData) {
  315. fileName = strrchr(pThreadData->LogModule, '\\');
  316. } else {
  317. fileName = NULL;
  318. }
  319. if (fileName) {
  320. fileName++; // Point at just the source file name (no path)
  321. int len = strlen(fileName);
  322. newData = (char *) malloc(dataSize + len + 128);
  323. if (newData) {
  324. if (data) {
  325. memcpy(newData, data, dataSize);
  326. }
  327. // Align the record data on even 8 byte boundary for viewing
  328. len = (len>8) ? 16 : 8;
  329. sprintf(&newData[dataSize], "%-*.*s@%7luNt%6luRs%6.6ls", len,
  330. len, fileName, pThreadData->LogModuleLine, pThreadData->LogNTBuild,
  331. RsBuildVersionAsString(pThreadData->LogRSBuild) );
  332. newDataSize = dataSize + strlen(&newData[dataSize]);
  333. }
  334. }
  335. //
  336. // Determine type of event
  337. //
  338. switch ( eventId & 0xc0000000 ) {
  339. case ERROR_SEVERITY_INFORMATIONAL:
  340. logType = EVENTLOG_INFORMATION_TYPE;
  341. bLog = (g_WsbLogLevel >= WSB_LOG_LEVEL_INFORMATION) ? TRUE : FALSE;
  342. bSnapShot = (g_WsbLogSnapShotLevel >= WSB_LOG_LEVEL_INFORMATION) ? TRUE : FALSE;
  343. break;
  344. case ERROR_SEVERITY_WARNING:
  345. logType = EVENTLOG_WARNING_TYPE;
  346. bLog = (g_WsbLogLevel >= WSB_LOG_LEVEL_WARNING) ? TRUE : FALSE;
  347. bSnapShot = (g_WsbLogSnapShotLevel >= WSB_LOG_LEVEL_WARNING) ? TRUE : FALSE;
  348. break;
  349. case ERROR_SEVERITY_ERROR:
  350. logType = EVENTLOG_ERROR_TYPE;
  351. bLog = (g_WsbLogLevel >= WSB_LOG_LEVEL_ERROR) ? TRUE : FALSE;
  352. bSnapShot = (g_WsbLogSnapShotLevel >= WSB_LOG_LEVEL_ERROR) ? TRUE : FALSE;
  353. break;
  354. default:
  355. logType = EVENTLOG_INFORMATION_TYPE;
  356. bLog = (g_WsbLogLevel >= WSB_LOG_LEVEL_COMMENT) ? TRUE : FALSE;
  357. bSnapShot = (g_WsbLogSnapShotLevel >= WSB_LOG_LEVEL_COMMENT) ? TRUE : FALSE;
  358. break;
  359. }
  360. WsbAffirm ( bLog, S_OK );
  361. WsbTracef(OLESTR("\r\n"));
  362. WsbTracef(OLESTR("!!!!! EVENT !!!!! - File: %hs @ Line: %d (%lu-%ls)\r\n"),
  363. (pThreadData ? pThreadData->LogModule : ""),
  364. (pThreadData ? pThreadData->LogModuleLine : 0),
  365. (pThreadData ? pThreadData->LogNTBuild : 0),
  366. RsBuildVersionAsString((pThreadData ? pThreadData->LogRSBuild : 0)) );
  367. //
  368. // Determine source facility and category of message
  369. //
  370. switch ( HRESULT_FACILITY( eventId ) ) {
  371. case WSB_FACILITY_PLATFORM:
  372. facilityName = WSB_FACILITY_PLATFORM_NAME;
  373. category = WSB_CATEGORY_PLATFORM;
  374. break;
  375. case WSB_FACILITY_RMS:
  376. facilityName = WSB_FACILITY_PLATFORM_NAME;
  377. category = WSB_CATEGORY_RMS;
  378. break;
  379. case WSB_FACILITY_HSMENG:
  380. facilityName = WSB_FACILITY_PLATFORM_NAME;
  381. category = WSB_CATEGORY_HSMENG;
  382. break;
  383. case WSB_FACILITY_JOB:
  384. facilityName = WSB_FACILITY_PLATFORM_NAME;
  385. category = WSB_CATEGORY_JOB;
  386. break;
  387. case WSB_FACILITY_HSMTSKMGR:
  388. facilityName = WSB_FACILITY_PLATFORM_NAME;
  389. category = WSB_CATEGORY_HSMTSKMGR;
  390. break;
  391. case WSB_FACILITY_FSA:
  392. facilityName = WSB_FACILITY_PLATFORM_NAME;
  393. category = WSB_CATEGORY_FSA;
  394. break;
  395. case WSB_FACILITY_GUI:
  396. facilityName = WSB_FACILITY_PLATFORM_NAME;
  397. category = WSB_CATEGORY_GUI;
  398. break;
  399. case WSB_FACILITY_MOVER:
  400. facilityName = WSB_FACILITY_PLATFORM_NAME;
  401. category = WSB_CATEGORY_MOVER;
  402. break;
  403. case WSB_FACILITY_LAUNCH:
  404. facilityName = WSB_FACILITY_PLATFORM_NAME;
  405. category = WSB_CATEGORY_LAUNCH;
  406. break;
  407. case WSB_FACILITY_USERLINK:
  408. facilityName = WSB_FACILITY_PLATFORM_NAME;
  409. category = WSB_CATEGORY_USERLINK;
  410. break;
  411. case WSB_FACILITY_TEST:
  412. facilityName = WSB_FACILITY_TEST_NAME;
  413. category = WSB_CATEGORY_TEST;
  414. break;
  415. case HRESULT_FACILITY(FACILITY_NT_BIT):
  416. facilityName = WSB_FACILITY_NTDLL_NAME;
  417. eventId &= ~FACILITY_NT_BIT;
  418. break;
  419. default:
  420. facilityName = WSB_FACILITY_NTDLL_NAME;
  421. break;
  422. }
  423. //
  424. // Trace the message
  425. //
  426. if ( g_pWsbTrace ) {
  427. if ( facilityName ) {
  428. OLECHAR * messageText = 0;
  429. // NOTE: Positional parameters in the inserts are not processed. These
  430. // are done by ReportEvent() only.
  431. vaList = *inserts;
  432. HMODULE hModule;
  433. hModule = LoadLibraryEx( facilityName, NULL, LOAD_LIBRARY_AS_DATAFILE );
  434. if (hModule) {
  435. FormatMessage( FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_ALLOCATE_BUFFER,
  436. hModule,
  437. eventId,
  438. MAKELANGID ( LANG_NEUTRAL, SUBLANG_DEFAULT ),
  439. (LPTSTR) &messageText,
  440. 0,
  441. &vaList );
  442. if ( messageText ) {
  443. WsbTracef( OLESTR("%ls"), messageText ); // Format messages come with \n
  444. LocalFree( messageText );
  445. } else {
  446. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Message <0x%08lx> could not be translated.\r\n"), eventId );
  447. }
  448. FreeLibrary(hModule);
  449. } else {
  450. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Could not load facility name DLL %ls. \r\n"), facilityName);
  451. }
  452. } else {
  453. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Message File for <0x%08lx> could not be found.\r\n"), eventId );
  454. }
  455. if ( data && dataSize > 0 )
  456. WsbTraceBufferAsBytes( dataSize, data );
  457. }
  458. // Prepare arguments for ReportEvent
  459. // First count the number of arguments
  460. vaList = *inserts;
  461. for( count = 0; (va_arg( vaList, OLECHAR *)) != NULL; count++ );
  462. if ( count ) {
  463. OLECHAR* tmpArg;
  464. // Allocate a array to hold the string arguments.
  465. //
  466. // IMPORTANT NOTE: Don't try anything fancy here. va_list is different
  467. // on various platforms. We'll need to build the string
  468. // argument required by ReportEvent (too bad ReportEvent
  469. // doesn't take va_list like FormatMessage does.
  470. //
  471. logString = (OLECHAR **)malloc( count*sizeof(OLECHAR *) );
  472. WsbAffirmAlloc( logString );
  473. // load in the strings
  474. vaList = *inserts;
  475. for( count = 0; (tmpArg = va_arg( vaList, OLECHAR *)) != NULL; count++ ) {
  476. logString[count] = tmpArg;
  477. }
  478. }
  479. // Get a handle to the event source
  480. HANDLE hEventSource = RegisterEventSource(NULL, WSB_LOG_SOURCE_NAME );
  481. // Get the time in case we need to snap shot this event's logs and traces
  482. GetLocalTime(&stime);
  483. if (hEventSource != NULL) {
  484. // Write to event log
  485. DWORD recordDataSize = (newData) ? newDataSize : dataSize;
  486. LPVOID recordData = (newData) ? newData : data;
  487. if ( ReportEvent(hEventSource, logType, category, eventId, NULL, count, recordDataSize, (LPCTSTR *)&logString[0], recordData) ) {
  488. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Event <0x%08lx> was logged.\r\n"), eventId );
  489. WsbTracef( OLESTR("\r\n") );
  490. } else {
  491. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Event <0x%08lx> could not be logged due to the following error: %ls\r\n"), eventId, WsbHrAsString(HRESULT_FROM_WIN32(GetLastError())) );
  492. WsbTracef( OLESTR("\r\n") );
  493. }
  494. DeregisterEventSource(hEventSource);
  495. }
  496. try {
  497. HRESULT hr2 = S_OK;
  498. //
  499. // See if we are to take a snap shot of the event and trace logs when an event of this level is logged.
  500. //
  501. if ( (TRUE == bSnapShot) &&
  502. (TRUE == g_WsbLogSnapShotOn) ) {
  503. SnapShotTraceAndEvent(stime);
  504. }
  505. } WsbCatchAndDo(hr, hr=S_OK; );
  506. } WsbCatch( hr );
  507. if (newData) {
  508. free(newData);
  509. }
  510. if (logString) {
  511. free(logString);
  512. }
  513. }
  514. const OLECHAR*
  515. WsbBoolAsString(
  516. BOOL boolean
  517. )
  518. /*++
  519. Routine Description:
  520. This routine provides a string repesentation (e.g. TRUE, FALSE) for
  521. the value of the boolean supplied.
  522. NOTE: This method does not support localization of the strings.
  523. Arguments:
  524. boolean - A boolean value.
  525. Return Value:
  526. A string representation of the value of the boolean.
  527. --*/
  528. {
  529. return(boolean ? OLESTR("TRUE") : OLESTR("FALSE"));
  530. }
  531. const OLECHAR*
  532. WsbLongAsString(
  533. LONG inLong
  534. )
  535. /*++
  536. Routine Description:
  537. This routine provides a string repesentation for the value of the
  538. long supplied.
  539. NOTE: This method shares memory between subsequent calls of the function.
  540. Arguments:
  541. long - A long value
  542. Return Value:
  543. A string representation of the value of the GUID.
  544. --*/
  545. {
  546. static OLECHAR defaultString[40];
  547. swprintf( defaultString, OLESTR("%ld"), inLong );
  548. return(defaultString);
  549. }
  550. const OLECHAR*
  551. WsbFiletimeAsString(
  552. IN BOOL isRelative,
  553. IN FILETIME time
  554. )
  555. /*++
  556. Routine Description:
  557. This routine provides a string repesentation for the value of the
  558. FILETIME supplied.
  559. NOTE: This method shares memory between subsequent calls of the function.
  560. Arguments:
  561. isRelatice - A boolean that indicates whether the time is absolute (e.g 1/1/1987 ...)
  562. or relative (e.g. 1 hour).
  563. time - A FILETIME.
  564. Return Value:
  565. A string representation of the value of the FILETIME.
  566. --*/
  567. {
  568. static OLECHAR defaultString[80];
  569. OLECHAR* tmpString = 0;
  570. HRESULT hr;
  571. hr = WsbFTtoWCS(isRelative, time, &tmpString, sizeof(defaultString));
  572. if (hr == S_OK) {
  573. wcscpy(defaultString, tmpString);
  574. } else {
  575. wcscpy(defaultString, L"BADFILETIME");
  576. }
  577. WsbFree(tmpString);
  578. return(defaultString);
  579. }
  580. const OLECHAR*
  581. WsbGuidAsString(
  582. GUID guid
  583. )
  584. /*++
  585. Routine Description:
  586. This routine provides a string repesentation for the value of the
  587. GUID supplied.
  588. NOTE: This method shares memory between subsequent calls of the function.
  589. Arguments:
  590. guid - A GUID.
  591. Return Value:
  592. A string representation of the value of the GUID.
  593. --*/
  594. {
  595. static OLECHAR defaultString[40];
  596. swprintf( defaultString, OLESTR("{%.8x-%.4x-%.4hx-%.2x%.2x-%.2x%.2x%.2x%.2x%.2x%.2x}"),
  597. guid.Data1, (UINT)guid.Data2, (UINT)guid.Data3,
  598. (UINT) guid.Data4[0], (UINT) guid.Data4[1],
  599. (UINT) guid.Data4[2], (UINT) guid.Data4[3], (UINT) guid.Data4[4],
  600. (UINT) guid.Data4[5], (UINT) guid.Data4[6], (UINT) guid.Data4[7]);
  601. return(defaultString);
  602. }
  603. const OLECHAR*
  604. WsbHrAsString(
  605. HRESULT hr
  606. )
  607. /*++
  608. Routine Description:
  609. This routine provides a string repesentation (e.g. S_OK, E_POINTER) for
  610. the value of the HRESULT supplied.
  611. NOTE: This method shares memory between subsequent calls of the function.
  612. Arguments:
  613. hr - An HRESULT.
  614. Return Value:
  615. A string representation of the value of the HRESULT.
  616. --*/
  617. {
  618. const OLECHAR *returnString = 0;
  619. const OLECHAR *facilityName = 0;
  620. const DWORD cSize = 1024;
  621. DWORD stringSize = (cSize - 20);
  622. static OLECHAR defaultString[cSize];
  623. DWORD lastError;
  624. // Handle a few special cases which are not in the message table resource
  625. switch ( hr ) {
  626. case S_OK:
  627. returnString = OLESTR("Ok"); // This overloads Win32 NO_ERROR.
  628. break;
  629. case S_FALSE:
  630. returnString = OLESTR("False"); // This overloads Win32 ERROR_INVALID_FUNCTION
  631. break;
  632. default:
  633. break;
  634. }
  635. if ( 0 == returnString ) {
  636. returnString = defaultString;
  637. swprintf( defaultString, OLESTR("0x%08lx"), hr );
  638. //
  639. // First, try getting the message from the system
  640. //
  641. if ( 0 == FormatMessage( FORMAT_MESSAGE_FROM_SYSTEM,
  642. NULL,
  643. hr,
  644. MAKELANGID ( LANG_NEUTRAL, SUBLANG_DEFAULT ),
  645. defaultString,
  646. stringSize,
  647. NULL ) ) {
  648. lastError = GetLastError(); // For debugging
  649. // Next, try the module executing this code.
  650. if ( 0 == FormatMessage( FORMAT_MESSAGE_FROM_HMODULE,
  651. NULL,
  652. hr,
  653. MAKELANGID ( LANG_NEUTRAL, SUBLANG_DEFAULT ),
  654. defaultString,
  655. stringSize,
  656. NULL ) ) {
  657. lastError = GetLastError(); // For debugging
  658. // Finally, try to identify the module based on the facility code
  659. switch ( HRESULT_FACILITY( hr ) ) {
  660. case WSB_FACILITY_PLATFORM:
  661. case WSB_FACILITY_RMS:
  662. case WSB_FACILITY_HSMENG:
  663. case WSB_FACILITY_JOB:
  664. case WSB_FACILITY_HSMTSKMGR:
  665. case WSB_FACILITY_FSA:
  666. case WSB_FACILITY_GUI:
  667. case WSB_FACILITY_MOVER:
  668. case WSB_FACILITY_LAUNCH:
  669. case WSB_FACILITY_USERLINK:
  670. facilityName = WSB_FACILITY_PLATFORM_NAME;
  671. break;
  672. case WSB_FACILITY_TEST:
  673. facilityName = WSB_FACILITY_TEST_NAME;
  674. break;
  675. case HRESULT_FACILITY(FACILITY_NT_BIT):
  676. facilityName = WSB_FACILITY_NTDLL_NAME;
  677. hr &= ~FACILITY_NT_BIT;
  678. break;
  679. default:
  680. facilityName = WSB_FACILITY_NTDLL_NAME;
  681. break;
  682. }
  683. if ( facilityName ) {
  684. HMODULE hModule;
  685. hModule = LoadLibraryEx( facilityName, NULL, LOAD_LIBRARY_AS_DATAFILE );
  686. if (hModule) {
  687. FormatMessage( FORMAT_MESSAGE_FROM_HMODULE,
  688. hModule,
  689. hr,
  690. MAKELANGID ( LANG_NEUTRAL, SUBLANG_DEFAULT ),
  691. defaultString,
  692. stringSize,
  693. NULL );
  694. FreeLibrary(hModule);
  695. } else {
  696. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Could not load facility name DLL %ls. \r\n"), facilityName);
  697. }
  698. }
  699. }
  700. }
  701. //
  702. // remove trailing \r\n ( this makes things nice for tracing and asserts )
  703. //
  704. if ( defaultString[ wcslen(defaultString)-1 ] == OLESTR('\n') ) {
  705. defaultString[ wcslen(defaultString)-1 ] = OLESTR('\0');
  706. if ( defaultString[ wcslen(defaultString)-1 ] == OLESTR('\r') ) {
  707. defaultString[ wcslen(defaultString)-1 ] = OLESTR('\0');
  708. swprintf( &defaultString[ wcslen(defaultString) ], OLESTR(" (0x%08lx)"), hr );
  709. }
  710. }
  711. }
  712. return ( returnString );
  713. }
  714. const OLECHAR*
  715. WsbLonglongAsString(
  716. LONGLONG llong
  717. )
  718. /*++
  719. Routine Description:
  720. This routine provides a string repesentation for the value of the
  721. LONGLONG supplied.
  722. NOTE: This method shares memory between subsequent calls of the function.
  723. Arguments:
  724. llong - A LONGLONG value.
  725. Return Value:
  726. A string representation of the value.
  727. --*/
  728. {
  729. static OLECHAR defaultString[128];
  730. OLECHAR* ptr = &defaultString[0];
  731. WsbLLtoWCS(llong, &ptr, 128);
  732. return(defaultString);
  733. }
  734. const OLECHAR*
  735. WsbStringAsString(
  736. OLECHAR* pStr
  737. )
  738. /*++
  739. Routine Description:
  740. This routine provides a string repesentation for the value of the
  741. String supplied.
  742. NOTE: This method shares memory between subsequent calls of the function.
  743. Arguments:
  744. pStr - A string value.
  745. Return Value:
  746. A string representation of the value.
  747. --*/
  748. {
  749. OLECHAR* returnString;
  750. if (0 == pStr) {
  751. returnString = OLESTR("NULL");
  752. } else {
  753. returnString = pStr;
  754. }
  755. return(returnString);
  756. }
  757. const OLECHAR*
  758. WsbPtrToBoolAsString(
  759. BOOL* pBool
  760. )
  761. /*++
  762. Routine Description:
  763. This routine provides a string repesentation for the value of the
  764. pointer to a BOOL supplied.
  765. NOTE: This method does not support localization of the strings.
  766. Arguments:
  767. pBool - A pointer to a BOOL or NULL.
  768. Return Value:
  769. A string representation of the value of the BOOL or "NULL" if the
  770. pointer was null.
  771. --*/
  772. {
  773. OLECHAR* returnString;
  774. if (0 == pBool) {
  775. returnString = OLESTR("NULL");
  776. } else {
  777. returnString = (OLECHAR*) WsbBoolAsString(*pBool);
  778. }
  779. return(returnString);
  780. }
  781. const OLECHAR*
  782. WsbPtrToFiletimeAsString(
  783. IN BOOL isRelative,
  784. IN FILETIME *pTime
  785. )
  786. /*++
  787. Routine Description:
  788. This routine provides a string repesentation for the value of the
  789. FILETIME supplied.
  790. NOTE: This method shares memory between subsequent calls of the function.
  791. Arguments:
  792. iselatice - A boolean that indicates whether the time is absolute (e.g 1/1/1987 ...)
  793. or relative (e.g. 1 hour).
  794. pTime - A pointer to a FILETIME.
  795. Return Value:
  796. A string representation of the value of the FILETIME.
  797. --*/
  798. {
  799. OLECHAR* returnString;
  800. if (0 == pTime) {
  801. returnString = OLESTR("NULL");
  802. } else {
  803. returnString = (OLECHAR*) WsbFiletimeAsString(isRelative, *pTime);
  804. }
  805. return(returnString);
  806. }
  807. const OLECHAR*
  808. WsbPtrToGuidAsString(
  809. GUID* pGuid
  810. )
  811. /*++
  812. Routine Description:
  813. This routine provides a string repesentation for the value of the
  814. pointer to a GUID supplied.
  815. NOTE: This method does not support localization of the strings.
  816. Arguments:
  817. pGuid - A pointer to a GUID or NULL.
  818. Return Value:
  819. A string representation of the value of the GUID or "NULL" if the
  820. pointer was null.
  821. --*/
  822. {
  823. OLECHAR* returnString;
  824. if (0 == pGuid) {
  825. returnString = OLESTR("NULL");
  826. } else {
  827. returnString = (OLECHAR*) WsbGuidAsString(*pGuid);
  828. }
  829. return(returnString);
  830. }
  831. const OLECHAR*
  832. WsbPtrToHrAsString(
  833. HRESULT * pHr
  834. )
  835. /*++
  836. Routine Description:
  837. This routine provides a string repesentation for the value of the
  838. pointer to a HRESULT supplied.
  839. NOTE: This method does not support localization of the strings.
  840. Arguments:
  841. pHr - A pointer to an HRESULT.
  842. Return Value:
  843. A string representation of the value of the HRESULT.
  844. --*/
  845. {
  846. OLECHAR* returnString;
  847. if (0 == pHr) {
  848. returnString = OLESTR("NULL");
  849. } else {
  850. returnString = (OLECHAR*) WsbHrAsString(*pHr);
  851. }
  852. return(returnString);
  853. }
  854. const OLECHAR*
  855. WsbPtrToLonglongAsString(
  856. LONGLONG* pLlong
  857. )
  858. /*++
  859. Routine Description:
  860. This routine provides a string repesentation for the value of the
  861. pointer to a LONGLONG supplied.
  862. NOTE: This method does not support localization of the strings.
  863. Arguments:
  864. pLonglong - A pointer to a LONGLONG or NULL.
  865. Return Value:
  866. A string representation of the value of the LONGLONG or "NULL" if the
  867. pointer was null.
  868. --*/
  869. {
  870. OLECHAR* returnString;
  871. if (0 == pLlong) {
  872. returnString = OLESTR("NULL");
  873. } else {
  874. returnString = (OLECHAR*) WsbLonglongAsString(*pLlong);
  875. }
  876. return(returnString);
  877. }
  878. const OLECHAR*
  879. WsbPtrToLongAsString(
  880. LONG* pLong
  881. )
  882. /*++
  883. Routine Description:
  884. This routine provides a string repesentation for the value of the
  885. pointer to a LONG supplied.
  886. NOTE: This method does not support localization of the strings, and
  887. shares memory between subsequent calls of the function.
  888. Arguments:
  889. pLong - A pointer to a LONG or NULL.
  890. Return Value:
  891. A string representation of the value of the LONG or "NULL" if the
  892. pointer was null.
  893. --*/
  894. {
  895. OLECHAR* returnString;
  896. static OLECHAR defaultString[20];
  897. if (0 == pLong) {
  898. returnString = OLESTR("NULL");
  899. } else {
  900. swprintf(defaultString, OLESTR("%ld"), *pLong);
  901. returnString = defaultString;
  902. }
  903. return(returnString);
  904. }
  905. const OLECHAR*
  906. WsbPtrToShortAsString(
  907. SHORT* pShort
  908. )
  909. /*++
  910. Routine Description:
  911. This routine provides a string repesentation for the value of the
  912. pointer to a SHORT supplied.
  913. NOTE: This method does not support localization of the strings, and
  914. shares memory between subsequent calls of the function.
  915. Arguments:
  916. pShort - A pointer to a SHORT or NULL.
  917. Return Value:
  918. A string representation of the value of the SHORT or "NULL" if the
  919. pointer was null.
  920. --*/
  921. {
  922. OLECHAR* returnString;
  923. static OLECHAR defaultString[20];
  924. if (0 == pShort) {
  925. returnString = OLESTR("NULL");
  926. } else {
  927. swprintf(defaultString, OLESTR("%d"), *pShort);
  928. returnString = defaultString;
  929. }
  930. return(returnString);
  931. }
  932. const OLECHAR*
  933. WsbPtrToByteAsString(
  934. BYTE* pByte
  935. )
  936. /*++
  937. Routine Description:
  938. This routine provides a string repesentation for the value of the
  939. pointer to a BYTE supplied.
  940. NOTE: This method does not support localization of the strings, and
  941. shares memory between subsequent calls of the function.
  942. Arguments:
  943. pByte - A pointer to a BYTE or NULL.
  944. Return Value:
  945. A string representation of the value of the BYTE or "NULL" if the
  946. pointer was null.
  947. --*/
  948. {
  949. OLECHAR* returnString;
  950. static OLECHAR defaultString[20];
  951. if (0 == pByte) {
  952. returnString = OLESTR("NULL");
  953. } else {
  954. swprintf(defaultString, OLESTR("%d"), *pByte);
  955. returnString = defaultString;
  956. }
  957. return(returnString);
  958. }
  959. const OLECHAR*
  960. WsbPtrToStringAsString(
  961. OLECHAR** pString
  962. )
  963. /*++
  964. Routine Description:
  965. This routine provides a string repesentation for the value of the
  966. pointer to a string supplied.
  967. NOTE: This method does not support localization ofthe strings.
  968. Arguments
  969. pString - A pointer to a OLECHAR* or NULL.
  970. Return Value:
  971. The string or "NULL" if the pointer was null.
  972. --*/
  973. {
  974. OLECHAR* returnString;
  975. if( (0 == pString) || (0 == *pString) ) {
  976. returnString = OLESTR("NULL");
  977. } else {
  978. returnString = *pString;
  979. }
  980. return(returnString);
  981. }
  982. const OLECHAR*
  983. WsbPtrToUliAsString(
  984. ULARGE_INTEGER* pUli
  985. )
  986. /*++
  987. Routine Description:
  988. This routine provides a string repesentation for the value of the
  989. pointer to a ULARGE_INTEGER supplied.
  990. NOTE: This method does not support localization of the strings, and
  991. shares memory between subsequent calls of the function.
  992. Arguments:
  993. pUli - A pointer to a ULARGE_INTEGER or NULL.
  994. Return Value:
  995. A string representation of the value of the ULARGE_INTEGER or "NULL" if the
  996. pointer was null.
  997. --*/
  998. {
  999. OLECHAR* returnString;
  1000. if (0 == pUli) {
  1001. returnString = OLESTR("NULL");
  1002. } else {
  1003. returnString = (OLECHAR*) WsbLonglongAsString( pUli->QuadPart );
  1004. }
  1005. return(returnString);
  1006. }
  1007. const OLECHAR*
  1008. WsbPtrToUlongAsString(
  1009. ULONG* pUlong
  1010. )
  1011. /*++
  1012. Routine Description:
  1013. This routine provides a string repesentation for the value of the
  1014. pointer to a ULONG supplied.
  1015. NOTE: This method does not support localization of the strings, and
  1016. shares memory between subsequent calls of the function.
  1017. Arguments:
  1018. pUlong - A pointer to a ULONG or NULL.
  1019. Return Value:
  1020. A string representation of the value of the ULONG or "NULL" if the
  1021. pointer was null.
  1022. --*/
  1023. {
  1024. OLECHAR* returnString;
  1025. static OLECHAR defaultString[20];
  1026. if (0 == pUlong) {
  1027. returnString = OLESTR("NULL");
  1028. } else {
  1029. swprintf(defaultString, OLESTR("%lu"), *pUlong);
  1030. returnString = defaultString;
  1031. }
  1032. return(returnString);
  1033. }
  1034. const OLECHAR*
  1035. WsbPtrToUshortAsString(
  1036. USHORT* pUshort
  1037. )
  1038. /*++
  1039. Routine Description:
  1040. This routine provides a string repesentation for the value of the
  1041. pointer to a USHORT supplied.
  1042. NOTE: This method does not support localization of the strings, and
  1043. shares memory between subsequent calls of the function.
  1044. Arguments:
  1045. pUshort - A pointer to a USHORT or NULL.
  1046. Return Value:
  1047. A string representation of the value of the USHORT or "NULL" if the
  1048. pointer was null.
  1049. --*/
  1050. {
  1051. OLECHAR* returnString;
  1052. static OLECHAR defaultString[20];
  1053. if (0 == pUshort) {
  1054. returnString = OLESTR("NULL");
  1055. } else {
  1056. swprintf(defaultString, OLESTR("%u"), *pUshort);
  1057. returnString = defaultString;
  1058. }
  1059. return(returnString);
  1060. }
  1061. const OLECHAR*
  1062. WsbPtrToPtrAsString(
  1063. void** ppVoid
  1064. )
  1065. /*++
  1066. Routine Description:
  1067. This routine provides a string repesentation for the value of the
  1068. pointer to a ULONG supplied.
  1069. NOTE: This method does not support localization of the strings, and
  1070. shares memory between subsequent calls of the function.
  1071. Arguments:
  1072. pUlong - A pointer to a ULONG or NULL.
  1073. Return Value:
  1074. A string representation of the value of the ULONG or "NULL" if the
  1075. pointer was null.
  1076. --*/
  1077. {
  1078. OLECHAR* returnString;
  1079. static OLECHAR defaultString[20];
  1080. if (0 == ppVoid) {
  1081. returnString = OLESTR("NULL");
  1082. } else {
  1083. swprintf(defaultString, OLESTR("0x%p"), *ppVoid);
  1084. returnString = defaultString;
  1085. }
  1086. return(returnString);
  1087. }
  1088. const OLECHAR*
  1089. WsbAbbreviatePath(
  1090. const OLECHAR* path,
  1091. USHORT length
  1092. )
  1093. /*++
  1094. Routine Description:
  1095. This routine condenses a path from it's original length to the requested
  1096. length by chopping out it's middle characters
  1097. NOTE: This method does not support localization of the strings, and
  1098. shares memory between subsequent calls of the function.
  1099. Arguments:
  1100. path - A pointer to the path
  1101. length - The condensed path length including the \0
  1102. Return Value:
  1103. A string representation of the value of the BYTE or "NULL" if the
  1104. pointer was null. This function also returns "NULL" if the length is less
  1105. than 4 bytes.
  1106. --*/
  1107. {
  1108. HRESULT hr = S_OK;
  1109. OLECHAR* returnString;
  1110. static CWsbStringPtr tmpString;
  1111. returnString = OLESTR("ERROR");
  1112. try {
  1113. //
  1114. // Check to see if we have anything to work with
  1115. //
  1116. if ((0 == path) || (length < 4)) {
  1117. returnString = OLESTR("NULL");
  1118. } else {
  1119. //
  1120. // Get enough space for the return
  1121. //
  1122. USHORT pathlen;
  1123. pathlen = (USHORT)wcslen(path);
  1124. hr = tmpString.Realloc(length);
  1125. if (S_OK != hr) {
  1126. returnString = OLESTR("No memory");
  1127. WsbAffirmHr(hr);
  1128. }
  1129. if (pathlen < length) {
  1130. swprintf(tmpString, OLESTR("%s"), path);
  1131. } else {
  1132. USHORT partlength = (USHORT) ( (length - 4) / 2 );
  1133. wcsncpy(tmpString, path, partlength);
  1134. tmpString[(int) partlength] = L'\0';
  1135. wcscat(tmpString, OLESTR("..."));
  1136. wcscat(tmpString, &(path[pathlen - partlength]));
  1137. }
  1138. returnString = tmpString;
  1139. }
  1140. } WsbCatch(hr);
  1141. return(returnString);
  1142. }
  1143. void WsbTraceBufferAsBytes(
  1144. DWORD size,
  1145. LPVOID data
  1146. )
  1147. /*++
  1148. Routine Description:
  1149. This routine traces an arbitrary size buffer of bytes in hex and asci.
  1150. A similar routine could be written trace a buffer in words.
  1151. Arguments:
  1152. size - The size of buffer to trace.
  1153. data - The data to trace.
  1154. Return Value:
  1155. None.
  1156. --*/
  1157. {
  1158. HRESULT hr = S_OK;
  1159. try {
  1160. // Make sure we are supposed to trace
  1161. WsbAffirm( 0 != g_pWsbTrace, S_OK);
  1162. // Make sure we have something to trace
  1163. WsbAssertPointer( data );
  1164. CWsbStringPtr traceString;
  1165. char *output;
  1166. unsigned char *bufferP = (unsigned char *)data;
  1167. // IMPORTANT NOTE: Changing these may mean the last line processing need to be changed.
  1168. char *beginAsci = " [";
  1169. char *endAsci = "]";
  1170. char *charFmt = "%02x";
  1171. char *addFmt = "%04x:";
  1172. char *between8 = " ";
  1173. char *between4 = " ";
  1174. char noPrintChar = 0x2e;
  1175. const int ll = 16; // IMPORTANT NOTE: line length, a multiple of 8 - if this changes, the last line processing needs to be fixed.
  1176. int lineCount = 0;
  1177. output = (char *)malloc( (/*address*/6+/*data*/(ll*3)+/*asci*/4+ll+3/*between*/+7+1)*sizeof(char) );
  1178. WsbAffirmAlloc( output );
  1179. if ( size > 0 ) {
  1180. unsigned long i, ii, j, k;
  1181. long repeat;
  1182. unsigned char c;
  1183. for ( i = 0; i < size; i++ ) {
  1184. if ( (0 == i % ll) && (i != 0) ) {
  1185. // print asci interpretation
  1186. sprintf( output, beginAsci );
  1187. traceString.Append(output);
  1188. for ( j = 0; j < ll; j++ ) {
  1189. c = bufferP[i-ll+j];
  1190. if ( c < ' ' || c > '~' ) {
  1191. c = noPrintChar;
  1192. }
  1193. sprintf( output, "%c", c );
  1194. traceString.Append(output);
  1195. }
  1196. sprintf( output, endAsci );
  1197. traceString.Append(output);
  1198. WsbTracef( OLESTR("%ls\n"), (WCHAR *) traceString );
  1199. lineCount++;
  1200. // now check if the next line is the same as the one just printed
  1201. repeat = 0;
  1202. ii = i;
  1203. while ( (0 == memcmp( &bufferP[ii-ll], &bufferP[ii], ll )) && (ii+ll < size) ) {
  1204. repeat++;
  1205. ii += ll;
  1206. }
  1207. if ( repeat > 1 ) {
  1208. sprintf( output, " previous line repeats %ld times", repeat);
  1209. traceString = output;
  1210. WsbTracef( OLESTR("%ls\n"), (WCHAR *) traceString );
  1211. lineCount++;
  1212. i = ii;
  1213. }
  1214. }
  1215. if ( 0 == i % ll ) {
  1216. // print address
  1217. sprintf( output, addFmt, i );
  1218. traceString = output;
  1219. }
  1220. // add alignment spacing
  1221. if ( (0 == (i + 8) % ll) ) {
  1222. sprintf( output, between8 );
  1223. traceString.Append(output);
  1224. }
  1225. else if ( 0 == i % 4 ) {
  1226. sprintf( output, between4 );
  1227. traceString.Append(output);
  1228. }
  1229. else {
  1230. sprintf( output, " " );
  1231. traceString.Append(output);
  1232. }
  1233. // print byte in hex
  1234. sprintf( output, charFmt, bufferP[i] );
  1235. traceString.Append(output);
  1236. }
  1237. // handle the last line; i allways > 0 here
  1238. // NOTE: This is only good for upto 16 chars per line.
  1239. if ( i % ll ) {
  1240. k = (ll - (i % ll)) * 3 + ( (i % ll) < 5 ? 1 : 0 )+ ( (i % ll) < 9 ? 2 : 0 )+ ( (i % ll) < 13 ? 1 : 0 );
  1241. for ( j = 0; j < k ; j++ ) {
  1242. sprintf( output, " ");
  1243. traceString.Append(output);
  1244. }
  1245. }
  1246. k = (i % ll) ? (i % ll) : ll ;
  1247. sprintf( output, beginAsci );
  1248. traceString.Append(output);
  1249. for ( j = 0; j < k; j++ ) {
  1250. c = bufferP[i-k+j];
  1251. if ( c < ' ' || c > '~' ) {
  1252. c = noPrintChar;
  1253. }
  1254. sprintf( output, "%c", c );
  1255. traceString.Append(output);
  1256. }
  1257. sprintf( output, endAsci); lineCount++;
  1258. traceString.Append(output);
  1259. WsbTracef( OLESTR("%ls\n"), (WCHAR *) traceString );
  1260. }
  1261. }
  1262. WsbCatch( hr );
  1263. }
  1264. void
  1265. WsbTraceTerminate(
  1266. void
  1267. )
  1268. /*++
  1269. Routine Description:
  1270. Terminate (cleanup) this module because the process is ending
  1271. Arguments:
  1272. None.
  1273. Return Value:
  1274. None.
  1275. --*/
  1276. {
  1277. if (BOGUS_TLS_INDEX != TlsIndex) {
  1278. TlsFree(TlsIndex);
  1279. TlsIndex = BOGUS_TLS_INDEX;
  1280. }
  1281. }
  1282. ULONG
  1283. WsbTraceThreadOff(
  1284. void
  1285. )
  1286. /*++
  1287. Routine Description:
  1288. Increment the trace-off count for this thread
  1289. Arguments:
  1290. None.
  1291. Return Value:
  1292. The final trace-off count.
  1293. --*/
  1294. {
  1295. ULONG count = 0;
  1296. THREAD_DATA* pThreadData = NULL;
  1297. if (S_OK == GetThreadDataPointer(&pThreadData)) {
  1298. count = ++(pThreadData->TraceOffCount);
  1299. }
  1300. return(count);
  1301. }
  1302. ULONG
  1303. WsbTraceThreadOffCount(
  1304. void
  1305. )
  1306. /*++
  1307. Routine Description:
  1308. Return the current trace-off count for this thread
  1309. Arguments:
  1310. None.
  1311. Return Value:
  1312. The current trace-off count.
  1313. --*/
  1314. {
  1315. ULONG count = 0;
  1316. THREAD_DATA* pThreadData = NULL;
  1317. if (S_OK == GetThreadDataPointer(&pThreadData)) {
  1318. count = pThreadData->TraceOffCount;
  1319. }
  1320. return(count);
  1321. }
  1322. ULONG
  1323. WsbTraceThreadOn(
  1324. void
  1325. )
  1326. /*++
  1327. Routine Description:
  1328. Decrement the trace-off count for this thread
  1329. Arguments:
  1330. None.
  1331. Return Value:
  1332. The final trace-off count.
  1333. --*/
  1334. {
  1335. ULONG count = 0;
  1336. THREAD_DATA* pThreadData = NULL;
  1337. if (S_OK == GetThreadDataPointer(&pThreadData)) {
  1338. if (0 < pThreadData->TraceOffCount) {
  1339. pThreadData->TraceOffCount--;
  1340. }
  1341. count = pThreadData->TraceOffCount;
  1342. }
  1343. return(count);
  1344. }
  1345. static HRESULT
  1346. OutputTraceString(
  1347. IN ULONG indentLevel,
  1348. IN OLECHAR* introString,
  1349. IN OLECHAR* format,
  1350. IN va_list vaList
  1351. )
  1352. /*++
  1353. Routine Description:
  1354. Build and output the trace string.
  1355. Arguments:
  1356. indentLevel - Count of indentation strings to output
  1357. introString - String to add before variable list
  1358. vaList - Variable list to format
  1359. Return Value:
  1360. The data pointer.
  1361. --*/
  1362. {
  1363. HRESULT hr = S_OK;
  1364. OLECHAR traceString[WSB_TRACE_BUFF_SIZE];
  1365. try {
  1366. LONG incSize;
  1367. LONG traceSize = 0;
  1368. // Initialize the string
  1369. swprintf(traceString, OLESTR(""));
  1370. // Add indentation
  1371. incSize = wcslen(WSB_INDENT_STRING);
  1372. for(ULONG level = 0; level < indentLevel; level++) {
  1373. if ((traceSize + incSize) < WSB_TRACE_BUFF_SIZE) {
  1374. wcscat(traceString, WSB_INDENT_STRING);
  1375. traceSize += incSize;
  1376. }
  1377. }
  1378. // Add the intro string
  1379. if (introString) {
  1380. incSize = wcslen(introString);
  1381. } else {
  1382. incSize = 0;
  1383. }
  1384. if (incSize && ((traceSize + incSize) < WSB_TRACE_BUFF_SIZE)) {
  1385. wcscat(traceString, introString);
  1386. traceSize += incSize;
  1387. }
  1388. // Format the arguments (leave room for EOL and EOS)
  1389. incSize = _vsnwprintf(&traceString[traceSize],
  1390. (WSB_TRACE_BUFF_SIZE - traceSize - 3), format, vaList);
  1391. if (incSize < 0) {
  1392. // This means we filled the buffer and would have overflowed
  1393. // Need to add EOS
  1394. traceString[WSB_TRACE_BUFF_SIZE - 3] = OLECHAR('\0');
  1395. traceSize = WSB_TRACE_BUFF_SIZE - 3;
  1396. } else {
  1397. traceSize += incSize;
  1398. }
  1399. // Add EOL if needed
  1400. if (introString) {
  1401. wcscat(&traceString[traceSize], OLESTR("\r\n"));
  1402. }
  1403. WsbAffirmHr(g_pWsbTrace->Print(traceString));
  1404. } WsbCatch (hr);
  1405. return(hr);
  1406. }
  1407. static HRESULT
  1408. GetThreadDataPointer(
  1409. OUT THREAD_DATA** ppTD
  1410. )
  1411. /*++
  1412. Routine Description:
  1413. Return a pointer to the data specific to the current thread. This
  1414. function will allocate space for the thread data (and initialize it)
  1415. if needed.
  1416. Arguments:
  1417. ppTD - Pointer to pointer to thread data.
  1418. Return Value:
  1419. The data pointer.
  1420. --*/
  1421. {
  1422. HRESULT hr = E_FAIL;
  1423. THREAD_DATA* pThreadData = NULL;
  1424. // Make sure the TLS index is valid
  1425. if (BOGUS_TLS_INDEX != TlsIndex) {
  1426. // Try to get the data pointer for this thread
  1427. pThreadData = static_cast<THREAD_DATA*>(TlsGetValue(TlsIndex));
  1428. if (pThreadData) {
  1429. hr = S_OK;
  1430. } else {
  1431. // Allocate data for this thread yet
  1432. pThreadData = static_cast<THREAD_DATA*>(WsbAlloc(sizeof(THREAD_DATA)));
  1433. if (pThreadData) {
  1434. if (TlsSetValue(TlsIndex, pThreadData)) {
  1435. // Initialize the data for this thread
  1436. pThreadData->TraceOffCount = 0;
  1437. pThreadData->IndentLevel = 0;
  1438. pThreadData->LogModule = NULL;
  1439. pThreadData->LogModuleLine = 0;
  1440. pThreadData->LogNTBuild = 0;
  1441. pThreadData->LogRSBuild = 0;
  1442. hr = S_OK;
  1443. } else {
  1444. // TlsSetValue failed!
  1445. WsbFree(pThreadData);
  1446. pThreadData = NULL;
  1447. }
  1448. }
  1449. }
  1450. }
  1451. *ppTD = pThreadData;
  1452. return(hr);
  1453. }
  1454. static void
  1455. SnapShotTraceAndEvent(
  1456. SYSTEMTIME stime
  1457. )
  1458. /*++
  1459. Routine Description:
  1460. This routine saves the trace files and event logs
  1461. Arguments:
  1462. Return Value:
  1463. None.
  1464. --*/
  1465. {
  1466. HRESULT hr = S_OK;
  1467. try {
  1468. OLECHAR dataString[256];
  1469. OLECHAR tmpString[50];
  1470. OLECHAR mutexName[50] = L"WsbTraceSnapShotMutex";
  1471. DWORD sizeGot;
  1472. HANDLE mutexHandle = INVALID_HANDLE_VALUE;
  1473. //
  1474. // The level is one to snap shot and snap shot is on. Now make sure there is a
  1475. // path specified where we are to copy the logs
  1476. //
  1477. WsbAffirm(0 != g_pWsbLogSnapShotPath, E_POINTER);
  1478. WsbAffirm(0 != wcslen(g_pWsbLogSnapShotPath), E_POINTER);
  1479. //
  1480. // Get the system root string from the registry
  1481. //
  1482. WsbAffirmHr(WsbGetRegistryValueString(NULL, WSB_CURRENT_VERSION_REGISTRY_KEY, WSB_SYSTEM_ROOT_REGISTRY_VALUE, dataString, 256, &sizeGot));
  1483. CWsbStringPtr snapShotSubDir;
  1484. CWsbStringPtr snapShotFile;
  1485. snapShotSubDir = g_pWsbLogSnapShotPath;
  1486. //
  1487. // Make sure there is a "\" at the end of the path
  1488. //
  1489. int len;
  1490. len = wcslen(snapShotSubDir);
  1491. if (snapShotSubDir[len] != '\\') {
  1492. snapShotSubDir.Append(L"\\");
  1493. }
  1494. // Build the path to the subdirectory that will contain the logs from the input path
  1495. // and the time of the event.
  1496. swprintf(tmpString, OLESTR("%2.02u.%2.02u-%2.2u.%2.2u.%2.2u.%3.3u"),
  1497. stime.wMonth, stime.wDay,
  1498. stime.wHour, stime.wMinute,
  1499. stime.wSecond, stime.wMilliseconds);
  1500. snapShotSubDir.Append(tmpString);
  1501. //
  1502. // Make sure the subdirectory can be created
  1503. //
  1504. WsbAffirmHr(WsbCreateAllDirectories(snapShotSubDir));
  1505. //
  1506. // We need to synchronize around the creating of the
  1507. // event backup files and copying them. Since all three
  1508. // services will access this code, use a mutex to
  1509. // synchronize them.
  1510. mutexHandle = CreateMutex(NULL, TRUE, mutexName);
  1511. if (mutexHandle) {
  1512. //
  1513. // Copy the event logs
  1514. // First back them up and then copy the backup file.
  1515. //
  1516. HANDLE eventLogHandle = INVALID_HANDLE_VALUE;
  1517. try {
  1518. CWsbStringPtr computerName;
  1519. CWsbStringPtr logName;
  1520. WsbAffirmHr( WsbGetComputerName( computerName ) );
  1521. //
  1522. // Open the application event log and back it up
  1523. //
  1524. logName = dataString;
  1525. logName.Append(WSB_APP_EVENT_LOG);
  1526. eventLogHandle = OpenEventLog((LPCTSTR)computerName, (LPCTSTR)logName);
  1527. if (INVALID_HANDLE_VALUE != eventLogHandle) {
  1528. logName = dataString;
  1529. logName.Append(WSB_APP_EVENT_LOG_BKUP);
  1530. DeleteFile(logName);
  1531. WsbAffirmStatus(BackupEventLog(eventLogHandle, (LPCTSTR)logName));
  1532. WsbAffirmStatus(CloseEventLog(eventLogHandle));
  1533. snapShotFile = snapShotSubDir;
  1534. snapShotFile.Append(WSB_APP_EVENT_LOG_NAME);
  1535. //
  1536. // Now copy the backup file
  1537. //
  1538. WsbAffirmStatus(CopyFile(logName, snapShotFile, FALSE));
  1539. }
  1540. //
  1541. // Open the system event log and back it up
  1542. //
  1543. logName = dataString;
  1544. logName.Append(WSB_SYS_EVENT_LOG);
  1545. eventLogHandle = OpenEventLog((LPCTSTR)computerName, (LPCTSTR)logName);
  1546. if (INVALID_HANDLE_VALUE != eventLogHandle) {
  1547. logName = dataString;
  1548. logName.Append(WSB_SYS_EVENT_LOG_BKUP);
  1549. DeleteFile(logName);
  1550. WsbAffirmStatus(BackupEventLog(eventLogHandle, (LPCTSTR)logName));
  1551. WsbAffirmStatus(CloseEventLog(eventLogHandle));
  1552. snapShotFile = snapShotSubDir;
  1553. snapShotFile.Append(WSB_SYS_EVENT_LOG_NAME);
  1554. //
  1555. // Now copy the backup file
  1556. //
  1557. WsbAffirmStatus(CopyFile(logName, snapShotFile, FALSE));
  1558. }
  1559. } WsbCatchAndDo(hr,if (INVALID_HANDLE_VALUE != eventLogHandle) {
  1560. CloseEventLog(eventLogHandle);}; hr = S_OK; );
  1561. (void)ReleaseMutex(mutexHandle);
  1562. }
  1563. //
  1564. // Copy the trace files if there are any
  1565. //
  1566. try {
  1567. WIN32_FIND_DATA findData;
  1568. HANDLE handle;
  1569. CWsbStringPtr traceFile;
  1570. CWsbStringPtr searchString;
  1571. BOOL foundFile;
  1572. //
  1573. // Find the file(s)
  1574. //
  1575. WsbAffirmHr(WsbGetMetaDataPath(searchString));
  1576. searchString.Append(WSB_RS_TRACE_FILES);
  1577. handle = FindFirstFile(searchString, &findData);
  1578. snapShotFile = snapShotSubDir;
  1579. snapShotFile.Append(L"\\");
  1580. WsbAffirmHr(WsbGetMetaDataPath(traceFile));
  1581. traceFile.Append(WSB_RS_TRACE_PATH);
  1582. WsbAffirmHr(snapShotFile.Append((OLECHAR *)(findData.cFileName)));
  1583. WsbAffirmHr(traceFile.Append((OLECHAR *)(findData.cFileName)));
  1584. // If we found a file, then remember the scan handle and
  1585. // return the scan item.
  1586. foundFile = TRUE;
  1587. while ((INVALID_HANDLE_VALUE != handle) && (foundFile == TRUE)) {
  1588. if ((FILE_ATTRIBUTE_DIRECTORY & findData.dwFileAttributes) != FILE_ATTRIBUTE_DIRECTORY) {
  1589. WsbAffirmStatus(CopyFile(traceFile, snapShotFile, FALSE));
  1590. }
  1591. foundFile = FindNextFile(handle, &findData);
  1592. snapShotFile = snapShotSubDir;
  1593. snapShotFile.Append(L"\\");
  1594. WsbAffirmHr(WsbGetMetaDataPath(traceFile));
  1595. traceFile.Append(WSB_RS_TRACE_PATH);
  1596. WsbAffirmHr(snapShotFile.Append((OLECHAR *)(findData.cFileName)));
  1597. WsbAffirmHr(traceFile.Append((OLECHAR *)(findData.cFileName)));
  1598. }
  1599. } WsbCatchAndDo(hr, hr = S_OK; );
  1600. }
  1601. WsbCatch( hr );
  1602. }
  1603. #include "winnls.h"
  1604. #include "resource.h"
  1605. #define HIDWORD(_qw) (DWORD)((_qw)>>32)
  1606. #define LODWORD(_qw) (DWORD)(_qw)
  1607. #define ARRAYSIZE(a) (sizeof(a)/sizeof(a[0]))
  1608. LPTSTR AddCommas(DWORD dw, LPTSTR pszResult, int nResLen);
  1609. const int pwOrders[] = {IDS_WSB_BYTES, IDS_WSB_ORDERKB, IDS_WSB_ORDERMB,
  1610. IDS_WSB_ORDERGB, IDS_WSB_ORDERTB, IDS_WSB_ORDERPB, IDS_WSB_ORDEREB};
  1611. HRESULT WsbShortSizeFormat64(__int64 dw64, LPTSTR szBuf)
  1612. /*++
  1613. Routine Description:
  1614. Converts numbers into sort formats
  1615. 532 -> 523 bytes
  1616. 1340 -> 1.3KB
  1617. 23506 -> 23.5KB
  1618. -> 2.4MB
  1619. -> 5.2GB
  1620. Arguments:
  1621. Return Value:
  1622. Note:
  1623. This code is cloned from MS source /shell/shelldll/util.c - AHB
  1624. --*/
  1625. {
  1626. int i;
  1627. UINT wInt, wLen, wDec;
  1628. TCHAR szTemp[10], szOrder[20], szFormat[5];
  1629. HMODULE hModule;
  1630. if (dw64 < 1000) {
  1631. wsprintf(szTemp, TEXT("%d"), LODWORD(dw64));
  1632. i = 0;
  1633. goto AddOrder;
  1634. }
  1635. for (i = 1; i<ARRAYSIZE(pwOrders)-1 && dw64 >= 1000L * 1024L; dw64 >>= 10, i++);
  1636. /* do nothing */
  1637. wInt = LODWORD(dw64 >> 10);
  1638. AddCommas(wInt, szTemp, 10);
  1639. wLen = lstrlen(szTemp);
  1640. if (wLen < 3)
  1641. {
  1642. wDec = LODWORD(dw64 - (__int64)wInt * 1024L) * 1000 / 1024;
  1643. // At this point, wDec should be between 0 and 1000
  1644. // we want get the top one (or two) digits.
  1645. wDec /= 10;
  1646. if (wLen == 2)
  1647. wDec /= 10;
  1648. // Note that we need to set the format before getting the
  1649. // intl char.
  1650. lstrcpy(szFormat, TEXT("%02d"));
  1651. szFormat[2] = (TCHAR)( TEXT('0') + 3 - wLen );
  1652. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_SDECIMAL,
  1653. szTemp+wLen, ARRAYSIZE(szTemp)-wLen);
  1654. wLen = lstrlen(szTemp);
  1655. wLen += wsprintf(szTemp+wLen, szFormat, wDec);
  1656. }
  1657. AddOrder:
  1658. hModule = LoadLibraryEx(WSB_FACILITY_PLATFORM_NAME, NULL, LOAD_LIBRARY_AS_DATAFILE);
  1659. if (hModule) {
  1660. LoadString(hModule,
  1661. pwOrders[i],
  1662. szOrder,
  1663. ARRAYSIZE(szOrder));
  1664. wsprintf(szBuf, szOrder, (LPTSTR)szTemp);
  1665. FreeLibrary(hModule);
  1666. }
  1667. return S_OK;
  1668. }
  1669. LPTSTR AddCommas(DWORD dw, LPTSTR pszResult, int nResLen)
  1670. /*++
  1671. Routine Description:
  1672. Takes a DWORD add commas etc to it and puts the result in the buffer
  1673. Arguments:
  1674. Return Value:
  1675. Note:
  1676. This code is cloned from MS source /shell/shelldll/util.c - AHB
  1677. --*/
  1678. {
  1679. TCHAR szTemp[20]; // more than enough for a DWORD
  1680. TCHAR szSep[5];
  1681. NUMBERFMT nfmt;
  1682. nfmt.NumDigits=0;
  1683. nfmt.LeadingZero=0;
  1684. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_SGROUPING, szSep, ARRAYSIZE(szSep));
  1685. nfmt.Grouping = _tcstol(szSep, NULL, 10);
  1686. GetLocaleInfo(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, szSep, ARRAYSIZE(szSep));
  1687. nfmt.lpDecimalSep = nfmt.lpThousandSep = szSep;
  1688. nfmt.NegativeOrder= 0;
  1689. wsprintf(szTemp, TEXT("%lu"), dw);
  1690. if (GetNumberFormat(LOCALE_USER_DEFAULT, 0, szTemp, &nfmt, pszResult, nResLen) == 0)
  1691. lstrcpy(pszResult, szTemp);
  1692. return pszResult;
  1693. }
  1694. void
  1695. WsbTraceAndPrint(
  1696. DWORD eventId,
  1697. ...
  1698. )
  1699. /*++
  1700. Routine Description:
  1701. This routine writes a message into standard output. The message
  1702. is also written to the application trace file.
  1703. Arguments:
  1704. eventId - The message Id to log.
  1705. Inserts - Message inserts that are merged with the message description specified by
  1706. eventId. The number of inserts must match the number specified by the
  1707. message description. The last insert must be NULL to indicate the
  1708. end of the insert list.
  1709. Return Value:
  1710. None.
  1711. --*/
  1712. {
  1713. HRESULT hr = S_OK;
  1714. try {
  1715. va_list vaList;
  1716. va_start(vaList, eventId);
  1717. WsbTraceAndPrintV(eventId, &vaList );
  1718. va_end(vaList);
  1719. } WsbCatch( hr );
  1720. }
  1721. void
  1722. WsbTraceAndPrintV(
  1723. DWORD eventId,
  1724. va_list * inserts
  1725. )
  1726. /*++
  1727. Routine Description:
  1728. This routine writes a message into standard output. The message
  1729. is also written to the application trace file.
  1730. Arguments:
  1731. eventId - The message Id to log.
  1732. inserts - An array of message inserts that are merged with the message description
  1733. specified by eventId. The number of inserts must match the number
  1734. specified by the message description. The last insert must be NULL,
  1735. to indicate the end of the insert list.
  1736. Return Value:
  1737. None.
  1738. --*/
  1739. {
  1740. HRESULT hr = S_OK;
  1741. try {
  1742. WsbAssertPointer( inserts );
  1743. const OLECHAR * facilityName = 0;
  1744. BOOL bPrint;
  1745. OLECHAR * messageText = 0;
  1746. //
  1747. // Determine type of event
  1748. //
  1749. switch ( eventId & 0xc0000000 ) {
  1750. case ERROR_SEVERITY_INFORMATIONAL:
  1751. bPrint = (g_WsbPrintLevel >= WSB_LOG_LEVEL_INFORMATION) ? TRUE : FALSE;
  1752. break;
  1753. case ERROR_SEVERITY_WARNING:
  1754. bPrint = (g_WsbPrintLevel >= WSB_LOG_LEVEL_WARNING) ? TRUE : FALSE;
  1755. break;
  1756. case ERROR_SEVERITY_ERROR:
  1757. bPrint = (g_WsbPrintLevel >= WSB_LOG_LEVEL_ERROR) ? TRUE : FALSE;
  1758. break;
  1759. default:
  1760. bPrint = (g_WsbPrintLevel >= WSB_LOG_LEVEL_COMMENT) ? TRUE : FALSE;
  1761. break;
  1762. }
  1763. WsbAffirm (bPrint, S_OK);
  1764. //
  1765. // Determine source facility of message
  1766. //
  1767. switch ( HRESULT_FACILITY( eventId ) ) {
  1768. case WSB_FACILITY_PLATFORM:
  1769. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1770. break;
  1771. case WSB_FACILITY_RMS:
  1772. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1773. break;
  1774. case WSB_FACILITY_HSMENG:
  1775. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1776. break;
  1777. case WSB_FACILITY_JOB:
  1778. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1779. break;
  1780. case WSB_FACILITY_HSMTSKMGR:
  1781. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1782. break;
  1783. case WSB_FACILITY_FSA:
  1784. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1785. break;
  1786. case WSB_FACILITY_GUI:
  1787. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1788. break;
  1789. case WSB_FACILITY_MOVER:
  1790. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1791. break;
  1792. case WSB_FACILITY_LAUNCH:
  1793. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1794. break;
  1795. case WSB_FACILITY_USERLINK:
  1796. facilityName = WSB_FACILITY_PLATFORM_NAME;
  1797. break;
  1798. case WSB_FACILITY_CLI:
  1799. facilityName = WSB_FACILITY_CLI_NAME;
  1800. break;
  1801. case WSB_FACILITY_TEST:
  1802. facilityName = WSB_FACILITY_TEST_NAME;
  1803. break;
  1804. case HRESULT_FACILITY(FACILITY_NT_BIT):
  1805. facilityName = WSB_FACILITY_NTDLL_NAME;
  1806. eventId &= ~FACILITY_NT_BIT;
  1807. break;
  1808. default:
  1809. facilityName = WSB_FACILITY_NTDLL_NAME;
  1810. break;
  1811. }
  1812. if ( facilityName ) {
  1813. HMODULE hModule;
  1814. hModule = LoadLibraryEx( facilityName, NULL, LOAD_LIBRARY_AS_DATAFILE );
  1815. if (hModule) {
  1816. //
  1817. // Load and format the message
  1818. //
  1819. FormatMessage(FORMAT_MESSAGE_FROM_HMODULE | FORMAT_MESSAGE_ALLOCATE_BUFFER,
  1820. hModule,
  1821. eventId,
  1822. MAKELANGID ( LANG_NEUTRAL, SUBLANG_DEFAULT ),
  1823. (LPTSTR) &messageText,
  1824. 0,
  1825. inserts);
  1826. if ( messageText ) {
  1827. //
  1828. // Print the message
  1829. //
  1830. wprintf( messageText ); // Format messages come with \n
  1831. // TEMPORARY: Should we convert here using WideCharToMultiByte ???
  1832. //
  1833. // Trace the message
  1834. //
  1835. if ( g_pWsbTrace ) {
  1836. WsbTracef( OLESTR("!!!!! PRINT - Event <0x%08lx> is printed\n"), eventId );
  1837. WsbTracef( OLESTR("%ls"), messageText ); // Format messages come with \n
  1838. }
  1839. LocalFree( messageText );
  1840. } else {
  1841. if ( g_pWsbTrace ) {
  1842. WsbTracef( OLESTR("!!!!! PRINT !!!!! - Message <0x%08lx> could not be translated.\r\n"), eventId );
  1843. }
  1844. }
  1845. FreeLibrary(hModule);
  1846. } else {
  1847. WsbTracef( OLESTR("!!!!! EVENT !!!!! - Could not load facility name DLL %ls. \r\n"), facilityName);
  1848. }
  1849. } else {
  1850. if ( g_pWsbTrace ) {
  1851. WsbTracef( OLESTR("!!!!! PRINT !!!!! - Message File for <0x%08lx> could not be found.\r\n"), eventId );
  1852. }
  1853. }
  1854. } WsbCatch( hr );
  1855. }