2429 lines
70 KiB
C
2429 lines
70 KiB
C
/*++
|
||
|
||
Copyright (c) 1989-1993 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
cache.c
|
||
|
||
Abstract:
|
||
|
||
This module contains the name cache routines for the Netbios
|
||
module of the ISN transport.
|
||
|
||
Author:
|
||
|
||
Adam Barr (adamba) 20-December-1993
|
||
|
||
Environment:
|
||
|
||
Kernel mode
|
||
|
||
Revision History:
|
||
|
||
|
||
--*/
|
||
|
||
#include "precomp.h"
|
||
#pragma hdrstop
|
||
|
||
#ifdef ALLOC_PRAGMA
|
||
#pragma alloc_text(PAGE,CreateNetbiosCacheTable)
|
||
#endif
|
||
|
||
#ifdef RASAUTODIAL
|
||
#include <acd.h>
|
||
#include <acdapi.h>
|
||
|
||
extern BOOLEAN fAcdLoadedG;
|
||
extern ACD_DRIVER AcdDriverG;
|
||
|
||
BOOLEAN
|
||
NbiAttemptAutoDial(
|
||
IN PDEVICE pDevice,
|
||
IN PCONNECTION pConnection,
|
||
IN ULONG ulFlags,
|
||
IN ACD_CONNECT_CALLBACK pProc,
|
||
IN PREQUEST pRequest
|
||
);
|
||
|
||
VOID
|
||
NbiRetryTdiConnect(
|
||
IN BOOLEAN fSuccess,
|
||
IN PVOID *pArgs
|
||
);
|
||
#endif // RASAUTODIAL
|
||
|
||
//
|
||
// We should change to monitor add name packets better,
|
||
// so if we get an add for a different place we attempt to determine
|
||
// if it is real or bogus and update if possible.
|
||
//
|
||
|
||
|
||
NTSTATUS
|
||
CacheFindName(
|
||
IN PDEVICE Device,
|
||
IN FIND_NAME_TYPE Type,
|
||
IN PUCHAR RemoteName OPTIONAL,
|
||
OUT PNETBIOS_CACHE * CacheName
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine looks up a particular remote name in the
|
||
Netbios name cache. If it cannot find it, a find name
|
||
request is queued up.
|
||
|
||
THIS REQUEST IS CALLED WITH THE DEVICE LOCK HELD AND
|
||
RETURNS WITH IT HELD.
|
||
|
||
Arguments:
|
||
|
||
Device - The netbios device.
|
||
|
||
Type - Defines the type. The effect this has is:
|
||
FindNameConnect - On connects we will ignore an existing
|
||
cache entry if it got no response before.
|
||
FindNameNetbiosFindName - For these we ignore an existing
|
||
cache entry if it is for a group name -- this is
|
||
because the find name wants the address of every
|
||
machine, not just the network list.
|
||
FindNameOther - Normal handling is done.
|
||
|
||
RemoteName - The name to be discovered -- will be NULL if it
|
||
is the broadcast address.
|
||
|
||
CacheName - Returns the cache entry that was discovered.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
PLIST_ENTRY p;
|
||
PSLIST_ENTRY s;
|
||
PNETBIOS_CACHE FoundCacheName;
|
||
PNB_SEND_RESERVED Reserved;
|
||
PUCHAR RealRemoteName; // RemoteName or NetbiosBroadcastName
|
||
|
||
//
|
||
// First scan the netbios name cache to see if we know
|
||
// about this remote.
|
||
//
|
||
|
||
if (RemoteName) {
|
||
RealRemoteName = RemoteName;
|
||
} else {
|
||
RealRemoteName = NetbiosBroadcastName;
|
||
}
|
||
|
||
if ( FindInNetbiosCacheTable ( Device->NameCache,
|
||
RealRemoteName,
|
||
&FoundCacheName ) == STATUS_SUCCESS ) {
|
||
|
||
//
|
||
// If this is a netbios find name, we only can use unique
|
||
// names in the cache; for the group ones we need to requery
|
||
// because the cache only lists networks, not individual machines.
|
||
// For connect requests, if we find an empty cache entry we
|
||
// remove it and requery.
|
||
//
|
||
|
||
if ( FoundCacheName->Unique || (Type != FindNameNetbiosFindName) ) {
|
||
|
||
if (FoundCacheName->NetworksUsed > 0) {
|
||
|
||
*CacheName = FoundCacheName;
|
||
NB_DEBUG2 (CACHE, ("Found cache name <%.16s>\n", RemoteName ? RemoteName : "<broadcast>"));
|
||
return STATUS_SUCCESS;
|
||
|
||
} else {
|
||
|
||
if (Type != FindNameConnect) {
|
||
|
||
if (FoundCacheName->FailedOnDownWan) {
|
||
NB_DEBUG2 (CACHE, ("Found cache name, but down wan <%.16s>\n", RemoteName ? RemoteName : "<broadcast>"));
|
||
return STATUS_DEVICE_DOES_NOT_EXIST;
|
||
} else {
|
||
NB_DEBUG2 (CACHE, ("Found cache name, but no nets <%.16s>\n", RemoteName ? RemoteName : "<broadcast>"));
|
||
return STATUS_BAD_NETWORK_PATH;
|
||
}
|
||
|
||
} else {
|
||
|
||
//
|
||
// This is a connect and the current cache entry
|
||
// has zero names; delete it.
|
||
//
|
||
|
||
RemoveFromNetbiosCacheTable ( Device->NameCache, FoundCacheName );
|
||
CTEAssert (FoundCacheName->ReferenceCount == 1);
|
||
if (--FoundCacheName->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Free unneeded empty cache entry %lx\n", FoundCacheName));
|
||
NbiFreeMemory(
|
||
FoundCacheName,
|
||
sizeof(NETBIOS_CACHE) + ((FoundCacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Free due to replacement");
|
||
}
|
||
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
//
|
||
// There was no suitable cache entry for this network, first see
|
||
// if there is one pending.
|
||
//
|
||
|
||
for (p = Device->WaitingFindNames.Flink;
|
||
p != &Device->WaitingFindNames;
|
||
p = p->Flink) {
|
||
|
||
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
|
||
|
||
//
|
||
// For this purpose we ignore a packet if a route
|
||
// has been found and it was for a unique name. This
|
||
// is because the cache information has already been
|
||
// inserted for this name. Otherwise if the name has
|
||
// since been deleted from the cache, the request
|
||
// that is looking for this name will starve because
|
||
// FindNameTimeout will just destroy the packet.
|
||
//
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseUnique) {
|
||
continue;
|
||
}
|
||
|
||
if (RtlEqualMemory(
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
RealRemoteName, 16)) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Cache name already pending <%.16s>\n", RemoteName ? RemoteName : "<broadcast>"));
|
||
|
||
//
|
||
// There is already one pending. If it is for a group
|
||
// name and this is a netbios find name, we make sure
|
||
// the retry count is such that at least one more
|
||
// query will be sent, so the netbios find name
|
||
// buffer can be filled with the responses from this.
|
||
//
|
||
|
||
if ((Type == FindNameNetbiosFindName) &&
|
||
(NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseGroup) &&
|
||
(Reserved->u.SR_FN.RetryCount == Device->BroadcastCount)) {
|
||
|
||
--Reserved->u.SR_FN.RetryCount;
|
||
}
|
||
|
||
return STATUS_PENDING;
|
||
}
|
||
}
|
||
|
||
s = NbiPopSendPacket(Device, TRUE);
|
||
|
||
if (s == NULL) {
|
||
NB_DEBUG (CACHE, ("Couldn't get packet to find <%.16s>\n", RemoteName ? RemoteName : "<broadcast>"));
|
||
return STATUS_INSUFFICIENT_RESOURCES;
|
||
}
|
||
|
||
Reserved = CONTAINING_RECORD (s, NB_SEND_RESERVED, PoolLinkage);
|
||
|
||
//
|
||
// We have the packet, fill it in for this request.
|
||
//
|
||
|
||
CTEAssert (Reserved->SendInProgress == FALSE);
|
||
Reserved->SendInProgress = FALSE;
|
||
Reserved->Type = SEND_TYPE_FIND_NAME;
|
||
RtlCopyMemory (Reserved->u.SR_FN.NetbiosName, RealRemoteName, 16);
|
||
Reserved->u.SR_FN.StatusAndSentOnUpLine = FNStatusNoResponse; // SentOnUpLine is FALSE
|
||
Reserved->u.SR_FN.RetryCount = 0;
|
||
Reserved->u.SR_FN.NewCache = NULL;
|
||
Reserved->u.SR_FN.SendTime = Device->FindNameTime;
|
||
#if !defined(_PNP_POWER)
|
||
Reserved->u.SR_FN.CurrentNicId = 1;
|
||
|
||
(VOID)(*Device->Bind.QueryHandler)( // Check return code ?
|
||
IPX_QUERY_MAX_TYPE_20_NIC_ID,
|
||
(USHORT)0,
|
||
&Reserved->u.SR_FN.MaximumNicId,
|
||
sizeof(USHORT),
|
||
NULL);
|
||
|
||
if (Reserved->u.SR_FN.MaximumNicId == 0) {
|
||
Reserved->u.SR_FN.MaximumNicId = 1; // code assumes at least one
|
||
}
|
||
#endif !_PNP_POWER
|
||
NB_DEBUG2 (CACHE, ("Queued FIND_NAME %lx for <%.16s>\n",
|
||
Reserved, RemoteName ? RemoteName : "<broadcast>"));
|
||
|
||
|
||
InsertHeadList(
|
||
&Device->WaitingFindNames,
|
||
&Reserved->WaitLinkage);
|
||
|
||
++Device->FindNamePacketCount;
|
||
|
||
if (!Device->FindNameTimerActive) {
|
||
|
||
Device->FindNameTimerActive = TRUE;
|
||
NbiReferenceDevice (Device, DREF_FN_TIMER);
|
||
|
||
CTEStartTimer(
|
||
&Device->FindNameTimer,
|
||
1, // 1 ms, i.e. expire immediately
|
||
FindNameTimeout,
|
||
(PVOID)Device);
|
||
}
|
||
|
||
NbiReferenceDevice (Device, DREF_FIND_NAME);
|
||
|
||
return STATUS_PENDING;
|
||
|
||
} /* CacheFindName */
|
||
|
||
|
||
VOID
|
||
FindNameTimeout(
|
||
CTEEvent * Event,
|
||
PVOID Context
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine is called when the find name timer expires.
|
||
It is called every FIND_NAME_GRANULARITY milliseconds unless there
|
||
is nothing to do.
|
||
|
||
Arguments:
|
||
|
||
Event - The event used to queue the timer.
|
||
|
||
Context - The context, which is the device pointer.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
PDEVICE Device = (PDEVICE)Context;
|
||
PLIST_ENTRY p, q;
|
||
PNB_SEND_RESERVED Reserved;
|
||
PNDIS_PACKET Packet;
|
||
NB_CONNECTIONLESS UNALIGNED * Header;
|
||
PNETBIOS_CACHE FoundCacheName;
|
||
NDIS_STATUS NdisStatus;
|
||
#if !defined(_PNP_POWER)
|
||
static IPX_LOCAL_TARGET BroadcastTarget = { 0, { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff } };
|
||
#endif !_PNP_POWER
|
||
NB_DEFINE_LOCK_HANDLE (LockHandle)
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
|
||
|
||
++Device->FindNameTime;
|
||
|
||
if (Device->FindNamePacketCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("FindNameTimeout exiting\n"));
|
||
|
||
Device->FindNameTimerActive = FALSE;
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
NbiDereferenceDevice (Device, DREF_FN_TIMER);
|
||
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Check what is on the queue; this is set up as a
|
||
// loop but in fact it rarely does (under no
|
||
// circumstances can we send more than one packet
|
||
// each time this function executes).
|
||
//
|
||
while (TRUE) {
|
||
|
||
p = Device->WaitingFindNames.Flink;
|
||
if (p == &Device->WaitingFindNames) {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
break;
|
||
}
|
||
|
||
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
|
||
|
||
if (Reserved->SendInProgress) {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
break;
|
||
}
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseUnique) {
|
||
|
||
//
|
||
// This was a find name for a unique name which got a
|
||
// response but was not freed at the time (because
|
||
// SendInProgress was still TRUE) so we free it now.
|
||
//
|
||
|
||
(VOID)RemoveHeadList (&Device->WaitingFindNames);
|
||
ExInterlockedPushEntrySList(
|
||
&Device->SendPacketList,
|
||
&Reserved->PoolLinkage,
|
||
&NbiGlobalPoolInterlock);
|
||
--Device->FindNamePacketCount;
|
||
|
||
//
|
||
// It is OK to do this with the lock held because
|
||
// it won't be the last one (we have the RIP_TIMER ref).
|
||
//
|
||
|
||
NbiDereferenceDevice (Device, DREF_FIND_NAME);
|
||
continue;
|
||
}
|
||
|
||
if (((SHORT) (Device->FindNameTime - Reserved->u.SR_FN.SendTime)) < 0) {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
break;
|
||
}
|
||
|
||
(VOID)RemoveHeadList (&Device->WaitingFindNames);
|
||
|
||
|
||
//
|
||
// Increment the counter and see if we have sent
|
||
// all the frames we need to (we will age out
|
||
// here if we got no response for a unique query,
|
||
// or if we are doing a global name or broadcast
|
||
// search). We also kill the query right now if
|
||
// we have not found anything but down wan lines
|
||
// to send it on.
|
||
//
|
||
|
||
if ((++Reserved->u.SR_FN.RetryCount > Device->BroadcastCount) ||
|
||
((Reserved->u.SR_FN.RetryCount > 1) && (!NB_GET_SR_FN_SENT_ON_UP_LINE(Reserved)))) {
|
||
|
||
#if DBG
|
||
if (Reserved->u.SR_FN.RetryCount > Device->BroadcastCount) {
|
||
NB_DEBUG2 (CACHE, ("FindNameTimeout aging out %lx\n", Reserved));
|
||
} else {
|
||
NB_DEBUG2 (CACHE, ("FindNameTimeout no active nets %lx\n", Reserved));
|
||
}
|
||
#endif
|
||
|
||
//
|
||
// This packet is stale, clean it up and continue.
|
||
//
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseGroup) {
|
||
|
||
CTEAssert (Reserved->u.SR_FN.NewCache != NULL);
|
||
|
||
//
|
||
// If this was a group name and we have a new
|
||
// cache entry that we have been building for it,
|
||
// then insert that in the queue and use it
|
||
// to succeed any pending connects. Because
|
||
// netbios find name requests can cause cache
|
||
// requests for group names to be queued even
|
||
// if we already have on in the database, we
|
||
// first scan for old ones and remove them.
|
||
//
|
||
|
||
if ( FindInNetbiosCacheTable( Device->NameCache,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
&FoundCacheName ) == STATUS_SUCCESS ) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Found old group cache name <%.16s>\n", FoundCacheName->NetbiosName));
|
||
|
||
RemoveFromNetbiosCacheTable ( Device->NameCache, FoundCacheName );
|
||
|
||
if (--FoundCacheName->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Free replaced cache entry %lx\n", FoundCacheName));
|
||
NbiFreeMemory(
|
||
FoundCacheName,
|
||
sizeof(NETBIOS_CACHE) + ((FoundCacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Free due to replacement");
|
||
|
||
}
|
||
|
||
}
|
||
|
||
Reserved->u.SR_FN.NewCache->TimeStamp = Device->CacheTimeStamp;
|
||
|
||
InsertInNetbiosCacheTable(
|
||
Device->NameCache,
|
||
Reserved->u.SR_FN.NewCache);
|
||
|
||
//
|
||
// Reference it for the moment since CacheHandlePending
|
||
// uses it after releasing the lock. CacheHandlePending
|
||
// will dereference it.
|
||
//
|
||
|
||
++Reserved->u.SR_FN.NewCache->ReferenceCount;
|
||
|
||
//
|
||
// This call releases the locks
|
||
//
|
||
|
||
CacheHandlePending(
|
||
Device,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
NetbiosNameFound,
|
||
Reserved->u.SR_FN.NewCache
|
||
NB_LOCK_HANDLE_ARG(LockHandle));
|
||
|
||
} else {
|
||
|
||
CTEAssert (Reserved->u.SR_FN.NewCache == NULL);
|
||
|
||
//
|
||
// Allocate an empty cache entry to record the
|
||
// fact that we could not find this name, unless
|
||
// there is already an entry for this name.
|
||
//
|
||
|
||
if ( FindInNetbiosCacheTable( Device->NameCache,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
&FoundCacheName ) == STATUS_SUCCESS ) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Don't replace old group cache name with empty <%16.16s>\n", FoundCacheName->NetbiosName));
|
||
} else {
|
||
|
||
PNETBIOS_CACHE EmptyCache;
|
||
|
||
//
|
||
// Nothing found.
|
||
//
|
||
|
||
EmptyCache = NbiAllocateMemory (sizeof(NETBIOS_CACHE), MEMORY_CACHE, "Cache Entry");
|
||
if (EmptyCache != NULL) {
|
||
|
||
RtlZeroMemory (EmptyCache, sizeof(NETBIOS_CACHE));
|
||
|
||
NB_DEBUG2 (CACHE, ("Allocate new empty cache %lx for <%.16s>\n",
|
||
EmptyCache, Reserved->u.SR_FN.NetbiosName));
|
||
|
||
RtlCopyMemory (EmptyCache->NetbiosName, Reserved->u.SR_FN.NetbiosName, 16);
|
||
EmptyCache->Unique = TRUE; // so we'll delete it if we see an add name
|
||
EmptyCache->ReferenceCount = 1;
|
||
EmptyCache->NetworksAllocated = 1;
|
||
EmptyCache->TimeStamp = Device->CacheTimeStamp;
|
||
EmptyCache->NetworksUsed = 0;
|
||
EmptyCache->FailedOnDownWan = (BOOLEAN)
|
||
!NB_GET_SR_FN_SENT_ON_UP_LINE(Reserved);
|
||
|
||
InsertInNetbiosCacheTable (
|
||
Device->NameCache,
|
||
EmptyCache);
|
||
}
|
||
}
|
||
|
||
//
|
||
// Fail all datagrams, etc. that were waiting for
|
||
// this route. This call releases the lock.
|
||
//
|
||
|
||
CacheHandlePending(
|
||
Device,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
NB_GET_SR_FN_SENT_ON_UP_LINE(Reserved) ?
|
||
NetbiosNameNotFoundNormal :
|
||
NetbiosNameNotFoundWanDown,
|
||
NULL
|
||
NB_LOCK_HANDLE_ARG(LockHandle));
|
||
|
||
}
|
||
|
||
ExInterlockedPushEntrySList(
|
||
&Device->SendPacketList,
|
||
&Reserved->PoolLinkage,
|
||
&NbiGlobalPoolInterlock);
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
|
||
|
||
--Device->FindNamePacketCount;
|
||
NbiDereferenceDevice (Device, DREF_FIND_NAME);
|
||
continue;
|
||
}
|
||
|
||
|
||
|
||
//
|
||
// Send the packet out again. We first set the time so
|
||
// it won't be sent again until the appropriate timeout.
|
||
//
|
||
|
||
Reserved->u.SR_FN.SendTime = (USHORT)(Device->FindNameTime + Device->FindNameTimeout);
|
||
|
||
InsertTailList (&Device->WaitingFindNames, &Reserved->WaitLinkage);
|
||
|
||
CTEAssert (Reserved->Identifier == IDENTIFIER_NB);
|
||
CTEAssert (!Reserved->SendInProgress);
|
||
Reserved->SendInProgress = TRUE;
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
//
|
||
// If this is the first retry, we need to initialize the packet
|
||
//
|
||
if ( Reserved->u.SR_FN.RetryCount == 1 ) {
|
||
//
|
||
// Fill in the IPX header -- the default header has the broadcast
|
||
// address on net 0 as the destination IPX address, which is
|
||
// what we want.
|
||
//
|
||
|
||
Header = (NB_CONNECTIONLESS UNALIGNED *)(&Reserved->Header[Device->Bind.IncludedHeaderOffset]);
|
||
RtlCopyMemory((PVOID)&Header->IpxHeader, &Device->ConnectionlessHeader, sizeof(IPX_HEADER));
|
||
Header->IpxHeader.PacketLength[0] = (sizeof(IPX_HEADER)+sizeof(NB_NAME_FRAME)) / 256;
|
||
Header->IpxHeader.PacketLength[1] = (sizeof(IPX_HEADER)+sizeof(NB_NAME_FRAME)) % 256;
|
||
|
||
Header->IpxHeader.PacketType = (UCHAR)(Device->Internet ? 0x014 : 0x04);
|
||
|
||
//
|
||
// Now fill in the Netbios header.
|
||
//
|
||
|
||
RtlZeroMemory (Header->NameFrame.RoutingInfo, 32);
|
||
Header->NameFrame.ConnectionControlFlag = 0x00;
|
||
// Header->NameFrame.DataStreamType = NB_CMD_FIND_NAME;
|
||
Header->NameFrame.DataStreamType2 = NB_CMD_FIND_NAME;
|
||
Header->NameFrame.NameTypeFlag = 0x00;
|
||
|
||
RtlCopyMemory(
|
||
Header->NameFrame.Name,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
16);
|
||
|
||
|
||
}
|
||
//
|
||
// Now submit the packet to IPX.
|
||
//
|
||
|
||
Packet = CONTAINING_RECORD (Reserved, NDIS_PACKET, ProtocolReserved[0]);
|
||
|
||
NB_DEBUG2 (CACHE, ("FindNameTimeout sending %lx\n", Reserved));
|
||
|
||
NdisAdjustBufferLength(NB_GET_NBHDR_BUFF(Packet), sizeof(IPX_HEADER) +
|
||
sizeof(NB_NAME_FRAME));
|
||
if ((NdisStatus =
|
||
(*Device->Bind.SendHandler)(
|
||
&BroadcastTarget,
|
||
Packet,
|
||
sizeof(IPX_HEADER) + sizeof(NB_NAME_FRAME),
|
||
sizeof(IPX_HEADER) + sizeof(NB_NAME_FRAME))) != STATUS_PENDING) {
|
||
|
||
NbiSendComplete(
|
||
Packet,
|
||
NdisStatus);
|
||
|
||
}
|
||
|
||
|
||
break;
|
||
|
||
}
|
||
|
||
//
|
||
// Since we did something this time, we restart the timer.
|
||
//
|
||
|
||
CTEStartTimer(
|
||
&Device->FindNameTimer,
|
||
FIND_NAME_GRANULARITY,
|
||
FindNameTimeout,
|
||
(PVOID)Device);
|
||
|
||
} /* FindNameTimeout */
|
||
|
||
|
||
VOID
|
||
CacheHandlePending(
|
||
IN PDEVICE Device,
|
||
IN PUCHAR RemoteName,
|
||
IN NETBIOS_NAME_RESULT Result,
|
||
IN PNETBIOS_CACHE CacheName OPTIONAL
|
||
IN NB_LOCK_HANDLE_PARAM(LockHandle)
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine cleans up pending datagrams and connects
|
||
that were waiting for a route to be discovered to a
|
||
given Netbios NAME. THIS ROUTINE IS CALLED WITH
|
||
DEVICE->LOCK ACQUIRED AND RETURNS WITH IT RELEASED.
|
||
|
||
Arguments:
|
||
|
||
Device - The device.
|
||
|
||
RemoteName - The netbios name that was being searched for.
|
||
|
||
Result - Indicates if the name was found, or not found due
|
||
to no response or wan lines being down.
|
||
|
||
CacheName - If Result is NetbiosNameFound, the cache entry for this name.
|
||
This entry has been referenced and this routine will deref it.
|
||
|
||
LockHandle - The handle used to acquire the lock.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
|
||
LIST_ENTRY DatagramList;
|
||
LIST_ENTRY ConnectList;
|
||
LIST_ENTRY AdapterStatusList;
|
||
LIST_ENTRY NetbiosFindNameList;
|
||
PNB_SEND_RESERVED Reserved;
|
||
PNDIS_PACKET Packet;
|
||
PLIST_ENTRY p;
|
||
PREQUEST ConnectRequest, DatagramRequest, AdapterStatusRequest, NetbiosFindNameRequest;
|
||
PCONNECTION Connection;
|
||
PADDRESS_FILE AddressFile;
|
||
TDI_ADDRESS_NETBIOS * RemoteAddress;
|
||
CTELockHandle CancelLH;
|
||
NB_DEFINE_LOCK_HANDLE (LockHandle1)
|
||
|
||
|
||
InitializeListHead (&DatagramList);
|
||
InitializeListHead (&ConnectList);
|
||
InitializeListHead (&AdapterStatusList);
|
||
InitializeListHead (&NetbiosFindNameList);
|
||
|
||
//
|
||
// Put all connect requests on ConnectList. They will
|
||
// be continued or failed later.
|
||
//
|
||
|
||
p = Device->WaitingConnects.Flink;
|
||
|
||
while (p != &Device->WaitingConnects) {
|
||
|
||
ConnectRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(ConnectRequest);
|
||
p = p->Flink;
|
||
|
||
if (RtlEqualMemory (Connection->RemoteName, RemoteName, 16)) {
|
||
|
||
RemoveEntryList (REQUEST_LINKAGE(ConnectRequest));
|
||
InsertTailList (&ConnectList, REQUEST_LINKAGE(ConnectRequest));
|
||
|
||
Connection->SubState = CONNECTION_SUBSTATE_C_W_ACK;
|
||
}
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// Put all the datagrams on Datagram list. They will be
|
||
// sent or failed later.
|
||
//
|
||
|
||
p = Device->WaitingDatagrams.Flink;
|
||
|
||
while (p != &Device->WaitingDatagrams) {
|
||
|
||
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
|
||
|
||
p = p->Flink;
|
||
|
||
//
|
||
// Check differently based on whether we were looking for
|
||
// the broadcast address or not.
|
||
//
|
||
|
||
if (Reserved->u.SR_DG.RemoteName == (PVOID)-1) {
|
||
if (!RtlEqualMemory (RemoteName, NetbiosBroadcastName, 16)) {
|
||
continue;
|
||
}
|
||
} else {
|
||
|
||
if (!RtlEqualMemory (RemoteName, Reserved->u.SR_DG.RemoteName->NetbiosName, 16)) {
|
||
continue;
|
||
}
|
||
}
|
||
|
||
RemoveEntryList (&Reserved->WaitLinkage);
|
||
InsertTailList (&DatagramList, &Reserved->WaitLinkage);
|
||
|
||
//
|
||
// Reference this here with the lock held.
|
||
//
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
++CacheName->ReferenceCount;
|
||
}
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// Put all the adapter status requests on AdapterStatus
|
||
// list. They will be sent or failed later.
|
||
//
|
||
|
||
p = Device->WaitingAdapterStatus.Flink;
|
||
|
||
while (p != &Device->WaitingAdapterStatus) {
|
||
|
||
AdapterStatusRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
|
||
p = p->Flink;
|
||
|
||
RemoteAddress = (TDI_ADDRESS_NETBIOS *)REQUEST_INFORMATION(AdapterStatusRequest);
|
||
|
||
if (!RtlEqualMemory(
|
||
RemoteName,
|
||
RemoteAddress->NetbiosName,
|
||
16)) {
|
||
continue;
|
||
}
|
||
|
||
RemoveEntryList (REQUEST_LINKAGE(AdapterStatusRequest));
|
||
InsertTailList (&AdapterStatusList, REQUEST_LINKAGE(AdapterStatusRequest));
|
||
|
||
//
|
||
// Reference this here with the lock held.
|
||
//
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
++CacheName->ReferenceCount;
|
||
}
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// Put all the netbios find name requests on NetbiosFindName
|
||
// list. They will be completed later.
|
||
//
|
||
|
||
p = Device->WaitingNetbiosFindName.Flink;
|
||
|
||
while (p != &Device->WaitingNetbiosFindName) {
|
||
|
||
NetbiosFindNameRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
|
||
p = p->Flink;
|
||
|
||
RemoteAddress = (TDI_ADDRESS_NETBIOS *)REQUEST_INFORMATION(NetbiosFindNameRequest);
|
||
|
||
if (!RtlEqualMemory(
|
||
RemoteName,
|
||
RemoteAddress->NetbiosName,
|
||
16)) {
|
||
continue;
|
||
}
|
||
|
||
RemoveEntryList (REQUEST_LINKAGE(NetbiosFindNameRequest));
|
||
InsertTailList (&NetbiosFindNameList, REQUEST_LINKAGE(NetbiosFindNameRequest));
|
||
|
||
}
|
||
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
|
||
//
|
||
// Now that the lock is free, process all the packets on
|
||
// the various lists.
|
||
//
|
||
|
||
for (p = ConnectList.Flink; p != &ConnectList; ) {
|
||
|
||
ConnectRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
p = p->Flink;
|
||
|
||
Connection = (PCONNECTION)REQUEST_OPEN_CONTEXT(ConnectRequest);
|
||
|
||
NB_GET_CANCEL_LOCK( &CancelLH );
|
||
NB_SYNC_GET_LOCK (&Connection->Lock, &LockHandle1);
|
||
|
||
if ((Connection->State == CONNECTION_STATE_CONNECTING) &&
|
||
(Connection->SubState != CONNECTION_SUBSTATE_C_DISCONN)) {
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
|
||
NB_DEBUG2 (CONNECTION, ("Found queued connect %lx on %lx\n", ConnectRequest, Connection));
|
||
|
||
//
|
||
// Continue with the connection sequence.
|
||
//
|
||
|
||
Connection->SubState = CONNECTION_SUBSTATE_C_W_ROUTE;
|
||
}
|
||
|
||
|
||
if ((Result == NetbiosNameFound) && (!ConnectRequest->Cancel)) {
|
||
|
||
IoSetCancelRoutine (ConnectRequest, NbiCancelConnectWaitResponse);
|
||
|
||
NB_SYNC_SWAP_IRQL( CancelLH, LockHandle1 );
|
||
NB_FREE_CANCEL_LOCK ( CancelLH );
|
||
|
||
Connection->LocalTarget = CacheName->Networks[0].LocalTarget;
|
||
RtlCopyMemory(&Connection->RemoteHeader.DestinationNetwork, &CacheName->FirstResponse, 12);
|
||
NbiReferenceConnectionSync (Connection, CREF_FIND_ROUTE);
|
||
|
||
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle1);
|
||
|
||
*(UNALIGNED ULONG *)Connection->FindRouteRequest.Network = CacheName->FirstResponse.NetworkAddress;
|
||
RtlCopyMemory(Connection->FindRouteRequest.Node,CacheName->FirstResponse.NodeAddress,6);
|
||
Connection->FindRouteRequest.Identifier = IDENTIFIER_NB;
|
||
Connection->FindRouteRequest.Type = IPX_FIND_ROUTE_RIP_IF_NEEDED;
|
||
|
||
//
|
||
// When this completes, we will send the session init.
|
||
// We don't call it if the client is for network 0,
|
||
// instead just fake as if no route could be found
|
||
// and we will use the local target we got here.
|
||
//
|
||
|
||
if (CacheName->FirstResponse.NetworkAddress != 0) {
|
||
(*Device->Bind.FindRouteHandler) (&Connection->FindRouteRequest);
|
||
} else {
|
||
NbiFindRouteComplete( &Connection->FindRouteRequest, FALSE);
|
||
}
|
||
|
||
} else {
|
||
BOOLEAN bAutodialAttempt = FALSE;
|
||
|
||
if (ConnectRequest->Cancel) {
|
||
NB_DEBUG2 (CONNECTION, ("Cancelling connect %lx on %lx\n", ConnectRequest, Connection));
|
||
}
|
||
else
|
||
{
|
||
NB_DEBUG2 (CONNECTION, ("Timing out connect %lx on %lx\n", ConnectRequest, Connection));
|
||
}
|
||
|
||
ASSERT (Connection->ConnectRequest == ConnectRequest);
|
||
|
||
#ifdef RASAUTODIAL
|
||
if (fAcdLoadedG) {
|
||
CTELockHandle adirql;
|
||
BOOLEAN fEnabled;
|
||
|
||
//
|
||
// See if the automatic connection driver knows
|
||
// about this address before we search the
|
||
// network. If it does, we return STATUS_PENDING,
|
||
// and we will come back here via NbfRetryTdiConnect().
|
||
//
|
||
CTEGetLock(&AcdDriverG.SpinLock, &adirql);
|
||
fEnabled = AcdDriverG.fEnabled;
|
||
CTEFreeLock(&AcdDriverG.SpinLock, adirql);
|
||
if (fEnabled && NbiAttemptAutoDial(
|
||
Device,
|
||
Connection,
|
||
0,
|
||
NbiRetryTdiConnect,
|
||
ConnectRequest))
|
||
{
|
||
NB_SYNC_FREE_LOCK(&Connection->Lock, LockHandle1);
|
||
NB_FREE_CANCEL_LOCK(CancelLH);
|
||
|
||
bAutodialAttempt = TRUE;
|
||
}
|
||
}
|
||
#endif // RASAUTODIAL
|
||
|
||
if (!bAutodialAttempt) {
|
||
Connection->ConnectRequest = NULL;
|
||
Connection->SubState = CONNECTION_SUBSTATE_C_DISCONN;
|
||
|
||
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle1);
|
||
|
||
IoSetCancelRoutine( ConnectRequest, (PDRIVER_CANCEL)NULL );
|
||
NB_FREE_CANCEL_LOCK( CancelLH );
|
||
|
||
REQUEST_STATUS(ConnectRequest) = STATUS_BAD_NETWORK_PATH;
|
||
|
||
NbiCompleteRequest(ConnectRequest);
|
||
NbiFreeRequest (Device, ConnectRequest);
|
||
}
|
||
|
||
NbiDereferenceConnection (Connection, CREF_CONNECT);
|
||
|
||
}
|
||
|
||
} else {
|
||
|
||
CTEAssert (0); // What happens to the IRP? Who completes it?
|
||
|
||
NB_SYNC_FREE_LOCK (&Connection->Lock, LockHandle1);
|
||
NB_FREE_CANCEL_LOCK( CancelLH );
|
||
|
||
}
|
||
|
||
NbiDereferenceConnection (Connection, CREF_WAIT_CACHE);
|
||
|
||
}
|
||
|
||
|
||
for (p = DatagramList.Flink; p != &DatagramList; ) {
|
||
|
||
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
|
||
p = p->Flink;
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
|
||
NB_DEBUG2 (DATAGRAM, ("Found queued datagram %lx on %lx\n", Reserved->u.SR_DG.DatagramRequest, Reserved->u.SR_DG.AddressFile));
|
||
|
||
Reserved->u.SR_DG.Cache = CacheName;
|
||
Reserved->u.SR_DG.CurrentNetwork = 0;
|
||
|
||
//
|
||
// CacheName was referenced above.
|
||
//
|
||
|
||
Packet = CONTAINING_RECORD (Reserved, NDIS_PACKET, ProtocolReserved[0]);
|
||
if ( REQUEST_NDIS_BUFFER( Reserved->u.SR_DG.DatagramRequest )) {
|
||
NdisChainBufferAtBack (Packet, REQUEST_NDIS_BUFFER(Reserved->u.SR_DG.DatagramRequest));
|
||
}
|
||
|
||
NbiTransmitDatagram (Reserved);
|
||
|
||
} else {
|
||
|
||
//
|
||
// Should we send it once as a broadcast
|
||
// on net 0, just in case??
|
||
//
|
||
|
||
AddressFile = Reserved->u.SR_DG.AddressFile;
|
||
DatagramRequest = Reserved->u.SR_DG.DatagramRequest;
|
||
|
||
NB_DEBUG2 (DATAGRAM, ("Timing out datagram %lx on %lx\n", DatagramRequest, AddressFile));
|
||
|
||
//
|
||
// If the failure was due to a down wan line indicate
|
||
// that, otherwise return success (so the browser won't
|
||
// confuse this with a down wan line).
|
||
//
|
||
|
||
if (Result == NetbiosNameNotFoundWanDown) {
|
||
REQUEST_STATUS(DatagramRequest) = STATUS_DEVICE_DOES_NOT_EXIST;
|
||
} else {
|
||
REQUEST_STATUS(DatagramRequest) = STATUS_BAD_NETWORK_PATH;
|
||
}
|
||
REQUEST_INFORMATION(DatagramRequest) = 0;
|
||
|
||
NbiCompleteRequest(DatagramRequest);
|
||
NbiFreeRequest (Device, DatagramRequest);
|
||
|
||
NbiDereferenceAddressFile (AddressFile, AFREF_SEND_DGRAM);
|
||
|
||
ExInterlockedPushEntrySList(
|
||
&Device->SendPacketList,
|
||
&Reserved->PoolLinkage,
|
||
&NbiGlobalPoolInterlock);
|
||
}
|
||
|
||
}
|
||
|
||
|
||
for (p = AdapterStatusList.Flink; p != &AdapterStatusList; ) {
|
||
|
||
AdapterStatusRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
p = p->Flink;
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
|
||
NB_DEBUG2 (QUERY, ("Found queued AdapterStatus %lx\n", AdapterStatusRequest));
|
||
|
||
//
|
||
// Continue with the AdapterStatus sequence. We put
|
||
// it in ActiveAdapterStatus, it will either get
|
||
// completed when a response is received or timed
|
||
// out by the long timeout.
|
||
//
|
||
|
||
REQUEST_STATUSPTR(AdapterStatusRequest) = (PVOID)CacheName;
|
||
|
||
//
|
||
// CacheName was referenced above.
|
||
//
|
||
|
||
REQUEST_INFORMATION (AdapterStatusRequest) = 0;
|
||
|
||
NB_INSERT_TAIL_LIST(
|
||
&Device->ActiveAdapterStatus,
|
||
REQUEST_LINKAGE (AdapterStatusRequest),
|
||
&Device->Lock);
|
||
|
||
NbiSendStatusQuery (AdapterStatusRequest);
|
||
|
||
} else {
|
||
|
||
NB_DEBUG2 (QUERY, ("Timing out AdapterStatus %lx\n", AdapterStatusRequest));
|
||
|
||
REQUEST_STATUS(AdapterStatusRequest) = STATUS_IO_TIMEOUT;
|
||
|
||
NbiCompleteRequest(AdapterStatusRequest);
|
||
NbiFreeRequest (Device, AdapterStatusRequest);
|
||
|
||
NbiDereferenceDevice (Device, DREF_STATUS_QUERY);
|
||
|
||
}
|
||
|
||
}
|
||
|
||
|
||
for (p = NetbiosFindNameList.Flink; p != &NetbiosFindNameList; ) {
|
||
|
||
NetbiosFindNameRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
p = p->Flink;
|
||
|
||
//
|
||
// In fact there is not much difference between success or
|
||
// failure, since in the successful case the information
|
||
// will already have been written to the buffer. Just
|
||
// complete the request with the appropriate status,
|
||
// which will already be stored in the request.
|
||
//
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
|
||
if (CacheName->Unique) {
|
||
|
||
NB_DEBUG2 (QUERY, ("Found queued unique NetbiosFindName %lx\n", NetbiosFindNameRequest));
|
||
|
||
} else {
|
||
|
||
NB_DEBUG2 (QUERY, ("Found queued group NetbiosFindName %lx\n", NetbiosFindNameRequest));
|
||
|
||
}
|
||
|
||
} else {
|
||
|
||
CTEAssert (REQUEST_STATUS(NetbiosFindNameRequest) == STATUS_IO_TIMEOUT);
|
||
NB_DEBUG2 (QUERY, ("Timed out NetbiosFindName %lx\n", NetbiosFindNameRequest));
|
||
|
||
}
|
||
|
||
//
|
||
// This sets REQUEST_INFORMATION(Request) to the correct value.
|
||
//
|
||
|
||
NbiSetNetbiosFindNameInformation (NetbiosFindNameRequest);
|
||
|
||
NbiCompleteRequest(NetbiosFindNameRequest);
|
||
NbiFreeRequest (Device, NetbiosFindNameRequest);
|
||
|
||
NbiDereferenceDevice (Device, DREF_NB_FIND_NAME);
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// We referenced this temporarily so we could use it in here,
|
||
// deref and check if we need to delete it.
|
||
//
|
||
|
||
if (Result == NetbiosNameFound) {
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle1);
|
||
|
||
if (--CacheName->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Free newly allocated cache entry %lx\n", CacheName));
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Free in CacheHandlePending");
|
||
|
||
}
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle1);
|
||
|
||
}
|
||
|
||
} /* CacheHandlePending */
|
||
|
||
|
||
VOID
|
||
NbiProcessNameRecognized(
|
||
IN PIPX_LOCAL_TARGET RemoteAddress,
|
||
IN ULONG MacOptions,
|
||
IN PUCHAR PacketBuffer,
|
||
IN UINT PacketSize
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine handles NB_CMD_NAME_RECOGNIZED frames.
|
||
|
||
Arguments:
|
||
|
||
RemoteAddress - The local target this packet was received from.
|
||
|
||
MacOptions - The MAC options for the underlying NDIS binding.
|
||
|
||
LookaheadBuffer - The packet data, starting at the IPX
|
||
header.
|
||
|
||
PacketSize - The total length of the packet, starting at the
|
||
IPX header.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
PLIST_ENTRY p;
|
||
PDEVICE Device = NbiDevice;
|
||
PNETBIOS_CACHE NameCache;
|
||
PREQUEST NetbiosFindNameRequest;
|
||
PNB_SEND_RESERVED Reserved;
|
||
TDI_ADDRESS_NETBIOS * RemoteNetbiosAddress;
|
||
NB_CONNECTIONLESS UNALIGNED * Connectionless =
|
||
(NB_CONNECTIONLESS UNALIGNED *)PacketBuffer;
|
||
NB_DEFINE_LOCK_HANDLE(LockHandle)
|
||
|
||
|
||
#if 0
|
||
//
|
||
// We should handle responses from network 0
|
||
// differently -- if they are for a group name, we should
|
||
// keep them around but only until we get a non-zero
|
||
// response from the same card.
|
||
//
|
||
|
||
if (*(UNALIGNED ULONG *)(Connectionless->IpxHeader.SourceNetwork) == 0) {
|
||
return;
|
||
}
|
||
#endif
|
||
|
||
|
||
//
|
||
// We need to scan our queue of pending find name packets
|
||
// to see if someone is waiting for this name.
|
||
//
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
|
||
|
||
for (p = Device->WaitingFindNames.Flink;
|
||
p != &Device->WaitingFindNames;
|
||
p = p->Flink) {
|
||
|
||
Reserved = CONTAINING_RECORD (p, NB_SEND_RESERVED, WaitLinkage);
|
||
|
||
//
|
||
// Find names which have already found unique names are
|
||
// "dead", waiting for FindNameTimeout to remove them,
|
||
// and should be ignored when scanning the list.
|
||
//
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseUnique) {
|
||
|
||
continue;
|
||
}
|
||
|
||
if (RtlEqualMemory (Reserved->u.SR_FN.NetbiosName, Connectionless->NameFrame.Name, 16)) {
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (p == &Device->WaitingFindNames)
|
||
{
|
||
if ((FindInNetbiosCacheTable (Device->NameCache,
|
||
Connectionless->NameFrame.Name,
|
||
&NameCache ) == STATUS_SUCCESS) &&
|
||
(NameCache->NetworksUsed == 0))
|
||
{
|
||
//
|
||
// Update our information about this network if needed.
|
||
//
|
||
NameCache->Unique = (BOOLEAN)((Connectionless->NameFrame.NameTypeFlag & NB_NAME_GROUP) == 0);
|
||
if (RtlEqualMemory (Connectionless->NameFrame.Name, NetbiosBroadcastName, 16))
|
||
{
|
||
NameCache->Unique = FALSE;
|
||
}
|
||
|
||
RtlCopyMemory (&NameCache->FirstResponse, Connectionless->IpxHeader.SourceNetwork, 12);
|
||
NameCache->NetworksUsed = 1;
|
||
NameCache->Networks[0].Network = *(UNALIGNED ULONG*)(Connectionless->IpxHeader.SourceNetwork);
|
||
|
||
//
|
||
// If this packet was not routed to us and is for a group name,
|
||
// rather than use whatever local target it happened to come
|
||
// from we set it up so that it is broadcast on that net.
|
||
//
|
||
|
||
if ((RtlEqualMemory (RemoteAddress->MacAddress, Connectionless->IpxHeader.SourceNode, 6)) &&
|
||
(!NameCache->Unique))
|
||
{
|
||
NameCache->Networks[0].LocalTarget.NicHandle = RemoteAddress->NicHandle;
|
||
RtlCopyMemory (NameCache->Networks[0].LocalTarget.MacAddress, BroadcastAddress, 6);
|
||
RtlCopyMemory (NameCache->FirstResponse.NodeAddress, BroadcastAddress, 6);
|
||
}
|
||
else
|
||
{
|
||
NameCache->Networks[0].LocalTarget = *RemoteAddress;
|
||
}
|
||
}
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Scan for any netbios find name requests on the queue, and
|
||
// inform them about this remote. We need to do this on every
|
||
// response because group names need every computer recorded,
|
||
// but the normal cache only includes one entry per network.
|
||
//
|
||
|
||
for (p = Device->WaitingNetbiosFindName.Flink;
|
||
p != &Device->WaitingNetbiosFindName;
|
||
p = p->Flink) {
|
||
|
||
NetbiosFindNameRequest = LIST_ENTRY_TO_REQUEST(p);
|
||
|
||
RemoteNetbiosAddress = (TDI_ADDRESS_NETBIOS *)REQUEST_INFORMATION(NetbiosFindNameRequest);
|
||
|
||
if (!RtlEqualMemory(
|
||
Connectionless->NameFrame.Name,
|
||
RemoteNetbiosAddress->NetbiosName,
|
||
16)) {
|
||
continue;
|
||
}
|
||
|
||
//
|
||
// This will update the request status if needed.
|
||
//
|
||
|
||
NbiUpdateNetbiosFindName(
|
||
NetbiosFindNameRequest,
|
||
#if defined(_PNP_POWER)
|
||
&RemoteAddress->NicHandle,
|
||
#else
|
||
RemoteAddress->NicId,
|
||
#endif _PNP_POWER
|
||
(TDI_ADDRESS_IPX UNALIGNED *)Connectionless->IpxHeader.SourceNetwork,
|
||
(BOOLEAN)((Connectionless->NameFrame.NameTypeFlag & NB_NAME_GROUP) == 0));
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// See what is up with this pending find name packet.
|
||
//
|
||
|
||
if (Reserved->u.SR_FN.NewCache == NULL) {
|
||
//
|
||
// This is the first response we have received, so we
|
||
// allocate the initial entry with room for a single
|
||
// entry.
|
||
//
|
||
|
||
NameCache = NbiAllocateMemory (sizeof(NETBIOS_CACHE), MEMORY_CACHE, "Cache Entry");
|
||
if (NameCache == NULL) {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
return;
|
||
}
|
||
|
||
NB_DEBUG2 (CACHE, ("Alloc new cache %lx for <%.16s>, net %lx\n",
|
||
NameCache, Reserved->u.SR_FN.NetbiosName,
|
||
*(UNALIGNED ULONG *)(Connectionless->IpxHeader.SourceNetwork)));
|
||
|
||
RtlCopyMemory (NameCache->NetbiosName, Connectionless->NameFrame.Name, 16);
|
||
NameCache->Unique = (BOOLEAN)((Connectionless->NameFrame.NameTypeFlag & NB_NAME_GROUP) == 0);
|
||
NameCache->ReferenceCount = 1;
|
||
RtlCopyMemory (&NameCache->FirstResponse, Connectionless->IpxHeader.SourceNetwork, 12);
|
||
NameCache->NetworksAllocated = 1;
|
||
NameCache->NetworksUsed = 1;
|
||
NameCache->Networks[0].Network = *(UNALIGNED ULONG *)(Connectionless->IpxHeader.SourceNetwork);
|
||
|
||
if (RtlEqualMemory (Connectionless->NameFrame.Name, NetbiosBroadcastName, 16)) {
|
||
|
||
NB_SET_SR_FN_STATUS (Reserved, FNStatusResponseGroup);
|
||
NameCache->Unique = FALSE;
|
||
|
||
} else {
|
||
|
||
NB_SET_SR_FN_STATUS(
|
||
Reserved,
|
||
NameCache->Unique ? FNStatusResponseUnique : FNStatusResponseGroup);
|
||
|
||
}
|
||
|
||
Reserved->u.SR_FN.NewCache = NameCache;
|
||
|
||
//
|
||
// If this packet was not routed to us and is for a group name,
|
||
// rather than use whatever local target it happened to come
|
||
// from we set it up so that it is broadcast on that net.
|
||
//
|
||
|
||
if ((RtlEqualMemory (RemoteAddress->MacAddress, Connectionless->IpxHeader.SourceNode, 6)) &&
|
||
(NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseGroup)) {
|
||
#if defined(_PNP_POWER)
|
||
NameCache->Networks[0].LocalTarget.NicHandle = RemoteAddress->NicHandle;
|
||
#else
|
||
NameCache->Networks[0].LocalTarget.NicId = RemoteAddress->NicId;
|
||
#endif _PNP_POWER
|
||
RtlCopyMemory (NameCache->Networks[0].LocalTarget.MacAddress, BroadcastAddress, 6);
|
||
RtlCopyMemory (NameCache->FirstResponse.NodeAddress, BroadcastAddress, 6);
|
||
} else {
|
||
NameCache->Networks[0].LocalTarget = *RemoteAddress;
|
||
}
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseUnique) {
|
||
|
||
//
|
||
// Complete pending requests now, since it is a unique
|
||
// name we have all the information we will get.
|
||
//
|
||
|
||
NameCache->TimeStamp = Device->CacheTimeStamp;
|
||
|
||
InsertInNetbiosCacheTable(
|
||
Device->NameCache,
|
||
NameCache);
|
||
|
||
//
|
||
// Reference it since CacheHandlePending uses it
|
||
// with the lock released. CacheHandlePending
|
||
// will dereference it.
|
||
//
|
||
|
||
++NameCache->ReferenceCount;
|
||
|
||
//
|
||
// This call releases the lock.
|
||
//
|
||
|
||
CacheHandlePending(
|
||
Device,
|
||
Reserved->u.SR_FN.NetbiosName,
|
||
NetbiosNameFound,
|
||
NameCache
|
||
NB_LOCK_HANDLE_ARG(LockHandle));
|
||
|
||
} else {
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
}
|
||
|
||
} else {
|
||
|
||
//
|
||
// We already have a response to this frame.
|
||
//
|
||
|
||
if (NB_GET_SR_FN_STATUS(Reserved) == FNStatusResponseUnique) {
|
||
|
||
//
|
||
// Should we check that the response is also
|
||
// unique? Not much to do since I don't know of an
|
||
// equivalent to the netbeui NAME_IN_CONFLICT.
|
||
//
|
||
|
||
} else {
|
||
|
||
//
|
||
// This is a group name.
|
||
//
|
||
|
||
if (Connectionless->NameFrame.NameTypeFlag & NB_NAME_GROUP) {
|
||
|
||
//
|
||
// Update our information about this network if needed.
|
||
// This may free the existing cache and allocate a new one.
|
||
//
|
||
|
||
Reserved->u.SR_FN.NewCache =
|
||
CacheUpdateNameCache(
|
||
Reserved->u.SR_FN.NewCache,
|
||
RemoteAddress,
|
||
(TDI_ADDRESS_IPX UNALIGNED *)
|
||
Connectionless->IpxHeader.SourceNetwork,
|
||
FALSE);
|
||
|
||
} else {
|
||
|
||
//
|
||
// Hmmm... This respondent thinks it is a unique name
|
||
// but we think it is group, should we do something?
|
||
//
|
||
|
||
}
|
||
}
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
}
|
||
|
||
} /* NbiProcessNameRecognized */
|
||
|
||
|
||
PNETBIOS_CACHE
|
||
CacheUpdateNameCache(
|
||
IN PNETBIOS_CACHE NameCache,
|
||
IN PIPX_LOCAL_TARGET RemoteAddress,
|
||
IN TDI_ADDRESS_IPX UNALIGNED * SourceAddress,
|
||
IN BOOLEAN ModifyQueue
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine is called to update a netbios cache entry
|
||
with a new network, if it is does not already contain
|
||
information about the network. It is called when a frame
|
||
is received advertising the appropriate cache entry, which
|
||
is either a group name or the broadcast name.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH IT HELD.
|
||
|
||
Arguments:
|
||
|
||
NameCache - The name cache entry to update.
|
||
|
||
RemoteAddress - The remote address on which a frame was received.
|
||
|
||
IpxAddress - The source IPX address of the frame.
|
||
|
||
ModifyQueue - TRUE if we should update the queue which this
|
||
cache entry is in, if we reallocate it.
|
||
|
||
Return Value:
|
||
|
||
The netbios cache entry, either the original or a reallocated one.
|
||
|
||
--*/
|
||
|
||
{
|
||
|
||
PDEVICE Device = NbiDevice;
|
||
USHORT NewNetworks;
|
||
PNETBIOS_CACHE NewNameCache;
|
||
PLIST_ENTRY OldPrevious;
|
||
UINT i;
|
||
|
||
//
|
||
// See if we already know about this network.
|
||
//
|
||
|
||
for (i = 0; i < NameCache->NetworksUsed; i++) {
|
||
if (NameCache->Networks[i].Network == SourceAddress->NetworkAddress) {
|
||
return NameCache;
|
||
}
|
||
}
|
||
|
||
//
|
||
// We need to add information about this network
|
||
// to the name cache entry. If we have to allocate
|
||
// a new one we do that.
|
||
//
|
||
|
||
NB_DEBUG2 (CACHE, ("Got new net %lx for <%.16s>\n",
|
||
SourceAddress->NetworkAddress,
|
||
NameCache->NetbiosName));
|
||
|
||
if (NameCache->NetworksUsed == NameCache->NetworksAllocated) {
|
||
|
||
//
|
||
// We double the number of entries allocated until
|
||
// we hit 16, then add 8 at a time.
|
||
//
|
||
|
||
if (NameCache->NetworksAllocated < 16) {
|
||
NewNetworks = NameCache->NetworksAllocated * 2;
|
||
} else {
|
||
NewNetworks = NameCache->NetworksAllocated + 8;
|
||
}
|
||
|
||
|
||
NewNameCache = NbiAllocateMemory(
|
||
sizeof(NETBIOS_CACHE) + ((NewNetworks-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Enlarge cache entry");
|
||
|
||
if (NewNameCache == NULL) {
|
||
return NameCache;
|
||
}
|
||
|
||
NB_DEBUG2 (CACHE, ("Expand cache %lx to %lx for <%.16s>\n",
|
||
NameCache, NewNameCache, NameCache->NetbiosName));
|
||
|
||
//
|
||
// Copy the new current data to the new one.
|
||
//
|
||
|
||
RtlCopyMemory(
|
||
NewNameCache,
|
||
NameCache,
|
||
sizeof(NETBIOS_CACHE) + ((NameCache->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)));
|
||
|
||
NewNameCache->NetworksAllocated = NewNetworks;
|
||
NewNameCache->ReferenceCount = 1;
|
||
|
||
if (ModifyQueue) {
|
||
|
||
//
|
||
// Insert at the same place as the old one. The time
|
||
// stamp is the same as the old one.
|
||
//
|
||
|
||
|
||
ReinsertInNetbiosCacheTable( Device->NameCache, NameCache, NewNameCache );
|
||
|
||
}
|
||
|
||
if (--NameCache->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Free replaced cache entry %lx\n", NameCache));
|
||
NbiFreeMemory(
|
||
NameCache,
|
||
sizeof(NETBIOS_CACHE) + ((NameCache->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Enlarge existing");
|
||
|
||
}
|
||
|
||
NameCache = NewNameCache;
|
||
|
||
}
|
||
|
||
NameCache->Networks[NameCache->NetworksUsed].Network =
|
||
SourceAddress->NetworkAddress;
|
||
|
||
//
|
||
// If this packet was not routed to us, then store the local
|
||
// target for a correct broadcast.
|
||
//
|
||
|
||
if (RtlEqualMemory (RemoteAddress->MacAddress, SourceAddress->NodeAddress, 6)) {
|
||
#if defined(_PNP_POWER)
|
||
NameCache->Networks[NameCache->NetworksUsed].LocalTarget.NicHandle = RemoteAddress->NicHandle;
|
||
#else
|
||
NameCache->Networks[NameCache->NetworksUsed].LocalTarget.NicId = RemoteAddress->NicId;
|
||
#endif _PNP_POWER
|
||
RtlCopyMemory (NameCache->Networks[NameCache->NetworksUsed].LocalTarget.MacAddress, BroadcastAddress, 6);
|
||
} else {
|
||
NameCache->Networks[NameCache->NetworksUsed].LocalTarget = *RemoteAddress;
|
||
}
|
||
|
||
++NameCache->NetworksUsed;
|
||
return NameCache;
|
||
|
||
} /* CacheUpdateNameCache */
|
||
|
||
|
||
VOID
|
||
CacheUpdateFromAddName(
|
||
IN PIPX_LOCAL_TARGET RemoteAddress,
|
||
IN NB_CONNECTIONLESS UNALIGNED * Connectionless,
|
||
IN BOOLEAN LocalFrame
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine is called when an add name frame is received.
|
||
If it is for a group name it checks if our cache entry for
|
||
that group name needs to be updated to include a new network;
|
||
for all frames it checks if our broadcast cache entry needs
|
||
to be updated to include a new network.
|
||
|
||
Arguments:
|
||
|
||
RemoteAddress - The address the frame was received from.
|
||
|
||
Connectionless - The header of the received add name.
|
||
|
||
LocalFrame - TRUE if the frame was sent locally.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
PUCHAR NetbiosName;
|
||
PNETBIOS_CACHE NameCache;
|
||
PLIST_ENTRY p;
|
||
PDEVICE Device = NbiDevice;
|
||
NB_DEFINE_LOCK_HANDLE (LockHandle)
|
||
|
||
|
||
NetbiosName = (PUCHAR)Connectionless->NameFrame.Name;
|
||
|
||
//
|
||
// First look up the broadcast name.
|
||
//
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
|
||
|
||
if (!LocalFrame) {
|
||
|
||
if ( FindInNetbiosCacheTable( Device->NameCache,
|
||
NetbiosBroadcastName,
|
||
&NameCache ) == STATUS_SUCCESS ) {
|
||
//
|
||
// This will reallocate a cache entry and update the
|
||
// queue if necessary.
|
||
//
|
||
|
||
(VOID)CacheUpdateNameCache(
|
||
NameCache,
|
||
RemoteAddress,
|
||
(TDI_ADDRESS_IPX UNALIGNED *)(Connectionless->IpxHeader.SourceNetwork),
|
||
TRUE);
|
||
}
|
||
|
||
}
|
||
|
||
|
||
//
|
||
// Now see if our database needs to be updated based on this.
|
||
//
|
||
|
||
if ( FindInNetbiosCacheTable( Device->NameCache,
|
||
Connectionless->NameFrame.Name,
|
||
&NameCache ) == STATUS_SUCCESS ) {
|
||
|
||
|
||
if (!NameCache->Unique) {
|
||
|
||
if (!LocalFrame) {
|
||
|
||
//
|
||
// This will reallocate a cache entry and update the
|
||
// queue if necessary.
|
||
//
|
||
|
||
(VOID)CacheUpdateNameCache(
|
||
NameCache,
|
||
RemoteAddress,
|
||
(TDI_ADDRESS_IPX UNALIGNED *)(Connectionless->IpxHeader.SourceNetwork),
|
||
TRUE);
|
||
|
||
}
|
||
|
||
} else {
|
||
|
||
//
|
||
// To be safe, delete any unique names we get add
|
||
// names for (we will requery next time we need it).
|
||
//
|
||
|
||
RemoveFromNetbiosCacheTable ( Device->NameCache, NameCache );
|
||
|
||
if (--NameCache->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Free add named cache entry %lx\n", NameCache));
|
||
NbiFreeMemory(
|
||
NameCache,
|
||
sizeof(NETBIOS_CACHE) + ((NameCache->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Enlarge existing");
|
||
|
||
}
|
||
|
||
}
|
||
|
||
}
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
} /* CacheUpdateFromAddName */
|
||
|
||
|
||
VOID
|
||
NbiProcessDeleteName(
|
||
IN PIPX_LOCAL_TARGET RemoteAddress,
|
||
IN ULONG MacOptions,
|
||
IN PUCHAR PacketBuffer,
|
||
IN UINT PacketSize
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine handles NB_CMD_DELETE_NAME frames.
|
||
|
||
Arguments:
|
||
|
||
RemoteAddress - The local target this packet was received from.
|
||
|
||
MacOptions - The MAC options for the underlying NDIS binding.
|
||
|
||
LookaheadBuffer - The packet data, starting at the IPX
|
||
header.
|
||
|
||
PacketSize - The total length of the packet, starting at the
|
||
IPX header.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
NB_CONNECTIONLESS UNALIGNED * Connectionless =
|
||
(NB_CONNECTIONLESS UNALIGNED *)PacketBuffer;
|
||
PUCHAR NetbiosName;
|
||
PNETBIOS_CACHE CacheName;
|
||
PDEVICE Device = NbiDevice;
|
||
NB_DEFINE_LOCK_HANDLE (LockHandle)
|
||
|
||
|
||
if (PacketSize != sizeof(IPX_HEADER) + sizeof(NB_NAME_FRAME)) {
|
||
return;
|
||
}
|
||
|
||
//
|
||
// We want to update our netbios cache to reflect the
|
||
// fact that this name is no longer valid.
|
||
//
|
||
|
||
NetbiosName = (PUCHAR)Connectionless->NameFrame.Name;
|
||
|
||
NB_SYNC_GET_LOCK (&Device->Lock, &LockHandle);
|
||
|
||
if ( FindInNetbiosCacheTable( Device->NameCache,
|
||
NetbiosName,
|
||
&CacheName ) == STATUS_SUCCESS ) {
|
||
|
||
//
|
||
// We don't track group names since we don't know if
|
||
// this is the last person that owns it. We also drop
|
||
// the frame if does not come from the person we think
|
||
// owns this name.
|
||
//
|
||
|
||
if ((!CacheName->Unique) ||
|
||
(CacheName->NetworksUsed == 0) ||
|
||
(!RtlEqualMemory (&CacheName->FirstResponse, Connectionless->IpxHeader.SourceNetwork, 12))) {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
return;
|
||
}
|
||
|
||
NB_DEBUG2 (CACHE, ("Found cache name to delete <%.16s>\n", NetbiosName));
|
||
|
||
}else {
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
return;
|
||
}
|
||
|
||
|
||
//
|
||
// We have a cache entry, take it out of the list. If no
|
||
// one else is using it, delete it; if not, they will delete
|
||
// it when they are done.
|
||
//
|
||
|
||
|
||
RemoveFromNetbiosCacheTable ( Device->NameCache, CacheName);
|
||
|
||
if (--CacheName->ReferenceCount == 0) {
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
NB_DEBUG2 (CACHE, ("Free delete name cache entry %lx\n", CacheName));
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Name deleted");
|
||
|
||
} else {
|
||
|
||
NB_SYNC_FREE_LOCK (&Device->Lock, LockHandle);
|
||
|
||
}
|
||
|
||
} /* NbiProcessDeleteName */
|
||
|
||
VOID
|
||
InsertInNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN PNETBIOS_CACHE CacheEntry
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine inserts a new cache entry in the hash table
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
CacheEntry - Entry to be inserted.
|
||
|
||
Return Value:
|
||
|
||
None
|
||
|
||
--*/
|
||
|
||
{
|
||
USHORT HashIndex;
|
||
|
||
//
|
||
// Keep a threshold of how many entries do we keep in the table.
|
||
// If it crosses the threshold, just remove the oldest entry
|
||
//
|
||
if ( CacheTable->CurrentEntries >= CacheTable->MaxHashIndex * NB_MAX_AVG_CACHE_ENTRIES_PER_BUCKET ) {
|
||
PNETBIOS_CACHE OldestCacheEntry = NULL;
|
||
PNETBIOS_CACHE NextEntry;
|
||
PLIST_ENTRY p;
|
||
|
||
for ( HashIndex = 0; HashIndex < CacheTable->MaxHashIndex; HashIndex++) {
|
||
if ( (p = CacheTable->Bucket[ HashIndex ].Blink ) != &CacheTable->Bucket[ HashIndex ] ) {
|
||
NextEntry = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
|
||
if ( OldestCacheEntry ) {
|
||
if ( NextEntry->TimeStamp < OldestCacheEntry->TimeStamp ) {
|
||
OldestCacheEntry = NextEntry;
|
||
}
|
||
} else {
|
||
OldestCacheEntry = NextEntry;
|
||
}
|
||
}
|
||
}
|
||
|
||
CTEAssert( OldestCacheEntry );
|
||
|
||
NB_DEBUG2 (CACHE, ("Threshold exceeded, removing oldest cache entry %lx\n", OldestCacheEntry));
|
||
RemoveEntryList (&OldestCacheEntry->Linkage);
|
||
CacheTable->CurrentEntries--;
|
||
|
||
if (--OldestCacheEntry->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Freed cache entry %lx\n", OldestCacheEntry));
|
||
|
||
NbiFreeMemory(
|
||
OldestCacheEntry,
|
||
sizeof(NETBIOS_CACHE) + ((OldestCacheEntry->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Aged out");
|
||
|
||
}
|
||
|
||
}
|
||
HashIndex = ( ( CacheEntry->NetbiosName[0] & 0x0f ) << 4 ) + ( CacheEntry->NetbiosName[1] & 0x0f );
|
||
HashIndex = HashIndex % CacheTable->MaxHashIndex;
|
||
|
||
InsertHeadList( &CacheTable->Bucket[HashIndex], &CacheEntry->Linkage );
|
||
CacheTable->CurrentEntries++;
|
||
} /* InsertInNetbiosCacheTable */
|
||
|
||
|
||
__inline
|
||
VOID
|
||
ReinsertInNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN PNETBIOS_CACHE OldEntry,
|
||
IN PNETBIOS_CACHE NewEntry
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine inserts a new cache entry at the same place where
|
||
the old entry was.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
CacheEntry - Entry to be inserted.
|
||
|
||
Return Value:
|
||
|
||
None
|
||
|
||
--*/
|
||
|
||
{
|
||
PLIST_ENTRY OldPrevious;
|
||
|
||
OldPrevious = OldEntry->Linkage.Blink;
|
||
RemoveEntryList (&OldEntry->Linkage);
|
||
InsertHeadList (OldPrevious, &NewEntry->Linkage);
|
||
} /* ReinsertInNetbiosCacheTable */
|
||
|
||
__inline
|
||
VOID
|
||
RemoveFromNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN PNETBIOS_CACHE CacheEntry
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine removes an entry from the cache table.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
CacheEntry - Entry to be removed.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
--*/
|
||
|
||
{
|
||
RemoveEntryList( &CacheEntry->Linkage );
|
||
CacheTable->CurrentEntries--;
|
||
} /* RemoveFromNetbiosCacheTable */
|
||
|
||
|
||
|
||
VOID
|
||
FlushOldFromNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN USHORT AgeLimit
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine removes all the old entries from the hash table.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
AgeLimit - All the entries older than AgeLimit will be removed.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
--*/
|
||
|
||
{
|
||
USHORT HashIndex;
|
||
PLIST_ENTRY p;
|
||
PNETBIOS_CACHE CacheName;
|
||
|
||
//
|
||
// run the hash table looking for old entries. Since new entries
|
||
// are stored at the head and all entries are time stamped when
|
||
// they are inserted, we scan backwards and stop once we find
|
||
// an entry which does not need to be aged.
|
||
// we repeat this for each bucket.
|
||
|
||
for ( HashIndex = 0; HashIndex < CacheTable->MaxHashIndex; HashIndex++) {
|
||
for (p = CacheTable->Bucket[ HashIndex ].Blink;
|
||
p != &CacheTable->Bucket[ HashIndex ];
|
||
) {
|
||
|
||
CacheName = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
p = p->Blink;
|
||
|
||
//
|
||
// see if any entries have been around for more than agelimit
|
||
//
|
||
|
||
if ((USHORT)(NbiDevice->CacheTimeStamp - CacheName->TimeStamp) >= AgeLimit ) {
|
||
|
||
RemoveEntryList (&CacheName->Linkage);
|
||
CacheTable->CurrentEntries--;
|
||
|
||
if (--CacheName->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Aging out name cache entry %lx\n", CacheName));
|
||
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Aged out");
|
||
|
||
}
|
||
|
||
} else {
|
||
|
||
break;
|
||
|
||
}
|
||
} // for loop
|
||
} // for loop
|
||
} /* FlushOldFromNetbiosCacheTable */
|
||
|
||
VOID
|
||
FlushFailedNetbiosCacheEntries(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine removes all the failed entries from the hash table.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
--*/
|
||
|
||
{
|
||
USHORT HashIndex;
|
||
PLIST_ENTRY p;
|
||
PNETBIOS_CACHE CacheName;
|
||
|
||
if (NULL == CacheTable) {
|
||
return;
|
||
}
|
||
|
||
//
|
||
// run the hash table looking for old entries. Since new entries
|
||
// are stored at the head and all entries are time stamped when
|
||
// they are inserted, we scan backwards and stop once we find
|
||
// an entry which does not need to be aged.
|
||
// we repeat this for each bucket.
|
||
|
||
for ( HashIndex = 0; HashIndex < CacheTable->MaxHashIndex; HashIndex++) {
|
||
for (p = CacheTable->Bucket[ HashIndex ].Blink;
|
||
p != &CacheTable->Bucket[ HashIndex ];
|
||
) {
|
||
|
||
CacheName = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
p = p->Blink;
|
||
|
||
//
|
||
// flush all the failed cache entries.
|
||
// We do this when a new adapter appears, and there's a possiblity that
|
||
// the failed entries might succeed now on the new adapter.
|
||
//
|
||
|
||
if (CacheName->NetworksUsed == 0) {
|
||
RemoveEntryList (&CacheName->Linkage);
|
||
CacheTable->CurrentEntries--;
|
||
CTEAssert( CacheName->ReferenceCount == 1 );
|
||
CTEAssert( CacheName->NetworksAllocated == 1 );
|
||
|
||
NB_DEBUG2 (CACHE, ("Flushing out failed name cache entry %lx\n", CacheName));
|
||
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE),
|
||
MEMORY_CACHE,
|
||
"Aged out");
|
||
|
||
}
|
||
} // for loop
|
||
} // for loop
|
||
} /* FlushFailedNetbiosCacheEntries */
|
||
|
||
VOID
|
||
RemoveInvalidRoutesFromNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN NIC_HANDLE UNALIGNED *InvalidNicHandle
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine removes all invalid route entries from the hash table.
|
||
Routes become invalid when the binding is deleted in Ipx due to PnP
|
||
event.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
InvalidRouteNicId - NicId of the invalid routes.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
--*/
|
||
|
||
{
|
||
PLIST_ENTRY p;
|
||
PNETBIOS_CACHE CacheName;
|
||
USHORT i,j,NetworksRemoved;
|
||
USHORT HashIndex;
|
||
PDEVICE Device = NbiDevice;
|
||
|
||
//
|
||
// Flush all the cache entries that are using this NicId in the local
|
||
// target.
|
||
//
|
||
|
||
for ( HashIndex = 0; HashIndex < Device->NameCache->MaxHashIndex; HashIndex++) {
|
||
for (p = Device->NameCache->Bucket[ HashIndex ].Flink;
|
||
p != &Device->NameCache->Bucket[ HashIndex ];
|
||
) {
|
||
|
||
CacheName = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
p = p->Flink;
|
||
|
||
|
||
//
|
||
// Remove each of those routes which is using this NicId.
|
||
// if no routes left, then flush the cache entry also.
|
||
// ( unique names have only one route anyways )
|
||
//
|
||
for ( i = 0, NetworksRemoved = 0; i < CacheName->NetworksUsed; i++ ) {
|
||
if ( CacheName->Networks[i].LocalTarget.NicHandle.NicId == InvalidNicHandle->NicId ) {
|
||
CTEAssert( RtlEqualMemory( &CacheName->Networks[i].LocalTarget.NicHandle, InvalidNicHandle, sizeof(NIC_HANDLE)));
|
||
for ( j = i+1; j < CacheName->NetworksUsed; j++ ) {
|
||
CacheName->Networks[j-1] = CacheName->Networks[j];
|
||
}
|
||
NetworksRemoved++;
|
||
} else if ( CacheName->Networks[i].LocalTarget.NicHandle.NicId > InvalidNicHandle->NicId ) {
|
||
CacheName->Networks[i].LocalTarget.NicHandle.NicId--;
|
||
}
|
||
}
|
||
CTEAssert( NetworksRemoved <= CacheName->NetworksUsed );
|
||
if ( ! ( CacheName->NetworksUsed -= NetworksRemoved ) ) {
|
||
RemoveEntryList (&CacheName->Linkage);
|
||
CacheTable->CurrentEntries--;
|
||
|
||
NB_DEBUG2 (CACHE, ("Removed cache entry %lx bcoz route(NicId %d) deleted\n", CacheName, InvalidNicHandle->NicId ));
|
||
if (--CacheName->ReferenceCount == 0) {
|
||
|
||
NB_DEBUG2 (CACHE, ("Freed name cache entry %lx\n", CacheName));
|
||
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Aged out");
|
||
|
||
}
|
||
}
|
||
} // for loop
|
||
} // for loop
|
||
} /* RemoveInvalidRoutesFromNetbiosCacheTable */
|
||
|
||
|
||
NTSTATUS
|
||
FindInNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable,
|
||
IN PUCHAR NameToBeFound,
|
||
OUT PNETBIOS_CACHE *CacheEntry
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine finds a netbios name in the Hash Table and returns
|
||
the corresponding cache entry.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
CacheEntry - Pointer to the netbios cache entry if found.
|
||
|
||
Return Value:
|
||
|
||
STATUS_SUCCESS - if successful.
|
||
|
||
STATUS_UNSUCCESSFUL - otherwise.
|
||
|
||
--*/
|
||
|
||
{
|
||
USHORT HashIndex;
|
||
PLIST_ENTRY p;
|
||
PNETBIOS_CACHE FoundCacheName;
|
||
|
||
|
||
HashIndex = ( ( NameToBeFound[0] & 0x0f ) << 4 ) + ( NameToBeFound[1] & 0x0f );
|
||
HashIndex = HashIndex % CacheTable->MaxHashIndex;
|
||
|
||
for (p = ( CacheTable->Bucket[ HashIndex ] ).Flink;
|
||
p != &CacheTable->Bucket[ HashIndex ];
|
||
p = p->Flink) {
|
||
|
||
FoundCacheName = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
|
||
//
|
||
// See if this entry is for the same name we are looking for.
|
||
|
||
if ( RtlEqualMemory (FoundCacheName->NetbiosName, NameToBeFound, 16) ) {
|
||
*CacheEntry = FoundCacheName;
|
||
return STATUS_SUCCESS;
|
||
}
|
||
}
|
||
|
||
return STATUS_UNSUCCESSFUL;
|
||
} /* FindInNetbiosCacheTable */
|
||
|
||
NTSTATUS
|
||
CreateNetbiosCacheTable(
|
||
IN OUT PNETBIOS_CACHE_TABLE *NewTable,
|
||
IN USHORT MaxHashIndex
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine creates a new hash table for netbios cache
|
||
and initializes it.
|
||
|
||
THIS ROUTINE IS CALLED WITH THE DEVICE LOCK HELD AND RETURNS
|
||
WITH THE LOCK HELD.
|
||
|
||
Arguments:
|
||
|
||
NewTable - The pointer of the table to be created.
|
||
|
||
MaxHashIndex - Number of buckets in the hash table.
|
||
|
||
Return Value:
|
||
|
||
STATUS_SUCCESS - if successful.
|
||
|
||
STATUS_INSUFFICIENT_RESOURCES - If cannot allocate memory.
|
||
|
||
--*/
|
||
|
||
{
|
||
USHORT i;
|
||
|
||
*NewTable = NbiAllocateMemory (sizeof(NETBIOS_CACHE_TABLE) + sizeof(LIST_ENTRY) * ( MaxHashIndex - 1) ,
|
||
MEMORY_CACHE, "Cache Table");
|
||
|
||
if ( *NewTable ) {
|
||
for ( i = 0; i < MaxHashIndex; i++ ) {
|
||
InitializeListHead(& (*NewTable)->Bucket[i] );
|
||
}
|
||
|
||
(*NewTable)->MaxHashIndex = MaxHashIndex;
|
||
(*NewTable)->CurrentEntries = 0;
|
||
return STATUS_SUCCESS;
|
||
}
|
||
else {
|
||
NB_DEBUG( CACHE, ("Cannot create Netbios Cache Table\n") );
|
||
return STATUS_INSUFFICIENT_RESOURCES;
|
||
}
|
||
|
||
} /* CreateNetbiosCacheTable */
|
||
|
||
|
||
VOID
|
||
DestroyNetbiosCacheTable(
|
||
IN PNETBIOS_CACHE_TABLE CacheTable
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine removes all entries from the hash table.
|
||
and free up the hash table.
|
||
|
||
Arguments:
|
||
|
||
CacheTable - The pointer of the Hash Table.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
--*/
|
||
|
||
{
|
||
USHORT HashIndex;
|
||
PLIST_ENTRY p;
|
||
PNETBIOS_CACHE CacheName;
|
||
|
||
|
||
for ( HashIndex = 0; HashIndex < CacheTable->MaxHashIndex; HashIndex++) {
|
||
while (!IsListEmpty ( &( CacheTable->Bucket[ HashIndex ] ) ) ) {
|
||
|
||
p = RemoveHeadList ( &( CacheTable->Bucket[ HashIndex ] ));
|
||
CacheTable->CurrentEntries--;
|
||
CacheName = CONTAINING_RECORD (p, NETBIOS_CACHE, Linkage);
|
||
|
||
NB_DEBUG2 (CACHE, ("Free cache entry %lx\n", CacheName));
|
||
|
||
NbiFreeMemory(
|
||
CacheName,
|
||
sizeof(NETBIOS_CACHE) + ((CacheName->NetworksAllocated-1) * sizeof(NETBIOS_NETWORK)),
|
||
MEMORY_CACHE,
|
||
"Free entries");
|
||
|
||
}
|
||
} // for loop
|
||
|
||
CTEAssert( CacheTable->CurrentEntries == 0 );
|
||
|
||
NbiFreeMemory (CacheTable, sizeof(NETBIOS_CACHE_TABLE) + sizeof(LIST_ENTRY) * ( CacheTable->MaxHashIndex - 1) ,
|
||
MEMORY_CACHE, "Free Cache Table");
|
||
|
||
} /* DestroyNetbiosCacheTable */
|
||
|
||
|