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.
 
 
 
 
 
 

773 lines
17 KiB

/*++
Copyright (c) 1997 Microsoft Corporation
Module Name:
stilog.cpp
Abstract:
Class to handle file based logging
Author:
Vlad Sadovsky (VladS) 01-Sep-1997
History:
--*/
//
// Include Headers
//
#include "cplusinc.h"
#include "sticomm.h"
//
// Static definitions and variables
//
static const TCHAR szDefaultName[] = TEXT("Sti_Trace.log");
static const TCHAR szDefaultTracerName[] = TEXT("STI");
static const TCHAR szMutexNamePrefix[] = TEXT("StiTraceMutex");
static const TCHAR szColumns[] = TEXT("Severity TracerName [Process::ThreadId] Time MessageText\r\n\r\n");
static const TCHAR szOpenedLog[] = TEXT("\n****** Opened file log at %s %s .Tracer (%s) , called from [%#s::%#lx]\n");
static const TCHAR szClosedLog[] = TEXT("\n******Closed trace log on %s %s Tracer (%s) , called from [%#s::%#lx]\n");
//
// Functions
//
//
inline BOOL
FormatStdTime(
IN const SYSTEMTIME * pstNow,
IN OUT TCHAR * pchBuffer,
IN int cbBuffer
)
{
return ( GetTimeFormat( LOCALE_SYSTEM_DEFAULT,
( LOCALE_NOUSEROVERRIDE | TIME_FORCE24HOURFORMAT|
TIME_NOTIMEMARKER),
pstNow, NULL, pchBuffer, cbBuffer)
!= 0);
} // FormatStdTime()
inline BOOL
FormatStdDate( IN const SYSTEMTIME * pstNow,
IN OUT TCHAR * pchBuffer,
IN int cbBuffer)
{
return ( GetDateFormat( LOCALE_SYSTEM_DEFAULT, LOCALE_NOUSEROVERRIDE,
pstNow, NULL, pchBuffer, cbBuffer)
!= 0);
} // FormatStdDate()
inline TCHAR
TraceFlagToLetter(
IN DWORD dwType
)
{
if (dwType & STI_TRACE_ERROR) {
return TEXT('e');
}
else if (dwType & STI_TRACE_WARNING) {
return TEXT('w');
}
else if (dwType & STI_TRACE_INFORMATION) {
return TEXT('i');
}
else {
return TEXT('t');
}
}
STI_FILE_LOG::STI_FILE_LOG(
IN LPCTSTR lpszTracerName,
IN LPCTSTR lpszLogName,
IN DWORD dwFlags, // = 0
IN HMODULE hMessageModule // =NULL
) :
m_hMutex(INVALID_HANDLE_VALUE)
/*++
Description
Constructor function for given log object.
Arguments:
lpszSource: file name for log. Relative to Windows directory
Note:
--*/
{
LPCTSTR lpActualLogName;
TCHAR szTempName[MAX_PATH];
LPTSTR pszFilePart;
DWORD dwError;
DWORD cbName;
DWORD dwLevel;
DWORD dwMode;
BOOL fTruncate = FALSE;
m_dwSignature = SIGNATURE_FILE_LOG;
lpActualLogName = szDefaultName;
m_hLogWindow = NULL;
m_lWrittenHeader = FALSE;
m_hDefaultMessageModule = hMessageModule ? hMessageModule : GetModuleHandle(NULL);
dwLevel = STI_TRACE_ERROR;
dwMode = STI_TRACE_ADD_TIME | STI_TRACE_ADD_THREAD;
m_hLogFile = INVALID_HANDLE_VALUE;
*m_szLogFilePath = TEXT('\0');
m_dwMaxSize = STI_MAX_LOG_SIZE;
ReportError(NO_ERROR);
//
// Read global settings from the registry
//
RegEntry re(REGSTR_PATH_STICONTROL REGSTR_PATH_LOGGING,HKEY_LOCAL_MACHINE);
if (re.IsValid()) {
m_dwMaxSize = re.GetNumber(REGSTR_VAL_LOG_MAXSIZE,STI_MAX_LOG_SIZE);
//
// If we need to check append flag, read it from the registry
//
if (dwFlags & STIFILELOG_CHECK_TRUNCATE_ON_BOOT) {
fTruncate = re.GetNumber(REGSTR_VAL_LOG_TRUNCATE_ON_BOOT,FALSE);
}
}
//
// Open file for logging
//
if (lpszLogName && *lpszLogName ) {
lpActualLogName = lpszLogName;
}
//
// lpszTracerName is ANSI.
//
if (lpszTracerName && *lpszTracerName ) {
lstrcpyn(m_szTracerName,lpszTracerName,sizeof(m_szTracerName) / sizeof(m_szTracerName[0]));
m_szTracerName[sizeof(m_szTracerName) / sizeof(m_szTracerName[0]) -1] = TEXT('\0');
}
else {
lstrcpy(m_szTracerName,szDefaultTracerName);
}
//
// Approximate process name with name of the executable binary used to create process
//
*szTempName = TEXT('\0');
::GetModuleFileName(NULL,szTempName,sizeof(szTempName) / sizeof(szTempName[0]));
pszFilePart = _tcsrchr(szTempName,TEXT('\\'));
pszFilePart = (pszFilePart && *pszFilePart) ? ::CharNext(pszFilePart) : szTempName;
lstrcpyn(m_szProcessName,pszFilePart,sizeof(m_szProcessName)/sizeof(TCHAR));
m_szProcessName[sizeof(m_szProcessName)/sizeof(TCHAR) - 1] = TEXT('\0');
//
// Read flags for this logger
//
re.MoveToSubKey(lpszTracerName);
if (re.IsValid()) {
dwLevel = re.GetNumber(REGSTR_VAL_LOG_LEVEL,STI_TRACE_ERROR)
& STI_TRACE_MESSAGE_TYPE_MASK;
dwMode = re.GetNumber(REGSTR_VAL_LOG_MODE,STI_TRACE_ADD_THREAD)
& STI_TRACE_MESSAGE_FLAGS_MASK;
}
m_dwReportMode = dwLevel | dwMode;
//
// Open log file
//
cbName = ::GetWindowsDirectory(m_szLogFilePath,sizeof(m_szLogFilePath)/sizeof(m_szLogFilePath[0]));
if (( cbName == 0) || !*m_szLogFilePath ) {
ReportError(::GetLastError());
return;
}
lstrcat(lstrcat(m_szLogFilePath,TEXT("\\")),lpActualLogName);
m_hLogFile = ::CreateFile(m_szLogFilePath,
GENERIC_WRITE,
FILE_SHARE_WRITE | FILE_SHARE_READ,
NULL, // security attributes
OPEN_ALWAYS,
FILE_ATTRIBUTE_NORMAL,
NULL); // template file handle
// if ( m_hLogFile!= INVALID_HANDLE_VALUE) {
if (IS_VALID_HANDLE(m_hLogFile)) {
if(fTruncate) {
::SetFilePointer( m_hLogFile, 0, NULL, FILE_BEGIN );
::SetEndOfFile( m_hLogFile );
}
::SetFilePointer( m_hLogFile, 0, NULL, FILE_END);
lstrcpy(szTempName,szMutexNamePrefix);
lstrcat(szTempName,lpActualLogName);
m_hMutex = ::CreateMutex(NULL,
FALSE,
szTempName
);
if ( !IS_VALID_HANDLE(m_hMutex ) ) {
// ASSERT
}
}
if (!IS_VALID_HANDLE(m_hLogFile) || !IS_VALID_HANDLE(m_hMutex )) {
ReportError(::GetLastError());
}
else {
//
// Success
//
}
} /* STI_FILE_LOG::STI_FILE_LOG() */
STI_FILE_LOG::~STI_FILE_LOG( VOID)
/*++
Description:
Destructor function for given STI_FILE_LOG object.
--*/
{
SYSTEMTIME stCurrentTime;
TCHAR szFmtDate[20];
TCHAR szFmtTime[32];
TCHAR szTextBuffer[128];
if(m_lWrittenHeader) {
GetLocalTime(&stCurrentTime);
FormatStdDate( &stCurrentTime, szFmtDate, 15);
FormatStdTime( &stCurrentTime, szFmtTime, sizeof(szFmtTime) / sizeof(szFmtTime[0]));
::wsprintf(szTextBuffer,szClosedLog,
szFmtDate,szFmtTime,
m_szTracerName,
m_szProcessName,
::GetCurrentThreadId()
);
WriteStringToLog(szTextBuffer);
}
if (IS_VALID_HANDLE(m_hMutex)) {
::CloseHandle(m_hMutex);
m_hMutex = INVALID_HANDLE_VALUE;
}
if (IS_VALID_HANDLE(m_hLogFile)) {
::FlushFileBuffers( m_hLogFile);
::CloseHandle(m_hLogFile);
m_hLogFile = INVALID_HANDLE_VALUE;
}
m_dwSignature = SIGNATURE_FILE_LOG_FREE;
} /* STI_FILE_LOG::~STI_FILE_LOG() */
//
// IUnknown methods. Used only for reference counting
//
STDMETHODIMP
STI_FILE_LOG::QueryInterface( REFIID riid, LPVOID * ppvObj)
{
return E_FAIL;
}
STDMETHODIMP_(ULONG)
STI_FILE_LOG::AddRef( void)
{
::InterlockedIncrement(&m_cRef);
return m_cRef;
}
STDMETHODIMP_(ULONG)
STI_FILE_LOG::Release( void)
{
LONG cNew;
if(!(cNew = ::InterlockedDecrement(&m_cRef))) {
delete this;
}
return cNew;
}
void
STI_FILE_LOG::
WriteLogSessionHeader(
VOID
)
/*++
Description
Arguments:
Note:
--*/
{
SYSTEMTIME stCurrentTime;
TCHAR szFmtDate[32];
TCHAR szFmtTime[32];
TCHAR szTextBuffer[128];
GetLocalTime(&stCurrentTime);
FormatStdDate( &stCurrentTime, szFmtDate, sizeof(szFmtDate) / sizeof(szFmtDate[0]));
FormatStdTime( &stCurrentTime, szFmtTime, sizeof(szFmtTime) / sizeof(szFmtTime[0]));
::wsprintf(szTextBuffer,szOpenedLog,
szFmtDate,szFmtTime,
m_szTracerName,
m_szProcessName,
::GetCurrentThreadId()
);
WriteStringToLog(szTextBuffer);
WriteStringToLog(szColumns);
}
void
STI_FILE_LOG::
ReportMessage(
DWORD dwType,
LPCTSTR pszMsg,
...
)
/*++
Description
Arguments:
Note:
--*/
{
va_list list;
TCHAR *pchBuff = NULL;
DWORD cch;
va_start (list, pszMsg);
pchBuff = NULL;
cch = ::FormatMessage(FORMAT_MESSAGE_MAX_WIDTH_MASK |
FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_FROM_STRING,
pszMsg,
0,
0,
(LPTSTR) &pchBuff,
1024,
&list
);
vReportMessage(dwType,pszMsg,list);
va_end(list);
}
void
STI_FILE_LOG::
ReportMessage(
DWORD dwType,
DWORD idMessage,
...
)
/*++
Description
Arguments:
Note:
--*/
{
va_list list;
TCHAR *pchBuff = NULL;
DWORD cch;
va_start (list, idMessage);
//
// Read message and add inserts
//
pchBuff = NULL;
cch = ::FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER |
FORMAT_MESSAGE_MAX_WIDTH_MASK |
FORMAT_MESSAGE_FROM_HMODULE,
m_hDefaultMessageModule ,
idMessage,
0,
(LPTSTR) &pchBuff,
1024,
&list
);
if (pchBuff && cch) {
vReportMessage(dwType,pchBuff,list);
LocalFree(pchBuff);
}
va_end(list);
}
void
STI_FILE_LOG::
vReportMessage(
DWORD dwType,
LPCTSTR pszMsg,
va_list arglist
)
/*++
Description
Arguments:
Note:
--*/
{
TCHAR achTextBuffer[1024];
DWORD dwReportMode;
if ((QueryError() != NOERROR) ||
(!m_hLogFile || m_hLogFile == INVALID_HANDLE_VALUE)) {
return;
}
// Do we need to display this message ?
if (!((dwType & m_dwReportMode) & STI_TRACE_MESSAGE_TYPE_MASK)) {
return;
}
// Start message with type
*achTextBuffer = TraceFlagToLetter(dwType);
*(achTextBuffer+1) = TEXT(' ');
*(achTextBuffer+2) = TEXT('\0');
// Add name of tracer source
lstrcat(lstrcat(achTextBuffer,m_szTracerName),TEXT(" "));;
dwReportMode = m_dwReportMode | (dwType & STI_TRACE_MESSAGE_FLAGS_MASK);
//
// Prepare header if needed
//
if (dwReportMode & STI_TRACE_MESSAGE_FLAGS_MASK ) {
if (dwReportMode & STI_TRACE_ADD_THREAD ) {
//
// Add process::thread ID
//
TCHAR szThreadId[40];
::wsprintf(szThreadId,TEXT("[%#s::%#lx] "),m_szProcessName,::GetCurrentThreadId());
::lstrcat(::lstrcat(achTextBuffer,szThreadId),TEXT(" "));
}
if (dwReportMode & STI_TRACE_ADD_TIME ) {
// Add current time
SYSTEMTIME stCurrentTime;
TCHAR szFmtTime[32];
*szFmtTime = TEXT('\0');
::GetLocalTime(&stCurrentTime);
FormatStdTime( &stCurrentTime, szFmtTime, 15);
lstrcat(lstrcat(achTextBuffer,szFmtTime),TEXT(" "));;
}
}
lstrcat(achTextBuffer,TEXT("\t"));
//
// Now add message text itself
//
::wvsprintf(achTextBuffer+::lstrlen(achTextBuffer), pszMsg, arglist);
::lstrcat(achTextBuffer , TEXT("\r\n"));
//
// Write to the file, flushing buffers if message type is ERROR
//
WriteStringToLog(achTextBuffer,(dwType & STI_TRACE_MESSAGE_TYPE_MASK) & STI_TRACE_ERROR);
}
VOID
STI_FILE_LOG::
WriteStringToLog(
LPCTSTR pszTextBuffer,
BOOL fFlush // = FALSE
)
{
DWORD dwcbWritten;
LONG lHeaderWrittenFlag;
//
// Integrity check. Make sure the mutex and file handles are valid - if not,
// then bail.
//
if (!(IS_VALID_HANDLE(m_hMutex) && IS_VALID_HANDLE(m_hLogFile))) {
#ifdef DEBUG
//OutputDebugString(TEXT("STILOG File or Mutex handle is invalid. "));
#endif
return ;
}
//
// If had not done yet, write session header here. Note this is recursive call
// and flag marking header should be set first.
//
lHeaderWrittenFlag = InterlockedExchange(&m_lWrittenHeader,1L);
if (!lHeaderWrittenFlag) {
WriteLogSessionHeader();
}
TAKE_MUTEX t(m_hMutex);
//
// Check that log file size is not exceeding the limit. Assuming log file size is set to fit
// into 32bit field, so we don't bother looking at high dword.
//
// Nb: We are not saving backup copy of the log, expecting that it never grows that large
//
BY_HANDLE_FILE_INFORMATION fi;
if (!GetFileInformationByHandle(m_hLogFile,&fi)) {
#ifdef DEBUG
OutputDebugString(TEXT("STILOG could not get file size for log file. "));
#endif
return ;
}
if ( fi.nFileSizeHigh !=0 || (fi.nFileSizeLow > m_dwMaxSize) ){
::SetFilePointer( m_hLogFile, 0, NULL, FILE_BEGIN );
::SetEndOfFile( m_hLogFile );
::GetFileInformationByHandle(m_hLogFile,&fi);
}
#ifdef USE_FILE_LOCK
::LockFile(m_hLogFile,fi.nFileSizeLow,fi.nFileSizeHigh,4096,0);
#endif
::SetFilePointer( m_hLogFile, 0, NULL, FILE_END);
#ifdef _UNICODE
UINT len = lstrlen(pszTextBuffer);
CHAR *ansiBuffer = (CHAR *)alloca(len + 2);
if(ansiBuffer) {
::WideCharToMultiByte(CP_ACP,
0,
pszTextBuffer,
-1,
ansiBuffer,
len + 1,
NULL,
NULL);
::WriteFile(m_hLogFile,
ansiBuffer,
len,
&dwcbWritten,
NULL);
}
#else
::WriteFile(m_hLogFile,
pszTextBuffer,
lstrlen(pszTextBuffer),
&dwcbWritten,
NULL);
#endif
#ifdef USE_FILE_LOCK
::UnlockFile(m_hLogFile,fi.nFileSizeLow,fi.nFileSizeHigh,4096,0);
#endif
if (fFlush) {
// Flush buffers to disk
FlushFileBuffers(m_hLogFile);
}
if(QueryReportMode() & STI_TRACE_LOG_TOUI) {
ULONG_PTR dwResult;
SendMessageTimeout(m_hLogWindow,
STIMON_MSG_LOG_MESSAGE,
0,
(LPARAM)pszTextBuffer,
0,
100,
&dwResult
);
}
#ifdef DEBUG
::OutputDebugString(pszTextBuffer);
#endif
}
//
// C-APIs
//
HANDLE
WINAPI
CreateStiFileLog(
IN LPCTSTR lpszTracerName,
IN LPCTSTR lpszLogName,
IN DWORD dwReportMode
)
/*++
Description
Arguments:
Note:
--*/
{
HANDLE hRet = INVALID_HANDLE_VALUE;
STI_FILE_LOG* pStiFileLog = NULL;
pStiFileLog = new STI_FILE_LOG(lpszTracerName,lpszLogName);
if(pStiFileLog){
if (pStiFileLog->IsValid()) {
hRet = static_cast<HANDLE>(pStiFileLog);
pStiFileLog->SetReportMode(pStiFileLog->QueryReportMode() | dwReportMode);
} else {
//
// Notice that we delete this object rather than calling
// CloseStiFileLog. We do this because it the filelog is
// not valid (maybe it had an internal creation error),
// CloseStiFileLog won't try to delete it, hence we delete
// it here.
//
delete pStiFileLog;
pStiFileLog = NULL;
hRet = INVALID_HANDLE_VALUE;
}
} else {
::SetLastError(ERROR_NOT_ENOUGH_MEMORY);
}
return hRet;
}
DWORD
WINAPI
CloseStiFileLog(
IN HANDLE hFileLog
)
{
STI_FILE_LOG* pStiFileLog = NULL;
pStiFileLog = static_cast<STI_FILE_LOG*>(hFileLog);
if (IsBadWritePtr(pStiFileLog,sizeof(STI_FILE_LOG)) ||
!pStiFileLog->IsValid()) {
::SetLastError(ERROR_INVALID_HANDLE);
return ERROR_INVALID_HANDLE;
}
delete pStiFileLog;
return NOERROR;
}
DWORD
WINAPI
ReportStiLogMessage(
IN HANDLE hFileLog,
IN DWORD dwType,
IN LPCTSTR psz,
...
)
{
STI_FILE_LOG* pStiFileLog = NULL;
pStiFileLog = static_cast<STI_FILE_LOG*>(hFileLog);
if (IsBadWritePtr(pStiFileLog,sizeof(STI_FILE_LOG)) ||
!pStiFileLog->IsValid()) {
::SetLastError(ERROR_INVALID_HANDLE);
return ERROR_INVALID_HANDLE;
}
va_list list;
va_start (list, psz);
pStiFileLog->vReportMessage(dwType,psz,list);
va_end(list);
return NOERROR;
}
/********************************* End of File ***************************/