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
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
|
|
|