|
|
/*++
Copyright (c) 1991 Microsoft Corporation
Module Name:
common.c
Abstract:
Utility routines used by Lodctr and/or UnLodCtr
Author:
Bob Watson (a-robw) 12 Feb 93
Revision History:
--*/ #define UNICODE 1
#define _UNICODE 1
//
// "C" Include files
//
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <malloc.h>
#include <locale.h>
//
// Windows Include files
//
#include <windows.h>
#include <winperf.h>
#include <tchar.h>
#include <initguid.h>
#include <guiddef.h>
#include "wmistr.h"
#include "evntrace.h"
//
// local include files
//
#define _INIT_WINPERFP_
#include "winperfp.h"
#include "common.h"
//
//
// Text string Constant definitions
//
LPCTSTR NamesKey = (LPCTSTR)TEXT("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Perflib"); LPCTSTR DefaultLangId = (LPCTSTR)TEXT("009"); LPCTSTR DefaultLangTag = (LPCTSTR)TEXT("000"); LPCTSTR Counters = (LPCTSTR)TEXT("Counters"); LPCTSTR Help = (LPCTSTR)TEXT("Help"); LPCTSTR VersionStr = (LPCTSTR)TEXT("Version"); LPCTSTR LastHelp = (LPCTSTR)TEXT("Last Help"); LPCTSTR LastCounter = (LPCTSTR)TEXT("Last Counter"); LPCTSTR FirstHelp = (LPCTSTR)TEXT("First Help"); LPCTSTR cszFirstCounter = (LPCTSTR)TEXT("First Counter"); LPCTSTR Busy = (LPCTSTR)TEXT("Updating"); LPCTSTR Slash = (LPCTSTR)TEXT("\\"); LPCTSTR BlankString = (LPCTSTR)TEXT(" "); LPCTSTR DriverPathRoot = (LPCTSTR)TEXT("SYSTEM\\CurrentControlSet\\Services"); LPCTSTR Performance = (LPCTSTR)TEXT("Performance"); LPCTSTR CounterNameStr = (LPCTSTR)TEXT("Counter "); LPCTSTR HelpNameStr = (LPCTSTR)TEXT("Explain "); LPCTSTR AddCounterNameStr = (LPCTSTR)TEXT("Addcounter "); LPCTSTR AddHelpNameStr = (LPCTSTR)TEXT("Addexplain "); LPCTSTR szObjectList = (LPCTSTR)TEXT("Object List"); LPCTSTR szLibraryValidationCode = (LPCTSTR)TEXT("Library Validation Code");
BOOLEAN g_bCheckTraceLevel = FALSE;
// Global (to this module) Buffers
//
static TCHAR DisplayStringBuffer[DISP_BUFF_SIZE]; static TCHAR TextFormat[DISP_BUFF_SIZE]; static HANDLE hMod = NULL; // process handle
static DWORD dwLastError = ERROR_SUCCESS;
HANDLE hEventLog = NULL; HANDLE hLoadPerfMutex = NULL; //
// local static data
//
static TCHAR cDoubleQuote = TEXT('\"');
BOOL __stdcall DllEntryPoint( IN HANDLE DLLHandle, IN DWORD Reason, IN LPVOID ReservedAndUnused ) { BOOL bReturn = FALSE;
ReservedAndUnused;
DisableThreadLibraryCalls (DLLHandle);
switch(Reason) { case DLL_PROCESS_ATTACH: setlocale(LC_ALL, ".OCP");
hMod = DLLHandle; // use DLL handle , not APP handle
// register eventlog source
hEventLog = RegisterEventSourceW ( NULL, (LPCWSTR)L"LoadPerf");
bReturn = TRUE; break;
case DLL_PROCESS_DETACH: if (hEventLog != NULL) { if (DeregisterEventSource(hEventLog)) { hEventLog = NULL; } } if (hLoadPerfMutex != NULL) { CloseHandle(hLoadPerfMutex); hLoadPerfMutex = NULL; } bReturn = TRUE; break;
case DLL_THREAD_ATTACH: case DLL_THREAD_DETACH: bReturn = TRUE; break; }
return bReturn; }
LPCTSTR GetStringResource ( UINT wStringId ) /*++
Retrived UNICODE strings from the resource file for display
--*/ {
if (!hMod) { hMod = (HINSTANCE)GetModuleHandle(NULL); // get instance ID of this module;
} if (hMod) { if ((LoadString(hMod, wStringId, DisplayStringBuffer, DISP_BUFF_SIZE)) > 0) { return (LPTSTR)&DisplayStringBuffer[0]; } else { dwLastError = GetLastError(); return BlankString; } } else { return BlankString; } } LPCWSTR GetFormatResource ( UINT wStringId ) /*++
Returns an ANSI string for use as a format string in a printf fn.
--*/ {
if (!hMod) { hMod = (HINSTANCE)GetModuleHandle(NULL); // get instance ID of this module;
} if (hMod) { if ((LoadStringW(hMod, wStringId, TextFormat, DISP_BUFF_SIZE)) > 0) { return (LPCTSTR)&TextFormat[0]; } else { dwLastError = GetLastError(); return BlankString; } } else { return BlankString; } }
VOID DisplayCommandHelp ( UINT iFirstLine, UINT iLastLine ) /*++
DisplayCommandHelp
displays usage of command line arguments
Arguments
NONE
Return Value
NONE
--*/ { UINT iThisLine; WCHAR StringBuffer[DISP_BUFF_SIZE]; CHAR OemStringBuffer[DISP_BUFF_SIZE]; int nStringBufferLen; int nOemStringBufferLen;
// clear the buffer
memset (StringBuffer, 0, sizeof(StringBuffer));
if (!hMod) { hMod = (HINSTANCE)GetModuleHandle(NULL); // get instance ID of this module;
} if (hMod) { for (iThisLine = iFirstLine; iThisLine <= iLastLine; iThisLine++) { nStringBufferLen = LoadStringW(hMod, iThisLine, StringBuffer, DISP_BUFF_SIZE); if (nStringBufferLen > 0) { nOemStringBufferLen = sizeof(OemStringBuffer) / sizeof(OemStringBuffer[0]); WideCharToMultiByte (CP_OEMCP, 0, StringBuffer, // string to convert
nStringBufferLen +1, // string length + null
OemStringBuffer, nOemStringBufferLen, NULL, NULL);
fprintf (stdout, "\n%s", OemStringBuffer); } } } // else do nothing
} // DisplayCommandHelp
BOOL TrimSpaces ( IN OUT LPTSTR szString ) /*++
Routine Description:
Trims leading and trailing spaces from szString argument, modifying the buffer passed in
Arguments:
IN OUT LPTSTR szString buffer to process
Return Value:
TRUE if string was modified FALSE if not
--*/ { LPTSTR szSource; LPTSTR szDest; LPTSTR szLast; BOOL bChars;
szLast = szSource = szDest = szString; bChars = FALSE;
while (*szSource != 0) { // skip leading non-space chars
if (!_istspace(*szSource)) { szLast = szDest; bChars = TRUE; } if (bChars) { // remember last non-space character
// copy source to destination & increment both
*szDest++ = *szSource++; } else { szSource++; } }
if (bChars) { *++szLast = 0; // terminate after last non-space char
} else { // string was all spaces so return an empty (0-len) string
*szString = 0; }
return (szLast != szSource); }
BOOL IsDelimiter ( IN TCHAR cChar, IN TCHAR cDelimiter ) /*++
Routine Description:
compares the characte to the delimiter. If the delimiter is a whitespace character then any whitespace char will match otherwise an exact match is required --*/ { if (_istspace(cDelimiter)) { // delimiter is whitespace so any whitespace char will do
return (_istspace(cChar)); } else { // delimiter is not white space so use an exact match
return (cChar == cDelimiter); } }
LPCTSTR GetItemFromString ( IN LPCTSTR szEntry, IN DWORD dwItem, IN TCHAR cDelimiter
) /*++
Routine Description:
returns nth item from a list delimited by the cDelimiter Char. Leaves (double)quoted strings intact.
Arguments:
IN LPCTSTR szEntry Source string returned to parse
IN DWORD dwItem 1-based index indicating which item to return. (i.e. 1= first item in list, 2= second, etc.)
IN TCHAR cDelimiter character used to separate items. Note if cDelimiter is WhiteSpace (e.g. a tab or a space) then any white space will serve as a delim.
Return Value:
pointer to buffer containing desired entry in string. Note, this routine may only be called 4 times before the string buffer is re-used. (i.e. don't use this function more than 4 times in single function call!!)
--*/ { static TCHAR szReturnBuffer[4][MAX_PATH]; static LONG dwBuff; LPTSTR szSource, szDest; DWORD dwThisItem; DWORD dwStrLeft;
dwBuff = ++dwBuff % 4; // wrap buffer index
szSource = (LPTSTR)szEntry; szDest = &szReturnBuffer[dwBuff][0];
// clear previous contents
memset (szDest, 0, (MAX_PATH * sizeof(TCHAR)));
// find desired entry in string
dwThisItem = 1; while (dwThisItem < dwItem) { if (*szSource != 0) { while (!IsDelimiter(*szSource, cDelimiter) && (*szSource != 0)) { if (*szSource == cDoubleQuote) { // if this is a quote, then go to the close quote
szSource++; while ((*szSource != cDoubleQuote) && (*szSource != 0)) szSource++; } if (*szSource != 0) szSource++; } } dwThisItem++; if (*szSource != 0) szSource++; } // copy this entry to the return buffer
if (*szSource != 0) { dwStrLeft = MAX_PATH-1; while (!IsDelimiter(*szSource, cDelimiter) && (*szSource != 0)) { if (*szSource == cDoubleQuote) { // if this is a quote, then go to the close quote
// don't copy quotes!
szSource++; while ((*szSource != cDoubleQuote) && (*szSource != 0)) { *szDest++ = *szSource++; dwStrLeft--; if (!dwStrLeft) break; // dest is full (except for term NULL
} if (*szSource != 0) szSource++; } else { *szDest++ = *szSource++; dwStrLeft--; if (!dwStrLeft) break; // dest is full (except for term NULL
} } *szDest = 0; }
// remove any leading and/or trailing spaces
TrimSpaces (&szReturnBuffer[dwBuff][0]);
return &szReturnBuffer[dwBuff][0]; }
void ReportLoadPerfEvent( IN WORD EventType, IN DWORD EventID, IN DWORD dwDataCount, IN DWORD dwData1, IN DWORD dwData2, IN DWORD dwData3, IN WORD wStringCount, IN LPWSTR szString1, IN LPWSTR szString2, IN LPWSTR szString3 ) { WCHAR szDbg[1024]; DWORD dwData[4]; LPWSTR szMessageArray[4]; BOOL bResult = FALSE; DWORD dwLastError = GetLastError();
if (dwDataCount > 3) dwDataCount = 3; if (wStringCount > 3) wStringCount = 3;
if (dwDataCount > 0) dwData[0] = dwData1; if (dwDataCount > 1) dwData[1] = dwData2; if (dwDataCount > 2) dwData[2] = dwData3; dwDataCount *= sizeof(DWORD);
if (wStringCount > 0 && szString1) szMessageArray[0] = szString1; if (wStringCount > 1 && szString2) szMessageArray[1] = szString2; if (wStringCount > 2 && szString3) szMessageArray[2] = szString3;
if (hEventLog == NULL) { hEventLog = RegisterEventSourceW (NULL, (LPCWSTR)L"LoadPerf"); }
if (dwDataCount > 0 && wStringCount > 0) { bResult = ReportEventW(hEventLog, EventType, // event type
0, // category (not used)
EventID, // event,
NULL, // SID (not used),
wStringCount, // number of strings
dwDataCount, // sizeof raw data
szMessageArray, // message text array
(LPVOID) & dwData[0]); // raw data
} else if (dwDataCount > 0) { bResult = ReportEventW(hEventLog, EventType, // event type
0, // category (not used)
EventID, // event,
NULL, // SID (not used),
0, // number of strings
dwDataCount, // sizeof raw data
NULL, // message text array
(LPVOID) & dwData[0]); // raw data
} else if (wStringCount > 0) { bResult = ReportEventW(hEventLog, EventType, // event type
0, // category (not used)
EventID, // event,
NULL, // SID (not used),
wStringCount, // number of strings
0, // sizeof raw data
szMessageArray, // message text array
NULL); // raw data
} else { bResult = ReportEventW(hEventLog, EventType, // event type
0, // category (not used)
EventID, // event,
NULL, // SID (not used),
0, // number of strings
0, // sizeof raw data
NULL, // message text array
NULL); // raw data
}
if (! bResult) { swprintf(szDbg, (LPCWSTR) L"LOADPERF(0x%08X)::(%d,0x%08X,%d)(%d,%d,%d,%d)(%d,%ws,%ws,%ws)\n", GetCurrentThreadId(), EventType, EventID, GetLastError(), dwDataCount, dwData1, dwData2, dwData3, wStringCount, szString1, szString2, szString3); OutputDebugString(szDbg); }
SetLastError(dwLastError); }
BOOLEAN LoadPerfGrabMutex() { BOOLEAN bResult = TRUE; HANDLE hLocalMutex = NULL; DWORD dwWaitStatus = 0;
if (hLoadPerfMutex == NULL) { hLocalMutex = CreateMutex(NULL, TRUE, TEXT("LOADPERF_MUTEX")); if (hLocalMutex == NULL) { bResult = FALSE; } else { InterlockedCompareExchangePointer(& hLoadPerfMutex, hLocalMutex, NULL); } }
if (hLocalMutex == NULL) { dwWaitStatus = WaitForSingleObject(hLoadPerfMutex, H_MUTEX_TIMEOUT); if (dwWaitStatus != WAIT_OBJECT_0 && dwWaitStatus != WAIT_ABANDONED) { SetLastError(dwWaitStatus); bResult = FALSE; } }
if (! bResult) { TRACE((WINPERF_DBG_TRACE_FATAL), (& LoadPerfGuid, __LINE__, LOADPERF_LOADPERFGRABMUTEX, 0, GetLastError(), NULL)); }
return bResult; }
|