547 lines
13 KiB
C
547 lines
13 KiB
C
/*++
|
||
|
||
Copyright (c) 1991 Microsoft Corporation
|
||
|
||
Module Name:
|
||
|
||
svcconn.c
|
||
|
||
Abstract:
|
||
|
||
This module contains routines for supporting the connection APIs in
|
||
the server service, SrvNetConnectionEnum.
|
||
|
||
Author:
|
||
|
||
David Treadwell (davidtr) 23-Feb-1991
|
||
|
||
Revision History:
|
||
|
||
--*/
|
||
|
||
#include "precomp.h"
|
||
#pragma hdrstop
|
||
|
||
#define BugCheckFileId SRV_FILE_SVCCONN
|
||
|
||
//
|
||
// Forward declarations.
|
||
//
|
||
|
||
VOID
|
||
FillConnectionInfoBuffer (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block,
|
||
IN OUT PVOID *FixedStructurePointer,
|
||
IN OUT LPTSTR *EndOfVariableData
|
||
);
|
||
|
||
BOOLEAN
|
||
FilterConnections (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block
|
||
);
|
||
|
||
ULONG
|
||
SizeConnections (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block
|
||
);
|
||
|
||
#ifdef ALLOC_PRAGMA
|
||
#pragma alloc_text( PAGE, SrvNetConnectionEnum )
|
||
#pragma alloc_text( PAGE, FillConnectionInfoBuffer )
|
||
#pragma alloc_text( PAGE, FilterConnections )
|
||
#pragma alloc_text( PAGE, SizeConnections )
|
||
#endif
|
||
|
||
//
|
||
// Macros to determine the size a share would take up at one of the
|
||
// levels of share information.
|
||
//
|
||
|
||
#define TOTAL_SIZE_OF_CONNECTION(treeConnect,level,user,netname) \
|
||
( (level) == 0 ? sizeof(CONNECTION_INFO_0) : \
|
||
sizeof(CONNECTION_INFO_1) + \
|
||
SrvLengthOfStringInApiBuffer((user)) + \
|
||
SrvLengthOfStringInApiBuffer((netname)) )
|
||
|
||
#define FIXED_SIZE_OF_CONNECTION(level) \
|
||
( (level) == 0 ? sizeof(CONNECTION_INFO_0) : \
|
||
sizeof(CONNECTION_INFO_1) )
|
||
|
||
|
||
NTSTATUS
|
||
SrvNetConnectionEnum (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Buffer,
|
||
IN ULONG BufferLength
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine processes the NetConnectionEnum API in the server FSD.
|
||
|
||
Arguments:
|
||
|
||
Srp - a pointer to the server request packet that contains all
|
||
the information necessary to satisfy the request. This includes:
|
||
|
||
INPUT:
|
||
|
||
Name1 - qualifier for determining the basis for the search. It
|
||
is either a computer name, in which case information about
|
||
tree connects from the specified client is returned, or
|
||
a share name, in which case information about tree connects
|
||
to the specified share is returned.
|
||
|
||
Level - level of information to return, 0 or 1.
|
||
|
||
OUTPUT:
|
||
|
||
Parameters.Get.EntriesRead - the number of entries that fit in
|
||
the output buffer.
|
||
|
||
Parameters.Get.TotalEntries - the total number of entries that
|
||
would be returned with a large enough buffer.
|
||
|
||
Parameters.Get.TotalBytesNeeded - the buffer size that would be
|
||
required to hold all the entries.
|
||
|
||
Buffer - a pointer to the buffer for results.
|
||
|
||
BufferLength - the length of this buffer.
|
||
|
||
Return Value:
|
||
|
||
NTSTATUS - result of operation to return to the server service.
|
||
|
||
--*/
|
||
|
||
{
|
||
PAGED_CODE( );
|
||
|
||
return SrvEnumApiHandler(
|
||
Srp,
|
||
Buffer,
|
||
BufferLength,
|
||
&SrvTreeConnectList,
|
||
FilterConnections,
|
||
SizeConnections,
|
||
FillConnectionInfoBuffer
|
||
);
|
||
|
||
} // SrvNetConnectionEnum
|
||
|
||
|
||
STATIC
|
||
VOID
|
||
FillConnectionInfoBuffer (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block,
|
||
IN OUT PVOID *FixedStructure,
|
||
IN LPTSTR *EndOfVariableData
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine puts a single fixed session structure and, if it fits,
|
||
associated variable data, into a buffer. Fixed data goes at the
|
||
beginning of the buffer, variable data at the end.
|
||
|
||
*** This routine must be called with Connection->Lock held!
|
||
|
||
Arguments:
|
||
|
||
Level - the level of information to copy from the connection.
|
||
|
||
Connection - the tree connect from which to get information.
|
||
|
||
FixedStructure - where the ine buffer to place the fixed structure.
|
||
This pointer is updated to point to the next available
|
||
position for a fixed structure.
|
||
|
||
EndOfVariableData - the last position on the buffer that variable
|
||
data for this structure can occupy. The actual variable data
|
||
is written before this position as long as it won't overwrite
|
||
fixed structures. It is would overwrite fixed structures, it
|
||
is not written.
|
||
|
||
Return Value:
|
||
|
||
None.
|
||
|
||
--*/
|
||
|
||
{
|
||
PTREE_CONNECT treeConnect = Block;
|
||
PSESSION session;
|
||
PCONNECTION_INFO_1 coni1;
|
||
|
||
LARGE_INTEGER currentTime;
|
||
ULONG currentSecondsSince1980;
|
||
ULONG startTimeInSecondsSince1980;
|
||
ULONG secondsAlive;
|
||
|
||
PAGED_CODE();
|
||
|
||
//
|
||
// Get the current time and use this to determine how long the
|
||
// tree connection has been alive.
|
||
//
|
||
|
||
KeQuerySystemTime( ¤tTime );
|
||
|
||
RtlTimeToSecondsSince1980(
|
||
¤tTime,
|
||
¤tSecondsSince1980
|
||
);
|
||
|
||
RtlTimeToSecondsSince1980(
|
||
&treeConnect->StartTime,
|
||
&startTimeInSecondsSince1980
|
||
);
|
||
|
||
secondsAlive = currentSecondsSince1980 - startTimeInSecondsSince1980;
|
||
|
||
//
|
||
// Set up the fixed structure pointer and find out where the fixed
|
||
// structure ends.
|
||
//
|
||
|
||
coni1 = *FixedStructure;
|
||
|
||
*FixedStructure = (PCHAR)*FixedStructure +
|
||
FIXED_SIZE_OF_CONNECTION( Srp->Level );
|
||
ASSERT( (ULONG)*EndOfVariableData >= (ULONG)*FixedStructure );
|
||
|
||
//
|
||
// Case on the level to fill in the fixed structure appropriately.
|
||
// We fill in actual pointers in the output structure. This is
|
||
// possible because we are in the server FSD, hence the server
|
||
// service's process and address space.
|
||
//
|
||
// *** This routine assumes that the fixed structure will fit in the
|
||
// buffer!
|
||
//
|
||
// *** Using the switch statement in this fashion relies on the fact
|
||
// that the first fields on the different session structures are
|
||
// identical.
|
||
//
|
||
|
||
switch( Srp->Level ) {
|
||
|
||
case 1:
|
||
|
||
//
|
||
// Convert the server's internal representation of share types
|
||
// to the expected format.
|
||
//
|
||
|
||
switch ( treeConnect->Share->ShareType ) {
|
||
|
||
case ShareTypeDisk:
|
||
|
||
coni1->coni1_type = STYPE_DISKTREE;
|
||
break;
|
||
|
||
case ShareTypePrint:
|
||
|
||
coni1->coni1_type = STYPE_PRINTQ;
|
||
break;
|
||
|
||
#if SRV_COMM_DEVICES
|
||
case ShareTypeComm:
|
||
|
||
coni1->coni1_type = STYPE_DEVICE;
|
||
break;
|
||
#endif
|
||
case ShareTypePipe:
|
||
|
||
coni1->coni1_type = STYPE_IPC;
|
||
break;
|
||
|
||
default:
|
||
|
||
//
|
||
// This should never happen. It means that somebody
|
||
// stomped on the share block.
|
||
//
|
||
|
||
INTERNAL_ERROR(
|
||
ERROR_LEVEL_UNEXPECTED,
|
||
"FillConnectionInfoBuffer: invalid share type in share: %ld",
|
||
treeConnect->Share->ShareType,
|
||
NULL
|
||
);
|
||
|
||
SrvLogInvalidSmb( NULL );
|
||
return;
|
||
}
|
||
|
||
//
|
||
// Set up the count of opens done on this tree connect. Do not include
|
||
// cached opens, as they are transparent to users and administrators
|
||
//
|
||
|
||
coni1->coni1_num_opens = treeConnect->CurrentFileOpenCount;
|
||
|
||
if( coni1->coni1_num_opens > 0 ) {
|
||
|
||
ULONG count = SrvCountCachedRfcbsForTid(
|
||
treeConnect->Connection,
|
||
treeConnect->Tid );
|
||
|
||
if( coni1->coni1_num_opens > count ) {
|
||
coni1->coni1_num_opens -= count;
|
||
} else {
|
||
coni1->coni1_num_opens = 0;
|
||
}
|
||
|
||
}
|
||
|
||
//
|
||
// There is always exactly one user on a tree connect.
|
||
//
|
||
// !!! Is this correct???
|
||
|
||
coni1->coni1_num_users = 1;
|
||
|
||
//
|
||
// Set up the alive time.
|
||
//
|
||
|
||
coni1->coni1_time = secondsAlive;
|
||
|
||
//
|
||
// Attempt to find a reasonable user name. Since the SMB
|
||
// protocol does not link tree connects with users, only with
|
||
// sessions, it may not be possible to return a user name.
|
||
//
|
||
|
||
ACQUIRE_LOCK( &treeConnect->Connection->Lock );
|
||
|
||
session = SrvFindUserOnConnection( treeConnect->Connection );
|
||
|
||
if ( session != NULL ) {
|
||
|
||
SrvCopyUnicodeStringToBuffer(
|
||
&session->UserName,
|
||
*FixedStructure,
|
||
EndOfVariableData,
|
||
&coni1->coni1_username
|
||
);
|
||
|
||
} else {
|
||
|
||
coni1->coni1_username = NULL;
|
||
}
|
||
|
||
RELEASE_LOCK( &treeConnect->Connection->Lock );
|
||
|
||
//
|
||
// Set up the net name. If the qualifier passed in the
|
||
// SRP is a computer name, then the net name is the share
|
||
// name. If the qualifier is a share name, the net name
|
||
// is a computer name.
|
||
//
|
||
|
||
if ( Srp->Name1.Length > 2 && *Srp->Name1.Buffer == '\\' &&
|
||
*(Srp->Name1.Buffer+1) == '\\' ) {
|
||
|
||
SrvCopyUnicodeStringToBuffer(
|
||
&treeConnect->Share->ShareName,
|
||
*FixedStructure,
|
||
EndOfVariableData,
|
||
&coni1->coni1_netname
|
||
);
|
||
|
||
} else {
|
||
|
||
UNICODE_STRING clientName;
|
||
PUNICODE_STRING clientMachineName;
|
||
|
||
clientMachineName =
|
||
&treeConnect->Connection->PagedConnection->ClientMachineNameString;
|
||
|
||
//
|
||
// Make a string that does not contain the leading
|
||
// backslashes.
|
||
//
|
||
|
||
clientName.Buffer = clientMachineName->Buffer + 2;
|
||
clientName.Length =
|
||
(USHORT) (clientMachineName->Length - 2 * sizeof(WCHAR));
|
||
clientName.MaximumLength = clientName.Length;
|
||
|
||
SrvCopyUnicodeStringToBuffer(
|
||
&clientName,
|
||
*FixedStructure,
|
||
EndOfVariableData,
|
||
&coni1->coni1_netname
|
||
);
|
||
}
|
||
|
||
// *** Lack of break is intentional!
|
||
|
||
case 0:
|
||
|
||
//
|
||
// Set up the tree connect ID.
|
||
//
|
||
|
||
coni1->coni1_id = SrvGetResumeHandle( &SrvTreeConnectList, treeConnect );
|
||
|
||
break;
|
||
|
||
default:
|
||
|
||
//
|
||
// This should never happen. The server service should have
|
||
// checked for an invalid level.
|
||
//
|
||
|
||
INTERNAL_ERROR(
|
||
ERROR_LEVEL_UNEXPECTED,
|
||
"FillConnectionInfoBuffer: invalid level number: %ld",
|
||
Srp->Level,
|
||
NULL
|
||
);
|
||
|
||
SrvLogInvalidSmb( NULL );
|
||
}
|
||
|
||
return;
|
||
|
||
} // FillConnectionInfoBuffer
|
||
|
||
|
||
BOOLEAN
|
||
FilterConnections (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine is intended to be called by SrvEnumApiHandler to check
|
||
whether a particular tree connect should be returned.
|
||
|
||
Arguments:
|
||
|
||
Srp - a pointer to the SRP for the operation. Name1 ("qualifier"
|
||
on NetConnectionEnum) is used to do the filtering.
|
||
|
||
Block - a pointer to the tree connect to check.
|
||
|
||
Return Value:
|
||
|
||
TRUE if the block should be placed in the output buffer, FALSE
|
||
if it should be passed over.
|
||
|
||
--*/
|
||
|
||
{
|
||
PTREE_CONNECT treeConnect = Block;
|
||
PUNICODE_STRING compareName;
|
||
|
||
PAGED_CODE( );
|
||
|
||
//
|
||
// We're going to compare the Name1 field against the share name
|
||
// if a computer name is the qualifier or against the computer
|
||
// name if the share name was the qualifier.
|
||
//
|
||
|
||
if ( Srp->Name1.Length > 2*sizeof(WCHAR) && *Srp->Name1.Buffer == '\\' &&
|
||
*(Srp->Name1.Buffer+1) == '\\' ) {
|
||
compareName =
|
||
&treeConnect->Connection->PagedConnection->ClientMachineNameString;
|
||
} else {
|
||
compareName = &treeConnect->Share->ShareName;
|
||
}
|
||
|
||
return RtlEqualUnicodeString(
|
||
&Srp->Name1,
|
||
compareName,
|
||
TRUE
|
||
);
|
||
|
||
} // FilterConnections
|
||
|
||
|
||
ULONG
|
||
SizeConnections (
|
||
IN PSERVER_REQUEST_PACKET Srp,
|
||
IN PVOID Block
|
||
)
|
||
|
||
/*++
|
||
|
||
Routine Description:
|
||
|
||
This routine returns the size the passed-in tree connect would take
|
||
up in an API output buffer.
|
||
|
||
Arguments:
|
||
|
||
Srp - a pointer to the SRP for the operation. The level and Name1
|
||
("qualifier" on NetConnectionEnum) are used.
|
||
|
||
Block - a pointer to the tree connect to size.
|
||
|
||
Return Value:
|
||
|
||
ULONG - The number of bytes the tree connect would take up in the
|
||
output buffer.
|
||
|
||
--*/
|
||
|
||
{
|
||
PTREE_CONNECT treeConnect = Block;
|
||
PUNICODE_STRING netName;
|
||
PUNICODE_STRING userName;
|
||
PSESSION session;
|
||
ULONG size;
|
||
|
||
PAGED_CODE( );
|
||
|
||
if ( Srp->Name1.Length > 2 && *Srp->Name1.Buffer == '\\' &&
|
||
*(Srp->Name1.Buffer+1) == '\\' ) {
|
||
netName = &treeConnect->Share->ShareName;
|
||
} else {
|
||
netName =
|
||
&treeConnect->Connection->PagedConnection->ClientMachineNameString;
|
||
}
|
||
|
||
//
|
||
// Attempt to find a reasonable user name. Since the SMB protocol
|
||
// does not link tree connects with users, only with sessions, it
|
||
// may not be possible to return a user name.
|
||
//
|
||
|
||
ACQUIRE_LOCK( &treeConnect->Connection->Lock );
|
||
|
||
session = SrvFindUserOnConnection( treeConnect->Connection );
|
||
|
||
if ( session != NULL ) {
|
||
userName = &session->UserName;
|
||
} else {
|
||
userName = NULL;
|
||
}
|
||
|
||
size = TOTAL_SIZE_OF_CONNECTION( treeConnect, Srp->Level, userName, netName );
|
||
|
||
RELEASE_LOCK( &treeConnect->Connection->Lock );
|
||
|
||
return size;
|
||
|
||
} // SizeConnections
|
||
|