mirror of https://github.com/tongzx/nt5src
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.
5421 lines
115 KiB
5421 lines
115 KiB
/*++
|
|
|
|
Copyright (c) 1997-2001 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
print.c
|
|
|
|
Abstract:
|
|
|
|
Domain Name System (DNS) Library
|
|
|
|
Print routines.
|
|
|
|
Author:
|
|
|
|
Jim Gilroy (jamesg) February 8, 1997
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include "local.h"
|
|
#include "svcguid.h" // RnR guids
|
|
#include "..\dnsapi\dnsapip.h" // Private query stuff
|
|
#include "..\resolver\idl\resrpc.h" // Resolver interface structs
|
|
|
|
|
|
//
|
|
// Print globals
|
|
//
|
|
|
|
CRITICAL_SECTION DnsAtomicPrintCs;
|
|
PCRITICAL_SECTION pDnsAtomicPrintCs = NULL;
|
|
|
|
//
|
|
// Empty string for simple switching of UTF-8/Unicode print
|
|
// (macros in dnslib.h)
|
|
//
|
|
|
|
DWORD DnsEmptyString = 0;
|
|
|
|
|
|
//
|
|
// Indenting
|
|
//
|
|
// Serve up as many indenting tabs as indent level indicates
|
|
//
|
|
|
|
CHAR IndentString[] = "\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t\t";
|
|
|
|
#define INDENT_STRING( level ) (IndentString + (sizeof(IndentString) - 1 - (level)))
|
|
|
|
|
|
|
|
//
|
|
// Print Locking
|
|
//
|
|
// Unless caller initilizes print locking by supplying lock,
|
|
// print locking is disabled.
|
|
//
|
|
|
|
VOID
|
|
DnsPrint_InitLocking(
|
|
IN PCRITICAL_SECTION pLock
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Setup DNS printing to use a lock.
|
|
|
|
Can use already initialized lock from caller, or will
|
|
create default lock.
|
|
|
|
Arguments:
|
|
|
|
pLock - ptr to CS to use as lock; if NULL, create one
|
|
|
|
Return Value:
|
|
|
|
None
|
|
|
|
--*/
|
|
{
|
|
if ( pLock )
|
|
{
|
|
pDnsAtomicPrintCs = pLock;
|
|
}
|
|
else if ( !pDnsAtomicPrintCs )
|
|
{
|
|
InitializeCriticalSection( &DnsAtomicPrintCs );
|
|
pDnsAtomicPrintCs = &DnsAtomicPrintCs;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Lock(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Lock to get atomic DNS print.
|
|
|
|
--*/
|
|
{
|
|
if ( pDnsAtomicPrintCs )
|
|
{
|
|
EnterCriticalSection( pDnsAtomicPrintCs );
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
DnsPrint_Unlock(
|
|
VOID
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Unlock to debug print.
|
|
|
|
--*/
|
|
{
|
|
if ( pDnsAtomicPrintCs )
|
|
{
|
|
LeaveCriticalSection( pDnsAtomicPrintCs );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Print routines for general types and structures
|
|
//
|
|
|
|
VOID
|
|
DnsPrint_String(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader OPTIONAL,
|
|
IN PSTR pszString,
|
|
IN BOOL fUnicode,
|
|
IN PSTR pszTrailer OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print DNS string given in either Unicode or UTF-8 format.
|
|
|
|
--*/
|
|
{
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "";
|
|
}
|
|
if ( !pszTrailer )
|
|
{
|
|
pszTrailer = "";
|
|
}
|
|
|
|
if ( ! pszString )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s(NULL DNS string ptr)%s",
|
|
pszHeader,
|
|
pszTrailer );
|
|
}
|
|
else if (fUnicode)
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%S%s",
|
|
pszHeader,
|
|
(PWSTR ) pszString,
|
|
pszTrailer );
|
|
}
|
|
else
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s%s",
|
|
pszHeader,
|
|
(PSTR ) pszString,
|
|
pszTrailer );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_StringCharSet(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader OPTIONAL,
|
|
IN PSTR pszString,
|
|
IN DNS_CHARSET CharSet,
|
|
IN PSTR pszTrailer OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print string of given CHARSET.
|
|
|
|
--*/
|
|
{
|
|
DnsPrint_String(
|
|
PrintRoutine,
|
|
pContext,
|
|
pszHeader,
|
|
pszString,
|
|
(CharSet == DnsCharSetUnicode),
|
|
pszTrailer );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_UnicodeStringBytes(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PWCHAR pUnicode,
|
|
IN DWORD Length
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print chars (WORDs) of unicode string.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\twide string %S\r\n"
|
|
"\tlength %d\r\n"
|
|
"\tbytes ",
|
|
pszHeader,
|
|
pUnicode,
|
|
Length );
|
|
|
|
for ( i=0; i<Length; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%04hx ",
|
|
pUnicode[i] );
|
|
}
|
|
printf( "\r\n" );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Utf8StringBytes(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PCHAR pUtf8,
|
|
IN DWORD Length
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print bytes of UTF8 string.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tUTF8 string %s\r\n"
|
|
"\tlength %d\r\n"
|
|
"\tbytes ",
|
|
pszHeader,
|
|
pUtf8,
|
|
Length );
|
|
|
|
for ( i=0; i<Length; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%02x ",
|
|
(UCHAR) pUtf8[i] );
|
|
}
|
|
printf( "\r\n" );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_StringArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR * StringArray,
|
|
IN DWORD Count, OPTIONAL
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print string array.
|
|
|
|
--*/
|
|
{
|
|
DWORD i = 0;
|
|
PCHAR pstr;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "StringArray:";
|
|
}
|
|
if ( !StringArray )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL pointer!\r\n",
|
|
pszHeader );
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( Count )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s Count = %d\r\n",
|
|
pszHeader,
|
|
Count );
|
|
}
|
|
else
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n",
|
|
pszHeader );
|
|
}
|
|
|
|
//
|
|
// print args
|
|
// - stop at Count (if given)
|
|
// OR
|
|
// - on NULL arg (if no count given)
|
|
//
|
|
|
|
while ( (!Count || i < Count) )
|
|
{
|
|
pstr = StringArray[i++];
|
|
if ( !pstr && !Count )
|
|
{
|
|
break;
|
|
}
|
|
PrintRoutine(
|
|
pContext,
|
|
(fUnicode) ? "\t%S\r\n" : "\t%s\r\n",
|
|
pstr );
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Argv(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN CHAR ** Argv,
|
|
IN DWORD Argc, OPTIONAL
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print Argv array.
|
|
|
|
--*/
|
|
{
|
|
//
|
|
// this is just special case of string print
|
|
//
|
|
|
|
DnsPrint_StringArray(
|
|
PrintRoutine,
|
|
pContext,
|
|
pszHeader
|
|
? pszHeader
|
|
: "Argv:",
|
|
Argv,
|
|
Argc,
|
|
fUnicode );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_DwordArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR pszName,
|
|
IN DWORD dwCount,
|
|
IN PDWORD adwArray
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print DWORD array.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
pszHeader );
|
|
}
|
|
|
|
if ( !pszName )
|
|
{
|
|
pszName = "DWORD";
|
|
}
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s Array Count = %d\r\n",
|
|
pszName,
|
|
dwCount );
|
|
|
|
for( i=0; i<dwCount; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t%s[%d] => 0x%p (%d)\r\n",
|
|
pszName,
|
|
i,
|
|
adwArray[i],
|
|
adwArray[i] );
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_IpAddressArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR pszName,
|
|
IN DWORD dwIpAddrCount,
|
|
IN PIP_ADDRESS pIpAddrs
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print IP address array.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( !pszName )
|
|
{
|
|
pszName = "IP Addr";
|
|
}
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s Count = %d\r\n",
|
|
pszHeader ? pszHeader : "",
|
|
pszName,
|
|
dwIpAddrCount );
|
|
|
|
if ( dwIpAddrCount != 0 && pIpAddrs != NULL )
|
|
{
|
|
// print array with count
|
|
// use character print so works even if NOT DWORD aligned
|
|
|
|
for( i=0; i<dwIpAddrCount; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t%s[%d] => %d.%d.%d.%d\r\n",
|
|
pszName,
|
|
i,
|
|
* ( (PUCHAR) &pIpAddrs[i] + 0 ),
|
|
* ( (PUCHAR) &pIpAddrs[i] + 1 ),
|
|
* ( (PUCHAR) &pIpAddrs[i] + 2 ),
|
|
* ( (PUCHAR) &pIpAddrs[i] + 3 ) );
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
// this spins if printing zero length IP_ARRAY struct
|
|
|
|
else if ( pIpAddrs != NULL )
|
|
{
|
|
// print NULL terminated array (ex. hostents IPs)
|
|
|
|
i = 0;
|
|
while ( pIpAddrs[i] )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t%s[%d] => %s\r\n",
|
|
pszName,
|
|
i,
|
|
inet_ntoa( *(struct in_addr *) &pIpAddrs[i] ) );
|
|
}
|
|
}
|
|
#endif
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_IpArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR pszName,
|
|
IN PIP_ARRAY pIpArray
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print IP address array struct
|
|
|
|
Just pass through to more generic print routine.
|
|
|
|
--*/
|
|
{
|
|
// protect against NULL case
|
|
|
|
if ( !pIpArray )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\tNULL IP Array.\r\n",
|
|
pszHeader ? pszHeader : "" );
|
|
}
|
|
|
|
// call uncoupled IP array routine
|
|
|
|
else
|
|
{
|
|
DnsPrint_IpAddressArray(
|
|
PrintRoutine,
|
|
pContext,
|
|
pszHeader,
|
|
pszName,
|
|
pIpArray->AddrCount,
|
|
pIpArray->AddrArray );
|
|
}
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_DnsAddrArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_ADDR_ARRAY pAddrArray
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print IP address info array.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Addr Array";
|
|
}
|
|
if ( !pAddrArray )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL AddrArray ptr.\r\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
" Ptr = %p\r\n"
|
|
" Count = %d\r\n",
|
|
pszHeader ? pszHeader : "IP Address Info",
|
|
pAddrArray,
|
|
pAddrArray->AddrCount );
|
|
|
|
//
|
|
// print array up to count
|
|
//
|
|
// DCR: IP6 address info blob
|
|
//
|
|
|
|
for( i=0; i<pAddrArray->AddrCount; i++ )
|
|
{
|
|
IP_ADDRESS ip = pAddrArray->AddrArray[i].IpAddr;
|
|
IP_ADDRESS subnetMask = pAddrArray->AddrArray[i].SubnetMask;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tIP Address[%d] => %d.%d.%d.%d\r\n"
|
|
"\tSubnetMask[%d] => %d.%d.%d.%d\r\n",
|
|
i,
|
|
(ip & 0xff),
|
|
((ip>>8) & 0xff),
|
|
((ip>>16) & 0xff),
|
|
((ip>>24) & 0xff),
|
|
i,
|
|
(subnetMask & 0xff),
|
|
((subnetMask>>8) & 0xff),
|
|
((subnetMask>>16) & 0xff),
|
|
((subnetMask>>24) & 0xff)
|
|
);
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Ip6Address(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PIP6_ADDRESS pIp6Address,
|
|
IN PSTR pszTrailer
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print IP6 address.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine -- print routine to call
|
|
|
|
pContext -- first argument to print routine
|
|
|
|
pszHeader -- header to print
|
|
NOTE: unlike other print routines this routine
|
|
requires header to contain newline,tab, etc if
|
|
multiline print is desired; the reason is to allow
|
|
use of this routine for single line print
|
|
|
|
pIp6Address -- ptr to IP6 address to print
|
|
|
|
pszTrailer -- trailer to print
|
|
NOTE: again this routine is designed to allow single
|
|
line print; if newline required after print, send
|
|
newline in trailer
|
|
|
|
Return Value:
|
|
|
|
Ptr to next location in buffer (the terminating NULL).
|
|
|
|
--*/
|
|
{
|
|
CHAR buffer[ IP6_ADDRESS_STRING_BUFFER_LENGTH ];
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "IP6 Address: ";
|
|
}
|
|
if ( !pszTrailer )
|
|
{
|
|
pszHeader = "\r\n";
|
|
}
|
|
|
|
if ( !pIp6Address )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL IP6 address ptr.%s",
|
|
pszHeader,
|
|
pszTrailer );
|
|
}
|
|
|
|
// convert IP6 address to string
|
|
|
|
Dns_Ip6AddressToString_A(
|
|
buffer,
|
|
pIp6Address );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s%s",
|
|
pszHeader,
|
|
buffer,
|
|
pszTrailer );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Print routines for DNS types and structures
|
|
//
|
|
|
|
INT
|
|
Dns_WritePacketNameToBuffer(
|
|
OUT PCHAR pBuffer,
|
|
OUT PCHAR * ppBufferOut,
|
|
IN PBYTE pMsgName,
|
|
IN PDNS_HEADER pMsgHead, OPTIONAL
|
|
IN PBYTE pMsgEnd OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Write packet name into buffer.
|
|
|
|
Arguments:
|
|
|
|
pBuffer - buffer to print to,
|
|
bounds checking done only on each loop to limit to
|
|
twice DNS_MAX_NAME_LENGTH, recommend input buffer
|
|
of at least that length
|
|
|
|
ppBufferOut - ptr to terminating NULL in buffer conversion, this
|
|
is position at which additional printing to buffer
|
|
could resume
|
|
|
|
pMsgName - ptr to name in packet to print
|
|
|
|
pMsgHead - ptr to DNS message; need for offsetting, if not given
|
|
names are not printed past first offset
|
|
|
|
pMsgEnd - ptr to end of message, specifically byte immediately after
|
|
message
|
|
|
|
Return Value:
|
|
|
|
Count of bytes in packet name occupied.
|
|
This offset from pMsgName is the next field in the packet.
|
|
|
|
Zero return indicates error in message name.
|
|
|
|
--*/
|
|
{
|
|
register PUCHAR pchbuf;
|
|
register PUCHAR pchmsg;
|
|
register UCHAR cch;
|
|
PCHAR pbufStop;
|
|
PCHAR pnextLabel;
|
|
UCHAR compressionType;
|
|
WORD offset;
|
|
PCHAR pbyteAfterFirstOffset = NULL;
|
|
|
|
//
|
|
// no message end specified?
|
|
// make it max ptr so we can do single test for ptr validity
|
|
// rather than testing for pMsgEnd existence first
|
|
//
|
|
|
|
if ( !pMsgEnd )
|
|
{
|
|
pMsgEnd = (PVOID)(INT_PTR)(-1);
|
|
}
|
|
|
|
//
|
|
// loop until copy as printable name, or hit compression or name error
|
|
//
|
|
|
|
pchbuf = pBuffer;
|
|
pbufStop = pchbuf + DNS_MAX_NAME_LENGTH + DNS_MAX_LABEL_LENGTH;
|
|
pchmsg = pMsgName;
|
|
|
|
while ( 1 )
|
|
{
|
|
// bounds checking to survive bad packet
|
|
//
|
|
// DEVNOTE: note this is not strictly a bad packet (could be just a
|
|
// heck of a lot of labels) and we could
|
|
// a) let packet processing proceed without printing
|
|
// or
|
|
// b) require buffer that could contain max legal DNS name
|
|
// but not worth the effort
|
|
|
|
if ( pchbuf >= pbufStop )
|
|
{
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[ERROR name exceeds safe print buffer length]\r\n" );
|
|
pchmsg = pMsgName;
|
|
break;
|
|
}
|
|
|
|
cch = (UCHAR) *pchmsg++;
|
|
compressionType = cch & 0xC0;
|
|
|
|
DNSDBG( OFF, (
|
|
"byte = (%d) (0x%02x)\r\n"
|
|
"compress flag = (%d) (0x%02x)\r\n",
|
|
cch, cch,
|
|
compressionType, compressionType ));
|
|
|
|
//
|
|
// normal length byte
|
|
// - write length field
|
|
// - copy label to print buffer
|
|
//
|
|
|
|
if ( compressionType == 0 )
|
|
{
|
|
pchbuf += sprintf( pchbuf, "(%d)", (INT)cch );
|
|
|
|
// terminate at root name
|
|
|
|
if ( ! cch )
|
|
{
|
|
break;
|
|
}
|
|
|
|
// check that within packet
|
|
|
|
pnextLabel = pchmsg + cch;
|
|
if ( pnextLabel >= pMsgEnd )
|
|
{
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[ERROR length byte: 0x%02X at %p leads outside message]\r\n",
|
|
cch,
|
|
pchmsg );
|
|
|
|
// force zero byte return
|
|
|
|
pchmsg = pMsgName;
|
|
break;
|
|
}
|
|
|
|
// copy label to output string
|
|
|
|
memcpy(
|
|
pchbuf,
|
|
pchmsg,
|
|
cch );
|
|
|
|
pchbuf += cch;
|
|
pchmsg = pnextLabel;
|
|
continue;
|
|
}
|
|
|
|
//
|
|
// valid compression
|
|
//
|
|
|
|
else if ( compressionType == (UCHAR)0xC0 )
|
|
{
|
|
// check that compression word not straddling message end
|
|
|
|
if ( pchmsg >= pMsgEnd )
|
|
{
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[ERROR compression word at %p is outside message]\r\n",
|
|
pchmsg );
|
|
|
|
// force zero byte return
|
|
|
|
pchmsg = pMsgName;
|
|
break;
|
|
}
|
|
|
|
// calculate offset
|
|
|
|
offset = cch; // high byte
|
|
offset <<= 8;
|
|
offset |= *pchmsg++; // low byte
|
|
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[%04hX]",
|
|
offset );
|
|
|
|
if ( pMsgHead )
|
|
{
|
|
//
|
|
// on first compression, save ptr to byte immediately after
|
|
// name, so can calculate next byte
|
|
//
|
|
// save ptr to next byte in mess, to calculate actual length
|
|
// name takes up in packet
|
|
//
|
|
|
|
if ( ! pbyteAfterFirstOffset )
|
|
{
|
|
pbyteAfterFirstOffset = pchmsg;
|
|
}
|
|
|
|
//
|
|
// jump to offset for continuation of name
|
|
// - clear two highest bits to get length
|
|
//
|
|
|
|
offset = offset ^ 0xC000;
|
|
DNS_ASSERT( (offset & 0xC000) == 0 );
|
|
|
|
pnextLabel = (PCHAR)pMsgHead + offset;
|
|
if ( pnextLabel >= pchmsg - sizeof(WORD) )
|
|
{
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[ERROR offset at %p to higher byte in packet %p]\r\n",
|
|
pchmsg - sizeof(WORD),
|
|
pnextLabel );
|
|
break;
|
|
}
|
|
pchmsg = pnextLabel;
|
|
continue;
|
|
}
|
|
|
|
// if no ptr to message head, can not continue at offset
|
|
// NULL terminate previous label
|
|
|
|
else
|
|
{
|
|
*pchbuf++ = 0;
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// invalid compression
|
|
// - force zero byte return to indicate error
|
|
|
|
else
|
|
{
|
|
pchbuf += sprintf(
|
|
pchbuf,
|
|
"[ERROR length byte: 0x%02X]",
|
|
cch );
|
|
pchmsg = pMsgName;
|
|
break;
|
|
}
|
|
}
|
|
|
|
//
|
|
// return ptr to next position in output buffer
|
|
//
|
|
|
|
if ( ppBufferOut )
|
|
{
|
|
*ppBufferOut = pchbuf;
|
|
}
|
|
|
|
//
|
|
// return number of bytes read from message
|
|
//
|
|
|
|
if ( pbyteAfterFirstOffset )
|
|
{
|
|
pchmsg = pbyteAfterFirstOffset;
|
|
}
|
|
return (INT)( pchmsg - pMsgName );
|
|
}
|
|
|
|
|
|
|
|
INT
|
|
DnsPrint_PacketName(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader, OPTIONAL
|
|
IN PBYTE pMsgName,
|
|
IN PDNS_HEADER pMsgHead, OPTIONAL
|
|
IN PBYTE pMsgEnd, OPTIONAL
|
|
IN PSTR pszTrailer OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print DNS name given in packet format.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header to print
|
|
|
|
pMsgHead - ptr to DNS message; need for offsetting, if not given
|
|
names are not printed past first offset
|
|
|
|
pMsgName - ptr to name in packet to print
|
|
|
|
pMsgEnd - ptr to end of message; OPTIONAL, but needed to protect
|
|
against AV accessing bad packet names
|
|
|
|
pszTrailer - trailer to print after name
|
|
|
|
Return Value:
|
|
|
|
Count of bytes in packet name occupied.
|
|
This offset from pMsgName is the next field in the packet.
|
|
|
|
Zero return indicates error in message name.
|
|
|
|
--*/
|
|
{
|
|
INT countNameBytes;
|
|
|
|
// name buffer, allow space for full name, plus parens on length
|
|
// fields plus several compression flags
|
|
|
|
CHAR PrintName[ 2*DNS_MAX_NAME_LENGTH ];
|
|
|
|
|
|
if ( ! pMsgName )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s(NULL packet name ptr)%s\r\n",
|
|
pszHeader ? pszHeader : "",
|
|
pszTrailer ? pszTrailer : ""
|
|
);
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// build packet name into buffer, then print
|
|
//
|
|
|
|
countNameBytes = Dns_WritePacketNameToBuffer(
|
|
PrintName,
|
|
NULL,
|
|
pMsgName,
|
|
pMsgHead,
|
|
pMsgEnd
|
|
);
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s%s",
|
|
pszHeader ? pszHeader : "",
|
|
PrintName,
|
|
pszTrailer ? pszTrailer : ""
|
|
);
|
|
|
|
return( countNameBytes );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Message(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_MSG_BUF pMsg
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print DNS message buffer.
|
|
Includes context information as well as actual DNS message.
|
|
|
|
--*/
|
|
{
|
|
PDNS_HEADER pmsgHeader;
|
|
PCHAR pchRecord;
|
|
PBYTE pmsgEnd;
|
|
INT i;
|
|
INT isection;
|
|
INT cchName;
|
|
WORD wLength;
|
|
WORD wOffset;
|
|
WORD wXid;
|
|
WORD wQuestionCount;
|
|
WORD wAnswerCount;
|
|
WORD wNameServerCount;
|
|
WORD wAdditionalCount;
|
|
WORD countSectionRR;
|
|
BOOL fFlipped = FALSE;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n",
|
|
pszHeader );
|
|
}
|
|
|
|
// get message info
|
|
//
|
|
// note: length may not be correctly set while building message,
|
|
// so make pmsgEnd greater of given length and pCurrent ptr
|
|
// but allow for case where set back to pre-OPT length
|
|
//
|
|
|
|
wLength = pMsg->MessageLength;
|
|
pmsgHeader = &pMsg->MessageHead;
|
|
pmsgEnd = ((PBYTE)pmsgHeader) + wLength;
|
|
|
|
if ( pmsgEnd < pMsg->pCurrent &&
|
|
pmsgEnd != pMsg->pPreOptEnd )
|
|
{
|
|
pmsgEnd = pMsg->pCurrent;
|
|
}
|
|
|
|
//
|
|
// print header info
|
|
//
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s info at %p\r\n"
|
|
" Socket = %d\r\n"
|
|
" Remote addr %s, port %ld\r\n"
|
|
" Buf length = 0x%04x\r\n"
|
|
" Msg length = 0x%04x\r\n"
|
|
" Message:\r\n",
|
|
( pMsg->fTcp
|
|
? "TCP"
|
|
: "UDP" ),
|
|
( pmsgHeader->IsResponse
|
|
? "response"
|
|
: "question" ),
|
|
pMsg,
|
|
pMsg->Socket,
|
|
MSG_REMOTE_IP_STRING( pMsg ),
|
|
MSG_REMOTE_IP_PORT( pMsg ),
|
|
pMsg->BufferLength,
|
|
wLength
|
|
);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
" XID 0x%04hx\r\n"
|
|
" Flags 0x%04hx\r\n"
|
|
" QR 0x%lx (%s)\r\n"
|
|
" OPCODE 0x%lx (%s)\r\n"
|
|
" AA 0x%lx\r\n"
|
|
" TC 0x%lx\r\n"
|
|
" RD 0x%lx\r\n"
|
|
" RA 0x%lx\r\n"
|
|
" Z 0x%lx\r\n"
|
|
" RCODE 0x%lx (%s)\r\n"
|
|
" QCOUNT 0x%hx\r\n"
|
|
" ACOUNT 0x%hx\r\n"
|
|
" NSCOUNT 0x%hx\r\n"
|
|
" ARCOUNT 0x%hx\r\n",
|
|
|
|
pmsgHeader->Xid,
|
|
ntohs((*((PWORD)pmsgHeader + 1))),
|
|
|
|
pmsgHeader->IsResponse,
|
|
(pmsgHeader->IsResponse ? "response" : "question"),
|
|
pmsgHeader->Opcode,
|
|
Dns_OpcodeString( pmsgHeader->Opcode ),
|
|
pmsgHeader->Authoritative,
|
|
pmsgHeader->Truncation,
|
|
pmsgHeader->RecursionDesired,
|
|
pmsgHeader->RecursionAvailable,
|
|
pmsgHeader->Reserved,
|
|
pmsgHeader->ResponseCode,
|
|
Dns_ResponseCodeString( pmsgHeader->ResponseCode ),
|
|
|
|
pmsgHeader->QuestionCount,
|
|
pmsgHeader->AnswerCount,
|
|
pmsgHeader->NameServerCount,
|
|
pmsgHeader->AdditionalCount );
|
|
|
|
//
|
|
// determine if byte flipped and get correct count
|
|
//
|
|
|
|
wXid = pmsgHeader->Xid;
|
|
wQuestionCount = pmsgHeader->QuestionCount;
|
|
wAnswerCount = pmsgHeader->AnswerCount;
|
|
wNameServerCount = pmsgHeader->NameServerCount;
|
|
wAdditionalCount = pmsgHeader->AdditionalCount;
|
|
|
|
if ( wQuestionCount )
|
|
{
|
|
fFlipped = wQuestionCount & 0xff00;
|
|
}
|
|
else if ( wNameServerCount )
|
|
{
|
|
fFlipped = wNameServerCount & 0xff00;
|
|
}
|
|
if ( fFlipped )
|
|
{
|
|
wXid = ntohs( wXid );
|
|
wQuestionCount = ntohs( wQuestionCount );
|
|
wAnswerCount = ntohs( wAnswerCount );
|
|
wNameServerCount = ntohs( wNameServerCount );
|
|
wAdditionalCount = ntohs( wAdditionalCount );
|
|
}
|
|
|
|
//
|
|
// catch record flipping problems -- all are flipped or none at all
|
|
// and no record count should be > 256 EXCEPT answer count
|
|
// during FAST zone transfer
|
|
//
|
|
|
|
DNS_ASSERT( ! (wQuestionCount & 0xff00) );
|
|
DNS_ASSERT( ! (wNameServerCount & 0xff00) );
|
|
DNS_ASSERT( ! (wAdditionalCount & 0xff00) );
|
|
|
|
#if 0
|
|
//
|
|
// stop here if WINS response -- don't have parsing ready
|
|
//
|
|
|
|
if ( pmsgHeader->IsResponse && IS_WINS_XID(wXid) )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
" WINS Response packet.\r\n\r\n" );
|
|
goto Unlock;
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// print questions and resource records
|
|
//
|
|
|
|
pchRecord = (PCHAR)(pmsgHeader + 1);
|
|
|
|
for ( isection=0; isection<4; isection++)
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
" %s Section:\r\n",
|
|
Dns_SectionNameString( isection, pmsgHeader->Opcode ) );
|
|
|
|
if ( isection==0 )
|
|
{
|
|
countSectionRR = wQuestionCount;
|
|
}
|
|
else if ( isection==1 )
|
|
{
|
|
countSectionRR = wAnswerCount;
|
|
}
|
|
else if ( isection==2 )
|
|
{
|
|
countSectionRR = wNameServerCount;
|
|
}
|
|
else if ( isection==3 )
|
|
{
|
|
countSectionRR = wAdditionalCount;
|
|
}
|
|
|
|
for ( i=0; i < countSectionRR; i++ )
|
|
{
|
|
//
|
|
// verify not overrunning length
|
|
// - check against pCurrent as well as message length
|
|
// so can print packets while being built
|
|
//
|
|
|
|
wOffset = (WORD)(pchRecord - (PCHAR)pmsgHeader);
|
|
if ( wOffset >= wLength
|
|
&&
|
|
pchRecord >= pMsg->pCurrent )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: BOGUS PACKET:\r\n"
|
|
"\tFollowing RR (offset %d) past packet length (%d).\r\n",
|
|
wOffset,
|
|
wLength
|
|
);
|
|
goto Unlock;
|
|
}
|
|
|
|
//
|
|
// print RR name
|
|
//
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
" Name Offset = 0x%04x\r\n",
|
|
wOffset
|
|
);
|
|
|
|
cchName = DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
" Name \"",
|
|
pchRecord,
|
|
pmsgHeader,
|
|
pmsgEnd,
|
|
"\"\r\n" );
|
|
if ( ! cchName )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: Invalid name length, stop packet print\r\n" );
|
|
DNS_ASSERT( FALSE );
|
|
break;
|
|
}
|
|
pchRecord += cchName;
|
|
|
|
// print question or resource record
|
|
|
|
if ( isection == 0 )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
" QTYPE %d\r\n"
|
|
" QCLASS %d\r\n",
|
|
FlipUnalignedWord( pchRecord ),
|
|
FlipUnalignedWord( pchRecord + sizeof(WORD) )
|
|
);
|
|
pchRecord += sizeof( DNS_WIRE_QUESTION );
|
|
}
|
|
else
|
|
{
|
|
pchRecord += DnsPrint_PacketRecord(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
(PDNS_WIRE_RECORD) pchRecord,
|
|
pmsgHeader,
|
|
pmsgEnd
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// check that at proper end of packet
|
|
|
|
wOffset = (WORD)(pchRecord - (PCHAR)pmsgHeader);
|
|
if ( pchRecord < pMsg->pCurrent || wOffset < wLength )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"WARNING: message continues beyond these records\r\n"
|
|
"\tpch = %p, pCurrent = %p, %d bytes\r\n"
|
|
"\toffset = %d, msg length = %d, %d bytes\r\n",
|
|
pchRecord,
|
|
pMsg->pCurrent,
|
|
pMsg->pCurrent - pchRecord,
|
|
wOffset,
|
|
wLength,
|
|
wLength - wOffset );
|
|
}
|
|
PrintRoutine(
|
|
pContext,
|
|
" Message length = %04x\n\r\n",
|
|
wOffset );
|
|
|
|
Unlock:
|
|
DnsPrint_Unlock();
|
|
|
|
|
|
} // DnsPrint_Message
|
|
|
|
|
|
|
|
INT
|
|
DnsPrint_PacketRecord(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_WIRE_RECORD pMsgRR,
|
|
IN PDNS_HEADER pMsgHead, OPTIONAL
|
|
IN PBYTE pMsgEnd OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print RR in packet format.
|
|
|
|
Arguments:
|
|
|
|
pszHeader - Header message/name for RR.
|
|
|
|
pMsgRR - resource record to print
|
|
|
|
pMsgHead - ptr to DNS message; need for offsetting, if not given
|
|
names are not printed past first offset
|
|
|
|
pMsgEnd - ptr to end of message, specifically byte immediately after
|
|
message
|
|
|
|
Return Value:
|
|
|
|
Number of bytes in record.
|
|
|
|
--*/
|
|
{
|
|
PCHAR pdata = (PCHAR)(pMsgRR + 1);
|
|
PCHAR pdataStop;
|
|
WORD dlen = FlipUnalignedWord( &pMsgRR->DataLength );
|
|
WORD type;
|
|
PCHAR pRRString;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
//
|
|
// print RR fixed fields
|
|
//
|
|
|
|
type = FlipUnalignedWord( &pMsgRR->RecordType );
|
|
pRRString = Dns_RecordStringForType( type );
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n",
|
|
pszHeader );
|
|
}
|
|
PrintRoutine(
|
|
pContext,
|
|
" TYPE %s (%u)\r\n"
|
|
" CLASS %u\r\n"
|
|
" TTL %lu\r\n"
|
|
" DLEN %u\r\n"
|
|
" DATA ",
|
|
pRRString,
|
|
type,
|
|
FlipUnalignedWord( &pMsgRR->RecordClass ),
|
|
FlipUnalignedDword( &pMsgRR->TimeToLive ),
|
|
dlen );
|
|
|
|
//
|
|
// update records may not have data
|
|
//
|
|
|
|
if ( dlen == 0 )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"(none)\r\n" );
|
|
goto Done;
|
|
}
|
|
|
|
// stop byte after RR data
|
|
|
|
pdataStop = pdata + dlen;
|
|
if ( pMsgEnd < pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: record at %p extends past end of packet!\n"
|
|
"\tpmsg = %p\n"
|
|
"\tpmsgEnd = %p\n"
|
|
"\trecord end = %p\n",
|
|
pMsgRR,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
pdataStop );
|
|
goto Done;
|
|
}
|
|
|
|
//
|
|
// print RR data
|
|
//
|
|
|
|
switch ( type )
|
|
{
|
|
|
|
case DNS_TYPE_A:
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%d.%d.%d.%d\r\n",
|
|
* (PUCHAR)( pdata + 0 ),
|
|
* (PUCHAR)( pdata + 1 ),
|
|
* (PUCHAR)( pdata + 2 ),
|
|
* (PUCHAR)( pdata + 3 )
|
|
);
|
|
break;
|
|
|
|
case DNS_TYPE_AAAA:
|
|
{
|
|
IP6_ADDRESS ip6;
|
|
|
|
RtlCopyMemory(
|
|
&ip6,
|
|
pdata,
|
|
sizeof(ip6) );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%04x:%04x:%04x:%04x:%04x:%04x:%04x:%04x\n",
|
|
ip6.IP6Word[0],
|
|
ip6.IP6Word[1],
|
|
ip6.IP6Word[2],
|
|
ip6.IP6Word[3],
|
|
ip6.IP6Word[4],
|
|
ip6.IP6Word[5],
|
|
ip6.IP6Word[6],
|
|
ip6.IP6Word[7]
|
|
);
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_PTR:
|
|
case DNS_TYPE_NS:
|
|
case DNS_TYPE_CNAME:
|
|
case DNS_TYPE_MD:
|
|
case DNS_TYPE_MB:
|
|
case DNS_TYPE_MF:
|
|
case DNS_TYPE_MG:
|
|
case DNS_TYPE_MR:
|
|
|
|
//
|
|
// these RRs contain single domain name
|
|
//
|
|
|
|
DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
break;
|
|
|
|
case DNS_TYPE_MX:
|
|
case DNS_TYPE_RT:
|
|
case DNS_TYPE_AFSDB:
|
|
|
|
//
|
|
// these RR contain
|
|
// - one preference value
|
|
// - one domain name
|
|
//
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%d ",
|
|
FlipUnalignedWord( pdata )
|
|
);
|
|
DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
pdata + sizeof(WORD),
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
break;
|
|
|
|
case DNS_TYPE_SOA:
|
|
|
|
pdata += DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\r\n\t\tPrimaryServer: ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
NULL );
|
|
pdata += DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\r\n\t\tAdministrator: ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t\tSerialNo = %d\r\n"
|
|
"\t\tRefresh = %d\r\n"
|
|
"\t\tRetry = %d\r\n"
|
|
"\t\tExpire = %d\r\n"
|
|
"\t\tMinimumTTL = %d\r\n",
|
|
FlipUnalignedDword( pdata ),
|
|
FlipUnalignedDword( (PDWORD)pdata+1 ),
|
|
FlipUnalignedDword( (PDWORD)pdata+2 ),
|
|
FlipUnalignedDword( (PDWORD)pdata+3 ),
|
|
FlipUnalignedDword( (PDWORD)pdata+4 )
|
|
);
|
|
break;
|
|
|
|
case DNS_TYPE_MINFO:
|
|
case DNS_TYPE_RP:
|
|
|
|
//
|
|
// these RRs contain two domain names
|
|
//
|
|
|
|
pdata += DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
NULL );
|
|
DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
" ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
break;
|
|
|
|
case DNS_TYPE_TEXT:
|
|
case DNS_TYPE_HINFO:
|
|
case DNS_TYPE_ISDN:
|
|
case DNS_TYPE_X25:
|
|
{
|
|
//
|
|
// all these are simply text string(s)
|
|
//
|
|
|
|
PCHAR pch = pdata;
|
|
PCHAR pchStop = pch + dlen;
|
|
UCHAR cch;
|
|
|
|
while ( pch < pchStop )
|
|
{
|
|
cch = (UCHAR) *pch++;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t%.*s\r\n",
|
|
cch,
|
|
pch );
|
|
|
|
pch += cch;
|
|
}
|
|
if ( pch != pchStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: Bad RR. "
|
|
"Text strings do not add to RR length.\r\n" );
|
|
}
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_WKS:
|
|
{
|
|
INT i;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"WKS: Address %d.%d.%d.%d\r\n"
|
|
"\t\tProtocol %d\r\n"
|
|
"\t\tBitmask\r\n",
|
|
* (PUCHAR)( pdata + 0 ),
|
|
* (PUCHAR)( pdata + 1 ),
|
|
* (PUCHAR)( pdata + 2 ),
|
|
* (PUCHAR)( pdata + 3 ),
|
|
* (PUCHAR)( pdata + 4 ) );
|
|
|
|
pdata += SIZEOF_WKS_FIXED_DATA;
|
|
|
|
for ( i=0; i < (INT)(dlen-SIZEOF_WKS_FIXED_DATA); i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t\t\tbyte[%d] = %x\r\n",
|
|
i,
|
|
(UCHAR) pdata[i] );
|
|
}
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_NULL:
|
|
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
"\t\t",
|
|
pdata,
|
|
dlen );
|
|
break;
|
|
|
|
case DNS_TYPE_SRV:
|
|
|
|
// SRV <priority> <weight> <port> <target host>
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t\tPriority = %d\r\n"
|
|
"\t\tWeight = %d\r\n"
|
|
"\t\tPort = %d\r\n",
|
|
FlipUnalignedWord( pdata ),
|
|
FlipUnalignedWord( (PWORD)pdata+1 ),
|
|
FlipUnalignedWord( (PWORD)pdata+2 )
|
|
);
|
|
DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tTarget host ",
|
|
pdata + 3*sizeof(WORD),
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
break;
|
|
|
|
case DNS_TYPE_OPT:
|
|
|
|
//
|
|
// OPT
|
|
// - RR class is buffer size
|
|
// - RR TTL contains
|
|
// <extended RCODE> low byte
|
|
// <version> second byte
|
|
// <flags-zero> high word
|
|
//
|
|
|
|
{
|
|
BYTE version;
|
|
BYTE extendedRcode;
|
|
DWORD fullRcode = 0;
|
|
WORD flags;
|
|
|
|
extendedRcode = *( (PBYTE) &pMsgRR->TimeToLive );
|
|
version = *( (PBYTE) &pMsgRR->TimeToLive + 1 );
|
|
flags = *( (PWORD) &pMsgRR->TimeToLive + 1 );
|
|
|
|
if ( pMsgHead->ResponseCode )
|
|
{
|
|
fullRcode = ((DWORD)extendedRcode << 4) +
|
|
(DWORD)pMsgHead->ResponseCode;
|
|
}
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t\tBuffer Size = %d\r\n"
|
|
"\t\tRcode Ext = %d (%x)\r\n"
|
|
"\t\tRcode Full = %d\r\n"
|
|
"\t\tVersion = %d\r\n"
|
|
"\t\tFlags = %x\r\n",
|
|
FlipUnalignedWord( &pMsgRR->RecordClass ),
|
|
extendedRcode, extendedRcode,
|
|
fullRcode,
|
|
version,
|
|
flags );
|
|
}
|
|
break;
|
|
|
|
case DNS_TYPE_TKEY:
|
|
{
|
|
DWORD beginTime;
|
|
DWORD expireTime;
|
|
WORD keyLength;
|
|
WORD mode;
|
|
WORD extRcode;
|
|
WORD otherLength;
|
|
|
|
otherLength = (WORD)DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\r\n\t\tAlgorithm: ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
NULL );
|
|
if ( !otherLength )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid algorithm name in TKEY RR!\r\n" );
|
|
}
|
|
pdata += otherLength;
|
|
|
|
beginTime = InlineFlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
expireTime = InlineFlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
|
|
mode = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
extRcode = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
keyLength = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tCreate time = %d\r\n"
|
|
"\t\tExpire time = %d\r\n"
|
|
"\t\tMode = %d\r\n"
|
|
"\t\tExtended RCODE = %d\r\n"
|
|
"\t\tKey Length = %d\r\n",
|
|
beginTime,
|
|
expireTime,
|
|
mode,
|
|
extRcode,
|
|
keyLength );
|
|
|
|
if ( pdata + keyLength > pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid key length: exceeds record data!\r\n" );
|
|
break;
|
|
}
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tKey:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
keyLength );
|
|
|
|
pdata += keyLength;
|
|
if ( pdata + sizeof(WORD) > pdataStop )
|
|
{
|
|
break;
|
|
}
|
|
otherLength = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tOther Length = %d\r\n",
|
|
otherLength );
|
|
|
|
if ( pdata + otherLength > pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid other data length: exceeds record data!\r\n" );
|
|
break;
|
|
}
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tOther Data:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
otherLength );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_TSIG:
|
|
{
|
|
ULONGLONG signTime;
|
|
WORD fudgeTime;
|
|
WORD sigLength;
|
|
WORD extRcode;
|
|
WORD wOriginalId;
|
|
WORD otherLength;
|
|
|
|
otherLength = (WORD) DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\r\n\t\tAlgorithm: ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
NULL );
|
|
if ( !otherLength )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid algorithm name in TSIG RR!\r\n" );
|
|
}
|
|
pdata += otherLength;
|
|
|
|
signTime = InlineFlipUnaligned48Bits( pdata );
|
|
pdata += sizeof(DWORD) + sizeof(WORD);
|
|
|
|
fudgeTime = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
sigLength = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tSigned time = %I64u\r\n"
|
|
"\t\tFudge time = %u\r\n"
|
|
"\t\tSig Length = %u\r\n",
|
|
signTime,
|
|
fudgeTime,
|
|
sigLength );
|
|
|
|
if ( pdata + sigLength > pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid signature length: exceeds record data!\r\n" );
|
|
break;
|
|
}
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tSignature:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
sigLength );
|
|
|
|
pdata += sigLength;
|
|
if ( pdata + sizeof(DWORD) > pdataStop )
|
|
{
|
|
break;
|
|
}
|
|
wOriginalId = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
extRcode = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
otherLength = InlineFlipUnalignedWord( pdata );
|
|
pdata += sizeof(WORD);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tOriginal XID = %x\r\n"
|
|
"\t\tExtended RCODE = %u\r\n"
|
|
"\t\tOther Length = %u\r\n",
|
|
wOriginalId,
|
|
extRcode,
|
|
otherLength );
|
|
|
|
if ( pdata + otherLength > pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"Invalid other data length: exceeds record data!\r\n" );
|
|
break;
|
|
}
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tOther Data:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
otherLength );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_WINS:
|
|
{
|
|
DWORD i;
|
|
DWORD winsFlags;
|
|
DWORD lookupTimeout;
|
|
DWORD cacheTimeout;
|
|
DWORD winsCount;
|
|
CHAR flagString[ WINS_FLAG_MAX_LENGTH ];
|
|
|
|
//
|
|
// WINS
|
|
// - scope/domain mapping flag
|
|
// - lookup timeout
|
|
// - cache timeout
|
|
// - WINS server count
|
|
// - WINS server list
|
|
//
|
|
|
|
winsFlags = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
lookupTimeout = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
cacheTimeout = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
winsCount = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
|
|
Dns_WinsRecordFlagString(
|
|
winsFlags,
|
|
flagString );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tWINS flags = %s (%08x)\r\n"
|
|
"\t\tLookup timeout = %d\r\n"
|
|
"\t\tCaching TTL = %d\r\n",
|
|
flagString,
|
|
winsFlags,
|
|
lookupTimeout,
|
|
cacheTimeout );
|
|
|
|
if ( pdata + (winsCount * SIZEOF_IP_ADDRESS) > pdataStop )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: WINS server count leads beyond record data length!\n"
|
|
"\tpmsg = %p\n"
|
|
"\tpmsgEnd = %p\n"
|
|
"\tpRR = %p\n"
|
|
"\trecord data end = %p\n"
|
|
"\twins count = %d\n"
|
|
"\tend of wins IPs = %p\n",
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
pMsgRR,
|
|
pdataStop,
|
|
winsCount,
|
|
pdata + (winsCount * SIZEOF_IP_ADDRESS)
|
|
);
|
|
goto Done;
|
|
}
|
|
|
|
DnsPrint_IpAddressArray(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
"\tWINS",
|
|
winsCount,
|
|
(PIP_ADDRESS) pdata );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_WINSR:
|
|
{
|
|
DWORD winsFlags;
|
|
DWORD lookupTimeout;
|
|
DWORD cacheTimeout;
|
|
CHAR flagString[ WINS_FLAG_MAX_LENGTH ];
|
|
|
|
//
|
|
// NBSTAT
|
|
// - scope/domain mapping flag
|
|
// - lookup timeout
|
|
// - cache timeout
|
|
// - result domain -- optional
|
|
//
|
|
|
|
winsFlags = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
lookupTimeout = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
cacheTimeout = FlipUnalignedDword( pdata );
|
|
pdata += sizeof(DWORD);
|
|
|
|
Dns_WinsRecordFlagString(
|
|
winsFlags,
|
|
flagString );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tWINS-R flags = %s (%08x)\r\n"
|
|
"\t\tLookup timeout = %d\r\n"
|
|
"\t\tCaching TTL = %d\r\n",
|
|
flagString,
|
|
winsFlags,
|
|
lookupTimeout,
|
|
cacheTimeout );
|
|
|
|
DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tResult domain = ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_KEY:
|
|
{
|
|
WORD flags;
|
|
BYTE protocol;
|
|
BYTE algorithm;
|
|
INT keyLength;
|
|
CHAR szKeyFlags[ 100 ];
|
|
|
|
keyLength = dlen - SIZEOF_KEY_FIXED_DATA;
|
|
|
|
flags = FlipUnalignedWord( pdata );
|
|
pdata += sizeof( WORD );
|
|
protocol = * ( PBYTE ) pdata;
|
|
++pdata;
|
|
algorithm = * ( PBYTE ) pdata;
|
|
++pdata;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tKEY flags = 0x%04x %s\r\n"
|
|
"\t\tKEY protocol = %s (%d)\r\n"
|
|
"\t\tKEY algorithm = %s (%d)\r\n",
|
|
(INT) flags,
|
|
Dns_KeyFlagString( szKeyFlags, flags ),
|
|
Dns_GetKeyProtocolString( protocol ),
|
|
(INT) protocol,
|
|
Dns_GetDnssecAlgorithmString( algorithm ),
|
|
(INT) algorithm );
|
|
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tPublic key:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
keyLength );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_SIG:
|
|
{
|
|
WORD typeCovered;
|
|
BYTE algorithm;
|
|
BYTE labelCount;
|
|
DWORD originalTTL;
|
|
DWORD sigInception;
|
|
DWORD sigExpiration;
|
|
WORD keyTag;
|
|
CHAR szSigInception[ 100 ];
|
|
CHAR szSigExpiration[ 100 ];
|
|
INT sigLength;
|
|
|
|
typeCovered = FlipUnalignedWord( pdata );
|
|
pdata += sizeof( WORD );
|
|
algorithm = * ( PBYTE ) pdata;
|
|
++pdata;
|
|
labelCount = * ( PBYTE ) pdata;
|
|
++pdata;
|
|
originalTTL = FlipUnalignedDword( pdata );
|
|
pdata += sizeof( DWORD );
|
|
sigExpiration = FlipUnalignedDword( pdata );
|
|
pdata += sizeof( DWORD );
|
|
sigInception = FlipUnalignedDword( pdata );
|
|
pdata += sizeof( DWORD );
|
|
keyTag = FlipUnalignedWord( pdata );
|
|
pdata += sizeof( WORD );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\r\n"
|
|
"\t\tSIG type covered = %s\r\n"
|
|
"\t\tSIG algorithm = %s (%d)\r\n"
|
|
"\t\tSIG label count = %d\r\n"
|
|
"\t\tSIG original TTL = %d\r\n"
|
|
"\t\tSIG expiration = %s\r\n"
|
|
"\t\tSIG inception = %s\r\n"
|
|
"\t\tSIG key tag = %d\r\n",
|
|
Dns_RecordStringForType( typeCovered ),
|
|
Dns_GetDnssecAlgorithmString( ( BYTE ) algorithm ),
|
|
( INT ) algorithm,
|
|
( INT ) labelCount,
|
|
( INT ) originalTTL,
|
|
Dns_SigTimeString( sigExpiration, szSigExpiration ),
|
|
Dns_SigTimeString( sigInception, szSigInception ),
|
|
( INT ) keyTag );
|
|
|
|
pdata += DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tSIG signer's name = ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
|
|
sigLength = ( INT ) ( pdataStop - pdata );
|
|
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\t\tSignature:",
|
|
"\t\t ", // line header
|
|
pdata,
|
|
sigLength );
|
|
break;
|
|
}
|
|
|
|
case DNS_TYPE_NXT:
|
|
{
|
|
INT bitmapLength;
|
|
INT byteIdx;
|
|
INT bitIdx;
|
|
|
|
pdata += DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\r\n\t\tNXT next name = ",
|
|
pdata,
|
|
pMsgHead,
|
|
pMsgEnd,
|
|
"\r\n" );
|
|
|
|
bitmapLength = ( INT ) ( pdataStop - pdata );
|
|
|
|
PrintRoutine( pContext, "\t\tNXT types covered = " );
|
|
|
|
for ( byteIdx = 0; byteIdx < bitmapLength; ++byteIdx )
|
|
{
|
|
for ( bitIdx = ( byteIdx ? 0 : 1 ); bitIdx < 8; ++bitIdx )
|
|
{
|
|
PCHAR pszType;
|
|
|
|
if ( !( pdata[ byteIdx ] & ( 1 << bitIdx ) ) )
|
|
{
|
|
continue; // Bit value is zero - do not write string.
|
|
}
|
|
pszType = Dns_RecordStringForType( byteIdx * 8 + bitIdx );
|
|
if ( !pszType )
|
|
{
|
|
ASSERT( FALSE );
|
|
continue; // This type has no string - do not write.
|
|
}
|
|
PrintRoutine( pContext, "%s ", pszType );
|
|
}
|
|
}
|
|
|
|
PrintRoutine( pContext, "\r\n" );
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"Unknown resource record type %d at %p.\r\n",
|
|
type,
|
|
pMsgRR );
|
|
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
"\t\t",
|
|
pdata,
|
|
dlen );
|
|
break;
|
|
}
|
|
|
|
Done:
|
|
|
|
DnsPrint_Unlock();
|
|
|
|
return( sizeof(DNS_WIRE_RECORD) + dlen );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Print related utilities
|
|
//
|
|
|
|
|
|
INT
|
|
Dns_WriteFormattedSystemTimeToBuffer(
|
|
OUT PCHAR pBuffer,
|
|
IN PSYSTEMTIME pSystemTime
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Write SYSTEMTIME structure to buffer.
|
|
|
|
Arguments:
|
|
|
|
pBuffer -- buffer to write into, assumed to have at least 50
|
|
bytes available
|
|
|
|
pSystemTime -- system time to convert; assumed to be local, no
|
|
time zone conversion is done
|
|
|
|
Return Value:
|
|
|
|
Bytes in formatted string.
|
|
|
|
--*/
|
|
{
|
|
PCHAR pend = pBuffer + 60;
|
|
PCHAR pstart = pBuffer;
|
|
INT count;
|
|
|
|
pBuffer += GetDateFormat(
|
|
LOCALE_SYSTEM_DEFAULT,
|
|
LOCALE_NOUSEROVERRIDE,
|
|
(PSYSTEMTIME) pSystemTime,
|
|
NULL,
|
|
pBuffer,
|
|
(int)(pend - pBuffer) );
|
|
|
|
// Replace NULL from GetDateFormat with a space.
|
|
|
|
*( pBuffer - 1 ) = ' ';
|
|
|
|
pBuffer += GetTimeFormat(
|
|
LOCALE_SYSTEM_DEFAULT,
|
|
LOCALE_NOUSEROVERRIDE,
|
|
(PSYSTEMTIME) pSystemTime,
|
|
NULL,
|
|
pBuffer,
|
|
(int)(pend - pBuffer) );
|
|
|
|
if ( pBuffer <= pstart+1 )
|
|
{
|
|
return( 0 );
|
|
}
|
|
return (INT)( pBuffer - pstart );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Response code print
|
|
//
|
|
|
|
#define DNS_RCODE_UNKNOWN (DNS_RCODE_BADTIME + 1)
|
|
|
|
PCHAR ResponseCodeStringTable[] =
|
|
{
|
|
"NOERROR",
|
|
"FORMERR",
|
|
"SERVFAIL",
|
|
"NXDOMAIN",
|
|
"NOTIMP",
|
|
"REFUSED",
|
|
"YXDOMAIN",
|
|
"YXRRSET",
|
|
"NXRRSET",
|
|
"NOTAUTH",
|
|
"NOTZONE",
|
|
"11 - unknown\r\n",
|
|
"12 - unknown\r\n",
|
|
"13 - unknown\r\n",
|
|
"14 - unknown\r\n",
|
|
"15 - unknown\r\n",
|
|
|
|
// DNS RCODEs stop at 15 -- these extended errors are available for security
|
|
|
|
"BADSIG",
|
|
"BADKEY",
|
|
"BADTIME",
|
|
"UNKNOWN"
|
|
};
|
|
|
|
|
|
PCHAR
|
|
Dns_ResponseCodeString(
|
|
IN INT ResponseCode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get string corresponding to a response code.
|
|
|
|
Arguments:
|
|
|
|
ResponseCode - response code
|
|
|
|
Return Value:
|
|
|
|
Ptr to string for code.
|
|
|
|
--*/
|
|
{
|
|
if ( ResponseCode > DNS_RCODE_UNKNOWN )
|
|
{
|
|
ResponseCode = DNS_RCODE_UNKNOWN;
|
|
}
|
|
return( ResponseCodeStringTable[ ResponseCode ] );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// More detailed RCODE strings
|
|
//
|
|
|
|
PCHAR ResponseCodeExplanationStringTable[] =
|
|
{
|
|
"NOERROR: no error",
|
|
"FORMERR: format error",
|
|
"SERVFAIL: server failure",
|
|
"NXDOMAIN: name error",
|
|
"NOTIMP: not implemented",
|
|
"REFUSED",
|
|
"YXDOMAIN: name exists that should not",
|
|
"YXRRSET: RR set exists that should not",
|
|
"NXRRSET: required RR set does not exist",
|
|
"NOTAUTH: not authoritative",
|
|
"NOTZONE: name not in zone",
|
|
"11 - unknown",
|
|
"12 - unknown",
|
|
"13 - unknown",
|
|
"14 - unknown",
|
|
"15 - unknown",
|
|
|
|
// DNS RCODEs stop at 15 -- these extended errors are available for security
|
|
|
|
"BADSIG: bad signature",
|
|
"BADKEY: bad signature",
|
|
"BADTIME: invalid or expired time on signature or key",
|
|
"UNKNOWN"
|
|
};
|
|
|
|
|
|
PCHAR
|
|
Dns_ResponseCodeExplanationString(
|
|
IN INT ResponseCode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get string corresponding to a response code.
|
|
Basically for use by packet debug routine above.
|
|
|
|
Arguments:
|
|
|
|
ResponseCode - response code
|
|
|
|
Return Value:
|
|
|
|
Ptr to string for code.
|
|
|
|
--*/
|
|
{
|
|
if ( ResponseCode > DNS_RCODE_UNKNOWN )
|
|
{
|
|
ResponseCode = DNS_RCODE_UNKNOWN;
|
|
}
|
|
return( ResponseCodeExplanationStringTable[ ResponseCode ] );
|
|
}
|
|
|
|
|
|
|
|
PCHAR
|
|
Dns_KeyFlagString(
|
|
IN OUT PCHAR pszBuff,
|
|
IN WORD Flags
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Formats a human-readable string based on the flags value
|
|
(DNSSEC KEY RR flags). See RFC2535 section 3.2.1.
|
|
|
|
Arguments:
|
|
|
|
pszBuff - buffer to dump string into should be min 100 chars
|
|
|
|
flags - flag value to generate string for
|
|
|
|
Return Value:
|
|
|
|
pszBuff
|
|
|
|
--*/
|
|
{
|
|
BOOL fZoneKey = FALSE;
|
|
|
|
*pszBuff = '\0';
|
|
|
|
// "type" bits
|
|
|
|
if ( ( Flags & 0xC000 ) == 0xC000 )
|
|
{
|
|
strcat( pszBuff, "NOKEY " );
|
|
}
|
|
else if ( ( Flags & 0xC000 ) == 0x8000 )
|
|
{
|
|
strcat( pszBuff, "NOAUTH " );
|
|
}
|
|
else if ( ( Flags & 0xC000 ) == 0x4000 )
|
|
{
|
|
strcat( pszBuff, "NOCONF " );
|
|
}
|
|
else
|
|
{
|
|
strcat( pszBuff, "NOAUTH NOCONF " );
|
|
}
|
|
|
|
// extended bit
|
|
|
|
if ( Flags & 0x1000 )
|
|
{
|
|
strcat( pszBuff, "EXTEND " );
|
|
}
|
|
|
|
// name type bits
|
|
|
|
if ( ( Flags & 0x0300 ) == 0x0300 )
|
|
{
|
|
strcat( pszBuff, "RESNT " ); // reserved name type
|
|
}
|
|
else if ( ( Flags & 0x0200 ) == 0x0100 )
|
|
{
|
|
strcat( pszBuff, "ENTITY " );
|
|
}
|
|
else if ( ( Flags & 0x0100 ) == 0x4000 )
|
|
{
|
|
strcat( pszBuff, "ZONE " );
|
|
fZoneKey = TRUE;
|
|
}
|
|
else
|
|
{
|
|
strcat( pszBuff, "USER " );
|
|
}
|
|
|
|
// signatory bits
|
|
|
|
if ( fZoneKey )
|
|
{
|
|
strcat( pszBuff, ( Flags & 0x0008 ) ? "MODEA " : "MODEB " );
|
|
if ( Flags & 0x0004 )
|
|
{
|
|
strcat( pszBuff, "STRONG " );
|
|
}
|
|
if ( Flags & 0x0002 )
|
|
{
|
|
strcat( pszBuff, "UNIQ " );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if ( Flags & 0x0008 )
|
|
{
|
|
strcat( pszBuff, "ZCTRL " );
|
|
}
|
|
if ( Flags & 0x0004 )
|
|
{
|
|
strcat( pszBuff, "STRONG " );
|
|
}
|
|
if ( Flags & 0x0002 )
|
|
{
|
|
strcat( pszBuff, "UNIQ " );
|
|
}
|
|
}
|
|
|
|
return pszBuff;
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Opcode print
|
|
//
|
|
|
|
PCHAR OpcodeStringTable[] =
|
|
{
|
|
"QUERY",
|
|
"IQUERY",
|
|
"SRV_STATUS",
|
|
"UNKNOWN",
|
|
"NOTIFY",
|
|
"UPDATE",
|
|
"UNKNOWN?"
|
|
};
|
|
|
|
CHAR OpcodeCharacterTable[] =
|
|
{
|
|
'Q',
|
|
'I',
|
|
'S',
|
|
'K',
|
|
'N',
|
|
'U',
|
|
'?'
|
|
};
|
|
|
|
#define DNS_OPCODE_UNSPEC (DNS_OPCODE_UPDATE + 1)
|
|
|
|
|
|
PCHAR
|
|
Dns_OpcodeString(
|
|
IN INT Opcode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get string corresponding to a response code.
|
|
|
|
Arguments:
|
|
|
|
Opcode - response code
|
|
|
|
Return Value:
|
|
|
|
Ptr to string for code.
|
|
|
|
--*/
|
|
{
|
|
if ( Opcode > DNS_OPCODE_UNSPEC )
|
|
{
|
|
Opcode = DNS_OPCODE_UNSPEC;
|
|
}
|
|
return( OpcodeStringTable[ Opcode ] );
|
|
}
|
|
|
|
|
|
|
|
CHAR
|
|
Dns_OpcodeCharacter(
|
|
IN INT Opcode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get string corresponding to an opcode.
|
|
|
|
Arguments:
|
|
|
|
Opcode - response code
|
|
|
|
Return Value:
|
|
|
|
Ptr to string for code.
|
|
|
|
--*/
|
|
{
|
|
if ( Opcode > DNS_OPCODE_UNSPEC )
|
|
{
|
|
Opcode = DNS_OPCODE_UNSPEC;
|
|
}
|
|
return( OpcodeCharacterTable[ Opcode ] );
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Section names
|
|
//
|
|
// With update get a new set of section names.
|
|
// Provide single interface to putting a name on them.
|
|
//
|
|
|
|
PSTR SectionNameArray[5] =
|
|
{
|
|
"Question",
|
|
"Answer",
|
|
"Authority",
|
|
"Additional",
|
|
"ERROR: Invalid Section"
|
|
};
|
|
|
|
PSTR UpdateSectionNameArray[5] =
|
|
{
|
|
"Zone",
|
|
"Prerequisite",
|
|
"Update",
|
|
"Additional",
|
|
"ERROR: Invalid Section"
|
|
};
|
|
|
|
PCHAR
|
|
Dns_SectionNameString(
|
|
IN INT iSection,
|
|
IN INT iOpcode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Get string corresponding to name of RR section id.
|
|
For use by packet debug routine above.
|
|
|
|
Arguments:
|
|
|
|
iSection - section id (0-3 for Question-Additional)
|
|
|
|
iOpcode - opcode
|
|
|
|
Return Value:
|
|
|
|
Ptr to string for section name.
|
|
|
|
--*/
|
|
{
|
|
if ( iSection >= 4 )
|
|
{
|
|
iSection = 4;
|
|
}
|
|
|
|
if ( iOpcode == DNS_OPCODE_UPDATE )
|
|
{
|
|
return( UpdateSectionNameArray[iSection] );
|
|
}
|
|
else
|
|
{
|
|
return( SectionNameArray[iSection] );
|
|
}
|
|
}
|
|
|
|
|
|
VOID
|
|
DnsPrint_MessageNoContext(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_HEADER pMsgHead,
|
|
IN WORD wLength OPTIONAL
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print DNS message buffer.
|
|
Includes context information as well as actual DNS message.
|
|
|
|
--*/
|
|
{
|
|
PCHAR pchRecord;
|
|
PBYTE pmsgEnd;
|
|
INT i;
|
|
INT isection;
|
|
INT cchName;
|
|
WORD wOffset;
|
|
WORD wXid;
|
|
WORD wQuestionCount;
|
|
WORD wAnswerCount;
|
|
WORD wNameServerCount;
|
|
WORD wAdditionalCount;
|
|
WORD countSectionRR;
|
|
BOOL fFlipped = FALSE;
|
|
|
|
//
|
|
// processing limits
|
|
// - if length given set stop limit
|
|
// - if length not given set wLength so checks against
|
|
// length overrun always fail (are ok)
|
|
//
|
|
|
|
if ( wLength )
|
|
{
|
|
pmsgEnd = ((PBYTE)pMsgHead) + wLength;
|
|
}
|
|
else
|
|
{
|
|
wLength = MAXWORD;
|
|
pmsgEnd = NULL;
|
|
}
|
|
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine( pContext, "%s\r\n", pszHeader );
|
|
}
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"DNS message header at %p\r\n",
|
|
pMsgHead );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
" XID 0x%04hx\r\n"
|
|
" Flags 0x%04hx\r\n"
|
|
" QR 0x%lx (%s)\r\n"
|
|
" OPCODE 0x%lx (%s)\r\n"
|
|
" AA 0x%lx\r\n"
|
|
" TC 0x%lx\r\n"
|
|
" RD 0x%lx\r\n"
|
|
" RA 0x%lx\r\n"
|
|
" Z 0x%lx\r\n"
|
|
" RCODE 0x%lx (%s)\r\n"
|
|
" QCOUNT 0x%hx\r\n"
|
|
" ACOUNT 0x%hx\r\n"
|
|
" NSCOUNT 0x%hx\r\n"
|
|
" ARCOUNT 0x%hx\r\n",
|
|
|
|
pMsgHead->Xid,
|
|
ntohs((*((PWORD)pMsgHead + 1))),
|
|
|
|
pMsgHead->IsResponse,
|
|
(pMsgHead->IsResponse ? "response" : "question"),
|
|
pMsgHead->Opcode,
|
|
Dns_OpcodeString( pMsgHead->Opcode ),
|
|
pMsgHead->Authoritative,
|
|
pMsgHead->Truncation,
|
|
pMsgHead->RecursionDesired,
|
|
pMsgHead->RecursionAvailable,
|
|
pMsgHead->Reserved,
|
|
pMsgHead->ResponseCode,
|
|
Dns_ResponseCodeString( pMsgHead->ResponseCode ),
|
|
|
|
pMsgHead->QuestionCount,
|
|
pMsgHead->AnswerCount,
|
|
pMsgHead->NameServerCount,
|
|
pMsgHead->AdditionalCount );
|
|
|
|
//
|
|
// determine if byte flipped and get correct count
|
|
//
|
|
|
|
wXid = pMsgHead->Xid;
|
|
wQuestionCount = pMsgHead->QuestionCount;
|
|
wAnswerCount = pMsgHead->AnswerCount;
|
|
wNameServerCount = pMsgHead->NameServerCount;
|
|
wAdditionalCount = pMsgHead->AdditionalCount;
|
|
|
|
if ( wQuestionCount )
|
|
{
|
|
fFlipped = wQuestionCount & 0xff00;
|
|
}
|
|
else if ( wNameServerCount )
|
|
{
|
|
fFlipped = wNameServerCount & 0xff00;
|
|
}
|
|
if ( fFlipped )
|
|
{
|
|
wXid = ntohs( wXid );
|
|
wQuestionCount = ntohs( wQuestionCount );
|
|
wAnswerCount = ntohs( wAnswerCount );
|
|
wNameServerCount = ntohs( wNameServerCount );
|
|
wAdditionalCount = ntohs( wAdditionalCount );
|
|
}
|
|
|
|
//
|
|
// catch record flipping problems -- all are flipped or none at all
|
|
// and no record count should be > 256 EXCEPT answer count
|
|
// during FAST zone transfer
|
|
//
|
|
|
|
DNS_ASSERT( ! (wQuestionCount & 0xff00) );
|
|
DNS_ASSERT( ! (wNameServerCount & 0xff00) );
|
|
DNS_ASSERT( ! (wAdditionalCount & 0xff00) );
|
|
|
|
#if 0
|
|
//
|
|
// stop here if WINS response -- don't have parsing ready
|
|
//
|
|
|
|
if ( pMsgHead->IsResponse && IS_WINS_XID(wXid) )
|
|
{
|
|
PrintRoutine( pContext, " WINS Response packet.\r\n\r\n" );
|
|
goto Unlock;
|
|
}
|
|
#endif
|
|
|
|
//
|
|
// print questions and resource records
|
|
//
|
|
|
|
pchRecord = (PCHAR)(pMsgHead + 1);
|
|
|
|
for ( isection=0; isection<4; isection++)
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
" %s Section:\r\n",
|
|
Dns_SectionNameString( isection, pMsgHead->Opcode ) );
|
|
|
|
if ( isection==0 )
|
|
{
|
|
countSectionRR = wQuestionCount;
|
|
}
|
|
else if ( isection==1 )
|
|
{
|
|
countSectionRR = wAnswerCount;
|
|
}
|
|
else if ( isection==2 )
|
|
{
|
|
countSectionRR = wNameServerCount;
|
|
}
|
|
else if ( isection==3 )
|
|
{
|
|
countSectionRR = wAdditionalCount;
|
|
}
|
|
|
|
for ( i=0; i < countSectionRR; i++ )
|
|
{
|
|
//
|
|
// verify not overrunning length
|
|
// - check against pCurrent as well as message length
|
|
// so can print packets while being built
|
|
//
|
|
|
|
wOffset = (WORD)(pchRecord - (PCHAR)pMsgHead);
|
|
|
|
if ( wOffset >= wLength )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: BOGUS PACKET:\r\n"
|
|
"\tFollowing RR (offset %d) past packet length (%d).\r\n",
|
|
wOffset,
|
|
wLength
|
|
);
|
|
goto Unlock;
|
|
}
|
|
|
|
//
|
|
// print RR name
|
|
//
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
" Name Offset = 0x%04x\r\n",
|
|
wOffset
|
|
);
|
|
|
|
cchName = DnsPrint_PacketName(
|
|
PrintRoutine,
|
|
pContext,
|
|
" Name \"",
|
|
pchRecord,
|
|
pMsgHead,
|
|
pmsgEnd,
|
|
"\"\r\n" );
|
|
if ( !cchName )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"ERROR: Invalid name length, stop packet print\r\n" );
|
|
DNS_ASSERT( FALSE );
|
|
break;
|
|
}
|
|
pchRecord += cchName;
|
|
|
|
// print question or resource record
|
|
|
|
if ( isection == 0 )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
" QTYPE %d\r\n"
|
|
" QCLASS %d\r\n",
|
|
FlipUnalignedWord( pchRecord ),
|
|
FlipUnalignedWord( pchRecord + sizeof(WORD) )
|
|
);
|
|
pchRecord += sizeof( DNS_WIRE_QUESTION );
|
|
}
|
|
else
|
|
{
|
|
pchRecord += DnsPrint_PacketRecord(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
(PDNS_WIRE_RECORD) pchRecord,
|
|
pMsgHead,
|
|
pmsgEnd
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
// check that at proper end of packet
|
|
|
|
wOffset = (WORD)(pchRecord - (PCHAR)pMsgHead);
|
|
PrintRoutine(
|
|
pContext,
|
|
" Message length = %04x\r\n\r\n",
|
|
wOffset );
|
|
|
|
// print warning if given message length and did not end up
|
|
// at end of message
|
|
// note: pmsgEnd test in case passed wLength==0, in which case
|
|
// wLength set to MAXDWORD above
|
|
|
|
if ( pmsgEnd && wOffset < wLength )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"WARNING: message continues beyond these records\r\n"
|
|
"\tpch = %p\r\n"
|
|
"\toffset = %d, msg length = %d, %d bytes\r\n",
|
|
pchRecord,
|
|
wOffset,
|
|
wLength,
|
|
wLength - wOffset );
|
|
|
|
DnsPrint_RawOctets(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Remaining bytes:",
|
|
NULL,
|
|
pchRecord,
|
|
(wLength - wOffset) );
|
|
}
|
|
|
|
Unlock:
|
|
DnsPrint_Unlock();
|
|
|
|
|
|
} // DnsPrint_MessageNoContext
|
|
|
|
|
|
|
|
DWORD
|
|
DnsStringPrint_Guid(
|
|
OUT PCHAR pBuffer,
|
|
IN PGUID pGuid
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print GUID to buffer.
|
|
|
|
Arguments:
|
|
|
|
pBuffer - buffer to print to
|
|
buffer must be big enough for GUID string
|
|
GUID_STRING_BUFFER_LENGTH covers it
|
|
|
|
pGuid - GUID to print
|
|
|
|
Return Value:
|
|
|
|
Count of bytes printed to string.
|
|
|
|
--*/
|
|
{
|
|
if ( !pGuid )
|
|
{
|
|
*pBuffer = 0;
|
|
return 0;
|
|
}
|
|
|
|
return sprintf(
|
|
pBuffer,
|
|
"%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
|
|
pGuid->Data1,
|
|
pGuid->Data2,
|
|
pGuid->Data3,
|
|
*(PWORD) &pGuid->Data4[0],
|
|
pGuid->Data4[2],
|
|
pGuid->Data4[3],
|
|
pGuid->Data4[4],
|
|
pGuid->Data4[5],
|
|
pGuid->Data4[6],
|
|
pGuid->Data4[7],
|
|
pGuid->Data4[8] );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Guid(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PGUID pGuid
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print GUID
|
|
|
|
Arguments:
|
|
|
|
pszHeader - Header message/name for RR.
|
|
|
|
pGuid -- ptr to GUID to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
CHAR guidBuffer[ GUID_STRING_BUFFER_LENGTH ];
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Guid";
|
|
}
|
|
if ( !pGuid )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s: NULL GUID pointer!\r\n",
|
|
pszHeader );
|
|
}
|
|
|
|
// convert GUID to string
|
|
|
|
DnsStringPrint_Guid(
|
|
guidBuffer,
|
|
pGuid );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s: (%p) %s\r\n",
|
|
pszHeader,
|
|
pGuid,
|
|
guidBuffer );
|
|
}
|
|
|
|
|
|
|
|
DWORD
|
|
DnsStringPrint_RawOctets(
|
|
OUT PCHAR pBuffer,
|
|
IN PCHAR pchData,
|
|
IN DWORD dwLength,
|
|
IN PSTR pszLineHeader,
|
|
IN DWORD dwLineLength
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print raw octect data to sting
|
|
|
|
Arguments:
|
|
|
|
pBuffer - buffer to print to
|
|
|
|
pchData - data to print
|
|
|
|
dwLength - length of data to print
|
|
|
|
pszLineHeader - header on each line.
|
|
|
|
dwLineLength - number of bytes to print on line; default is
|
|
|
|
Return Value:
|
|
|
|
Count of bytes printed to string.
|
|
|
|
--*/
|
|
{
|
|
INT i;
|
|
INT lineCount = 0;
|
|
PCHAR pch = pBuffer;
|
|
|
|
*pch = 0;
|
|
|
|
//
|
|
// catch NULL pointer
|
|
// - return is null terminated
|
|
// - but indicate no bytes written
|
|
//
|
|
|
|
if ( !pchData )
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
//
|
|
// write each byte in hex
|
|
// - if dwLineLength set break into lines with count
|
|
// or optional header
|
|
//
|
|
|
|
for ( i = 0; i < (INT)dwLength; i++ )
|
|
{
|
|
if ( dwLineLength && (i % dwLineLength) == 0 )
|
|
{
|
|
if ( pszLineHeader )
|
|
{
|
|
pch += sprintf( pch, "\r\n%s", pszLineHeader );
|
|
}
|
|
else
|
|
{
|
|
pch += sprintf( pch, "\r\n%3d> ", i );
|
|
}
|
|
lineCount++;
|
|
}
|
|
|
|
pch += sprintf( pch, "%02x ", (UCHAR)pchData[i] );
|
|
}
|
|
|
|
return( (DWORD)(pch - pBuffer) );
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_RawBinary(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR pszLineHeader,
|
|
IN PCHAR pchData,
|
|
IN DWORD dwLength,
|
|
IN DWORD PrintSize
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print raw data.
|
|
|
|
Arguments:
|
|
|
|
pszHeader - Header message/name for RR.
|
|
|
|
pszLineHeader - Header on each line.
|
|
|
|
pchData - data to print
|
|
|
|
dwLength - length of data to print
|
|
|
|
PrintSize - size to print in
|
|
size(QWORD)
|
|
size(DWORD)
|
|
size(WORD)
|
|
defaults to bytes
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
DWORD lineCount = 0;
|
|
CHAR buf[ 2000 ];
|
|
PCHAR pch = buf;
|
|
PCHAR pbyte;
|
|
PCHAR pend;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s",
|
|
pszHeader );
|
|
}
|
|
|
|
buf[0] = 0;
|
|
|
|
//
|
|
// print bytes
|
|
// - write 16 bytes a line
|
|
// - buffer up 10 lines for speed
|
|
//
|
|
// note: we'll write a partial (<16 byte) line the first
|
|
// time if data is unaligned with PrintSize, then we'll
|
|
// write at 16 a pop
|
|
//
|
|
|
|
if ( PrintSize == 0 )
|
|
{
|
|
PrintSize = 1;
|
|
}
|
|
|
|
i = 0;
|
|
pch = buf;
|
|
pend = (PBYTE)pchData + dwLength;
|
|
|
|
while ( i < dwLength )
|
|
{
|
|
DWORD lineBytes = (i%16);
|
|
|
|
if ( lineBytes==0 || lineBytes > (16-PrintSize) )
|
|
{
|
|
if ( lineCount > 10 )
|
|
{
|
|
PrintRoutine( pContext, buf );
|
|
lineCount = 0;
|
|
pch = buf;
|
|
}
|
|
|
|
if ( pszLineHeader )
|
|
{
|
|
pch += sprintf( pch, "\r\n%s", pszLineHeader );
|
|
}
|
|
else
|
|
{
|
|
pch += sprintf( pch, "\r\n\t%3d> ", i );
|
|
}
|
|
lineCount++;
|
|
|
|
//if ( i >= 128 && dlen > 256 )
|
|
//{
|
|
// PrintRoutine( pContext, "skipping remaining bytes ...\r\n" ));
|
|
//}
|
|
}
|
|
|
|
pbyte = &pchData[i];
|
|
|
|
if ( PrintSize == sizeof(QWORD) &&
|
|
POINTER_IS_ALIGNED( pbyte, ALIGN_QUAD ) &&
|
|
pbyte + sizeof(QWORD) <= pend )
|
|
{
|
|
pch += sprintf( pch, "%I64x ", *(PQWORD)pbyte );
|
|
i += sizeof(QWORD);
|
|
}
|
|
else if ( PrintSize == sizeof(DWORD) &&
|
|
POINTER_IS_ALIGNED( pbyte, ALIGN_DWORD ) &&
|
|
pbyte + sizeof(DWORD) <= pend )
|
|
{
|
|
pch += sprintf( pch, "%08x ", *(PDWORD)pbyte );
|
|
i += sizeof(DWORD);
|
|
}
|
|
else if ( PrintSize == sizeof(WORD) &&
|
|
POINTER_IS_ALIGNED( pbyte, ALIGN_WORD ) &&
|
|
pbyte + sizeof(WORD) <= pend )
|
|
{
|
|
pch += sprintf( pch, "%04x ", *(PWORD)pbyte );
|
|
i += sizeof(WORD);
|
|
}
|
|
else // default to byte print
|
|
{
|
|
pch += sprintf( pch, "%02x ", *pbyte );
|
|
i++;
|
|
}
|
|
}
|
|
|
|
// print remaining bytes in buffer
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n",
|
|
buf );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_RawOctets(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PSTR pszLineHeader,
|
|
IN PCHAR pchData,
|
|
IN DWORD dwLength
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print raw octect data.
|
|
|
|
Arguments:
|
|
|
|
pszHeader - Header message/name for RR.
|
|
|
|
pszLineHeader - Header on each line.
|
|
|
|
pchData - data to print
|
|
|
|
dwLength - length of data to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
INT i;
|
|
INT lineCount = 0;
|
|
CHAR buf[ 2000 ];
|
|
PCHAR pch = buf;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
if ( pszHeader )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s",
|
|
pszHeader );
|
|
}
|
|
|
|
buf[0] = 0;
|
|
|
|
// buffer every 20 lines for speed
|
|
|
|
for ( i = 0; i < (INT)dwLength; i++ )
|
|
{
|
|
if ( !(i%16) )
|
|
{
|
|
if ( lineCount > 10 )
|
|
{
|
|
PrintRoutine( pContext, buf );
|
|
lineCount = 0;
|
|
pch = buf;
|
|
}
|
|
|
|
if ( pszLineHeader )
|
|
{
|
|
pch += sprintf( pch, "\r\n%s", pszLineHeader );
|
|
}
|
|
else
|
|
{
|
|
pch += sprintf( pch, "\r\n%3d> ", i );
|
|
}
|
|
lineCount++;
|
|
|
|
//if ( i >= 128 && dlen > 256 )
|
|
//{
|
|
// PrintRoutine( pContext, "skipping remaining bytes ...\r\n" ));
|
|
//}
|
|
}
|
|
|
|
pch += sprintf( pch, "%02x ", (UCHAR)pchData[i] );
|
|
}
|
|
|
|
// print remaining bytes in buffer
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n",
|
|
buf );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_ParsedRecord(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_PARSED_RR pParsedRR
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print parsed RR structure.
|
|
|
|
Arguments:
|
|
|
|
pszHeader - Header message/name for RR.
|
|
|
|
pParsedRR - parsed RR to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Parsed RR:";
|
|
}
|
|
|
|
if ( !pParsedRR )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL ParsedRR ptr." );
|
|
return;
|
|
}
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tpchName = %p\r\n"
|
|
"\tpchRR = %p\r\n"
|
|
"\tpchData = %p\r\n"
|
|
"\tpchNextRR = %p\r\n"
|
|
"\twType = %d\r\n"
|
|
"\twClass = %d\r\n"
|
|
"\tTTL = %d\r\n"
|
|
"\twDataLength = %d\r\n",
|
|
pszHeader,
|
|
pParsedRR->pchName,
|
|
pParsedRR->pchRR,
|
|
pParsedRR->pchData,
|
|
pParsedRR->pchNextRR,
|
|
pParsedRR->Type,
|
|
pParsedRR->Class,
|
|
pParsedRR->Ttl,
|
|
pParsedRR->DataLength
|
|
);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Winsock RnR structures
|
|
//
|
|
|
|
VOID
|
|
DnsPrint_FdSet(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN struct fd_set * pfd_set
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print sockets in FD_SET.
|
|
|
|
--*/
|
|
{
|
|
INT count;
|
|
INT i;
|
|
|
|
DNS_ASSERT( pfd_set );
|
|
|
|
count = (INT) pfd_set->fd_count;
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s (count = %d)\r\n",
|
|
pszHeader ? pszHeader : "FD_SET:",
|
|
count );
|
|
|
|
for (i=0; i<count; i++)
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tsocket[%d] = %d\r\n",
|
|
i,
|
|
pfd_set->fd_array[i] );
|
|
}
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Sockaddr(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN DWORD Indent,
|
|
IN PSOCKADDR pSockaddr,
|
|
IN INT iSockaddrLength
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print sockaddr structure and length used in call.
|
|
|
|
--*/
|
|
{
|
|
PSTR pindent = INDENT_STRING( Indent );
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Sockaddr:";
|
|
}
|
|
|
|
if ( !pSockaddr )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\tNULL Sockaddr passed to print.\r\n",
|
|
pindent,
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\r\n"
|
|
"%s\tpointer = %p\r\n"
|
|
"%s\tlength = %d\r\n"
|
|
"%s\tsa_family = %d\r\n",
|
|
pindent, pszHeader,
|
|
pindent, pSockaddr,
|
|
pindent, iSockaddrLength,
|
|
pindent, pSockaddr->sa_family
|
|
);
|
|
|
|
switch ( pSockaddr->sa_family )
|
|
{
|
|
|
|
case AF_INET:
|
|
{
|
|
PSOCKADDR_IN psin = (PSOCKADDR_IN) pSockaddr;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\tsin_port = %04x\r\n"
|
|
"%s\tsin_addr = %s (%08x)\r\n"
|
|
"%s\tsin_zero = %08x %08x\r\n",
|
|
pindent, psin->sin_port,
|
|
pindent, inet_ntoa( psin->sin_addr ),
|
|
psin->sin_addr.s_addr,
|
|
pindent, *(PDWORD) &psin->sin_zero[0],
|
|
*(PDWORD) &psin->sin_zero[4]
|
|
);
|
|
break;
|
|
}
|
|
|
|
case AF_INET6:
|
|
{
|
|
PSOCKADDR_IN6 psin = (PSOCKADDR_IN6) pSockaddr;
|
|
|
|
CHAR buffer[ IP6_ADDRESS_STRING_BUFFER_LENGTH ];
|
|
|
|
Dns_Ip6AddressToString_A(
|
|
buffer,
|
|
(PIP6_ADDRESS) &psin->sin6_addr );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\tsin6_port = %04x\r\n"
|
|
"%s\tsin6_flowinfo = %08x\r\n"
|
|
"%s\tsin6_addr = %s\r\n"
|
|
"%s\tsin6_scope_id = %08x\r\n",
|
|
pindent, psin->sin6_port,
|
|
pindent, psin->sin6_flowinfo,
|
|
pindent, buffer,
|
|
pindent, psin->sin6_scope_id
|
|
);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
|
|
// print unknown in WORDs
|
|
// limit print as this is probably a busted sockaddr due to bug
|
|
{
|
|
DnsPrint_RawBinary(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tdata: ",
|
|
pindent, // line header
|
|
pSockaddr->sa_data,
|
|
iSockaddrLength < 100
|
|
? iSockaddrLength
|
|
: 100,
|
|
sizeof(WORD)
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_AddrInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN DWORD Indent,
|
|
IN PADDRINFO pAddrInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print ADDRINFO structure.
|
|
|
|
--*/
|
|
{
|
|
PSTR pindent = INDENT_STRING( Indent );
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "AddrInfo:";
|
|
}
|
|
|
|
if ( !pAddrInfo )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s NULL AddrInfo.\r\n",
|
|
pindent,
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\n"
|
|
"%s\tPtr = %p\r\n"
|
|
"%s\tNext Ptr = %p\r\n"
|
|
"%s\tFlags = %08x\r\n"
|
|
"%s\tFamily = %d\r\n"
|
|
"%s\tSockType = %d\r\n"
|
|
"%s\tProtocol = %d\r\n"
|
|
"%s\tAddrLength = %d\r\n"
|
|
"%s\tName = %s\r\n",
|
|
pindent, pszHeader,
|
|
pindent, pAddrInfo,
|
|
pindent, pAddrInfo->ai_next,
|
|
pindent, pAddrInfo->ai_flags,
|
|
pindent, pAddrInfo->ai_family,
|
|
pindent, pAddrInfo->ai_socktype,
|
|
pindent, pAddrInfo->ai_protocol,
|
|
pindent, pAddrInfo->ai_addrlen,
|
|
pindent, pAddrInfo->ai_canonname
|
|
);
|
|
|
|
DnsPrint_Sockaddr(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
Indent + 1,
|
|
pAddrInfo->ai_addr,
|
|
pAddrInfo->ai_addrlen );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_AddrInfoList(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN DWORD Indent,
|
|
IN PADDRINFO pAddrInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print ADDRINFO structure.
|
|
|
|
--*/
|
|
{
|
|
PADDRINFO paddr = pAddrInfo;
|
|
PSTR pindent = INDENT_STRING( Indent );
|
|
|
|
//
|
|
// list header
|
|
//
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "AddrInfo List:";
|
|
}
|
|
|
|
if ( !paddr )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s NULL AddrInfo List.\r\n",
|
|
pindent,
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\n",
|
|
pindent, pszHeader
|
|
);
|
|
|
|
//
|
|
// print each ADDRINFO in list
|
|
//
|
|
|
|
while ( paddr )
|
|
{
|
|
DnsPrint_AddrInfo(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
Indent,
|
|
paddr );
|
|
|
|
paddr = paddr->ai_next;
|
|
}
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"End of AddrInfo list\n\n"
|
|
);
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_SocketAddress(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN DWORD Indent,
|
|
IN PSOCKET_ADDRESS pSocketAddress
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print SOCKET_ADDRESS structure.
|
|
|
|
--*/
|
|
{
|
|
PSTR pindent = INDENT_STRING( Indent );
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "SocketAddress:";
|
|
}
|
|
|
|
if ( !pSocketAddress )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s NULL SocketAddress.\r\n",
|
|
pindent,
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\n"
|
|
"%s\tpSockaddr = %p\r\n"
|
|
"%s\tiSockaddrLength = %d\r\n",
|
|
pindent, pszHeader,
|
|
pindent, pSocketAddress->lpSockaddr,
|
|
pindent, pSocketAddress->iSockaddrLength );
|
|
|
|
DnsPrint_Sockaddr(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
Indent,
|
|
pSocketAddress->lpSockaddr,
|
|
pSocketAddress->iSockaddrLength );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_CsAddr(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN DWORD Indent,
|
|
IN PCSADDR_INFO pCsAddr
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print CSADDR_INFO structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pParam - ptr to print context
|
|
|
|
pszHeader - header
|
|
|
|
Indent - indent count, for formatting CSADDR inside larger struct
|
|
|
|
pCsAddr - ptr to CSADDRINFO to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
PSTR pindent = INDENT_STRING( Indent );
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "CSAddrInfo:";
|
|
}
|
|
|
|
if ( !pCsAddr )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s \tNULL CSADDR_INFO ptr.\r\n",
|
|
pindent, pszHeader
|
|
);
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s%s\r\n"
|
|
"%s\tPtr = %p\n"
|
|
"%s\tSocketType = %d\n"
|
|
"%s\tProtocol = %d\n",
|
|
pindent, pszHeader,
|
|
pindent, pCsAddr,
|
|
pindent, pCsAddr->iSocketType,
|
|
pindent, pCsAddr->iProtocol
|
|
);
|
|
|
|
DnsPrint_SocketAddress(
|
|
PrintRoutine,
|
|
pContext,
|
|
"LocalAddress:",
|
|
Indent,
|
|
& pCsAddr->LocalAddr
|
|
);
|
|
|
|
DnsPrint_SocketAddress(
|
|
PrintRoutine,
|
|
pContext,
|
|
"RemoteAddress:",
|
|
Indent,
|
|
& pCsAddr->RemoteAddr
|
|
);
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_AfProtocolsArray(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PAFPROTOCOLS pProtocolArray,
|
|
IN DWORD ProtocolCount
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print AFPROTOCOLS array.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pProtocolArray - protocols array
|
|
|
|
ProtocolCount - array count
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "AFPROTOCOLS Array:";
|
|
}
|
|
|
|
// print
|
|
// - array + count
|
|
// - each protocol element
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tProtocol Array = %p\r\n"
|
|
"\tProtocol Count = %d\r\n",
|
|
pszHeader,
|
|
pProtocolArray,
|
|
ProtocolCount );
|
|
|
|
if ( pProtocolArray )
|
|
{
|
|
for ( i=0; i<ProtocolCount; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\t\tfamily = %d; proto = %d\r\n",
|
|
pProtocolArray[i].iAddressFamily,
|
|
pProtocolArray[i].iProtocol );
|
|
}
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_WsaQuerySet(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN LPWSAQUERYSET pQuerySet,
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print WSAQUERYSET structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pQuerySet - ptr to WSAQUERYSET to print
|
|
|
|
fUnicode - TRUE if WSAQUERYSET is wide (WSAQUERYSETW)
|
|
FALSE if ANSI
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
CHAR serviceGuidBuffer[ GUID_STRING_BUFFER_LENGTH ];
|
|
CHAR nameSpaceGuidBuffer[ GUID_STRING_BUFFER_LENGTH ];
|
|
DWORD i;
|
|
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "WSAQuerySet:";
|
|
}
|
|
|
|
if ( !pQuerySet )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL QuerySet ptr\r\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
// convert GUIDs to strings
|
|
|
|
DnsStringPrint_Guid(
|
|
serviceGuidBuffer,
|
|
pQuerySet->lpServiceClassId
|
|
);
|
|
DnsStringPrint_Guid(
|
|
nameSpaceGuidBuffer,
|
|
pQuerySet->lpNSProviderId
|
|
);
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tSize = %d\r\n"
|
|
"\tServiceInstanceName = %S%s\r\n"
|
|
"\tService GUID = (%p) %s\r\n"
|
|
"\tWSA version = %p %x %d\r\n"
|
|
"\tComment = %S%s\r\n"
|
|
"\tName Space = %d %s\r\n"
|
|
"\tName Space GUID = (%p) %s\r\n"
|
|
"\tContext = %S%s\r\n"
|
|
"\tNumberOfProtocols = %d\r\n"
|
|
"\tProtocol Array = %p\r\n"
|
|
"\tQueryString = %S%s\r\n"
|
|
"\tCS Addr Count = %d\r\n"
|
|
"\tCS Addr Array = %p\r\n"
|
|
"\tOutput Flags = %08x\r\n"
|
|
"\tpBlob = %p\r\n",
|
|
|
|
pszHeader,
|
|
pQuerySet->dwSize,
|
|
DNSSTRING_WIDE( fUnicode, pQuerySet->lpszServiceInstanceName ),
|
|
DNSSTRING_ANSI( fUnicode, pQuerySet->lpszServiceInstanceName ),
|
|
pQuerySet->lpServiceClassId,
|
|
serviceGuidBuffer,
|
|
pQuerySet->lpVersion,
|
|
( pQuerySet->lpVersion ) ? pQuerySet->lpVersion->dwVersion : 0,
|
|
( pQuerySet->lpVersion ) ? pQuerySet->lpVersion->ecHow : 0,
|
|
|
|
DNSSTRING_WIDE( fUnicode, pQuerySet->lpszComment ),
|
|
DNSSTRING_ANSI( fUnicode, pQuerySet->lpszComment ),
|
|
pQuerySet->dwNameSpace,
|
|
Dns_GetRnrNameSpaceIdString( pQuerySet->dwNameSpace ),
|
|
pQuerySet->lpNSProviderId,
|
|
nameSpaceGuidBuffer,
|
|
DNSSTRING_WIDE( fUnicode, pQuerySet->lpszContext ),
|
|
DNSSTRING_ANSI( fUnicode, pQuerySet->lpszContext ),
|
|
|
|
pQuerySet->dwNumberOfProtocols,
|
|
pQuerySet->lpafpProtocols,
|
|
DNSSTRING_WIDE( fUnicode, pQuerySet->lpszQueryString ),
|
|
DNSSTRING_ANSI( fUnicode, pQuerySet->lpszQueryString ),
|
|
|
|
pQuerySet->dwNumberOfCsAddrs,
|
|
pQuerySet->lpcsaBuffer,
|
|
pQuerySet->dwOutputFlags,
|
|
pQuerySet->lpBlob
|
|
);
|
|
|
|
// print address-family\protocols array
|
|
|
|
if ( pQuerySet->lpafpProtocols )
|
|
{
|
|
DnsPrint_AfProtocolsArray(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tAFPROTOCOLS Array:",
|
|
pQuerySet->lpafpProtocols,
|
|
pQuerySet->dwNumberOfProtocols );
|
|
}
|
|
|
|
// print CSADDR_INFO array
|
|
|
|
if ( pQuerySet->dwNumberOfCsAddrs &&
|
|
pQuerySet->lpcsaBuffer )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"--- CS_ADDR array:\r\n" );
|
|
|
|
for ( i=0; i<pQuerySet->dwNumberOfCsAddrs; i++ )
|
|
{
|
|
DnsPrint_CsAddr(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
1, // indent one level
|
|
& pQuerySet->lpcsaBuffer[i] );
|
|
}
|
|
}
|
|
|
|
// print blob (the hostent)
|
|
|
|
//
|
|
// DCR_FIX0: need some sort of test for blob type?
|
|
// - most blobs are hostent, but some are servent
|
|
//
|
|
|
|
if ( pQuerySet->lpBlob )
|
|
{
|
|
GUID ianaGuid = SVCID_INET_SERVICEBYNAME;
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"--- BLOB:\n"
|
|
"\tcbSize = %d\r\n"
|
|
"\tpBlobData = %p\r\n",
|
|
pQuerySet->lpBlob->cbSize,
|
|
pQuerySet->lpBlob->pBlobData
|
|
);
|
|
|
|
// note: can't print blob as hostent
|
|
// 1) may not be hostent
|
|
// 2) is passed with offsets rather than pointers
|
|
|
|
DnsPrint_RawBinary(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
"\t\t",
|
|
pQuerySet->lpBlob->pBlobData,
|
|
pQuerySet->lpBlob->cbSize,
|
|
sizeof(DWORD)
|
|
);
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_WsaNsClassInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PWSANSCLASSINFO pInfo,
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print WSACLASSINFO structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pInfo - ptr to WSACLASSINFO to print
|
|
|
|
fUnicode - TRUE if WSACLASSINFO is wide (WSACLASSINFOW)
|
|
FALSE if ANSI
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "WSANsClassInfo:";
|
|
}
|
|
|
|
if ( !pInfo )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL NsClassInfo ptr\r\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tPtr = %d\r\n"
|
|
"\tName = %S%s\r\n"
|
|
"\tName Space = %d\r\n"
|
|
"\tValue Type = %d\r\n"
|
|
"\tValue Size = %d\r\n"
|
|
"\tpValue = %p\r\n",
|
|
pszHeader,
|
|
pInfo,
|
|
DNSSTRING_WIDE( fUnicode, pInfo->lpszName ),
|
|
DNSSTRING_ANSI( fUnicode, pInfo->lpszName ),
|
|
pInfo->dwNameSpace,
|
|
pInfo->dwValueType,
|
|
pInfo->dwValueSize,
|
|
pInfo->lpValue
|
|
);
|
|
|
|
if ( pInfo->lpValue )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"--- Value:\r\n"
|
|
);
|
|
|
|
DnsPrint_RawBinary(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
"\t\t",
|
|
pInfo->lpValue,
|
|
pInfo->dwValueSize,
|
|
sizeof(BYTE) // print in bytes
|
|
);
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_WsaServiceClassInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN LPWSASERVICECLASSINFO pInfo,
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print WSASERVICECLASSINFO structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pInfo - ptr to WSASERVICECLASSINFO to print
|
|
|
|
fUnicode - TRUE if WSASERVICECLASSINFO is wide (WSASERVICECLASSINFOW)
|
|
FALSE if ANSI
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
CHAR serviceClassGuidBuffer[ GUID_STRING_BUFFER_LENGTH ];
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "WSAServiceClassInfo:";
|
|
}
|
|
|
|
if ( !pInfo )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s NULL ServiceClassInfo ptr\r\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
// convert GUID to strings
|
|
|
|
DnsStringPrint_Guid(
|
|
serviceClassGuidBuffer,
|
|
pInfo->lpServiceClassId
|
|
);
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tPtr = %p\r\n"
|
|
"\tClass GUID = (%p) %s\r\n"
|
|
"\tClassName = %S%s\r\n"
|
|
"\tClass Info Count = %d\r\n"
|
|
"\tClass Info Array = %p\r\n",
|
|
pszHeader,
|
|
pInfo,
|
|
serviceClassGuidBuffer,
|
|
DNSSTRING_WIDE( fUnicode, pInfo->lpszServiceClassName ),
|
|
DNSSTRING_ANSI( fUnicode, pInfo->lpszServiceClassName ),
|
|
pInfo->dwCount,
|
|
pInfo->lpClassInfos
|
|
);
|
|
|
|
if ( pInfo->lpClassInfos )
|
|
{
|
|
DWORD i;
|
|
|
|
for ( i=0; i<pInfo->dwCount; i++ )
|
|
{
|
|
DnsPrint_WsaNsClassInfo(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL, // default header
|
|
& pInfo->lpClassInfos[i],
|
|
fUnicode
|
|
);
|
|
}
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_Hostent(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PHOSTENT pHostent,
|
|
IN BOOL fUnicode
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print hostent structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pHostent - ptr to hostent
|
|
|
|
fUnicode - TRUE if hostent is unicode
|
|
FALSE if ANSI
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Hostent:";
|
|
}
|
|
|
|
if ( !pHostent )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Hostent ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\th_name = %p %S%s\n"
|
|
"\th_aliases = %p\n"
|
|
"\th_addrtype = %d\n"
|
|
"\th_length = %d\n"
|
|
"\th_addrlist = %p\n",
|
|
pszHeader,
|
|
pHostent->h_name,
|
|
DNSSTRING_WIDE( fUnicode, pHostent->h_name ),
|
|
DNSSTRING_ANSI( fUnicode, pHostent->h_name ),
|
|
pHostent->h_aliases,
|
|
pHostent->h_addrtype,
|
|
pHostent->h_length,
|
|
pHostent->h_addr_list
|
|
);
|
|
|
|
// print the aliases
|
|
|
|
if ( pHostent->h_aliases )
|
|
{
|
|
PSTR * paliasArray = pHostent->h_aliases;
|
|
PSTR palias;
|
|
|
|
while ( palias = *paliasArray++ )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tAlias = (%p) %S%s\n",
|
|
palias,
|
|
DNSSTRING_WIDE( fUnicode, palias ),
|
|
DNSSTRING_ANSI( fUnicode, palias ) );
|
|
}
|
|
}
|
|
|
|
// print the addresses
|
|
|
|
if ( pHostent->h_addr_list )
|
|
{
|
|
PCHAR * ppaddr = pHostent->h_addr_list;
|
|
PCHAR pip;
|
|
INT i = 0;
|
|
INT family = pHostent->h_addrtype;
|
|
INT addrLength = pHostent->h_length;
|
|
CHAR stringBuf[ IP6_ADDRESS_STRING_BUFFER_LENGTH ];
|
|
|
|
while ( pip = ppaddr[i] )
|
|
{
|
|
DWORD bufLength = IP6_ADDRESS_STRING_BUFFER_LENGTH;
|
|
|
|
Dns_AddressToString_A(
|
|
stringBuf,
|
|
& bufLength,
|
|
pip,
|
|
addrLength,
|
|
family );
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tAddr[%d] = %s \t(ptr=%p)\n",
|
|
i,
|
|
stringBuf,
|
|
pip );
|
|
i++;
|
|
}
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Query print routines
|
|
//
|
|
|
|
VOID
|
|
DnsPrint_QueryBlob(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PQUERY_BLOB pBlob
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print query blob.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pContext - print context
|
|
|
|
pszHeader - header
|
|
|
|
pBlob - query info
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Query Blob:";
|
|
}
|
|
|
|
if ( !pBlob )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Query Blob ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tname orig %S\n"
|
|
"\tname orig wire %s\n"
|
|
"\tname wire %s\n"
|
|
"\ttype %d\n"
|
|
"\tflags %08x\n"
|
|
|
|
"\tname length %d\n"
|
|
"\tname attributes %08x\n"
|
|
"\tquery count %d\n"
|
|
"\tname flags %08x\n"
|
|
"\tfappendedName %d\n"
|
|
|
|
"\tstatus %d\n"
|
|
"\trcode %d\n"
|
|
"\tnetfail status %d\n"
|
|
"\tcache negative %d\n"
|
|
"\tno ip local %d\n"
|
|
"\trecords %p\n"
|
|
"\tlocal records %p\n"
|
|
|
|
"\tnetwork info %p\n"
|
|
"\tserver IPs %p\n"
|
|
"\tpmsg %p\n"
|
|
"\tevent %p\n",
|
|
|
|
pBlob->pNameOrig,
|
|
pBlob->pNameOrigWire,
|
|
pBlob->pNameWire,
|
|
pBlob->wType,
|
|
pBlob->Flags,
|
|
|
|
pBlob->NameLength,
|
|
pBlob->NameAttributes,
|
|
pBlob->QueryCount,
|
|
pBlob->NameFlags,
|
|
pBlob->fAppendedName,
|
|
|
|
pBlob->Status,
|
|
pBlob->Rcode,
|
|
pBlob->NetFailureStatus,
|
|
pBlob->fCacheNegative,
|
|
pBlob->fNoIpLocal,
|
|
pBlob->pRecords,
|
|
pBlob->pLocalRecords,
|
|
|
|
pBlob->pNetworkInfo,
|
|
pBlob->pDnsServers,
|
|
pBlob->pRecvMsg,
|
|
pBlob->hEvent
|
|
);
|
|
|
|
// DCR_FIX0: cleanup when results in use
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Records:\n",
|
|
pBlob->pRecords );
|
|
|
|
// DCR_FIX0: use results when ready
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_QueryResults(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_RESULTS pResults
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print query results.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pContext - print context
|
|
|
|
pszHeader - header
|
|
|
|
pResults - results info
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Results:";
|
|
}
|
|
|
|
if ( !pResults )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Results ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tstatus %d\n"
|
|
"\trcode %d\n"
|
|
"\tserver %s\n"
|
|
"\tpanswer %p\n"
|
|
"\tpalias %p\n"
|
|
"\tpauthority %p\n"
|
|
"\tpadditional %p\n"
|
|
"\tpsig %p\n"
|
|
"\tpmsg %p\n",
|
|
pResults->Status,
|
|
pResults->Rcode,
|
|
IP_STRING( pResults->ServerIp ),
|
|
pResults->pAnswerRecords,
|
|
pResults->pAliasRecords,
|
|
pResults->pAuthorityRecords,
|
|
pResults->pAdditionalRecords,
|
|
pResults->pSigRecords,
|
|
pResults->pMessage
|
|
);
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tAnswer records:\n",
|
|
pResults->pAnswerRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tAlias records:\n",
|
|
pResults->pAliasRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tAuthority records:\n",
|
|
pResults->pAuthorityRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tAdditional records:\n",
|
|
pResults->pAdditionalRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"\tSignature records:\n",
|
|
pResults->pSigRecords );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_ParsedMessage(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_PARSED_MESSAGE pParsed
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print parsed message.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pContext - print context
|
|
|
|
pszHeader - header
|
|
|
|
pResults - query info
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Parsed Message:";
|
|
}
|
|
|
|
if ( !pParsed )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Parsed Message ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tstatus %d (%08x)\n"
|
|
"\tchar set %d\n",
|
|
pParsed->Status, pParsed->Status,
|
|
pParsed->CharSet
|
|
);
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"\tquestion:\n"
|
|
"\t\tname %S%s\n"
|
|
"\t\ttype %d\n"
|
|
"\t\tclass %d\n",
|
|
PRINT_STRING_WIDE_CHARSET( pParsed->pQuestionName, pParsed->CharSet ),
|
|
PRINT_STRING_ANSI_CHARSET( pParsed->pQuestionName, pParsed->CharSet ),
|
|
pParsed->QuestionType,
|
|
pParsed->QuestionClass
|
|
);
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Answer records:\n",
|
|
pParsed->pAnswerRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Alias records:\n",
|
|
pParsed->pAliasRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Authority records:\n",
|
|
pParsed->pAuthorityRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Additional records:\n",
|
|
pParsed->pAdditionalRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Signature records:\n",
|
|
pParsed->pSigRecords );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_QueryInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PDNS_QUERY_INFO pQueryInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print query info
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pContext - print context
|
|
|
|
pszHeader - header
|
|
|
|
pQueryInfo - query info
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Query Info:";
|
|
}
|
|
|
|
if ( !pQueryInfo )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Query Info ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\n"
|
|
"\tpointer %p\n"
|
|
"\tstatus %d (%08x)\n"
|
|
"\tchar set %d\n"
|
|
"\tname %S%s\n"
|
|
"\tname resv. %s\n"
|
|
"\ttype %d\n"
|
|
"\trcode %d\n"
|
|
"\tflags %08x\n"
|
|
|
|
"\tpanswer %p\n"
|
|
"\tpalias %p\n"
|
|
"\tpauthority %p\n"
|
|
"\tpadditional %p\n"
|
|
//"\tpsig %p\n"
|
|
|
|
"\tevent %p\n"
|
|
"\tpDnsServers %p\n"
|
|
"\tpmsg %p\n",
|
|
|
|
pszHeader,
|
|
pQueryInfo,
|
|
pQueryInfo->Status, pQueryInfo->Status,
|
|
pQueryInfo->CharSet,
|
|
PRINT_STRING_WIDE_CHARSET( pQueryInfo->pName, pQueryInfo->CharSet ),
|
|
PRINT_STRING_ANSI_CHARSET( pQueryInfo->pName, pQueryInfo->CharSet ),
|
|
pQueryInfo->pReservedName,
|
|
pQueryInfo->Type,
|
|
pQueryInfo->Rcode,
|
|
pQueryInfo->Flags,
|
|
|
|
pQueryInfo->pAnswerRecords,
|
|
pQueryInfo->pAliasRecords,
|
|
pQueryInfo->pAuthorityRecords,
|
|
pQueryInfo->pAdditionalRecords,
|
|
//pQueryInfo->pSigRecords,
|
|
|
|
pQueryInfo->hEvent,
|
|
pQueryInfo->pDnsServers,
|
|
pQueryInfo->pMessage
|
|
);
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Answer records:\n",
|
|
pQueryInfo->pAnswerRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Alias records:\n",
|
|
pQueryInfo->pAliasRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Authority records:\n",
|
|
pQueryInfo->pAuthorityRecords );
|
|
|
|
DnsPrint_RecordSet(
|
|
PrintRoutine,
|
|
pContext,
|
|
"Additional records:\n",
|
|
pQueryInfo->pAdditionalRecords );
|
|
|
|
//DnsPrint_RecordSet(
|
|
// "Signature records:\n",
|
|
// pQueryInfo->pSigRecords );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_EnvarInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PENVAR_DWORD_INFO pEnvar
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print envar data
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pContext - print context
|
|
|
|
pszHeader -- header to print with
|
|
|
|
pEnvar -- ptr to envar info
|
|
|
|
Return Value:
|
|
|
|
ERROR_SUCCESS if successful.
|
|
ErrorCode on failure.
|
|
|
|
--*/
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\n"
|
|
"\tId = %d\n"
|
|
"\tValue = %p\n"
|
|
"\tfFound = %d\n",
|
|
pszHeader ? pszHeader : "Envar Info:",
|
|
pEnvar->Id,
|
|
pEnvar->Value,
|
|
pEnvar->fFound
|
|
);
|
|
}
|
|
|
|
|
|
|
|
//
|
|
// Network info print routines.
|
|
//
|
|
|
|
VOID
|
|
DnsPrint_NetworkInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pPrintContext,
|
|
IN LPSTR pszHeader,
|
|
IN PDNS_NETINFO pNetworkInfo
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prints and validates network info structure.
|
|
Should also touch all the memory and AV when bogus.
|
|
|
|
Arguments:
|
|
|
|
pNetworkInfo -- network info to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "NetworkInfo:";
|
|
}
|
|
if ( !pNetworkInfo )
|
|
{
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s NULL NetworkInfo.\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s\n"
|
|
"\tpointer = %p\n"
|
|
"\tpszHostName = %s\n"
|
|
"\tpszDomainName = %s\n"
|
|
"\tpSearchList = %p\n"
|
|
"\tTimeStamp = %d\n"
|
|
"\tInfoFlags = %08x\n"
|
|
"\tReturnFlags = %08x\n"
|
|
"\tAdapterCount = %d\n"
|
|
"\tAdapterArraySize = %d\n",
|
|
pszHeader,
|
|
pNetworkInfo,
|
|
pNetworkInfo->pszHostName,
|
|
pNetworkInfo->pszDomainName,
|
|
pNetworkInfo->pSearchList,
|
|
pNetworkInfo->TimeStamp,
|
|
pNetworkInfo->InfoFlags,
|
|
pNetworkInfo->ReturnFlags,
|
|
pNetworkInfo->AdapterCount,
|
|
pNetworkInfo->MaxAdapterCount );
|
|
|
|
// print search list
|
|
|
|
DnsPrint_SearchList(
|
|
PrintRoutine,
|
|
pPrintContext,
|
|
"Search List: ",
|
|
pNetworkInfo->pSearchList );
|
|
|
|
// print server lists
|
|
|
|
for ( i=0; i < pNetworkInfo->AdapterCount; i++ )
|
|
{
|
|
CHAR header[60];
|
|
|
|
sprintf( header, "AdapterInfo[%d]:", i );
|
|
|
|
DnsPrint_AdapterInfo(
|
|
PrintRoutine,
|
|
pPrintContext,
|
|
header,
|
|
pNetworkInfo->AdapterArray[i] );
|
|
}
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"\n" );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_AdapterInfo(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pPrintContext,
|
|
IN LPSTR pszHeader,
|
|
IN PDNS_ADAPTER pAdapter
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prints and validates DNS adapter info.
|
|
Should also touch all the memory and AV when bogus.
|
|
|
|
Arguments:
|
|
|
|
pAdapter -- DNS adapter to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Adapter Info:";
|
|
}
|
|
if ( !pAdapter )
|
|
{
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s NULL Adapter info.\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s\n"
|
|
"\tpointer = %p\n"
|
|
"\tDomain = %s\n"
|
|
"\tGuid Name = %s\n"
|
|
"\tInterfaceIndex = %d\n"
|
|
"\tInfoFlags = %08x\n"
|
|
"\tStatus = %d\n"
|
|
"\tRunFlags = %08x\n"
|
|
"\tServerIndex = %d\n"
|
|
"\tServerCount = %d\n"
|
|
"\tServerArraySize = %d\n",
|
|
pszHeader,
|
|
pAdapter,
|
|
pAdapter->pszAdapterDomain,
|
|
pAdapter->pszAdapterGuidName,
|
|
pAdapter->InterfaceIndex,
|
|
pAdapter->InfoFlags,
|
|
pAdapter->Status,
|
|
pAdapter->RunFlags,
|
|
pAdapter->ServerIndex,
|
|
pAdapter->ServerCount,
|
|
pAdapter->MaxServerCount );
|
|
|
|
// DNS server info
|
|
|
|
for ( i=0; i < pAdapter->ServerCount; i++ )
|
|
{
|
|
// DCR: IP6 DNS server address
|
|
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"\tDNS Server [%d]:\n"
|
|
"\t\tIpAddr = %s\n"
|
|
"\t\tPriority = %d\n"
|
|
"\t\tStatus = %d (%08x)\n",
|
|
i,
|
|
IP_STRING( pAdapter->ServerArray[i].IpAddress ),
|
|
pAdapter->ServerArray[i].Priority,
|
|
pAdapter->ServerArray[i].Status,
|
|
pAdapter->ServerArray[i].Status
|
|
);
|
|
}
|
|
|
|
// IP address info
|
|
|
|
DnsPrint_IpArray(
|
|
PrintRoutine,
|
|
pPrintContext,
|
|
"IP Addrs",
|
|
"IP",
|
|
pAdapter->pAdapterIPAddresses );
|
|
|
|
DnsPrint_IpArray(
|
|
PrintRoutine,
|
|
pPrintContext,
|
|
"IP Addr Subnets",
|
|
"Mask",
|
|
pAdapter->pAdapterIPSubnetMasks );
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_SearchList(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pPrintContext,
|
|
IN LPSTR pszHeader,
|
|
IN PSEARCH_LIST pSearchList
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Prints and validates DNS search list.
|
|
Should also touch all the memory and AV when bogus.
|
|
|
|
Arguments:
|
|
|
|
pSearchList -- search list to print
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "DNS Search List:";
|
|
}
|
|
|
|
if ( ! pSearchList )
|
|
{
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s NULL search list.\n",
|
|
pszHeader );
|
|
return;
|
|
}
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"%s\n"
|
|
"\tpointer = %p\n"
|
|
"\tpszDomain = %s\n"
|
|
"\tcNameCount = %d\n"
|
|
"\tcTotalListSize = %d\n"
|
|
"\tCurrentName = %d\n"
|
|
"\tSearchListNames:\n",
|
|
pszHeader,
|
|
pSearchList,
|
|
pSearchList->pszDomainOrZoneName,
|
|
pSearchList->NameCount,
|
|
pSearchList->MaxNameCount,
|
|
pSearchList->CurrentNameIndex
|
|
);
|
|
|
|
for ( i=0; i < pSearchList->NameCount; i++ )
|
|
{
|
|
PrintRoutine(
|
|
pPrintContext,
|
|
"\t\t%s (Flags: %08x)\n",
|
|
pSearchList->SearchNameArray[i].pszName,
|
|
pSearchList->SearchNameArray[i].Flags );
|
|
}
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
DnsPrint_HostentBlob(
|
|
IN PRINT_ROUTINE PrintRoutine,
|
|
IN OUT PPRINT_CONTEXT pContext,
|
|
IN PSTR pszHeader,
|
|
IN PHOSTENT_BLOB pBlob
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Print hostent structure.
|
|
|
|
Arguments:
|
|
|
|
PrintRoutine - routine to print with
|
|
|
|
pszHeader - header
|
|
|
|
pBlob - ptr to hostent blob
|
|
|
|
Return Value:
|
|
|
|
None.
|
|
|
|
--*/
|
|
{
|
|
DWORD i;
|
|
|
|
if ( !pszHeader )
|
|
{
|
|
pszHeader = "Hostent Blob:";
|
|
}
|
|
|
|
if ( !pBlob )
|
|
{
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s %s\r\n",
|
|
pszHeader,
|
|
"NULL Hostent blob ptr." );
|
|
return;
|
|
}
|
|
|
|
// print the struct
|
|
|
|
DnsPrint_Lock();
|
|
|
|
PrintRoutine(
|
|
pContext,
|
|
"%s\r\n"
|
|
"\tPtr = %p\n"
|
|
"\tpHostent = %p\n"
|
|
"\tfAllocatedBlob = %d\n"
|
|
"\tfAllocatedBuf = %d\n"
|
|
"\tpBuffer = %p\n"
|
|
"\tBufferLength = %d\n"
|
|
"\tAvailLength = %d\n"
|
|
"\tpAvailBuffer = %p\n"
|
|
"\tpCurrent = %p\n"
|
|
"\tBytesLeft = %d\n"
|
|
"\tMaxAliasCount = %d\n"
|
|
"\tAliasCount = %d\n"
|
|
"\tMaxAddrCount = %d\n"
|
|
"\tAddrCount = %d\n"
|
|
"\tfWroteName = %d\n"
|
|
"\tfUnicode = %d\n"
|
|
"\tCharSet = %d\n",
|
|
pszHeader,
|
|
pBlob,
|
|
pBlob->pHostent,
|
|
pBlob->fAllocatedBlob,
|
|
pBlob->fAllocatedBuf,
|
|
pBlob->pBuffer,
|
|
pBlob->BufferLength,
|
|
pBlob->AvailLength,
|
|
pBlob->pAvailBuffer,
|
|
pBlob->pCurrent,
|
|
pBlob->BytesLeft,
|
|
pBlob->MaxAliasCount,
|
|
pBlob->AliasCount,
|
|
pBlob->MaxAddrCount,
|
|
pBlob->AddrCount,
|
|
pBlob->fWroteName,
|
|
pBlob->fUnicode,
|
|
pBlob->CharSet
|
|
);
|
|
|
|
// print the hostent
|
|
|
|
if ( pBlob->pHostent )
|
|
{
|
|
DnsPrint_Hostent(
|
|
PrintRoutine,
|
|
pContext,
|
|
NULL,
|
|
pBlob->pHostent,
|
|
pBlob->fUnicode
|
|
);
|
|
}
|
|
|
|
DnsPrint_Unlock();
|
|
}
|
|
|
|
//
|
|
// End of print.c
|
|
//
|
|
|