windows-nt/Source/XPSP1/NT/base/fs/rdr2/rdbss/close.c
2020-09-26 16:20:57 +08:00

515 lines
16 KiB
C

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
Close.c
Abstract:
This module implements the File Close routine for Rx called by the
dispatch driver.
Author:
Joe Linn [JoeLinn] sep-9-1994
Revision History:
--*/
#include "precomp.h"
#pragma hdrstop
//
// The Bug check file id for this module
//
#define BugCheckFileId (RDBSS_BUG_CHECK_CLOSE)
//
// The local debug trace level
//
#define Dbg (DEBUG_TRACE_CLOSE)
enum _CLOSE_DEBUG_BREAKPOINTS {
CloseBreakPoint_BeforeCloseFakeFcb = 1,
CloseBreakPoint_AfterCloseFakeFcb
};
VOID
RxPurgeNetFcb(
IN OUT PFCB pFcb,
IN OUT PRX_CONTEXT pRxContext
);
#ifdef ALLOC_PRAGMA
#pragma alloc_text(PAGE, RxCommonClose)
#pragma alloc_text(PAGE, RxPurgeNetFcb)
#pragma alloc_text(PAGE, RxCloseAssociatedSrvOpen)
#endif
NTSTATUS
RxCommonClose ( RXCOMMON_SIGNATURE )
/*++
Routine Description:
Close is invoked whenever the last reference to a file object is deleted.
Cleanup is invoked when the last handle to a file object is closed, and
is called before close.
Arguments:
Return Value:
RXSTATUS - The return status for the operation
Notes:
The CLOSE handling strategy in RDBSS is predicated upon the axiom that the
workload on the server should be minimized as and when possible.
There are a number of applications which repeatedly close and open the same
file, e.g., batch file processing. In these cases the same file is opened,
a line/buffer is read, the file is closed and the same set of operations are
repeated over and over again.
This is handled in RDBSS by a delayed processing of the CLOSE request. There
is a delay ( of about 10 seconds ) between completing the request and initiating
processing on the request. This opens up a window during which a subsequent
OPEN can be collapsed onto an existing SRV_OPEN. The time interval can be tuned
to meet these requirements.
--*/
{
NTSTATUS Status;
RxCaptureRequestPacket;
RxCaptureFcb;
RxCaptureFobx;
RxCaptureParamBlock;
RxCaptureFileObject;
TYPE_OF_OPEN TypeOfOpen = NodeType(capFcb);
BOOLEAN AcquiredFcb = FALSE;
PAGED_CODE();
RxDbgTrace(+1, Dbg, ("RxCommonClose IrpC/Fobx/Fcb = %08lx %08lx %08lx\n",
RxContext,capFobx,capFcb));
RxLog(("CClose %lx %lx %lx %lx\n",RxContext,capFobx,capFcb,capFileObject));
RxWmiLog(LOG,
RxCommonClose_1,
LOGPTR(RxContext)
LOGPTR(capFobx)
LOGPTR(capFcb)
LOGPTR(capFileObject));
Status = RxAcquireExclusiveFcb( RxContext, capFcb );
if (Status != STATUS_SUCCESS) {
RxDbgTrace(-1, Dbg, ("RxCommonClose Cannot acquire FCB(%lx) %lx\n",capFcb,Status));
return Status;
}
AcquiredFcb = TRUE;
try {
switch (TypeOfOpen) {
case RDBSS_NTC_STORAGE_TYPE_UNKNOWN:
case RDBSS_NTC_STORAGE_TYPE_FILE:
case RDBSS_NTC_STORAGE_TYPE_DIRECTORY:
case RDBSS_NTC_OPENTARGETDIR_FCB:
case RDBSS_NTC_IPC_SHARE:
case RDBSS_NTC_MAILSLOT:
case RDBSS_NTC_SPOOLFILE:
{
PSRV_OPEN SrvOpen = NULL;
BOOLEAN fDelayClose = FALSE;
RxDbgTrace(0, Dbg, ("Close UserFileOpen/UserDirectoryOpen/OpenTargetDir %04lx\n", TypeOfOpen));
RxReferenceNetFcb(capFcb);
if (capFobx) {
SrvOpen = capFobx->SrvOpen;
if ((NodeType(capFcb) != RDBSS_NTC_STORAGE_TYPE_DIRECTORY) &&
(!FlagOn(capFcb->FcbState,FCB_STATE_ORPHANED)) &&
(!FlagOn(capFcb->FcbState,FCB_STATE_DELETE_ON_CLOSE)) &&
(FlagOn(capFcb->FcbState,FCB_STATE_COLLAPSING_ENABLED))) {
PSRV_CALL pSrvCall = capFcb->NetRoot->SrvCall;
RxLog(("@@@@DelayCls FOBX %lx SrvOpen %lx@@\n",capFobx,SrvOpen));
RxWmiLog(LOG,
RxCommonClose_2,
LOGPTR(capFobx)
LOGPTR(SrvOpen));
// If this is the last open instance and the close is being delayed
// mark the SRV_OPEN. This will enable us to respond to buffering
// state change requests with a close operation as opposed to
// the regular flush/purge response.
// We also check the COLLAPSING_DISABLED flag to determine whether its even necessary to delay
// close the file. If we cannot collapse the open, no reason to delay its closure. Delaying here
// caused us to stall for 10 seconds on an oplock break to a delay closed file because the final close
// caused by the break was delay-closed again, resulting in a delay before the oplock break is satisfied.
if ( (SrvOpen->OpenCount == 1) && !FlagOn( SrvOpen->Flags, SRVOPEN_FLAG_COLLAPSING_DISABLED ) ) {
if (InterlockedIncrement(&pSrvCall->NumberOfCloseDelayedFiles) <
pSrvCall->MaximumNumberOfCloseDelayedFiles) {
fDelayClose = TRUE;
SrvOpen->Flags |= SRVOPEN_FLAG_CLOSE_DELAYED;
} else {
RxDbgTrace(0,Dbg,("Not delaying files because count exceeded limit\n"));
InterlockedDecrement(&pSrvCall->NumberOfCloseDelayedFiles);
}
}
}
if (!fDelayClose) {
PNET_ROOT NetRoot = (PNET_ROOT)capFcb->pNetRoot;
if (NetRoot->Type != NET_ROOT_PRINT &&
FlagOn(capFobx->Flags, FOBX_FLAG_DELETE_ON_CLOSE)) {
RxScavengeRelatedFobxs(capFcb);
RxSynchronizeWithScavenger(RxContext);
RxReleaseFcb(NULL,capFcb);
RxAcquireFcbTableLockExclusive( &NetRoot->FcbTable, TRUE);
RxOrphanThisFcb(capFcb);
RxReleaseFcbTableLock( &NetRoot->FcbTable );
Status = RxAcquireExclusiveFcb(NULL,capFcb);
ASSERT(Status == STATUS_SUCCESS);
}
}
RxMarkFobxOnClose(capFobx);
}
if (!fDelayClose) {
Status = RxCloseAssociatedSrvOpen(capFobx,RxContext);
if (capFobx != NULL) {
RxDereferenceNetFobx(capFobx,LHS_ExclusiveLockHeld);
}
} else {
ASSERT(capFobx != NULL);
RxDereferenceNetFobx(capFobx,LHS_SharedLockHeld);
}
AcquiredFcb = !RxDereferenceAndFinalizeNetFcb(capFcb,RxContext,FALSE,FALSE);
capFileObject->FsContext = IntToPtr(0xffffffff);
if (AcquiredFcb) {
AcquiredFcb = FALSE;
RxReleaseFcb( RxContext, capFcb );
} else {
//the tracker gets very unhappy if you don't do this!
RxTrackerUpdateHistory(RxContext,NULL,'rrCr',__LINE__,__FILE__,0);
}
}
break;
default:
RxBugCheck( TypeOfOpen, 0, 0 );
break;
}
} finally {
if (AbnormalTermination()) {
if (AcquiredFcb) {
RxReleaseFcb( RxContext, capFcb );
}
} else {
ASSERT( !AcquiredFcb );
}
RxDbgTrace(-1, Dbg, ("RxCommonClose -> %08lx\n", Status));
}
return Status;
}
VOID
RxPurgeNetFcb(
IN OUT PFCB pFcb,
IN OUT PRX_CONTEXT pRxContext
)
/*++
Routine Description:
This routine initiates the purge on an FCB instance
Arguments:
pFcb - the FOBX instance for which close processing is to be initiated
RxContext - the context
Return Value:
Notes:
On entry to this routine the FCB must have been accquired exclusive.
On exit there is no change in resource ownership
--*/
{
NTSTATUS Status;
PAGED_CODE();
RxDbgTrace(0, Dbg, ("CleanupPurge:MmFlushImage\n", 0));
MmFlushImageSection(
&pFcb->NonPaged->SectionObjectPointers,
MmFlushForWrite);
// we dont pass in the context here because it is not necessary to track this
// release because of the subsequent acquire...........
RxReleaseFcb( NULL, pFcb );
MmForceSectionClosed(
&pFcb->NonPaged->SectionObjectPointers,
TRUE);
Status = RxAcquireExclusiveFcb(NULL,pFcb);
ASSERT(Status == STATUS_SUCCESS);
}
NTSTATUS
RxCloseAssociatedSrvOpen(
IN OUT PFOBX pFobx,
IN OUT PRX_CONTEXT RxContext)
/*++
Routine Description:
This routine initiates the close processing for an FOBX. The FOBX close
processing can be trigerred in one of three ways ....
1) Regular close processing on receipt of the IRP_MJ_CLOSE for the associated
file object.
2) Delayed close processing while scavenging the FOBX. This happens when the
close processing was delayed in anticipation of an open and no opens are
forthcoming.
3) Delayed close processing on receipt of a buffering state change request
for a close that was delayed.
Arguments:
pFobx - the FOBX instance for which close processing is to be initiated.
It is NULL for MAILSLOT files.
RxContext - the context parameter is NULL for case (2).
Return Value:
Notes:
On entry to this routine the FCB must have been accquired exclusive.
On exit there is no change in resource ownership
--*/
{
NTSTATUS Status = STATUS_MORE_PROCESSING_REQUIRED;
PFCB pFcb;
PSRV_OPEN pSrvOpen;
PRX_CONTEXT pLocalRxContext;
PAGED_CODE();
// Distinguish between those cases where there is a real SRV_OPEN instance
// from those that do not have one, e.g., mailslot files.
if (pFobx == NULL) {
if (RxContext != NULL) {
pFcb = (PFCB)(RxContext->pFcb);
pSrvOpen = NULL;
} else {
Status = STATUS_SUCCESS;
}
} else {
if (pFobx->Flags & FOBX_FLAG_SRVOPEN_CLOSED) {
RxMarkFobxOnClose(pFobx);
Status = STATUS_SUCCESS;
} else {
pSrvOpen = pFobx->SrvOpen;
if (pSrvOpen->Flags & SRVOPEN_FLAG_CLOSED) {
pFcb = pSrvOpen->Fcb;
ASSERT(RxIsFcbAcquiredExclusive( pFcb ));
pFobx->Flags |= FOBX_FLAG_SRVOPEN_CLOSED;
if (pSrvOpen->OpenCount > 0) {
pSrvOpen->OpenCount--;
}
RxMarkFobxOnClose(pFobx);
Status = STATUS_SUCCESS;
} else {
pFcb = pSrvOpen->Fcb;
}
}
}
// If there is no corresponding open on the server side or if the close
// processing has already been accomplished there is no further processing
// required. In other cases w.r.t scavenged close processing a new
// context might have to be created.
if ((Status == STATUS_MORE_PROCESSING_REQUIRED &&
(pLocalRxContext = RxContext) == NULL)) {
pLocalRxContext = RxCreateRxContext(
NULL,
pSrvOpen->Fcb->RxDeviceObject,
RX_CONTEXT_FLAG_WAIT|RX_CONTEXT_FLAG_MUST_SUCCEED_NONBLOCKING);
if (pLocalRxContext != NULL) {
pLocalRxContext->MajorFunction = IRP_MJ_CLOSE;
pLocalRxContext->pFcb = (PMRX_FCB)pFcb;
pLocalRxContext->pFobx = (PMRX_FOBX)pFobx;
if (pFobx != NULL)
{
pLocalRxContext->pRelevantSrvOpen = (PMRX_SRV_OPEN)(pFobx->SrvOpen);
}
Status = STATUS_MORE_PROCESSING_REQUIRED;
} else {
Status = STATUS_INSUFFICIENT_RESOURCES;
}
}
// if the context creation was successful and the close processing for
// the SRV_OPEN instance needs to be initiated with the mini rdr
// proceed.
if (Status == STATUS_MORE_PROCESSING_REQUIRED) {
ASSERT(RxIsFcbAcquiredExclusive( pFcb ));
// Mark the Fobx instance on the initiation of the close operation. This
// is the complement to the action taken on cleanup. It ensures
// that the infrastructure setup for delayed close processing is undone.
// For those instances in which the FOBS is NULL the FCB is manipulated
// directly
if (pFobx != NULL) {
RxMarkFobxOnClose(pFobx);
} else {
InterlockedDecrement(&pFcb->OpenCount);
}
if (pSrvOpen != NULL) {
if (pSrvOpen->Condition == Condition_Good) {
if (pSrvOpen->OpenCount > 0) {
pSrvOpen->OpenCount--;
}
if (pSrvOpen->OpenCount == 1) {
if (!IsListEmpty(&pSrvOpen->FobxList)) {
PFOBX remainingFobx;
remainingFobx = CONTAINING_RECORD(
pSrvOpen->FobxList.Flink,
FOBX,
FobxQLinks);
if (!IsListEmpty(&remainingFobx->ScavengerFinalizationList)) {
pSrvOpen->Flags |= SRVOPEN_FLAG_CLOSE_DELAYED;
}
}
}
// Purge the FCB before initiating the close processing with
// the mini redirectors
if ((pSrvOpen->OpenCount == 0) &&
(Status == STATUS_MORE_PROCESSING_REQUIRED) &&
(RxContext == NULL)) {
RxPurgeNetFcb(pFcb,pLocalRxContext);
}
// Since PurgeNetFcb drops and reacquires the resource, ensure that
// the SrvOpen is still valid before proceeding with the
// finalization.
pSrvOpen = pFobx->SrvOpen;
if ((pSrvOpen != NULL) &&
((pSrvOpen->OpenCount == 0) ||
(FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_ORPHANED))) &&
!FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_CLOSED) &&
(Status == STATUS_MORE_PROCESSING_REQUIRED)) {
ASSERT(RxIsFcbAcquiredExclusive( pFcb ));
MINIRDR_CALL(
Status,
pLocalRxContext,
pFcb->MRxDispatch,
MRxCloseSrvOpen,
(pLocalRxContext));
RxLog(("MRXClose %lx %lx %lx %lx %lx\n",RxContext,pFcb,pSrvOpen,pFobx,Status));
RxWmiLog(LOG,
RxCloseAssociatedSrvOpen,
LOGPTR(RxContext)
LOGPTR(pFcb)
LOGPTR(pSrvOpen)
LOGPTR(pFobx)
LOGULONG(Status));
pSrvOpen->Flags |= SRVOPEN_FLAG_CLOSED;
if (FlagOn(pSrvOpen->Flags,SRVOPEN_FLAG_CLOSE_DELAYED)) {
InterlockedDecrement(&pFcb->NetRoot->SrvCall->NumberOfCloseDelayedFiles);
}
RxRemoveShareAccessPerSrvOpens(pSrvOpen);
// Ensure that any buffering state change requests for this
// SRV_OPEN instance which was closed is purged from the
// buffering manager data structures.
RxPurgeChangeBufferingStateRequestsForSrvOpen(pSrvOpen);
RxDereferenceSrvOpen(pSrvOpen,LHS_ExclusiveLockHeld);
} else {
Status = STATUS_SUCCESS;
}
pFobx->Flags |= FOBX_FLAG_SRVOPEN_CLOSED;
} else {
Status = STATUS_SUCCESS;
}
} else {
ASSERT((NodeType(pFcb) == RDBSS_NTC_OPENTARGETDIR_FCB) ||
(NodeType(pFcb) == RDBSS_NTC_IPC_SHARE) ||
(NodeType(pFcb) == RDBSS_NTC_MAILSLOT));
RxDereferenceNetFcb(pFcb);
Status = STATUS_SUCCESS;
}
if (pLocalRxContext != RxContext) {
RxDereferenceAndDeleteRxContext(pLocalRxContext);
}
}
return Status;
}