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.
1866 lines
66 KiB
1866 lines
66 KiB
/*++
|
|
|
|
Copyright (c) 1995 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
NTPNP.c
|
|
|
|
Abstract:
|
|
|
|
This module implements the DRIVER_INITIALIZATION routine for the
|
|
NBT Transport and other routines that are specific to the NT implementation
|
|
of a driver.
|
|
|
|
Author:
|
|
|
|
Earle R. Horton (earleh) 08-Nov-1995
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
|
|
#include "precomp.h"
|
|
#include "ntddip.h" // Needed for PNETBT_PNP_RECONFIG_REQUEST
|
|
#include "ntprocs.h"
|
|
#include <tcpinfo.h>
|
|
#include <tdiinfo.h>
|
|
#include "ntpnp.tmh"
|
|
|
|
#ifdef _NETBIOSLESS
|
|
NTSTATUS
|
|
NbtSpecialDeviceAdd(
|
|
PUNICODE_STRING pucBindName,
|
|
PUNICODE_STRING pucExportName,
|
|
PWSTR pKeyName,
|
|
USHORT DefaultSessionPort,
|
|
USHORT DefaultDatagramPort
|
|
);
|
|
|
|
NTSTATUS
|
|
NbtSpecialReadRegistry(
|
|
PWSTR pKeyName,
|
|
tDEVICECONTEXT *pDeviceContext,
|
|
USHORT DefaultSessionPort,
|
|
USHORT DefaultDatagramPort
|
|
);
|
|
#endif
|
|
|
|
tDEVICECONTEXT *
|
|
CheckAddrNotification(
|
|
IN PTA_ADDRESS Addr,
|
|
IN PUNICODE_STRING pDeviceName,
|
|
OUT ULONG* IpAddr
|
|
);
|
|
|
|
extern HANDLE TdiClientHandle;
|
|
extern HANDLE TdiProviderHandle;
|
|
DWORD AddressCount = 0;
|
|
|
|
NET_DEVICE_POWER_STATE LastSystemPowerState = NetDeviceStateD0; // by default
|
|
|
|
//******************* Pageable Routine Declarations ****************
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma CTEMakePageable(PAGE, NbtNotifyTdiClients)
|
|
#pragma CTEMakePageable(PAGE, NbtAddressAdd)
|
|
#pragma CTEMakePageable(PAGE, NbtAddNewInterface)
|
|
#pragma CTEMakePageable(PAGE, NbtDeviceAdd)
|
|
#pragma CTEMakePageable(PAGE, TdiAddressArrival)
|
|
#pragma CTEMakePageable(PAGE, TdiAddressDeletion)
|
|
#pragma CTEMakePageable(PAGE, TdiBindHandler)
|
|
#pragma CTEMakePageable(PAGE, NbtCreateSmbDevice)
|
|
#pragma CTEMakePageable(PAGE, NbtSpecialReadRegistry)
|
|
#pragma CTEMakePageable(PAGE, NbtPnPPowerComplete)
|
|
#pragma CTEMakePageable(PAGE, TdiPnPPowerHandler)
|
|
#pragma CTEMakePageable(PAGE, LookupDeviceInRegistry)
|
|
#pragma CTEMakePageable(PAGE, CheckAddrNotification)
|
|
#endif
|
|
//******************* Pageable Routine Declarations ****************
|
|
|
|
|
|
|
|
//
|
|
// This used at the boot time.
|
|
// We shouldn't call TdiProviderReady until all the interfaces
|
|
// we know so far have been initialized
|
|
//
|
|
// TcpipReady: set to TRUE when we receive TdiProviderReady from IP
|
|
// NumIfBeingIndicated: the # of interfaces being indicated to our clients
|
|
// JustBooted: set to FALSE after we call TdiProviderReady
|
|
//
|
|
|
|
DWORD JustBooted = TRUE;
|
|
#define IsBootTime() (InterlockedExchange(&JustBooted, FALSE))
|
|
|
|
//#if DBG
|
|
//
|
|
// TcpipReady is for debugging purpose only.
|
|
//
|
|
// BootTimeCounter is initialized to ONE which
|
|
// take it into account.
|
|
//
|
|
int TcpipReady = FALSE;
|
|
//#endif
|
|
|
|
LONG BootTimeCounter = 1; // For the IP's ProviderReady
|
|
|
|
|
|
|
|
void
|
|
NbtUpBootCounter(void)
|
|
{
|
|
if (!JustBooted) {
|
|
return;
|
|
}
|
|
|
|
ASSERT(BootTimeCounter >= 0);
|
|
|
|
InterlockedIncrement(&BootTimeCounter);
|
|
}
|
|
|
|
void
|
|
NbtDownBootCounter(void)
|
|
{
|
|
LONG CounterSnapshot;
|
|
|
|
if (!JustBooted) {
|
|
return;
|
|
}
|
|
|
|
ASSERT(BootTimeCounter > 0);
|
|
CounterSnapshot = InterlockedDecrement(&BootTimeCounter);
|
|
|
|
if (!CounterSnapshot && IsBootTime()) {
|
|
|
|
//
|
|
// Just try our best
|
|
//
|
|
// The caller always call us at PASSIVE_LEVEL except from
|
|
// StartProcessNbtDhcpRequests, a timer routine which could
|
|
// be called at DISPATCH_LEVEL
|
|
//
|
|
if (KeGetCurrentIrql() == PASSIVE_LEVEL) {
|
|
|
|
TdiProviderReady (TdiProviderHandle); // Notify our clients now
|
|
|
|
} else {
|
|
|
|
//
|
|
// Although this is a benign assert, we still want it
|
|
// to capture the normal cases in which this function
|
|
// should be called at PASSIVE_LEVEL.
|
|
//
|
|
ASSERT (0);
|
|
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
static
|
|
IsIPv6Interface(
|
|
PUNICODE_STRING pDeviceName
|
|
)
|
|
{
|
|
return (pDeviceName->Length >= 28 && pDeviceName->Buffer[13] == '6');
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
tDEVICECONTEXT *
|
|
NbtFindAndReferenceDevice(
|
|
PUNICODE_STRING pucBindName,
|
|
BOOLEAN fNameIsBindName
|
|
)
|
|
{
|
|
PLIST_ENTRY pHead;
|
|
PLIST_ENTRY pEntry;
|
|
tDEVICECONTEXT *pDeviceContext;
|
|
CTELockHandle OldIrq;
|
|
PUNICODE_STRING pucNameToCompare;
|
|
|
|
CTESpinLock(&NbtConfig.JointLock,OldIrq);
|
|
|
|
pHead = &NbtConfig.DeviceContexts;
|
|
pEntry = pHead->Flink;
|
|
while (pEntry != pHead)
|
|
{
|
|
pDeviceContext = CONTAINING_RECORD(pEntry,tDEVICECONTEXT,Linkage);
|
|
//
|
|
// Reference this device so that it doesn't disappear when we release the lock!
|
|
//
|
|
NBT_REFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, TRUE);
|
|
CTESpinFree(&NbtConfig.JointLock,OldIrq);
|
|
|
|
//
|
|
// Set the right type of name to compare against
|
|
//
|
|
if (fNameIsBindName)
|
|
{
|
|
pucNameToCompare = &pDeviceContext->BindName;
|
|
}
|
|
else
|
|
{
|
|
pucNameToCompare = &pDeviceContext->ExportName;
|
|
}
|
|
|
|
//
|
|
// Use case-insensitive compare since registry is case-insensitive
|
|
//
|
|
if (RtlCompareUnicodeString(pucBindName, pucNameToCompare, TRUE) == 0)
|
|
{
|
|
//
|
|
// We have already Referenced this device above
|
|
//
|
|
return (pDeviceContext);
|
|
}
|
|
|
|
CTESpinLock(&NbtConfig.JointLock,OldIrq);
|
|
|
|
pEntry = pEntry->Flink;
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, TRUE);
|
|
}
|
|
|
|
CTESpinFree(&NbtConfig.JointLock,OldIrq);
|
|
return (tDEVICECONTEXT *)NULL;
|
|
}
|
|
|
|
VOID
|
|
NbtNotifyTdiClients(
|
|
IN tDEVICECONTEXT *pDeviceContext,
|
|
IN enum eTDI_ACTION Action
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This is where all Tdi registrations and Deregistrations occur
|
|
ASSUMPTION: Only 1 thread will running this request at any time
|
|
|
|
Arguments:
|
|
|
|
None.
|
|
|
|
Return Value:
|
|
|
|
None (since this is a Worker thread)
|
|
|
|
--*/
|
|
|
|
{
|
|
CTELockHandle OldIrq;
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
HANDLE NetAddressRegistrationHandle, DeviceRegistrationHandle;
|
|
PLIST_ENTRY pEntry;
|
|
|
|
CTEPagedCode();
|
|
|
|
NbtTrace(NBT_TRACE_PNP, ("ExportName=%Z BindName=%Z Action=%d",
|
|
&pDeviceContext->ExportName, &pDeviceContext->BindName, Action));
|
|
switch (Action)
|
|
{
|
|
case NBT_TDI_REGISTER:
|
|
{
|
|
//
|
|
// Add the "permanent" name to the local name table. This is the IP
|
|
// address of the node padded out to 16 bytes with zeros.
|
|
//
|
|
#ifdef _NETBIOSLESS
|
|
if (!IsDeviceNetbiosless(pDeviceContext))
|
|
#endif
|
|
{
|
|
NbtAddPermanentName(pDeviceContext);
|
|
}
|
|
|
|
//
|
|
// If the device was not registered with TDI, do so now.
|
|
//
|
|
if (!pDeviceContext->DeviceRegistrationHandle)
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtNotifyTdiClients: Calling TdiRegisterDeviceObject ...\n"));
|
|
|
|
status = TdiRegisterDeviceObject( &pDeviceContext->ExportName,
|
|
&pDeviceContext->DeviceRegistrationHandle);
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
pDeviceContext->DeviceRegistrationHandle = NULL;
|
|
}
|
|
NbtTrace(NBT_TRACE_PNP, ("RegisterDeviceObject: ExportName=%Z BindName=%Z status=%!status!",
|
|
&pDeviceContext->ExportName, &pDeviceContext->BindName, status));
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.NbtNotifyTdiClients: TdiRegisterDeviceObject for <%x> returned <%x>\n",
|
|
pDeviceContext, status));
|
|
}
|
|
|
|
//
|
|
// If the Net address was not registered with TDI, do so now.
|
|
//
|
|
if ((!pDeviceContext->NetAddressRegistrationHandle) &&
|
|
#ifdef _NETBIOSLESS
|
|
(!IsDeviceNetbiosless(pDeviceContext)) &&
|
|
#endif
|
|
(pDeviceContext->pPermClient))
|
|
{
|
|
TA_NETBIOS_ADDRESS PermAddress;
|
|
|
|
PermAddress.Address[0].AddressLength = sizeof(TDI_ADDRESS_NETBIOS);
|
|
PermAddress.Address[0].AddressType = TDI_ADDRESS_TYPE_NETBIOS;
|
|
PermAddress.Address[0].Address[0].NetbiosNameType = TDI_ADDRESS_NETBIOS_TYPE_UNIQUE;
|
|
CTEMemCopy( PermAddress.Address[0].Address[0].NetbiosName,
|
|
pDeviceContext->pPermClient->pAddress->pNameAddr->Name,
|
|
NETBIOS_NAME_SIZE);
|
|
|
|
status = TdiRegisterNetAddress(
|
|
(PTA_ADDRESS) PermAddress.Address,
|
|
&pDeviceContext->ExportName,
|
|
(PTDI_PNP_CONTEXT) &pDeviceContext->Context2,
|
|
&pDeviceContext->NetAddressRegistrationHandle);
|
|
|
|
if (!NT_SUCCESS(status))
|
|
{
|
|
pDeviceContext->NetAddressRegistrationHandle = NULL;
|
|
}
|
|
NbtTrace(NBT_TRACE_PNP, ("RegisterNetAddress: ExportName=%Z BindName=%Z status=%!status!",
|
|
&pDeviceContext->ExportName, &pDeviceContext->BindName, status));
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtNotifyTdiClients: TdiRegisterNetAddress for <%x> returned <%x>\n",
|
|
pDeviceContext, status));
|
|
}
|
|
break;
|
|
}
|
|
|
|
case NBT_TDI_DEREGISTER:
|
|
{
|
|
if (NetAddressRegistrationHandle = pDeviceContext->NetAddressRegistrationHandle)
|
|
{
|
|
pDeviceContext->NetAddressRegistrationHandle = NULL;
|
|
status = TdiDeregisterNetAddress (NetAddressRegistrationHandle);
|
|
|
|
NbtTrace(NBT_TRACE_PNP, ("DeregisterNetAddress: ExportName=%Z BindName=%Z status=%!status!",
|
|
&pDeviceContext->ExportName, &pDeviceContext->BindName, status));
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtNbtNotifyTdiClients: TdiDeregisterNetAddress<%x> returned<%x>\n",
|
|
pDeviceContext, status));
|
|
}
|
|
|
|
if (DeviceRegistrationHandle = pDeviceContext->DeviceRegistrationHandle)
|
|
{
|
|
pDeviceContext->DeviceRegistrationHandle = NULL;
|
|
status = TdiDeregisterDeviceObject (DeviceRegistrationHandle);
|
|
|
|
NbtTrace(NBT_TRACE_PNP, ("DeregisterDeviceObject: ExportName=%Z BindName=%Z status=%!status!",
|
|
&pDeviceContext->ExportName, &pDeviceContext->BindName, status));
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtNotifyTdiClients: TdiDeregisterDeviceObject<%x> returned<%x>\n",
|
|
pDeviceContext, status));
|
|
}
|
|
|
|
//
|
|
// The permanent name is a function of the MAC address so remove
|
|
// it since the Address is going away
|
|
//
|
|
#ifdef _NETBIOSLESS
|
|
if (!IsDeviceNetbiosless(pDeviceContext))
|
|
#endif
|
|
{
|
|
NbtRemovePermanentName(pDeviceContext);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
KdPrint(("Nbt.NbtNotifyTdiClients: ERROR: Invalid Action=<%x> on Device <%x>\n",
|
|
Action, pDeviceContext));
|
|
}
|
|
}
|
|
|
|
VOID
|
|
NbtUpdateSmbBinding(
|
|
VOID
|
|
)
|
|
{
|
|
tDEVICECONTEXT *pSavedSmbDevice = NULL;
|
|
KIRQL OldIrq = 0;
|
|
|
|
CTESpinLock(&NbtConfig.JointLock,OldIrq);
|
|
if (pNbtSmbDevice && !gbDestroyingSmbDevice) {
|
|
pSavedSmbDevice = pNbtSmbDevice;
|
|
NBT_REFERENCE_DEVICE (pSavedSmbDevice, REF_DEV_SMB_BIND, TRUE);
|
|
}
|
|
CTESpinFree(&NbtConfig.JointLock,OldIrq);
|
|
|
|
if (pSavedSmbDevice) {
|
|
|
|
NETBT_SMB_BIND_REQUEST SmbRequest = { 0 };
|
|
|
|
SmbRequest.RequestType = SMB_SERVER;
|
|
SmbRequest.MultiSZBindList = NbtConfig.pServerBindings;
|
|
SmbRequest.pDeviceName = NULL;
|
|
SmbRequest.PnPOpCode = TDI_PNP_OP_ADD;
|
|
|
|
if (SmbRequest.MultiSZBindList) {
|
|
NbtSetSmbBindingInfo2(
|
|
pSavedSmbDevice,
|
|
&SmbRequest
|
|
);
|
|
}
|
|
|
|
SmbRequest.RequestType = SMB_CLIENT;
|
|
SmbRequest.MultiSZBindList = NbtConfig.pClientBindings;
|
|
SmbRequest.pDeviceName = NULL;
|
|
SmbRequest.PnPOpCode = TDI_PNP_OP_ADD;
|
|
if (SmbRequest.MultiSZBindList) {
|
|
NbtSetSmbBindingInfo2(
|
|
pSavedSmbDevice,
|
|
&SmbRequest
|
|
);
|
|
}
|
|
NBT_DEREFERENCE_DEVICE (pSavedSmbDevice, REF_DEV_SMB_BIND, FALSE);
|
|
}
|
|
}
|
|
|
|
VOID
|
|
SmbNotifyTdiClients (
|
|
IN enum eTDI_ACTION Action
|
|
)
|
|
{
|
|
tDEVICECONTEXT *pSavedSmbDevice = NULL;
|
|
KIRQL OldIrq = 0;
|
|
|
|
CTESpinLock(&NbtConfig.JointLock,OldIrq);
|
|
if (pNbtSmbDevice && !gbDestroyingSmbDevice) {
|
|
pSavedSmbDevice = pNbtSmbDevice;
|
|
NBT_REFERENCE_DEVICE (pSavedSmbDevice, REF_DEV_SMB_BIND, TRUE);
|
|
}
|
|
CTESpinFree(&NbtConfig.JointLock,OldIrq);
|
|
|
|
if (pSavedSmbDevice) {
|
|
NbtNotifyTdiClients (pSavedSmbDevice, Action);
|
|
NBT_DEREFERENCE_DEVICE (pSavedSmbDevice, REF_DEV_SMB_BIND, FALSE);
|
|
}
|
|
}
|
|
|
|
//----------------------------------------------------------------------------
|
|
NTSTATUS
|
|
NbtAddressAdd(
|
|
ULONG IpAddr,
|
|
tDEVICECONTEXT *pDeviceContext,
|
|
PUNICODE_STRING pucBindString
|
|
)
|
|
{
|
|
NTSTATUS status = STATUS_SUCCESS;
|
|
tADDRARRAY DeviceAddressArray;
|
|
tIPADDRESS pIpAddresses[MAX_IP_ADDRS];
|
|
tIPADDRESS SubnetMask;
|
|
ULONG NumAddressesRead;
|
|
|
|
CTEPagedCode();
|
|
|
|
ASSERT(pucBindString && IpAddr);
|
|
|
|
//
|
|
// Find the bind and export devices to use from the device
|
|
// described in the registry that uses this address.
|
|
//
|
|
if (status != STATUS_SUCCESS) {
|
|
return status;
|
|
}
|
|
|
|
status = LookupDeviceInRegistry(pucBindString, &DeviceAddressArray, NULL);
|
|
if (!NT_SUCCESS(status)) {
|
|
KdPrint(("netbt!NbtAddressAdd: Cannot find device in the registry: status <%x>\n", status));
|
|
NbtTrace(NBT_TRACE_PNP, ("BindName=%Z IP=%!ipaddr! status=%!status!",
|
|
pucBindString, IpAddr, status));
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
//
|
|
// Fetch a static IP address from the registry.
|
|
//
|
|
*pIpAddresses = 0;
|
|
status = GetIPFromRegistry (pucBindString,
|
|
pIpAddresses,
|
|
&SubnetMask,
|
|
MAX_IP_ADDRS,
|
|
&NumAddressesRead,
|
|
NBT_IP_STATIC);
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtAddressAdd: GetIPFromRegistry for NBT_IP_STATIC returned <%x>\n",status));
|
|
NbtTrace(NBT_TRACE_PNP, ("GetIPFromRegistry return status=%!status! for NBT_IP_STATIC BindName=%Z IP=%!ipaddr!",
|
|
status, pucBindString, IpAddr));
|
|
|
|
if ((status != STATUS_SUCCESS) || (*pIpAddresses != IpAddr)) {
|
|
//
|
|
// This one doesn't have a valid static address. Try DHCP.
|
|
//
|
|
*pIpAddresses = 0; // Cleanup any previously-read entries!
|
|
status = GetIPFromRegistry (pucBindString,
|
|
pIpAddresses,
|
|
&SubnetMask,
|
|
MAX_IP_ADDRS,
|
|
&NumAddressesRead,
|
|
NBT_IP_DHCP);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtAddressAdd: GetIPFromRegistry for NBT_IP_DHCP returned <%x>\n",status));
|
|
NbtTrace(NBT_TRACE_PNP, ("GetIPFromRegistry return status=%!status! for NBT_IP_DHCP BindName=%Z IP=%!ipaddr!",
|
|
status, pucBindString, IpAddr));
|
|
}
|
|
|
|
if ((status != STATUS_SUCCESS) || (*pIpAddresses != IpAddr)) {
|
|
//
|
|
// Check for Autoconfiguration IP address
|
|
//
|
|
*pIpAddresses = 0; // Cleanup any previously-read entries!
|
|
status = GetIPFromRegistry (pucBindString,
|
|
pIpAddresses,
|
|
&SubnetMask,
|
|
MAX_IP_ADDRS,
|
|
&NumAddressesRead,
|
|
NBT_IP_AUTOCONFIGURATION);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.NbtAddressAdd: GetIPFromRegistry for NBT_IP_AUTOCONFIGURATION returned <%x>\n",
|
|
status));
|
|
NbtTrace(NBT_TRACE_PNP, ("GetIPFromRegistry return status=%!status! for NBT_IP_AUTO BindName=%Z IP=%!ipaddr!",
|
|
status, pucBindString, IpAddr));
|
|
}
|
|
|
|
//
|
|
// The Device must have been created beforehand by using the BindHandler
|
|
//
|
|
if ((status == STATUS_SUCCESS) && (*pIpAddresses == IpAddr)) {
|
|
BOOLEAN IsDuplicateNotification = FALSE;
|
|
#ifdef MULTIPLE_WINS
|
|
int i;
|
|
#endif
|
|
|
|
pDeviceContext->RasProxyFlags = DeviceAddressArray.RasProxyFlags;
|
|
pDeviceContext->EnableNagling = DeviceAddressArray.EnableNagling;
|
|
//
|
|
// Initialize the WINs server addresses
|
|
//
|
|
if ((IpAddr == pDeviceContext->IpAddress) &&
|
|
(DeviceAddressArray.NetbiosEnabled == pDeviceContext->NetbiosEnabled))
|
|
{
|
|
IsDuplicateNotification = TRUE;
|
|
NbtTrace(NBT_TRACE_PNP, ("Duplicate notification: %Z %!ipaddr!", pucBindString, IpAddr));
|
|
}
|
|
|
|
pDeviceContext->lNameServerAddress = DeviceAddressArray.NameServerAddress;
|
|
pDeviceContext->lBackupServer = DeviceAddressArray.BackupServer;
|
|
pDeviceContext->RefreshToBackup = 0;
|
|
pDeviceContext->SwitchedToBackup = 0;
|
|
#ifdef MULTIPLE_WINS
|
|
pDeviceContext->lNumOtherServers = DeviceAddressArray.NumOtherServers;
|
|
pDeviceContext->lLastResponsive = 0;
|
|
for (i = 0; i < DeviceAddressArray.NumOtherServers; i++) {
|
|
pDeviceContext->lOtherServers[i] = DeviceAddressArray.Others[i];
|
|
}
|
|
#endif
|
|
#ifdef _NETBIOSLESS
|
|
pDeviceContext->NetbiosEnabled = DeviceAddressArray.NetbiosEnabled;
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("NbtAddressAdd: %wZ, enabled = %d\n",
|
|
&pDeviceContext->ExportName, pDeviceContext->NetbiosEnabled));
|
|
NbtTrace(NBT_TRACE_PNP, ("NetbiosEnabled=%x: %Z %!ipaddr!",
|
|
pDeviceContext->NetbiosEnabled, pucBindString, IpAddr));
|
|
#endif
|
|
|
|
//
|
|
// Open the addresses with the transports
|
|
// these are passed into here in the reverse byte order, wrt to the IOCTL
|
|
// from DHCP.
|
|
//
|
|
if (IsDuplicateNotification) {
|
|
status = STATUS_UNSUCCESSFUL;
|
|
} else {
|
|
ULONG i;
|
|
|
|
//
|
|
// We may have read more than Ip address for this Device
|
|
// so save all of them
|
|
//
|
|
if (NumAddressesRead > 1) {
|
|
for (i=1; i<NumAddressesRead; i++) {
|
|
pDeviceContext->AdditionalIpAddresses[i-1] = pIpAddresses[i];
|
|
}
|
|
}
|
|
ASSERT (NumAddressesRead > 0);
|
|
#if 0
|
|
//
|
|
// TcpIp does not support opening of multiple addresses
|
|
// per handle, so disable this option for now!
|
|
//
|
|
pDeviceContext->NumAdditionalIpAddresses = NumAddressesRead - 1;
|
|
#endif
|
|
pDeviceContext->AssignedIpAddress = IpAddr;
|
|
|
|
NbtNewDhcpAddress(pDeviceContext,htonl(*pIpAddresses),htonl(SubnetMask));
|
|
|
|
NbtUpdateSmbBinding();
|
|
}
|
|
} else {
|
|
KdPrint (("Nbt.NbtAddressAdd: ERROR -- pDeviceContext=<%x>, status=<%x>, IpAddr=<%x>, ulIpAddress=<%x>\n",
|
|
pDeviceContext, status, IpAddr, *pIpAddresses));
|
|
status = STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
NTSTATUS
|
|
NbtAddNewInterface (
|
|
IN PIRP pIrp,
|
|
IN PVOID *pBuffer,
|
|
IN ULONG Size
|
|
)
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
Creates a device context by coming up with a unique export string to name
|
|
the device.
|
|
|
|
Arguments:
|
|
|
|
Return Value:
|
|
|
|
Notes:
|
|
|
|
|
|
--*/
|
|
{
|
|
ULONG nextIndex = InterlockedIncrement(&NbtConfig.InterfaceIndex);
|
|
WCHAR Suffix[16];
|
|
WCHAR Bind[60] = L"\\Device\\If";
|
|
WCHAR Export[60] = L"\\Device\\NetBt_If";
|
|
UNICODE_STRING ucSuffix;
|
|
UNICODE_STRING ucBindStr;
|
|
UNICODE_STRING ucExportStr;
|
|
NTSTATUS status;
|
|
ULONG OutSize;
|
|
BOOLEAN Attached = FALSE;
|
|
tADDRARRAY *pAddrArray = NULL;
|
|
tDEVICECONTEXT *pDeviceContext = NULL;
|
|
PNETBT_ADD_DEL_IF pAddDelIf = (PNETBT_ADD_DEL_IF)pBuffer;
|
|
|
|
CTEPagedCode();
|
|
|
|
//
|
|
// Validate output buffer size
|
|
//
|
|
if (Size < sizeof(NETBT_ADD_DEL_IF))
|
|
{
|
|
KdPrint(("Nbt.NbtAddNewInterface: Output buffer too small for struct\n"));
|
|
NbtTrace(NBT_TRACE_PNP, ("Output buffer too small for struct size=%d, required=%d",
|
|
Size, sizeof(NETBT_ADD_DEL_IF)));
|
|
return(STATUS_INVALID_PARAMETER);
|
|
}
|
|
//
|
|
// Create the bind/export strings as:
|
|
// Bind: \Device\IF<1> Export: \Device\NetBt_IF<1>
|
|
// where 1 is a unique interface index.
|
|
//
|
|
ucSuffix.Buffer = Suffix;
|
|
ucSuffix.Length = 0;
|
|
ucSuffix.MaximumLength = sizeof(Suffix);
|
|
|
|
RtlIntegerToUnicodeString(nextIndex, 10, &ucSuffix);
|
|
|
|
RtlInitUnicodeString(&ucBindStr, Bind);
|
|
ucBindStr.MaximumLength = sizeof(Bind);
|
|
RtlInitUnicodeString(&ucExportStr, Export);
|
|
ucExportStr.MaximumLength = sizeof(Export);
|
|
|
|
RtlAppendUnicodeStringToString(&ucBindStr, &ucSuffix);
|
|
RtlAppendUnicodeStringToString(&ucExportStr, &ucSuffix);
|
|
|
|
OutSize = FIELD_OFFSET (NETBT_ADD_DEL_IF, IfName[0]) +
|
|
ucExportStr.Length + sizeof(UNICODE_NULL);
|
|
|
|
if (Size < OutSize)
|
|
{
|
|
KdPrint(("Nbt.NbtAddNewInterface: Buffer too small for name\n"));
|
|
NbtTrace(NBT_TRACE_PNP, ("Buffer too small for name size=%d, required=%d", Size, OutSize));
|
|
pAddDelIf->Length = ucExportStr.Length + sizeof(UNICODE_NULL);
|
|
pAddDelIf->Status = STATUS_BUFFER_TOO_SMALL;
|
|
pIrp->IoStatus.Information = sizeof(NETBT_ADD_DEL_IF);
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(( "Nbt.NbtAddNewInterface: Creating ucBindStr: %ws ucExportStr: %ws\n",
|
|
ucBindStr.Buffer, ucExportStr.Buffer ));
|
|
//
|
|
// Attach to the system process so that all handles are created in the
|
|
// proper context.
|
|
//
|
|
CTEAttachFsp(&Attached, REF_FSP_ADD_INTERFACE);
|
|
|
|
status = NbtCreateDeviceObject (&ucBindStr,
|
|
&ucExportStr,
|
|
&pAddrArray[0],
|
|
&pDeviceContext,
|
|
NBT_DEVICE_CLUSTER);
|
|
|
|
CTEDetachFsp(Attached, REF_FSP_ADD_INTERFACE);
|
|
|
|
if (pDeviceContext)
|
|
{
|
|
//
|
|
// Fill up the output buffer with the export name
|
|
//
|
|
RtlCopyMemory(&pAddDelIf->IfName[0], ucExportStr.Buffer, ucExportStr.Length+sizeof(UNICODE_NULL));
|
|
pAddDelIf->Length = ucExportStr.Length + sizeof(UNICODE_NULL);
|
|
pAddDelIf->InstanceNumber = pDeviceContext->InstanceNumber;
|
|
pAddDelIf->Status = STATUS_SUCCESS;
|
|
pIrp->IoStatus.Information = OutSize;
|
|
//
|
|
// By-pass the TDI PnP mechanism for logical interfaces (ie don't register with TDI)
|
|
//
|
|
return (STATUS_SUCCESS);
|
|
}
|
|
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtCreateDeviceObject return %!status! for BindName=%Z ExportName=%Z",
|
|
status, &ucBindStr, &ucExportStr));
|
|
|
|
return status;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
NbtDeviceAdd(
|
|
PUNICODE_STRING pucBindString
|
|
)
|
|
{
|
|
tDEVICECONTEXT *pDeviceContext;
|
|
UNICODE_STRING ucExportString;
|
|
tADDRARRAY DeviceAddressArray;
|
|
BOOLEAN Attached = FALSE;
|
|
NTSTATUS Status;
|
|
PLIST_ENTRY pHead, pEntry;
|
|
NTSTATUS dontcarestatus;
|
|
int i;
|
|
|
|
|
|
CTEPagedCode();
|
|
|
|
//
|
|
// Ignore it if we already bind to the device
|
|
//
|
|
if (pDeviceContext = NbtFindAndReferenceDevice (pucBindString, TRUE)) {
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, FALSE);
|
|
|
|
KdPrint (("Nbt.NbtDeviceAdd: ERROR: Device=<%ws> already exists!\n", pucBindString->Buffer));
|
|
NbtTrace(NBT_TRACE_PNP, ("Device %Z already exists!", pucBindString));
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
//
|
|
// Can we find the new device in registry file? If not, ignore it.
|
|
//
|
|
Status = LookupDeviceInRegistry(pucBindString, &DeviceAddressArray, &ucExportString);
|
|
if (!NT_SUCCESS(Status)) {
|
|
KdPrint(("netbt!NbtDeviceAdd: Cannot find device in the registry: status <%x>\n", Status));
|
|
NbtTrace(NBT_TRACE_PNP, ("LookupDeviceInRegistry return %!status! for device %Z",
|
|
Status, pucBindString));
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
//
|
|
// Attach to the system process so that all handles are created in the
|
|
// proper context.
|
|
//
|
|
CTEAttachFsp(&Attached, REF_FSP_DEVICE_ADD);
|
|
|
|
Status = NbtCreateDeviceObject (pucBindString,
|
|
&ucExportString,
|
|
&DeviceAddressArray,
|
|
&pDeviceContext,
|
|
NBT_DEVICE_REGULAR);
|
|
CTEMemFree(ucExportString.Buffer);
|
|
|
|
CTEDetachFsp(Attached, REF_FSP_DEVICE_ADD);
|
|
|
|
//
|
|
// Call Tdi to re-enumerate the addresses for us
|
|
//
|
|
if (NT_SUCCESS (Status)) {
|
|
TdiEnumerateAddresses(TdiClientHandle);
|
|
} else {
|
|
KdPrint (("Nbt.NbtDeviceAdd: ERROR: NbtCreateDeviceObject returned <%x>\n", Status));
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtCreateDeviceObject return %!status! for device %Z",
|
|
Status, pucBindString));
|
|
}
|
|
|
|
return (Status);
|
|
}
|
|
|
|
|
|
// TdiAddressArrival - PnP TDI_ADD_ADDRESS_HANDLER
|
|
// Handles an IP address arriving
|
|
// Called by TDI when an address arrives.
|
|
//
|
|
// Input: Addr - IP address that's coming.
|
|
//
|
|
// Returns: Nothing.
|
|
//
|
|
VOID
|
|
TdiAddressArrival(
|
|
PTA_ADDRESS Addr,
|
|
PUNICODE_STRING pDeviceName,
|
|
PTDI_PNP_CONTEXT Context
|
|
)
|
|
{
|
|
ULONG IpAddr, LastAssignedIpAddress;
|
|
tDEVICECONTEXT *pDeviceContext;
|
|
PTDI_PNP_CONTEXT pTdiContext;
|
|
NTSTATUS status;
|
|
|
|
CTEPagedCode();
|
|
|
|
pDeviceContext = CheckAddrNotification(Addr, pDeviceName, &IpAddr);
|
|
if (pDeviceContext == NULL) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Now the device is referenced!!!
|
|
//
|
|
NbtTrace(NBT_TRACE_PNP, ("TdiAddressArrival for %Z, IpAddr=%!ipaddr!, "
|
|
"pDeviceContext->AssignedIpAddress=%!ipaddr!, pDeviceContext->IpAddress=%!ipaddr!",
|
|
pDeviceName, IpAddr, pDeviceContext->AssignedIpAddress, pDeviceContext->IpAddress));
|
|
|
|
//
|
|
// Update the PDO in Context2
|
|
//
|
|
pTdiContext = (PTDI_PNP_CONTEXT) &pDeviceContext->Context2;
|
|
pTdiContext->ContextSize = Context->ContextSize;
|
|
pTdiContext->ContextType = Context->ContextType;
|
|
*(PVOID UNALIGNED*) pTdiContext->ContextData = *(PVOID UNALIGNED*) Context->ContextData;
|
|
|
|
LastAssignedIpAddress = pDeviceContext->AssignedIpAddress;
|
|
if (NT_SUCCESS (status = NbtAddressAdd(IpAddr, pDeviceContext, pDeviceName))) {
|
|
// Register Smb Device
|
|
// Assumption 1: tdi can assign multiple addresses to same device
|
|
// Assumption 2: tdi will always delete an assignment it made
|
|
// THIS IS CODED FOR ONE ADDRESS PER DEVICE
|
|
// First assigned address wins (and is reference counted)
|
|
if (LastAssignedIpAddress == 0) {
|
|
if ((1 == InterlockedIncrement (&AddressCount)) && (NbtConfig.SMBDeviceEnabled)) {
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiAddressArrival: Registering NetbiosSmb Device\n"));
|
|
SmbNotifyTdiClients (NBT_TDI_REGISTER);
|
|
}
|
|
}
|
|
} else {
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtAddressAdd return %!status! for device %Z",
|
|
status, pDeviceName));
|
|
}
|
|
|
|
//
|
|
// Derefenece the device
|
|
//
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, FALSE);
|
|
SetNodeType();
|
|
}
|
|
|
|
// TdiAddressDeletion - PnP TDI_DEL_ADDRESS_HANDLER
|
|
// Handles an IP address going away.
|
|
// Called by TDI when an address is deleted. If it's an address we
|
|
// care about we'll clean up appropriately.
|
|
//
|
|
// Input: Addr - IP address that's going.
|
|
//
|
|
// Returns: Nothing.
|
|
//
|
|
VOID
|
|
TdiAddressDeletion(
|
|
PTA_ADDRESS Addr,
|
|
PUNICODE_STRING pDeviceName,
|
|
PTDI_PNP_CONTEXT Context
|
|
)
|
|
{
|
|
ULONG IpAddr;
|
|
tDEVICECONTEXT *pDeviceContext;
|
|
|
|
CTEPagedCode();
|
|
|
|
pDeviceContext = CheckAddrNotification(Addr, pDeviceName, &IpAddr);
|
|
if (pDeviceContext == NULL) {
|
|
return;
|
|
}
|
|
|
|
//
|
|
// Now the device is referenced!!!
|
|
//
|
|
NbtTrace(NBT_TRACE_PNP, ("TdiAddressDeletion for %Z, IpAddr=%!ipaddr!, "
|
|
"pDeviceContext->AssignedIpAddress=%!ipaddr!, pDeviceContext->IpAddress=%!ipaddr!",
|
|
pDeviceName, IpAddr, pDeviceContext->AssignedIpAddress, pDeviceContext->IpAddress));
|
|
|
|
// Deregister Smb Device
|
|
// THIS IS CODED FOR ONE ADDRESS PER DEVICE
|
|
// Only deletion of an assigned address wins (and is ref counted)
|
|
if (pDeviceContext->AssignedIpAddress == IpAddr) {
|
|
if ((0 == InterlockedDecrement (&AddressCount))) {
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiAddressDeletion: Deregistering NetbiosSmb Device\n"));
|
|
SmbNotifyTdiClients (NBT_TDI_DEREGISTER);
|
|
}
|
|
|
|
pDeviceContext->AssignedIpAddress = 0;
|
|
if (IpAddr == pDeviceContext->IpAddress) {
|
|
NbtNewDhcpAddress(pDeviceContext, 0, 0);
|
|
}
|
|
}
|
|
|
|
//
|
|
// Derefenece the device
|
|
//
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, FALSE);
|
|
SetNodeType();
|
|
}
|
|
|
|
VOID
|
|
TdiBindHandler(
|
|
TDI_PNP_OPCODE PnPOpCode,
|
|
PUNICODE_STRING pDeviceName,
|
|
PWSTR MultiSZBindList)
|
|
{
|
|
NTSTATUS Status;
|
|
tDEVICECONTEXT *pDeviceContext;
|
|
|
|
CTEPagedCode();
|
|
|
|
switch (PnPOpCode)
|
|
{
|
|
case (TDI_PNP_OP_ADD):
|
|
{
|
|
//
|
|
// Ignore TCPIP6 interface
|
|
//
|
|
if (IsIPv6Interface(pDeviceName)) {
|
|
return;
|
|
}
|
|
|
|
Status = NbtDeviceAdd (pDeviceName);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_ADD]: ERROR <%x>, AdapterCount=<%x>\n",
|
|
Status, NbtConfig.AdapterCount));
|
|
NbtLogEvent (EVENT_NBT_CREATE_DEVICE, Status, 0x111);
|
|
}
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtDeviceAdd return %!status! for %Z", Status, pDeviceName));
|
|
|
|
break;
|
|
}
|
|
|
|
case (TDI_PNP_OP_DEL):
|
|
{
|
|
//
|
|
// If the Device is Valid, Dereference it!
|
|
//
|
|
if (pDeviceContext = NbtFindAndReferenceDevice (pDeviceName, TRUE))
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_DEL]: Dereferencing Device <%wZ>\n",
|
|
&pDeviceContext->BindName));
|
|
|
|
// Deref it since we referenced it above!
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, FALSE);
|
|
|
|
Status = NbtDestroyDevice (pDeviceContext, TRUE);
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtDestoryDevice return %!status! for %Z", Status, pDeviceName));
|
|
}
|
|
else
|
|
{
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_DEL]: ERROR -- Device=<%wZ>\n", pDeviceName));
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtFindAndReferenceDevice return NULL for %Z", pDeviceName));
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case (TDI_PNP_OP_UPDATE):
|
|
{
|
|
tDEVICES *pBindDevices = NULL;
|
|
tDEVICES *pExportDevices = NULL;
|
|
tADDRARRAY *pAddrArray = NULL;
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_UPDATE]: Got Update Notification\n"));
|
|
//
|
|
// Re-read the registry
|
|
//
|
|
CTEExAcquireResourceExclusive(&NbtConfig.Resource,TRUE);
|
|
NbtReadRegistry (&pBindDevices, &pExportDevices, &pAddrArray);
|
|
NbtReadRegistryCleanup (&pBindDevices, &pExportDevices, &pAddrArray);
|
|
CTEExReleaseResource(&NbtConfig.Resource);
|
|
SetNodeType();
|
|
NbtTrace(NBT_TRACE_PNP, ("[TDI_PNP_OP_UPDATE]"));
|
|
break;
|
|
}
|
|
|
|
case (TDI_PNP_OP_PROVIDERREADY):
|
|
{
|
|
WCHAR wcIpDeviceName[60] = DD_IP_DEVICE_NAME;
|
|
UNICODE_STRING ucIpDeviceName;
|
|
|
|
RtlInitUnicodeString(&ucIpDeviceName, wcIpDeviceName);
|
|
ucIpDeviceName.MaximumLength = sizeof (wcIpDeviceName);
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_NETREADY]: Comparing <%wZ> with <%wZ>\n",
|
|
pDeviceName, &ucIpDeviceName));
|
|
NbtTrace(NBT_TRACE_PNP, ("[TDI_PNP_OP_NETREADY]: <%Z> <==> <%Z>", pDeviceName, &ucIpDeviceName));
|
|
|
|
//
|
|
// Use case-insensitive compare since registry is case-insensitive
|
|
//
|
|
if (RtlCompareUnicodeString(pDeviceName, &ucIpDeviceName, TRUE) == 0)
|
|
{
|
|
//
|
|
// This is the notification we were waiting for from Ip, so now
|
|
// notify our clients of our completion status as a provider!
|
|
//
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiBindHandler[TDI_PNP_OP_NETREADY]: Got Ip Notification\n"));
|
|
|
|
//#if DBG
|
|
TcpipReady = TRUE;
|
|
//#endif
|
|
NbtDownBootCounter();
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case (TDI_PNP_OP_NETREADY):
|
|
{
|
|
// Nothing to do!
|
|
NbtTrace(NBT_TRACE_PNP, ("[TDI_PNP_OP_NETREADY]"));
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
KdPrint(("Nbt.TdiBindHandler: Unknown Opcode=<%x>\n", PnPOpCode));
|
|
NbtTrace(NBT_TRACE_PNP, ("Unknown Opcode=<%x>", PnPOpCode));
|
|
ASSERT (0);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
tDEVICECONTEXT *
|
|
NbtCreateSmbDevice(
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
The model of this device is different from the rest of the Netbt devices in that netbt devices
|
|
are per-adapter. For this device there is only one instance across all adapters.
|
|
|
|
This is the code that creates the Smb device. We create it at DriverEntry and Destory it
|
|
at driver Unload time.
|
|
|
|
We try to call existing routines to create the new device, so that we can reuse as much code
|
|
as possible and have the new device initialized identically to the other netbt devices. Then
|
|
we customize this device by setting some variables controlling port and endpoint.
|
|
|
|
In the current design, a message-only Netbt device is only single session. Different sessions,
|
|
or applications, use different Tcp ports. Each message-only Netbt device is assigned a single
|
|
port, such as Smb. If you want to
|
|
support a different application over Netbt, The easiest thing is to instantiate a new message-only
|
|
device with a different Tcp port. Perhaps there is a way to delay the binding of the port on the
|
|
client side from device creation to connection creation?
|
|
|
|
Another idea to consider is to modularize the construction of these message-only devices. You
|
|
could have a table in the registry naming the device, with its unique initialization parameters.
|
|
This code could then read the table.
|
|
|
|
Create and initialize the message special device.
|
|
This function is not driven by Pnp because it is not adapter specific.
|
|
|
|
The idea here was to abstract the details of special devices as much as possible. The way
|
|
the current code is written, you must have a single port for each device. This means you
|
|
typically will get one application for each special device. Right now the only case is
|
|
rdr/srv using message-mode for smb traffic. In the future, if you had another netbios session
|
|
application that wanted an internet pure device, you could just call this routine with the
|
|
new parameters.
|
|
|
|
Two issues that I can think of:
|
|
1. The default session name is still hardcoded. You might want to pass that in here if you
|
|
didn't want *smbserver as the session name.
|
|
2. Binding is per application. Currently there is a .Inf file to get Smb bound to the rdr
|
|
and srv. If you have a new application and a new special device, you will need another .inf
|
|
file.
|
|
|
|
Arguments:
|
|
|
|
|
|
Return Value:
|
|
|
|
NTSTATUS -
|
|
|
|
--*/
|
|
|
|
{
|
|
NTSTATUS Status;
|
|
BOOLEAN Attached = FALSE;
|
|
tDEVICECONTEXT *pDeviceContext = NULL;
|
|
NBT_WORK_ITEM_CONTEXT *pContext;
|
|
UNICODE_STRING ucSmbDeviceBindName;
|
|
UNICODE_STRING ucSmbDeviceExportName;
|
|
WCHAR Path[MAX_PATH];
|
|
UNICODE_STRING ParamPath;
|
|
OBJECT_ATTRIBUTES TmpObjectAttributes;
|
|
HANDLE Handle;
|
|
ULONG Metric;
|
|
|
|
CTEPagedCode();
|
|
|
|
NbtTrace(NBT_TRACE_PNP, ("Creating Smb device"));
|
|
|
|
RtlInitUnicodeString(&ucSmbDeviceBindName, WC_SMB_DEVICE_BIND_NAME);
|
|
ucSmbDeviceBindName.MaximumLength = sizeof (WC_SMB_DEVICE_BIND_NAME);
|
|
RtlInitUnicodeString(&ucSmbDeviceExportName, WC_SMB_DEVICE_EXPORT_NAME);
|
|
ucSmbDeviceExportName.MaximumLength = sizeof (WC_SMB_DEVICE_EXPORT_NAME);
|
|
|
|
CTEAttachFsp(&Attached, REF_FSP_CREATE_SMB_DEVICE);
|
|
|
|
//
|
|
// Create the SMBDevice
|
|
//
|
|
Status = NbtCreateDeviceObject (&ucSmbDeviceBindName, // Bind name, ignored, but must match for delete
|
|
&ucSmbDeviceExportName, // Export name
|
|
NULL,
|
|
&pDeviceContext,
|
|
NBT_DEVICE_NETBIOSLESS);// message-only Netbt device
|
|
|
|
if (NT_SUCCESS(Status))
|
|
{
|
|
pDeviceContext->SessionPort = NbtConfig.DefaultSmbSessionPort;
|
|
pDeviceContext->DatagramPort = NbtConfig.DefaultSmbDatagramPort;
|
|
pDeviceContext->NameServerPort = 0; // Disable this port for security reasons
|
|
|
|
RtlCopyMemory (pDeviceContext->MessageEndpoint, "*SMBSERVER ", NETBIOS_NAME_SIZE );
|
|
|
|
//
|
|
// Here is where we initialize the handles in the special device
|
|
// Create the handles to the transport. This does not depend on dhcp
|
|
// Use LOOP_BACK because we need to put something non-zero here
|
|
//
|
|
// This device is registered based on address notifications
|
|
//
|
|
// Status = NbtCreateAddressObjects (LOOP_BACK, 0, pDeviceContext);
|
|
Status = NbtCreateAddressObjects (INADDR_LOOPBACK, 0, pDeviceContext);
|
|
pDeviceContext->BroadcastAddress = LOOP_BACK; // Make sure no broadcasts
|
|
if (NT_SUCCESS(Status))
|
|
{
|
|
//
|
|
// Now clear the If lists and add the INADDR_LOOPBACK address
|
|
//
|
|
if (pDeviceContext->hSession)
|
|
{
|
|
NbtSetTcpInfo (pDeviceContext->hSession,
|
|
AO_OPTION_IFLIST,
|
|
INFO_TYPE_ADDRESS_OBJECT,
|
|
(ULONG) TRUE);
|
|
NbtSetTcpInfo (pDeviceContext->hSession,
|
|
AO_OPTION_ADD_IFLIST,
|
|
INFO_TYPE_ADDRESS_OBJECT,
|
|
pDeviceContext->IPInterfaceContext);
|
|
}
|
|
|
|
//
|
|
// Now, set the same for the Datagram port
|
|
//
|
|
if ((pDeviceContext->pFileObjects) &&
|
|
(pDeviceContext->pFileObjects->hDgram))
|
|
{
|
|
NbtSetTcpInfo (pDeviceContext->pFileObjects->hDgram,
|
|
AO_OPTION_IFLIST,
|
|
INFO_TYPE_ADDRESS_OBJECT,
|
|
(ULONG) TRUE);
|
|
NbtSetTcpInfo (pDeviceContext->pFileObjects->hDgram,
|
|
AO_OPTION_ADD_IFLIST,
|
|
INFO_TYPE_ADDRESS_OBJECT,
|
|
pDeviceContext->IPInterfaceContext);
|
|
}
|
|
NbtTrace(NBT_TRACE_PNP, ("Successful creating Smb device"));
|
|
}
|
|
else
|
|
{
|
|
KdPrint (("Nbt.NbtCreateSmbDevice: NbtCreateAddressObjects Failed, status = <%x>\n", Status));
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtCreateAddressObject Failed with %!status!", Status));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
KdPrint (("Nbt.NbtCreateSmbDevice: NbtCreateDeviceObject Failed, status = <%x>\n", Status));
|
|
NbtTrace(NBT_TRACE_PNP, ("NbtCreateDeviceObject Failed with %!status!", Status));
|
|
}
|
|
|
|
CTEDetachFsp(Attached, REF_FSP_CREATE_SMB_DEVICE);
|
|
|
|
return (pDeviceContext);
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
NbtPnPPowerComplete(
|
|
IN PNET_PNP_EVENT NetEvent,
|
|
IN NTSTATUS ProviderStatus
|
|
)
|
|
{
|
|
CTEPagedCode();
|
|
|
|
TdiPnPPowerComplete (TdiClientHandle, NetEvent, ProviderStatus);
|
|
NbtTrace(NBT_TRACE_PNP, ("[NbtPnPPowerComplete]"));
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
TdiPnPPowerHandler(
|
|
IN PUNICODE_STRING pDeviceName,
|
|
IN PNET_PNP_EVENT PnPEvent,
|
|
IN PTDI_PNP_CONTEXT Context1,
|
|
IN PTDI_PNP_CONTEXT Context2
|
|
)
|
|
{
|
|
tDEVICECONTEXT *pDeviceContext = NULL;
|
|
NTSTATUS status = STATUS_SUCCESS; // Success by default!
|
|
PNETBT_PNP_RECONFIG_REQUEST PnPEventBuffer = (PNETBT_PNP_RECONFIG_REQUEST) PnPEvent->Buffer;
|
|
PNET_DEVICE_POWER_STATE pPowerState = (PNET_DEVICE_POWER_STATE) PnPEventBuffer; // Power requests
|
|
BOOLEAN fWait = FALSE;
|
|
#ifdef _NETBIOSLESS
|
|
BOOLEAN fOldNetbiosEnabledState;
|
|
#endif
|
|
|
|
CTEPagedCode();
|
|
|
|
//
|
|
// Pass the request up first
|
|
//
|
|
if ((pDeviceName) && (pDeviceName->Length)) {
|
|
if (!(pDeviceContext = NbtFindAndReferenceDevice (pDeviceName, TRUE))) {
|
|
return (STATUS_SUCCESS);
|
|
}
|
|
|
|
#ifdef _NETBIOSLESS
|
|
fOldNetbiosEnabledState = pDeviceContext->NetbiosEnabled;
|
|
#endif
|
|
} else if (PnPEvent->NetEvent != NetEventReconfigure) {
|
|
//
|
|
// pDeviceName is not set for Reconfigure events
|
|
// The only valid case for no Device to be specified is Reconfigure!
|
|
//
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.NbtTdiPnpPowerHandler: Device=<%wZ>, Event=<%d>, C1=<%p>, C2=<%p>\n",
|
|
pDeviceName, PnPEvent->NetEvent, Context1, Context2 ));
|
|
NbtTrace(NBT_TRACE_PNP, ("Device=<%Z>, Event=<%d>, C1=<%p>, C2=<%p>",
|
|
pDeviceName, PnPEvent->NetEvent, Context1, Context2));
|
|
|
|
switch (PnPEvent->NetEvent)
|
|
{
|
|
case (NetEventQueryPower):
|
|
{
|
|
//
|
|
// Check if we should veto this request
|
|
//
|
|
if ((*pPowerState != NetDeviceStateD0) &&
|
|
(NbtConfig.MinimumRefreshSleepTimeout == 0))
|
|
{
|
|
status = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
}
|
|
|
|
status = TdiPnPPowerRequest (&pDeviceContext->ExportName, PnPEvent, Context1, Context2, NbtPnPPowerComplete);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler[QueryPower]: Device=<%x>, PowerState=<%x>, status=<%x>\n",
|
|
pDeviceContext, *pPowerState, status));
|
|
NbtTrace(NBT_TRACE_PNP, ("[QueryPower]: Device=<%Z>, PowerState=<%x>, status=%!status!",
|
|
pDeviceName, *pPowerState, status));
|
|
|
|
//
|
|
// NetBt doesn't need to do anything here, so we'll just return!
|
|
//
|
|
break;
|
|
}
|
|
case (NetEventSetPower):
|
|
{
|
|
//
|
|
// Check if we should veto this request (if requested by user)
|
|
//
|
|
if ((*pPowerState != NetDeviceStateD0) &&
|
|
(NbtConfig.MinimumRefreshSleepTimeout == 0))
|
|
{
|
|
status = STATUS_UNSUCCESSFUL;
|
|
break;
|
|
}
|
|
|
|
status = TdiPnPPowerRequest (&pDeviceContext->ExportName, PnPEvent, Context1, Context2, NbtPnPPowerComplete);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler[SetPower]: Device=<%x>, PowerState=<%d=>%d>, status=<%x>\n",
|
|
pDeviceContext, LastSystemPowerState, *pPowerState, status));
|
|
NbtTrace(NBT_TRACE_PNP, ("[SetPower]: Device=<%Z>, PowerState=<%x>, status=%!status!",
|
|
pDeviceName, *pPowerState, status));
|
|
|
|
CTEExAcquireResourceExclusive(&NbtConfig.Resource,TRUE);
|
|
|
|
if (*pPowerState != LastSystemPowerState) // this is a state transition
|
|
{
|
|
switch (*pPowerState)
|
|
{
|
|
case NetDeviceStateD0:
|
|
{
|
|
NbtConfig.GlobalRefreshState &= ~NBT_G_REFRESH_SLEEPING;
|
|
|
|
if (NbtConfig.pWakeupRefreshTimer)
|
|
{
|
|
if (NT_SUCCESS (NTQueueToWorkerThread(NULL, DelayedNbtStopWakeupTimer,
|
|
NULL,
|
|
NbtConfig.pWakeupRefreshTimer,
|
|
NULL, NULL, FALSE)))
|
|
{
|
|
NbtConfig.pWakeupRefreshTimer->RefCount++;
|
|
NbtConfig.pWakeupRefreshTimer = NULL;
|
|
}
|
|
|
|
// Ignore the return status! (Best effort!)
|
|
StartTimer(RefreshTimeout,
|
|
NbtConfig.InitialRefreshTimeout/NbtConfig.RefreshDivisor,
|
|
NULL, // context value
|
|
NULL, // context2 value
|
|
NULL,
|
|
NULL,
|
|
NULL, // This Timer is Global!
|
|
&NbtConfig.pRefreshTimer,
|
|
0,
|
|
FALSE);
|
|
}
|
|
break;
|
|
}
|
|
|
|
case NetDeviceStateD1:
|
|
case NetDeviceStateD2:
|
|
case NetDeviceStateD3:
|
|
{
|
|
if (LastSystemPowerState != NetDeviceStateD0) // Don't differentiate bw D1, D2, & D3
|
|
{
|
|
break;
|
|
}
|
|
|
|
//
|
|
// Reset the Refresh Timer to function accordingly
|
|
//
|
|
NbtStopRefreshTimer();
|
|
ASSERT (!NbtConfig.pWakeupRefreshTimer);
|
|
NbtConfig.GlobalRefreshState |= NBT_G_REFRESH_SLEEPING;
|
|
|
|
KeClearEvent (&NbtConfig.WakeupTimerStartedEvent);
|
|
if (NT_SUCCESS (NTQueueToWorkerThread(NULL, DelayedNbtStartWakeupTimer,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
NULL,
|
|
FALSE)))
|
|
{
|
|
fWait = TRUE;
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
ASSERT (0);
|
|
}
|
|
}
|
|
|
|
LastSystemPowerState = *pPowerState;
|
|
}
|
|
|
|
CTEExReleaseResource(&NbtConfig.Resource);
|
|
|
|
if (fWait)
|
|
{
|
|
NTSTATUS status;
|
|
status = KeWaitForSingleObject (&NbtConfig.WakeupTimerStartedEvent, // Object to wait on.
|
|
Executive, // Reason for waiting
|
|
KernelMode, // Processor mode
|
|
FALSE, // Alertable
|
|
NULL); // Timeout
|
|
ASSERT(status == STATUS_SUCCESS);
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
case (NetEventQueryRemoveDevice):
|
|
{
|
|
status = TdiPnPPowerRequest (&pDeviceContext->ExportName, PnPEvent, Context1, Context2, NbtPnPPowerComplete);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler: NetEventQueryRemoveDevice -- status=<%x>\n",status));
|
|
NbtTrace(NBT_TRACE_PNP, ("[NetEventQueryRemoveDevice]: Device=<%Z>, status=%!status!",
|
|
pDeviceName, status));
|
|
break;
|
|
}
|
|
case (NetEventCancelRemoveDevice):
|
|
{
|
|
status = TdiPnPPowerRequest (&pDeviceContext->ExportName, PnPEvent, Context1, Context2, NbtPnPPowerComplete);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler: NetEventCancelRemoveDevice -- status=<%x>\n",status));
|
|
NbtTrace(NBT_TRACE_PNP, ("[NetEventCancelRemoveDevice]: Device=<%Z>, status=%!status!",
|
|
pDeviceName, status));
|
|
break;
|
|
}
|
|
case (NetEventReconfigure):
|
|
{
|
|
//
|
|
// First check if the WINs server entries have been modified
|
|
//
|
|
if (pDeviceContext)
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.TdiPnPPowerHandler: WINs servers have changed for <%x>\n",pDeviceContext));
|
|
status = NTReReadRegistry (pDeviceContext, TRUE);
|
|
NbtTrace(NBT_TRACE_PNP, ("[NetEventReconfigure]: WINs servers have changed for %Z, status=%!status!",
|
|
pDeviceName, status));
|
|
}
|
|
else // check the rest of the options
|
|
{
|
|
#if 0
|
|
// EnumDnsOption is no longer set through the UI, so we can ignore this!
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.TdiPnPPowerHandler: Checking EnumDNS option for <%x>\n",pDeviceContext));
|
|
|
|
switch (PnPEventBuffer->enumDnsOption)
|
|
{
|
|
case (WinsOnly):
|
|
NbtConfig.UseDnsOnly = FALSE;
|
|
NbtConfig.ResolveWithDns = FALSE;
|
|
break;
|
|
|
|
case (DnsOnly):
|
|
NbtConfig.UseDnsOnly = TRUE;
|
|
NbtConfig.ResolveWithDns = TRUE;
|
|
break;
|
|
|
|
case (WinsThenDns):
|
|
NbtConfig.UseDnsOnly = FALSE;
|
|
NbtConfig.ResolveWithDns = TRUE;
|
|
break;
|
|
|
|
default:
|
|
KdPrint (("Nbt.TdiPnPPowerHandler: ERROR bad option for enumDnsOption <%x>\n",
|
|
PnPEventBuffer->enumDnsOption));
|
|
}
|
|
#endif // 0
|
|
|
|
if (PnPEventBuffer->fLmhostsEnabled)
|
|
{
|
|
if ((!NbtConfig.EnableLmHosts) || // if the user is re-enabling LmHosts
|
|
(PnPEventBuffer->fLmhostsFileSet)) // the user wants to use a new LmHosts file
|
|
{
|
|
tDEVICES *pBindDevices=NULL;
|
|
tDEVICES *pExportDevices=NULL;
|
|
tADDRARRAY *pAddrArray=NULL;
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.TdiPnPPowerHandler: Reading LmHosts file\n"));
|
|
|
|
|
|
//
|
|
// ReRead the registry for the LmHost options
|
|
//
|
|
CTEExAcquireResourceExclusive(&NbtConfig.Resource,TRUE);
|
|
status = NbtReadRegistry (&pBindDevices, &pExportDevices, &pAddrArray);
|
|
NbtReadRegistryCleanup(&pBindDevices, &pExportDevices, &pAddrArray);
|
|
CTEExReleaseResource(&NbtConfig.Resource);
|
|
|
|
DelayedNbtResyncRemoteCache(NULL, NULL, NULL, NULL);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
NbtConfig.EnableLmHosts = PnPEventBuffer->fLmhostsEnabled;
|
|
}
|
|
}
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler: NetEventReconfigure -- status=<%x>\n",status));
|
|
NbtTrace(NBT_TRACE_PNP, ("NetEventReconfigure -- %Z status=%!status!", pDeviceName, status));
|
|
|
|
break;
|
|
}
|
|
case (NetEventBindList):
|
|
{
|
|
//
|
|
// Just do a general reread of the registry parameters since we could
|
|
// get WINS address change notifications through here!
|
|
//
|
|
if (pDeviceContext)
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.TdiPnPPowerHandler: NetEventBindList request for <%x>\n",pDeviceContext));
|
|
status = NTReReadRegistry (pDeviceContext, TRUE);
|
|
}
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.TdiPnPPowerHandler: NetEventBindList -- status=<%x>\n",status));
|
|
break;
|
|
}
|
|
case (NetEventPnPCapabilities):
|
|
{
|
|
//
|
|
// Query into TcpIp to get the latest Pnp properties on this device!
|
|
//
|
|
if (pDeviceContext)
|
|
{
|
|
PULONG pResult = NULL;
|
|
ULONG BufferLen = sizeof (ULONG);
|
|
ULONG Input = pDeviceContext->IPInterfaceContext;
|
|
|
|
//
|
|
// Query the latest WOL capabilities on this adapter!
|
|
//
|
|
if (NT_SUCCESS (status = NbtProcessIPRequest (IOCTL_IP_GET_WOL_CAPABILITY,
|
|
&Input, // Input buffer
|
|
BufferLen,
|
|
(PVOID) &pResult,
|
|
&BufferLen)))
|
|
{
|
|
ASSERT (pResult);
|
|
pDeviceContext->WOLProperties = *pResult;
|
|
CTEMemFree (pResult);
|
|
}
|
|
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint (("Nbt.TdiPnPPowerHandler[NetEventPnPCapabilities] <%x>, pDeviceContext=<%p>, Input=<%x>, Result=<%x>\n",status, pDeviceContext, Input, pDeviceContext->WOLProperties));
|
|
|
|
status = STATUS_SUCCESS;
|
|
}
|
|
break;
|
|
}
|
|
|
|
default:
|
|
KdPrint(("Nbt.TdiPnPPowerHandler: Invalid NetEvent=<%x> -- status=<%x>\n",
|
|
PnPEvent->NetEvent,status));
|
|
|
|
}
|
|
|
|
if (pDeviceContext) {
|
|
#ifdef _NETBIOSLESS
|
|
//
|
|
// Check for transition in Netbios enable state
|
|
//
|
|
if (fOldNetbiosEnabledState != pDeviceContext->NetbiosEnabled)
|
|
{
|
|
if (pDeviceContext->NetbiosEnabled)
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.NbtTdiPnpPowerHandler: Enabling address on %wZ\n",
|
|
&pDeviceContext->ExportName));
|
|
|
|
// We don't know what the right IP address is,
|
|
// so we tell TDI to Enumerate!
|
|
TdiEnumerateAddresses(TdiClientHandle);
|
|
}
|
|
else
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("NbtTdiPnp: disabling address on %wZ", &pDeviceContext->ExportName ));
|
|
NbtNewDhcpAddress(pDeviceContext, 0, 0); // Get rid of IP address to disable adapter
|
|
}
|
|
}
|
|
#endif
|
|
|
|
NBT_DEREFERENCE_DEVICE (pDeviceContext, REF_DEV_FIND_REF, FALSE);
|
|
SetNodeType();
|
|
}
|
|
|
|
return (status);
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------------------------
|
|
NTSTATUS
|
|
CheckSetWakeupPattern(
|
|
tDEVICECONTEXT *pDeviceContext,
|
|
PUCHAR pName,
|
|
BOOLEAN RequestAdd
|
|
)
|
|
{
|
|
NTSTATUS Status = STATUS_UNSUCCESSFUL;
|
|
CTELockHandle OldIrq;
|
|
ULONG InBufLen = 0;
|
|
|
|
IP_WAKEUP_PATTERN_REQUEST IPWakeupPatternReq; // Ioctl Data Format (12 bytes)
|
|
NET_PM_WAKEUP_PATTERN_DESC WakeupPatternDesc; // IP data (8 Bytes)
|
|
NETBT_WAKEUP_PATTERN PatternData; // Data Storage for the Wakeup Data itself (72)
|
|
|
|
BOOLEAN fAttached = FALSE;
|
|
|
|
if (pDeviceContext->DeviceType != NBT_DEVICE_REGULAR)
|
|
{
|
|
return STATUS_UNSUCCESSFUL;
|
|
}
|
|
|
|
CTESpinLock(pDeviceContext,OldIrq);
|
|
//
|
|
// Only 1 pattern (the first one) can be set at any time
|
|
//
|
|
if (RequestAdd)
|
|
{
|
|
if (pDeviceContext->WakeupPatternRefCount)
|
|
{
|
|
//
|
|
// There is already a pattern registered on this device
|
|
//
|
|
if (CTEMemEqu (pDeviceContext->WakeupPatternName, pName, NETBIOS_NAME_SIZE-1))
|
|
{
|
|
pDeviceContext->WakeupPatternRefCount++;
|
|
Status = STATUS_SUCCESS;
|
|
}
|
|
|
|
CTESpinFree(pDeviceContext,OldIrq);
|
|
return (Status);
|
|
}
|
|
|
|
// This is the first pattern
|
|
CTEMemCopy(&pDeviceContext->WakeupPatternName,pName,NETBIOS_NAME_SIZE);
|
|
pDeviceContext->WakeupPatternRefCount++;
|
|
}
|
|
//
|
|
// This is a Delete pattern request
|
|
//
|
|
else
|
|
{
|
|
if ((!pDeviceContext->WakeupPatternRefCount) || // No pattern currently registered
|
|
(!CTEMemEqu (pDeviceContext->WakeupPatternName, pName, NETBIOS_NAME_SIZE-1))) // Not this pattern
|
|
{
|
|
CTESpinFree(pDeviceContext,OldIrq);
|
|
return (STATUS_UNSUCCESSFUL);
|
|
}
|
|
//
|
|
// The pattern for deletion matched the pattern that was set earlier
|
|
//
|
|
else if (--pDeviceContext->WakeupPatternRefCount)
|
|
{
|
|
//
|
|
// This pattern is still referenced
|
|
//
|
|
CTESpinFree(pDeviceContext,OldIrq);
|
|
return (STATUS_SUCCESS);
|
|
}
|
|
}
|
|
CTESpinFree(pDeviceContext,OldIrq);
|
|
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("Nbt.SetWakeupPattern: %s<%-16.16s:%x> on Device=<%wZ>\n",
|
|
(RequestAdd ? "Add" : "Remove"), pName, pName[15], &pDeviceContext->BindName));
|
|
|
|
//
|
|
// Initialize the Pattern Data
|
|
//
|
|
CTEZeroMemory((PVOID) &PatternData, sizeof(NETBT_WAKEUP_PATTERN));
|
|
ConvertToHalfAscii((PCHAR) &PatternData.nbt_NameRR, pName, NULL, 0);
|
|
PatternData.iph_protocol = 0x11; // UDP Protocol
|
|
PatternData.udph_src = htons (NBT_NAMESERVICE_UDP_PORT);
|
|
PatternData.udph_dest = htons (NBT_NAMESERVICE_UDP_PORT);
|
|
PatternData.nbt_OpCodeFlags = htons (0x0010);
|
|
//
|
|
// Initialize the WakeupPattern Description
|
|
//
|
|
WakeupPatternDesc.Next = NULL;
|
|
WakeupPatternDesc.Ptrn = (PUCHAR) &PatternData;
|
|
WakeupPatternDesc.Mask = NetBTPatternMask;
|
|
WakeupPatternDesc.PtrnLen = NetBTPatternLen;
|
|
//
|
|
// Initialize the WakeupPattern Request
|
|
//
|
|
IPWakeupPatternReq.PtrnDesc = &WakeupPatternDesc;
|
|
IPWakeupPatternReq.AddPattern = RequestAdd; // Add = TRUE, Remove = FALSE
|
|
|
|
IPWakeupPatternReq.InterfaceContext = pDeviceContext->IPInterfaceContext;
|
|
|
|
//
|
|
// Now, register the Wakeup pattern on this adapter
|
|
//
|
|
Status = NbtProcessIPRequest (IOCTL_IP_WAKEUP_PATTERN,
|
|
&IPWakeupPatternReq, // Input buffer
|
|
sizeof (IP_WAKEUP_PATTERN_REQUEST),
|
|
NULL,
|
|
NULL
|
|
);
|
|
|
|
//
|
|
// If we were doing an add, we need to Deref since we failed to register this pattern
|
|
//
|
|
if ((RequestAdd) &&
|
|
(!NT_SUCCESS (Status)))
|
|
{
|
|
CTESpinLock(pDeviceContext,OldIrq);
|
|
pDeviceContext->WakeupPatternRefCount--;
|
|
CTESpinFree(pDeviceContext,OldIrq);
|
|
}
|
|
|
|
return Status;
|
|
}
|
|
|
|
/*
|
|
* bug #88696
|
|
* Set the global variable NodeType based on RegistryNodeType and WINS configuration
|
|
*/
|
|
void
|
|
SetNodeType(void)
|
|
{
|
|
/* We only need to check if the registry NodeType is broadcast */
|
|
if (RegistryNodeType & (BNODE| DEFAULT_NODE_TYPE)) {
|
|
/*
|
|
* If there exist at least one active link with WINS server,
|
|
* we set NodeType to hybrid.
|
|
*/
|
|
PLIST_ENTRY head, item;
|
|
CTELockHandle OldIrq;
|
|
|
|
CTESpinLock(&NbtConfig.JointLock,OldIrq);
|
|
NodeType = RegistryNodeType;
|
|
head = &NbtConfig.DeviceContexts;
|
|
for (item = head->Flink; item != head; item = item->Flink) {
|
|
tDEVICECONTEXT* dev;
|
|
|
|
dev = CONTAINING_RECORD(item, tDEVICECONTEXT, Linkage);
|
|
if (dev->IsDestroyed || dev->IpAddress == 0 ||
|
|
!dev->NetbiosEnabled) {
|
|
continue;
|
|
}
|
|
if ((dev->lNameServerAddress!=LOOP_BACK && dev->lNameServerAddress) || (dev->lBackupServer!=LOOP_BACK && dev->lBackupServer)) {
|
|
NodeType = (MSNODE | (NodeType & PROXY));
|
|
/* We don't need to check further */
|
|
break;
|
|
}
|
|
}
|
|
|
|
// A broadcast node cannot have proxy
|
|
if ((NodeType & BNODE) && (NodeType & PROXY)) {
|
|
NodeType &= (~PROXY);
|
|
}
|
|
CTESpinFree(&NbtConfig.JointLock,OldIrq);
|
|
}
|
|
}
|
|
|
|
NTSTATUS
|
|
LookupDeviceInRegistry(
|
|
IN PUNICODE_STRING pBindName,
|
|
OUT tADDRARRAY* pAddrs,
|
|
OUT PUNICODE_STRING pExportName
|
|
)
|
|
{
|
|
tDEVICES *pBindDevices = NULL;
|
|
tDEVICES *pExportDevices = NULL;
|
|
tADDRARRAY *pAddrArray = NULL;
|
|
NTSTATUS Status;
|
|
int i;
|
|
|
|
CTEPagedCode();
|
|
|
|
CTEExAcquireResourceExclusive(&NbtConfig.Resource,TRUE);
|
|
Status = NbtReadRegistry (&pBindDevices, &pExportDevices, &pAddrArray);
|
|
|
|
if (!NT_SUCCESS(Status) || !pBindDevices || !pExportDevices || !pAddrArray) {
|
|
KdPrint (("NetBT!LookupDeviceInRegistry: Registry incomplete: pBind=<%x>, pExport=<%x>, pAddrArray=<%x>\n",
|
|
pBindDevices, pExportDevices, pAddrArray));
|
|
CTEExReleaseResource(&NbtConfig.Resource);
|
|
NbtReadRegistryCleanup(&pBindDevices, &pExportDevices, &pAddrArray);
|
|
|
|
return STATUS_REGISTRY_CORRUPT;
|
|
}
|
|
|
|
Status = STATUS_UNSUCCESSFUL;
|
|
for (i=0; i<pNbtGlobConfig->uNumDevicesInRegistry; i++ ) {
|
|
if (RtlCompareUnicodeString(pBindName, &pBindDevices->Names[i], TRUE) == 0) {
|
|
Status = STATUS_SUCCESS;
|
|
if (pAddrs) {
|
|
RtlCopyMemory(pAddrs, &pAddrArray[i], sizeof(pAddrArray[i]));
|
|
}
|
|
if (pExportName) {
|
|
pExportName->MaximumLength = pExportDevices->Names[i].MaximumLength;
|
|
pExportName->Buffer = NbtAllocMem(pExportDevices->Names[i].MaximumLength, NBT_TAG2('17'));
|
|
if (pExportName->Buffer == NULL) {
|
|
KdPrint (("NetBT!LookupDeviceInRegistry: fail to allocate memory\n"));
|
|
Status = STATUS_INSUFFICIENT_RESOURCES;
|
|
} else {
|
|
RtlCopyUnicodeString(pExportName, &pExportDevices->Names[i]);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
NbtReadRegistryCleanup(&pBindDevices, &pExportDevices, &pAddrArray);
|
|
CTEExReleaseResource(&NbtConfig.Resource);
|
|
return Status;
|
|
}
|
|
|
|
tDEVICECONTEXT *
|
|
CheckAddrNotification(
|
|
IN PTA_ADDRESS Addr,
|
|
IN PUNICODE_STRING pDeviceName,
|
|
OUT ULONG *IpAddr
|
|
)
|
|
/*++
|
|
Check if the TDI address notification is for us,
|
|
if so, return a Referenced device context and the IP address
|
|
otherwise, return NULL.
|
|
|
|
Note: it is the caller's responsibility to dereference the device context.
|
|
--*/
|
|
{
|
|
CTEPagedCode();
|
|
|
|
if (IsIPv6Interface(pDeviceName)) {
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Ignore any other type of address except IP
|
|
//
|
|
if (Addr->AddressType != TDI_ADDRESS_TYPE_IP) {
|
|
return NULL;
|
|
}
|
|
|
|
*IpAddr = ntohl(((PTDI_ADDRESS_IP)&Addr->Address[0])->in_addr);
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
{
|
|
IF_DBG(NBT_DEBUG_PNP_POWER)
|
|
KdPrint(("netbt!CheckAddrNotification: %d.%d.%d.%d\n",
|
|
((*IpAddr)>>24)&0xFF,((*IpAddr)>>16)&0xFF,((*IpAddr)>>8)&0xFF,(*IpAddr)&0xFF));
|
|
}
|
|
|
|
//
|
|
// Filter out zero address notifications
|
|
//
|
|
if (*IpAddr == 0) {
|
|
KdPrint (("Nbt.TdiAddressDeletion: ERROR: Address <%x> not assigned to any device!\n", IpAddr));
|
|
return NULL;
|
|
}
|
|
|
|
//
|
|
// Ignore this notification if we don't bind to this device
|
|
//
|
|
return NbtFindAndReferenceDevice (pDeviceName, TRUE);
|
|
}
|