windows-nt/Source/XPSP1/NT/net/ndis/samples/isdnwan/tpidebug.c
2020-09-26 16:20:57 +08:00

506 lines
13 KiB
C

/*
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
(C) Copyright 1998
All rights reserved.
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
Portions of this software are:
(C) Copyright 1994 TriplePoint, Inc. -- http://www.TriplePoint.com
License to use this software is granted under the same terms
outlined in the Microsoft Windows Device Driver Development Kit.
(C) Copyright 1992 Microsoft Corp. -- http://www.Microsoft.com
License to use this software is granted under the terms outlined in
the Microsoft Windows Device Driver Development Kit.
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@doc INTERNAL TpiDebug TpiDebug_c
@module TpiDebug.c |
This module, along with <f TpiDebug\.h>, implements code and macros to
support NDIS driver debugging. This file must be linked with the driver
to support debug dumps and logging.
@comm
The code and macros defined by these modules is only generated during
development debugging when the C pre-processor macro flag (DBG == 1).
If (DBG == 0) no code will be generated, and all debug strings will be
removed from the image.
This is a device independent module which can be re-used, without
change, by any driver or application.
@head3 Contents |
@index class,mfunc,func,msg,mdata,struct,enum | TpiDebug_c
@end
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
*/
#if defined(_EXE_) || defined(_DLL_)
typedef char CHAR, *PCHAR;
typedef unsigned char UCHAR, *PUCHAR;
typedef unsigned short USHORT, *PUSHORT;
typedef unsigned long ULONG, *PULONG;
typedef unsigned int *PUINT;
# include <windows.h>
#elif defined(_VXD_)
# include <basedef.h>
# include <vmm.h>
# pragma VxD_LOCKED_CODE_SEG
# pragma VxD_LOCKED_DATA_SEG
#else
# include <windef.h>
#endif
#include "TpiDebug.h"
#if DBG
/*
// Sometimes the debug output seriously impacts the run-time performance,
// so it is necessary to turn off the debug output. In this case, you can
// capture some debug trace information into the DbgLogBuffer, and it can
// be examined later without impacting the run-time performance.
*/
#define DBG_LOG_ENTRIES 100 // Maximum number of FIFO log entries.
#define DBG_LOG_SIZE 128 // Maximum number of bytes per entry.
#if defined(_VXD_)
DBG_SETTINGS DbgSettings = { DBG_DEFAULTS, {'V','X','D',0 } };
#elif defined(_EXE_)
DBG_SETTINGS DbgSettings = { DBG_DEFAULTS, {'E','X','E',0 } };
#elif defined(_DLL_)
DBG_SETTINGS DbgSettings = { DBG_DEFAULTS, {'D','L','L',0 } };
#elif defined(_SYS_)
DBG_SETTINGS DbgSettings = { DBG_DEFAULTS, {'S','Y','S',0 } };
#else
DBG_SETTINGS DbgSettings = { DBG_DEFAULTS, {'T','P','I',0 } };
#endif
PDBG_SETTINGS DbgInfo = &DbgSettings;
UINT DbgLogIndex = 0;
UCHAR DbgLogBuffer[DBG_LOG_ENTRIES][DBG_LOG_SIZE] = { { 0 } };
/* @doc INTERNAL TpiDebug TpiDebug_c DbgPrintData
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f DbgPrintData> outputs data to the debug display formatted in HEX and
ASCII for easy viewing.
<f Note>: This routine is used for debug output only.
It is not compiled into the retail version.
@ex <tab> |
DbgPrintData(ReceiveBuffer, 14, 0); // Packet header
DbgPrintData(ReceiveBuffer+14, BytesReceived-14, 14); // Packet data
0000: ff ff ff ff ff ff 0a 22 23 01 02 03 00 10 ......."#.....
000E: 41 42 43 44 45 46 47 48 49 4a 4b 4c 4d 4e 4f 40 ABCDEFGHIJKMNOPQ
*/
VOID DbgPrintData(
IN PUCHAR Data, // @parm
// Pointer to first byte of data to be displayed.
IN UINT NumBytes, // @parm
// Number of bytes to be displayed.
IN ULONG Offset // @parm
// Value to be added to the offset counter displayed at the start of each
// line. This is useful for viewing data whose base offset is relative to
// another, non-zero starting address.
)
{
UINT LineStart;
UINT LineIndex;
/*
// Display the caller's buffer with up to 16 bytes per line.
*/
for (LineStart = 0; LineStart < NumBytes; LineStart += 16)
{
/*
// Display the starting offset of the line.
*/
DbgPrint("%04lx: ", LineStart + Offset);
/*
// Display a line of HEX byte values.
*/
for (LineIndex = LineStart; LineIndex < (LineStart+16); LineIndex++)
{
if (LineIndex < NumBytes)
{
DbgPrint("%02x ",(UINT)((UCHAR)*(Data+LineIndex)));
}
else
{
DbgPrint(" ");
}
}
DbgPrint(" "); // A little white space between HEX and ASCII.
/*
// Display the corresponding ASCII byte values if they are printable.
// (i.e. 0x20 <= N <= 0x7F).
*/
for (LineIndex = LineStart; LineIndex < (LineStart+16); LineIndex++)
{
if (LineIndex < NumBytes)
{
char c = *(Data+LineIndex);
if (c < ' ' || c > 'z')
{
c = '.';
}
DbgPrint("%c", (UINT)c);
}
else
{
DbgPrint(" ");
}
}
DbgPrint("\n"); // End of line.
}
}
/* @doc INTERNAL TpiDebug TpiDebug_c DbgQueueData
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f DbgQueueData> saves data to the DbgLogBuffer so it can be viewed later
with the debugger.
<f Note>: This routine is used for debug output only.
It is not compiled into the retail version.
*/
VOID DbgQueueData(
IN PUCHAR Data, // @parm
// Pointer to first byte of data to be displayed.
IN UINT NumBytes, // @parm
// Number of bytes to be displayed.
IN UINT Flags // @parm
// A flag descriptor to help identify the log entry.
)
{
/*
// Point to the next available entry in the DbgLogBuffer.
*/
PUCHAR LogEntry = &DbgLogBuffer[DbgLogIndex++][0];
/*
// Wrap around on the next entry if needed.
*/
if (DbgLogIndex >= DBG_LOG_ENTRIES)
{
DbgLogIndex = 0;
}
/*
// Save the flags parameter in the first WORD of the log buffer.
*/
*((PUSHORT) LogEntry) = (USHORT) Flags;
LogEntry += sizeof(PUSHORT);
/*
// Save the NumBytes parameter in the second WORD of the log buffer.
*/
*((PUSHORT) LogEntry) = (USHORT) NumBytes;
LogEntry += sizeof(NumBytes);
/*
// Don't try to save more than we have room for.
*/
if (NumBytes > DBG_LOG_SIZE - sizeof(USHORT) * 2)
{
NumBytes = DBG_LOG_SIZE - sizeof(USHORT) * 2;
}
/*
// Save the rest of the data in the remaining portion of the log buffer.
*/
while (NumBytes--)
{
*LogEntry++ = *Data++;
}
}
/* @doc INTERNAL TpiDebug TpiDebug_c DbgBreakPoint
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func VOID | DbgBreakPoint |
<f DbgBreakPoint> is defined in the NT kernel for SYS drivers, but we
override it here so we can support for SYS's, EXE's, VXD's, and DLL's.
*/
#if defined(_MSC_VER) && (_MSC_VER <= 800)
// Must be building with 16-bit compiler
VOID __cdecl DbgBreakPoint(VOID)
#else
// Must be building with 32-bit compiler
VOID __stdcall DbgBreakPoint(VOID)
#endif
{
#if !defined(_WIN64)
__asm int 3;
#endif
}
/* @doc INTERNAL TpiDebug TpiDebug_c DbgPrint
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func ULONG __cdecl | DbgPrint |
<f DbgPrint> is defined in the kernel for SYS drivers, otherwise it is
supported here for EXE's, VXD's, and DLL's.
@parm PCHAR | Format |
printf style format string.
@parm OPTIONAL | Params |
Zero or more optional parameters as needed by the format string.
*/
#if defined(_VXD_)
#if !defined(NDIS_DOS)
ULONG __cdecl DbgPrint(PCHAR Format, ...)
{
ULONG result = 0;
__asm lea eax, (Format + 4)
__asm push eax
__asm push Format
VMMCall(_Debug_Printf_Service)
__asm add esp, 4*2
__asm mov result, eax
return (result);
}
#endif
#elif defined(_EXE_) || defined(_DLL_)
UCHAR DbgString[1024];
ULONG __cdecl DbgPrint(PCHAR Format, ...)
{
ULONG result;
result = wvsprintf(DbgString, Format, ((PCHAR) &Format) + sizeof(PCHAR));
OutputDebugString(DbgString);
if (result >= sizeof(DbgString))
{
// We just blew the stack!
// Since we can't return, we have to generate a stack-fault interrupt
__asm int 1;
__asm int 3;
__asm int 12;
}
return (result);
}
#endif // DbgPrint
/*
* If DBG_SILENT is set, all TERSE debug goes here. An assertion
* will dump the block.
*/
#define DBG_QUEUE_LEN 4096
UINT DbgIndex=0;
UINT DbgLen=0;
UCHAR DbgQueue[DBG_QUEUE_LEN] = {0};
UCHAR DbgLock=0;
/* @doc INTERNAL TpiDebug TpiDebug_c DbgDumpSilentQueue
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f DbgDumpSilentQueue> dumps the contents of the silent debug queue to
the monitor.
*/
void DbgDumpSilentQueue(
void
)
{
if (DbgLen >= DBG_QUEUE_LEN)
{
DbgPrintData(
&DbgQueue[DbgIndex],
DBG_QUEUE_LEN-DbgIndex,
0);
if (DbgIndex)
{
DbgPrint("\n");
DbgPrintData(
DbgQueue,
DbgIndex-1,
0);
}
DbgPrint("\n");
}
else if (DbgLen)
{
DbgPrintData(
DbgQueue,
DbgIndex-1,
0);
DbgPrint("\n");
}
}
#if NDIS_NT
/* @doc INTERNAL TpiDebug TpiDebug_c _assert
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f _assert> overrides the assertion function provided by the operating
system. Dumps the contents of debug queue, prints the assertion, and
then traps to the debugger. Used for debugging only.
*/
void _CRTAPI1 _assert(
void * exp, // @parm
// ASCIIZ pointer to the expression causing the fault.
void * file, // @parm
// ASCIIZ pointer to the name of the file.
unsigned line // @parm
// Line offset within the file where the assertion is defined.
)
{
DbgDumpSilentQueue();
DbgPrint("Assertion Failed: %s at %s:%d\n",exp,file,line);
DbgBreakPoint();
}
#endif
/* @doc INTERNAL TpiDebug TpiDebug_c DbgSilentQueue
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f DbgSilentQueue> logs a string to the debug queue which can be
displayed later using <f DbgDumpSilentQueue>. Used for debugging only.
*/
void DbgSilentQueue(
PUCHAR str // @parm
// Pointer to string to be placed in DbgQueue.
)
{
/*
// If the debug queue is busy, just
// bail out.
*/
if ((++DbgLock) > 1)
{
goto exit;
}
while (str && *str)
{
DbgQueue[DbgIndex] = *str++;
DbgLen++;
if ((++DbgIndex) >= DBG_QUEUE_LEN)
{
DbgIndex = 0;
}
}
exit:
DbgLock--;
}
/* @doc INTERNAL TpiDebug TpiDebug_c DbgPrintFieldTable
§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§§
@func
<f DbgPrintFieldTable> displays the contents of a C data structure in
a formatted output to the debugger. This can be used when symbolic
debugging is not available on the target platform.
*/
void DbgPrintFieldTable(
IN PDBG_FIELD_TABLE pFields, // @parm
// A pointer to an array of field records <t DBG_FIELD_TABLE>.
IN PUCHAR pBaseContext, // @parm
// References the base of the structure where the values will be displayed
// from. This should be a pointer to the first byte of the structure.
IN PUCHAR pBaseName // @parm
// Pointer to C string containing the name of the structure being displayed.
)
{
DbgPrint("STRUCTURE: @0x%08X %s\n", pBaseContext, pBaseName);
while (pFields->FieldName)
{
switch (pFields->FieldType)
{
case sizeof(ULONG):
DbgPrint("\t%04X: %-32s=0x%08X\n", pFields->FieldOffset,
pFields->FieldName,
*(PULONG)(pBaseContext+pFields->FieldOffset));
break;
case sizeof(USHORT):
DbgPrint("\t%04X: %-32s=0x%04X\n", pFields->FieldOffset,
pFields->FieldName,
*(PUSHORT)(pBaseContext+pFields->FieldOffset));
break;
case sizeof(UCHAR):
DbgPrint("\t%04X: %-32s=0x%02X\n", pFields->FieldOffset,
pFields->FieldName,
*(PUCHAR)(pBaseContext+pFields->FieldOffset));
break;
default:
ASSERT(0);
break;
}
pFields++;
}
}
#endif // DBG