Leaked source code of windows server 2003
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.
 
 
 
 
 
 

409 lines
12 KiB

/*++
Copyright (c) 1998, Microsoft Corporation
Module Name:
dnsio.c
Abstract:
This module contains code for the DNS allocator's network I/O completion
routines.
Author:
Abolade Gbadegesin (aboladeg) 9-Mar-1998
Revision History:
Raghu Gatta (rgatta) 28-Mar-2002
Note: bunch of reposts - due to accumulated patching - need to cleanup.
--*/
#include "precomp.h"
#pragma hdrstop
#include "dnsmsg.h"
VOID
DnsReadCompletionRoutine(
ULONG ErrorCode,
ULONG BytesTransferred,
PNH_BUFFER Bufferp
)
/*++
Routine Description:
This routine is invoked upon completion of a read operation
on a datagram socket bound to the DNS server UDP port.
The message read is validated and processed; the processing may involve
creating a query-record and forwarding the query to a server, or
matching a response to an existing query-record and forwarding the
response to the appropriate client.
Arguments:
ErrorCode - Win32 status code for the I/O operation
BytesTransferred - number of bytes in 'Bufferp'
Bufferp - holds data read from the datagram socket
Return Value:
none.
Environment:
Runs in the context of an RTUTILS.DLL worker-thread which has just
dequeued an I/O completion packet from the common I/O completion port
with which our datagram sockets are associated.
A reference to the component will have been made on our behalf
by 'NhReadDatagramSocket'.
--*/
{
ULONG Error;
PDNS_HEADER Headerp;
PDNS_INTERFACE Interfacep;
PROFILE("DnsReadCompletionRoutine");
do {
//
// There are two cases where we don't process the message;
// (a) the I/O operation failed
// (b) the interface is no longer active
// In case (a), we repost the buffer; in case (b), we do not.
//
Interfacep = (PDNS_INTERFACE)Bufferp->Context;
//
// First look for an error code
//
if (ErrorCode) {
NhTrace(
TRACE_FLAG_IO,
"DnsReadCompletionRoutine: error %d for read-context %x",
ErrorCode,
Bufferp->Context
);
//
// See if the interface is still active
//
ACQUIRE_LOCK(Interfacep);
if (!DNS_INTERFACE_ACTIVE(Interfacep)) {
RELEASE_LOCK(Interfacep);
NhReleaseBuffer(Bufferp);
} else {
RELEASE_LOCK(Interfacep);
EnterCriticalSection(&DnsInterfaceLock);
if (!DNS_REFERENCE_INTERFACE(Interfacep)) {
LeaveCriticalSection(&DnsInterfaceLock);
NhReleaseBuffer(Bufferp);
} else {
LeaveCriticalSection(&DnsInterfaceLock);
//
// Repost the buffer for another read operation
//
do {
Error =
NhReadDatagramSocket(
&DnsComponentReference,
Bufferp->Socket,
Bufferp,
DnsReadCompletionRoutine,
Bufferp->Context,
Bufferp->Context2
);
//
// A connection-reset error indicates that our last
// *send* could not be delivered at its destination.
// We could hardly care less; so issue the read again,
// immediately.
//
} while (Error == WSAECONNRESET);
if (Error) {
ACQUIRE_LOCK(Interfacep);
DnsDeferReadInterface(Interfacep, Bufferp->Socket);
RELEASE_LOCK(Interfacep);
DNS_DEREFERENCE_INTERFACE(Interfacep);
NhWarningLog(
IP_DNS_PROXY_LOG_RECEIVE_FAILED,
Error,
"%I",
NhQueryAddressSocket(Bufferp->Socket)
);
NhReleaseBuffer(Bufferp);
}
}
}
break;
}
//
// Now see if the interface is operational
//
ACQUIRE_LOCK(Interfacep);
if (!DNS_INTERFACE_ACTIVE(Interfacep)) {
RELEASE_LOCK(Interfacep);
NhReleaseBuffer(Bufferp);
NhTrace(
TRACE_FLAG_IO,
"DnsReadCompletionRoutine: interface %d inactive",
Interfacep->Index
);
break;
}
RELEASE_LOCK(Interfacep);
//
// Ensure minimum DNS_HEADER size
//
if (BytesTransferred < sizeof(DNS_HEADER)) {
NhTrace(
TRACE_FLAG_DNS,
"DnsReadCompletionRoutine: message size %d too small",
BytesTransferred
);
NhWarningLog(
IP_DNS_PROXY_LOG_MESSAGE_TOO_SMALL,
0,
""
);
InterlockedIncrement(
reinterpret_cast<LPLONG>(&DnsStatistics.MessagesIgnored)
);
//
// Repost read
//
EnterCriticalSection(&DnsInterfaceLock);
if (!DNS_REFERENCE_INTERFACE(Interfacep)) {
LeaveCriticalSection(&DnsInterfaceLock);
NhReleaseBuffer(Bufferp);
} else {
LeaveCriticalSection(&DnsInterfaceLock);
do {
Error =
NhReadDatagramSocket(
&DnsComponentReference,
Bufferp->Socket,
Bufferp,
DnsReadCompletionRoutine,
Bufferp->Context,
Bufferp->Context2
);
//
// A connection-reset error indicates that our last
// *send* could not be delivered at its destination.
// We could hardly care less; so issue the read again,
// immediately.
//
} while (Error == WSAECONNRESET);
if (Error) {
ACQUIRE_LOCK(Interfacep);
DnsDeferReadInterface(Interfacep, Bufferp->Socket);
RELEASE_LOCK(Interfacep);
DNS_DEREFERENCE_INTERFACE(Interfacep);
NhWarningLog(
IP_DNS_PROXY_LOG_RECEIVE_FAILED,
Error,
"%I",
NhQueryAddressSocket(Bufferp->Socket)
);
NhReleaseBuffer(Bufferp);
}
}
break;
}
//
// Now look at the message
//
Headerp = (PDNS_HEADER)Bufferp->Buffer;
if (Headerp->IsResponse == DNS_MESSAGE_QUERY) {
DnsProcessQueryMessage(
Interfacep,
Bufferp
);
} else {
DnsProcessResponseMessage(
Interfacep,
Bufferp
);
}
} while(FALSE);
DNS_DEREFERENCE_INTERFACE(Interfacep);
DEREFERENCE_DNS();
} // DnsReadCompletionRoutine
VOID
DnsWriteCompletionRoutine(
ULONG ErrorCode,
ULONG BytesTransferred,
PNH_BUFFER Bufferp
)
/*++
Routine Description:
This routine is invoked upon completion of a write-operation
on a datagram socket bound to the DNS server UDP port.
The write-context for all writes is a 'DNS_QUERY'. Our handling
is dependent on whether the message written was a query or a response.
Upon completion of a query, we may need to do a resend if there was
an error. Upon completion of a response, we delete the query-record.
Arguments:
ErrorCode - Win32 status code for the I/O operation
BytesTransferred - number of bytes in 'Bufferp'
Bufferp - holds data read from the datagram socket
Return Value:
none.
Environment:
Runs in the context of an RTUTILS.DLL worker-thread which has just
dequeued an I/O completion packet from the common I/O completion port
with which our datagram sockets are associated.
A reference to the component will have been made on our behalf
by 'NhReadDatagramSocket'.
--*/
{
ULONG Error;
PDNS_HEADER Headerp;
PDNS_INTERFACE Interfacep;
USHORT QueryId;
PDNS_QUERY Queryp;
PULONG Server;
PROFILE("DnsWriteCompletionRoutine");
Interfacep = (PDNS_INTERFACE)Bufferp->Context;
QueryId = (USHORT)Bufferp->Context2;
Headerp = (PDNS_HEADER)Bufferp->Buffer;
ACQUIRE_LOCK(Interfacep);
//
// Obtain the query associated with the send.
//
Queryp = DnsMapResponseToQuery(Interfacep, QueryId);
if (Headerp->IsResponse == DNS_MESSAGE_RESPONSE) {
if (ErrorCode) {
//
// An error occurred sending the message to the client
//
NhTrace(
TRACE_FLAG_DNS,
"DnsWriteCompletionRoutine: error %d response %d interface %d",
ErrorCode,
Queryp ? Queryp->QueryId : -1,
Interfacep->Index
);
NhWarningLog(
IP_DNS_PROXY_LOG_RESPONSE_FAILED,
ErrorCode,
"%I",
NhQueryAddressSocket(Bufferp->Socket)
);
} else if (Queryp && Headerp->ResponseCode == DNS_RCODE_NOERROR) {
//
// We're done with this query since it succeeded; remove it.
//
NhTrace(
TRACE_FLAG_DNS,
"DnsWriteCompletionRoutine: removing query %d interface %d",
Queryp->QueryId,
Interfacep->Index
);
DnsDeleteQuery(Interfacep, Queryp);
}
} else {
if (!ErrorCode) {
//
// No errors, so just return.
//
NhTrace(
TRACE_FLAG_DNS,
"DnsWriteCompletionRoutine: sent query %d interface %d",
Queryp ? Queryp->QueryId : -1,
Interfacep->Index
);
} else {
//
// The query just went out and it failed.
//
NhTrace(
TRACE_FLAG_DNS,
"DnsWriteCompletionRoutine: error %d for query %d interface %d",
ErrorCode,
Queryp ? Queryp->QueryId : -1,
Interfacep->Index
);
NhWarningLog(
IP_DNS_PROXY_LOG_QUERY_FAILED,
ErrorCode,
"%I%I%I",
Queryp ? Queryp->SourceAddress : -1,
Bufferp->WriteAddress.sin_addr.s_addr,
NhQueryAddressSocket(Bufferp->Socket)
);
}
}
RELEASE_LOCK(Interfacep);
DNS_DEREFERENCE_INTERFACE(Interfacep);
NhReleaseBuffer(Bufferp);
DEREFERENCE_DNS();
} // DnsWriteCompletionRoutine