Source code of Windows XP (NT5)
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.
 
 
 
 
 
 

1903 lines
49 KiB

/*++
Copyright (c) 1992 Microsoft Corporation
Module Name:
nbp.c
Abstract:
This module contains nbp code
Author:
Jameel Hyder ([email protected])
Nikhil Kamkolkar ([email protected])
Revision History:
19 Jun 1992 Initial Version
Notes: Tab stop: 4
--*/
#include <atalk.h>
#pragma hdrstop
#define FILENUM NBP
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE_NZ, AtalkNbpAction)
#pragma alloc_text(PAGE_NZ, atalkNbpLinkPendingNameInList)
#pragma alloc_text(PAGE_NZ, atalkNbpSendRequest)
#endif
/*** AtalkNbpPacketIn
*
*/
VOID
AtalkNbpPacketIn(
IN PPORT_DESCRIPTOR pPortDesc,
IN PDDP_ADDROBJ pDdpAddr,
IN PBYTE pPkt,
IN USHORT PktLen,
IN PATALK_ADDR pSrcAddr,
IN PATALK_ADDR pDstAddr,
IN ATALK_ERROR ErrorCode,
IN BYTE DdpType,
IN PVOID pHandlerCtx,
IN BOOLEAN OptimizedPath,
IN PVOID OptimizeCtx
)
{
PPEND_NAME pPendName;
PATALK_NODE pNode;
TIME TimeS, TimeE, TimeD;
PNBPHDR pNbpHdr = (PNBPHDR)pPkt;
PRTE pRte;
SHORT i, NbpId, NbpCmd, TupleCnt;
PNBPTUPLE pNbpTuple = NULL, pInBufTuple = NULL;
BOOLEAN DefZone = FALSE, RestartTimer = FALSE;
BOOLEAN fWeCancelledTimer = TRUE;
BOOLEAN Found;
ASSERT(KeGetCurrentIrql() == DISPATCH_LEVEL);
TimeS = KeQueryPerformanceCounter(NULL);
do
{
if ((ErrorCode == ATALK_SOCKET_CLOSED) || (DdpType != DDPPROTO_NBP))
break;
else if ((ErrorCode != ATALK_NO_ERROR) || (PktLen < sizeof(NBPHDR)))
{
break;
}
// Get NBP header information and decide what to do
NbpCmd = (SHORT)((pNbpHdr->_CmdAndTupleCnt >> 4) & 0x0F);
TupleCnt = (SHORT)(pNbpHdr->_CmdAndTupleCnt & 0x0F);
NbpId = (SHORT)pNbpHdr->_NbpId;
if ((pNbpTuple = AtalkAllocMemory(sizeof(NBPTUPLE))) == NULL)
{
TMPLOGERR();
break;
}
switch (NbpCmd)
{
case NBP_LOOKUP:
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Cmd Lookup\n"));
if ((TupleCnt == 1) &&
(atalkNbpDecodeTuple(pPkt + sizeof(NBPHDR),
(USHORT)(PktLen - sizeof(NBPHDR)),
pNbpTuple) > 0))
{
atalkNbpLookupNames(pPortDesc, pDdpAddr, pNbpTuple, NbpId);
}
else
{
#if 0
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
#endif
break;
}
break;
case NBP_BROADCAST_REQUEST:
case NBP_FORWARD_REQUEST:
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Cmd %sRequest\n",
(NbpCmd == NBP_BROADCAST_REQUEST) ? "Broadcast" : "Forward"));
// We don't care if we are not a router
if ((pPortDesc->pd_Flags & PD_ROUTER_RUNNING) == 0)
break;
if (TupleCnt != 1)
{
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
if (atalkNbpDecodeTuple(pPkt + sizeof(NBPHDR),
(USHORT)(PktLen - sizeof(NBPHDR)),
pNbpTuple) == 0)
{
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
if ((pNbpTuple->tpl_ZoneLen == 0) ||
((pNbpTuple->tpl_Zone[0] == '*') && (pNbpTuple->tpl_ZoneLen == 1)))
DefZone = TRUE;
if (EXT_NET(pPortDesc))
{
if (DefZone)
{
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
}
else // Non-extended network
{
if (DefZone)
{
if (pPortDesc->pd_NetworkRange.anr_FirstNetwork != pSrcAddr->ata_Network)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("AtalkNbpPacketIn: LT Port, '*' zone - SrcAddr %d.%d, Net %d\n",
pSrcAddr->ata_Network, pSrcAddr->ata_Node,
pPortDesc->pd_NetworkRange.anr_FirstNetwork));
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
if (!(pPortDesc->pd_Flags & PD_VALID_DESIRED_ZONE))
break;
ACQUIRE_SPIN_LOCK_DPC(&pPortDesc->pd_Lock);
pNbpTuple->tpl_ZoneLen = pPortDesc->pd_DesiredZone->zn_ZoneLen;
RtlCopyMemory(pNbpTuple->tpl_Zone,
pPortDesc->pd_DesiredZone->zn_Zone,
pPortDesc->pd_DesiredZone->zn_ZoneLen);
RELEASE_SPIN_LOCK_DPC(&pPortDesc->pd_Lock);
}
}
// For a forward request send a lookup datagram
if (NbpCmd == NBP_FORWARD_REQUEST)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("AtalkNbpPacketIn: Sending NbpLookup for a NbpForwardRequest\n"));
atalkNbpSendLookupDatagram(pPortDesc, pDdpAddr, NbpId, pNbpTuple);
break;
}
// We have a broadcast request. Walk through the routing tables
// sending either a forward request or a lookup (broadcast) to
// each network that contains the specified zone
ACQUIRE_SPIN_LOCK_DPC(&AtalkRteLock);
for (i = 0;i < NUM_RTMP_HASH_BUCKETS; i++)
{
for (pRte = AtalkRoutingTable[i];
pRte != NULL;
pRte = pRte->rte_Next)
{
ATALK_ERROR Status;
// If the network is directly connected i.e. 0 hops away
// use the zone-list in the PortDesc rather than the
// routing table - the routing table may not be filled
// in with a zone list (due to the normal ZipQuery mechanism)
ACQUIRE_SPIN_LOCK_DPC(&pRte->rte_Lock);
if (!(pRte->rte_Flags & RTE_ZONELIST_VALID))
{
if ((pRte->rte_NumHops != 0) ||
!AtalkZoneNameOnList(pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
pRte->rte_PortDesc->pd_ZoneList))
{
RELEASE_SPIN_LOCK_DPC(&pRte->rte_Lock);
continue;
}
}
else if (!AtalkZoneNameOnList(pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
pRte->rte_ZoneList))
{
RELEASE_SPIN_LOCK_DPC(&pRte->rte_Lock);
continue;
}
pRte->rte_RefCount ++;
RELEASE_SPIN_LOCK_DPC(&pRte->rte_Lock);
// If not a local network, send a forward request
if (pRte->rte_NumHops != 0)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("AtalkNbpPacketIn: Sending NbpForwardRequest for a broadcast\n"));
// Do not hold the Rte lock during a DdpSend
RELEASE_SPIN_LOCK_DPC(&AtalkRteLock);
atalkNbpSendForwardRequest(pDdpAddr,
pRte,
NbpId,
pNbpTuple);
ACQUIRE_SPIN_LOCK_DPC(&AtalkRteLock);
AtalkRtmpDereferenceRte(pRte, TRUE);
}
else
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("AtalkNbpPacketIn: Sending Lookup for a broadcast\n"));
// Send a lookup
RELEASE_SPIN_LOCK_DPC(&AtalkRteLock);
atalkNbpSendLookupDatagram(pRte->rte_PortDesc,
NULL,
NbpId,
pNbpTuple);
ACQUIRE_SPIN_LOCK_DPC(&AtalkRteLock);
AtalkRtmpDereferenceRte(pRte, TRUE);
}
}
}
RELEASE_SPIN_LOCK_DPC(&AtalkRteLock);
break;
case NBP_LOOKUP_REPLY:
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Cmd LookupReply\n"));
// This had better be a response to a previous lookup
// Look for a pending name on all open sockets on this node
if (TupleCnt == 0)
break;
// Decode the tuple for Register/Confirm case
if (atalkNbpDecodeTuple(pPkt + sizeof(NBPHDR),
(USHORT)(PktLen - sizeof(NBPHDR)),
pNbpTuple) == 0)
{
break;
}
pNode = pDdpAddr->ddpao_Node;
ACQUIRE_SPIN_LOCK_DPC(&pNode->an_Lock);
Found = FALSE;
for (i = 0; (i < NODE_DDPAO_HASH_SIZE) && !Found; i++)
{
PDDP_ADDROBJ pSkt;
for (pSkt = pNode->an_DdpAoHash[i];
(pSkt != NULL) && !Found;
pSkt = pSkt->ddpao_Next)
{
PPEND_NAME * ppPendName;
ACQUIRE_SPIN_LOCK_DPC(&pSkt->ddpao_Lock);
for (ppPendName = &pSkt->ddpao_PendNames;
(pPendName = *ppPendName) != NULL;
ppPendName = &pPendName->pdn_Next)
{
ASSERT (VALID_PENDNAME(pPendName));
if (pPendName->pdn_Flags & PDN_CLOSING)
{
continue;
}
if (pPendName->pdn_NbpId == NbpId)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: LookupReply Found name\n"));
Found = TRUE;
ACQUIRE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
pPendName->pdn_RefCount ++;
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
break;
}
}
RELEASE_SPIN_LOCK_DPC(&pSkt->ddpao_Lock);
}
}
RELEASE_SPIN_LOCK_DPC(&pNode->an_Lock);
// If the timer fired just before we could find and cancel it
if (pPendName == NULL)
break;
do
{
if (AtalkTimerCancelEvent(&pPendName->pdn_Timer, NULL))
{
// if the timer was successfully cancelled, take away
// the reference for it
atalkNbpDerefPendName(pPendName);
RestartTimer = TRUE;
}
else
{
fWeCancelledTimer = FALSE;
}
if ((pPendName->pdn_Reason == FOR_REGISTER) ||
(pPendName->pdn_Reason == FOR_CONFIRM))
{
BOOLEAN NoMatch;
// Does the reply match the one we're trying to register ?
NoMatch = ( (TupleCnt != 1) ||
(pPendName->pdn_pRegdName == NULL) ||
!AtalkFixedCompareCaseInsensitive(
pPendName->pdn_pRegdName->rdn_Tuple.tpl_Object,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_ObjectLen,
pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen) ||
!AtalkFixedCompareCaseInsensitive(
pPendName->pdn_pRegdName->rdn_Tuple.tpl_Type,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_TypeLen,
pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen));
if (NoMatch)
{
if (TupleCnt != 1)
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
// If we are registering, we're done as someone already
// has our name
if (pPendName->pdn_Reason == FOR_REGISTER)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Register failure\n"));
ACQUIRE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
pPendName->pdn_Status = ATALK_SHARING_VIOLATION;
pPendName->pdn_Flags |= PDN_CLOSING;
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
if (fWeCancelledTimer)
{
atalkNbpDerefPendName(pPendName); // Take away creation ref
}
RestartTimer = FALSE;
break;
}
// We're confirming, if no match get out
if ((pPendName->pdn_ConfirmAddr.ata_Network != pNbpTuple->tpl_Address.ata_Network) ||
(pPendName->pdn_ConfirmAddr.ata_Node != pNbpTuple->tpl_Address.ata_Node))
{
break;
}
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Confirm success\n"));
ACQUIRE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
pPendName->pdn_Status = ATALK_NO_ERROR;
((PNBP_CONFIRM_PARAMS)(pPendName->pdn_pActReq->ar_pParms))->ConfirmTuple.Address.Address =
pNbpTuple->tpl_Address.ata_Address;
if (pPendName->pdn_ConfirmAddr.ata_Socket != pNbpTuple->tpl_Address.ata_Socket)
{
pPendName->pdn_Status = ATALK_NEW_SOCKET;
}
pPendName->pdn_Flags |= PDN_CLOSING;
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
atalkNbpDerefPendName(pPendName); // Take away creation ref
RestartTimer = FALSE;
}
else // FOR_LOOKUP
{
int i, j, tmp, NextTupleOff = sizeof(NBPHDR);
BOOLEAN Done = FALSE;
ULONG BytesCopied;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Lookup searching...\n"));
// Allocate space for an NBP tuple for copying and comparing
// Failure to allocate can result in duplicates - so be it
if (pInBufTuple == NULL)
pInBufTuple = AtalkAllocMemory(sizeof(NBPTUPLE));
for (i = 0; i < TupleCnt && !Done; i++)
{
BOOLEAN Duplicate = FALSE;
// If we encounter a bad tuple, ignore the rest. Drop tuples which are
// our names and we are set to drop them !!!
if (((tmp = atalkNbpDecodeTuple(pPkt + NextTupleOff,
(USHORT)(PktLen - NextTupleOff),
pNbpTuple)) == 0) ||
(AtalkFilterOurNames &&
(((pNbpTuple->tpl_Address.ata_Network == AtalkUserNode1.atn_Network) &&
(pNbpTuple->tpl_Address.ata_Node == AtalkUserNode1.atn_Node)) ||
((pNbpTuple->tpl_Address.ata_Network == AtalkUserNode2.atn_Network) &&
(pNbpTuple->tpl_Address.ata_Node == AtalkUserNode2.atn_Node)))))
break;
NextTupleOff += tmp;
ACQUIRE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
// Now walk through the tuples that we already picked
// up and drop duplicates
if (pInBufTuple != NULL)
{
for (j = 0; j < pPendName->pdn_TotalTuples; j++)
{
TdiCopyMdlToBuffer(pPendName->pdn_pAMdl,
j * sizeof(NBPTUPLE),
(PBYTE)pInBufTuple,
0,
sizeof(NBPTUPLE),
&BytesCopied);
ASSERT (BytesCopied == sizeof(NBPTUPLE));
if ((pInBufTuple->tpl_Address.ata_Network ==
pNbpTuple->tpl_Address.ata_Network) &&
(pInBufTuple->tpl_Address.ata_Node ==
pNbpTuple->tpl_Address.ata_Node) &&
(pInBufTuple->tpl_Address.ata_Socket ==
pNbpTuple->tpl_Address.ata_Socket))
{
Duplicate = TRUE;
break;
}
}
if (Duplicate)
{
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
continue;
}
}
// We are guaranteed that there is space available
// for another tuple.
TdiCopyBufferToMdl((PBYTE)pNbpTuple,
0,
sizeof(NBPTUPLE),
pPendName->pdn_pAMdl,
pPendName->pdn_TotalTuples * sizeof(NBPTUPLE),
&BytesCopied);
ASSERT (BytesCopied == sizeof(NBPTUPLE));
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Lookup, found a tuple\n"));
pPendName->pdn_TotalTuples ++;
if ((pPendName->pdn_TotalTuples == pPendName->pdn_MaxTuples) ||
(pPendName->pdn_MdlLength -
(pPendName->pdn_TotalTuples * sizeof(NBPTUPLE)) <
sizeof(NBPTUPLE)))
{
Done = TRUE;
((PNBP_LOOKUP_PARAMS)(pPendName->pdn_pActReq->ar_pParms))->NoTuplesRead =
pPendName->pdn_TotalTuples;
pPendName->pdn_Status = ATALK_NO_ERROR;
pPendName->pdn_Flags |= PDN_CLOSING;
RestartTimer = FALSE;
}
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
if (Done)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Lookup calling completion\n"));
if (fWeCancelledTimer)
{
atalkNbpDerefPendName(pPendName); // Take away creation ref
}
break;
}
}
}
} while (FALSE);
if (RestartTimer)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpPacketIn: Restarting timer\n"));
ACQUIRE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
pPendName->pdn_RefCount ++;
RELEASE_SPIN_LOCK_DPC(&pPendName->pdn_Lock);
AtalkTimerScheduleEvent(&pPendName->pdn_Timer);
}
atalkNbpDerefPendName(pPendName);
break;
default:
AtalkLogBadPacket(pPortDesc,
pSrcAddr,
pDstAddr,
pPkt,
PktLen);
break;
}
} while (FALSE);
if (pNbpTuple != NULL)
AtalkFreeMemory(pNbpTuple);
if (pInBufTuple != NULL)
AtalkFreeMemory(pInBufTuple);
TimeE = KeQueryPerformanceCounter(NULL);
TimeD.QuadPart = TimeE.QuadPart - TimeS.QuadPart;
INTERLOCKED_ADD_LARGE_INTGR(
&pPortDesc->pd_PortStats.prtst_NbpPacketInProcessTime,
TimeD,
&AtalkStatsLock.SpinLock);
INTERLOCKED_INCREMENT_LONG_DPC(
&pPortDesc->pd_PortStats.prtst_NumNbpPacketsIn,
&AtalkStatsLock.SpinLock);
}
/*** atalkNbpTimer
*
*/
LOCAL LONG FASTCALL
atalkNbpTimer(
IN PTIMERLIST pTimer,
IN BOOLEAN TimerShuttingDown
)
{
PPEND_NAME pCurrPendName;
ATALK_ERROR error;
PDDP_ADDROBJ pDdpAddr;
BOOLEAN RestartTimer = TRUE;
BYTE Reason;
pCurrPendName = (PPEND_NAME)CONTAINING_RECORD(pTimer, PEND_NAME, pdn_Timer);
ASSERT (VALID_PENDNAME(pCurrPendName));
Reason = pCurrPendName->pdn_Reason;
ASSERT ((Reason == FOR_REGISTER) ||
(Reason == FOR_LOOKUP) ||
(Reason == FOR_CONFIRM));
pDdpAddr = pCurrPendName->pdn_pDdpAddr;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: For Socket %lx, PendName %lx\n",
pDdpAddr, pCurrPendName));
ACQUIRE_SPIN_LOCK_DPC(&pCurrPendName->pdn_Lock);
if (TimerShuttingDown ||
(pCurrPendName->pdn_Flags & PDN_CLOSING))
pCurrPendName->pdn_RemainingBroadcasts = 1;
if (--(pCurrPendName->pdn_RemainingBroadcasts) == 0)
{
RestartTimer = FALSE;
if (Reason == FOR_REGISTER)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: Register success\n"));
ACQUIRE_SPIN_LOCK_DPC(&pDdpAddr->ddpao_Lock);
pCurrPendName->pdn_pRegdName->rdn_Next = pDdpAddr->ddpao_RegNames;
pDdpAddr->ddpao_RegNames = pCurrPendName->pdn_pRegdName;
RELEASE_SPIN_LOCK_DPC(&pDdpAddr->ddpao_Lock);
pCurrPendName->pdn_Flags &= ~PDN_FREE_REGDNAME;
}
pCurrPendName->pdn_Flags |= PDN_CLOSING;
}
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: Remaining broadcasts %d\n",
pCurrPendName->pdn_RemainingBroadcasts));
if (RestartTimer)
{
pCurrPendName->pdn_RefCount ++;
RELEASE_SPIN_LOCK_DPC(&pCurrPendName->pdn_Lock);
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: Sending another request\n"));
if (!atalkNbpSendRequest(pCurrPendName))
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("atalkNbpTimer: atalkNbpSendRequest failed\n"));
}
}
else
{
RELEASE_SPIN_LOCK_DPC(&pCurrPendName->pdn_Lock);
error = ATALK_NO_ERROR;
if (Reason == FOR_CONFIRM)
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: Confirm Failure\n"));
error = ATALK_TIMEOUT;
}
else if (Reason == FOR_LOOKUP)
{
((PNBP_LOOKUP_PARAMS)(pCurrPendName->pdn_pActReq->ar_pParms))->NoTuplesRead =
pCurrPendName->pdn_TotalTuples;
}
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpTimer: Calling completion routine\n"));
pCurrPendName->pdn_Status = error;
atalkNbpDerefPendName(pCurrPendName); // Take away creation reference
}
atalkNbpDerefPendName(pCurrPendName);
return (RestartTimer ? ATALK_TIMER_REQUEUE : ATALK_TIMER_NO_REQUEUE);
}
/*** atalkNbpLookupNames
*
*/
LOCAL VOID
atalkNbpLookupNames(
IN PPORT_DESCRIPTOR pPortDesc,
IN PDDP_ADDROBJ pDdpAddr,
IN PNBPTUPLE pNbpTuple,
IN SHORT NbpId
)
{
int i, index, TupleCnt;
BOOLEAN AllocNewBuffDesc = TRUE;
PBUFFER_DESC pBuffDesc,
pBuffDescStart = NULL,
*ppBuffDesc = &pBuffDescStart;
PATALK_NODE pNode = pDdpAddr->ddpao_Node;
SEND_COMPL_INFO SendInfo;
PBYTE Datagram;
ASSERT(KeGetCurrentIrql() == DISPATCH_LEVEL);
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Entered\n"));
// Does the requestor atleast has the right zone ?
if ((pNbpTuple->tpl_Zone[0] != '*') ||
(pNbpTuple->tpl_ZoneLen != 1))
{
// If either we do not know our zone or if it does not match or
// we are an extended network, return - we have nothing to do
if (EXT_NET(pPortDesc))
{
if (!(pPortDesc->pd_Flags & PD_VALID_DESIRED_ZONE) ||
((pPortDesc->pd_DesiredZone == NULL) ?1:
(!AtalkFixedCompareCaseInsensitive(pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
pPortDesc->pd_DesiredZone->zn_Zone,
pPortDesc->pd_DesiredZone->zn_ZoneLen))))
{
return;
}
}
}
// Walk the registered names list on all sockets open on this node and
// see if we have a matching name. We have to walk the pending names
// list also (should not answer, if we the node trying to register the
// name).
ACQUIRE_SPIN_LOCK_DPC(&pDdpAddr->ddpao_Node->an_Lock);
for (i = 0; i < NODE_DDPAO_HASH_SIZE; i++)
{
PDDP_ADDROBJ pSkt;
for (pSkt = pNode->an_DdpAoHash[i];
pSkt != NULL;
pSkt = pSkt->ddpao_Next)
{
PREGD_NAME pRegdName;
PPEND_NAME pPendName;
ACQUIRE_SPIN_LOCK_DPC(&pSkt->ddpao_Lock);
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Checking Socket %lx\n", pSkt));
// First check registered names
for (pRegdName = pSkt->ddpao_RegNames;
pRegdName != NULL;
pRegdName = pRegdName->rdn_Next)
{
ASSERT (VALID_REGDNAME(pRegdName));
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Checking RegdName %lx\n", pRegdName));
if (!atalkNbpMatchWild(pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen,
pRegdName->rdn_Tuple.tpl_Object,
pRegdName->rdn_Tuple.tpl_ObjectLen) ||
!atalkNbpMatchWild(pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen,
pRegdName->rdn_Tuple.tpl_Type,
pRegdName->rdn_Tuple.tpl_TypeLen))
continue;
// Allocate a new buffer descriptor, if we must
if (AllocNewBuffDesc)
{
if ((pBuffDesc = AtalkAllocBuffDesc(NULL,
MAX_DGRAM_SIZE,
BD_CHAR_BUFFER | BD_FREE_BUFFER)) == NULL)
break;
Datagram = pBuffDesc->bd_CharBuffer;
index = sizeof(NBPHDR);
TupleCnt = 0;
*ppBuffDesc = pBuffDesc;
pBuffDesc->bd_Next = NULL;
ppBuffDesc = &pBuffDesc->bd_Next;
AllocNewBuffDesc = FALSE;
}
// We have a match. Build complete Nbp tuple
index += atalkNbpEncodeTuple(&pRegdName->rdn_Tuple,
"*",
1,
0,
Datagram+index);
TupleCnt ++;
if (((index + MAX_NBP_TUPLELENGTH) > MAX_DGRAM_SIZE) ||
(TupleCnt == 0x0F))
{
((PNBPHDR)Datagram)->_NbpId = (BYTE)NbpId;
((PNBPHDR)Datagram)->_CmdAndTupleCnt =
(NBP_LOOKUP_REPLY << 4) + TupleCnt;
AllocNewBuffDesc = TRUE;
pBuffDesc->bd_Length = (SHORT)index;
}
}
// Now check pending names
for (pPendName = pSkt->ddpao_PendNames;
pPendName != NULL;
pPendName = pPendName->pdn_Next)
{
ASSERT (VALID_PENDNAME(pPendName));
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Checking PendName %lx\n", pPendName));
// Ignore all but the ones that are being registered
if (pPendName->pdn_Reason != FOR_REGISTER)
continue;
// Also those that we are registering
if ((pSkt->ddpao_Node->an_NodeAddr.atn_Network ==
pNbpTuple->tpl_Address.ata_Network) &&
(pSkt->ddpao_Node->an_NodeAddr.atn_Node ==
pNbpTuple->tpl_Address.ata_Node) &&
(pPendName->pdn_NbpId == (BYTE)NbpId))
continue;
if ((pPendName->pdn_pRegdName == NULL) ||
!atalkNbpMatchWild(
pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_Object,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_ObjectLen) ||
!atalkNbpMatchWild(
pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_Type,
pPendName->pdn_pRegdName->rdn_Tuple.tpl_TypeLen))
{
continue;
}
// Allocate a new buffer descriptor, if we must
if (AllocNewBuffDesc)
{
if ((pBuffDesc = AtalkAllocBuffDesc(NULL,
MAX_DGRAM_SIZE,
BD_CHAR_BUFFER | BD_FREE_BUFFER)) == NULL)
break;
Datagram = pBuffDesc->bd_CharBuffer;
index = sizeof(NBPHDR);
TupleCnt = 0;
*ppBuffDesc = pBuffDesc;
pBuffDesc->bd_Next = NULL;
ppBuffDesc = &pBuffDesc->bd_Next;
AllocNewBuffDesc = FALSE;
}
// We have a match. Build complete Nbp tuple
index += atalkNbpEncodeTuple(&pPendName->pdn_pRegdName->rdn_Tuple,
"*",
1,
0,
Datagram+index);
TupleCnt ++;
if (((index + MAX_NBP_TUPLELENGTH) > MAX_DGRAM_SIZE) ||
(TupleCnt == 0x0F))
{
((PNBPHDR)Datagram)->_NbpId = (BYTE)NbpId;
((PNBPHDR)Datagram)->_CmdAndTupleCnt =
(NBP_LOOKUP_REPLY << 4) + TupleCnt;
AllocNewBuffDesc = TRUE;
pBuffDesc->bd_Length = (SHORT)index;
}
}
RELEASE_SPIN_LOCK_DPC(&pSkt->ddpao_Lock);
}
}
RELEASE_SPIN_LOCK_DPC(&pDdpAddr->ddpao_Node->an_Lock);
// Close the current buffdesc
if (!AllocNewBuffDesc)
{
((PNBPHDR)Datagram)->_NbpId = (BYTE)NbpId;
((PNBPHDR)Datagram)->_CmdAndTupleCnt = (NBP_LOOKUP_REPLY << 4) + TupleCnt;
pBuffDesc->bd_Length = (SHORT)index;
}
// Now blast off all the datagrams that we have filled up
SendInfo.sc_TransmitCompletion = atalkNbpSendComplete;
// SendInfo.sc_Ctx2 = NULL;
// SendInfo.sc_Ctx3 = NULL;
for (pBuffDesc = pBuffDescStart;
pBuffDesc != NULL;
pBuffDesc = pBuffDescStart)
{
ATALK_ERROR ErrorCode;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Sending lookup response\n"));
pBuffDescStart = pBuffDesc->bd_Next;
pBuffDesc->bd_Next = NULL;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLookupNames: Sending %lx\n", pBuffDesc));
ASSERT(pBuffDesc->bd_Length > 0);
// Length is already properly set in the buffer descriptor.
SendInfo.sc_Ctx1 = pBuffDesc;
ErrorCode = AtalkDdpSend(pDdpAddr,
&pNbpTuple->tpl_Address,
DDPPROTO_NBP,
FALSE,
pBuffDesc,
NULL,
0,
NULL,
&SendInfo);
if (!ATALK_SUCCESS(ErrorCode))
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_WARN,
("atalkNbpLookupNames: DdpSend failed %ld\n", ErrorCode));
AtalkFreeBuffDesc(pBuffDesc);
}
}
}
/*** AtalkNbpAction
*
*/
ATALK_ERROR
AtalkNbpAction(
IN PDDP_ADDROBJ pDdpAddr,
IN BYTE Reason,
IN PNBPTUPLE pNbpTuple,
OUT PAMDL pAMdl OPTIONAL, // FOR_LOOKUP
IN USHORT MaxTuples OPTIONAL, // FOR_LOOKUP
IN PACTREQ pActReq
)
{
PPORT_DESCRIPTOR pPortDesc;
PPEND_NAME pPendName = NULL;
PREGD_NAME pRegdName = NULL;
ATALK_ERROR Status = ATALK_INVALID_PARAMETER;
LONG MdlLen = 0;
BOOLEAN DefZone = FALSE;
ASSERT (Reason == FOR_REGISTER ||
Reason == FOR_CONFIRM ||
Reason == FOR_LOOKUP);
do
{
if ((pNbpTuple->tpl_ObjectLen == 0) ||
(pNbpTuple->tpl_ObjectLen > MAX_ENTITY_LENGTH) ||
(pNbpTuple->tpl_TypeLen == 0) ||
(pNbpTuple->tpl_TypeLen > MAX_ENTITY_LENGTH))
break;
if ((Reason == FOR_LOOKUP) &&
((pAMdl == NULL) ||
((MdlLen = AtalkSizeMdlChain(pAMdl)) < sizeof(NBPTUPLE))))
{
Status = ATALK_BUFFER_TOO_SMALL;
break;
}
pPortDesc = pDdpAddr->ddpao_Node->an_Port;
if (pNbpTuple->tpl_ZoneLen != 0)
{
if (pNbpTuple->tpl_ZoneLen > MAX_ENTITY_LENGTH)
break;
if (((pNbpTuple->tpl_Zone[0] == '*') && (pNbpTuple->tpl_ZoneLen == 1)) ||
((pPortDesc->pd_DesiredZone != NULL) &&
AtalkFixedCompareCaseInsensitive(pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
pPortDesc->pd_DesiredZone->zn_Zone,
pPortDesc->pd_DesiredZone->zn_ZoneLen)))
{
DefZone = TRUE;
}
}
else
{
pNbpTuple->tpl_Zone[0] = '*';
pNbpTuple->tpl_ZoneLen = 1;
DefZone = TRUE;
}
if (Reason != FOR_LOOKUP) // i.e. REGISTER or CONFIRM
{
if ((pNbpTuple->tpl_Object[0] == '=') ||
(pNbpTuple->tpl_Type[0] == '=') ||
(AtalkSearchBuf(pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen,
NBP_WILD_CHARACTER) != NULL) ||
(AtalkSearchBuf(pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen,
NBP_WILD_CHARACTER) != NULL))
break;
if ((Reason == FOR_REGISTER) && !DefZone)
break;
}
// For extended networks, set the zone name correctly
if (DefZone &&
(pPortDesc->pd_Flags & (PD_EXT_NET | PD_VALID_DESIRED_ZONE)) ==
(PD_EXT_NET | PD_VALID_DESIRED_ZONE))
{
RtlCopyMemory(pNbpTuple->tpl_Zone,
pPortDesc->pd_DesiredZone->zn_Zone,
pPortDesc->pd_DesiredZone->zn_ZoneLen);
pNbpTuple->tpl_ZoneLen = pPortDesc->pd_DesiredZone->zn_ZoneLen;
}
// Start by building the pending name structure. This needs to be linked
// to the socket holding the spin lock and getting a unique enumerator
// and an nbp id. If either of these fail, then we undo the stuff.
if (((pPendName = (PPEND_NAME)AtalkAllocZeroedMemory(sizeof(PEND_NAME))) == NULL) ||
((pRegdName = (PREGD_NAME)AtalkAllocZeroedMemory(sizeof(REGD_NAME))) == NULL))
{
if (pPendName != NULL)
{
AtalkFreeMemory(pPendName);
pPendName = NULL;
}
Status = ATALK_RESR_MEM;
break;
}
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("AtalkNbpAction: %s, Socket %lx, PendName %lx, RegdName %lx\n",
(Reason == FOR_REGISTER) ? "Register" :
((Reason == FOR_CONFIRM) ? "Confirm" : "Lookup"),
pDdpAddr, pPendName, pRegdName));
#if DBG
pRegdName->rdn_Signature = RDN_SIGNATURE;
pPendName->pdn_Signature = PDN_SIGNATURE;
#endif
pRegdName->rdn_Tuple.tpl_ObjectLen = pNbpTuple->tpl_ObjectLen;;
RtlCopyMemory(pRegdName->rdn_Tuple.tpl_Object,
pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen);
pRegdName->rdn_Tuple.tpl_TypeLen = pNbpTuple->tpl_TypeLen;;
RtlCopyMemory(pRegdName->rdn_Tuple.tpl_Type,
pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen);
pRegdName->rdn_Tuple.tpl_ZoneLen = pNbpTuple->tpl_ZoneLen;;
RtlCopyMemory(pRegdName->rdn_Tuple.tpl_Zone,
pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen);
pRegdName->rdn_Tuple.tpl_Address.ata_Address = pDdpAddr->ddpao_Addr.ata_Address;
pPendName->pdn_pRegdName = pRegdName;
INITIALIZE_SPIN_LOCK(&pPendName->pdn_Lock);
pPendName->pdn_RefCount = 3; // Reference for creation, timer & for ourselves
pPendName->pdn_pDdpAddr = pDdpAddr;
AtalkDdpReferenceByPtr(pDdpAddr, &Status);
ASSERT(ATALK_SUCCESS(Status));
pPendName->pdn_Flags = PDN_FREE_REGDNAME;
pPendName->pdn_Reason = Reason;
pPendName->pdn_pActReq = pActReq;
pPendName->pdn_RemainingBroadcasts = NBP_NUM_BROADCASTS;
AtalkTimerInitialize(&pPendName->pdn_Timer,
atalkNbpTimer,
NBP_BROADCAST_INTERVAL);
if (Reason == FOR_CONFIRM)
pPendName->pdn_ConfirmAddr = pNbpTuple->tpl_Address;
else if (Reason == FOR_LOOKUP)
{
pPendName->pdn_pAMdl = pAMdl;
pPendName->pdn_MdlLength = (USHORT)MdlLen;
pPendName->pdn_TotalTuples = 0;
pPendName->pdn_MaxTuples = MaxTuples;
// If we are not doing a wild card search, restrict
// the tuples to one so we get out early instead of
// the max. time-out since we are never going to
// fill the buffer
if (!((pNbpTuple->tpl_Object[0] == '=') ||
(pNbpTuple->tpl_Type[0] == '=') ||
(pNbpTuple->tpl_Zone[0] == '=') ||
(AtalkSearchBuf(pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen,
NBP_WILD_CHARACTER) != NULL) ||
(AtalkSearchBuf(pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen,
NBP_WILD_CHARACTER) != NULL) ||
(AtalkSearchBuf(pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
NBP_WILD_CHARACTER) != NULL)))
{
pPendName->pdn_MaxTuples = 1;
}
}
// We're going to send a directed lookup for confirms, or either a
// broadcast request or a lookup for registers or lookup depending
// on whether we know about a router or not. We do not have to bother
// checking the registered names list, for register, in our node
// because the broadcast will eventually get to us and we'll handle
// it then ! Request packet, with one tuple
if (Reason == FOR_CONFIRM) // Send to confirming node
((PNBPHDR)(pPendName->pdn_Datagram))->_CmdAndTupleCnt =
(NBP_LOOKUP << 4) + 1;
else if (pPortDesc->pd_Flags & PD_SEEN_ROUTER_RECENTLY)
((PNBPHDR)(pPendName->pdn_Datagram))->_CmdAndTupleCnt =
(NBP_BROADCAST_REQUEST << 4) + 1;
else ((PNBPHDR)(pPendName->pdn_Datagram))->_CmdAndTupleCnt =
(NBP_LOOKUP << 4) + 1;
pPendName->pdn_DatagramLength = sizeof(NBPHDR) +
atalkNbpEncodeTuple(&pPendName->pdn_pRegdName->rdn_Tuple,
NULL,
0,
// NAMESINFORMATION_SOCKET,
LAST_DYNAMIC_SOCKET,
pPendName->pdn_Datagram + sizeof(NBPHDR));
// Alloc an Nbp Id and an enumerator and link it into the list
atalkNbpLinkPendingNameInList(pDdpAddr, pPendName);
((PNBPHDR)(pPendName->pdn_Datagram))->_NbpId = (BYTE)(pPendName->pdn_NbpId);
AtalkTimerScheduleEvent(&pPendName->pdn_Timer);
atalkNbpSendRequest(pPendName);
atalkNbpDerefPendName(pPendName); // We are done now.
Status = ATALK_PENDING;
} while (FALSE);
return Status;
}
/*** AtalkNbpRemove
*
*/
ATALK_ERROR
AtalkNbpRemove(
IN PDDP_ADDROBJ pDdpAddr,
IN PNBPTUPLE pNbpTuple,
IN PACTREQ pActReq
)
{
PREGD_NAME pRegdName, *ppRegdName;
KIRQL OldIrql;
ATALK_ERROR Status = ATALK_INVALID_PARAMETER;
do
{
// Remove a registered NBP name. Zone must either be NULL or "*"
if ((pNbpTuple->tpl_ObjectLen == 0) ||
(pNbpTuple->tpl_ObjectLen > MAX_ENTITY_LENGTH) ||
(pNbpTuple->tpl_TypeLen == 0) ||
(pNbpTuple->tpl_TypeLen > MAX_ENTITY_LENGTH))
break;
if (pNbpTuple->tpl_ZoneLen == 0)
{
pNbpTuple->tpl_ZoneLen = 1;
pNbpTuple->tpl_Zone[0] = '*';
}
else
{
if ((pNbpTuple->tpl_ZoneLen != 1) ||
(pNbpTuple->tpl_Zone[0] != '*'))
break;
}
if ((pNbpTuple->tpl_Object[0] == '=') || (pNbpTuple->tpl_Type[0] == '=') ||
AtalkSearchBuf(pNbpTuple->tpl_Object, pNbpTuple->tpl_ObjectLen, NBP_WILD_CHARACTER) ||
AtalkSearchBuf(pNbpTuple->tpl_Type, pNbpTuple->tpl_TypeLen, NBP_WILD_CHARACTER))
break;
// Search in the registered names list in the open socket
// Lock down the structure first
ACQUIRE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, &OldIrql);
for (ppRegdName = &pDdpAddr->ddpao_RegNames;
(pRegdName = *ppRegdName) != NULL;
ppRegdName = &pRegdName->rdn_Next)
{
ASSERT (VALID_REGDNAME(pRegdName));
if (AtalkFixedCompareCaseInsensitive(pNbpTuple->tpl_Object,
pNbpTuple->tpl_ObjectLen,
pRegdName->rdn_Tuple.tpl_Object,
pRegdName->rdn_Tuple.tpl_ObjectLen) &&
AtalkFixedCompareCaseInsensitive(pNbpTuple->tpl_Type,
pNbpTuple->tpl_TypeLen,
pRegdName->rdn_Tuple.tpl_Type,
pRegdName->rdn_Tuple.tpl_TypeLen))
{
*ppRegdName = pRegdName->rdn_Next;
break;
}
}
RELEASE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, OldIrql);
Status = ATALK_FAILURE;
if (pRegdName != NULL)
{
AtalkFreeMemory(pRegdName);
Status = ATALK_NO_ERROR;
}
} while (FALSE);
AtalkUnlockNbpIfNecessary();
(*pActReq->ar_Completion)(Status, pActReq);
return (ATALK_PENDING);
}
/*** atalkNbpMatchWild
*
*/
LOCAL BOOLEAN
atalkNbpMatchWild(
IN PBYTE WildString,
IN BYTE WildStringLen,
IN PBYTE String,
IN BYTE StringLen
)
/*++
There are two kinds of wild card searches. An '=' by itself matches anything.
Partial matches use the 'curly equals' or 0xC5. So representing that by the
'=' character below.
foo= will match any name which starts with foo.
=foo will match any name which ends in foo.
=foo= will match any name with foo in it.
foo=bar will match any name that starts with foo and ends with bar.
--*/
{
PBYTE pTarget, pTokStr;
int TargetLen, TokStrLen;
PBYTE pWildCard, pCurStr, pCurWild;
int Len;
int i;
BOOLEAN fWildCharPresent = FALSE;
// first see if it's a 'match any' request
if ((WildString[0] == 0) ||
((WildString[0] == '=') && (WildStringLen == 1)))
return TRUE;
// now, check to see if there is any wild char in the requested name
for (i=0; i<WildStringLen; i++)
{
if (WildString[i] == NBP_WILD_CHARACTER)
{
fWildCharPresent = TRUE;
break;
}
}
// if there is no wild character in the requested name, this is
// a straight forward string compare!
if (!fWildCharPresent)
{
if (WildStringLen != StringLen)
return FALSE;
if (SubStringMatch(WildString,String,StringLen,WildStringLen))
return TRUE;
else
return FALSE;
}
// ok, now deal with the wild character mess
pTarget = String;
pTokStr = WildString;
TargetLen = StringLen;
while (WildStringLen > 0 && StringLen > 0)
{
// find length of substring until the next wild-char
TokStrLen = GetTokenLen(pTokStr,WildStringLen,NBP_WILD_CHARACTER);
if (TokStrLen > 0)
{
if (!SubStringMatch(pTarget,pTokStr,StringLen,TokStrLen))
{
return (FALSE);
}
pTokStr += TokStrLen;
WildStringLen -= (BYTE)TokStrLen;
pTarget += TokStrLen;
StringLen -= (BYTE)TokStrLen;
}
// the very first char was wild-char: skip over it
else
{
pTokStr++;
WildStringLen--;
}
}
// if we survived all the checks, this string is a match!
return (TRUE);
}
/*** atalkNbpEncodeTuple
*
*/
LOCAL SHORT
atalkNbpEncodeTuple(
IN PNBPTUPLE pNbpTuple,
IN PBYTE pZone OPTIONAL, // Override zone
IN BYTE ZoneLen OPTIONAL, // Valid only if pZone != NULL
IN BYTE Socket OPTIONAL,
OUT PBYTE pBuffer
)
{
typedef struct
{
BYTE _NetNum[2];
BYTE _Node;
BYTE _Socket;
BYTE _Enumerator;
} HDR, *PHDR;
SHORT Len = sizeof(HDR);
if (pZone == NULL)
{
pZone = pNbpTuple->tpl_Zone;
ZoneLen = pNbpTuple->tpl_ZoneLen;
}
PUTSHORT2SHORT(((PHDR)pBuffer)->_NetNum, pNbpTuple->tpl_Address.ata_Network);
((PHDR)pBuffer)->_Node = pNbpTuple->tpl_Address.ata_Node;
((PHDR)pBuffer)->_Socket = pNbpTuple->tpl_Address.ata_Socket;
if (Socket != 0)
((PHDR)pBuffer)->_Socket = Socket;
PUTSHORT2BYTE(&((PHDR)pBuffer)->_Enumerator, pNbpTuple->tpl_Enumerator);
pBuffer += sizeof(HDR);
*pBuffer++ = pNbpTuple->tpl_ObjectLen;
RtlCopyMemory(pBuffer, pNbpTuple->tpl_Object, pNbpTuple->tpl_ObjectLen);
pBuffer += pNbpTuple->tpl_ObjectLen;
Len += (pNbpTuple->tpl_ObjectLen + 1);
*pBuffer++ = pNbpTuple->tpl_TypeLen;
RtlCopyMemory(pBuffer, pNbpTuple->tpl_Type, pNbpTuple->tpl_TypeLen);
pBuffer += pNbpTuple->tpl_TypeLen;
Len += (pNbpTuple->tpl_TypeLen + 1);
*pBuffer++ = ZoneLen;
RtlCopyMemory(pBuffer, pZone, ZoneLen);
// pBuffer += ZoneLen;
Len += (ZoneLen + 1);
return (Len);
}
/*** atalkNbpDecodeTuple
*
*/
LOCAL SHORT
atalkNbpDecodeTuple(
IN PBYTE pBuffer,
IN USHORT PktLen,
OUT PNBPTUPLE pNbpTuple
)
{
typedef struct
{
BYTE _NetNum[2];
BYTE _Node;
BYTE _Socket;
BYTE _Enumerator;
} HDR, *PHDR;
SHORT Len = 0;
do
{
if (PktLen < MIN_NBP_TUPLELENGTH)
{
break;
}
GETSHORT2SHORT(&pNbpTuple->tpl_Address.ata_Network,
((PHDR)pBuffer)->_NetNum);
pNbpTuple->tpl_Address.ata_Node = ((PHDR)pBuffer)->_Node;
pNbpTuple->tpl_Address.ata_Socket = ((PHDR)pBuffer)->_Socket;
GETBYTE2SHORT(&pNbpTuple->tpl_Enumerator,
&((PHDR)pBuffer)->_Enumerator);
// Get past the header
pBuffer += sizeof(HDR);
PktLen -= sizeof(HDR);
Len = sizeof(HDR);
pNbpTuple->tpl_ObjectLen = *pBuffer++;
PktLen --;
if ((pNbpTuple->tpl_ObjectLen > PktLen) ||
(pNbpTuple->tpl_ObjectLen > MAX_ENTITY_LENGTH))
{
Len = 0;
break;
}
RtlCopyMemory(pNbpTuple->tpl_Object, pBuffer, pNbpTuple->tpl_ObjectLen);
pBuffer += pNbpTuple->tpl_ObjectLen;
PktLen -= pNbpTuple->tpl_ObjectLen;
Len += (pNbpTuple->tpl_ObjectLen + 1);
if (PktLen == 0)
{
Len = 0;
break;
}
pNbpTuple->tpl_TypeLen = *pBuffer++;
PktLen --;
if ((pNbpTuple->tpl_TypeLen > PktLen) ||
(pNbpTuple->tpl_TypeLen > MAX_ENTITY_LENGTH))
{
Len = 0;
break;
}
RtlCopyMemory(pNbpTuple->tpl_Type, pBuffer, pNbpTuple->tpl_TypeLen);
pBuffer += pNbpTuple->tpl_TypeLen;
PktLen -= pNbpTuple->tpl_TypeLen;
Len += (pNbpTuple->tpl_TypeLen + 1);
if (PktLen == 0)
{
Len = 0;
break;
}
pNbpTuple->tpl_ZoneLen = *pBuffer++;
PktLen --;
if ((pNbpTuple->tpl_ZoneLen > PktLen) ||
(pNbpTuple->tpl_ZoneLen > MAX_ENTITY_LENGTH))
{
Len = 0;
break;
}
RtlCopyMemory(pNbpTuple->tpl_Zone, pBuffer, pNbpTuple->tpl_ZoneLen);
Len += (pNbpTuple->tpl_ZoneLen + 1);
} while (FALSE);
return (Len);
}
/*** atalkNbpLinkPendingNameInList
*
*/
LOCAL VOID
atalkNbpLinkPendingNameInList(
IN PDDP_ADDROBJ pDdpAddr,
IN OUT PPEND_NAME pPendName
)
{
PATALK_NODE pNode = pDdpAddr->ddpao_Node;
KIRQL OldIrql;
ASSERT (VALID_PENDNAME(pPendName));
ACQUIRE_SPIN_LOCK(&pNode->an_Lock, &OldIrql);
// Use the next consecutive values. If there are > 256 pending names on a node, we'll
// end up re-using the ids and enums. Still ok unless all of them are of the form
// =:=@=. Well lets just keep it simple.
pPendName->pdn_NbpId = ++(pNode->an_NextNbpId);
pPendName->pdn_pRegdName->rdn_Tuple.tpl_Enumerator = ++(pNode->an_NextNbpEnum);
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpLinkPendingNameInList: Linking PendingName %lx in socket %lx\n",
pPendName, pDdpAddr));
pPendName->pdn_Next = pDdpAddr->ddpao_PendNames;
pDdpAddr->ddpao_PendNames = pPendName;
RELEASE_SPIN_LOCK(&pNode->an_Lock, OldIrql);
}
/*** AtalkNbpCloseSocket
*
*/
VOID
AtalkNbpCloseSocket(
IN PDDP_ADDROBJ pDdpAddr
)
{
PPEND_NAME pPendName, *ppPendName;
PREGD_NAME pRegdName, *ppRegdName;
KIRQL OldIrql;
ACQUIRE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, &OldIrql);
// Free the pending names from the open socket.
for (ppPendName = &pDdpAddr->ddpao_PendNames;
(pPendName = *ppPendName) != NULL;
NOTHING)
{
ASSERT (VALID_PENDNAME(pPendName));
if (pPendName->pdn_Flags & PDN_CLOSING)
{
ppPendName = &pPendName->pdn_Next;
continue;
}
pPendName->pdn_Flags |= PDN_CLOSING;
pPendName->pdn_Status = ATALK_SOCKET_CLOSED;
// Cancel outstanding timers on the pending names
if (AtalkTimerCancelEvent(&pPendName->pdn_Timer, NULL))
{
atalkNbpDerefPendName(pPendName);
}
RELEASE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, OldIrql);
ASSERT (pPendName->pdn_RefCount > 0);
atalkNbpDerefPendName(pPendName);
ACQUIRE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, &OldIrql);
ppPendName = &pDdpAddr->ddpao_PendNames;
}
// Free the registered names from the open socket.
for (ppRegdName = &pDdpAddr->ddpao_RegNames;
(pRegdName = *ppRegdName) != NULL;
NOTHING)
{
ASSERT (VALID_REGDNAME(pRegdName));
*ppRegdName = pRegdName->rdn_Next;
AtalkFreeMemory(pRegdName);
}
RELEASE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, OldIrql);
}
/*** atalkNbpSendRequest
*
*/
LOCAL BOOLEAN
atalkNbpSendRequest(
IN PPEND_NAME pPendName
)
{
PDDP_ADDROBJ pDdpAddr;
PBUFFER_DESC pBuffDesc;
ATALK_ADDR DestAddr;
ATALK_ADDR SrcAddr;
ATALK_ERROR Status;
PPORT_DESCRIPTOR pPortDesc;
SEND_COMPL_INFO SendInfo;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpSendRequest: Sending request for PendName %lx\n", pPendName));
ASSERT(!(pPendName->pdn_Flags & PDN_CLOSING));
pPortDesc = pPendName->pdn_pDdpAddr->ddpao_Node->an_Port;
DestAddr.ata_Socket = NAMESINFORMATION_SOCKET;
if (pPendName->pdn_Reason == FOR_CONFIRM)
{
DestAddr.ata_Network = pPendName->pdn_ConfirmAddr.ata_Network;
DestAddr.ata_Node = pPendName->pdn_ConfirmAddr.ata_Node;
}
else
{
if (pPortDesc->pd_Flags & PD_SEEN_ROUTER_RECENTLY)
{
DestAddr.ata_Network = pPortDesc->pd_ARouter.atn_Network;
DestAddr.ata_Node = pPortDesc->pd_ARouter.atn_Node;
}
else
{
DestAddr.ata_Network = CABLEWIDE_BROADCAST_NETWORK;
DestAddr.ata_Node = ATALK_BROADCAST_NODE;
}
}
SrcAddr.ata_Address = pPendName->pdn_pDdpAddr->ddpao_Addr.ata_Address;
// SrcAddr.ata_Socket = NAMESINFORMATION_SOCKET;
SrcAddr.ata_Socket = LAST_DYNAMIC_SOCKET;
AtalkDdpReferenceByAddr(pPendName->pdn_pDdpAddr->ddpao_Node->an_Port,
&SrcAddr,
&pDdpAddr,
&Status);
if (!ATALK_SUCCESS(Status))
{
return FALSE;
}
if ((pBuffDesc = AtalkAllocBuffDesc(pPendName->pdn_Datagram,
pPendName->pdn_DatagramLength,
BD_CHAR_BUFFER)) == NULL)
{
AtalkDdpDereference(pDdpAddr);
return FALSE;
}
ASSERT(pBuffDesc->bd_Length == pPendName->pdn_DatagramLength);
ASSERT(pBuffDesc->bd_Length > 0);
SendInfo.sc_TransmitCompletion = atalkNbpSendComplete;
SendInfo.sc_Ctx1 = pBuffDesc;
// SendInfo.sc_Ctx2 = NULL;
// SendInfo.sc_Ctx3 = NULL;
Status = AtalkDdpSend(pDdpAddr,
&DestAddr,
DDPPROTO_NBP,
FALSE,
pBuffDesc,
NULL,
0,
NULL,
&SendInfo);
if (!ATALK_SUCCESS(Status))
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("atalkNbpSendRequest: AtalkDdpSend Failed %lx\n", Status));
AtalkFreeBuffDesc(pBuffDesc);
}
AtalkDdpDereference(pDdpAddr);
return (ATALK_SUCCESS(Status));
}
/*** atalkNbpSendLookupDatagram
*
*/
LOCAL VOID
atalkNbpSendLookupDatagram(
IN PPORT_DESCRIPTOR pPortDesc,
IN PDDP_ADDROBJ pDdpAddr OPTIONAL,
IN SHORT NbpId,
IN PNBPTUPLE pNbpTuple
)
{
PBYTE Datagram = NULL;
BYTE MulticastAddr[ELAP_ADDR_LEN];
PBUFFER_DESC pBuffDesc = NULL;
BOOLEAN DerefDdp = FALSE;
ULONG Len;
ATALK_ADDR Dst, Src;
ATALK_ERROR Status;
SEND_COMPL_INFO SendInfo;
if (pDdpAddr == NULL)
{
Src.ata_Network = pPortDesc->pd_ARouter.atn_Network;
Src.ata_Node = pPortDesc->pd_ARouter.atn_Node;
Src.ata_Socket = NAMESINFORMATION_SOCKET;
AtalkDdpReferenceByAddr(pPortDesc, &Src, &pDdpAddr, &Status);
if (!ATALK_SUCCESS(Status))
{
return;
}
DerefDdp = TRUE;
}
do
{
if ((pBuffDesc = AtalkAllocBuffDesc(NULL,
sizeof(NBPHDR) + MAX_NBP_TUPLELENGTH,
BD_CHAR_BUFFER | BD_FREE_BUFFER)) == NULL)
break;
Datagram = pBuffDesc->bd_CharBuffer;
((PNBPHDR)Datagram)->_NbpId = (BYTE)NbpId;
((PNBPHDR)Datagram)->_CmdAndTupleCnt = (NBP_LOOKUP << 4) + 1;
Len = sizeof(NBPHDR) +
atalkNbpEncodeTuple(pNbpTuple,
NULL,
0,
0,
Datagram+sizeof(NBPHDR));
Dst.ata_Node = ATALK_BROADCAST_NODE;
Dst.ata_Socket = NAMESINFORMATION_SOCKET;
if (EXT_NET(pPortDesc))
{
// Send to "0000FF" at correct zone multicast address
Dst.ata_Network = CABLEWIDE_BROADCAST_NETWORK;
AtalkZipMulticastAddrForZone(pPortDesc,
pNbpTuple->tpl_Zone,
pNbpTuple->tpl_ZoneLen,
MulticastAddr);
}
else
{
// Send to "nnnnFF" as broadcast
Dst.ata_Network = pPortDesc->pd_NetworkRange.anr_FirstNetwork;
}
// Set the length in the buffer descriptor.
AtalkSetSizeOfBuffDescData(pBuffDesc, (USHORT)Len);
ASSERT(pBuffDesc->bd_Length > 0);
SendInfo.sc_TransmitCompletion = atalkNbpSendComplete;
SendInfo.sc_Ctx1 = pBuffDesc;
// SendInfo.sc_Ctx2 = NULL;
// SendInfo.sc_Ctx3 = NULL;
if (!ATALK_SUCCESS(Status = AtalkDdpSend(pDdpAddr,
&Dst,
DDPPROTO_NBP,
FALSE,
pBuffDesc,
NULL,
0,
MulticastAddr,
&SendInfo)))
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("atalkNbpSendLookupDatagram: DdpSend failed %ld\n", Status));
break;
}
Datagram = NULL;
pBuffDesc = NULL;
} while (FALSE);
if (DerefDdp)
AtalkDdpDereference(pDdpAddr);
if (pBuffDesc != NULL)
AtalkFreeBuffDesc(pBuffDesc);
}
/*** atalkNbpSendForwardRequest
*
*/
LOCAL VOID
atalkNbpSendForwardRequest(
IN PDDP_ADDROBJ pDdpAddr,
IN PRTE pRte,
IN SHORT NbpId,
IN PNBPTUPLE pNbpTuple
)
{
PBYTE Datagram = NULL;
PBUFFER_DESC pBuffDesc = NULL;
SEND_COMPL_INFO SendInfo;
ATALK_ERROR ErrorCode;
ULONG Len;
ATALK_ADDR Dst;
do
{
if ((pBuffDesc = AtalkAllocBuffDesc(NULL,
sizeof(NBPHDR) + MAX_NBP_TUPLELENGTH,
BD_CHAR_BUFFER | BD_FREE_BUFFER)) == NULL)
break;
Datagram = pBuffDesc->bd_CharBuffer;
((PNBPHDR)Datagram)->_NbpId = (BYTE)NbpId;
((PNBPHDR)Datagram)->_CmdAndTupleCnt = (NBP_FORWARD_REQUEST << 4) + 1;
Len = sizeof(NBPHDR) +
atalkNbpEncodeTuple(pNbpTuple,
NULL,
0,
0,
Datagram+sizeof(NBPHDR));
Dst.ata_Network = pRte->rte_NwRange.anr_FirstNetwork;
Dst.ata_Node = ANY_ROUTER_NODE;
Dst.ata_Socket = NAMESINFORMATION_SOCKET;
// Set the length in the buffer descriptor.
AtalkSetSizeOfBuffDescData(pBuffDesc, (USHORT)Len);
ASSERTMSG("Dest in rte 0\n", Dst.ata_Network != CABLEWIDE_BROADCAST_NETWORK);
ASSERT(pBuffDesc->bd_Length > 0);
SendInfo.sc_TransmitCompletion = atalkNbpSendComplete;
SendInfo.sc_Ctx1 = pBuffDesc;
// SendInfo.sc_Ctx2 = NULL;
// SendInfo.sc_Ctx3 = NULL;
ErrorCode = AtalkDdpSend(pDdpAddr,
&Dst,
DDPPROTO_NBP,
FALSE,
pBuffDesc,
NULL,
0,
NULL,
&SendInfo);
if (!ATALK_SUCCESS(ErrorCode))
{
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("atalkNbpSendForwardRequest: DdpSend failed %ld\n", ErrorCode));
break;
}
Datagram = NULL;
pBuffDesc = NULL;
} while (FALSE);
if (pBuffDesc != NULL)
AtalkFreeBuffDesc(pBuffDesc);
}
/*** atalkNbpDerefPendName
*
*/
VOID
atalkNbpDerefPendName(
IN PPEND_NAME pPendName
)
{
PPEND_NAME * ppPendName;
PDDP_ADDROBJ pDdpAddr = pPendName->pdn_pDdpAddr;
BOOLEAN Unlink, Found = FALSE;
KIRQL OldIrql;
ACQUIRE_SPIN_LOCK(&pPendName->pdn_Lock, &OldIrql);
Unlink = (--(pPendName->pdn_RefCount) == 0);
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpDerefPendName: New Count %d\n", pPendName->pdn_RefCount));
RELEASE_SPIN_LOCK(&pPendName->pdn_Lock, OldIrql);
if (!Unlink)
return;
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_INFO,
("atalkNbpDerefPendName: Unlinking pPendName\n"));
ACQUIRE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, &OldIrql);
for (ppPendName = &pDdpAddr->ddpao_PendNames;
*ppPendName != NULL;
ppPendName = &(*ppPendName)->pdn_Next)
{
if (*ppPendName == pPendName)
{
*ppPendName = pPendName->pdn_Next;
Found = TRUE;
break;
}
}
RELEASE_SPIN_LOCK(&pDdpAddr->ddpao_Lock, OldIrql);
if (Found)
{
AtalkDdpDereference(pDdpAddr);
}
else ASSERTMSG("atalkNbpDerefPendName: Could not find\n", 0);
AtalkUnlockNbpIfNecessary();
(*pPendName->pdn_pActReq->ar_Completion)(pPendName->pdn_Status, pPendName->pdn_pActReq);
if (pPendName->pdn_Flags & PDN_FREE_REGDNAME)
AtalkFreeMemory(pPendName->pdn_pRegdName);
AtalkFreeMemory(pPendName);
}
/*** atalkNbpSendComplete
*
*/
VOID FASTCALL
atalkNbpSendComplete(
IN NDIS_STATUS Status,
IN PSEND_COMPL_INFO pSendInfo
)
{
PBUFFER_DESC pBuffDesc = (PBUFFER_DESC)(pSendInfo->sc_Ctx1);
if (!ATALK_SUCCESS(Status))
DBGPRINT(DBG_COMP_NBP, DBG_LEVEL_ERR,
("atalkNbpSendComplete: Failed %lx, pBuffDesc %lx\n",
Status, pBuffDesc));
AtalkFreeBuffDesc(pBuffDesc);
}