2606 lines
68 KiB
C
2606 lines
68 KiB
C
|
/*++
|
|||
|
|
|||
|
Copyright (c) 1989 Microsoft Corporation
|
|||
|
|
|||
|
Module Name:
|
|||
|
|
|||
|
network.c
|
|||
|
|
|||
|
Abstract:
|
|||
|
|
|||
|
This module contains routines for interfacing the LAN Manager server
|
|||
|
to the network.
|
|||
|
|
|||
|
Author:
|
|||
|
|
|||
|
Chuck Lenzmeier (chuckl) 7-Oct-1989
|
|||
|
|
|||
|
Environment:
|
|||
|
|
|||
|
File System Process, kernel mode only
|
|||
|
|
|||
|
Revision History:
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
#include "precomp.h"
|
|||
|
#pragma hdrstop
|
|||
|
|
|||
|
#define BugCheckFileId SRV_FILE_NETWORK
|
|||
|
|
|||
|
//
|
|||
|
// Local declarations
|
|||
|
//
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
GetNetworkAddress (
|
|||
|
IN PENDPOINT Endpoint
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenEndpoint (
|
|||
|
OUT PENDPOINT *Endpoint,
|
|||
|
IN PUNICODE_STRING NetworkName,
|
|||
|
IN PUNICODE_STRING TransportName,
|
|||
|
IN PANSI_STRING TransportAddress,
|
|||
|
IN PUNICODE_STRING DomainName,
|
|||
|
IN ULONG PrimaryMachineFlags,
|
|||
|
IN BOOLEAN AlternateEndpoint
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNetbiosAddress (
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNetbiosExAddress (
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNonNetbiosAddress (
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
);
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenIpxSocket (
|
|||
|
OUT PHANDLE Handle,
|
|||
|
OUT PFILE_OBJECT *FileObject,
|
|||
|
OUT PDEVICE_OBJECT *DeviceObject,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN USHORT Socket
|
|||
|
);
|
|||
|
|
|||
|
#ifdef ALLOC_PRAGMA
|
|||
|
#pragma alloc_text( PAGE, SrvAddServedNet )
|
|||
|
#pragma alloc_text( PAGE, SrvDeleteServedNet )
|
|||
|
#pragma alloc_text( PAGE, SrvDoDisconnect )
|
|||
|
#pragma alloc_text( PAGE, GetNetworkAddress )
|
|||
|
#pragma alloc_text( PAGE, OpenEndpoint )
|
|||
|
#pragma alloc_text( PAGE, OpenNetbiosAddress )
|
|||
|
#pragma alloc_text( PAGE, OpenNonNetbiosAddress )
|
|||
|
#pragma alloc_text( PAGE, OpenIpxSocket )
|
|||
|
#pragma alloc_text( PAGE, SrvRestartAccept )
|
|||
|
#pragma alloc_text( PAGE, RestartStartSend )
|
|||
|
#pragma alloc_text( PAGE, GetIpxMaxBufferSize )
|
|||
|
|
|||
|
#ifdef SRV_PNP_POWER
|
|||
|
#pragma alloc_text( PAGE, SrvPnpProcessor )
|
|||
|
#endif
|
|||
|
|
|||
|
#endif
|
|||
|
#if 0
|
|||
|
NOT PAGEABLE -- SrvOpenConnection
|
|||
|
NOT PAGEABLE -- SrvPrepareReceiveWorkItem
|
|||
|
NOT PAGEABLE -- SrvStartSend
|
|||
|
NOT PAGEABLE -- SrvStartSend2
|
|||
|
#endif
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SrvAddServedNet(
|
|||
|
IN PUNICODE_STRING NetworkName,
|
|||
|
IN PUNICODE_STRING TransportName,
|
|||
|
IN PANSI_STRING TransportAddress,
|
|||
|
IN PUNICODE_STRING DomainName,
|
|||
|
IN ULONG fPrimaryMach
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function initializes the server on a network. This
|
|||
|
involves making the server known by creating a transport endpoint,
|
|||
|
posting a Listen request, and setting up event handlers.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
NetworkName - The administrative name of the network (e.g., NET1)
|
|||
|
|
|||
|
TransportName - The fully qualified name of the transport device.
|
|||
|
For example, "\Device\Nbf".
|
|||
|
|
|||
|
TransportAddress - The fully qualified address (or name ) of the
|
|||
|
server's endpoint. This name is used exactly as specified. For
|
|||
|
NETBIOS-compatible networks, the caller must upcase and
|
|||
|
blank-fill the name. For example, "NTSERVERbbbbbbbb".
|
|||
|
|
|||
|
DomainName - The name of the domain to service
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Indicates whether the network was successfully started.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
PENDPOINT endpoint;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) KdPrint(( "SrvAddServedNet entered\n" ));
|
|||
|
|
|||
|
//
|
|||
|
// Call OpenEndpoint to open the transport provider, bind to the
|
|||
|
// server address, and register the FSD receive event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = OpenEndpoint(
|
|||
|
&endpoint,
|
|||
|
NetworkName,
|
|||
|
TransportName,
|
|||
|
TransportAddress,
|
|||
|
DomainName,
|
|||
|
fPrimaryMach,
|
|||
|
FALSE); // primary endpoint
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
|
|||
|
IF_DEBUG(ERRORS) {
|
|||
|
KdPrint(( "SrvAddServedNet: unable to open endpoint \"%wZ%Z\", "
|
|||
|
"status %X\n", TransportName, TransportAddress, status ));
|
|||
|
}
|
|||
|
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Dereference the endpoint. (When it was created, the reference
|
|||
|
// count was incremented to account for our pointer.)
|
|||
|
//
|
|||
|
|
|||
|
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
|
|||
|
if (fPrimaryMach) {
|
|||
|
NTSTATUS LocalStatus;
|
|||
|
|
|||
|
//
|
|||
|
// Call OpenEndpoint to open the transport provider, bind to the
|
|||
|
// server address, and register the FSD receive event handler. This is
|
|||
|
// the auxillary endpoint registration in the new TDI address format. SInce
|
|||
|
// this is not supported by all the transports it cannot be deemed an error.
|
|||
|
//
|
|||
|
//
|
|||
|
|
|||
|
LocalStatus = OpenEndpoint(
|
|||
|
&endpoint,
|
|||
|
NetworkName,
|
|||
|
TransportName,
|
|||
|
TransportAddress,
|
|||
|
DomainName,
|
|||
|
fPrimaryMach,
|
|||
|
TRUE); // Alternate endpoint
|
|||
|
|
|||
|
if ( !NT_SUCCESS(LocalStatus) ) {
|
|||
|
|
|||
|
IF_DEBUG(ERRORS) {
|
|||
|
KdPrint(( "SrvAddServedNet: unable to open endpoint \"%wZ%Z\", "
|
|||
|
"status %X\n", TransportName, TransportAddress, LocalStatus ));
|
|||
|
}
|
|||
|
} else {
|
|||
|
SrvDereferenceEndpoint( endpoint );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) {
|
|||
|
KdPrint(( "SrvAddServedNet complete: %X\n", STATUS_SUCCESS ));
|
|||
|
}
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
} // SrvAddServedNet
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SrvDeleteServedNet(
|
|||
|
IN PUNICODE_STRING TransportName,
|
|||
|
IN PANSI_STRING TransportAddress
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function causes the server to stop listening to a network.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
TransportAddress - the transport address (e.g. \Device\Nbf\POPCORN
|
|||
|
of the endpoint to delete.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Indicates whether the network was successfully stopped.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PLIST_ENTRY listEntry;
|
|||
|
PENDPOINT endpoint;
|
|||
|
BOOLEAN match;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) KdPrint(( "SrvDeleteServedNet entered\n" ));
|
|||
|
|
|||
|
//
|
|||
|
// Find the endpoint block with the specified name.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
listEntry = SrvEndpointList.ListHead.Flink;
|
|||
|
|
|||
|
while ( listEntry != &SrvEndpointList.ListHead ) {
|
|||
|
|
|||
|
endpoint = CONTAINING_RECORD(
|
|||
|
listEntry,
|
|||
|
ENDPOINT,
|
|||
|
GlobalEndpointListEntry
|
|||
|
);
|
|||
|
|
|||
|
match = (BOOLEAN)(
|
|||
|
RtlEqualUnicodeString(
|
|||
|
TransportName,
|
|||
|
&endpoint->TransportName,
|
|||
|
TRUE // case insensitive compare
|
|||
|
)
|
|||
|
&&
|
|||
|
RtlEqualString(
|
|||
|
(PSTRING)TransportAddress,
|
|||
|
(PSTRING)&endpoint->TransportAddress,
|
|||
|
TRUE // case insensitive compare
|
|||
|
)
|
|||
|
);
|
|||
|
|
|||
|
if ( match ) {
|
|||
|
|
|||
|
//
|
|||
|
// The specified network name (endpoint) exists. Close the
|
|||
|
// endpoint. This releases the endpoint lock.
|
|||
|
//
|
|||
|
|
|||
|
SrvCloseEndpoint( endpoint );
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// The current endpoint's name doesn't match. Get the next one.
|
|||
|
//
|
|||
|
|
|||
|
listEntry = listEntry->Flink;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// No matching endpoint was found.
|
|||
|
//
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
return STATUS_NONEXISTENT_NET_NAME;
|
|||
|
|
|||
|
} // SrvDeleteServedNet
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SrvDoDisconnect (
|
|||
|
IN OUT PCONNECTION Connection
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function issues a Disconnect request on a network. The request
|
|||
|
is performed synchronously -- control is not returned to the caller
|
|||
|
until the request completes.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Connection - Supplies a pointer to an Connection Block
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Indicates whether the disconnect was successful.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvDoDisconnect entered\n" ));
|
|||
|
#if SRVDBG29
|
|||
|
UpdateConnectionHistory( "SDSC", Connection->Endpoint, Connection );
|
|||
|
#endif
|
|||
|
|
|||
|
ASSERT( !Connection->Endpoint->IsConnectionless );
|
|||
|
|
|||
|
//
|
|||
|
// Issue the disconnect request.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueDisconnectRequest(
|
|||
|
Connection->FileObject,
|
|||
|
&Connection->DeviceObject,
|
|||
|
TDI_DISCONNECT_ABORT
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"SrvDoDisconnect: NtDeviceIoControlFile failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
#if SRVDBG29
|
|||
|
if (status != STATUS_LINK_FAILED && status != STATUS_REMOTE_DISCONNECT) {
|
|||
|
KdPrint(( "SRV: SrvDoDisconnect: SrvIssueDisconnectRequest failed\n" ));
|
|||
|
DbgBreakPoint();
|
|||
|
}
|
|||
|
#endif
|
|||
|
//
|
|||
|
// Mark the connection as not reusable, because the transport
|
|||
|
// probably still thinks it's active.
|
|||
|
//
|
|||
|
|
|||
|
Connection->NotReusable = TRUE;
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Return the status of the I/O operation.
|
|||
|
//
|
|||
|
|
|||
|
return status;
|
|||
|
|
|||
|
} // SrvDoDisconnect
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SrvOpenConnection (
|
|||
|
IN PENDPOINT Endpoint
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function opens a connection for an endpoint and queues it to
|
|||
|
the endpoint's free connection list.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Endpoint - Supplies a pointer to an Endpoint Block
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Indicates whether the connection was successfully opened.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
PCONNECTION connection;
|
|||
|
PPAGED_CONNECTION pagedConnection;
|
|||
|
CHAR eaBuffer[sizeof(FILE_FULL_EA_INFORMATION) - 1 +
|
|||
|
TDI_CONNECTION_CONTEXT_LENGTH + 1 +
|
|||
|
sizeof(CONNECTION_CONTEXT)];
|
|||
|
PFILE_FULL_EA_INFORMATION ea;
|
|||
|
CONNECTION_CONTEXT UNALIGNED *ctx;
|
|||
|
OBJECT_ATTRIBUTES objectAttributes;
|
|||
|
IO_STATUS_BLOCK iosb;
|
|||
|
KIRQL oldIrql;
|
|||
|
PTABLE_HEADER tableHeader;
|
|||
|
SHORT sidIndex;
|
|||
|
CSHORT i;
|
|||
|
PTABLE_ENTRY entry = NULL;
|
|||
|
TDI_PROVIDER_INFO providerInfo;
|
|||
|
|
|||
|
//
|
|||
|
// Allocate a connection block.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateConnection( &connection );
|
|||
|
|
|||
|
if ( connection == NULL ) {
|
|||
|
return STATUS_INSUFF_SERVER_RESOURCES;
|
|||
|
}
|
|||
|
|
|||
|
pagedConnection = connection->PagedConnection;
|
|||
|
|
|||
|
//
|
|||
|
// Allocate an entry in the endpoint's connection table.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(0), &oldIrql );
|
|||
|
for ( i = 1; i < ENDPOINT_LOCK_COUNT ; i++ ) {
|
|||
|
ACQUIRE_DPC_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(i) );
|
|||
|
}
|
|||
|
|
|||
|
tableHeader = &Endpoint->ConnectionTable;
|
|||
|
|
|||
|
if ( tableHeader->FirstFreeEntry == -1 &&
|
|||
|
SrvGrowTable(
|
|||
|
tableHeader,
|
|||
|
8,
|
|||
|
0x7fff ) == FALSE ) {
|
|||
|
|
|||
|
for ( i = ENDPOINT_LOCK_COUNT-1 ; i > 0 ; i-- ) {
|
|||
|
RELEASE_DPC_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(i) );
|
|||
|
}
|
|||
|
RELEASE_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(0), oldIrql );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
sidIndex = tableHeader->FirstFreeEntry;
|
|||
|
entry = &tableHeader->Table[sidIndex];
|
|||
|
tableHeader->FirstFreeEntry = entry->NextFreeEntry;
|
|||
|
DEBUG entry->NextFreeEntry = -2;
|
|||
|
if ( tableHeader->LastFreeEntry == sidIndex ) {
|
|||
|
tableHeader->LastFreeEntry = -1;
|
|||
|
}
|
|||
|
|
|||
|
for ( i = ENDPOINT_LOCK_COUNT-1 ; i > 0 ; i-- ) {
|
|||
|
RELEASE_DPC_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(i) );
|
|||
|
}
|
|||
|
RELEASE_SPIN_LOCK( &ENDPOINT_SPIN_LOCK(0), oldIrql );
|
|||
|
|
|||
|
if ( !Endpoint->IsConnectionless ) {
|
|||
|
|
|||
|
//
|
|||
|
// Create the EA for the connection context.
|
|||
|
//
|
|||
|
|
|||
|
ea = (PFILE_FULL_EA_INFORMATION)eaBuffer;
|
|||
|
ea->NextEntryOffset = 0;
|
|||
|
ea->Flags = 0;
|
|||
|
ea->EaNameLength = TDI_CONNECTION_CONTEXT_LENGTH;
|
|||
|
ea->EaValueLength = sizeof(CONNECTION_CONTEXT);
|
|||
|
|
|||
|
RtlCopyMemory( ea->EaName, StrConnectionContext, ea->EaNameLength + 1 );
|
|||
|
|
|||
|
ctx = (CONNECTION_CONTEXT UNALIGNED *)&ea->EaName[ea->EaNameLength + 1];
|
|||
|
*ctx = connection;
|
|||
|
|
|||
|
//
|
|||
|
// Create the connection file object.
|
|||
|
//
|
|||
|
|
|||
|
SrvInitializeObjectAttributes_U(
|
|||
|
&objectAttributes,
|
|||
|
&Endpoint->TransportName,
|
|||
|
OBJ_CASE_INSENSITIVE,
|
|||
|
NULL,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
status = NtCreateFile(
|
|||
|
&pagedConnection->ConnectionHandle,
|
|||
|
0,
|
|||
|
&objectAttributes,
|
|||
|
&iosb,
|
|||
|
NULL,
|
|||
|
0,
|
|||
|
0,
|
|||
|
0,
|
|||
|
0,
|
|||
|
eaBuffer,
|
|||
|
FIELD_OFFSET( FILE_FULL_EA_INFORMATION, EaName[0] ) +
|
|||
|
ea->EaNameLength + 1 + ea->EaValueLength
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
IF_DEBUG(ERRORS) {
|
|||
|
KdPrint(( "SrvOpenConnection: NtCreateFile failed: %X\n", status ));
|
|||
|
}
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
SRVDBG_CLAIM_HANDLE( pagedConnection->ConnectionHandle, "CON", 7, connection );
|
|||
|
|
|||
|
//
|
|||
|
// Obtain a referenced pointer to the file object.
|
|||
|
//
|
|||
|
|
|||
|
status = ObReferenceObjectByHandle(
|
|||
|
pagedConnection->ConnectionHandle,
|
|||
|
0,
|
|||
|
(POBJECT_TYPE) NULL,
|
|||
|
KernelMode,
|
|||
|
(PVOID *)&connection->FileObject,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_OB_REF_BY_HANDLE, status );
|
|||
|
|
|||
|
//
|
|||
|
// This internal error bugchecks the system.
|
|||
|
//
|
|||
|
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_IMPOSSIBLE,
|
|||
|
"SrvOpenConnection: ObReferenceObjectByHandle failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
goto cleanup;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Get the address of the device object for the endpoint.
|
|||
|
//
|
|||
|
|
|||
|
connection->DeviceObject = IoGetRelatedDeviceObject(
|
|||
|
connection->FileObject
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Associate the connection with the endpoint's address.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueAssociateRequest(
|
|||
|
connection->FileObject,
|
|||
|
&connection->DeviceObject,
|
|||
|
Endpoint->EndpointHandle
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_UNEXPECTED,
|
|||
|
"SrvOpenConnection: SrvIssueAssociateRequest failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
IF_DEBUG(NET1) {
|
|||
|
KdPrint(( "SrvOpenConnection: Connection on \"%wZ%Z\" opened; handle %lx, "
|
|||
|
"pointer %lx\n", &Endpoint->TransportName,
|
|||
|
&Endpoint->TransportAddress,
|
|||
|
connection->PagedConnection->ConnectionHandle,
|
|||
|
connection->FileObject ));
|
|||
|
}
|
|||
|
|
|||
|
} // if ( !Endpoint->IsConnectionless )
|
|||
|
|
|||
|
//
|
|||
|
// Initialize the MaximumSendSize for the transport that we're using
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueTdiQuery(
|
|||
|
connection->FileObject,
|
|||
|
&connection->DeviceObject,
|
|||
|
(PCHAR)&providerInfo,
|
|||
|
sizeof(providerInfo),
|
|||
|
TDI_QUERY_PROVIDER_INFO
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// If we got the provider info, make sure the maximum send size is at
|
|||
|
// least 1K-1. If we have no provider info, then maximum send size is 64KB.
|
|||
|
//
|
|||
|
|
|||
|
if ( NT_SUCCESS(status) ) {
|
|||
|
connection->MaximumSendSize = providerInfo.MaxSendSize;
|
|||
|
if ( connection->MaximumSendSize < MIN_SEND_SIZE ) {
|
|||
|
connection->MaximumSendSize = MIN_SEND_SIZE;
|
|||
|
}
|
|||
|
} else {
|
|||
|
connection->MaximumSendSize = MAX_PARTIAL_BUFFER_SIZE;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set the reference count on the connection to zero, in order to
|
|||
|
// put it on the free list. (SrvAllocateConnection initialized the
|
|||
|
// count to two.)
|
|||
|
//
|
|||
|
|
|||
|
connection->BlockHeader.ReferenceCount = 0;
|
|||
|
|
|||
|
UPDATE_REFERENCE_HISTORY( connection, TRUE );
|
|||
|
UPDATE_REFERENCE_HISTORY( connection, TRUE );
|
|||
|
|
|||
|
//
|
|||
|
// Reference the endpoint and link the connection into the
|
|||
|
// endpoint's free connection list.
|
|||
|
//
|
|||
|
|
|||
|
connection->Endpoint = Endpoint;
|
|||
|
connection->EndpointSpinLock =
|
|||
|
&ENDPOINT_SPIN_LOCK(sidIndex & ENDPOINT_LOCK_MASK);
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
SrvReferenceEndpoint( Endpoint );
|
|||
|
|
|||
|
ACQUIRE_SPIN_LOCK( connection->EndpointSpinLock, &oldIrql );
|
|||
|
INCREMENT_IPXSID_SEQUENCE( entry->SequenceNumber );
|
|||
|
if ( sidIndex == 0 && entry->SequenceNumber == 0 ) {
|
|||
|
INCREMENT_IPXSID_SEQUENCE( entry->SequenceNumber );
|
|||
|
}
|
|||
|
|
|||
|
connection->Sid = MAKE_IPXSID( sidIndex, entry->SequenceNumber );
|
|||
|
|
|||
|
entry->Owner = connection;
|
|||
|
RELEASE_SPIN_LOCK( connection->EndpointSpinLock, oldIrql );
|
|||
|
|
|||
|
ACQUIRE_GLOBAL_SPIN_LOCK( Fsd, &oldIrql );
|
|||
|
SrvInsertTailList(
|
|||
|
&Endpoint->FreeConnectionList,
|
|||
|
&connection->EndpointFreeListEntry
|
|||
|
);
|
|||
|
#if SRVDBG29
|
|||
|
UpdateConnectionHistory( "OPEN", Endpoint, connection );
|
|||
|
#endif
|
|||
|
Endpoint->FreeConnectionCount++;
|
|||
|
Endpoint->TotalConnectionCount++;
|
|||
|
|
|||
|
RELEASE_GLOBAL_SPIN_LOCK( Fsd, oldIrql );
|
|||
|
|
|||
|
IF_DEBUG(TDI) {
|
|||
|
KdPrint(( "SrvOpenConnection created connection %lx for endpoint "
|
|||
|
"%lx; free %ld, total %ld\n", connection, Endpoint,
|
|||
|
Endpoint->FreeConnectionCount,
|
|||
|
Endpoint->TotalConnectionCount ));
|
|||
|
}
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// The connection open was successful.
|
|||
|
//
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) {
|
|||
|
KdPrint(( "SrvOpenConnection complete: %X\n", STATUS_SUCCESS ));
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
//
|
|||
|
// Out-of-line error cleanup.
|
|||
|
//
|
|||
|
|
|||
|
cleanup:
|
|||
|
|
|||
|
//
|
|||
|
// Something failed. Clean up as appropriate.
|
|||
|
//
|
|||
|
|
|||
|
if ( !Endpoint->IsConnectionless ) {
|
|||
|
if ( connection->FileObject != NULL ) {
|
|||
|
ObDereferenceObject( connection->FileObject );
|
|||
|
}
|
|||
|
if ( pagedConnection->ConnectionHandle != NULL ) {
|
|||
|
SRVDBG_RELEASE_HANDLE( pagedConnection->ConnectionHandle, "CON", 12, connection );
|
|||
|
SrvNtClose( pagedConnection->ConnectionHandle, FALSE );
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
if ( entry != NULL ) {
|
|||
|
SrvRemoveEntryTable( tableHeader, sidIndex );
|
|||
|
}
|
|||
|
|
|||
|
SrvFreeConnection( connection );
|
|||
|
|
|||
|
return status;
|
|||
|
|
|||
|
} // SrvOpenConnection
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
GetNetworkAddress (
|
|||
|
IN PENDPOINT Endpoint
|
|||
|
)
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
PCHAR adapterStatus;
|
|||
|
PCHAR adapterAddress;
|
|||
|
ANSI_STRING ansiString;
|
|||
|
CHAR addressData[12+1];
|
|||
|
ULONG i;
|
|||
|
|
|||
|
struct {
|
|||
|
ULONG ActivityCount;
|
|||
|
TA_IPX_ADDRESS LocalAddress;
|
|||
|
} addressInfo;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
if ( !Endpoint->IsConnectionless ) {
|
|||
|
|
|||
|
//
|
|||
|
// Allocate a buffer to receive adapter information.
|
|||
|
//
|
|||
|
// *** We want to get the ADAPTER_STATUS structure, but it is
|
|||
|
// defined in the windows header file sdk\inc\nb30.h.
|
|||
|
// Rather than including all the windows header files, just
|
|||
|
// allocate about a page, which should always be enough for
|
|||
|
// that structure.
|
|||
|
//
|
|||
|
|
|||
|
adapterStatus = ALLOCATE_NONPAGED_POOL( 4080, BlockTypeDataBuffer );
|
|||
|
if ( adapterStatus == NULL ) {
|
|||
|
return STATUS_INSUFF_SERVER_RESOURCES;
|
|||
|
}
|
|||
|
|
|||
|
status = SrvIssueTdiQuery(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
adapterStatus,
|
|||
|
4080,
|
|||
|
TDI_QUERY_ADAPTER_STATUS
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_UNEXPECTED,
|
|||
|
"GetNetworkAddress: SrvIssueTdiQuery failed: %X\n",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
DEALLOCATE_NONPAGED_POOL( adapterStatus );
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
adapterAddress = adapterStatus;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
status = SrvIssueTdiQuery(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)&addressInfo,
|
|||
|
sizeof(addressInfo),
|
|||
|
TDI_QUERY_ADDRESS_INFO
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_UNEXPECTED,
|
|||
|
"GetNetworkAddress: SrvIssueTdiQuery failed: %X\n",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
Endpoint->LocalAddress = addressInfo.LocalAddress.Address[0].Address[0];
|
|||
|
|
|||
|
adapterAddress = addressInfo.LocalAddress.Address[0].Address[0].NodeAddress;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Get an ANSI string that contains the adapter address.
|
|||
|
//
|
|||
|
|
|||
|
ansiString.Buffer = addressData;
|
|||
|
ansiString.Length = 12;
|
|||
|
ansiString.MaximumLength = 13;
|
|||
|
|
|||
|
#define tohexdigit(a) ((CHAR)( (a) > 9 ? ((a) + 'a' - 0xA) : ((a) + '0') ))
|
|||
|
|
|||
|
for ( i = 0; i < 6; i++ ) {
|
|||
|
addressData[2*i] = tohexdigit( (adapterAddress[i] >> 4) & 0x0F );
|
|||
|
addressData[2*i+1] = tohexdigit( adapterAddress[i] & 0x0F );
|
|||
|
}
|
|||
|
|
|||
|
addressData[12] = '\0';
|
|||
|
|
|||
|
//
|
|||
|
// Convert the address string to Unicode.
|
|||
|
//
|
|||
|
|
|||
|
status = RtlAnsiStringToUnicodeString(
|
|||
|
&Endpoint->NetworkAddress,
|
|||
|
&ansiString,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
ASSERT( NT_SUCCESS(status) );
|
|||
|
|
|||
|
if ( !Endpoint->IsConnectionless ) {
|
|||
|
DEALLOCATE_NONPAGED_POOL( adapterStatus );
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
} // GetNetworkAddress
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenEndpoint (
|
|||
|
OUT PENDPOINT *Endpoint,
|
|||
|
IN PUNICODE_STRING NetworkName,
|
|||
|
IN PUNICODE_STRING TransportName,
|
|||
|
IN PANSI_STRING TransportAddress,
|
|||
|
IN PUNICODE_STRING DomainName,
|
|||
|
IN DWORD PrimaryMachine,
|
|||
|
IN BOOLEAN AlternateEndpoint
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function opens a transport provider, simultaneously binding the
|
|||
|
server's address to the transport endpoint, and registers a Receive
|
|||
|
event handler for the endpoint.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Endpoint - Returns a pointer to an Endpoint Block
|
|||
|
|
|||
|
NetworkName - Supplies the administrative name of the network (e.g.,
|
|||
|
NET1).
|
|||
|
|
|||
|
TransportName - The fully qualified name of the transport device.
|
|||
|
For example, "\Device\Nbf".
|
|||
|
|
|||
|
TransportAddress - The exact name of the server to be used on the
|
|||
|
specified transport. For NETBIOS-compatible networks, the
|
|||
|
caller must upcase and blank-fill the name. For example,
|
|||
|
"NTSERVERbbbbbbbb".
|
|||
|
|
|||
|
DomainName - name of domain to serve
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
NTSTATUS - Indicates whether the network was successfully opened.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
PENDPOINT endpoint = NULL; // local copy of Endpoint
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) KdPrint(( "OpenEndpoint entered\n" ));
|
|||
|
|
|||
|
//
|
|||
|
// Allocate an endpoint block.
|
|||
|
//
|
|||
|
|
|||
|
SrvAllocateEndpoint(
|
|||
|
&endpoint,
|
|||
|
NetworkName,
|
|||
|
TransportName,
|
|||
|
TransportAddress,
|
|||
|
DomainName
|
|||
|
);
|
|||
|
|
|||
|
if ( endpoint == NULL ) {
|
|||
|
IF_DEBUG(TRACE1) {
|
|||
|
KdPrint(( "OpenEndpoint complete: %X\n",
|
|||
|
STATUS_INSUFF_SERVER_RESOURCES ));
|
|||
|
}
|
|||
|
return STATUS_INSUFF_SERVER_RESOURCES;
|
|||
|
}
|
|||
|
|
|||
|
if(PrimaryMachine)
|
|||
|
{
|
|||
|
endpoint->IsPrimaryName = 1;
|
|||
|
}
|
|||
|
|
|||
|
if (AlternateEndpoint) {
|
|||
|
status = OpenNetbiosExAddress(
|
|||
|
endpoint,
|
|||
|
TransportName,
|
|||
|
TransportAddress->Buffer);
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// Assume that the transport is a NetBIOS provider, and try to
|
|||
|
// open the server's address using the NetBIOS name.
|
|||
|
//
|
|||
|
|
|||
|
status = OpenNetbiosAddress(
|
|||
|
endpoint,
|
|||
|
TransportName,
|
|||
|
TransportAddress->Buffer
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
|
|||
|
BOOLEAN isDuplicate = FALSE;
|
|||
|
PLIST_ENTRY listEntry;
|
|||
|
|
|||
|
//
|
|||
|
// Apparently the transport is not a NetBIOS provider. We can
|
|||
|
// not open multiple connectionless providers through the same
|
|||
|
// TransportName.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
for( listEntry = SrvEndpointList.ListHead.Flink;
|
|||
|
listEntry != &SrvEndpointList.ListHead;
|
|||
|
listEntry = listEntry->Flink ) {
|
|||
|
|
|||
|
PENDPOINT tmpEndpoint;
|
|||
|
|
|||
|
tmpEndpoint = CONTAINING_RECORD( listEntry, ENDPOINT, GlobalEndpointListEntry );
|
|||
|
|
|||
|
if( GET_BLOCK_STATE( tmpEndpoint ) == BlockStateActive &&
|
|||
|
tmpEndpoint->IsConnectionless &&
|
|||
|
RtlCompareUnicodeString( &tmpEndpoint->TransportName, TransportName, TRUE ) == 0 ) {
|
|||
|
|
|||
|
IF_DEBUG(ERRORS) {
|
|||
|
KdPrint(( "OpenEndpoint: Only one connectionless endpoint on %wZ allowed!\n",
|
|||
|
TransportName ));
|
|||
|
}
|
|||
|
|
|||
|
isDuplicate = TRUE;
|
|||
|
status = STATUS_TOO_MANY_NODES;
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
//
|
|||
|
// Try to open it as a connectionless provider.
|
|||
|
//
|
|||
|
if( isDuplicate == FALSE ) {
|
|||
|
status = OpenNonNetbiosAddress(
|
|||
|
endpoint,
|
|||
|
TransportName,
|
|||
|
TransportAddress->Buffer
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
|
|||
|
//
|
|||
|
// We couldn't open the provider as either a NetBIOS provider
|
|||
|
// or as a connectionless provider.
|
|||
|
//
|
|||
|
|
|||
|
IF_DEBUG(ERRORS) {
|
|||
|
KdPrint(( "OpenEndpoint: OpenAddress failed: %X\n", status ));
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close all free connections.
|
|||
|
//
|
|||
|
|
|||
|
EmptyFreeConnectionList( endpoint );
|
|||
|
|
|||
|
SrvFreeEndpoint( endpoint );
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
SrvEndpointCount--;
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Query the provider for the send entry point
|
|||
|
//
|
|||
|
|
|||
|
SrvQuerySendEntryPoint(
|
|||
|
endpoint->FileObject,
|
|||
|
&endpoint->DeviceObject,
|
|||
|
IOCTL_TDI_QUERY_DIRECT_SEND_HANDLER,
|
|||
|
(PVOID*)&endpoint->FastTdiSend
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Query the provider for the send entry point
|
|||
|
//
|
|||
|
|
|||
|
SrvQuerySendEntryPoint(
|
|||
|
endpoint->FileObject,
|
|||
|
&endpoint->DeviceObject,
|
|||
|
IOCTL_TDI_QUERY_DIRECT_SENDDG_HANDLER,
|
|||
|
(PVOID*)&endpoint->FastTdiSendDatagram
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// The network open was successful. Link the new endpoint into the
|
|||
|
// list of active endpoints. Return with a success status. (We
|
|||
|
// don't dereference the endpoint because we're returning a pointer
|
|||
|
// to the endpoint.)
|
|||
|
//
|
|||
|
|
|||
|
SrvInsertEntryOrderedList( &SrvEndpointList, endpoint );
|
|||
|
|
|||
|
*Endpoint = endpoint;
|
|||
|
|
|||
|
IF_DEBUG(TRACE1) {
|
|||
|
KdPrint(( "OpenEndpoint complete: %X\n", STATUS_SUCCESS ));
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
} // OpenEndpoint
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
SetupConnectionEndpointHandlers(
|
|||
|
IN OUT PENDPOINT Endpoint)
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
ULONG i;
|
|||
|
|
|||
|
Endpoint->IsConnectionless = FALSE;
|
|||
|
|
|||
|
status = SrvVerifyDeviceStackSize(
|
|||
|
Endpoint->EndpointHandle,
|
|||
|
TRUE,
|
|||
|
&Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS( status ) ) {
|
|||
|
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNetbiosAddress: Verify Device Stack Size failed: %X\n",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Find the network address of the adapter used by corresponding to
|
|||
|
// this endpoint.
|
|||
|
//
|
|||
|
|
|||
|
GetNetworkAddress( Endpoint );
|
|||
|
|
|||
|
//
|
|||
|
// Register the server's Receive event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
TDI_EVENT_RECEIVE,
|
|||
|
(PVOID)SrvFsdTdiReceiveHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNetbiosAddress: set receive event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Register the server's Disconnect event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
TDI_EVENT_DISCONNECT,
|
|||
|
(PVOID)SrvFsdTdiDisconnectHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_UNEXPECTED,
|
|||
|
"OpenNetbiosAddress: set disconnect event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Create a number of free connections for the endpoint. These
|
|||
|
// connections will be used to service Connect events.
|
|||
|
//
|
|||
|
// *** If we fail in an attempt to create a connection, but we can
|
|||
|
// successfully create at least one, we keep the endpoint. The
|
|||
|
// cleanup code below depends on this behavior.
|
|||
|
//
|
|||
|
|
|||
|
for ( i = 0; i < SrvFreeConnectionMinimum; i++ ) {
|
|||
|
|
|||
|
status = SrvOpenConnection( Endpoint );
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNetbiosAddress: SrvOpenConnection failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
if ( i == 0 ) {
|
|||
|
goto cleanup;
|
|||
|
} else {
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Register the server's Connect event handler.
|
|||
|
//
|
|||
|
// *** Note that Connect events can be delivered IMMEDIATELY upon
|
|||
|
// completion of this request!
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
TDI_EVENT_CONNECT,
|
|||
|
(PVOID)SrvFsdTdiConnectHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_UNEXPECTED,
|
|||
|
"OpenNetbiosAddress: set connect event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
//
|
|||
|
// Out-of-line error cleanup.
|
|||
|
//
|
|||
|
|
|||
|
cleanup:
|
|||
|
|
|||
|
//
|
|||
|
// Something failed. Clean up as appropriate.
|
|||
|
//
|
|||
|
|
|||
|
if ( Endpoint->FileObject != NULL ) {
|
|||
|
ObDereferenceObject( Endpoint->FileObject );
|
|||
|
Endpoint->FileObject = NULL;
|
|||
|
}
|
|||
|
if ( Endpoint->EndpointHandle != NULL ) {
|
|||
|
SRVDBG_RELEASE_HANDLE( Endpoint->EndpointHandle, "END", 14, Endpoint );
|
|||
|
SrvNtClose( Endpoint->EndpointHandle, FALSE );
|
|||
|
Endpoint->EndpointHandle = NULL;
|
|||
|
}
|
|||
|
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNetbiosAddress (
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
)
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
ULONG i;
|
|||
|
|
|||
|
CHAR eaBuffer[sizeof(FILE_FULL_EA_INFORMATION) +
|
|||
|
TDI_TRANSPORT_ADDRESS_LENGTH + 1 +
|
|||
|
sizeof(TA_NETBIOS_ADDRESS)];
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
status = TdiOpenNetbiosAddress(
|
|||
|
&Endpoint->EndpointHandle,
|
|||
|
eaBuffer,
|
|||
|
DeviceName,
|
|||
|
NetbiosName
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
status = SetupConnectionEndpointHandlers(Endpoint);
|
|||
|
|
|||
|
return status;
|
|||
|
} // OpenNetbiosAddress
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNetbiosExAddress(
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
)
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
|
|||
|
PFILE_FULL_EA_INFORMATION ea;
|
|||
|
OBJECT_ATTRIBUTES objectAttributes;
|
|||
|
IO_STATUS_BLOCK iosb;
|
|||
|
|
|||
|
ULONG length;
|
|||
|
CHAR buffer[sizeof(FILE_FULL_EA_INFORMATION) +
|
|||
|
TDI_TRANSPORT_ADDRESS_LENGTH + 1 +
|
|||
|
sizeof(TA_NETBIOS_EX_ADDRESS)];
|
|||
|
|
|||
|
TA_NETBIOS_EX_ADDRESS NetbiosExAddress;
|
|||
|
PTDI_ADDRESS_NETBIOS_EX pTdiNetbiosExAddress;
|
|||
|
PTDI_ADDRESS_NETBIOS pNetbiosAddress;
|
|||
|
|
|||
|
ULONG NetbiosExAddressLength;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Build the NETBIOS Extended address.
|
|||
|
//
|
|||
|
|
|||
|
NetbiosExAddress.TAAddressCount = 1;
|
|||
|
NetbiosExAddress.Address[0].AddressLength = TDI_ADDRESS_LENGTH_NETBIOS_EX;
|
|||
|
NetbiosExAddress.Address[0].AddressType = TDI_ADDRESS_TYPE_NETBIOS_EX;
|
|||
|
|
|||
|
pTdiNetbiosExAddress = NetbiosExAddress.Address[0].Address;
|
|||
|
pNetbiosAddress = &pTdiNetbiosExAddress->NetbiosAddress;
|
|||
|
pNetbiosAddress->NetbiosNameType = TDI_ADDRESS_NETBIOS_TYPE_UNIQUE;
|
|||
|
|
|||
|
NetbiosExAddressLength = FIELD_OFFSET(TRANSPORT_ADDRESS,Address)
|
|||
|
+ FIELD_OFFSET(TA_ADDRESS,Address)
|
|||
|
+ FIELD_OFFSET(TDI_ADDRESS_NETBIOS_EX,NetbiosAddress)
|
|||
|
+ TDI_ADDRESS_LENGTH_NETBIOS;
|
|||
|
|
|||
|
RtlCopyMemory(
|
|||
|
pNetbiosAddress->NetbiosName,
|
|||
|
NetbiosName,
|
|||
|
NETBIOS_NAME_LEN);
|
|||
|
|
|||
|
// Copy the default endpoint name onto the NETBIOS Extended address.
|
|||
|
RtlCopyMemory(
|
|||
|
pTdiNetbiosExAddress->EndpointName,
|
|||
|
SMBSERVER_LOCAL_ENDPOINT_NAME,
|
|||
|
NETBIOS_NAME_LEN);
|
|||
|
|
|||
|
length = FIELD_OFFSET( FILE_FULL_EA_INFORMATION, EaName[0] ) +
|
|||
|
TDI_TRANSPORT_ADDRESS_LENGTH + 1 +
|
|||
|
NetbiosExAddressLength;
|
|||
|
ea = (PFILE_FULL_EA_INFORMATION)buffer;
|
|||
|
|
|||
|
ea->NextEntryOffset = 0;
|
|||
|
ea->Flags = 0;
|
|||
|
ea->EaNameLength = TDI_TRANSPORT_ADDRESS_LENGTH;
|
|||
|
ea->EaValueLength = (USHORT)NetbiosExAddressLength;
|
|||
|
|
|||
|
RtlCopyMemory( ea->EaName, StrTransportAddress, ea->EaNameLength + 1 );
|
|||
|
|
|||
|
RtlCopyMemory(
|
|||
|
&ea->EaName[ea->EaNameLength + 1],
|
|||
|
&NetbiosExAddress,
|
|||
|
NetbiosExAddressLength
|
|||
|
);
|
|||
|
|
|||
|
InitializeObjectAttributes( &objectAttributes, DeviceName, OBJ_CASE_INSENSITIVE, NULL, NULL );
|
|||
|
|
|||
|
status = NtCreateFile (
|
|||
|
&Endpoint->EndpointHandle,
|
|||
|
FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES, // desired access
|
|||
|
&objectAttributes, // object attributes
|
|||
|
&iosb, // returned status information
|
|||
|
NULL, // block size (unused)
|
|||
|
0, // file attributes
|
|||
|
FILE_SHARE_READ | FILE_SHARE_WRITE, // share access
|
|||
|
FILE_CREATE, // create disposition
|
|||
|
0, // create options
|
|||
|
buffer, // EA buffer
|
|||
|
length // EA length
|
|||
|
);
|
|||
|
|
|||
|
IF_DEBUG(TDI) KdPrint(("Opening NETBIOS_EX address returns %lx\n",status));
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
Endpoint->IsNoNetBios = TRUE;
|
|||
|
status = SetupConnectionEndpointHandlers(Endpoint);
|
|||
|
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenNonNetbiosAddress (
|
|||
|
IN OUT PENDPOINT Endpoint,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN PVOID NetbiosName
|
|||
|
)
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
ULONG i;
|
|||
|
ULONG numAdapters;
|
|||
|
PULONG maxPktArray = NULL;
|
|||
|
UCHAR buffer[sizeof(NWLINK_ACTION) + sizeof(IPX_ADDRESS_DATA) - 1];
|
|||
|
PNWLINK_ACTION action;
|
|||
|
PIPX_ADDRESS_DATA ipxAddressData;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Open the NetBIOS name socket.
|
|||
|
//
|
|||
|
|
|||
|
status = OpenIpxSocket(
|
|||
|
&Endpoint->NameSocketHandle,
|
|||
|
&Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
DeviceName,
|
|||
|
SMB_IPX_NAME_SOCKET
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
Endpoint->IsConnectionless = TRUE;
|
|||
|
action = (PNWLINK_ACTION)buffer;
|
|||
|
|
|||
|
//
|
|||
|
// Put the endpoint in broadcast reception mode.
|
|||
|
//
|
|||
|
|
|||
|
action->Header.TransportId = 'XPIM'; // "MIPX"
|
|||
|
action->Header.ActionCode = 0;
|
|||
|
action->Header.Reserved = 0;
|
|||
|
action->OptionType = NWLINK_OPTION_ADDRESS;
|
|||
|
action->BufferLength = sizeof(action->Option);
|
|||
|
action->Option = MIPX_RCVBCAST;
|
|||
|
|
|||
|
status = SrvIssueTdiAction(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)action,
|
|||
|
sizeof(NWLINK_ACTION)
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Tell the transport to give you the extended receive info
|
|||
|
//
|
|||
|
|
|||
|
action->Header.TransportId = 'XPIM'; // "MIPX"
|
|||
|
action->Header.ActionCode = 0;
|
|||
|
action->Header.Reserved = 0;
|
|||
|
action->OptionType = NWLINK_OPTION_ADDRESS;
|
|||
|
action->BufferLength = sizeof(action->Option);
|
|||
|
action->Option = MIPX_SETRCVFLAGS;
|
|||
|
|
|||
|
status = SrvIssueTdiAction(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)action,
|
|||
|
sizeof(NWLINK_ACTION)
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Get the max adapter number
|
|||
|
//
|
|||
|
|
|||
|
action->Header.TransportId = 'XPIM'; // "MIPX"
|
|||
|
action->Header.ActionCode = 0;
|
|||
|
action->Header.Reserved = 0;
|
|||
|
action->OptionType = NWLINK_OPTION_ADDRESS;
|
|||
|
action->BufferLength = sizeof(action->Option) + sizeof(ULONG);
|
|||
|
action->Option = MIPX_ADAPTERNUM2;
|
|||
|
|
|||
|
status = SrvIssueTdiAction(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)action,
|
|||
|
sizeof(NWLINK_ACTION) + sizeof(ULONG) - 1
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
numAdapters = *((PULONG)action->Data);
|
|||
|
|
|||
|
//
|
|||
|
// Allocate an array to store the max pkt size for each adapter
|
|||
|
//
|
|||
|
|
|||
|
maxPktArray = ALLOCATE_HEAP( numAdapters * sizeof(ULONG), BlockTypeBuffer );
|
|||
|
|
|||
|
if ( maxPktArray == NULL ) {
|
|||
|
status = STATUS_INSUFF_SERVER_RESOURCES;
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
Endpoint->IpxMaxPacketSizeArray = maxPktArray;
|
|||
|
Endpoint->MaxAdapters = numAdapters;
|
|||
|
|
|||
|
//
|
|||
|
// Query the max pkt size for each adapter
|
|||
|
//
|
|||
|
|
|||
|
action->Header.TransportId = 'XPIM'; // "MIPX"
|
|||
|
action->Header.ActionCode = 0;
|
|||
|
action->Header.Reserved = 0;
|
|||
|
action->OptionType = NWLINK_OPTION_ADDRESS;
|
|||
|
action->BufferLength = sizeof(action->Option) + sizeof(IPX_ADDRESS_DATA);
|
|||
|
action->Option = MIPX_GETCARDINFO2;
|
|||
|
ipxAddressData = (PIPX_ADDRESS_DATA)action->Data;
|
|||
|
|
|||
|
for ( i = 0; i < numAdapters; i++ ) {
|
|||
|
|
|||
|
ipxAddressData->adapternum = i;
|
|||
|
|
|||
|
status = SrvIssueTdiAction(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)action,
|
|||
|
sizeof(NWLINK_ACTION) + sizeof(IPX_ADDRESS_DATA) - 1
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If this is a wan link, then we need to query the length each
|
|||
|
// time we get a connection.
|
|||
|
//
|
|||
|
|
|||
|
if ( ipxAddressData->wan ) {
|
|||
|
maxPktArray[i] = 0;
|
|||
|
} else {
|
|||
|
maxPktArray[i] = ipxAddressData->maxpkt;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Find the network address of the adapter used by corresponding to
|
|||
|
// this endpoint.
|
|||
|
//
|
|||
|
|
|||
|
GetNetworkAddress( Endpoint );
|
|||
|
|
|||
|
//
|
|||
|
// Register the name claim Receive Datagram event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
TDI_EVENT_RECEIVE_DATAGRAM,
|
|||
|
(PVOID)SrvIpxNameDatagramHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNonNetbiosAddress: set receive datagram event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Claim the server name.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIpxClaimServerName( Endpoint, NetbiosName );
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Open the server socket.
|
|||
|
//
|
|||
|
|
|||
|
status = OpenIpxSocket(
|
|||
|
&Endpoint->EndpointHandle,
|
|||
|
&Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
DeviceName,
|
|||
|
SMB_IPX_SERVER_SOCKET
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Create a number of free connections for the endpoint. These
|
|||
|
// connections will be used to service Connect events.
|
|||
|
//
|
|||
|
// *** If we fail in an attempt to create a connection, but we can
|
|||
|
// successfully create at least one, we keep the endpoint. The
|
|||
|
// cleanup code below depends on this behavior.
|
|||
|
//
|
|||
|
|
|||
|
for ( i = 0; i < SrvFreeConnectionMinimum; i++ ) {
|
|||
|
|
|||
|
status = SrvOpenConnection( Endpoint );
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNonNetbiosAddress: SrvOpenConnection failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
if ( i == 0 ) {
|
|||
|
goto cleanup;
|
|||
|
} else {
|
|||
|
break;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Register the server Receive Datagram event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
TDI_EVENT_RECEIVE_DATAGRAM,
|
|||
|
(PVOID)SrvIpxServerDatagramHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNonNetbiosAddress: set receive datagram event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Register the server Chained Receive Datagram event handler.
|
|||
|
//
|
|||
|
|
|||
|
status = SrvIssueSetEventHandlerRequest(
|
|||
|
Endpoint->FileObject,
|
|||
|
&Endpoint->DeviceObject,
|
|||
|
TDI_EVENT_CHAINED_RECEIVE_DATAGRAM,
|
|||
|
(PVOID)SrvIpxServerChainedDatagramHandler,
|
|||
|
Endpoint
|
|||
|
);
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenNonNetbiosAddress: set chained receive datagram event handler failed: %X",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
SrvLogServiceFailure( SRV_SVC_NT_IOCTL_FILE, status );
|
|||
|
goto cleanup;
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
//
|
|||
|
// Out-of-line error cleanup.
|
|||
|
//
|
|||
|
|
|||
|
cleanup:
|
|||
|
|
|||
|
//
|
|||
|
// Something failed. Clean up as appropriate.
|
|||
|
//
|
|||
|
|
|||
|
if ( maxPktArray != NULL ) {
|
|||
|
Endpoint->IpxMaxPacketSizeArray = NULL;
|
|||
|
FREE_HEAP( maxPktArray );
|
|||
|
}
|
|||
|
if ( Endpoint->FileObject != NULL ) {
|
|||
|
ObDereferenceObject( Endpoint->FileObject );
|
|||
|
Endpoint->FileObject = NULL;
|
|||
|
}
|
|||
|
if ( Endpoint->EndpointHandle != NULL ) {
|
|||
|
SRVDBG_RELEASE_HANDLE( Endpoint->EndpointHandle, "END", 14, Endpoint );
|
|||
|
SrvNtClose( Endpoint->EndpointHandle, FALSE );
|
|||
|
Endpoint->FileObject = NULL;
|
|||
|
}
|
|||
|
|
|||
|
if ( Endpoint->NameSocketFileObject != NULL ) {
|
|||
|
ObDereferenceObject( Endpoint->NameSocketFileObject );
|
|||
|
Endpoint->NameSocketFileObject = NULL;
|
|||
|
}
|
|||
|
if ( Endpoint->NameSocketHandle != NULL ) {
|
|||
|
SRVDBG_RELEASE_HANDLE( Endpoint->NameSocketHandle, "END", 14, Endpoint );
|
|||
|
SrvNtClose( Endpoint->NameSocketHandle, FALSE );
|
|||
|
Endpoint->NameSocketHandle = NULL;
|
|||
|
}
|
|||
|
|
|||
|
return status;
|
|||
|
|
|||
|
} // OpenNonNetbiosAddress
|
|||
|
|
|||
|
|
|||
|
NTSTATUS
|
|||
|
OpenIpxSocket (
|
|||
|
OUT PHANDLE Handle,
|
|||
|
OUT PFILE_OBJECT *FileObject,
|
|||
|
OUT PDEVICE_OBJECT *DeviceObject,
|
|||
|
IN PVOID DeviceName,
|
|||
|
IN USHORT Socket
|
|||
|
)
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
ULONG length;
|
|||
|
PFILE_FULL_EA_INFORMATION ea;
|
|||
|
TA_IPX_ADDRESS ipxAddress;
|
|||
|
OBJECT_ATTRIBUTES objectAttributes;
|
|||
|
IO_STATUS_BLOCK iosb;
|
|||
|
|
|||
|
CHAR buffer[sizeof(FILE_FULL_EA_INFORMATION) +
|
|||
|
TDI_TRANSPORT_ADDRESS_LENGTH + 1 +
|
|||
|
sizeof(TA_IPX_ADDRESS)];
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
//
|
|||
|
// Build the IPX socket address.
|
|||
|
//
|
|||
|
|
|||
|
length = FIELD_OFFSET( FILE_FULL_EA_INFORMATION, EaName[0] ) +
|
|||
|
TDI_TRANSPORT_ADDRESS_LENGTH + 1 +
|
|||
|
sizeof(TA_IPX_ADDRESS);
|
|||
|
ea = (PFILE_FULL_EA_INFORMATION)buffer;
|
|||
|
|
|||
|
ea->NextEntryOffset = 0;
|
|||
|
ea->Flags = 0;
|
|||
|
ea->EaNameLength = TDI_TRANSPORT_ADDRESS_LENGTH;
|
|||
|
ea->EaValueLength = sizeof (TA_IPX_ADDRESS);
|
|||
|
|
|||
|
RtlCopyMemory( ea->EaName, StrTransportAddress, ea->EaNameLength + 1 );
|
|||
|
|
|||
|
//
|
|||
|
// Create a copy of the NETBIOS address descriptor in a local
|
|||
|
// first, in order to avoid alignment problems.
|
|||
|
//
|
|||
|
|
|||
|
ipxAddress.TAAddressCount = 1;
|
|||
|
ipxAddress.Address[0].AddressType = TDI_ADDRESS_TYPE_IPX;
|
|||
|
ipxAddress.Address[0].AddressLength = sizeof (TDI_ADDRESS_IPX);
|
|||
|
ipxAddress.Address[0].Address[0].Socket = Socket;
|
|||
|
|
|||
|
RtlCopyMemory(
|
|||
|
&ea->EaName[ea->EaNameLength + 1],
|
|||
|
&ipxAddress,
|
|||
|
sizeof(TA_IPX_ADDRESS)
|
|||
|
);
|
|||
|
|
|||
|
InitializeObjectAttributes( &objectAttributes, DeviceName, 0, NULL, NULL );
|
|||
|
|
|||
|
status = NtCreateFile (
|
|||
|
Handle,
|
|||
|
FILE_READ_ATTRIBUTES | FILE_WRITE_ATTRIBUTES, // desired access
|
|||
|
&objectAttributes, // object attributes
|
|||
|
&iosb, // returned status information
|
|||
|
NULL, // block size (unused)
|
|||
|
0, // file attributes
|
|||
|
FILE_SHARE_READ | FILE_SHARE_WRITE, // share access
|
|||
|
FILE_CREATE, // create disposition
|
|||
|
0, // create options
|
|||
|
buffer, // EA buffer
|
|||
|
length // EA length
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
status = SrvVerifyDeviceStackSize(
|
|||
|
*Handle,
|
|||
|
TRUE,
|
|||
|
FileObject,
|
|||
|
DeviceObject,
|
|||
|
NULL
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS( status ) ) {
|
|||
|
INTERNAL_ERROR(
|
|||
|
ERROR_LEVEL_EXPECTED,
|
|||
|
"OpenIpxSocket: Verify Device Stack Size failed: %X\n",
|
|||
|
status,
|
|||
|
NULL
|
|||
|
);
|
|||
|
return status;
|
|||
|
}
|
|||
|
|
|||
|
return STATUS_SUCCESS;
|
|||
|
|
|||
|
} // OpenIpxSocket
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvPrepareReceiveWorkItem (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext,
|
|||
|
IN BOOLEAN QueueItemToFreeList
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine initializes a Receive work item and optionally queues
|
|||
|
it to a list anchored in the server FSD device object. The
|
|||
|
transport receive event handler in the FSD dequeues work items from
|
|||
|
this list and passes their associated IRPS to the transport
|
|||
|
provider.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - Supplies a pointer to the preallocated work context
|
|||
|
block that represents the work item.
|
|||
|
|
|||
|
QueueItemToFreeList - If TRUE queue this work item on the receive
|
|||
|
free queue.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PSMB_HEADER header;
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvPrepareReceiveWorkItem entered\n" ));
|
|||
|
|
|||
|
//
|
|||
|
// Set up pointers to the SMB header and parameters for the request
|
|||
|
// and the response. Note that we currently write the response over
|
|||
|
// the request. SMB processors must be able to handle this. We
|
|||
|
// maintain separate request and response pointers so that we can
|
|||
|
// use a separate buffer if necessary. Maintaining separate request
|
|||
|
// and response parameter pointers also allows us to process AndX
|
|||
|
// SMBs without having to pack the AndX commands as we go.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->ResponseBuffer = WorkContext->RequestBuffer;
|
|||
|
|
|||
|
header = (PSMB_HEADER)WorkContext->RequestBuffer->Buffer;
|
|||
|
|
|||
|
WorkContext->RequestHeader = header;
|
|||
|
WorkContext->RequestParameters = (PVOID)(header + 1);
|
|||
|
|
|||
|
WorkContext->ResponseHeader = header;
|
|||
|
WorkContext->ResponseParameters = (PVOID)(header + 1);
|
|||
|
|
|||
|
//
|
|||
|
// Set up the restart routine in the work context.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->FsdRestartRoutine = SrvQueueWorkToFspAtDpcLevel;
|
|||
|
WorkContext->FspRestartRoutine = SrvRestartReceive;
|
|||
|
|
|||
|
if ( QueueItemToFreeList ) {
|
|||
|
|
|||
|
//
|
|||
|
// Queue the prepared receive work item to the FSD list.
|
|||
|
//
|
|||
|
|
|||
|
GET_SERVER_TIME( WorkContext->CurrentWorkQueue, &WorkContext->Timestamp );
|
|||
|
RETURN_FREE_WORKITEM( WorkContext );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// Make the work item look like it's in use by setting its
|
|||
|
// reference count to 1.
|
|||
|
//
|
|||
|
|
|||
|
ASSERT( WorkContext->BlockHeader.ReferenceCount == 0 );
|
|||
|
WorkContext->BlockHeader.ReferenceCount = 1;
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvPrepareReceiveWorkItem
|
|||
|
|
|||
|
|
|||
|
VOID SRVFASTCALL
|
|||
|
SrvRestartAccept (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function is the worker thread restart routine for Accept
|
|||
|
requests. If the endpoint on which the connection was established
|
|||
|
is no longer active, this routine disconnects the connection. This
|
|||
|
is necessary because the connect indication handler cannot
|
|||
|
atomically verify that the endpoint is active and install the active
|
|||
|
connection. (This is because the handler runs at DPC level.)
|
|||
|
|
|||
|
This routine also checks the status of the TdiAccept. In case of
|
|||
|
an error, it frees the connection.
|
|||
|
|
|||
|
If all is well, but the endpoint is short of free connections, a new
|
|||
|
one is created.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - Supplies a pointer to the work context block describing
|
|||
|
server-specific context for the request.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PCONNECTION connection;
|
|||
|
PENDPOINT endpoint;
|
|||
|
PIRP irp;
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
IF_DEBUG(WORKER1) KdPrint(( " - SrvRestartAccept\n" ));
|
|||
|
|
|||
|
connection = WorkContext->Connection;
|
|||
|
endpoint = WorkContext->Endpoint;
|
|||
|
irp = WorkContext->Irp;
|
|||
|
IF_DEBUG(TRACE2) {
|
|||
|
KdPrint(( " connection %lx, endpoint %lx, IRP %lx\n",
|
|||
|
connection, endpoint, irp ));
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If the I/O request failed or was canceled, or if the endpoint
|
|||
|
// block is closing, clean up.
|
|||
|
//
|
|||
|
|
|||
|
ACQUIRE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
if ( irp->Cancel ||
|
|||
|
!NT_SUCCESS(irp->IoStatus.Status) ||
|
|||
|
(GET_BLOCK_STATE(endpoint) != BlockStateActive) ) {
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
DEBUG {
|
|||
|
KdPrint(( "SrvRestartAccept: Accept failed!" ));
|
|||
|
if ( irp->Cancel ) {
|
|||
|
KdPrint(( " I/O canceled\n" ));
|
|||
|
} else if ( !NT_SUCCESS(irp->IoStatus.Status) ) {
|
|||
|
KdPrint(( " I/O failed: %X\n", irp->IoStatus.Status ));
|
|||
|
} else {
|
|||
|
KdPrint(( " Endpoint no longer active\n" ));
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Close the connection. If the Accept succeeded, we need to
|
|||
|
// issue a Disconnect.
|
|||
|
//
|
|||
|
|
|||
|
#if SRVDBG29
|
|||
|
if (irp->Cancel) {
|
|||
|
UpdateConnectionHistory( "ACC1", endpoint, connection );
|
|||
|
} else if (!NT_SUCCESS(irp->IoStatus.Status)) {
|
|||
|
UpdateConnectionHistory( "ACC2", endpoint, connection );
|
|||
|
} else {
|
|||
|
UpdateConnectionHistory( "ACC3", endpoint, connection );
|
|||
|
}
|
|||
|
#endif
|
|||
|
|
|||
|
SrvCloseConnection(
|
|||
|
connection,
|
|||
|
(BOOLEAN)(irp->Cancel || !NT_SUCCESS(irp->IoStatus.Status) ?
|
|||
|
TRUE : FALSE) // RemoteDisconnect
|
|||
|
);
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
//
|
|||
|
// The Accept worked, and the endpoint is still active. Create
|
|||
|
// a new free connection, if necessary.
|
|||
|
//
|
|||
|
|
|||
|
if ( endpoint->FreeConnectionCount < SrvFreeConnectionMinimum ) {
|
|||
|
(VOID)SrvOpenConnection( endpoint );
|
|||
|
}
|
|||
|
|
|||
|
RELEASE_LOCK( &SrvEndpointLock );
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
SrvDereferenceWorkItem( WorkContext );
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvRestartAccept complete\n" ));
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvRestartAccept
|
|||
|
|
|||
|
|
|||
|
VOID
|
|||
|
SrvStartSend (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext,
|
|||
|
IN PIO_COMPLETION_ROUTINE SendCompletionRoutine,
|
|||
|
IN PMDL Mdl OPTIONAL,
|
|||
|
IN ULONG SendOptions
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function starts a Send request. It is started as an
|
|||
|
asynchronous I/O request. When the Send completes, it is delivered
|
|||
|
via the I/O completion routine to the server FSD, which routes it to
|
|||
|
the specified FsdRestartRoutine. (This may be
|
|||
|
SrvQueueWorkToFspAtDpcLevel, which queues the work item to the FSP
|
|||
|
at the FspRestartRoutine.)
|
|||
|
|
|||
|
Partial sends and chained sends are supported. A partial send is one
|
|||
|
that is not the last segment of a "message" or "record". A chained
|
|||
|
send is one made up of multiple virtually discontiguous buffers.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - Supplies a pointer to a Work Context block. The
|
|||
|
following fields of this structure must be valid:
|
|||
|
|
|||
|
TdiRequest
|
|||
|
Irp (optional; actual address copied here)
|
|||
|
Endpoint
|
|||
|
Endpoint->FileObject
|
|||
|
Endpoint->DeviceObject
|
|||
|
Connection
|
|||
|
Connection->ConnectionId
|
|||
|
|
|||
|
Mdl - Supplies a pointer to the first (or only) MDL describing the
|
|||
|
data that is to be sent. To effect a chained send, the Next
|
|||
|
pointer of each MDL in the chain must point to the next MDL;
|
|||
|
the end of the chain is indicated by the NULL Next pointer.
|
|||
|
|
|||
|
The total length of the send is calculated by summing the
|
|||
|
ByteCount fields of each MDL in the chain.
|
|||
|
|
|||
|
This parameter is optional. If it is omitted, a zero-length
|
|||
|
message is sent.
|
|||
|
|
|||
|
SendOptions - Supplied TDI send options, which indicate whether this
|
|||
|
send is the last (or only) in a "chain" of partial sends.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PTDI_REQUEST_KERNEL_SEND parameters;
|
|||
|
PIO_STACK_LOCATION irpSp;
|
|||
|
PIRP irp;
|
|||
|
// PMDL nextMdl;
|
|||
|
ULONG sendLength;
|
|||
|
PDEVICE_OBJECT deviceObject;
|
|||
|
PFILE_OBJECT fileObject;
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvStartSend entered\n" ));
|
|||
|
|
|||
|
ASSERT( !WorkContext->Endpoint->IsConnectionless );
|
|||
|
|
|||
|
//
|
|||
|
// Set ProcessingCount to zero so this send cannot be cancelled.
|
|||
|
// This is used together with setting the cancel flag to false below.
|
|||
|
//
|
|||
|
// WARNING: This still presents us with a tiny window where this
|
|||
|
// send could be cancelled.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->ProcessingCount = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Get the irp, device, and file objects
|
|||
|
//
|
|||
|
|
|||
|
irp = WorkContext->Irp;
|
|||
|
deviceObject = WorkContext->Connection->DeviceObject;
|
|||
|
fileObject = WorkContext->Connection->FileObject;
|
|||
|
|
|||
|
//
|
|||
|
// Count up the length of the data described by chained MDLs.
|
|||
|
//
|
|||
|
|
|||
|
#if 0
|
|||
|
sendLength = 0;
|
|||
|
nextMdl = Mdl;
|
|||
|
while ( nextMdl != NULL ) {
|
|||
|
sendLength += MmGetMdlByteCount( nextMdl );
|
|||
|
nextMdl = nextMdl->Next;
|
|||
|
}
|
|||
|
#else
|
|||
|
sendLength = WorkContext->ResponseBuffer->DataLength;
|
|||
|
#endif
|
|||
|
|
|||
|
//
|
|||
|
// Build the I/O request packet.
|
|||
|
//
|
|||
|
// *** Note that the connection block is not referenced to account
|
|||
|
// for this I/O request. The WorkContext block already has a
|
|||
|
// referenced pointer to the connection, and this pointer is not
|
|||
|
// dereferenced until after the I/O completes.
|
|||
|
//
|
|||
|
|
|||
|
ASSERT( irp->StackCount >= deviceObject->StackSize );
|
|||
|
|
|||
|
irp->Tail.Overlay.OriginalFileObject = fileObject;
|
|||
|
irp->Tail.Overlay.Thread = WorkContext->CurrentWorkQueue->IrpThread;
|
|||
|
DEBUG irp->RequestorMode = KernelMode;
|
|||
|
//
|
|||
|
// Get a pointer to the next stack location. This one is used to
|
|||
|
// hold the parameters for the device I/O control request.
|
|||
|
//
|
|||
|
|
|||
|
irpSp = IoGetNextIrpStackLocation( irp );
|
|||
|
|
|||
|
//
|
|||
|
// Set up the completion routine.
|
|||
|
//
|
|||
|
|
|||
|
IoSetCompletionRoutine(
|
|||
|
irp,
|
|||
|
SendCompletionRoutine,
|
|||
|
(PVOID)WorkContext,
|
|||
|
TRUE,
|
|||
|
TRUE,
|
|||
|
TRUE
|
|||
|
);
|
|||
|
|
|||
|
|
|||
|
irpSp->FileObject = fileObject;
|
|||
|
|
|||
|
parameters = (PTDI_REQUEST_KERNEL_SEND)&irpSp->Parameters;
|
|||
|
parameters->SendFlags = SendOptions;
|
|||
|
parameters->SendLength = sendLength;
|
|||
|
|
|||
|
//
|
|||
|
// For these two cases, InputBuffer is the buffered I/O "system
|
|||
|
// buffer". Build an MDL for either read or write access,
|
|||
|
// depending on the method, for the output buffer.
|
|||
|
//
|
|||
|
|
|||
|
irp->MdlAddress = Mdl;
|
|||
|
|
|||
|
//
|
|||
|
// If statistics are to be gathered for this work item, do so now.
|
|||
|
//
|
|||
|
|
|||
|
UPDATE_STATISTICS(
|
|||
|
WorkContext,
|
|||
|
sendLength,
|
|||
|
WorkContext->ResponseHeader->Command
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Pass the request to the transport provider.
|
|||
|
//
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) {
|
|||
|
KdPrint(( "SrvStartSend posting Send IRP %lx\n", irp ));
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If SmbTrace is active and we're in a context where the SmbTrace
|
|||
|
// shared section isn't accessible, send this off to the FSP.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->Irp->Cancel = FALSE;
|
|||
|
|
|||
|
if ( SmbTraceActive[SMBTRACE_SERVER] ) {
|
|||
|
|
|||
|
if ((KeGetCurrentIrql() == DISPATCH_LEVEL) ||
|
|||
|
(IoGetCurrentProcess() != SrvServerProcess) ) {
|
|||
|
|
|||
|
irpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
|||
|
irpSp->MinorFunction = TDI_SEND;
|
|||
|
irp->AssociatedIrp.SystemBuffer = NULL;
|
|||
|
irp->Flags = (ULONG)IRP_BUFFERED_IO;
|
|||
|
|
|||
|
WorkContext->Parameters2.StartSend.FspRestartRoutine =
|
|||
|
WorkContext->FspRestartRoutine;
|
|||
|
WorkContext->Parameters2.StartSend.SendLength = sendLength;
|
|||
|
|
|||
|
WorkContext->FspRestartRoutine = RestartStartSend;
|
|||
|
SrvQueueWorkToFsp( WorkContext );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
SMBTRACE_SRV( Mdl );
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set the cancel flag to FALSE in case this was cancelled by
|
|||
|
// the SrvSmbNtCancel routine.
|
|||
|
//
|
|||
|
|
|||
|
if ( WorkContext->Endpoint->FastTdiSend ) {
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT2( SrvDbgStatistics.DirectSendsAttempted );
|
|||
|
irpSp->MinorFunction = TDI_DIRECT_SEND;
|
|||
|
DEBUG irpSp->DeviceObject = deviceObject;
|
|||
|
IoSetNextIrpStackLocation( irp );
|
|||
|
WorkContext->Endpoint->FastTdiSend( deviceObject, irp );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
irpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
|||
|
irpSp->MinorFunction = TDI_SEND;
|
|||
|
irp->AssociatedIrp.SystemBuffer = NULL;
|
|||
|
irp->Flags = (ULONG)IRP_BUFFERED_IO;
|
|||
|
|
|||
|
(VOID)IoCallDriver( deviceObject, irp );
|
|||
|
}
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvStartSend complete\n" ));
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvStartSend
|
|||
|
|
|||
|
VOID
|
|||
|
SrvStartSend2 (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext,
|
|||
|
IN PIO_COMPLETION_ROUTINE SendCompletionRoutine
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This function starts a Send request. It is started as an
|
|||
|
asynchronous I/O request. When the Send completes, it is delivered
|
|||
|
via the I/O completion routine to the server FSD, which routes it to
|
|||
|
the specified FsdRestartRoutine. (This may be
|
|||
|
SrvQueueWorkToFspAtDpcLevel, which queues the work item to the FSP
|
|||
|
at the FspRestartRoutine.)
|
|||
|
|
|||
|
Partial sends and chained sends are supported. A partial send is one
|
|||
|
that is not the last segment of a "message" or "record". A chained
|
|||
|
send is one made up of multiple virtually discontiguous buffers.
|
|||
|
|
|||
|
** This is identical to SrvStartSend except that the parameter mdl
|
|||
|
is assumed to be ResponseBuffer->Mdl and sendOptions is assumed to be
|
|||
|
0 **
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - Supplies a pointer to a Work Context block. The
|
|||
|
following fields of this structure must be valid:
|
|||
|
|
|||
|
TdiRequest
|
|||
|
Irp (optional; actual address copied here)
|
|||
|
Endpoint
|
|||
|
Endpoint->FileObject
|
|||
|
Endpoint->DeviceObject
|
|||
|
Connection
|
|||
|
Connection->ConnectionId
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PTDI_REQUEST_KERNEL_SEND parameters;
|
|||
|
PIO_STACK_LOCATION irpSp;
|
|||
|
PIRP irp;
|
|||
|
PDEVICE_OBJECT deviceObject;
|
|||
|
PFILE_OBJECT fileObject;
|
|||
|
|
|||
|
PMDL mdl = WorkContext->ResponseBuffer->Mdl;
|
|||
|
ULONG sendLength = WorkContext->ResponseBuffer->DataLength;
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvStartSend2 entered\n" ));
|
|||
|
|
|||
|
ASSERT( !WorkContext->Endpoint->IsConnectionless );
|
|||
|
|
|||
|
//
|
|||
|
// Set ProcessingCount to zero so this send cannot be cancelled.
|
|||
|
// This is used together with setting the cancel flag to false below.
|
|||
|
//
|
|||
|
// WARNING: This still presents us with a tiny window where this
|
|||
|
// send could be cancelled.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->ProcessingCount = 0;
|
|||
|
|
|||
|
//
|
|||
|
// Get the irp, device, and file objects
|
|||
|
//
|
|||
|
|
|||
|
irp = WorkContext->Irp;
|
|||
|
deviceObject = WorkContext->Connection->DeviceObject;
|
|||
|
fileObject = WorkContext->Connection->FileObject;
|
|||
|
|
|||
|
//
|
|||
|
// Build the I/O request packet.
|
|||
|
//
|
|||
|
// *** Note that the connection block is not referenced to account
|
|||
|
// for this I/O request. The WorkContext block already has a
|
|||
|
// referenced pointer to the connection, and this pointer is not
|
|||
|
// dereferenced until after the I/O completes.
|
|||
|
//
|
|||
|
|
|||
|
ASSERT( irp->StackCount >= deviceObject->StackSize );
|
|||
|
|
|||
|
irp->Tail.Overlay.OriginalFileObject = fileObject;
|
|||
|
irp->Tail.Overlay.Thread = WorkContext->CurrentWorkQueue->IrpThread;
|
|||
|
DEBUG irp->RequestorMode = KernelMode;
|
|||
|
|
|||
|
//
|
|||
|
// Get a pointer to the next stack location. This one is used to
|
|||
|
// hold the parameters for the device I/O control request.
|
|||
|
//
|
|||
|
|
|||
|
irpSp = IoGetNextIrpStackLocation( irp );
|
|||
|
|
|||
|
//
|
|||
|
// Set up the completion routine.
|
|||
|
//
|
|||
|
|
|||
|
IoSetCompletionRoutine(
|
|||
|
irp,
|
|||
|
SendCompletionRoutine,
|
|||
|
(PVOID)WorkContext,
|
|||
|
TRUE,
|
|||
|
TRUE,
|
|||
|
TRUE
|
|||
|
);
|
|||
|
|
|||
|
irpSp->FileObject = fileObject;
|
|||
|
|
|||
|
parameters = (PTDI_REQUEST_KERNEL_SEND)&irpSp->Parameters;
|
|||
|
parameters->SendFlags = 0;
|
|||
|
parameters->SendLength = sendLength;
|
|||
|
|
|||
|
//
|
|||
|
// For these two cases, InputBuffer is the buffered I/O "system
|
|||
|
// buffer". Build an MDL for either read or write access,
|
|||
|
// depending on the method, for the output buffer.
|
|||
|
//
|
|||
|
|
|||
|
irp->MdlAddress = mdl;
|
|||
|
|
|||
|
//
|
|||
|
// If statistics are to be gathered for this work item, do so now.
|
|||
|
//
|
|||
|
|
|||
|
UPDATE_STATISTICS(
|
|||
|
WorkContext,
|
|||
|
sendLength,
|
|||
|
WorkContext->ResponseHeader->Command
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Pass the request to the transport provider.
|
|||
|
//
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) {
|
|||
|
KdPrint(( "SrvStartSend2 posting Send IRP %lx\n", irp ));
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If SmbTrace is active and we're in a context where the SmbTrace
|
|||
|
// shared section isn't accessible, send this off to the FSP.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->Irp->Cancel = FALSE;
|
|||
|
|
|||
|
if ( SmbTraceActive[SMBTRACE_SERVER] ) {
|
|||
|
|
|||
|
if ((KeGetCurrentIrql() == DISPATCH_LEVEL) ||
|
|||
|
(IoGetCurrentProcess() != SrvServerProcess) ) {
|
|||
|
|
|||
|
irpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
|||
|
irpSp->MinorFunction = TDI_SEND;
|
|||
|
irp->AssociatedIrp.SystemBuffer = NULL;
|
|||
|
irp->Flags = (ULONG)IRP_BUFFERED_IO;
|
|||
|
|
|||
|
WorkContext->Parameters2.StartSend.FspRestartRoutine =
|
|||
|
WorkContext->FspRestartRoutine;
|
|||
|
WorkContext->Parameters2.StartSend.SendLength = sendLength;
|
|||
|
|
|||
|
WorkContext->FspRestartRoutine = RestartStartSend;
|
|||
|
SrvQueueWorkToFsp( WorkContext );
|
|||
|
|
|||
|
return;
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
SMBTRACE_SRV( mdl );
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// Set the cancel flag to FALSE in case this was cancelled by
|
|||
|
// the SrvSmbNtCancel routine.
|
|||
|
//
|
|||
|
|
|||
|
if ( WorkContext->Endpoint->FastTdiSend ) {
|
|||
|
|
|||
|
INCREMENT_DEBUG_STAT2( SrvDbgStatistics.DirectSendsAttempted );
|
|||
|
irpSp->MinorFunction = TDI_DIRECT_SEND;
|
|||
|
DEBUG irpSp->DeviceObject = deviceObject;
|
|||
|
IoSetNextIrpStackLocation( irp );
|
|||
|
WorkContext->Endpoint->FastTdiSend( deviceObject, irp );
|
|||
|
|
|||
|
} else {
|
|||
|
|
|||
|
irpSp->MajorFunction = IRP_MJ_INTERNAL_DEVICE_CONTROL;
|
|||
|
irpSp->MinorFunction = TDI_SEND;
|
|||
|
irp->AssociatedIrp.SystemBuffer = NULL;
|
|||
|
irp->Flags = (ULONG)IRP_BUFFERED_IO;
|
|||
|
|
|||
|
(VOID)IoCallDriver( deviceObject, irp );
|
|||
|
}
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvStartSend2 complete\n" ));
|
|||
|
return;
|
|||
|
|
|||
|
} // SrvStartSend2
|
|||
|
|
|||
|
|
|||
|
VOID SRVFASTCALL
|
|||
|
RestartStartSend (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - Supplies a pointer to a Work Context block.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
None.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
WorkContext->FspRestartRoutine =
|
|||
|
WorkContext->Parameters2.StartSend.FspRestartRoutine;
|
|||
|
|
|||
|
SMBTRACE_SRV( WorkContext->Irp->MdlAddress );
|
|||
|
|
|||
|
//
|
|||
|
// Set the cancel flag to FALSE in case this was cancelled by
|
|||
|
// the SrvSmbNtCancel routine.
|
|||
|
//
|
|||
|
|
|||
|
WorkContext->Irp->Cancel = FALSE;
|
|||
|
(VOID)IoCallDriver(
|
|||
|
IoGetNextIrpStackLocation( WorkContext->Irp )->DeviceObject,
|
|||
|
WorkContext->Irp
|
|||
|
);
|
|||
|
|
|||
|
IF_DEBUG(TRACE2) KdPrint(( "SrvRestartStartSend complete\n" ));
|
|||
|
return;
|
|||
|
|
|||
|
} // RestartStartSend
|
|||
|
|
|||
|
ULONG
|
|||
|
GetIpxMaxBufferSize(
|
|||
|
PENDPOINT Endpoint,
|
|||
|
ULONG AdapterNumber,
|
|||
|
ULONG DefaultMaxBufferSize
|
|||
|
)
|
|||
|
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine computes the max buffer size the server negotiates
|
|||
|
with the client. It takes the smaller of DefaultMaxBufferSize
|
|||
|
and the max packet length returned by the ipx transport.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
Endpoint - pointer to the endpoint corresponding to the ipx transport
|
|||
|
AdapterNumber - the adapter number for which the max buffer size is to
|
|||
|
be computed for.
|
|||
|
DefaultMaxBufferSize - the maximum size that can be returned by this
|
|||
|
routine.
|
|||
|
|
|||
|
Return Value:
|
|||
|
|
|||
|
The max buffer size to be negotiated by the server.
|
|||
|
|
|||
|
--*/
|
|||
|
|
|||
|
{
|
|||
|
NTSTATUS status;
|
|||
|
ULONG maxBufferSize;
|
|||
|
PNWLINK_ACTION action;
|
|||
|
PIPX_ADDRESS_DATA ipxAddressData;
|
|||
|
UCHAR buffer[sizeof(NWLINK_ACTION) + sizeof(IPX_ADDRESS_DATA) - 1];
|
|||
|
|
|||
|
PAGED_CODE( );
|
|||
|
|
|||
|
action = (PNWLINK_ACTION)buffer;
|
|||
|
|
|||
|
//
|
|||
|
// Verify that the adapter number is within bounds
|
|||
|
//
|
|||
|
|
|||
|
if ( AdapterNumber > Endpoint->MaxAdapters ) {
|
|||
|
return DefaultMaxBufferSize;
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// If value in array is non-zero, then this is not a wan link.
|
|||
|
// Use that value.
|
|||
|
//
|
|||
|
|
|||
|
if ( Endpoint->IpxMaxPacketSizeArray[AdapterNumber-1] != 0 ) {
|
|||
|
|
|||
|
maxBufferSize = MIN(
|
|||
|
Endpoint->IpxMaxPacketSizeArray[AdapterNumber-1],
|
|||
|
DefaultMaxBufferSize
|
|||
|
);
|
|||
|
|
|||
|
return (maxBufferSize & ~3);
|
|||
|
}
|
|||
|
|
|||
|
//
|
|||
|
// This is a wan link, query the max packet size.
|
|||
|
//
|
|||
|
|
|||
|
action->Header.TransportId = 'XPIM'; // "MIPX"
|
|||
|
action->Header.ActionCode = 0;
|
|||
|
action->Header.Reserved = 0;
|
|||
|
action->OptionType = NWLINK_OPTION_ADDRESS;
|
|||
|
action->BufferLength = sizeof(action->Option) + sizeof(IPX_ADDRESS_DATA);
|
|||
|
action->Option = MIPX_GETCARDINFO2;
|
|||
|
ipxAddressData = (PIPX_ADDRESS_DATA)action->Data;
|
|||
|
|
|||
|
ipxAddressData->adapternum = AdapterNumber - 1;
|
|||
|
|
|||
|
status = SrvIssueTdiAction(
|
|||
|
Endpoint->NameSocketFileObject,
|
|||
|
&Endpoint->NameSocketDeviceObject,
|
|||
|
(PCHAR)action,
|
|||
|
sizeof(NWLINK_ACTION) + sizeof(IPX_ADDRESS_DATA) - 1
|
|||
|
);
|
|||
|
|
|||
|
if ( !NT_SUCCESS(status) ) {
|
|||
|
return DefaultMaxBufferSize;
|
|||
|
}
|
|||
|
|
|||
|
ASSERT( ipxAddressData->wan );
|
|||
|
|
|||
|
maxBufferSize = MIN(
|
|||
|
(ULONG)ipxAddressData->maxpkt,
|
|||
|
DefaultMaxBufferSize
|
|||
|
);
|
|||
|
|
|||
|
return (maxBufferSize & ~3);
|
|||
|
|
|||
|
} // GetMaxIpxPacketSize
|
|||
|
|
|||
|
#ifdef SRV_PNP_POWER
|
|||
|
|
|||
|
VOID SRVFASTCALL
|
|||
|
SrvPnpProcessor (
|
|||
|
IN OUT PWORK_CONTEXT WorkContext
|
|||
|
)
|
|||
|
/*++
|
|||
|
|
|||
|
Routine Description:
|
|||
|
|
|||
|
This routine gets called by a worker thread to handle
|
|||
|
PNP notifications.
|
|||
|
|
|||
|
Arguments:
|
|||
|
|
|||
|
WorkContext - contains information in Parameters.Pnp detailing the
|
|||
|
PNP activity we need to perform.
|
|||
|
--*/
|
|||
|
{
|
|||
|
PAGED_CODE();
|
|||
|
|
|||
|
//
|
|||
|
// Send the request to the srvsvc
|
|||
|
//
|
|||
|
SrvXsPnpOperation( WorkContext->Parameters.Pnp.Bind,
|
|||
|
WorkContext->Parameters.Pnp.Index
|
|||
|
);
|
|||
|
|
|||
|
//
|
|||
|
// Signal our caller that the bind is completed
|
|||
|
//
|
|||
|
KeSetEvent(
|
|||
|
WorkContext->Parameters.Pnp.Event,
|
|||
|
EVENT_INCREMENT,
|
|||
|
FALSE
|
|||
|
);
|
|||
|
|
|||
|
WorkContext->FspRestartRoutine = SrvRestartReceive;
|
|||
|
WorkContext->BlockHeader.ReferenceCount = 0;
|
|||
|
RETURN_FREE_WORKITEM( WorkContext );
|
|||
|
}
|
|||
|
|
|||
|
#endif
|
|||
|
|