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

332 lines
9.6 KiB
C

#include <windows.h>
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <pdh.h>
#include <pdhp.h>
#include "pdhidef.h"
#include "log_bin.h"
#include "log_wmi.h"
#include "log_text.h"
#include "log_sql.h"
#include "strings.h"
#include "pdhmsg.h"
LPWSTR
PlaiStringDuplicate( LPWSTR sz );
BOOL __stdcall
IsValidLogHandle (
IN HLOG hLog
);
PDH_FUNCTION
PdhiWriteRelogRecord(
IN PPDHI_LOG pLog,
IN SYSTEMTIME *st
)
{
PDH_STATUS pdhStatus = ERROR_SUCCESS;
LPWSTR szUserString = NULL;
pdhStatus = WAIT_FOR_AND_LOCK_MUTEX (pLog->hLogMutex);
if (pdhStatus == ERROR_SUCCESS) {
switch (LOWORD(pLog->dwLogFormat)) {
case PDH_LOG_TYPE_CSV:
case PDH_LOG_TYPE_TSV:
pdhStatus =PdhiWriteTextLogRecord (
pLog,
st,
(LPCWSTR)szUserString);
break;
case PDH_LOG_TYPE_RETIRED_BIN:
pdhStatus = PDH_NOT_IMPLEMENTED;
break;
case PDH_LOG_TYPE_BINARY:
pdhStatus = PdhiWriteWmiLogRecord(
pLog,
st,
(LPCWSTR) szUserString);
break;
case PDH_LOG_TYPE_SQL:
pdhStatus =PdhiWriteSQLLogRecord (
pLog,
st,
(LPCWSTR)szUserString);
break;
case PDH_LOG_TYPE_PERFMON:
default:
pdhStatus = PDH_UNKNOWN_LOG_FORMAT;
break;
}
RELEASE_MUTEX (pLog->hLogMutex);
}
return pdhStatus;
}
PDH_FUNCTION
PdhRelogA(
HLOG hLogIn,
PPDH_RELOG_INFO_A pRelogInfo
)
{
PDH_STATUS pdhStatus = ERROR_SUCCESS;
PDH_RELOG_INFO_W RelogInfo;
ZeroMemory( &RelogInfo, sizeof(PDH_RELOG_INFO_W) );
if( NULL == pRelogInfo ){
return PDH_INVALID_ARGUMENT;
}
__try{
RelogInfo.dwFlags = pRelogInfo->dwFlags;
RelogInfo.dwFileFormat = pRelogInfo->dwFileFormat;
memcpy( &RelogInfo.TimeInfo, &pRelogInfo->TimeInfo, sizeof(PDH_TIME_INFO) );
RelogInfo.ftInterval = pRelogInfo->ftInterval;
RelogInfo.StatusFunction = pRelogInfo->StatusFunction;
RelogInfo.Reserved1 = pRelogInfo->Reserved1;
RelogInfo.Reserved2 = pRelogInfo->Reserved2;
if( NULL != pRelogInfo->strLog ){
RelogInfo.strLog = (LPWSTR)G_ALLOC(
(strlen(pRelogInfo->strLog)+1) * sizeof(WCHAR) );
if( RelogInfo.strLog ){
mbstowcs(
RelogInfo.strLog,
pRelogInfo->strLog,
(strlen(pRelogInfo->strLog)+1)
);
}
}else{
pdhStatus = PDH_INVALID_ARGUMENT;
}
} __except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
if( ERROR_SUCCESS == pdhStatus ){
pdhStatus = PdhRelogW( hLogIn, &RelogInfo );
}
G_FREE( RelogInfo.strLog );
return pdhStatus;
}
PDH_FUNCTION
PdhRelogW(
HLOG hLogIn,
PPDH_RELOG_INFO_W pRelogInfo
)
{
PDH_STATUS pdhStatus = ERROR_SUCCESS;
HLOG hLogOut;
PPDHI_LOG pLogIn;
PPDHI_LOG pLogOut;
DWORD dwLogFormat;
PDH_RELOG_INFO_W RelogInfo;
SYSTEMTIME ut;
FILETIME lt;
ULONG nSampleCount = 0;
ULONG nSamplesWritten = 0;
PDH_TIME_INFO TimeInfo;
DWORD dwNumEntries = 1;
DWORD dwBufferSize = sizeof(PDH_TIME_INFO);
if( NULL == pRelogInfo ){
return PDH_INVALID_ARGUMENT;
}
ZeroMemory( &RelogInfo, sizeof(PDH_RELOG_INFO_W) );
ZeroMemory( &TimeInfo, sizeof( PDH_TIME_INFO ) );
pdhStatus = PdhGetDataSourceTimeRangeH (
hLogIn,
&dwNumEntries,
&TimeInfo,
&dwBufferSize
);
if( ERROR_SUCCESS == pdhStatus ){
if( TimeInfo.SampleCount == 0 ){
return PDH_NO_DATA;
}
}else{
return pdhStatus;
}
if( NULL != pRelogInfo ){
__try{
memcpy( &RelogInfo, pRelogInfo, sizeof(PDH_RELOG_INFO_W) );
RelogInfo.strLog = PlaiStringDuplicate( pRelogInfo->strLog );
} __except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
}else{
pdhStatus = PDH_INVALID_ARGUMENT;
}
if( IsValidLogHandle(hLogIn) && ERROR_SUCCESS == pdhStatus ){
HCOUNTER hCounter;
HQUERY hQuery;
ULONG nRecordSkip;
pLogIn = (PPDHI_LOG)hLogIn;
__try{
hQuery = (HQUERY)pLogIn->pQuery;
dwLogFormat = pLogIn->dwLogFormat;
if( NULL == pLogIn->pQuery->pCounterListHead ){
pdhStatus = PDH_NO_DATA;
}
} __except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
if( pdhStatus == ERROR_SUCCESS ){
pdhStatus = PdhOpenLogW(
RelogInfo.strLog,
RelogInfo.dwFlags,
&RelogInfo.dwFileFormat,
hQuery,
0,
NULL,
&hLogOut
);
}
if( pdhStatus == ERROR_SUCCESS ){
pLogOut= (PPDHI_LOG)hLogOut;
__try {
if( RelogInfo.TimeInfo.StartTime == 0 ||
RelogInfo.TimeInfo.StartTime < TimeInfo.StartTime ){
pLogIn->pQuery->TimeRange.StartTime = TimeInfo.StartTime;
pRelogInfo->TimeInfo.StartTime = TimeInfo.StartTime;
}else{
pLogIn->pQuery->TimeRange.StartTime = RelogInfo.TimeInfo.StartTime;
}
if( RelogInfo.TimeInfo.EndTime == 0 ||
RelogInfo.TimeInfo.EndTime > TimeInfo.EndTime ){
pLogIn->pQuery->TimeRange.EndTime = TimeInfo.EndTime;
pRelogInfo->TimeInfo.EndTime = TimeInfo.EndTime;
}else{
pLogIn->pQuery->TimeRange.EndTime = RelogInfo.TimeInfo.EndTime;
}
}__except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
nRecordSkip = RelogInfo.TimeInfo.SampleCount >= 1 ? RelogInfo.TimeInfo.SampleCount : 1;
PdhiResetLogBuffers(hLogIn);
while (ERROR_SUCCESS == pdhStatus) {
pdhStatus = PdhiCollectQueryData((PPDHI_QUERY) hQuery, (LONGLONG *) & lt);
FileTimeToSystemTime(& lt, & ut);
if( NULL != RelogInfo.StatusFunction ){
if( !(nSampleCount % 10) ){
__try{
RelogInfo.StatusFunction(
PDH_RELOG_STATUS_PROCESSING,
(double)nSampleCount/(double)TimeInfo.SampleCount
);
} __except (EXCEPTION_EXECUTE_HANDLER) {
RelogInfo.StatusFunction = NULL;
}
}
}
if (nSampleCount ++ % nRecordSkip) {
continue;
}
if (ERROR_SUCCESS == pdhStatus) {
pdhStatus = PdhiWriteRelogRecord(pLogOut, & ut);
nSamplesWritten++;
}
else if (PDH_NO_DATA == pdhStatus) {
// Reset pdhStatus. PDH_NO_DATA means that there are no new counter data
// for collected counters. Skip current record and continue.
//
pdhStatus = ERROR_SUCCESS;
}
}
//
// Check for valid exit status codes
//
if( PDH_NO_MORE_DATA == pdhStatus ){
pdhStatus = ERROR_SUCCESS;
}else{
switch( LOWORD(dwLogFormat) ){
case PDH_LOG_TYPE_BINARY:
case PDH_LOG_TYPE_PERFMON:
if( PDH_ENTRY_NOT_IN_LOG_FILE == pdhStatus ){
pdhStatus = ERROR_SUCCESS;
}
break;
}
//
// No Default:
// All other errors are really errors
//
}
if( ERROR_SUCCESS == pdhStatus ){
pdhStatus = PdhCloseLog( hLogOut, 0 );
}else{
PdhCloseLog( hLogOut, 0 );
}
__try {
((PPDHI_QUERY)hQuery)->hOutLog = NULL;
} __except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
}
}else{
pdhStatus = PDH_INVALID_HANDLE;
}
if( ERROR_SUCCESS == pdhStatus ){
__try {
pRelogInfo->TimeInfo.SampleCount = nSamplesWritten;
} __except (EXCEPTION_EXECUTE_HANDLER) {
pdhStatus = PDH_INVALID_ARGUMENT;
}
}
G_FREE( RelogInfo.strLog );
return pdhStatus;
}