307 lines
7.5 KiB
C
Raw Permalink Normal View History

2001-01-01 00:00:00 +01:00
/*++
Copyright (c) Microsoft Corporation. All rights reserved.
Module Name:
tracedp.c
Abstract:
Sample trace provider program.
--*/
#include <stdio.h>
#include <stdlib.h>
#include <windows.h>
#include <shellapi.h>
#include <tchar.h>
#include <wmistr.h>
#include <guiddef.h>
#include <evntrace.h>
#define DEFAULT_EVENTS 5000
#define MAXSTR 1024
TRACEHANDLE LoggerHandle;
// Control GUID for this provider.
GUID ControlGuid =
{0xd58c126f, 0xb309, 0x11d1, 0x96, 0x9e, 0x00, 0x00, 0xf8, 0x75, 0xa5, 0xbc};
// Only one transaction GUID will be registered for this provider.
GUID TransactionGuid =
{0xce5b1020, 0x8ea9, 0x11d0, 0xa4, 0xec, 0x00, 0xa0, 0xc9, 0x06, 0x29, 0x10};
// Array for transaction GUID registration.
TRACE_GUID_REGISTRATION TraceGuidReg[] =
{
{ (LPGUID)&TransactionGuid,
NULL
}
};
// User event layout: one ULONG.
typedef struct _USER_EVENT {
EVENT_TRACE_HEADER Header;
ULONG EventInfo;
} USER_EVENT, *PUSER_EVENT;
// Registration handle.
TRACEHANDLE RegistrationHandle;
// Trace on/off switch, level, and flag.
BOOLEAN TraceOnFlag;
ULONG EnableLevel = 0;
ULONG EnableFlags = 0;
// Number of events to be logged. The actual number can be less if disabled early.
ULONG MaxEvents = DEFAULT_EVENTS;
// To keep track of events logged.
ULONG EventCount = 0;
// ControlCallback function for enable/disable.
ULONG
ControlCallback(
IN WMIDPREQUESTCODE RequestCode,
IN PVOID Context,
IN OUT ULONG *InOutBufferSize,
IN OUT PVOID Buffer
);
// Main work to be traced is done here.
void
DoWork();
__cdecl main(argc, argv)
int argc;
char **argv;
/*++
Routine Description:
main() routine.
Arguments:
Usage: TraceDp [number of events]
[number of events] default is 5000
Return Value:
Error Code defined in winerror.h : If the function succeeds,
it returns ERROR_SUCCESS (== 0).
--*/
{
ULONG Status, i;
LPTSTR *targv, *utargv = NULL;
TraceOnFlag = FALSE;
#ifdef UNICODE
if ((targv = CommandLineToArgvW(
GetCommandLineW(), // pointer to a command-line string
&argc // receives the argument count
)) == NULL)
{
return (GetLastError());
};
utargv = targv;
#else
targv = argv;
#endif
// process command line arguments to override defaults
if (argc == 2) {
targv ++;
if (**targv >= _T('0') && **targv <= _T('9')) {
MaxEvents = _ttoi(targv[0]);
}
else if (!_tcsicmp(targv[0], _T("/?")) || !_tcsicmp(targv[0], _T("-?"))) {
printf("Usage: TraceDp [number of events]\n");
printf("\t[number of events] default is 5000\n");
return ERROR_SUCCESS;
}
}
// Free temporary argument buffer.
if (utargv != NULL) {
GlobalFree(utargv);
}
// Event provider registration.
Status = RegisterTraceGuids(
(WMIDPREQUEST)ControlCallback, // callback function
0,
&ControlGuid,
1,
TraceGuidReg,
NULL,
NULL,
&RegistrationHandle
);
if (Status != ERROR_SUCCESS) {
_tprintf(_T("Trace registration failed. Status=%d\n"), Status);
return(Status);
}
else {
_tprintf(_T("Trace registered successfully\n"));
}
_tprintf(_T("Testing Logger with %d events\n"),
MaxEvents);
// Sleep until enabled by a trace controller. In this sample, we wait to be
// enabled for tracing. However, normal applications should continue and log
// events when enabled later on.
while (!TraceOnFlag) {
_sleep(1000);
}
// Do the work while logging events. We trace two events (START and END) for
// each call to DoWork.
for (i = 0; i < MaxEvents / 2; i++) {
DoWork();
}
// Unregister.
UnregisterTraceGuids(RegistrationHandle);
return ERROR_SUCCESS;
}
ULONG
ControlCallback(
IN WMIDPREQUESTCODE RequestCode,
IN PVOID Context,
IN OUT ULONG *InOutBufferSize,
IN OUT PVOID Buffer
)
/*++
Routine Description:
Callback function when enabled.
Arguments:
RequestCode - Flag for either enable or disable.
Context - User-defined context.
InOutBufferSize - not used.
Buffer - WNODE_HEADER for the logger session.
Return Value:
Error Status. ERROR_SUCCESS if successful.
--*/
{
ULONG Status;
ULONG RetSize;
Status = ERROR_SUCCESS;
switch (RequestCode)
{
case WMI_ENABLE_EVENTS:
{
RetSize = 0;
LoggerHandle = GetTraceLoggerHandle( Buffer );
EnableLevel = GetTraceEnableLevel(LoggerHandle);
EnableFlags = GetTraceEnableFlags(LoggerHandle);
_tprintf(_T("Logging enabled to 0x%016I64x(%d,%d,%d)\n"),
LoggerHandle, RequestCode, EnableLevel, EnableFlags);
TraceOnFlag = TRUE;
break;
}
case WMI_DISABLE_EVENTS:
{
TraceOnFlag = FALSE;
RetSize = 0;
LoggerHandle = 0;
_tprintf(_T("\nLogging Disabled\n"));
break;
}
default:
{
RetSize = 0;
Status = ERROR_INVALID_PARAMETER;
break;
}
}
*InOutBufferSize = RetSize;
return(Status);
}
void
DoWork()
/*++
Routine Description:
Logs events.
Arguments:
Return Value:
None.
--*/
{
USER_EVENT UserEvent;
ULONG Status;
RtlZeroMemory(&UserEvent, sizeof(UserEvent));
UserEvent.Header.Size = sizeof(USER_EVENT);
UserEvent.Header.Flags = WNODE_FLAG_TRACED_GUID;
UserEvent.Header.Guid = TransactionGuid;
// Log START event to indicate a start of a routine or activity.
// We log EventCount as data.
if (TraceOnFlag) {
UserEvent.Header.Class.Type = EVENT_TRACE_TYPE_START;
UserEvent.EventInfo = ++EventCount;
Status = TraceEvent(
LoggerHandle,
(PEVENT_TRACE_HEADER) & UserEvent);
if (Status != ERROR_SUCCESS) {
// TraceEvent() failed. This could happen when the logger
// cannot keep up with too many events logged rapidly.
// In such cases, the return valid is
// ERROR_NOT_ENOUGH_MEMORY.
_tprintf(_T("TraceEvent failed. Status=%d\n"), Status);
}
}
//
// Main body of the work in this routine comes here.
//
_sleep(1);
// Log END event to indicate an end of a routine or activity.
// We log EventCount as data.
if (TraceOnFlag) {
UserEvent.Header.Class.Type = EVENT_TRACE_TYPE_END;
UserEvent.EventInfo = ++EventCount;
Status = TraceEvent(
LoggerHandle,
(PEVENT_TRACE_HEADER) & UserEvent);
if (Status != ERROR_SUCCESS) {
// TraceEvent() failed. This could happen when the logger
// cannot keep up with too many events logged rapidly.
// In such cases, the return valid is
// ERROR_NOT_ENOUGH_MEMORY.
_tprintf(_T("TraceEvent failed. Status=%d\n"), Status);
}
}
}