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.
480 lines
11 KiB
480 lines
11 KiB
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
|
|
Module Name:
|
|
|
|
net\routing\ip\wanarp2\rcv.c
|
|
|
|
Abstract:
|
|
|
|
|
|
|
|
Revision History:
|
|
|
|
|
|
--*/
|
|
|
|
|
|
#define __FILE_SIG__ RCV_SIG
|
|
|
|
#include "inc.h"
|
|
|
|
INT
|
|
WanNdisReceivePacket(
|
|
IN NDIS_HANDLE nhProtocolContext,
|
|
IN PNDIS_PACKET pnpPacket
|
|
)
|
|
{
|
|
PNDIS_BUFFER pnbBuffer;
|
|
PVOID pvFirstBuffer;
|
|
UINT uiFirstBufLen, uiTotalPacketLen;
|
|
INT iClientCount;
|
|
NDIS_STATUS nsStatus;
|
|
|
|
TraceEnter(RCV, "NdisReceivePacket");
|
|
|
|
NdisGetFirstBufferFromPacket(pnpPacket,
|
|
&pnbBuffer,
|
|
&pvFirstBuffer,
|
|
&uiFirstBufLen,
|
|
&uiTotalPacketLen);
|
|
if (pvFirstBuffer==NULL)
|
|
return 0;
|
|
|
|
|
|
//
|
|
// The first buffer better contain enough data
|
|
//
|
|
|
|
if (uiFirstBufLen < sizeof(ETH_HEADER) + sizeof(IP_HEADER))
|
|
return 0;
|
|
|
|
iClientCount = 0;
|
|
|
|
nsStatus = WanReceiveCommon(nhProtocolContext,
|
|
pnpPacket,
|
|
pvFirstBuffer,
|
|
sizeof(ETH_HEADER),
|
|
(PVOID)((ULONG_PTR)pvFirstBuffer + sizeof(ETH_HEADER)),
|
|
uiFirstBufLen - sizeof(ETH_HEADER),
|
|
uiTotalPacketLen - sizeof(ETH_HEADER),
|
|
pnbBuffer,
|
|
&iClientCount);
|
|
|
|
return iClientCount;
|
|
}
|
|
|
|
NDIS_STATUS
|
|
WanNdisReceive(
|
|
NDIS_HANDLE nhProtocolContext,
|
|
NDIS_HANDLE nhXferContext,
|
|
VOID UNALIGNED *pvHeader,
|
|
UINT uiHeaderLen,
|
|
VOID UNALIGNED *pvData,
|
|
UINT uiFirstBufferLen,
|
|
UINT uiTotalDataLen
|
|
)
|
|
{
|
|
TraceEnter(RCV, "NdisReceive");
|
|
|
|
return WanReceiveCommon(nhProtocolContext,
|
|
nhXferContext,
|
|
pvHeader,
|
|
uiHeaderLen,
|
|
pvData,
|
|
uiFirstBufferLen,
|
|
uiTotalDataLen,
|
|
NULL,
|
|
NULL);
|
|
}
|
|
|
|
NDIS_STATUS
|
|
WanReceiveCommon(
|
|
NDIS_HANDLE nhProtocolContext,
|
|
NDIS_HANDLE nhXferContext,
|
|
VOID UNALIGNED *pvHeader,
|
|
UINT uiHeaderLen,
|
|
VOID UNALIGNED *pvData,
|
|
UINT uiFirstBufferLen,
|
|
UINT uiTotalDataLen,
|
|
PMDL pMdl,
|
|
PINT piClientCount
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
The common receive handler for packet based or buffer based receives
|
|
|
|
Locks:
|
|
|
|
Called at DPC (usually).
|
|
Acquires the g_rlConnTable lock to get a pointer to the connection
|
|
entry. Then locks either the entry itself or the adapter.
|
|
|
|
Arguments:
|
|
|
|
nhProtocolContext
|
|
nhXferContext
|
|
pvHeader
|
|
uiHeaderLen
|
|
pvData
|
|
uiFirstBufferLen
|
|
uiTotalDataLen
|
|
pMdl
|
|
piClientCount
|
|
|
|
Return Value:
|
|
|
|
NDIS_STATUS_NOT_ACCEPTED
|
|
|
|
--*/
|
|
|
|
{
|
|
PCONN_ENTRY pConnEntry;
|
|
PADAPTER pAdapter;
|
|
PUMODE_INTERFACE pInterface;
|
|
ETH_HEADER UNALIGNED *pEthHeader;
|
|
KIRQL kiIrql;
|
|
WORD wType;
|
|
ULONG ulIndex;
|
|
BOOLEAN bNonUnicast;
|
|
|
|
#if DBG
|
|
|
|
IP_HEADER UNALIGNED *pIpHeader;
|
|
|
|
#endif
|
|
|
|
//
|
|
// Pick out connection index from the buffer
|
|
//
|
|
|
|
pEthHeader = (ETH_HEADER UNALIGNED *)pvHeader;
|
|
|
|
ulIndex = GetConnIndexFromAddr(pEthHeader->rgbyDestAddr);
|
|
|
|
RtAcquireSpinLock(&g_rlConnTableLock,
|
|
&kiIrql);
|
|
|
|
if(ulIndex >= g_ulConnTableSize)
|
|
{
|
|
Trace(RCV, ERROR,
|
|
("ReceiveCommon: Invalid index for conn entry %d\n",
|
|
ulIndex));
|
|
|
|
RtReleaseSpinLock(&g_rlConnTableLock,
|
|
kiIrql);
|
|
|
|
return NDIS_STATUS_NOT_ACCEPTED;
|
|
}
|
|
|
|
pConnEntry = GetConnEntryGivenIndex(ulIndex);
|
|
|
|
if(pConnEntry is NULL)
|
|
{
|
|
Trace(RCV, ERROR,
|
|
("ReceiveCommon: Couldnt find entry for conn %d\n",
|
|
ulIndex));
|
|
|
|
RtReleaseSpinLock(&g_rlConnTableLock,
|
|
kiIrql);
|
|
|
|
return NDIS_STATUS_NOT_ACCEPTED;
|
|
}
|
|
|
|
//
|
|
// Lock the connection entry or adapter
|
|
//
|
|
|
|
RtAcquireSpinLockAtDpcLevel(pConnEntry->prlLock);
|
|
|
|
RtReleaseSpinLockFromDpcLevel(&g_rlConnTableLock);
|
|
|
|
//
|
|
// We can get this only on a connected entry
|
|
//
|
|
|
|
RtAssert(pConnEntry->byState is CS_CONNECTED);
|
|
|
|
pAdapter = pConnEntry->pAdapter;
|
|
|
|
//
|
|
// A connected entry must have an adapter
|
|
//
|
|
|
|
RtAssert(pAdapter);
|
|
|
|
//
|
|
// The interface better also be present
|
|
//
|
|
|
|
pInterface = pAdapter->pInterface;
|
|
|
|
RtAssert(pInterface);
|
|
|
|
if(pConnEntry->duUsage isnot DU_CALLIN)
|
|
{
|
|
//
|
|
// For non clients, also lock the interface
|
|
//
|
|
|
|
RtAcquireSpinLockAtDpcLevel(&(pInterface->rlLock));
|
|
}
|
|
|
|
#if DBG
|
|
|
|
Trace(RCV, INFO,
|
|
("ReceiveCommon: Extracted adapter %x with name %s\n",
|
|
pAdapter,
|
|
pAdapter->asDeviceNameA.Buffer));
|
|
|
|
pIpHeader = (IP_HEADER UNALIGNED *)pvData;
|
|
|
|
RtAssert((pIpHeader->byVerLen & 0xF0) is 0x40);
|
|
RtAssert(LengthOfIpHeader(pIpHeader) >= 20);
|
|
|
|
//
|
|
// If the packet is not fragmented, then the data from its header
|
|
// should be <= the data ndis gives us (<= because there may be
|
|
// padding and ndis may be giving us trailing bytes)
|
|
//
|
|
|
|
//RtAssert(RtlUshortByteSwap(pIpHeader->wLength) <= uiTotalDataLen);
|
|
|
|
#endif
|
|
|
|
//
|
|
// Increment some stats. For the server interface, these can be
|
|
// inconsistent
|
|
//
|
|
|
|
pInterface->ulInOctets += (uiTotalDataLen + uiHeaderLen);
|
|
|
|
//
|
|
// Verify this is a well formed packet
|
|
//
|
|
|
|
wType = RtlUshortByteSwap(pEthHeader->wType);
|
|
|
|
if(wType isnot ARP_ETYPE_IP)
|
|
{
|
|
pInterface->ulInUnknownProto++;
|
|
|
|
Trace(RCV, ERROR,
|
|
("ReceiveCommon: Type %d is wrong\n", wType));
|
|
|
|
if(pConnEntry->duUsage isnot DU_CALLIN)
|
|
{
|
|
RtReleaseSpinLockFromDpcLevel(&(pInterface->rlLock));
|
|
}
|
|
|
|
RtReleaseSpinLock(pConnEntry->prlLock,
|
|
kiIrql);
|
|
|
|
DereferenceConnEntry(pConnEntry);
|
|
|
|
return NDIS_STATUS_NOT_RECOGNIZED;
|
|
}
|
|
|
|
//
|
|
// Need to figure out if this is a unicast or a broadcast. At the
|
|
// link layer we dont have a concept of broadcast. So, we always mark
|
|
// this as unicast. We can be smarter about this and look at the
|
|
// IPHeader and decide based on the IP dest addr
|
|
//
|
|
|
|
bNonUnicast = FALSE;
|
|
|
|
pInterface->ulInUniPkts++;
|
|
|
|
//
|
|
// Check if the filtering of the Netbios packets is enabled on this
|
|
// connection. If so then do not indicate the packet.
|
|
//
|
|
|
|
if((pConnEntry->bFilterNetBios is TRUE) and
|
|
(WanpDropNetbiosPacket((PBYTE)pvData,uiFirstBufferLen)))
|
|
{
|
|
pInterface->ulInDiscards++;
|
|
|
|
Trace(RCV, TRACE,
|
|
("ReceiveCommon: Dropping Netbios packet\n", wType));
|
|
|
|
if(pConnEntry->duUsage isnot DU_CALLIN)
|
|
{
|
|
RtReleaseSpinLockFromDpcLevel(&(pInterface->rlLock));
|
|
}
|
|
|
|
RtReleaseSpinLock(pConnEntry->prlLock,
|
|
kiIrql);
|
|
|
|
DereferenceConnEntry(pConnEntry);
|
|
|
|
return NDIS_STATUS_NOT_ACCEPTED;
|
|
}
|
|
|
|
//
|
|
// Release the lock BEFORE we tell IP
|
|
//
|
|
|
|
if(pConnEntry->duUsage isnot DU_CALLIN)
|
|
{
|
|
RtReleaseSpinLockFromDpcLevel(&(pInterface->rlLock));
|
|
}
|
|
|
|
RtReleaseSpinLock(pConnEntry->prlLock,
|
|
kiIrql);
|
|
|
|
#if PKT_DBG
|
|
|
|
Trace(RCV, ERROR,
|
|
("ReceiveCommon: \nMdl %x Pkt %x Hdr %x Data %x\n",
|
|
pMdl,
|
|
nhXferContext,
|
|
pvHeader,
|
|
pvData));
|
|
|
|
#endif // PKT_DBG
|
|
|
|
if(pMdl)
|
|
{
|
|
g_pfnIpRcvPkt(pAdapter->pvIpContext,
|
|
(PBYTE)pvData,
|
|
uiFirstBufferLen,
|
|
uiTotalDataLen,
|
|
nhXferContext,
|
|
0,
|
|
bNonUnicast,
|
|
sizeof(ETH_HEADER),
|
|
pMdl,
|
|
piClientCount,
|
|
pConnEntry->pvIpLinkContext);
|
|
}
|
|
else
|
|
{
|
|
g_pfnIpRcv(pAdapter->pvIpContext,
|
|
(PBYTE)pvData,
|
|
uiFirstBufferLen,
|
|
uiTotalDataLen,
|
|
nhXferContext,
|
|
0,
|
|
bNonUnicast,
|
|
pConnEntry->pvIpLinkContext);
|
|
}
|
|
|
|
DereferenceConnEntry(pConnEntry);
|
|
|
|
return NDIS_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
VOID
|
|
WanNdisReceiveComplete(
|
|
NDIS_HANDLE nhBindHandle
|
|
)
|
|
{
|
|
TraceEnter(RCV, "NdisReceiveComplete");
|
|
|
|
g_pfnIpRcvComplete();
|
|
}
|
|
|
|
NDIS_STATUS
|
|
WanIpTransferData(
|
|
PVOID pvContext,
|
|
NDIS_HANDLE nhMacContext,
|
|
UINT uiProtoOffset,
|
|
UINT uiTransferOffset,
|
|
UINT uiTransferLength,
|
|
PNDIS_PACKET pnpPacket,
|
|
PUINT puiTransferred
|
|
)
|
|
{
|
|
RtAssert(FALSE);
|
|
|
|
return NDIS_STATUS_SUCCESS;
|
|
}
|
|
|
|
VOID
|
|
WanNdisTransferDataComplete(
|
|
NDIS_HANDLE nhProtocolContext,
|
|
PNDIS_PACKET pnpPacket,
|
|
NDIS_STATUS nsStatus,
|
|
UINT uiBytesCopied
|
|
)
|
|
{
|
|
RtAssert(FALSE);
|
|
|
|
return;
|
|
}
|
|
|
|
UINT
|
|
WanIpReturnPacket(
|
|
PVOID pvContext,
|
|
PNDIS_PACKET pnpPacket
|
|
)
|
|
{
|
|
Trace(RCV, ERROR,
|
|
("IpReturnPacket: %x\n",
|
|
pnpPacket));
|
|
|
|
NdisReturnPackets(&pnpPacket,
|
|
1);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOLEAN
|
|
WanpDropNetbiosPacket(
|
|
PBYTE pbyBuffer,
|
|
ULONG ulBufferLen
|
|
)
|
|
{
|
|
IP_HEADER UNALIGNED *pIpHeader;
|
|
PBYTE pbyUdpPacket;
|
|
WORD wSrcPort;
|
|
ULONG ulIpHdrLen;
|
|
|
|
pIpHeader = (IP_HEADER UNALIGNED *)pbyBuffer;
|
|
|
|
if(pIpHeader->byProtocol is 0x11)
|
|
{
|
|
ulIpHdrLen = LengthOfIpHeader(pIpHeader);
|
|
|
|
//
|
|
// If we cant get to the 10th byte in the UDP packet in this
|
|
// buffer, we just let the packet go
|
|
//
|
|
|
|
if(ulBufferLen < ulIpHdrLen + 10)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
pbyUdpPacket = (PBYTE)((ULONG_PTR)pbyBuffer + ulIpHdrLen);
|
|
|
|
wSrcPort = *((WORD UNALIGNED *)pbyUdpPacket);
|
|
|
|
if(wSrcPort is PORT137_NBO)
|
|
{
|
|
//
|
|
// UDP port 137 is NETBIOS/IP traffic
|
|
//
|
|
|
|
//
|
|
// Allow only WINS Query Requests to go through
|
|
// WINS Query packets have x0000xxx in the 10th byte
|
|
//
|
|
|
|
if(((*(pbyUdpPacket + 10)) & 0x78) isnot 0)
|
|
{
|
|
return TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|