WindowsXP-SP1/base/fs/srv/smbopen.c
2020-09-30 16:53:49 +02:00

3952 lines
112 KiB
C
Raw Permalink Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
smbopen.c
Abstract:
This module contains routines for processing the following SMBs:
Open
Open and X
Create
Create New
Create Temporary
*** The SearchAttributes field in open/create SMBs is always
ignored. This duplicates the LM 2.0 server behavior.
Author:
David Treadwell (davidtr) 23-Nov-1989
Manny Weiser (mannyw) 15-Apr-1991 (oplock support)
Revision History:
16-Apr-1991 mannyw
--*/
#include "precomp.h"
#include "smbopen.tmh"
#pragma hdrstop
#define BugCheckFileId SRV_FILE_SMBOPEN
//
// in smbtrans.c
//
SMB_STATUS SRVFASTCALL
ExecuteTransaction (
IN OUT PWORK_CONTEXT WorkContext
);
//
// Local functions
//
VOID
SetEofToMatchAllocation (
IN HANDLE FileHandle,
IN ULONG AllocationSize
);
VOID SRVFASTCALL
RestartOpen (
PWORK_CONTEXT WorkContext
);
SMB_PROCESSOR_RETURN_TYPE
GenerateOpenResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
);
VOID SRVFASTCALL
RestartOpenAndX (
PWORK_CONTEXT WorkContext
);
SMB_PROCESSOR_RETURN_TYPE
GenerateOpenAndXResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
);
VOID SRVFASTCALL
RestartOpen2 (
PWORK_CONTEXT WorkContext
);
SMB_TRANS_STATUS
GenerateOpen2Response (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
);
VOID SRVFASTCALL
RestartNtCreateAndX (
PWORK_CONTEXT WorkContext
);
SMB_PROCESSOR_RETURN_TYPE
GenerateNtCreateAndXResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
);
VOID SRVFASTCALL
RestartCreateWithSdOrEa (
PWORK_CONTEXT WorkContext
);
SMB_TRANS_STATUS
GenerateCreateWithSdOrEaResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
);
VOID SRVFASTCALL
RestartSmbCreate(
IN PWORK_CONTEXT WorkContext
);
SMB_PROCESSOR_RETURN_TYPE
GenerateCreateResponse(
PWORK_CONTEXT WorkContext,
NTSTATUS CreateStatus
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text( PAGE, SrvSmbOpen )
#pragma alloc_text( PAGE, RestartOpen )
#pragma alloc_text( PAGE, GenerateOpenResponse )
#pragma alloc_text( PAGE, SrvSmbOpenAndX )
#pragma alloc_text( PAGE, RestartOpenAndX )
#pragma alloc_text( PAGE, GenerateOpenAndXResponse )
#pragma alloc_text( PAGE, SrvSmbOpen2 )
#pragma alloc_text( PAGE, RestartOpen2 )
#pragma alloc_text( PAGE, GenerateOpen2Response )
#pragma alloc_text( PAGE, SrvSmbNtCreateAndX )
#pragma alloc_text( PAGE, RestartNtCreateAndX )
#pragma alloc_text( PAGE, GenerateNtCreateAndXResponse )
#pragma alloc_text( PAGE, SrvSmbCreateWithSdOrEa )
#pragma alloc_text( PAGE, RestartCreateWithSdOrEa )
#pragma alloc_text( PAGE, GenerateCreateWithSdOrEaResponse )
#pragma alloc_text( PAGE, SrvSmbCreate )
#pragma alloc_text( PAGE, SrvSmbCreateTemporary )
#pragma alloc_text( PAGE, SetEofToMatchAllocation )
#pragma alloc_text( PAGE, RestartSmbCreate )
#pragma alloc_text( PAGE, GenerateCreateResponse )
#endif
SMB_PROCESSOR_RETURN_TYPE
SrvSmbOpen (
SMB_PROCESSOR_PARAMETERS
)
/*++
Routine Description:
Processes an Open SMB. (This is the 'core' Open.)
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbprocs.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbprocs.h
--*/
{
PREQ_OPEN request;
NTSTATUS status = STATUS_SUCCESS;
USHORT access;
SMB_STATUS SmbStatus = SmbStatusInProgress;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN;
SrvWmiStartContext(WorkContext);
IF_SMB_DEBUG(OPEN_CLOSE1) {
KdPrint(( "Open file request header at 0x%p, response header at 0x%p\n",
WorkContext->RequestHeader,
WorkContext->ResponseHeader ));
KdPrint(( "Open file request parameters at 0x%p, response parameters at 0x%p\n",
WorkContext->RequestParameters,
WorkContext->ResponseParameters ));
}
request = (PREQ_OPEN)WorkContext->RequestParameters;
access = SmbGetUshort( &request->DesiredAccess );
status = SrvCreateFile(
WorkContext,
(USHORT)(access & ~SMB_DA_WRITE_THROUGH), // Allow write behind
(USHORT)0, // SmbFileAttributes
SMB_OFUN_OPEN_OPEN | SMB_OFUN_CREATE_FAIL,
(ULONG)0, // SmbAllocationSize
(PCHAR)(request->Buffer + 1),
END_OF_REQUEST_SMB( WorkContext ),
NULL,
0L,
NULL,
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK_NOTIFY_ANY) != 0 ?
OplockTypeBatch :
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
RestartOpen
);
if (status == STATUS_OPLOCK_BREAK_IN_PROGRESS) {
// The open is blocked (waiting for a comm device or an oplock
// break), do not send a response.
//
SmbStatus = SmbStatusInProgress;
} else if ( WorkContext->Parameters2.Open.TemporaryOpen ) {
// The initial open failed due to a sharing violation, possibly
// caused by an batch oplock. Requeue the open to a blocking
// thread.
//
WorkContext->FspRestartRoutine = SrvRestartSmbReceived;
SrvQueueWorkToBlockingThread( WorkContext );
SmbStatus = SmbStatusInProgress;
} else {
// The open has completed. Generate and send the reply.
//
SmbStatus = GenerateOpenResponse( WorkContext, status );
}
SrvWmiEndContext(WorkContext);
return SmbStatus;
} // SrvSmbOpen
VOID SRVFASTCALL
RestartOpen (
IN PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of an Open SMB. (This is the 'core' Open.)
Arguments:
WorkContext - A pointer to the work context block for this SMB.
Return Value:
None.
--*/
{
SMB_PROCESSOR_RETURN_LOCAL smbStatus = SmbStatusInProgress;
NTSTATUS openStatus = STATUS_SUCCESS;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN;
SrvWmiStartContext(WorkContext);
openStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if ( NT_SUCCESS( openStatus ) ) {
openStatus = WorkContext->Irp->IoStatus.Status;
} else {
//
// This open was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the open.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateOpenResponse(
WorkContext,
openStatus
);
SrvEndSmbProcessing( WorkContext, smbStatus );
SrvWmiEndContext(WorkContext);
return;
} // RestartOpen
SMB_PROCESSOR_RETURN_TYPE
GenerateOpenResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
)
/*++
Routine Description:
Generates a response to an Open SMB. (This is the 'core' Open.)
Arguments:
WorkContext -
Status - The status of the open operation.
Return Value:
The status of the SMB processing.
--*/
{
PRESP_OPEN response;
PREQ_OPEN request;
NTSTATUS status;
SRV_FILE_INFORMATION_ABBREVIATED srvFileInformation;
PRFCB rfcb;
USHORT access;
PAGED_CODE( );
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS( OpenStatus ) ) {
SrvSetSmbError( WorkContext, OpenStatus );
return SmbStatusSendResponse;
}
rfcb = WorkContext->Rfcb;
response = (PRESP_OPEN)WorkContext->ResponseParameters;
request = (PREQ_OPEN)WorkContext->RequestParameters;
access = SmbGetUshort( &request->DesiredAccess ); // save for later use
//
// Get the additional information that needs to be returned in the
// response SMB. We always open with FILE_READ_ATTRIBUTES, so no
// access check is required.
//
status = SrvQueryInformationFileAbbreviated(
rfcb->Lfcb->FileHandle,
rfcb->Lfcb->FileObject,
&srvFileInformation,
FALSE,
WorkContext->TreeConnect->Share->ShareType
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(ERRORS) {
KdPrint(( "GenerateOpenResponse: SrvQueryInformationFile failed: %X\n",
status ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError( WorkContext, status );
return SmbStatusSendResponse;
}
rfcb->Mfcb->NonpagedMfcb->OpenFileSize = srvFileInformation.DataSize;
//
// Give the smart card a chance to get into the act
//
if( WorkContext->Endpoint->IsConnectionless && SrvIpxSmartCard.Open != NULL ) {
PVOID handle;
IF_DEBUG( SIPX ) {
KdPrint(( "Trying the smart card for %wZ\n", &rfcb->Mfcb->FileName ));
}
if( SrvIpxSmartCard.Open(
WorkContext->RequestBuffer->Buffer,
rfcb->Lfcb->FileObject,
&rfcb->Mfcb->FileName,
&(WorkContext->ClientAddress->IpxAddress.Address[0].Address[0]),
rfcb->Lfcb->FileObject->Flags & FO_CACHE_SUPPORTED,
&handle
) == TRUE ) {
IF_DEBUG( SIPX ) {
KdPrint(( "%wZ handled by Smart Card. Handle %p\n",
&rfcb->Mfcb->FileName, handle ));
}
rfcb->PagedRfcb->IpxSmartCardContext = handle;
}
}
//
// Set up fields of response SMB. Note that we copy the desired
// access to the granted access in the response. They must be the
// same, else the request would have failed.
//
// !!! This will not be the case for compatibility mode and FCB opens!
//
response->WordCount = 7;
SmbPutUshort( &response->Fid, rfcb->Fid );
SmbPutUshort( &response->FileAttributes, srvFileInformation.Attributes );
SmbPutUlong(
&response->LastWriteTimeInSeconds,
srvFileInformation.LastWriteTimeInSeconds
);
SmbPutUlong( &response->DataSize, srvFileInformation.DataSize.LowPart );
SmbPutUshort(
&response->GrantedAccess,
access );
SmbPutUshort( &response->ByteCount, 0 );
WorkContext->ResponseParameters = NEXT_LOCATION( response, RESP_OPEN, 0 );
IF_DEBUG(TRACE2) KdPrint(( "GenerateOpenResponse complete.\n" ));
return SmbStatusSendResponse;
} // GenerateOpenResponse
SMB_PROCESSOR_RETURN_TYPE
SrvSmbOpenAndX (
SMB_PROCESSOR_PARAMETERS
)
/*++
Routine Description:
Processes an OpenAndX SMB.
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbprocs.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbprocs.h
--*/
{
PREQ_OPEN_ANDX request;
NTSTATUS status = STATUS_SUCCESS;
USHORT access;
SMB_STATUS smbStatus = SmbStatusInProgress;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN_AND_X;
SrvWmiStartContext(WorkContext);
IF_SMB_DEBUG(OPEN_CLOSE1) {
KdPrint(( "Open file and X request header at 0x%p, response header at 0x%p\n",
WorkContext->RequestHeader,
WorkContext->ResponseHeader ));
KdPrint(( "Open file and X request parameters at 0x%p, response parameters at 0x%p\n",
WorkContext->RequestParameters,
WorkContext->ResponseParameters ));
}
//
// If we are not on a blocking thread and we don't have a license
// from the license server, shift the request over to the blocking work
// queue since acquiring a license is an expensive operation and we don't
// want to congest our nonblocking worker threads
//
if( WorkContext->UsingBlockingThread == 0 ) {
PSESSION session;
PTREE_CONNECT treeConnect;
status = SrvVerifyUidAndTid(
WorkContext,
&session,
&treeConnect,
ShareTypeWild
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(SMB_ERRORS) {
KdPrint(( "SrvSmbOpenAndX: Invalid UID or TID\n" ));
}
smbStatus = GenerateOpenAndXResponse( WorkContext, status );
goto Cleanup;
}
if( session->IsLSNotified == FALSE ) {
//
// Insert the work item at the tail of the blocking work queue.
//
SrvInsertWorkQueueTail(
&SrvBlockingWorkQueue,
(PQUEUEABLE_BLOCK_HEADER)WorkContext
);
smbStatus = SmbStatusInProgress;
goto Cleanup;
}
}
request = (PREQ_OPEN_ANDX)WorkContext->RequestParameters;
access = SmbGetUshort( &request->DesiredAccess ); // save for later use
status = SrvCreateFile(
WorkContext,
access,
SmbGetUshort( &request->FileAttributes ),
SmbGetUshort( &request->OpenFunction ),
SmbGetUlong( &request->AllocationSize ),
(PCHAR)request->Buffer,
END_OF_REQUEST_SMB( WorkContext ),
NULL,
0L,
NULL,
(SmbGetUshort(&request->Flags) & SMB_OPEN_OPBATCH) != 0 ?
OplockTypeBatch :
(SmbGetUshort(&request->Flags) & SMB_OPEN_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
RestartOpenAndX
);
if ( status == STATUS_OPLOCK_BREAK_IN_PROGRESS ) {
//
// The open is blocked (waiting for a comm device or an oplock
// break), do not send a reply.
//
smbStatus = SmbStatusInProgress;
} else if ( WorkContext->Parameters2.Open.TemporaryOpen ) {
//
// The initial open failed due to a sharing violation, possibly
// caused by an batch oplock. Requeue the open to a blocking
// thread.
//
WorkContext->FspRestartRoutine = SrvRestartSmbReceived;
SrvQueueWorkToBlockingThread( WorkContext );
smbStatus = SmbStatusInProgress;
} else {
//
// The open has completed. Generate and send the reply.
//
smbStatus = GenerateOpenAndXResponse( WorkContext, status );
}
Cleanup:
SrvWmiEndContext(WorkContext);
return smbStatus;
} // SrvSmbOpenAndX
VOID SRVFASTCALL
RestartOpenAndX (
PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of an Open and X SMB.
Arguments:
WorkContext - Work context block for the operation.
Return Value:
None.
--*/
{
SMB_PROCESSOR_RETURN_LOCAL smbStatus = SmbStatusInProgress;
NTSTATUS openStatus = STATUS_SUCCESS;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN_AND_X;
SrvWmiStartContext(WorkContext);
openStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if ( NT_SUCCESS( openStatus ) ) {
openStatus = WorkContext->Irp->IoStatus.Status;
if( NT_SUCCESS( openStatus ) ) {
//
// It's obvious that the file already existed, because we've
// been working on an oplock break. So set the
// IoStatus.Information field correctly.
//
WorkContext->Irp->IoStatus.Information = FILE_OPENED;
}
} else {
//
// This open was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the open.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateOpenAndXResponse(
WorkContext,
openStatus
);
if ( smbStatus == SmbStatusMoreCommands ) {
SrvProcessSmb( WorkContext );
} else {
SrvEndSmbProcessing( WorkContext, SmbStatusSendResponse );
}
SrvWmiEndContext(WorkContext);
return;
} // RestartOpenAndX
SMB_PROCESSOR_RETURN_TYPE
GenerateOpenAndXResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
)
/*++
Routine Description:
Generates a response to an Open and X SMB and setup for furthur
SMB processing.
Arguments:
WorkContext - Work context block for the operation.
OpenStatus - The status of the open operation.
Return Value:
None.
--*/
{
PREQ_OPEN_ANDX request;
PRESP_OPEN_ANDX response;
SRV_FILE_INFORMATION_ABBREVIATED srvFileInformation;
BOOLEAN reqAdditionalInformation;
BOOLEAN reqExtendedResponse;
PRFCB rfcb;
PLFCB lfcb;
PIO_STATUS_BLOCK ioStatusBlock;
UCHAR nextCommand;
USHORT reqAndXOffset;
USHORT access;
USHORT action = 0;
OPLOCK_TYPE oplockType;
NTSTATUS status;
PAGED_CODE( );
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS( OpenStatus ) ) {
SrvSetSmbError( WorkContext, OpenStatus );
//
// Remap the error if it is ERROR_ALREADY_EXISTS
//
if ( !CLIENT_CAPABLE_OF(NT_STATUS, WorkContext->Connection) &&
SmbGetUshort( &WorkContext->ResponseHeader->Error ) ==
ERROR_ALREADY_EXISTS ) {
SmbPutUshort(
&WorkContext->ResponseHeader->Error,
ERROR_FILE_EXISTS
);
}
return SmbStatusSendResponse;
}
request = (PREQ_OPEN_ANDX)WorkContext->RequestParameters;
response = (PRESP_OPEN_ANDX)WorkContext->ResponseParameters;
access = SmbGetUshort( &request->DesiredAccess ); // save for later use
rfcb = WorkContext->Rfcb;
lfcb = rfcb->Lfcb;
reqExtendedResponse = (BOOLEAN)( (SmbGetUshort(&request->Flags) &
SMB_OPEN_EXTENDED_RESPONSE) != 0);
//
// Attempt to acquire the oplock.
//
if ( WorkContext->TreeConnect->Share->ShareType != ShareTypePrint ) {
if ( (SmbGetUshort( &request->Flags ) & SMB_OPEN_OPBATCH) != 0 ) {
oplockType = OplockTypeBatch;
} else if ( (SmbGetUshort( &request->Flags ) & SMB_OPEN_OPLOCK) != 0 ) {
oplockType = OplockTypeExclusive;
} else {
oplockType = OplockTypeServerBatch;
}
if ( SrvRequestOplock( WorkContext, &oplockType, FALSE ) ) {
//
// The oplock was granted. Save in action so that we tell
// the client he has an oplock and update statistics.
//
action = SMB_OACT_OPLOCK;
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksGranted );
} else {
//
// The oplock request was denied. Update statistics.
//
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksDenied );
}
}
//
// If the file was created, set the EOF location to be the same as
// the size of the file. This is necessary for compatibility with
// OS/2, which only has EOF, not a separate allocation size.
//
ioStatusBlock = &WorkContext->Irp->IoStatus;
if ( (ioStatusBlock->Information == FILE_CREATED) ||
(ioStatusBlock->Information == FILE_OVERWRITTEN) ) {
//
// Extending EOF is only legal if the client has write access
// to the file. If the client doesn't have write access, don't
// extend the file.
//
// *** This is an incompatibility with OS/2.
if ( rfcb->WriteAccessGranted || rfcb->AppendAccessGranted ) {
SetEofToMatchAllocation(
lfcb->FileHandle,
SmbGetUlong( &request->AllocationSize )
);
} else {
SrvStatistics.GrantedAccessErrors++;
}
}
//
// If the consumer requested additional information, find it now.
//
reqAdditionalInformation = (BOOLEAN)( (SmbGetUshort(&request->Flags) &
SMB_OPEN_QUERY_INFORMATION) != 0);
if ( reqAdditionalInformation ||
( !rfcb->WriteAccessGranted && rfcb->AppendAccessGranted) ) {
//
// We always open with at least FILE_READ_ATTRIBUTES, so no
// access check is needed.
//
status = SrvQueryInformationFileAbbreviated(
lfcb->FileHandle,
lfcb->FileObject,
&srvFileInformation,
reqExtendedResponse,
WorkContext->TreeConnect->Share->ShareType
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbOpenAndX: SrvQueryInformationFile failed: "
"%X\n", status ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError( WorkContext, status );
return SmbStatusSendResponse;
}
rfcb->Mfcb->NonpagedMfcb->OpenFileSize = srvFileInformation.DataSize;
}
//
// Give the smart card a chance to get into the act
//
if( WorkContext->Endpoint->IsConnectionless && SrvIpxSmartCard.Open != NULL ) {
PVOID handle;
IF_DEBUG( SIPX ) {
KdPrint(( "Trying the smart card for %wZ\n", &rfcb->Mfcb->FileName ));
}
if( SrvIpxSmartCard.Open(
WorkContext->RequestBuffer->Buffer,
rfcb->Lfcb->FileObject,
&rfcb->Mfcb->FileName,
&(WorkContext->ClientAddress->IpxAddress.Address[0].Address[0]),
rfcb->Lfcb->FileObject->Flags & FO_CACHE_SUPPORTED,
&handle
) == TRUE ) {
IF_DEBUG( SIPX ) {
KdPrint(( "%wZ handled by Smart Card. Handle %p\n",
&rfcb->Mfcb->FileName, handle ));
}
rfcb->PagedRfcb->IpxSmartCardContext = handle;
}
}
//
// Set up response SMB.
//
nextCommand = request->AndXCommand;
reqAndXOffset = SmbGetUshort( &request->AndXOffset );
if (reqExtendedResponse) {
NTSTATUS ExtendedResponseStatus;
PRESP_EXTENDED_OPEN_ANDX ExtendedResponse;
ExtendedResponse = (PRESP_EXTENDED_OPEN_ANDX)response;
ExtendedResponseStatus = SrvUpdateMaximalAccessRightsInResponse(
WorkContext,
&ExtendedResponse->MaximalAccessRights,
&ExtendedResponse->GuestMaximalAccessRights);
// Fall back to regular response if there was an error in obtainaing
// the maximal access rights
reqExtendedResponse = (ExtendedResponseStatus == STATUS_SUCCESS);
}
response->AndXCommand = nextCommand;
response->AndXReserved = 0;
if (!reqExtendedResponse) {
SmbPutUshort(
&response->AndXOffset,
GET_ANDX_OFFSET(
WorkContext->ResponseHeader,
WorkContext->ResponseParameters,
RESP_OPEN_ANDX,
0
)
);
response->WordCount = 15;
} else {
SmbPutUshort(
&response->AndXOffset,
GET_ANDX_OFFSET(
WorkContext->ResponseHeader,
WorkContext->ResponseParameters,
RESP_EXTENDED_OPEN_ANDX,
0
)
);
response->WordCount = 19;
}
SmbPutUshort( &response->Fid, rfcb->Fid );
//
// If the consumer requested additional information, set appropiate
// fields, else set the fields to zero.
//
if ( reqAdditionalInformation ) {
SmbPutUshort(
&response->FileAttributes,
srvFileInformation.Attributes
);
SmbPutUlong(
&response->LastWriteTimeInSeconds,
srvFileInformation.LastWriteTimeInSeconds
);
SmbPutUlong( &response->DataSize, srvFileInformation.DataSize.LowPart );
access &= SMB_DA_SHARE_MASK;
if( rfcb->ReadAccessGranted && (rfcb->WriteAccessGranted || rfcb->AppendAccessGranted) ) {
access |= SMB_DA_ACCESS_READ_WRITE;
} else if( rfcb->ReadAccessGranted ) {
access |= SMB_DA_ACCESS_READ;
} else if( rfcb->WriteAccessGranted || rfcb->AppendAccessGranted ) {
access |= SMB_DA_ACCESS_WRITE;
}
SmbPutUshort( &response->GrantedAccess, access );
SmbPutUshort( &response->FileType, srvFileInformation.Type );
SmbPutUshort( &response->DeviceState, srvFileInformation.HandleState );
} else {
RtlZeroMemory( (PVOID)&response->FileAttributes, 16 );
}
//
// Bit field mapping of Action:
//
// Lrrr rrrr rrrr rrOO
//
// where:
//
// L - Lock (single-user total file lock status)
// 0 - file opened by another user
// 1 - file is opened only by this user at the present time
//
// O - Open (action taken on open)
// 1 - the file existed and was opened
// 2 - the file did not exist but was created
// 3 - the file existed and was truncated
//
switch ( ioStatusBlock->Information ) {
case FILE_OPENED:
action |= SMB_OACT_OPENED;
break;
case FILE_CREATED:
action |= SMB_OACT_CREATED;
break;
case FILE_OVERWRITTEN:
action |= SMB_OACT_TRUNCATED;
break;
default:
IF_DEBUG(ERRORS) {
KdPrint(( "Unknown Information value in IO status block: 0x%p\n",
(PVOID)(ioStatusBlock->Information) ));
}
}
SmbPutUshort( &response->Action, action );
SmbPutUlong( &response->ServerFid, (ULONG)0 );
SmbPutUshort( &response->Reserved, 0 );
SmbPutUshort( &response->ByteCount, 0 );
WorkContext->ResponseParameters = (PCHAR)WorkContext->ResponseHeader +
SmbGetUshort( &response->AndXOffset );
//
// Test for legal followon command.
//
switch ( nextCommand ) {
case SMB_COM_NO_ANDX_COMMAND:
break;
case SMB_COM_READ:
case SMB_COM_READ_ANDX:
case SMB_COM_IOCTL:
//
// Make sure the AndX command is still within the received SMB
//
if( (PCHAR)WorkContext->RequestHeader + reqAndXOffset <=
END_OF_REQUEST_SMB( WorkContext ) ) {
break;
}
/* Falls Through */
default: // Illegal followon command
IF_DEBUG(SMB_ERRORS) {
KdPrint(( "SrvSmbOpenAndX: Illegal followon command: 0x%lx\n",
nextCommand ));
}
//
// Return an error indicating that the followon command was bad.
// Note that the open is still considered successful, so the
// file remains open.
//
SrvSetSmbError( WorkContext, STATUS_INVALID_SMB );
return SmbStatusSendResponse;
}
//
// If there is an AndX command, set up to process it. Otherwise,
// indicate completion to the caller.
//
if ( nextCommand != SMB_COM_NO_ANDX_COMMAND ) {
WorkContext->NextCommand = nextCommand;
WorkContext->RequestParameters = (PCHAR)WorkContext->RequestHeader +
reqAndXOffset;
return SmbStatusMoreCommands;
}
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbOpenAndX complete.\n" ));
return SmbStatusSendResponse;
} // GenerateOpenAndXResponse
SMB_TRANS_STATUS
SrvSmbOpen2 (
IN OUT PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Processes an Open2 SMB. This request arrives in a Transaction2 SMB.
Arguments:
WorkContext - Supplies the address of a Work Context Block
describing the current request. See smbtypes.h for a more
complete description of the valid fields.
Return Value:
BOOLEAN - Indicates whether an error occurred. See smbtypes.h for a
more complete description.
--*/
{
PREQ_OPEN2 request;
PRESP_OPEN2 response;
NTSTATUS status = STATUS_SUCCESS;
SMB_TRANS_STATUS SmbStatus = SmbTransStatusInProgress;
USHORT access;
PTRANSACTION transaction;
ULONG eaErrorOffset = 0;
USHORT os2EaErrorOffset = 0;
PFILE_FULL_EA_INFORMATION ntFullEa;
ULONG ntFullEaBufferLength;
PFEALIST feaList;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN2;
SrvWmiStartContext(WorkContext);
transaction = WorkContext->Parameters.Transaction;
request = (PREQ_OPEN2)transaction->InParameters;
response = (PRESP_OPEN2)transaction->OutParameters;
//
// Verify that enough parameter bytes were sent and that we're allowed
// to return enough parameter bytes.
//
if ( (transaction->ParameterCount < sizeof(REQ_OPEN2)) ||
(transaction->MaxParameterCount < sizeof(RESP_OPEN2)) ) {
//
// Not enough parameter bytes were sent.
//
IF_DEBUG(SMB_ERRORS)
KdPrint(( "SrvSmbOpen2: bad parameter byte counts: %ld %ld\n",
transaction->ParameterCount,
transaction->MaxParameterCount ));
SrvLogInvalidSmb( WorkContext );
SrvSetSmbError2( WorkContext, STATUS_INVALID_SMB, TRUE );
status = STATUS_INVALID_SMB;
goto err_exit;
}
//
// Convert the EA list to NT style.
//
eaErrorOffset = 0;
feaList = (PFEALIST)transaction->InData;
//
// Make sure that the value in Fealist->cbList is legitimate and that
// there is at least sufficient data for a single EA.
//
// This code will fail if no EA list is indicated. Don't know
// if this is correct, but it was the previous behavior so
// it's been enforced.
//
if ( transaction->DataCount <= sizeof(FEALIST) ||
SmbGetUlong( &feaList->cbList ) <= sizeof(FEALIST) ||
SmbGetUlong( &feaList->cbList ) > transaction->DataCount) {
IF_DEBUG(SMB_ERRORS) {
KdPrint(( "EA list size bad: size = %ld, data sent was %ld\n",
SmbGetUlong( &feaList->cbList ), transaction->DataCount ));
}
SrvSetSmbError2( WorkContext, STATUS_OS2_EA_LIST_INCONSISTENT, TRUE );
status = STATUS_OS2_EA_LIST_INCONSISTENT;
goto err_exit;
}
//
// Convert the FEALIST to NT style.
//
status = SrvOs2FeaListToNt(
feaList,
&ntFullEa,
&ntFullEaBufferLength,
&os2EaErrorOffset
);
if ( !NT_SUCCESS(status) ) {
SrvSetSmbError2( WorkContext, status, TRUE );
goto err_exit;
}
access = SmbGetUshort( &request->DesiredAccess ); // save for later use
status = SrvCreateFile(
WorkContext,
access,
SmbGetUshort( &request->FileAttributes ),
SmbGetUshort( &request->OpenFunction ),
SmbGetUlong( &request->AllocationSize ),
(PCHAR)request->Buffer,
END_OF_TRANSACTION_PARAMETERS( transaction ),
ntFullEa,
ntFullEaBufferLength,
&eaErrorOffset,
(SmbGetUshort(&request->Flags) & SMB_OPEN_OPBATCH) != 0 ?
OplockTypeBatch :
(SmbGetUshort(&request->Flags) & SMB_OPEN_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
RestartOpen2
);
if ( status == STATUS_OPLOCK_BREAK_IN_PROGRESS ) {
//
// The open is blocked (waiting for a comm device or an oplock
// break), do not send a reply.
//
//
// Save a pointer to the full ea structure
//
WorkContext->Parameters2.Open.NtFullEa = ntFullEa;
WorkContext->Parameters2.Open.EaErrorOffset = eaErrorOffset;
SmbStatus = SmbTransStatusInProgress;
goto Cleanup;
} else if ( WorkContext->Parameters2.Open.TemporaryOpen ) {
//
// The initial open failed due to a sharing violation, possibly
// caused by an batch oplock. Requeue the open to a blocking
// thread.
//
// We need to free the EA buffer in this case.
//
DEALLOCATE_NONPAGED_POOL(ntFullEa);
WorkContext->FspRestartRoutine = (PRESTART_ROUTINE)ExecuteTransaction;
SrvQueueWorkToBlockingThread( WorkContext );
SmbStatus = SmbStatusInProgress;
goto Cleanup;
} else {
//
// Save a pointer to the full ea structure
//
WorkContext->Parameters2.Open.NtFullEa = ntFullEa;
WorkContext->Parameters2.Open.EaErrorOffset = eaErrorOffset;
//
// The open has completed. Generate and send the reply.
//
SmbStatus = GenerateOpen2Response( WorkContext, status );
goto Cleanup;
}
err_exit:
RtlZeroMemory( (PVOID)&response->Fid, 24 );
SmbPutUshort( &response->EaErrorOffset, os2EaErrorOffset );
SmbPutUlong( &response->EaLength, 0 );
transaction->SetupCount = 0;
transaction->ParameterCount = sizeof(RESP_OPEN2);
transaction->DataCount = 0;
SmbStatus = SmbTransStatusErrorWithData;
Cleanup:
SrvWmiEndContext(WorkContext);
return SmbStatus;
} // SrvSmbOpen2
VOID SRVFASTCALL
RestartOpen2 (
PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of an Open2 SMB.
Arguments:
WorkContext - Work context block for the operation.
Return Value:
None.
--*/
{
SMB_TRANS_STATUS smbStatus = SmbTransStatusInProgress;
NTSTATUS openStatus = STATUS_SUCCESS;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_OPEN2;
SrvWmiStartContext(WorkContext);
openStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if ( NT_SUCCESS( openStatus ) ) {
openStatus = WorkContext->Irp->IoStatus.Status;
if( NT_SUCCESS( openStatus ) ) {
//
// It's obvious that the file already existed, because we've
// been working on an oplock break. So set the
// IoStatus.Information field correctly.
//
WorkContext->Irp->IoStatus.Information = FILE_OPENED;
}
} else {
//
// This open was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the open.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateOpen2Response(
WorkContext,
openStatus
);
SrvCompleteExecuteTransaction( WorkContext, smbStatus );
SrvWmiEndContext(WorkContext);
return;
} // RestartOpen2
SMB_TRANS_STATUS
GenerateOpen2Response (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
)
/*++
Routine Description:
Generates a response to an Open and X SMB and setup for furthur
SMB processing.
Arguments:
WorkContext - Work context block for the operation.
OpenStatus - The status of the open operation.
Return Value:
None.
--*/
{
PREQ_OPEN2 request;
PRESP_OPEN2 response;
PRFCB rfcb;
PLFCB lfcb;
NTSTATUS status;
BOOLEAN reqAdditionalInformation;
SRV_FILE_INFORMATION_ABBREVIATED srvFileInformation;
BOOLEAN reqEaLength;
FILE_EA_INFORMATION fileEaInformation;
USHORT access;
USHORT action = 0;
PTRANSACTION transaction;
ULONG eaErrorOffset = 0;
USHORT os2EaErrorOffset = 0;
PFILE_FULL_EA_INFORMATION ntFullEa;
PFEALIST feaList;
OPLOCK_TYPE oplockType;
PAGED_CODE( );
transaction = WorkContext->Parameters.Transaction;
request = (PREQ_OPEN2)transaction->InParameters;
response = (PRESP_OPEN2)transaction->OutParameters;
feaList = (PFEALIST)transaction->InData;
ntFullEa = WorkContext->Parameters2.Open.NtFullEa;
eaErrorOffset = WorkContext->Parameters2.Open.EaErrorOffset;
access = SmbGetUshort( &request->DesiredAccess ); // save for later use
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS( OpenStatus ) ) {
SrvSetSmbError( WorkContext, OpenStatus );
//
// Remap the error if it is ERROR_ALREADY_EXISTS.
//
if ( !CLIENT_CAPABLE_OF(NT_STATUS,WorkContext->Connection) &&
SmbGetUshort( &WorkContext->ResponseHeader->Error ) ==
ERROR_ALREADY_EXISTS ) {
SmbPutUshort(
&WorkContext->ResponseHeader->Error,
ERROR_FILE_EXISTS
);
}
//
// If an EA error offset was returned, convert it from an offset
// into the NT full EA list to an offset in the OS/2 1.2 FEALIST.
//
if ( eaErrorOffset != 0 ) {
os2EaErrorOffset = SrvGetOs2FeaOffsetOfError(
eaErrorOffset,
ntFullEa,
feaList
);
}
DEALLOCATE_NONPAGED_POOL( ntFullEa );
goto err_exit;
}
DEALLOCATE_NONPAGED_POOL( ntFullEa );
//
// If the file was created, set the EOF location to be the same as
// the size of the file. This is necessary for compatibility with
// OS/2, which only has EOF, not a separate allocation size.
//
rfcb = WorkContext->Rfcb;
lfcb = rfcb->Lfcb;
if ( (WorkContext->Irp->IoStatus.Information == FILE_CREATED) ||
(WorkContext->Irp->IoStatus.Information == FILE_OVERWRITTEN) ) {
//
// Extending EOF is only legal if the client has write access
// to the file. If the client doesn't have write access, don't
// extend the file.
//
// *** This is an incompatibility with OS/2.
if ( rfcb->WriteAccessGranted || rfcb->AppendAccessGranted ) {
SetEofToMatchAllocation(
lfcb->FileHandle,
SmbGetUlong( &request->AllocationSize )
);
} else {
SrvStatistics.GrantedAccessErrors++;
}
}
//
// If the consumer requested additional information, find it now.
//
reqAdditionalInformation =
(BOOLEAN)((SmbGetUshort( &request->Flags ) &
SMB_OPEN_QUERY_INFORMATION) != 0);
reqEaLength =
(BOOLEAN)((SmbGetUshort( &request->Flags ) &
SMB_OPEN_QUERY_EA_LENGTH) != 0);
if ( reqAdditionalInformation ||
(!rfcb->WriteAccessGranted && rfcb->AppendAccessGranted) ) {
//
// We always open with at least FILE_READ_ATTRIBUTES, so no
// access check is needed.
//
status = SrvQueryInformationFileAbbreviated(
lfcb->FileHandle,
lfcb->FileObject,
&srvFileInformation,
FALSE,
WorkContext->TreeConnect->Share->ShareType
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbOpen2: SrvQueryInformationFile failed: "
"%X\n", status ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError2( WorkContext, status, TRUE );
goto err_exit;
}
rfcb->Mfcb->NonpagedMfcb->OpenFileSize = srvFileInformation.DataSize;
}
if ( reqEaLength ) {
IO_STATUS_BLOCK eaIoStatusBlock;
status = NtQueryInformationFile(
lfcb->FileHandle,
&eaIoStatusBlock,
&fileEaInformation,
sizeof(FILE_EA_INFORMATION),
FileEaInformation
);
if ( NT_SUCCESS(status) ) {
status = eaIoStatusBlock.Status;
}
if ( !NT_SUCCESS(status) ) {
INTERNAL_ERROR(
ERROR_LEVEL_UNEXPECTED,
"SrvSmbOpen2: NtQueryInformationFile (file information)"
"returned %X",
status,
NULL
);
SrvCloseRfcb( rfcb );
SrvSetSmbError2( WorkContext, status, TRUE );
goto err_exit;
} else {
//
// Adjust the EA size. If there are no EAs, OS/2 expects
// EA size = 4.
//
if (fileEaInformation.EaSize == 0) {
fileEaInformation.EaSize = 4;
}
}
} else {
fileEaInformation.EaSize = 0;
}
//
// Attempt to acquire the oplock.
//
if ( WorkContext->TreeConnect->Share->ShareType != ShareTypePrint ) {
if ( (SmbGetUshort( &request->Flags ) & SMB_OPEN_OPBATCH) != 0 ) {
oplockType = OplockTypeBatch;
} else if ( (SmbGetUshort( &request->Flags ) & SMB_OPEN_OPLOCK) != 0 ) {
oplockType = OplockTypeExclusive;
} else {
oplockType = OplockTypeServerBatch;
}
if ( SrvRequestOplock( WorkContext, &oplockType, FALSE ) ) {
action = SMB_OACT_OPLOCK;
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksGranted );
} else {
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksDenied );
}
}
//
// Set up response SMB.
//
SmbPutUshort( &response->Fid, rfcb->Fid );
//
// If the consumer requested additional information, set appropiate
// fields, else set the fields to zero.
//
if ( reqAdditionalInformation ) {
SmbPutUshort(
&response->FileAttributes,
srvFileInformation.Attributes
);
SmbPutUlong( &response->DataSize, srvFileInformation.DataSize.LowPart );
SmbPutUshort( &response->GrantedAccess, access );
SmbPutUshort( &response->FileType, srvFileInformation.Type );
SmbPutUshort( &response->DeviceState, srvFileInformation.HandleState );
} else {
RtlZeroMemory( (PVOID)&response->FileAttributes, 16 );
}
//
// Bit field mapping of Action:
//
// Lrrr rrrr rrrr rrOO
//
// where:
//
// L - Lock (single-user total file lock status)
// 0 - file opened by another user
// 1 - file is opened only by this user at the present time
//
// O - Open (action taken on open)
// 1 - the file existed and was opened
// 2 - the file did not exist but was created
// 3 - the file existed and was truncated
//
switch ( WorkContext->Irp->IoStatus.Information ) {
case FILE_OPENED:
action |= SMB_OACT_OPENED;
break;
case FILE_CREATED:
action |= SMB_OACT_CREATED;
break;
case FILE_OVERWRITTEN:
action |= SMB_OACT_TRUNCATED;
break;
default:
INTERNAL_ERROR(
ERROR_LEVEL_UNEXPECTED,
"SrvSmbOpen2: Unknown Information value in IO status"
"block: 0x%lx\n",
WorkContext->Irp->IoStatus.Information,
NULL
);
SrvLogServiceFailure( SRV_SVC_IO_CREATE_FILE,
(ULONG)WorkContext->Irp->IoStatus.Information );
}
SmbPutUshort( &response->Action, action );
SmbPutUlong( &response->ServerFid, (ULONG)0 );
SmbPutUshort( &response->EaErrorOffset, 0 );
SmbPutUlong( &response->EaLength, fileEaInformation.EaSize );
transaction->SetupCount = 0;
transaction->ParameterCount = sizeof(RESP_OPEN2);
transaction->DataCount = 0;
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbOpen2 complete.\n" ));
return SmbTransStatusSuccess;
err_exit:
RtlZeroMemory( (PVOID)&response->Fid, 24 );
SmbPutUshort( &response->EaErrorOffset, os2EaErrorOffset );
SmbPutUlong( &response->EaLength, 0 );
transaction->SetupCount = 0;
transaction->ParameterCount = sizeof(RESP_OPEN2);
transaction->DataCount = 0;
return SmbTransStatusErrorWithData;
} // GenerateOpen2Response
SMB_PROCESSOR_RETURN_TYPE
SrvSmbNtCreateAndX (
SMB_PROCESSOR_PARAMETERS
)
/*++
Routine Description:
Processes an NtCreateAndX SMB.
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbprocs.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbprocs.h
--*/
{
PREQ_NT_CREATE_ANDX request;
#ifdef INCLUDE_SMB_IFMODIFIED
PREQ_NT_CREATE_IF_MODIFIED_ANDX requestIfModified = NULL;
BOOLEAN ifModifiedCheck = FALSE;
#endif
ULONG flags;
NTSTATUS status = STATUS_SUCCESS;
SMB_STATUS SmbStatus = SmbStatusInProgress;
LARGE_INTEGER allocationSize;
UNICODE_STRING fileName;
PUCHAR name;
USHORT nameLength;
SECURITY_QUALITY_OF_SERVICE qualityOfService;
BOOLEAN isUnicode;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_NT_CREATE_AND_X;
SrvWmiStartContext(WorkContext);
IF_SMB_DEBUG(OPEN_CLOSE1) {
KdPrint(( "Create file and X request header at 0x%p, response header at 0x%p\n",
WorkContext->RequestHeader,
WorkContext->ResponseHeader ));
KdPrint(( "Create file and X request parameters at 0x%p, response parameters at 0x%p\n",
WorkContext->RequestParameters,
WorkContext->ResponseParameters ));
}
request = (PREQ_NT_CREATE_ANDX)WorkContext->RequestParameters;
allocationSize.HighPart = SmbGetUlong( &request->AllocationSize.HighPart );
allocationSize.LowPart = SmbGetUlong( &request->AllocationSize.LowPart );
flags = SmbGetUlong( &request->Flags );
#ifdef INCLUDE_SMB_IFMODIFIED
if (IS_POSTNT5_DIALECT( WorkContext->Connection->SmbDialect )) {
ifModifiedCheck = ((flags & NT_CREATE_IFMODIFIED_REQUEST) != 0 );
if (ifModifiedCheck) {
requestIfModified = (PREQ_NT_CREATE_IF_MODIFIED_ANDX) request;
}
}
#endif
//
// First verify that the file path name does not extend beyond the
// end of the SMB.
//
isUnicode = SMB_IS_UNICODE( WorkContext );
//
// if this request has IfModifiedSince fields, the request is of a
// slightly different format, the Name field is lower.
//
#ifdef INCLUDE_SMB_IFMODIFIED
if (requestIfModified != NULL) {
name = (PUCHAR)requestIfModified->Buffer;
} else {
name = (PUCHAR)request->Buffer;
}
#else
name = (PUCHAR)request->Buffer;
#endif
if ( isUnicode ) {
name = ALIGN_SMB_WSTR( name );
}
nameLength = SmbGetUshort( &request->NameLength );
if ( name + nameLength > ( END_OF_REQUEST_SMB( WorkContext ) + 1 ) ) {
status = STATUS_INVALID_SMB;
SmbStatus = GenerateNtCreateAndXResponse( WorkContext, STATUS_INVALID_SMB );
goto Cleanup;
}
//
// Convert the file name to a Unicode string.
//
status = SrvMakeUnicodeString(
isUnicode,
&fileName,
name,
&nameLength
);
if ( !NT_SUCCESS( status ) ) {
status = STATUS_INSUFF_SERVER_RESOURCES;
SmbStatus = GenerateNtCreateAndXResponse(
WorkContext,
STATUS_INSUFF_SERVER_RESOURCES
);
goto Cleanup;
}
//
// *** We always ask for STATIC tracking, not DYNAMIC, because we
// don't support dynamic tracking over the net yet.
//
// !!! Note that once we support dynamic tracking, we MUST CHANGE
// THE NAMED PIPE PROCESSING to not do writes/transceives at DPC
// level, because the NPFS needs to call SeCreateClientSecurity
// on every write when dynamic tracking is selected!
//
qualityOfService.Length = sizeof( qualityOfService );
qualityOfService.ImpersonationLevel =
SmbGetUlong( &request->ImpersonationLevel );
qualityOfService.ContextTrackingMode = FALSE;
//qualityOfService.ContextTrackingMode = (BOOLEAN)
// (request->SecurityFlags & SMB_SECURITY_DYNAMIC_TRACKING);
qualityOfService.EffectiveOnly = (BOOLEAN)
(request->SecurityFlags & SMB_SECURITY_EFFECTIVE_ONLY);
status = SrvNtCreateFile(
WorkContext,
SmbGetUlong( &request->RootDirectoryFid ),
SmbGetUlong( &request->DesiredAccess ),
allocationSize,
SmbGetUlong( &request->FileAttributes ),
SmbGetUlong( &request->ShareAccess ),
SmbGetUlong( &request->CreateDisposition ),
SmbGetUlong( &request->CreateOptions),
NULL,
&fileName,
NULL,
0,
NULL,
flags,
&qualityOfService,
#ifdef INCLUDE_SMB_IFMODIFIED
(flags & NT_CREATE_REQUEST_OPBATCH) != 0 ?
OplockTypeBatch :
(flags & NT_CREATE_REQUEST_OPLOCK) != 0 ?
#else
(request->Flags & NT_CREATE_REQUEST_OPBATCH) != 0 ?
OplockTypeBatch :
(request->Flags & NT_CREATE_REQUEST_OPLOCK) != 0 ?
#endif
OplockTypeExclusive : OplockTypeServerBatch,
RestartNtCreateAndX
);
//
// Free the unicode file name buffer if it has been allocated.
//
if ( !isUnicode ) {
RtlFreeUnicodeString( &fileName );
}
if ( status == STATUS_OPLOCK_BREAK_IN_PROGRESS ) {
//
// The open is blocked (waiting for a comm device or an oplock
// break), do not send a reply.
//
SmbStatus = SmbStatusInProgress;
} else if ( WorkContext->Parameters2.Open.TemporaryOpen ) {
//
// The initial open failed due to a sharing violation, possibly
// caused by an batch oplock. Requeue the open to a blocking
// thread.
//
WorkContext->FspRestartRoutine = SrvRestartSmbReceived;
SrvQueueWorkToBlockingThread( WorkContext );
SmbStatus = SmbStatusInProgress;
} else {
//
// The open has completed. Generate and send the reply.
//
SmbStatus = GenerateNtCreateAndXResponse( WorkContext, status );
}
Cleanup:
SrvWmiEndContext(WorkContext);
return SmbStatus;
} // SrvSmbNtCreateAndX
VOID SRVFASTCALL
RestartNtCreateAndX (
PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of an Nt Create and X SMB.
Arguments:
WorkContext - Work context block for the operation.
Return Value:
None.
--*/
{
SMB_PROCESSOR_RETURN_LOCAL smbStatus = SmbStatusInProgress;
NTSTATUS openStatus = STATUS_SUCCESS;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_NT_CREATE_AND_X;
SrvWmiStartContext(WorkContext);
openStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if ( NT_SUCCESS( openStatus ) ) {
openStatus = WorkContext->Irp->IoStatus.Status;
} else {
//
// This open was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the open.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateNtCreateAndXResponse(
WorkContext,
openStatus
);
if ( smbStatus == SmbStatusMoreCommands ) {
SrvProcessSmb( WorkContext );
} else {
#ifdef INCLUDE_SMB_IFMODIFIED
SrvEndSmbProcessing( WorkContext, smbStatus );
#else
SrvEndSmbProcessing( WorkContext, SmbStatusSendResponse );
#endif
}
SrvWmiEndContext(WorkContext);
return;
} // RestartNtCreateAndX
SMB_PROCESSOR_RETURN_TYPE
GenerateNtCreateAndXResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
)
/*++
Routine Description:
Generates a response to an Nt Create and X SMB and setup for furthur
SMB processing.
Arguments:
WorkContext - Work context block for the operation.
OpenStatus - The status of the open operation.
Return Value:
None.
Notes:
--*/
{
PREQ_NT_CREATE_ANDX request;
PRESP_NT_CREATE_ANDX response;
#ifdef INCLUDE_SMB_IFMODIFIED
PREQ_NT_CREATE_IF_MODIFIED_ANDX requestIfModified = NULL;
BOOLEAN GenerateExtendedResponse2 = FALSE;
BOOLEAN extendedRequested2 = FALSE;
BOOLEAN ifModifiedCheck = FALSE;
#ifdef INCLUDE_SMB_PERSISTENT
BOOLEAN persistentHandle = FALSE;
#endif
ULONG flags;
BOOLEAN byteCountSet = FALSE;
LARGE_INTEGER ourFileRefNumber;
LARGE_INTEGER ourUsnValue;
#endif
SMB_PROCESSOR_RETURN_LOCAL rc;
BOOLEAN GenerateExtendedResponse = FALSE;
SRV_NT_FILE_INFORMATION srvNtFileInformation;
PRFCB rfcb;
PIO_STATUS_BLOCK ioStatusBlock;
UCHAR nextCommand;
USHORT reqAndXOffset;
OPLOCK_TYPE oplockType;
UCHAR oplockLevel;
BOOLEAN allowLevelII;
BOOLEAN extendedRequested;
ULONG CreateAction = (ULONG)WorkContext->Irp->IoStatus.Information;
ULONG desiredAccess;
NTSTATUS status;
PAGED_CODE( );
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS( OpenStatus ) ) {
SrvSetSmbError( WorkContext, OpenStatus );
//
// Remap the error if it is ERROR_ALREADY_EXISTS
//
if ( !CLIENT_CAPABLE_OF(NT_STATUS, WorkContext->Connection) &&
SmbGetUshort( &WorkContext->ResponseHeader->Error ) ==
ERROR_ALREADY_EXISTS ) {
SmbPutUshort(
&WorkContext->ResponseHeader->Error,
ERROR_FILE_EXISTS
);
}
return SmbStatusSendResponse;
}
request = (PREQ_NT_CREATE_ANDX)WorkContext->RequestParameters;
response = (PRESP_NT_CREATE_ANDX)WorkContext->ResponseParameters;
#ifdef INCLUDE_SMB_IFMODIFIED
//
// these are the USN journal values for this file iff we need them.
//
ourFileRefNumber.QuadPart = 0;
ourUsnValue.QuadPart = 0;
flags = SmbGetUlong( &request->Flags );
//
// Does the client want extended info in the response?
//
extendedRequested = ((flags & NT_CREATE_REQUEST_EXTENDED_RESPONSE) != 0 );
if (IS_POSTNT5_DIALECT( WorkContext->Connection->SmbDialect )) {
extendedRequested2 = ((flags & NT_CREATE_REQUEST_EXTENDED_RESPONSE2) != 0 );
#ifdef INCLUDE_SMB_PERSISTENT
persistentHandle = ((flags & NT_CREATE_PERSISTANT_HANDLE) != 0 );
#endif
ifModifiedCheck = ((flags & NT_CREATE_IFMODIFIED_REQUEST) != 0 );
if (ifModifiedCheck) {
requestIfModified = (PREQ_NT_CREATE_IF_MODIFIED_ANDX) request;
}
}
#else
//
// Does the client want extended info in the response?
//
extendedRequested = ((request->Flags & NT_CREATE_REQUEST_EXTENDED_RESPONSE) != 0 );
#endif
desiredAccess = SmbGetUlong( &request->DesiredAccess );
rfcb = WorkContext->Rfcb;
//
// Attempt to acquire the oplock.
//
if ( desiredAccess != DELETE &&
!(request->CreateOptions & FILE_DIRECTORY_FILE) ) {
if ( request->Flags & NT_CREATE_REQUEST_OPLOCK ) {
allowLevelII = CLIENT_CAPABLE_OF( LEVEL_II_OPLOCKS, WorkContext->Connection );
if ( request->Flags & NT_CREATE_REQUEST_OPBATCH ) {
oplockType = OplockTypeBatch;
oplockLevel = SMB_OPLOCK_LEVEL_BATCH;
} else {
oplockType = OplockTypeExclusive;
oplockLevel = SMB_OPLOCK_LEVEL_EXCLUSIVE;
}
} else {
allowLevelII = FALSE;
oplockType = OplockTypeServerBatch;
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
}
if( SrvRequestOplock( WorkContext, &oplockType, allowLevelII ) ) {
//
// The oplock was granted. Check to see if it was a level 2.
//
if ( oplockType == OplockTypeShareRead ) {
oplockLevel = SMB_OPLOCK_LEVEL_II;
}
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksGranted );
} else {
//
// The oplock request was denied.
//
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksDenied );
}
} else {
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
}
//
// If the file was created, set the EOF location to be the same as
// the size of the file. This is necessary for compatibility with
// OS/2, which only has EOF, not a separate allocation size.
//
ioStatusBlock = &WorkContext->Irp->IoStatus;
//
// We always open with at least FILE_READ_ATTRIBUTES, so no
// access check is needed.
//
status = SrvQueryNtInformationFile(
rfcb->Lfcb->FileHandle,
rfcb->Lfcb->FileObject,
rfcb->ShareType,
extendedRequested,
&srvNtFileInformation
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: SrvQueryNtInformationFile failed: "
"%X\n", status ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError( WorkContext, status );
return SmbStatusSendResponse;
}
//
// Save parts of the file info in the MFCB for fast tests on Compressed
// operations.
//
rfcb->Mfcb->NonpagedMfcb->OpenFileSize.QuadPart =
srvNtFileInformation.NwOpenInfo.EndOfFile.QuadPart;
rfcb->Mfcb->NonpagedMfcb->OpenFileAttributes =
srvNtFileInformation.NwOpenInfo.FileAttributes;
//
// if the client is doing an "IfModifiedSince" open, then we need to check
// the following fields out of the request:
//
// CreationTime
// LastWriteTime
// ChangeTime
// EndOfFile
// FileAttributes
// UsnValue / USN file reference number
//
// if they all match with what is currently set for the file, we close
// the file and return STATUS_FILE_CLOSED.
//
#ifdef INCLUDE_SMB_IFMODIFIED
if (ifModifiedCheck) {
LARGE_INTEGER usnValue;
LARGE_INTEGER fileRefNumber;
LARGE_INTEGER endOfFile;
TIME creationTime;
TIME lastWriteTime;
TIME changeTime;
ULONG fileAttributes;
ULONG ourAttributes;
endOfFile.HighPart = SmbGetUlong( &requestIfModified->EndOfFile.HighPart );
endOfFile.LowPart = SmbGetUlong( &requestIfModified->EndOfFile.LowPart );
creationTime.HighPart = SmbGetUlong( &requestIfModified->CreationTime.HighPart );
creationTime.LowPart = SmbGetUlong( &requestIfModified->CreationTime.LowPart );
lastWriteTime.HighPart = SmbGetUlong( &requestIfModified->LastWriteTime.HighPart );
lastWriteTime.LowPart = SmbGetUlong( &requestIfModified->LastWriteTime.LowPart );
changeTime.HighPart = SmbGetUlong( &requestIfModified->ChangeTime.HighPart );
changeTime.LowPart = SmbGetUlong( &requestIfModified->ChangeTime.LowPart );
usnValue.HighPart = SmbGetUlong( &requestIfModified->UsnValue.HighPart );
usnValue.LowPart = SmbGetUlong( &requestIfModified->UsnValue.LowPart );
fileRefNumber.HighPart = SmbGetUlong( &requestIfModified->FileReferenceNumber.HighPart );
fileRefNumber.LowPart = SmbGetUlong( &requestIfModified->FileReferenceNumber.LowPart );
//
// we ignore the NORMAL and ARCHIVE attributes because the client
// doesn't care if the these attributes have changed.
//
fileAttributes = SmbGetUlong( &requestIfModified->FileAttributes );
fileAttributes &= ~FILE_ATTRIBUTE_NORMAL;
fileAttributes &= ~FILE_ATTRIBUTE_ARCHIVE;
ourAttributes = srvNtFileInformation.NwOpenInfo.FileAttributes;
ourAttributes &= ~FILE_ATTRIBUTE_NORMAL;
ourAttributes &= ~FILE_ATTRIBUTE_ARCHIVE;
if ( usnValue.QuadPart != 0 || fileRefNumber.QuadPart != 0 ) {
status = SrvIssueQueryUsnInfoRequest( rfcb,
FALSE,
&ourUsnValue,
&ourFileRefNumber );
if (! NT_SUCCESS(status)) {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: Query USN info (1) failed: 0x%X for handle %u\n",
status, rfcb->Lfcb->FileObject ));
}
}
}
if (creationTime.QuadPart == srvNtFileInformation.NwOpenInfo.CreationTime.QuadPart &&
lastWriteTime.QuadPart == srvNtFileInformation.NwOpenInfo.LastWriteTime.QuadPart &&
changeTime.QuadPart == srvNtFileInformation.NwOpenInfo.ChangeTime.QuadPart &&
endOfFile.QuadPart == srvNtFileInformation.NwOpenInfo.EndOfFile.QuadPart &&
usnValue.QuadPart == ourUsnValue.QuadPart &&
fileRefNumber.QuadPart == ourFileRefNumber.QuadPart &&
fileAttributes == ourAttributes ) {
//
// the client has the current version of this file, so he wants
// the open to fail.
//
// KdPrint(( "SmbNtCreateAndX IfModified worked. Closing file.\n" ));
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: File doesn't pass IfModifiedSince clause.\n" ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError( WorkContext, STATUS_FILE_CLOSED );
return SmbStatusSendResponse;
}
// KdPrint(( "SmbNtCreateAndX IfModified ignored, opened the file.\n" ));
}
#endif
// If the extended response is required then
if ( extendedRequested ) {
NTSTATUS ExtendedResponseStatus;
PRESP_EXTENDED_NT_CREATE_ANDX pExtendedResponse;
pExtendedResponse = (PRESP_EXTENDED_NT_CREATE_ANDX)response;
ExtendedResponseStatus = SrvUpdateMaximalAccessRightsInResponse(
WorkContext,
&pExtendedResponse->MaximalAccessRights,
&pExtendedResponse->GuestMaximalAccessRights);
GenerateExtendedResponse = (ExtendedResponseStatus == STATUS_SUCCESS);
}
//
// Set up response SMB.
//
nextCommand = request->AndXCommand;
reqAndXOffset = SmbGetUshort( &request->AndXOffset );
response->AndXCommand = nextCommand;
response->AndXReserved = 0;
if (GenerateExtendedResponse) {
#ifdef INCLUDE_SMB_IFMODIFIED
PRESP_EXTENDED_NT_CREATE_ANDX pExtendedResponse;
pExtendedResponse = (PRESP_EXTENDED_NT_CREATE_ANDX)response;
#endif
SmbPutUshort(
&response->AndXOffset,
GET_ANDX_OFFSET(
WorkContext->ResponseHeader,
WorkContext->ResponseParameters,
RESP_EXTENDED_NT_CREATE_ANDX,
0
)
);
response->WordCount = 42;
#ifdef INCLUDE_SMB_IFMODIFIED
SmbPutUshort( &pExtendedResponse->ByteCount, 0 );
byteCountSet = TRUE;
if (extendedRequested2 && (nextCommand == SMB_COM_NO_ANDX_COMMAND)) {
IO_STATUS_BLOCK ioStatus;
PFILE_NAME_INFORMATION fileNameInfo;
ULONG infoLength;
PRESP_EXTENDED_NT_CREATE_ANDX2 extendedResponse2;
ULONG bufLength;
extendedResponse2 = (PRESP_EXTENDED_NT_CREATE_ANDX2) request;
if (ourUsnValue.QuadPart == 0 && ourFileRefNumber.QuadPart == 0) {
status = SrvIssueQueryUsnInfoRequest( rfcb,
FALSE,
&ourUsnValue,
&ourFileRefNumber );
if (! NT_SUCCESS(status)) {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: Query USN info (2) failed: 0x%X for handle %u\n",
status, rfcb->Lfcb->FileObject ));
}
}
}
//
// Put in the short and long file name values here for CSC. It's
// a RESP_EXTENDED_NT_CREATE_ANDX2 response. These requests
// cannot currently be compounded because the file names will
// overwrite the next command.
//
// get the short name to put in the buffer
fileNameInfo = (PFILE_NAME_INFORMATION) ALIGN_UP( &extendedResponse2->Buffer[0], ULONG );
//
// the buffer length we can use is all that remains in the
// receive buffer.
//
ASSERT( WorkContext->RequestBuffer == WorkContext->ResponseBuffer );
bufLength = min( WorkContext->RequestBuffer->BufferLength,
WorkContext->Connection->MaximumSendSize );
infoLength = (ULONG) ( bufLength -
( (PCHAR) fileNameInfo -
(PCHAR) (WorkContext->RequestBuffer->Buffer) ));
infoLength -= sizeof(ULONG); // for alignment issues
status = NtQueryInformationFile(
rfcb->Lfcb->FileHandle,
&ioStatus,
(PVOID)fileNameInfo,
infoLength,
FileAlternateNameInformation
);
if (NT_SUCCESS(status)) {
if (fileNameInfo->FileNameLength <
(SMB_SHORT_NAME_LENGTH * sizeof(WCHAR))) {
ULONG fileNameLength = fileNameInfo->FileNameLength;
RtlZeroMemory( &extendedResponse2->ShortName[0],
sizeof(WCHAR) * SMB_SHORT_NAME_LENGTH );
RtlCopyMemory( &extendedResponse2->ShortName[0],
fileNameInfo->FileName,
fileNameLength
);
// null terminate the name
extendedResponse2->ShortName[fileNameLength/sizeof(WCHAR)] = L'\0';
} else {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: QueryInfoFile for short name returned length of %X\n", fileNameInfo->FileNameLength ));
}
extendedResponse2->ShortName[0] = L'\0';
}
} else {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: QueryInfoFile for short name failed: "
"%X\n", status ));
}
extendedResponse2->ShortName[0] = L'\0';
}
status = NtQueryInformationFile(
rfcb->Lfcb->FileHandle,
&ioStatus,
(PVOID)fileNameInfo,
infoLength,
FileNameInformation
);
if (NT_SUCCESS(status)) {
ULONG fileNameLength = 0;
//
// Use RtlMoveMemory as the source and destination will
// possibly overlap.
//
if (fileNameInfo->FileNameLength > 0) {
// only copy over the last component of the name.
PWCHAR lastBackSlash = &fileNameInfo->FileName[((fileNameInfo->FileNameLength)/sizeof(WCHAR))-1];
while (*lastBackSlash != UNICODE_DIR_SEPARATOR_CHAR &&
lastBackSlash > &fileNameInfo->FileName[0]) {
fileNameLength += sizeof(WCHAR);
lastBackSlash--;
}
lastBackSlash++;
RtlMoveMemory( &extendedResponse2->Buffer[0],
lastBackSlash,
fileNameLength
);
}
// null terminate the name
extendedResponse2->Buffer[fileNameLength/sizeof(WCHAR)] = L'\0';
SmbPutUshort( &extendedResponse2->ByteCount,
LOWORD(fileNameLength) + sizeof(WCHAR) );
byteCountSet = TRUE;
SmbPutUlong( &extendedResponse2->UsnValue.HighPart, ourUsnValue.HighPart );
SmbPutUlong( &extendedResponse2->UsnValue.LowPart, ourUsnValue.LowPart );
SmbPutUlong( &extendedResponse2->FileReferenceNumber.HighPart, ourFileRefNumber.HighPart );
SmbPutUlong( &extendedResponse2->FileReferenceNumber.LowPart, ourFileRefNumber.LowPart );
SmbPutUshort(
&response->AndXOffset,
GET_ANDX_OFFSET(
WorkContext->ResponseHeader,
WorkContext->ResponseParameters,
RESP_EXTENDED_NT_CREATE_ANDX2,
LOWORD(fileNameLength) + sizeof(WCHAR)
)
);
response->WordCount = SMB_RESP_EXTENDED_NT_CREATE_ANDX2_WORK_COUNT;
} else {
//
// if we couldn't get the name for some reason, let's not fail
// the whole open. Just leave the response as a
// RESP_EXTENDED_NT_CREATE_ANDX
//
IF_DEBUG(ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: QueryInfoFile for long name failed: "
"%X\n", status ));
}
status = STATUS_SUCCESS;
//
// reset the byte count in case the CrcValue was set
// and overwrote it.
//
SmbPutUshort( &pExtendedResponse->ByteCount, 0 );
byteCountSet = TRUE;
}
}
#endif
} else {
SmbPutUshort(
&response->AndXOffset,
GET_ANDX_OFFSET(
WorkContext->ResponseHeader,
WorkContext->ResponseParameters,
RESP_NT_CREATE_ANDX,
0
)
);
response->WordCount = 34;
}
response->OplockLevel = oplockLevel;
SmbPutUshort( &response->Fid, rfcb->Fid );
SmbPutUlong( &response->CreateAction, CreateAction );
SmbPutUlong(
&response->CreationTime.HighPart,
srvNtFileInformation.NwOpenInfo.CreationTime.HighPart
);
SmbPutUlong(
&response->CreationTime.LowPart,
srvNtFileInformation.NwOpenInfo.CreationTime.LowPart
);
SmbPutUlong(
&response->LastAccessTime.HighPart,
srvNtFileInformation.NwOpenInfo.LastAccessTime.HighPart
);
SmbPutUlong(
&response->LastAccessTime.LowPart,
srvNtFileInformation.NwOpenInfo.LastAccessTime.LowPart
);
SmbPutUlong(
&response->LastWriteTime.HighPart,
srvNtFileInformation.NwOpenInfo.LastWriteTime.HighPart
);
SmbPutUlong(
&response->LastWriteTime.LowPart,
srvNtFileInformation.NwOpenInfo.LastWriteTime.LowPart
);
SmbPutUlong(
&response->ChangeTime.HighPart,
srvNtFileInformation.NwOpenInfo.ChangeTime.HighPart
);
SmbPutUlong(
&response->ChangeTime.LowPart,
srvNtFileInformation.NwOpenInfo.ChangeTime.LowPart
);
SmbPutUlong( &response->FileAttributes, srvNtFileInformation.NwOpenInfo.FileAttributes );
SmbPutUlong(
&response->AllocationSize.HighPart,
srvNtFileInformation.NwOpenInfo.AllocationSize.HighPart
);
SmbPutUlong(
&response->AllocationSize.LowPart,
srvNtFileInformation.NwOpenInfo.AllocationSize.LowPart
);
SmbPutUlong(
&response->EndOfFile.HighPart,
srvNtFileInformation.NwOpenInfo.EndOfFile.HighPart
);
SmbPutUlong(
&response->EndOfFile.LowPart,
srvNtFileInformation.NwOpenInfo.EndOfFile.LowPart
);
SmbPutUshort( &response->FileType, srvNtFileInformation.Type );
SmbPutUshort( &response->DeviceState, srvNtFileInformation.HandleState );
response->Directory = (srvNtFileInformation.NwOpenInfo.FileAttributes & FILE_ATTRIBUTE_DIRECTORY) ? TRUE : FALSE;
#ifdef INCLUDE_SMB_IFMODIFIED
if (byteCountSet == FALSE) {
//
// if we haven't already put the bytecount into the extended response,
// do so now.
//
#endif
SmbPutUshort( &response->ByteCount, 0 );
#ifdef INCLUDE_SMB_IFMODIFIED
}
#endif
WorkContext->ResponseParameters = (PCHAR)WorkContext->ResponseHeader +
SmbGetUshort( &response->AndXOffset );
//
// Test for legal followon command.
//
switch ( nextCommand ) {
case SMB_COM_NO_ANDX_COMMAND:
break;
case SMB_COM_READ:
case SMB_COM_READ_ANDX:
case SMB_COM_IOCTL:
//
// Make sure the AndX command is still within the received SMB
//
if( (PCHAR)WorkContext->RequestHeader + reqAndXOffset <=
END_OF_REQUEST_SMB( WorkContext ) ) {
break;
}
/* Falls Through */
default: // Illegal followon command
IF_DEBUG(SMB_ERRORS) {
KdPrint(( "SrvSmbNtCreateAndX: Illegal followon command: 0x%lx\n",
nextCommand ));
}
//
// Return an error indicating that the followon command was bad.
// Note that the open is still considered successful, so the
// file remains open.
//
SrvSetSmbError( WorkContext, STATUS_INVALID_SMB );
return SmbStatusSendResponse;
}
#ifdef INCLUDE_SMB_PERSISTENT
if (persistentHandle && rfcb->Lfcb->TreeConnect->Share->AllowPersistentHandles) {
rfcb->PersistentHandle = TRUE;
} else {
persistentHandle = rfcb->PersistentHandle = FALSE;
}
#endif
//
// If there is an AndX command, set up to process it. Otherwise,
// indicate completion to the caller.
//
if ( nextCommand != SMB_COM_NO_ANDX_COMMAND ) {
WorkContext->NextCommand = nextCommand;
WorkContext->RequestParameters = (PCHAR)WorkContext->RequestHeader +
reqAndXOffset;
#ifdef INCLUDE_SMB_PERSISTENT
if (persistentHandle) {
rc = SrvPostPersistentOpen( WorkContext, SmbStatusMoreCommands );
} else {
rc = SmbStatusMoreCommands;
}
return rc;
#else
return SmbStatusMoreCommands;
#endif
}
#ifdef INCLUDE_SMB_PERSISTENT
if (persistentHandle) {
rc = SrvPostPersistentOpen( WorkContext, SmbStatusSendResponse );
} else {
rc = SmbStatusSendResponse;
}
#else
rc = SmbStatusSendResponse;
#endif
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbNtCreateAndX complete.\n" ));
return rc;
} // GenerateNtCreateAndXResponse
SMB_TRANS_STATUS
SrvSmbCreateWithSdOrEa (
IN OUT PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Processes an Create with SD or EA SMB. This request arrives in an
Nt Transaction SMB.
Arguments:
WorkContext - Supplies the address of a Work Context Block
describing the current request. See smbtypes.h for a more
complete description of the valid fields.
Return Value:
BOOLEAN - Indicates whether an error occurred. See smbtypes.h for a
more complete description.
--*/
{
PREQ_CREATE_WITH_SD_OR_EA request;
PRESP_CREATE_WITH_SD_OR_EA response;
NTSTATUS status;
PTRANSACTION transaction;
ULONG eaErrorOffset = 0;
LARGE_INTEGER allocationSize;
PVOID securityDescriptorBuffer;
ULONG sdLength;
ULONG eaLength;
PVOID eaBuffer;
UNICODE_STRING fileName;
PUCHAR name;
USHORT nameLength;
BOOLEAN isUnicode;
SECURITY_QUALITY_OF_SERVICE qualityOfService;
PAGED_CODE( );
transaction = WorkContext->Parameters.Transaction;
request = (PREQ_CREATE_WITH_SD_OR_EA)transaction->InParameters;
response = (PRESP_CREATE_WITH_SD_OR_EA)transaction->OutParameters;
//
// Verify that enough parameter bytes were sent and that we're allowed
// to return enough parameter bytes.
//
if ( (transaction->ParameterCount < sizeof(REQ_CREATE_WITH_SD_OR_EA)) ||
(transaction->MaxParameterCount < sizeof(RESP_CREATE_WITH_SD_OR_EA)) ) {
//
// Not enough parameter bytes were sent.
//
IF_DEBUG(SMB_ERRORS)
KdPrint(( "SrvSmbCreateWithSdOrEa: bad parameter byte counts: %ld %ld\n",
transaction->ParameterCount,
transaction->MaxParameterCount ));
SrvLogInvalidSmb( WorkContext );
return GenerateCreateWithSdOrEaResponse(
WorkContext,
STATUS_INVALID_SMB
);
}
eaErrorOffset = 0;
allocationSize.HighPart = SmbGetUlong( &request->AllocationSize.HighPart );
allocationSize.LowPart = SmbGetUlong( &request->AllocationSize.LowPart );
//
// First verify that the file path name doesnot extend beyond the
// end of the SMB.
//
isUnicode = SMB_IS_UNICODE( WorkContext );
name = (PUCHAR)request->Buffer;
if ( isUnicode ) {
name = ALIGN_SMB_WSTR( name );
}
nameLength = (USHORT)SmbGetUshort( &request->NameLength );
if ( name + nameLength > ((PCHAR)request + transaction->ParameterCount) ) {
IF_DEBUG(SMB_ERRORS) {
KdPrint(( "SrvSmbCreateWithSdOrEa: failed at line %u\n", __LINE__ ));
}
SrvLogInvalidSmb( WorkContext );
return GenerateCreateWithSdOrEaResponse(
WorkContext,
STATUS_INVALID_SMB
);
}
//
// Convert the file name to a Unicode string.
//
status = SrvMakeUnicodeString(
isUnicode,
&fileName,
name,
&nameLength
);
if ( !NT_SUCCESS( status ) ) {
return GenerateCreateWithSdOrEaResponse( WorkContext, status );
}
sdLength = SmbGetUlong( &request->SecurityDescriptorLength );
eaLength = SmbGetUlong( &request->EaLength );
securityDescriptorBuffer = transaction->InData;
eaBuffer = (PCHAR)securityDescriptorBuffer + ((sdLength + 3) & ~ 3);
if( eaLength > transaction->DataCount ||
sdLength > transaction->DataCount ||
(PCHAR)securityDescriptorBuffer + eaLength + sdLength >
(PCHAR)transaction->InData + transaction->DataCount ||
((sdLength != 0) && !RtlValidSecurityDescriptor(securityDescriptorBuffer))) {
IF_DEBUG( SMB_ERRORS ) {
KdPrint(( "SrvSmbCreateWithSdOrEa: failed at line %u\n", __LINE__ ));
KdPrint(( " eaLength %u, sdLength %u, DataCount %u\n",
eaLength, sdLength, transaction->DataCount ));
}
SrvLogInvalidSmb( WorkContext );
if ( !isUnicode ) {
RtlFreeUnicodeString( &fileName );
}
return GenerateCreateWithSdOrEaResponse(
WorkContext,
STATUS_INVALID_SECURITY_DESCR
);
}
//
// *** We always ask for STATIC tracking, not DYNAMIC, because we
// don't support dynamic tracking over the net yet.
//
// !!! Note that once we support dynamic tracking, we MUST CHANGE
// THE NAMED PIPE PROCESSING to not do writes/transceives at DPC
// level, because the NPFS needs to call SeCreateClientSecurity
// on every write when dynamic tracking is selected!
//
qualityOfService.Length = sizeof( qualityOfService );
qualityOfService.ImpersonationLevel =
SmbGetUlong( &request->ImpersonationLevel );
qualityOfService.ContextTrackingMode = FALSE;
//qualityOfService.ContextTrackingMode = (BOOLEAN)
// (request->SecurityFlags & SMB_SECURITY_DYNAMIC_TRACKING);
qualityOfService.EffectiveOnly = (BOOLEAN)
(request->SecurityFlags & SMB_SECURITY_EFFECTIVE_ONLY);
status = SrvNtCreateFile(
WorkContext,
SmbGetUlong( &request->RootDirectoryFid ),
SmbGetUlong( &request->DesiredAccess ),
allocationSize,
SmbGetUlong( &request->FileAttributes ),
SmbGetUlong( &request->ShareAccess ),
SmbGetUlong( &request->CreateDisposition ),
SmbGetUlong( &request->CreateOptions ),
(sdLength == 0) ? NULL : securityDescriptorBuffer,
&fileName,
(eaLength == 0) ? NULL : eaBuffer,
eaLength,
&eaErrorOffset,
SmbGetUlong( &request->Flags ),
&qualityOfService,
(request->Flags & NT_CREATE_REQUEST_OPBATCH) != 0 ?
OplockTypeBatch :
(request->Flags & NT_CREATE_REQUEST_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
RestartCreateWithSdOrEa
);
//
// Free the unicode file name buffer if it has been allocated.
//
if ( !isUnicode ) {
RtlFreeUnicodeString( &fileName );
}
if ( status == STATUS_OPLOCK_BREAK_IN_PROGRESS ) {
//
// The open is blocked (waiting for a comm device or an oplock
// break), do not send a reply.
//
//
// Save the ea error offset
//
WorkContext->Parameters2.Open.EaErrorOffset = eaErrorOffset;
return SmbTransStatusInProgress;
} else if ( WorkContext->Parameters2.Open.TemporaryOpen ) {
//
// The initial open failed due to a sharing violation, possibly
// caused by an batch oplock. Requeue the open to a blocking
// thread.
//
WorkContext->FspRestartRoutine = (PRESTART_ROUTINE)ExecuteTransaction;
SrvQueueWorkToBlockingThread( WorkContext );
return SmbStatusInProgress;
} else {
//
// Save the ea error offset
//
WorkContext->Parameters2.Open.EaErrorOffset = eaErrorOffset;
//
// The open has completed. Generate and send the reply.
//
return GenerateCreateWithSdOrEaResponse( WorkContext, status );
}
} // SrvSmbCreateWithSdOrEa
VOID SRVFASTCALL
RestartCreateWithSdOrEa (
PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of an Open2 SMB.
Arguments:
WorkContext - Work context block for the operation.
Return Value:
None.
--*/
{
SMB_TRANS_STATUS smbStatus;
NTSTATUS openStatus;
PAGED_CODE( );
openStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if ( NT_SUCCESS( openStatus ) ) {
openStatus = WorkContext->Irp->IoStatus.Status;
if( NT_SUCCESS( openStatus ) ) {
//
// It's obvious that the file already existed, because we've
// been working on an oplock break. So set the
// IoStatus.Information field correctly.
//
WorkContext->Irp->IoStatus.Information = FILE_OPENED;
}
} else {
//
// This open was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the open.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateCreateWithSdOrEaResponse( WorkContext, openStatus );
SrvCompleteExecuteTransaction( WorkContext, smbStatus );
return;
} // RestartCreateWithSdOrEa
SMB_TRANS_STATUS
GenerateCreateWithSdOrEaResponse (
PWORK_CONTEXT WorkContext,
NTSTATUS OpenStatus
)
/*++
Routine Description:
Generates a response to an Create With SD or EA SMB and setup for furthur
SMB processing.
Arguments:
WorkContext - Work context block for the operation.
OpenStatus - The status of the open operation.
Return Value:
None.
--*/
{
PREQ_CREATE_WITH_SD_OR_EA request;
PRESP_CREATE_WITH_SD_OR_EA response;
PRFCB rfcb;
NTSTATUS status;
SRV_NT_FILE_INFORMATION srvNtFileInformation;
PTRANSACTION transaction;
OPLOCK_TYPE oplockType;
UCHAR oplockLevel;
BOOLEAN allowLevelII;
BOOLEAN SendExtendedResponse = FALSE;
ACCESS_MASK MaximalAccessRights;
ACCESS_MASK GuestMaximalAccessRights;
ULONG eaErrorOffset;
BOOLEAN extendedRequested;
ULONG CreateAction = (ULONG)WorkContext->Irp->IoStatus.Information;
PAGED_CODE( );
transaction = WorkContext->Parameters.Transaction;
request = (PREQ_CREATE_WITH_SD_OR_EA)transaction->InParameters;
response = (PRESP_CREATE_WITH_SD_OR_EA)transaction->OutParameters;
extendedRequested = ((request->Flags & NT_CREATE_REQUEST_EXTENDED_RESPONSE) != 0 );
rfcb = WorkContext->Rfcb;
eaErrorOffset = WorkContext->Parameters2.Open.EaErrorOffset;
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS( OpenStatus ) ) {
SrvSetSmbError2( WorkContext, OpenStatus, TRUE );
//
// Remap the error if it is ERROR_ALREADY_EXISTS
//
if ( !CLIENT_CAPABLE_OF(NT_STATUS,WorkContext->Connection) &&
SmbGetUshort( &WorkContext->ResponseHeader->Error ) ==
ERROR_ALREADY_EXISTS ) {
SmbPutUshort(
&WorkContext->ResponseHeader->Error,
ERROR_FILE_EXISTS
);
}
goto err_exit;
}
//
// We always open with at least FILE_READ_ATTRIBUTES, so no
// access check is needed.
//
status = SrvQueryNtInformationFile(
rfcb->Lfcb->FileHandle,
rfcb->Lfcb->FileObject,
rfcb->ShareType,
extendedRequested,
&srvNtFileInformation
);
if ( !NT_SUCCESS(status) ) {
IF_DEBUG(ERRORS) {
KdPrint(( "GenerateCreateWithSdOrEaResponse: "
"SrvQueryNtInformationFile failed: %X\n", status ));
}
SrvCloseRfcb( rfcb );
SrvSetSmbError2( WorkContext, status, TRUE );
goto err_exit;
}
//
// Attempt to acquire the oplock.
//
if ( !(request->CreateOptions & FILE_DIRECTORY_FILE) ) {
if ( request->Flags & NT_CREATE_REQUEST_OPLOCK ) {
allowLevelII = CLIENT_CAPABLE_OF( LEVEL_II_OPLOCKS, WorkContext->Connection );
if ( request->Flags & NT_CREATE_REQUEST_OPBATCH ) {
oplockType = OplockTypeBatch;
oplockLevel = SMB_OPLOCK_LEVEL_BATCH;
} else {
oplockType = OplockTypeExclusive;
oplockLevel = SMB_OPLOCK_LEVEL_EXCLUSIVE;
}
} else {
allowLevelII = FALSE;
oplockType = OplockTypeServerBatch;
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
}
if ( SrvRequestOplock( WorkContext, &oplockType, allowLevelII ) ) {
//
// The oplock was granted. Check to see if it was a level 2.
//
if ( oplockType == OplockTypeShareRead ) {
oplockLevel = SMB_OPLOCK_LEVEL_II;
}
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksGranted );
} else {
//
// The oplock request was denied.
//
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
INCREMENT_DEBUG_STAT( SrvDbgStatistics.TotalOplocksDenied );
}
} else {
oplockLevel = SMB_OPLOCK_LEVEL_NONE;
}
if ( extendedRequested ) {
NTSTATUS ExtendedResponseStatus;
PRESP_EXTENDED_CREATE_WITH_SD_OR_EA ExtendedResponse;
ExtendedResponse = (PRESP_EXTENDED_CREATE_WITH_SD_OR_EA)response;
ExtendedResponseStatus = SrvUpdateMaximalAccessRightsInResponse(
WorkContext,
&ExtendedResponse->MaximalAccessRights,
&ExtendedResponse->GuestMaximalAccessRights);
SendExtendedResponse = (ExtendedResponseStatus == STATUS_SUCCESS);
}
//
// Set up response SMB.
//
response->OplockLevel = oplockLevel;
if (SendExtendedResponse) {
response->ExtendedResponse = 1;
} else {
response->ExtendedResponse = 0;
}
SmbPutUshort( &response->Fid, rfcb->Fid );
SmbPutUlong( &response->EaErrorOffset, eaErrorOffset );
SmbPutUlong( &response->CreateAction, CreateAction );
SmbPutUshort( &response->FileType, srvNtFileInformation.Type );
SmbPutUshort( &response->DeviceState, srvNtFileInformation.HandleState );
SmbPutUlong(
&response->CreationTime.HighPart,
srvNtFileInformation.NwOpenInfo.CreationTime.HighPart
);
SmbPutUlong(
&response->CreationTime.LowPart,
srvNtFileInformation.NwOpenInfo.CreationTime.LowPart
);
SmbPutUlong(
&response->LastAccessTime.HighPart,
srvNtFileInformation.NwOpenInfo.LastAccessTime.HighPart
);
SmbPutUlong(
&response->LastAccessTime.LowPart,
srvNtFileInformation.NwOpenInfo.LastAccessTime.LowPart
);
SmbPutUlong(
&response->LastWriteTime.HighPart,
srvNtFileInformation.NwOpenInfo.LastWriteTime.HighPart
);
SmbPutUlong(
&response->LastWriteTime.LowPart,
srvNtFileInformation.NwOpenInfo.LastWriteTime.LowPart
);
SmbPutUlong(
&response->ChangeTime.HighPart,
srvNtFileInformation.NwOpenInfo.ChangeTime.HighPart
);
SmbPutUlong(
&response->ChangeTime.LowPart,
srvNtFileInformation.NwOpenInfo.ChangeTime.LowPart
);
SmbPutUlong( &response->FileAttributes, srvNtFileInformation.NwOpenInfo.FileAttributes );
SmbPutUlong(
&response->AllocationSize.HighPart,
srvNtFileInformation.NwOpenInfo.AllocationSize.HighPart
);
SmbPutUlong(
&response->AllocationSize.LowPart,
srvNtFileInformation.NwOpenInfo.AllocationSize.LowPart
);
SmbPutUlong(
&response->EndOfFile.HighPart,
srvNtFileInformation.NwOpenInfo.EndOfFile.HighPart
);
SmbPutUlong(
&response->EndOfFile.LowPart,
srvNtFileInformation.NwOpenInfo.EndOfFile.LowPart
);
response->Directory = (srvNtFileInformation.NwOpenInfo.FileAttributes & FILE_ATTRIBUTE_DIRECTORY) ? TRUE : FALSE;
if (SendExtendedResponse) {
transaction->ParameterCount = sizeof(RESP_EXTENDED_CREATE_WITH_SD_OR_EA);
} else {
transaction->ParameterCount = sizeof(RESP_CREATE_WITH_SD_OR_EA);
}
transaction->SetupCount = 0;
transaction->DataCount = 0;
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbOpen2 complete.\n" ));
return SmbTransStatusSuccess;
err_exit:
RtlZeroMemory( (PVOID)response, sizeof(RESP_CREATE_WITH_SD_OR_EA) );
SmbPutUlong( &response->EaErrorOffset, eaErrorOffset );
transaction->SetupCount = 0;
transaction->ParameterCount = sizeof(RESP_CREATE_WITH_SD_OR_EA);
transaction->DataCount = 0;
return SmbTransStatusErrorWithData;
} // GenerateCreateWithSdOrEaResponse
SMB_PROCESSOR_RETURN_TYPE
SrvSmbCreate (
SMB_PROCESSOR_PARAMETERS
)
/*++
Routine Description:
Processes the Create and Create New SMBs.
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbprocs.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbprocs.h
--*/
{
PREQ_CREATE request;
UCHAR command;
NTSTATUS status = STATUS_SUCCESS;
SMB_STATUS SmbStatus = SmbStatusInProgress;
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_CREATE;
SrvWmiStartContext(WorkContext);
IF_SMB_DEBUG(OPEN_CLOSE1) {
KdPrint(( "Create file request header at 0x%p, response header at 0x%p\n",
WorkContext->RequestHeader,
WorkContext->ResponseHeader ));
KdPrint(( "Create file request parameters at 0x%p, response parameters at 0x%p\n",
WorkContext->RequestParameters,
WorkContext->ResponseParameters ));
}
request = (PREQ_CREATE)WorkContext->RequestParameters;
command = WorkContext->RequestHeader->Command;
//
// Open the file in compatibility mode, obtaining read/write access
// for this FID.
//
status = SrvCreateFile(
WorkContext,
SMB_DA_SHARE_COMPATIBILITY | SMB_DA_ACCESS_READ_WRITE,
SmbGetUshort( &request->FileAttributes ),
(USHORT) ( ( command == SMB_COM_CREATE ?
SMB_OFUN_OPEN_TRUNCATE : SMB_OFUN_OPEN_FAIL )
| SMB_OFUN_CREATE_CREATE ),
0, // SmbAllocationSize
(PCHAR)(request->Buffer + 1),
END_OF_REQUEST_SMB( WorkContext ),
NULL,
0L,
NULL,
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK_NOTIFY_ANY) != 0 ?
OplockTypeBatch :
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
RestartSmbCreate
);
if( status == STATUS_OPLOCK_BREAK_IN_PROGRESS ) {
//
// The open is blocked, perhaps because some other client had this
// file open and is holding onto the handle with a batch oplock.
//
SmbStatus = SmbStatusInProgress;
} else {
SmbStatus = GenerateCreateResponse( WorkContext, status );
}
SrvWmiEndContext(WorkContext);
return SmbStatus;
}
VOID SRVFASTCALL
RestartSmbCreate(
IN PWORK_CONTEXT WorkContext
)
/*++
Routine Description:
Completes processing of a Create SMB.
Arguments:
WorkContext - A pointer to the work context block for this SMB.
Return Value:
None.
--*/
{
SMB_PROCESSOR_RETURN_LOCAL smbStatus = SmbStatusInProgress;
NTSTATUS createStatus = STATUS_SUCCESS;
PAGED_CODE();
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_CREATE;
SrvWmiStartContext(WorkContext);
createStatus = SrvCheckOplockWaitState( WorkContext->WaitForOplockBreak );
if( NT_SUCCESS( createStatus ) ) {
createStatus = WorkContext->Irp->IoStatus.Status;
} else {
//
// The create was waiting for an oplock break to occur, but
// timed out. Close our handle to this file, then fail the create.
//
SrvCloseRfcb( WorkContext->Parameters2.Open.Rfcb );
}
WorkContext->Irp->IoStatus.Information = WorkContext->Parameters2.Open.IosbInformation;
smbStatus = GenerateCreateResponse( WorkContext, createStatus );
SrvEndSmbProcessing( WorkContext, smbStatus );
SrvWmiEndContext(WorkContext);
}
SMB_PROCESSOR_RETURN_TYPE
GenerateCreateResponse(
PWORK_CONTEXT WorkContext,
NTSTATUS CreateStatus
)
/*++
Routine Description:
Generates a response to a Create SMB
Arguments:
WorkContext -
Status - The status of the create operation
Return Value:
The status of the SMB processing.
--*/
{
PREQ_CREATE request;
PRESP_CREATE response;
PRFCB rfcb;
PAGED_CODE();
request = (PREQ_CREATE)WorkContext->RequestParameters;
response = (PRESP_CREATE)WorkContext->ResponseParameters;
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS(CreateStatus) ) {
SrvSetSmbError( WorkContext, CreateStatus );
//
// Remap the error if it is ERROR_ALREADY_EXISTS. In OS/2
// ERROR_ALREADY_EXISTS is used for resources like Semaphores.
// This cannot be passed back to the downlevel client and has
// to be remapped to ERROR_FILE_EXISTS
//
if ( !CLIENT_CAPABLE_OF(NT_STATUS,WorkContext->Connection) &&
SmbGetUshort( &WorkContext->ResponseHeader->Error ) ==
ERROR_ALREADY_EXISTS ) {
SmbPutUshort(
&WorkContext->ResponseHeader->Error,
ERROR_FILE_EXISTS
);
}
return SmbStatusSendResponse;
}
//
// Set the time on the file.
//
// !!! Should we do anything with the return code?
rfcb = WorkContext->Rfcb;
#ifdef INCLUDE_SMB_IFMODIFIED
(VOID)SrvSetLastWriteTime(
rfcb,
SmbGetUlong( &request->CreationTimeInSeconds ),
rfcb->Lfcb->GrantedAccess,
FALSE
);
#else
(VOID)SrvSetLastWriteTime(
rfcb,
SmbGetUlong( &request->CreationTimeInSeconds ),
rfcb->Lfcb->GrantedAccess
);
#endif
//
// Give the smart card a chance to get into the act
//
if( WorkContext->Endpoint->IsConnectionless && SrvIpxSmartCard.Open != NULL ) {
PVOID handle;
IF_DEBUG( SIPX ) {
KdPrint(( "Trying the smart card for %wZ\n", &rfcb->Mfcb->FileName ));
}
if( SrvIpxSmartCard.Open(
WorkContext->RequestBuffer->Buffer,
rfcb->Lfcb->FileObject,
&rfcb->Mfcb->FileName,
&(WorkContext->ClientAddress->IpxAddress.Address[0].Address[0]),
rfcb->Lfcb->FileObject->Flags & FO_CACHE_SUPPORTED,
&handle
) == TRUE ) {
IF_DEBUG( SIPX ) {
KdPrint(( "%wZ handled by Smart Card. Handle %p\n",
&rfcb->Mfcb->FileName, handle ));
}
rfcb->PagedRfcb->IpxSmartCardContext = handle;
}
}
//
// Set up response SMB.
//
response->WordCount = 1;
SmbPutUshort( &response->Fid, rfcb->Fid );
SmbPutUshort( &response->ByteCount, 0 );
WorkContext->ResponseParameters = NEXT_LOCATION(
response,
RESP_CREATE,
0
);
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbCreate complete.\n" ));
return SmbStatusSendResponse;
} // SrvSmbCreate
SMB_PROCESSOR_RETURN_TYPE
SrvSmbCreateTemporary (
SMB_PROCESSOR_PARAMETERS
)
/*++
Routine Description:
Processes a Create Temporary SMB.
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbprocs.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbprocs.h
--*/
{
PREQ_CREATE_TEMPORARY request;
PRESP_CREATE_TEMPORARY response;
PRFCB rfcb;
NTSTATUS status = STATUS_OBJECT_NAME_COLLISION;
SMB_STATUS SmbStatus = SmbStatusInProgress;
CLONG nameCounter;
USHORT i;
PSZ nameStart, ep;
CHAR name[9];
PAGED_CODE( );
if (WorkContext->PreviousSMB == EVENT_TYPE_SMB_LAST_EVENT)
WorkContext->PreviousSMB = EVENT_TYPE_SMB_CREATE_TEMPORARY;
SrvWmiStartContext(WorkContext);
IF_SMB_DEBUG(OPEN_CLOSE1) {
KdPrint(( "Create temporary file request header at 0x%p, response header at 0x%p\n",
WorkContext->RequestHeader,
WorkContext->ResponseHeader ));
KdPrint(( "Create temporary file request parameters at 0x%p, response parameters at 0x%p\n",
WorkContext->RequestParameters,
WorkContext->ResponseParameters ));
}
request = (PREQ_CREATE_TEMPORARY)WorkContext->RequestParameters;
response = (PRESP_CREATE_TEMPORARY)WorkContext->ResponseParameters;
//
// Find out where in the buffer the directory pathname ends. We will
// write the filename after this.
//
ep = END_OF_REQUEST_SMB( WorkContext );
for( nameStart = (PSZ)request->Buffer; nameStart <= ep && *nameStart; nameStart++ ) {
;
}
//
// If we didn't find a NULL, or we don't have room to add in the last part of the
// temporary filename, then we give up
//
if( *nameStart ||
WorkContext->RequestBuffer->BufferLength - WorkContext->RequestBuffer->DataLength < 9 ) {
//
// We did not find a valid pathname!
//
SrvSetSmbError( WorkContext, STATUS_INVALID_SMB );
status = STATUS_INVALID_SMB;
SmbStatus = SmbStatusSendResponse;
goto Cleanup;
}
//
// The temporary file will be created with a name like SRVxxxxx, where
// xxxxx is a hex integer. We first try to create SRV00000, and if it
// exists increment xxxxx until xxxxx = 0xFFFFF;
//
// !!! We may want to maintain a "last name" counter, to try to
// reduce the number of retries we need. We may also want to
// have an explicit bound the number of tries, like 16 or 32, or
// 293.
//
name[0] = 'S';
name[1] = 'R';
name[2] = 'V';
name[8] = '\0';
// *** for SrvCanonicalizePathName
WorkContext->RequestBuffer->DataLength += 9;
for ( nameCounter = 0;
status == STATUS_OBJECT_NAME_COLLISION &&
nameCounter < (CLONG)0xFFFFF;
nameCounter++ ) {
CCHAR j;
PSZ s;
name[3] = SrvHexChars[ (nameCounter & (CLONG)0xF0000) >> 16 ];
name[4] = SrvHexChars[ (nameCounter & (CLONG)0xF000) >> 12 ];
name[5] = SrvHexChars[ (nameCounter & (CLONG)0xF00) >> 8 ];
name[6] = SrvHexChars[ (nameCounter & (CLONG)0xF0) >> 4 ];
name[7] = SrvHexChars[ (nameCounter & (CLONG)0xF) ];
// *** We could get rid of this loop and the name[9] variable
// if we could put the name directly into the SMB buffer.
for ( j = 0, s = nameStart; j < 9; j++, s++ ) {
*s = name[j];
}
//
// Open the file in compatibility mode, obtaining read/write
// access for this FID.
//
status = SrvCreateFile(
WorkContext,
SMB_DA_SHARE_COMPATIBILITY | SMB_DA_ACCESS_READ_WRITE,
0, // SmbFileAttributes (normal)
SMB_OFUN_OPEN_FAIL | SMB_OFUN_CREATE_CREATE,
0, // SmbAllocationSize
(PCHAR)(request->Buffer + 1),
END_OF_REQUEST_SMB( WorkContext ),
NULL,
0L,
NULL,
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK_NOTIFY_ANY) != 0 ?
OplockTypeBatch :
(WorkContext->RequestHeader->Flags & SMB_FLAGS_OPLOCK) != 0 ?
OplockTypeExclusive : OplockTypeServerBatch,
NULL
);
ASSERT ( status != STATUS_OPLOCK_BREAK_IN_PROGRESS );
//
// If the open failed, send an error response.
//
if ( !NT_SUCCESS(status) && status != STATUS_OBJECT_NAME_COLLISION ) {
SrvSetSmbError( WorkContext, status );
SmbStatus = SmbStatusSendResponse;
goto Cleanup;
}
}
if ( nameCounter == (CLONG)0xFFFFF ) {
//
// This will probably never happen, and if it did, it would take
// a *long* time to attempt to open FFFFF == 1,048,575 files.
//
SmbStatus = SmbStatusSendResponse;
goto Cleanup;
}
//
// Set up response SMB.
//
rfcb = WorkContext->Rfcb;
rfcb->IsCacheable = FALSE;
response->WordCount = 1;
SmbPutUshort( &response->Fid, rfcb->Fid );
for ( i = 0; i < 9; i++ ) {
request->Buffer[i] = nameStart[i];
}
SmbPutUshort( &response->ByteCount, 9 );
WorkContext->ResponseParameters = NEXT_LOCATION(
response,
RESP_CREATE_TEMPORARY,
9
);
SmbStatus = SmbStatusSendResponse;
IF_DEBUG(TRACE2) KdPrint(( "SrvSmbCreateTemporary complete.\n" ));
Cleanup:
SrvWmiEndContext(WorkContext);
return SmbStatus;
} // SrvSmbCreateTemporary
VOID
SetEofToMatchAllocation (
IN HANDLE FileHandle,
IN ULONG AllocationSize
)
/*++
Routine Description:
Sets the EOF location for a file to match the size allocated when
the file was created. This routine is necessary in order to gain
compatibility with OS/2, which does not have separate concepts of
EOF and allocation size. When the server creates a file for an OS/2
client, if the allocation size is greater than 0, the server sets
the EOF to match that size.
This routine was created to allow the server to pass variations 17
and 18 of the filio003 test.
Arguments:
SMB_PROCESSOR_PARAMETERS - See smbtypes.h for a description
of the parameters to SMB processor routines.
Return Value:
SMB_PROCESSOR_RETURN_TYPE - See smbtypes.h
--*/
{
NTSTATUS status;
IO_STATUS_BLOCK iosb;
FILE_END_OF_FILE_INFORMATION newEndOfFile;
PAGED_CODE( );
//
// Don't bother doing this if the allocated size is zero.
//
if ( AllocationSize != 0 ) {
newEndOfFile.EndOfFile.QuadPart = AllocationSize;
status = NtSetInformationFile(
FileHandle,
&iosb,
&newEndOfFile,
sizeof( newEndOfFile ),
FileEndOfFileInformation
);
if ( !NT_SUCCESS(status) ) {
INTERNAL_ERROR(
ERROR_LEVEL_EXPECTED,
"SetEofToMatchAllocation: SetInformationFile returned %X",
status,
NULL
);
SrvLogServiceFailure( SRV_SVC_NT_SET_INFO_FILE, status );
}
}
return;
} // SetEofToMatchAllocation
#ifdef INCLUDE_SMB_IFMODIFIED
#if 0
NTSTATUS
SrvGetUsnInfoForFile(
IN PWORK_CONTEXT WorkContext,
IN PRFCB Rfcb,
IN BOOLEAN SubmitClose,
OUT PLARGE_INTEGER Usn,
OUT PLARGE_INTEGER FileRefNumber
)
{
PIRP irp;
ULONG fsControl;
Usn->QuadPart = 0;
FileRefNumber->QuadPart = 0;
if (! Rfcb->Lfcb->TreeConnect->Share->UsnCapable) {
return STATUS_SUCCESS;
}
// if the client opened the file with the short name, we could end up
// having a buffer that's woefully too small. We'll bump the
// allocation up to a large amount so that we don't have to
// go back and do it again.
sizeRequired = MAXIMUM_FILENAME_LENGTH * sizeof(WCHAR);
sizeRequired += sizeof( USN_RECORD );
sizeRequired = QuadAlign( sizeRequired );
usnRecord = ALLOCATE_NONPAGED_POOL( sizeRequired, BlockTypeDataBuffer );
if ( usnRecord == NULL) {
// IF_DEBUG(ERRORS) {
KdPrint(( "SrvGetUsnInfoForFile: unable to alloc block of size %u for handle 0x%x\n",
sizeRequired, Rfcb->Lfcb->FileObject ));
// }
return STATUS_INSUFF_SERVER_RESOURCES;
}
fsControl = SubmitClose ? FSCTL_WRITE_USN_CLOSE_RECORD : FSCTL_READ_FILE_USN_DATA;
//
// get the current USN number for this file.
//
irp = SrvBuildIoControlRequest(
NULL, // input IRP address
Rfcb->Lfcb->FileObject, // target file object address
WorkContext, // context
IRP_MJ_FILE_SYSTEM_CONTROL, // major function
fsControl, // minor function
NULL, // input buffer address
0, // input buffer length
usnRecord, // output buffer address
sizeRequired, // output buffer length
NULL, // MDL address
SrvMdlCompletionRoutine // completion routine
);
if (irp != NULL) {
status = IoCallDriver( Rfcb->Lfcb->DeviceObject, irp );
// if ( !NT_SUCCESS(status) ) {
ASSERT( status != STATUS_MORE_PROCESSING_REQUIRED );
ASSERT( status != STATUS_PENDING );
IF_DEBUG(ERRORS) {
KdPrint(( "SrvGetUsnInfoForFile: Query USN info failed: 0x%X for handle %u\n",
status, Rfcb->Lfcb->FileObject ));
// }
} else {
Usn->QuadPart = usnRecord->Usn;
FileRefNumber->QuadPart = usnRecord->FileReferenceNumber;
}
IoFreeIrp( irp );
} else {
IF_DEBUG(ERRORS) {
KdPrint(( "SrvGetUsnInfoForFile: unable to alloc irp for handle 0x%x\n",
Rfcb->Lfcb->FileObject ));
}
status = STATUS_INSUFF_SERVER_RESOURCES;
}
DEALLOCATE_NONPAGED_POOL( usnRecord );
return status;
}
#endif
#endif // def INCLUDE_SMB_IFMODIFIED