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

597 lines
13 KiB
C++

/******************************************************************************
Copyright (c) 1999 Microsoft Corporation
Module Name:
Logging.cpp
Abstract:
This file contains the implementation of a set of logging classes.
Revision History:
Davide Massarenti (Dmassare) 05/27/99
created
******************************************************************************/
#include "stdafx.h"
#define BUFFER_LINE_LENGTH (1024)
static WCHAR l_EndOfLine[] = L"\n";
/////////////////////////////////////////////////////////////////////////////
static DATE GetMidnight( /*[in]*/ SYSTEMTIME stTime )
{
DATE dTime;
stTime.wHour = 0;
stTime.wMinute = 0;
stTime.wSecond = 0;
stTime.wMilliseconds = 0;
::SystemTimeToVariantTime( &stTime, &dTime );
return dTime;
}
/////////////////////////////////////////////////////////////////////////////
MPC::FileLog::FileLog( /*[in]*/ bool fCacheHandle, /*[in]*/ bool fUseUnicode )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::FileLog" );
// MPC::wstring m_szLogFile;
m_hFile = INVALID_HANDLE_VALUE; // HANDLE m_hFile;
m_fCacheHandle = fCacheHandle; // bool m_fCacheHandle;
m_fUseUnicode = fUseUnicode; // bool m_fUseUnicode;
}
MPC::FileLog::FileLog( /*[in]*/ const FileLog& fl )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::FileLog" );
m_szLogFile = fl.m_szLogFile; // MPC::wstring m_szLogFile;
m_hFile = INVALID_HANDLE_VALUE; // HANDLE m_hFile;
m_fCacheHandle = fl.m_fCacheHandle; // bool m_fCacheHandle;
m_fUseUnicode = fl.m_fUseUnicode; // bool m_fUseUnicode;
}
MPC::FileLog::~FileLog()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::~FileLog" );
Close();
}
MPC::FileLog& MPC::FileLog::operator=( /*[in]*/ const FileLog& fl )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::operator=" );
Close();
m_szLogFile = fl.m_szLogFile; // MPC::wstring m_szLogFile;
m_hFile = INVALID_HANDLE_VALUE; // HANDLE m_hFile;
m_fCacheHandle = fl.m_fCacheHandle; // bool m_fCacheHandle;
m_fUseUnicode = fl.m_fUseUnicode; // bool m_fUseUnicode;
return *this;
}
/////////////////////////////////////////////////////////////////////////////
HRESULT MPC::FileLog::Open()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::Open" );
HRESULT hr;
Lock();
if(m_hFile == INVALID_HANDLE_VALUE)
{
// Ensure the directory exists.
__MPC_EXIT_IF_METHOD_FAILS(hr, MPC::MakeDir( m_szLogFile ));
__MPC_EXIT_IF_INVALID_HANDLE(hr, m_hFile, ::CreateFileW( m_szLogFile.c_str(), GENERIC_WRITE, FILE_SHARE_READ, NULL, OPEN_ALWAYS, FILE_FLAG_SEQUENTIAL_SCAN, NULL ));
::SetFilePointer( m_hFile, 0, NULL, FILE_END );
}
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::Close()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::Close" );
HRESULT hr;
Lock();
if(m_hFile != INVALID_HANDLE_VALUE)
{
::CloseHandle( m_hFile );
m_hFile = INVALID_HANDLE_VALUE;
}
hr = S_OK;
Unlock();
__MPC_FUNC_EXIT(hr);
}
/////////////////////////////////////////////////////////////////////////////
HRESULT MPC::FileLog::Rotate( /*[in]*/ DWORD dwDays )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::Rotate" );
HRESULT hr;
SYSTEMTIME st;
WCHAR rgTime[BUFFER_LINE_LENGTH];
MPC::wstring szLogFileNew;
Lock();
//
// Before rotating, check if it's time to do it.
//
if(dwDays)
{
FILETIME ftCreation;
SYSTEMTIME stCreation;
SYSTEMTIME stNow;
DATE dCreation;
DATE dNow;
__MPC_EXIT_IF_METHOD_FAILS(hr, Open());
if(::GetFileTime( m_hFile, &ftCreation, NULL, NULL ))
{
::FileTimeToSystemTime( &ftCreation, &stCreation );
::GetSystemTime ( &stNow );
dCreation = GetMidnight( stCreation );
dNow = GetMidnight( stNow );
//
// If it's not been 'dwDays' since the creation of the log, don't rotate.
//
if(dCreation + (DATE)dwDays > dNow)
{
__MPC_SET_ERROR_AND_EXIT(hr, S_OK);
}
}
}
__MPC_EXIT_IF_METHOD_FAILS(hr, Close());
//
// Append current time.
//
// <FileName>__<Year>_<Month>_<Day>_<hour>-<minute>-<second>
//
::GetLocalTime( &st );
StringCchPrintfW( rgTime, ARRAYSIZE(rgTime), L"__%04u-%02u-%02u_%02u-%02u-%02u", st.wYear, st.wMonth, st.wDay, st.wHour, st.wMinute, st.wSecond );
szLogFileNew = m_szLogFile;
szLogFileNew.append( rgTime );
__MPC_EXIT_IF_CALL_RETURNS_FALSE(hr, ::MoveFileExW( m_szLogFile.c_str(), szLogFileNew.c_str(), MOVEFILE_REPLACE_EXISTING ));
//
// After rotation, SET the date of creation. There's a BUG in NTFS that caches the date from the previous file...
//
if(dwDays)
{
FILETIME ftNow;
SYSTEMTIME stNow;
__MPC_EXIT_IF_METHOD_FAILS(hr, Open());
::GetSystemTime ( &stNow );
::SystemTimeToFileTime( &stNow, &ftNow );
::SetFileTime( m_hFile, &ftNow, NULL, NULL );
__MPC_EXIT_IF_METHOD_FAILS(hr, Close());
}
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::SetLocation( /*[in]*/ LPCWSTR szLogFile )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::SetLocation" );
HRESULT hr;
Lock();
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_STRING_NOT_EMPTY(szLogFile);
__MPC_PARAMCHECK_END();
__MPC_EXIT_IF_METHOD_FAILS(hr, Close());
m_szLogFile = szLogFile;
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::Terminate()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::Terminate" );
HRESULT hr;
__MPC_EXIT_IF_METHOD_FAILS(hr, Close());
hr = S_OK;
__MPC_FUNC_CLEANUP;
__MPC_FUNC_EXIT(hr);
}
/////////////////////////////////////////////////////////////////////////////
HRESULT MPC::FileLog::AppendString( /*[in]*/ LPCWSTR szLine )
{
_ASSERT(m_hFile != INVALID_HANDLE_VALUE);
USES_CONVERSION;
HRESULT hr;
DWORD dwWritten;
LPCVOID lpData;
DWORD dwSize;
if(m_fUseUnicode)
{
lpData = szLine;
dwSize = wcslen( szLine ) * sizeof(WCHAR);
}
else
{
LPCSTR szLineASCII = W2A( szLine );
lpData = szLineASCII;
dwSize = strlen( szLineASCII ) * sizeof(CHAR);
}
if(::WriteFile( m_hFile, lpData, dwSize, &dwWritten, NULL ) == FALSE)
{
hr = HRESULT_FROM_WIN32( ::GetLastError() );
}
else
{
hr = S_OK;
}
return hr;
}
HRESULT MPC::FileLog::WriteEntry( /*[in]*/ LPWSTR szLine )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::WriteEntry" );
HRESULT hr;
WCHAR rgTime[BUFFER_LINE_LENGTH];
SYSTEMTIME st;
Lock();
__MPC_EXIT_IF_METHOD_FAILS(hr, Open());
//
// Prepend current time.
//
::GetLocalTime( &st );
StringCchPrintfW( rgTime, ARRAYSIZE(rgTime), L"%04u/%02u/%02u %02u:%02u:%02u ", st.wYear, st.wMonth, st.wDay, st.wHour, st.wMinute, st.wSecond );
while(1)
{
LPWSTR szEndOfLine = wcschr( szLine, '\n' );
if(szEndOfLine) *szEndOfLine++ = 0;
__MPC_EXIT_IF_METHOD_FAILS(hr, AppendString( rgTime ));
__MPC_EXIT_IF_METHOD_FAILS(hr, AppendString( szLine ));
__MPC_EXIT_IF_METHOD_FAILS(hr, AppendString( L"\n" ));
//
// Two cases to stop: end of string or NewLine at the end of it.
//
if(!szEndOfLine || !szEndOfLine[0]) break;
szLine = szEndOfLine;
}
hr = S_OK;
__MPC_FUNC_CLEANUP;
if(m_fCacheHandle == false) (void)Close();
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::LogRecordV( /*[in]*/ LPCWSTR szFormat ,
/*[in]*/ va_list arglist )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::WriteEntry" );
HRESULT hr;
WCHAR rgLine[BUFFER_LINE_LENGTH];
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_STRING_NOT_EMPTY(szFormat);
__MPC_PARAMCHECK_END();
StringCchVPrintfW( rgLine, ARRAYSIZE(rgLine), szFormat, arglist );
__MPC_EXIT_IF_METHOD_FAILS(hr, WriteEntry( rgLine ));
hr = S_OK;
__MPC_FUNC_CLEANUP;
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::LogRecordV( /*[in]*/ LPCSTR szFormat ,
/*[in]*/ va_list arglist )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::FileLog::WriteEntry" );
USES_CONVERSION;
HRESULT hr;
CHAR rgLine[BUFFER_LINE_LENGTH];
__MPC_PARAMCHECK_BEGIN(hr)
__MPC_PARAMCHECK_STRING_NOT_EMPTY(szFormat);
__MPC_PARAMCHECK_END();
StringCchVPrintfA( rgLine, ARRAYSIZE(rgLine), szFormat, arglist );
__MPC_EXIT_IF_METHOD_FAILS(hr, WriteEntry( A2W(rgLine) ));
hr = S_OK;
__MPC_FUNC_CLEANUP;
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::FileLog::LogRecord( /*[in]*/ LPCWSTR szFormat,
/*[in]*/ ... )
{
va_list arglist;
va_start( arglist, szFormat );
return LogRecordV( szFormat, arglist );
}
HRESULT MPC::FileLog::LogRecord( /*[in]*/ LPCSTR szFormat,
/*[in]*/ ... )
{
va_list arglist;
va_start( arglist, szFormat );
return LogRecordV( szFormat, arglist );
}
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////
MPC::NTEvent::NTEvent()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::NTEvent" );
m_hEventSource = INVALID_HANDLE_VALUE;
}
MPC::NTEvent::NTEvent( /*[in]*/ const NTEvent& ne )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::NTEvent" );
m_hEventSource = INVALID_HANDLE_VALUE;
}
MPC::NTEvent::~NTEvent()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::~NTEvent" );
Terminate();
}
MPC::NTEvent& MPC::NTEvent::operator=( /*[in]*/ const NTEvent& fl )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::operator=" );
m_hEventSource = INVALID_HANDLE_VALUE;
return *this;
}
/////////////////////////////////////////////////////////////////////////////
HRESULT MPC::NTEvent::Init( /*[in]*/ LPCWSTR szEventSourceName )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::Init" );
HRESULT hr;
Lock();
//
// Validate params.
//
if(szEventSourceName == NULL)
{
return E_INVALIDARG;
}
if(m_hEventSource != INVALID_HANDLE_VALUE)
{
// only allow one init per lifetime of the object...
__MPC_SET_WIN32_ERROR_AND_EXIT(hr, ERROR_ALREADY_ASSIGNED);
}
__MPC_EXIT_IF_INVALID_HANDLE(hr, m_hEventSource, ::RegisterEventSourceW( NULL, szEventSourceName ));
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::NTEvent::Terminate()
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::Terminate" );
HRESULT hr;
Lock();
if(m_hEventSource != INVALID_HANDLE_VALUE)
{
__MPC_EXIT_IF_CALL_RETURNS_FALSE(hr, ::DeregisterEventSource( m_hEventSource ));
m_hEventSource = NULL;
}
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}
HRESULT MPC::NTEvent::LogEvent( /*[in]*/ WORD wEventType ,
/*[in]*/ DWORD dwEventID ,
/*[in]*/ ... )
{
__MPC_FUNC_ENTRY( COMMONID, "MPC::NTEvent::LogEvent" );
HRESULT hr;
va_list arglist;
LPCWSTR szParams;
LPCWSTR* pParams = NULL;
DWORD dwParams = 0;
int i;
Lock();
if(m_hEventSource == INVALID_HANDLE_VALUE)
{
__MPC_SET_ERROR_AND_EXIT(hr, E_FAIL);
}
//
// Walk through the parameters twice, the first time to count them, the second time to collect them.
//
for(i=0;i<2;i++)
{
va_start( arglist, dwEventID );
dwParams = 0;
while((szParams = va_arg( arglist, LPCWSTR )) != NULL)
{
if(pParams) pParams[dwParams] = szParams;
dwParams++;
}
va_end( arglist );
if(i == 0 && dwParams)
{
pParams = (LPCWSTR*)_alloca( sizeof(*pParams) * dwParams );
}
}
__MPC_EXIT_IF_CALL_RETURNS_FALSE(hr, ::ReportEventW( m_hEventSource, wEventType, 0, dwEventID, NULL, dwParams, 0, pParams, NULL ));
hr = S_OK;
__MPC_FUNC_CLEANUP;
Unlock();
__MPC_FUNC_EXIT(hr);
}