Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1840 lines
53 KiB

  1. /*++
  2. Copyright (c) 1989 Microsoft Corporation
  3. Module Name:
  4. scavengr.c
  5. Abstract:
  6. This module implements the scavenging routines in RDBSS.
  7. Author:
  8. Balan Sethu Raman [SethuR] 9-sep-1995
  9. Revision History:
  10. --*/
  11. #include "precomp.h"
  12. #pragma hdrstop
  13. KMUTEX RxScavengerMutex; //only one of these!
  14. VOID
  15. RxScavengerFinalizeEntries(
  16. PRDBSS_DEVICE_OBJECT RxDeviceObject
  17. );
  18. PRDBSS_DEVICE_OBJECT
  19. RxGetDeviceObjectOfInstance (
  20. PVOID pInstance
  21. );
  22. VOID
  23. RxScavengerTimerRoutine(
  24. PVOID pContext); //actually a rxdeviceobject
  25. #ifdef ALLOC_PRAGMA
  26. #pragma alloc_text(PAGE, RxPurgeFobxFromCache)
  27. #pragma alloc_text(PAGE, RxMarkFobxOnCleanup)
  28. #pragma alloc_text(PAGE, RxMarkFobxOnClose)
  29. #pragma alloc_text(PAGE, RxPurgeFobx)
  30. #pragma alloc_text(PAGE, RxInitializePurgeSyncronizationContext)
  31. #pragma alloc_text(PAGE, RxPurgeRelatedFobxs)
  32. #pragma alloc_text(PAGE, RxPurgeAllFobxs)
  33. #pragma alloc_text(PAGE, RxGetDeviceObjectOfInstance)
  34. #pragma alloc_text(PAGE, RxpMarkInstanceForScavengedFinalization)
  35. #pragma alloc_text(PAGE, RxpUndoScavengerFinalizationMarking)
  36. #pragma alloc_text(PAGE, RxScavengeVNetRoots)
  37. #pragma alloc_text(PAGE, RxScavengeRelatedFobxs)
  38. #pragma alloc_text(PAGE, RxScavengeAllFobxs)
  39. #pragma alloc_text(PAGE, RxScavengerFinalizeEntries)
  40. #pragma alloc_text(PAGE, RxScavengerTimerRoutine)
  41. #pragma alloc_text(PAGE, RxTerminateScavenging)
  42. #endif
  43. //
  44. // Local debug trace level
  45. //
  46. #define Dbg (DEBUG_TRACE_SCAVENGER)
  47. #ifndef WIN9X
  48. #define RxAcquireFcbScavengerMutex(pFcbScavenger) \
  49. RxAcquireScavengerMutex(); \
  50. (pFcbScavenger)->State |= RX_SCAVENGER_MUTEX_ACQUIRED
  51. #define RxReleaseFcbScavengerMutex(pFcbScavenger) \
  52. (pFcbScavenger)->State &= ~RX_SCAVENGER_MUTEX_ACQUIRED; \
  53. RxReleaseScavengerMutex()
  54. #else
  55. #define RxAcquireFcbScavengerMutex(pFcbScavenger) \
  56. (pFcbScavenger)->State |= RX_SCAVENGER_MUTEX_ACQUIRED
  57. #define RxReleaseFcbScavengerMutex(pFcbScavenger) \
  58. (pFcbScavenger)->State &= ~RX_SCAVENGER_MUTEX_ACQUIRED
  59. #endif
  60. #define RX_SCAVENGER_FINALIZATION_TIME_INTERVAL (10 * 1000 * 1000 * 10)
  61. extern VOID
  62. RxScavengerTimerRoutine(
  63. PVOID pContext);
  64. NTSTATUS
  65. RxPurgeFobxFromCache(
  66. PFOBX pFobxToBePurged)
  67. /*++
  68. Routine Description:
  69. This routine purges an FOBX for which a close is pending
  70. Arguments:
  71. pFobxToBePurged -- the FOBX instance
  72. Notes:
  73. At cleanup there are no more user handles associated with the file object.
  74. In such cases the time window between close and clanup is dictated by the
  75. additional references maintained by the memory manager / cache manager.
  76. This routine unlike the one that follows does not attempt to force the
  77. operations from the memory manager. It merely purges the underlying FCB
  78. from the cache
  79. The FOBX must have been referenced on entry to this routine and it will
  80. lose that reference on exit.
  81. --*/
  82. {
  83. NTSTATUS Status = STATUS_UNSUCCESSFUL;
  84. PFCB pFcbToBePurged = pFobxToBePurged->SrvOpen->Fcb;
  85. PAGED_CODE();
  86. ASSERT(pFcbToBePurged != NULL);
  87. Status = RxAcquireExclusiveFcb(NULL,pFcbToBePurged);
  88. if (Status == STATUS_SUCCESS) {
  89. BOOLEAN fResult;
  90. RxReferenceNetFcb(pFcbToBePurged);
  91. if (!FlagOn(pFobxToBePurged->Flags,FOBX_FLAG_SRVOPEN_CLOSED) &&
  92. (pFobxToBePurged->SrvOpen->UncleanFobxCount == 0)) {
  93. Status = RxPurgeFcbInSystemCache(
  94. pFcbToBePurged,
  95. NULL,
  96. 0,
  97. FALSE,
  98. TRUE);
  99. } else {
  100. RxLog(("Skipping Purging %lx\n",pFobxToBePurged));
  101. RxWmiLog(LOG,
  102. RxPurgeFobxFromCache,
  103. LOGPTR(pFobxToBePurged));
  104. }
  105. RxDereferenceNetFobx(pFobxToBePurged,LHS_ExclusiveLockHeld);
  106. if ( !RxDereferenceAndFinalizeNetFcb(pFcbToBePurged,NULL,FALSE,FALSE) ) {
  107. RxReleaseFcb( NULL, pFcbToBePurged );
  108. }
  109. } else {
  110. RxDereferenceNetFobx(pFobxToBePurged,LHS_LockNotHeld);
  111. }
  112. return Status;
  113. }
  114. VOID
  115. RxMarkFobxOnCleanup(
  116. PFOBX pFobx,
  117. BOOLEAN *pNeedPurge)
  118. /*++
  119. Routine Description:
  120. Thie routine marks a FOBX for special processing on cleanup
  121. Arguments:
  122. pFobx -- the FOBX instance
  123. Notes:
  124. At cleanup there are no more user handles associated with the file object.
  125. In such cases the time window between close and clanup is dictated by the
  126. additional references maintained by the memory manager / cache manager.
  127. On cleanup the FOBX is put on a close pending list and removed from it
  128. when the corresponding list when a close operation is received. In the interim
  129. if an OPEN is failing with ACCESS_DENIED status then the RDBSS can force a
  130. purge.
  131. Only diskfile type fobxs are placed on the delayed-close list.
  132. --*/
  133. {
  134. PAGED_CODE();
  135. if (pFobx != NULL) {
  136. PFCB pFcb = pFobx->SrvOpen->Fcb;
  137. PLIST_ENTRY pListEntry;
  138. PRDBSS_DEVICE_OBJECT RxDeviceObject;
  139. PRDBSS_SCAVENGER pRdbssScavenger;
  140. PFOBX pFobxToBePurged = NULL;
  141. ASSERT(NodeTypeIsFcb(pFcb));
  142. RxDeviceObject = pFcb->RxDeviceObject;
  143. pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  144. RxAcquireScavengerMutex();
  145. if ((NodeType(pFcb) != RDBSS_NTC_STORAGE_TYPE_FILE)
  146. || (pFcb->VNetRoot->NetRoot->DeviceType != RxDeviceType(DISK)) ) {
  147. //
  148. // the markfobxatclose will want to remove this from a list. just fix up
  149. // the list pointers and get out
  150. SetFlag(pFobx->Flags,FOBX_FLAG_MARKED_AS_DORMANT);
  151. InitializeListHead(&pFobx->ClosePendingList);
  152. pRdbssScavenger->NumberOfDormantFiles++ ;
  153. } else {
  154. // Ensure that the limit of dormant files specified for the given server is
  155. // not exceeded. If the limit will be exceeded pick an entry from the
  156. // list of files that are currently dormant and purge it.
  157. ASSERT(pRdbssScavenger->NumberOfDormantFiles >= 0);
  158. if (pRdbssScavenger->NumberOfDormantFiles >=
  159. pRdbssScavenger->MaximumNumberOfDormantFiles) {
  160. // If the number of dormant files exceeds the limit specified for the
  161. // given server a currently dormant file needs to be picked up for
  162. // purging.
  163. pListEntry = pRdbssScavenger->ClosePendingFobxsList.Flink;
  164. if (pListEntry != &pRdbssScavenger->ClosePendingFobxsList) {
  165. pFobxToBePurged = (PFOBX)(CONTAINING_RECORD(
  166. pListEntry,
  167. FOBX,
  168. ClosePendingList));
  169. if ((pFobxToBePurged->SrvOpen != NULL) &&
  170. (pFobxToBePurged->SrvOpen->Fcb == pFcb)) {
  171. // The first FOBX in the close pending list and the one about to be
  172. // inserted share the same FCB. Instaed of removing the first one
  173. // a purge is forced on the current FOBX. This avoids the resource
  174. // release/acquire that would have been required otherwise
  175. *pNeedPurge = TRUE;
  176. pFobxToBePurged = NULL;
  177. } else {
  178. RxReferenceNetFobx(pFobxToBePurged);
  179. }
  180. }
  181. }
  182. SetFlag(pFobx->Flags,FOBX_FLAG_MARKED_AS_DORMANT);
  183. InsertTailList(
  184. &pRdbssScavenger->ClosePendingFobxsList,
  185. &pFobx->ClosePendingList);
  186. if (pRdbssScavenger->NumberOfDormantFiles++ == 0) {
  187. BOOLEAN PostTimerRequest;
  188. if (pRdbssScavenger->State == RDBSS_SCAVENGER_INACTIVE) {
  189. pRdbssScavenger->State = RDBSS_SCAVENGER_DORMANT;
  190. PostTimerRequest = TRUE;
  191. } else {
  192. PostTimerRequest = FALSE;
  193. }
  194. if (PostTimerRequest) {
  195. LARGE_INTEGER TimeInterval;
  196. // Post a one shot timer request for scheduling the scavenger after a
  197. // predetermined amount of time.
  198. TimeInterval.QuadPart = RX_SCAVENGER_FINALIZATION_TIME_INTERVAL;
  199. RxPostOneShotTimerRequest(
  200. RxFileSystemDeviceObject,
  201. &pRdbssScavenger->WorkItem,
  202. RxScavengerTimerRoutine,
  203. RxDeviceObject,
  204. TimeInterval);
  205. }
  206. }
  207. }
  208. RxReleaseScavengerMutex();
  209. if (pFobxToBePurged != NULL) {
  210. NTSTATUS Status;
  211. Status = RxPurgeFobxFromCache(pFobxToBePurged);
  212. if (Status != STATUS_SUCCESS) {
  213. *pNeedPurge = TRUE;
  214. }
  215. }
  216. }
  217. }
  218. VOID
  219. RxMarkFobxOnClose(PFOBX pFobx)
  220. /*++
  221. Routine Description:
  222. This routine undoes the marking done on cleanup
  223. Arguments:
  224. pFobx -- the FOBX instance
  225. Notes:
  226. At cleanup there are no more user handles associated with the file object.
  227. In such cases the time window between close and clanup is dictated by the
  228. additional references maintained by the memory manager / cache manager.
  229. On cleanup the FOBX is put on a close pending list and removed from it
  230. when the corresponding list when a close operation is received. In the interim
  231. if an OPEN is failing with ACCESS_DENIED status then the RDBSS can force a
  232. purge.
  233. --*/
  234. {
  235. PAGED_CODE();
  236. if (pFobx != NULL) {
  237. PFCB pFcb = pFobx->SrvOpen->Fcb;
  238. PRDBSS_DEVICE_OBJECT RxDeviceObject;
  239. PRDBSS_SCAVENGER pRdbssScavenger;
  240. ASSERT(NodeTypeIsFcb(pFcb));
  241. RxDeviceObject = pFcb->RxDeviceObject;
  242. pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  243. RxAcquireScavengerMutex();
  244. if (BooleanFlagOn(pFobx->Flags,FOBX_FLAG_MARKED_AS_DORMANT)) {
  245. if (!pFobx->fOpenCountDecremented) {
  246. PFCB pFcb = pFobx->SrvOpen->Fcb;
  247. ASSERT(NodeTypeIsFcb(pFcb));
  248. InterlockedDecrement(&pFcb->OpenCount);
  249. pFobx->fOpenCountDecremented = TRUE;
  250. }
  251. InterlockedDecrement(&pRdbssScavenger->NumberOfDormantFiles);
  252. ClearFlag(pFobx->Flags,FOBX_FLAG_MARKED_AS_DORMANT);
  253. }
  254. if (!IsListEmpty(&pFobx->ClosePendingList)) {
  255. RemoveEntryList(&pFobx->ClosePendingList);
  256. InitializeListHead(&pFobx->ClosePendingList);
  257. }
  258. RxReleaseScavengerMutex();
  259. }
  260. }
  261. BOOLEAN
  262. RxPurgeFobx(
  263. PFOBX pFobx)
  264. /*++
  265. Routine Description:
  266. This routine purges an FOBX for which a close is pending
  267. Arguments:
  268. pFobx -- the FOBX instance
  269. Notes:
  270. At cleanup there are no more user handles associated with the file object.
  271. In such cases the time window between close and clanup is dictated by the
  272. additional references maintained by the memory manager / cache manager.
  273. On cleanup the FOBX is put on a close pending list and removed from it
  274. when the corresponding list when a close operation is received. In the interim
  275. if an OPEN is failing with ACCESS_DENIED status then the RDBSS can force a
  276. purge.
  277. --*/
  278. {
  279. NTSTATUS Status;
  280. BOOLEAN fResult = TRUE;
  281. PFCB pFcb = pFobx->SrvOpen->Fcb;
  282. PAGED_CODE();
  283. Status = RxAcquireExclusiveFcb(NULL,pFcb);
  284. ASSERT(Status == STATUS_SUCCESS);
  285. // Carry out the purge operation
  286. Status = RxPurgeFcbInSystemCache(
  287. pFcb,
  288. NULL,
  289. 0,
  290. FALSE,
  291. TRUE);
  292. RxReleaseFcb( NULL, pFcb );
  293. fResult = (Status == STATUS_SUCCESS);
  294. if (!fResult) {
  295. RxLog(("PurgeFobxCCFail %lx %lx %lx",pFobx,pFcb,fResult));
  296. RxWmiLog(LOG,
  297. RxPurgeFobx_1,
  298. LOGPTR(pFobx)
  299. LOGPTR(pFcb));
  300. return(fResult);
  301. }
  302. //try to flush the image section....it may fail
  303. if (!MmFlushImageSection( &pFcb->NonPaged->SectionObjectPointers, MmFlushForWrite )) {
  304. RxLog(("PurgeFobxImFail %lx %lx %lx",pFobx,pFcb,FALSE));
  305. RxWmiLog(LOG,
  306. RxPurgeFobx_2,
  307. LOGPTR(pFobx)
  308. LOGPTR(pFcb));
  309. return(FALSE);
  310. }
  311. //try to flush the user data sections section....it may fail
  312. if (!MmForceSectionClosed(&pFcb->NonPaged->SectionObjectPointers, TRUE)) {
  313. RxLog(("PurgeFobxUsFail %lx %lx %lx",pFobx,pFcb,FALSE));
  314. RxWmiLog(LOG,
  315. RxPurgeFobx_3,
  316. LOGPTR(pFobx)
  317. LOGPTR(pFcb));
  318. return(FALSE);
  319. }
  320. RxLog(("PurgeFobx %lx %lx %lx",pFobx,pFcb,TRUE));
  321. RxWmiLog(LOG,
  322. RxPurgeFobx_4,
  323. LOGPTR(pFobx)
  324. LOGPTR(pFcb));
  325. return TRUE;
  326. }
  327. VOID
  328. RxInitializePurgeSyncronizationContext (
  329. PPURGE_SYNCHRONIZATION_CONTEXT PurgeSyncronizationContext
  330. )
  331. {
  332. PAGED_CODE();
  333. InitializeListHead(&PurgeSyncronizationContext->ContextsAwaitingPurgeCompletion);
  334. PurgeSyncronizationContext->PurgeInProgress = FALSE;
  335. }
  336. VOID
  337. RxSynchronizeWithScavenger(
  338. PRX_CONTEXT RxContext)
  339. {
  340. NTSTATUS Status;
  341. RxCaptureFcb;
  342. BOOLEAN ReacquireFcbLock = FALSE;
  343. PRDBSS_SCAVENGER pRdbssScavenger = capFcb->RxDeviceObject->pRdbssScavenger;
  344. RxAcquireScavengerMutex();
  345. if ((pRdbssScavenger->CurrentScavengerThread != PsGetCurrentThread()) &&
  346. (pRdbssScavenger->CurrentFcbForClosePendingProcessing == capFcb)) {
  347. ReacquireFcbLock = TRUE;
  348. RxReleaseFcb(RxContext,capFcb);
  349. while (pRdbssScavenger->CurrentFcbForClosePendingProcessing == capFcb) {
  350. RxReleaseScavengerMutex();
  351. KeWaitForSingleObject(
  352. &(pRdbssScavenger->ClosePendingProcessingSyncEvent),
  353. Executive,
  354. KernelMode,
  355. TRUE,
  356. NULL);
  357. RxAcquireScavengerMutex();
  358. }
  359. }
  360. RxReleaseScavengerMutex();
  361. if (ReacquireFcbLock) {
  362. Status = RxAcquireExclusiveFcb( RxContext, capFcb );
  363. ASSERT(Status == STATUS_SUCCESS);
  364. }
  365. }
  366. NTSTATUS
  367. RxPurgeRelatedFobxs(
  368. PNET_ROOT pNetRoot,
  369. PRX_CONTEXT pRxContext,
  370. BOOLEAN AttemptFinalize,
  371. PFCB PurgingFcb
  372. )
  373. /*++
  374. Routine Description:
  375. This routine purges all the FOBX's associated with a NET_ROOT
  376. Arguments:
  377. pNetRoot -- the NET_ROOT for which the FOBX's need to be purged
  378. pRxContext -- the RX_CONTEXT instance
  379. Notes:
  380. At cleanup there are no more user handles associated with the file object.
  381. In such cases the time window between close and clanup is dictated by the
  382. additional references maintained by the memory manager / cache manager.
  383. On cleanup the FOBX is put on a close pending list and removed from it
  384. when the corresponding list when a close operation is received. In the interim
  385. if an OPEN is failing with ACCESS_DENIED status then the RDBSS can force a
  386. purge.
  387. This is a synchronous operation.
  388. --*/
  389. {
  390. BOOLEAN ScavengerMutexAcquired = FALSE;
  391. NTSTATUS Status;
  392. ULONG FobxsSuccessfullyPurged = 0;
  393. PPURGE_SYNCHRONIZATION_CONTEXT PurgeSyncronizationContext;
  394. LIST_ENTRY FailedToPurgeFobxList;
  395. PRDBSS_DEVICE_OBJECT RxDeviceObject = pRxContext->RxDeviceObject;
  396. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  397. PAGED_CODE();
  398. PurgeSyncronizationContext = &pNetRoot->PurgeSyncronizationContext;
  399. InitializeListHead(&FailedToPurgeFobxList);
  400. RxAcquireScavengerMutex();
  401. // If the purge operation for this net root is currently under way hold
  402. // this request till it completes else initiate the operation after
  403. // updating the state of the net root.
  404. if (PurgeSyncronizationContext->PurgeInProgress) {
  405. InsertTailList(
  406. &PurgeSyncronizationContext->ContextsAwaitingPurgeCompletion,
  407. &pRxContext->RxContextSerializationQLinks);
  408. RxReleaseScavengerMutex();
  409. RxWaitSync(pRxContext);
  410. RxAcquireScavengerMutex();
  411. }
  412. PurgeSyncronizationContext->PurgeInProgress = TRUE;
  413. RxWmiLog(LOG,
  414. RxPurgeRelatedFobxs_3,
  415. LOGPTR(pRxContext)
  416. LOGPTR(pNetRoot));
  417. while (pRdbssScavenger->CurrentNetRootForClosePendingProcessing == pNetRoot) {
  418. RxReleaseScavengerMutex();
  419. KeWaitForSingleObject(
  420. &(pRdbssScavenger->ClosePendingProcessingSyncEvent),
  421. Executive,
  422. KernelMode,
  423. TRUE,
  424. NULL);
  425. RxAcquireScavengerMutex();
  426. }
  427. ScavengerMutexAcquired = TRUE;
  428. // An attempt should be made to purge all the FOBX's that had a close
  429. // pending before the purge request was received.
  430. for (;;) {
  431. PLIST_ENTRY pListEntry = NULL;
  432. PFOBX pFobx;
  433. PFCB pFcb;
  434. BOOLEAN PurgeResult;
  435. pFobx = NULL;
  436. pListEntry = pRdbssScavenger->ClosePendingFobxsList.Flink;
  437. while (pListEntry != &pRdbssScavenger->ClosePendingFobxsList) {
  438. PFOBX pTempFobx;
  439. pTempFobx = (PFOBX)(CONTAINING_RECORD(
  440. pListEntry,
  441. FOBX,
  442. ClosePendingList));
  443. RxLog(("TempFobx=%lx",pTempFobx));
  444. RxWmiLog(LOG,
  445. RxPurgeRelatedFobxs_1,
  446. LOGPTR(pTempFobx));
  447. if ((pTempFobx->SrvOpen != NULL) &&
  448. (pTempFobx->SrvOpen->Fcb != NULL) &&
  449. (pTempFobx->SrvOpen->Fcb->VNetRoot != NULL) &&
  450. ((PNET_ROOT)pTempFobx->SrvOpen->Fcb->VNetRoot->NetRoot == pNetRoot)) {
  451. NTSTATUS PurgeStatus = STATUS_MORE_PROCESSING_REQUIRED;
  452. if ((PurgingFcb != NULL) &&
  453. (pTempFobx->SrvOpen->Fcb != PurgingFcb)) {
  454. MINIRDR_CALL_THROUGH(
  455. PurgeStatus,
  456. RxDeviceObject->Dispatch,
  457. MRxAreFilesAliased,(pTempFobx->SrvOpen->Fcb,PurgingFcb)
  458. );
  459. }
  460. if (PurgeStatus != STATUS_SUCCESS) {
  461. RemoveEntryList(pListEntry);
  462. InitializeListHead(pListEntry);
  463. pFobx = pTempFobx;
  464. break;
  465. } else {
  466. pListEntry = pListEntry->Flink;
  467. }
  468. } else {
  469. pListEntry = pListEntry->Flink;
  470. }
  471. }
  472. if (pFobx != NULL) {
  473. RxReferenceNetFobx(pFobx);
  474. } else {
  475. // Try to wake up the next waiter if any.
  476. if (!IsListEmpty(&PurgeSyncronizationContext->ContextsAwaitingPurgeCompletion)) {
  477. PLIST_ENTRY pContextListEntry;
  478. PRX_CONTEXT pNextContext;
  479. pContextListEntry = PurgeSyncronizationContext->ContextsAwaitingPurgeCompletion.Flink;
  480. RemoveEntryList(pContextListEntry);
  481. pNextContext = (PRX_CONTEXT)(CONTAINING_RECORD(
  482. pContextListEntry,
  483. RX_CONTEXT,
  484. RxContextSerializationQLinks));
  485. RxSignalSynchronousWaiter(pNextContext);
  486. } else {
  487. PurgeSyncronizationContext->PurgeInProgress = FALSE;
  488. }
  489. }
  490. ScavengerMutexAcquired = FALSE;
  491. RxReleaseScavengerMutex();
  492. if (pFobx == NULL) {
  493. break;
  494. }
  495. // Purge the FOBX.
  496. if (PurgeResult=RxPurgeFobx(pFobx)) {
  497. FobxsSuccessfullyPurged++;
  498. }
  499. pFcb = pFobx->SrvOpen->Fcb;
  500. if (AttemptFinalize
  501. && (RxAcquireExclusiveFcb(NULL,pFcb) == STATUS_SUCCESS)) {
  502. RxReferenceNetFcb(pFcb);
  503. RxDereferenceNetFobx(pFobx,LHS_ExclusiveLockHeld);
  504. if ( !RxDereferenceAndFinalizeNetFcb(pFcb,NULL,FALSE,FALSE) ) {
  505. RxReleaseFcb( NULL, pFcb );
  506. }
  507. } else {
  508. RxDereferenceNetFobx(pFobx,LHS_LockNotHeld);
  509. }
  510. if (!PurgeResult) {
  511. RxLog(("SCVNGR:FailedToPurge %lx\n", pFcb));
  512. RxWmiLog(LOG,
  513. RxPurgeRelatedFobxs_2,
  514. LOGPTR(pFcb));
  515. }
  516. RxAcquireScavengerMutex();
  517. ScavengerMutexAcquired = TRUE;
  518. }
  519. if (ScavengerMutexAcquired) {
  520. RxReleaseScavengerMutex();
  521. }
  522. Status = (FobxsSuccessfullyPurged > 0) ? (STATUS_SUCCESS) : (STATUS_UNSUCCESSFUL);
  523. return Status;
  524. }
  525. VOID
  526. RxPurgeAllFobxs(
  527. PRDBSS_DEVICE_OBJECT RxDeviceObject)
  528. /*++
  529. Routine Description:
  530. This routine purges all the FOBX's while stopping the scavenger
  531. Arguments:
  532. RxDeviceObject -- the mini redirector device for which the purge should be done
  533. --*/
  534. {
  535. PLIST_ENTRY pListEntry = NULL;
  536. PFOBX pFobx;
  537. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  538. PAGED_CODE();
  539. for (;;) {
  540. PFCB pFcb;
  541. RxAcquireScavengerMutex();
  542. pListEntry = pRdbssScavenger->ClosePendingFobxsList.Flink;
  543. ASSERT(pListEntry!=NULL);
  544. if (pListEntry != &pRdbssScavenger->ClosePendingFobxsList) {
  545. pFobx = (PFOBX)(CONTAINING_RECORD(
  546. pListEntry,
  547. FOBX,
  548. ClosePendingList));
  549. ASSERT ((pFobx->NodeTypeCode&(~RX_SCAVENGER_MASK))==RDBSS_NTC_FOBX);
  550. ASSERT(pListEntry->Flink!=NULL);
  551. ASSERT(pListEntry->Blink!=NULL);
  552. RemoveEntryList(pListEntry);
  553. InitializeListHead(pListEntry);
  554. RxReferenceNetFobx(pFobx);
  555. } else {
  556. pFobx = NULL;
  557. }
  558. RxReleaseScavengerMutex();
  559. if (pFobx == NULL) {
  560. break;
  561. }
  562. pFcb = pFobx->SrvOpen->Fcb;
  563. RxPurgeFobx(pFobx);
  564. if (RxAcquireExclusiveFcb(NULL,pFcb) == STATUS_SUCCESS) {
  565. RxReferenceNetFcb(pFcb);
  566. RxDereferenceNetFobx(pFobx,LHS_ExclusiveLockHeld);
  567. if ( !RxDereferenceAndFinalizeNetFcb(pFcb,NULL,FALSE,FALSE) ) {
  568. RxReleaseFcb( NULL, pFcb );
  569. }
  570. } else {
  571. RxLog(("RxPurgeAllFobxs: FCB %lx not accqd.\n",pFcb));
  572. RxWmiLog(LOG,
  573. RxPurgeAllFobxs,
  574. LOGPTR(pFcb));
  575. RxDereferenceNetFobx(pFobx,LHS_LockNotHeld);
  576. }
  577. }
  578. }
  579. PRDBSS_DEVICE_OBJECT
  580. RxGetDeviceObjectOfInstance (
  581. PVOID pInstance
  582. )
  583. /*++
  584. Routine Description:
  585. The routine finds out the device object of an upper structure.
  586. Arguments:
  587. pInstance - the instance
  588. Return Value:
  589. none.
  590. --*/
  591. {
  592. ULONG NodeTypeCode = NodeType(pInstance) & ~RX_SCAVENGER_MASK;
  593. PAGED_CODE();
  594. ASSERT( (NodeTypeCode == RDBSS_NTC_SRVCALL ) ||
  595. (NodeTypeCode == RDBSS_NTC_NETROOT ) ||
  596. (NodeTypeCode == RDBSS_NTC_V_NETROOT ) ||
  597. (NodeTypeCode == RDBSS_NTC_SRVOPEN ) ||
  598. (NodeTypeCode == RDBSS_NTC_FOBX )
  599. );
  600. switch (NodeTypeCode) {
  601. case RDBSS_NTC_SRVCALL:
  602. return((PSRV_CALL)pInstance)->RxDeviceObject;
  603. //no break;
  604. case RDBSS_NTC_NETROOT:
  605. return((PNET_ROOT)pInstance)->SrvCall->RxDeviceObject;
  606. //no break;
  607. case RDBSS_NTC_V_NETROOT:
  608. return((PV_NET_ROOT)pInstance)->NetRoot->SrvCall->RxDeviceObject;
  609. //no break;
  610. case RDBSS_NTC_SRVOPEN:
  611. return((PSRV_OPEN)pInstance)->Fcb->RxDeviceObject;
  612. //no break;
  613. case RDBSS_NTC_FOBX:
  614. return((PFOBX)pInstance)->SrvOpen->Fcb->RxDeviceObject;
  615. //no break;
  616. default:
  617. return(NULL);
  618. }
  619. }
  620. VOID
  621. RxpMarkInstanceForScavengedFinalization(
  622. PVOID pInstance)
  623. /*++
  624. Routine Description:
  625. Thie routine marks a reference counted instance for scavenging
  626. Arguments:
  627. pInstance -- the instance to be marked for finalization by the scavenger
  628. Notes:
  629. Currently scavenging has been implemented for SRV_CALL,NET_ROOT and V_NET_ROOT.
  630. The FCB scavenging is handled separately. The FOBX can and should always be
  631. synchronously finalized. The only data structure that will have to be potentially
  632. enabled for scavenged finalization are SRV_OPEN's.
  633. The Scavenger as it is implemented currently will not consume any system resources
  634. till there is a need for scavenged finalization. The first entry to be marked for
  635. scavenged finalization results in a timer request being posted for the scavenger.
  636. In the current implementation the timer requests are posted as one shot timer requests.
  637. This implies that there are no guarantess as regards the time interval within which
  638. the entries will be finalized. The scavenger activation mechanism is a potential
  639. candidate for fine tuning at a later stage.
  640. On Entry -- Scavenger Mutex must have been accquired
  641. On Exit -- no change in resource ownership.
  642. --*/
  643. {
  644. BOOLEAN PostTimerRequest = FALSE;
  645. PLIST_ENTRY pListHead = NULL;
  646. PLIST_ENTRY pListEntry = NULL;
  647. NODE_TYPE_CODE_AND_SIZE *pNode = (PNODE_TYPE_CODE_AND_SIZE)pInstance;
  648. USHORT InstanceType;
  649. PRDBSS_DEVICE_OBJECT RxDeviceObject = RxGetDeviceObjectOfInstance(pInstance);
  650. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  651. PAGED_CODE();
  652. RxDbgTrace(0,Dbg,("Marking %lx of type %lx for scavenged finalization\n",pInstance,NodeType(pInstance)));
  653. InstanceType = pNode->NodeTypeCode;
  654. if (pNode->NodeReferenceCount <= 1) {
  655. // Mark the entry for scavenging.
  656. pNode->NodeTypeCode |= RX_SCAVENGER_MASK;
  657. RxLog(("Marked for scavenging %lx",pNode));
  658. RxWmiLog(LOG,
  659. RxpMarkInstanceForScavengedFinalization,
  660. LOGPTR(pNode));
  661. switch (InstanceType) {
  662. case RDBSS_NTC_SRVCALL:
  663. {
  664. PSRV_CALL pSrvCall = (PSRV_CALL)pInstance;
  665. pRdbssScavenger->SrvCallsToBeFinalized++;
  666. pListHead = &pRdbssScavenger->SrvCallFinalizationList;
  667. pListEntry = &pSrvCall->ScavengerFinalizationList;
  668. }
  669. break;
  670. case RDBSS_NTC_NETROOT:
  671. {
  672. PNET_ROOT pNetRoot = (PNET_ROOT)pInstance;
  673. pRdbssScavenger->NetRootsToBeFinalized++;
  674. pListHead = &pRdbssScavenger->NetRootFinalizationList;
  675. pListEntry = &pNetRoot->ScavengerFinalizationList;
  676. }
  677. break;
  678. case RDBSS_NTC_V_NETROOT:
  679. {
  680. PV_NET_ROOT pVNetRoot = (PV_NET_ROOT)pInstance;
  681. pRdbssScavenger->VNetRootsToBeFinalized++;
  682. pListHead = &pRdbssScavenger->VNetRootFinalizationList;
  683. pListEntry = &pVNetRoot->ScavengerFinalizationList;
  684. }
  685. break;
  686. case RDBSS_NTC_SRVOPEN:
  687. {
  688. PSRV_OPEN pSrvOpen = (PSRV_OPEN)pInstance;
  689. pRdbssScavenger->SrvOpensToBeFinalized++;
  690. pListHead = &pRdbssScavenger->SrvOpenFinalizationList;
  691. pListEntry = &pSrvOpen->ScavengerFinalizationList;
  692. }
  693. break;
  694. case RDBSS_NTC_FOBX:
  695. {
  696. PFOBX pFobx = (PFOBX)pInstance;
  697. pRdbssScavenger->FobxsToBeFinalized++;
  698. pListHead = &pRdbssScavenger->FobxFinalizationList;
  699. pListEntry = &pFobx->ScavengerFinalizationList;
  700. }
  701. break;
  702. default:
  703. break;
  704. }
  705. InterlockedIncrement(&pNode->NodeReferenceCount);
  706. }
  707. if (pListHead != NULL) {
  708. InsertTailList(pListHead,pListEntry);
  709. if (pRdbssScavenger->State == RDBSS_SCAVENGER_INACTIVE) {
  710. pRdbssScavenger->State = RDBSS_SCAVENGER_DORMANT;
  711. PostTimerRequest = TRUE;
  712. } else {
  713. PostTimerRequest = FALSE;
  714. }
  715. }
  716. if (PostTimerRequest) {
  717. LARGE_INTEGER TimeInterval;
  718. // Post a one shot timer request for scheduling the scavenger after a
  719. // predetermined amount of time.
  720. TimeInterval.QuadPart = RX_SCAVENGER_FINALIZATION_TIME_INTERVAL;
  721. RxPostOneShotTimerRequest(
  722. RxFileSystemDeviceObject,
  723. &pRdbssScavenger->WorkItem,
  724. RxScavengerTimerRoutine,
  725. RxDeviceObject,
  726. TimeInterval);
  727. }
  728. }
  729. VOID
  730. RxpUndoScavengerFinalizationMarking(
  731. PVOID pInstance)
  732. /*++
  733. Routine Description:
  734. This routine undoes the marking for scavenged finalization
  735. Arguments:
  736. pInstance -- the instance to be unmarked
  737. Notes:
  738. This routine is typically invoked when a reference is made to an entry that has
  739. been marked for scavenging. Since the scavenger routine that does the finalization
  740. needs to acquire the exclusive lock this routine should be called with the
  741. appropriate lock held in a shared mode atleast. This routine removes it from the list
  742. of entries marked for scavenging and rips off the scavenger mask from the node type.
  743. --*/
  744. {
  745. PLIST_ENTRY pListEntry;
  746. PNODE_TYPE_CODE_AND_SIZE pNode = (PNODE_TYPE_CODE_AND_SIZE)pInstance;
  747. PRDBSS_DEVICE_OBJECT RxDeviceObject = RxGetDeviceObjectOfInstance(pInstance);
  748. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  749. PAGED_CODE();
  750. RxDbgTrace(0,Dbg,("SCAVENGER -- undoing the marking for %lx of type %lx\n",pNode,pNode->NodeTypeCode));
  751. if (pNode->NodeTypeCode & RX_SCAVENGER_MASK) {
  752. pNode->NodeTypeCode &= ~RX_SCAVENGER_MASK;
  753. switch (pNode->NodeTypeCode) {
  754. case RDBSS_NTC_SRVCALL:
  755. {
  756. PSRV_CALL pSrvCall = (PSRV_CALL)pInstance;
  757. pRdbssScavenger->SrvCallsToBeFinalized--;
  758. pListEntry = &pSrvCall->ScavengerFinalizationList;
  759. }
  760. break;
  761. case RDBSS_NTC_NETROOT:
  762. {
  763. PNET_ROOT pNetRoot = (PNET_ROOT)pInstance;
  764. pRdbssScavenger->NetRootsToBeFinalized--;
  765. pListEntry = &pNetRoot->ScavengerFinalizationList;
  766. }
  767. break;
  768. case RDBSS_NTC_V_NETROOT:
  769. {
  770. PV_NET_ROOT pVNetRoot = (PV_NET_ROOT)pInstance;
  771. pRdbssScavenger->VNetRootsToBeFinalized--;
  772. pListEntry = &pVNetRoot->ScavengerFinalizationList;
  773. }
  774. break;
  775. case RDBSS_NTC_SRVOPEN:
  776. {
  777. PSRV_OPEN pSrvOpen = (PSRV_OPEN)pInstance;
  778. pRdbssScavenger->SrvOpensToBeFinalized--;
  779. pListEntry = &pSrvOpen->ScavengerFinalizationList;
  780. }
  781. break;
  782. case RDBSS_NTC_FOBX:
  783. {
  784. PFOBX pFobx = (PFOBX)pInstance;
  785. pRdbssScavenger->FobxsToBeFinalized--;
  786. pListEntry = &pFobx->ScavengerFinalizationList;
  787. }
  788. break;
  789. default:
  790. return;
  791. }
  792. RemoveEntryList(pListEntry);
  793. InitializeListHead(pListEntry);
  794. InterlockedDecrement(&pNode->NodeReferenceCount);
  795. }
  796. }
  797. VOID
  798. RxUndoScavengerFinalizationMarking(
  799. PVOID pInstance)
  800. /*++
  801. Routine Description:
  802. This routine undoes the marking for scavenged finalization
  803. Arguments:
  804. pInstance -- the instance to be unmarked
  805. --*/
  806. {
  807. RxAcquireScavengerMutex();
  808. RxpUndoScavengerFinalizationMarking(pInstance);
  809. RxReleaseScavengerMutex();
  810. }
  811. BOOLEAN
  812. RxScavengeRelatedFobxs(PFCB pFcb)
  813. /*++
  814. Routine Description:
  815. Thie routine scavenges all the file objects pertaining to the given FCB.
  816. Notes:
  817. On Entry -- FCB must have been accquired exclusive.
  818. On Exit -- no change in resource acquistion.
  819. --*/
  820. {
  821. BOOLEAN ScavengerMutexAcquired = FALSE;
  822. BOOLEAN AtleastOneFobxScavenged = FALSE;
  823. PRDBSS_DEVICE_OBJECT RxDeviceObject = pFcb->RxDeviceObject;
  824. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  825. PAGED_CODE();
  826. RxAcquireScavengerMutex();
  827. ScavengerMutexAcquired = TRUE;
  828. if (pRdbssScavenger->FobxsToBeFinalized > 0) {
  829. PLIST_ENTRY pEntry;
  830. PFOBX pFobx;
  831. LIST_ENTRY FobxList;
  832. InitializeListHead(&FobxList);
  833. pEntry = pRdbssScavenger->FobxFinalizationList.Flink;
  834. while (pEntry != &pRdbssScavenger->FobxFinalizationList) {
  835. pFobx = (PFOBX)CONTAINING_RECORD(
  836. pEntry,
  837. FOBX,
  838. ScavengerFinalizationList);
  839. pEntry = pEntry->Flink;
  840. if (pFobx->SrvOpen != NULL &&
  841. pFobx->SrvOpen->Fcb == pFcb) {
  842. RxpUndoScavengerFinalizationMarking(pFobx);
  843. ASSERT(NodeType(pFobx) == RDBSS_NTC_FOBX);
  844. InsertTailList(&FobxList,&pFobx->ScavengerFinalizationList);
  845. }
  846. }
  847. ScavengerMutexAcquired = FALSE;
  848. RxReleaseScavengerMutex();
  849. AtleastOneFobxScavenged = !IsListEmpty(&FobxList);
  850. pEntry = FobxList.Flink;
  851. while (!IsListEmpty(&FobxList)) {
  852. pEntry = FobxList.Flink;
  853. RemoveEntryList(pEntry);
  854. pFobx = (PFOBX)CONTAINING_RECORD(
  855. pEntry,
  856. FOBX,
  857. ScavengerFinalizationList);
  858. RxFinalizeNetFobx(pFobx,TRUE,TRUE);
  859. }
  860. }
  861. if (ScavengerMutexAcquired) {
  862. RxReleaseScavengerMutex();
  863. }
  864. return AtleastOneFobxScavenged;
  865. }
  866. VOID
  867. RxpScavengeFobxs(
  868. PRDBSS_SCAVENGER pRdbssScavenger,
  869. PLIST_ENTRY pFobxList)
  870. /*++
  871. Routine Description:
  872. Thie routine scavenges all the file objects in the given list. This routine
  873. does the actual work of scavenging while RxScavengeFobxsForNetRoot and
  874. RxScavengeAllFobxs gather the file object extensions and call this routine
  875. Notes:
  876. --*/
  877. {
  878. while (!IsListEmpty(pFobxList)) {
  879. PFCB pFcb;
  880. PFOBX pFobx;
  881. NTSTATUS Status;
  882. PLIST_ENTRY pEntry;
  883. pEntry = pFobxList->Flink;
  884. RemoveEntryList(pEntry);
  885. pFobx = (PFOBX)CONTAINING_RECORD(
  886. pEntry,
  887. FOBX,
  888. ScavengerFinalizationList);
  889. pFcb = (PFCB)pFobx->pSrvOpen->pFcb;
  890. Status = RxAcquireExclusiveFcb( NULL, pFcb );
  891. if (Status == (STATUS_SUCCESS)) {
  892. BOOLEAN ReleaseFcb;
  893. RxReferenceNetFcb(pFcb);
  894. RxDereferenceNetFobx(pFobx,LHS_ExclusiveLockHeld);
  895. ReleaseFcb = !RxDereferenceAndFinalizeNetFcb(pFcb,NULL,FALSE,FALSE);
  896. if (ReleaseFcb) {
  897. RxReleaseFcb( NULL, pFcb );
  898. }
  899. } else {
  900. RxDereferenceNetFobx(pFobx,LHS_LockNotHeld);
  901. }
  902. }
  903. }
  904. VOID
  905. RxScavengeFobxsForNetRoot(
  906. PNET_ROOT pNetRoot,
  907. PFCB PurgingFcb)
  908. /*++
  909. Routine Description:
  910. Thie routine scavenges all the file objects pertaining to the given net root
  911. instance
  912. Notes:
  913. --*/
  914. {
  915. BOOLEAN ScavengerMutexAcquired = FALSE;
  916. PRDBSS_DEVICE_OBJECT RxDeviceObject = pNetRoot->pSrvCall->RxDeviceObject;
  917. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  918. PAGED_CODE();
  919. RxAcquireScavengerMutex();
  920. ScavengerMutexAcquired = TRUE;
  921. if (pRdbssScavenger->FobxsToBeFinalized > 0) {
  922. PLIST_ENTRY pEntry;
  923. PFOBX pFobx;
  924. LIST_ENTRY FobxList;
  925. InitializeListHead(&FobxList);
  926. pEntry = pRdbssScavenger->FobxFinalizationList.Flink;
  927. while (pEntry != &pRdbssScavenger->FobxFinalizationList) {
  928. pFobx = (PFOBX)CONTAINING_RECORD(
  929. pEntry,
  930. FOBX,
  931. ScavengerFinalizationList);
  932. pEntry = pEntry->Flink;
  933. if (pFobx->SrvOpen != NULL &&
  934. pFobx->SrvOpen->Fcb->pNetRoot == (PMRX_NET_ROOT)pNetRoot) {
  935. NTSTATUS PurgeStatus = STATUS_MORE_PROCESSING_REQUIRED;
  936. if ((PurgingFcb != NULL) &&
  937. (pFobx->SrvOpen->Fcb != PurgingFcb)) {
  938. MINIRDR_CALL_THROUGH(
  939. PurgeStatus,
  940. RxDeviceObject->Dispatch,
  941. MRxAreFilesAliased,(pFobx->SrvOpen->Fcb,PurgingFcb)
  942. );
  943. }
  944. if (PurgeStatus != STATUS_SUCCESS) {
  945. RxReferenceNetFobx(pFobx);
  946. ASSERT(NodeType(pFobx) == RDBSS_NTC_FOBX);
  947. InsertTailList(&FobxList,&pFobx->ScavengerFinalizationList);
  948. }
  949. }
  950. }
  951. ScavengerMutexAcquired = FALSE;
  952. RxReleaseScavengerMutex();
  953. RxpScavengeFobxs(pRdbssScavenger,&FobxList);
  954. }
  955. if (ScavengerMutexAcquired) {
  956. RxReleaseScavengerMutex();
  957. }
  958. return;
  959. }
  960. VOID
  961. RxScavengeAllFobxs(
  962. PRDBSS_DEVICE_OBJECT RxDeviceObject)
  963. /*++
  964. Routine Description:
  965. Thie routine scavenges all the file objects pertaining to the given mini
  966. redirector device object
  967. Notes:
  968. --*/
  969. {
  970. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  971. PAGED_CODE();
  972. if (pRdbssScavenger->FobxsToBeFinalized > 0) {
  973. PLIST_ENTRY pEntry;
  974. PFOBX pFobx;
  975. LIST_ENTRY FobxList;
  976. InitializeListHead(&FobxList);
  977. RxAcquireScavengerMutex();
  978. pEntry = pRdbssScavenger->FobxFinalizationList.Flink;
  979. while (pEntry != &pRdbssScavenger->FobxFinalizationList) {
  980. pFobx = (PFOBX)CONTAINING_RECORD(
  981. pEntry,
  982. FOBX,
  983. ScavengerFinalizationList);
  984. pEntry = pEntry->Flink;
  985. RxReferenceNetFobx(pFobx);
  986. ASSERT(NodeType(pFobx) == RDBSS_NTC_FOBX);
  987. InsertTailList(&FobxList,&pFobx->ScavengerFinalizationList);
  988. }
  989. RxReleaseScavengerMutex();
  990. RxpScavengeFobxs(pRdbssScavenger,&FobxList);
  991. }
  992. }
  993. BOOLEAN
  994. RxScavengeVNetRoots(
  995. PRDBSS_DEVICE_OBJECT RxDeviceObject)
  996. {
  997. BOOLEAN AtleastOneEntryScavenged = FALSE;
  998. PRX_PREFIX_TABLE pRxNetNameTable = RxDeviceObject->pRxNetNameTable;
  999. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  1000. PV_NET_ROOT pVNetRoot;
  1001. PAGED_CODE();
  1002. do {
  1003. PVOID pEntry;
  1004. RxDbgTrace(0,Dbg,("RDBSS SCAVENGER -- Scavenging VNetRoots\n"));
  1005. RxAcquirePrefixTableLockExclusive(pRxNetNameTable,TRUE);
  1006. RxAcquireScavengerMutex();
  1007. if (pRdbssScavenger->VNetRootsToBeFinalized > 0) {
  1008. pEntry = RemoveHeadList(&pRdbssScavenger->VNetRootFinalizationList);
  1009. pVNetRoot = (PV_NET_ROOT)
  1010. CONTAINING_RECORD(
  1011. pEntry,
  1012. V_NET_ROOT,
  1013. ScavengerFinalizationList);
  1014. RxpUndoScavengerFinalizationMarking(pVNetRoot);
  1015. ASSERT(NodeType(pVNetRoot) == RDBSS_NTC_V_NETROOT);
  1016. } else {
  1017. pVNetRoot = NULL;
  1018. }
  1019. RxReleaseScavengerMutex();
  1020. if (pVNetRoot != NULL) {
  1021. RxFinalizeVNetRoot(pVNetRoot,TRUE,TRUE);
  1022. AtleastOneEntryScavenged = TRUE;
  1023. }
  1024. RxReleasePrefixTableLock(pRxNetNameTable);
  1025. } while (pVNetRoot != NULL);
  1026. return AtleastOneEntryScavenged;
  1027. }
  1028. VOID
  1029. RxScavengerFinalizeEntries (
  1030. PRDBSS_DEVICE_OBJECT RxDeviceObject
  1031. )
  1032. /*++
  1033. Routine Description:
  1034. Thie routine initiates the delayed finalization of entries
  1035. Notes:
  1036. This routine must always be called only after acquiring the Scavenger Mutex.
  1037. On return from this routine it needs to be reacquired. This is required
  1038. to avoid redundant copying of data structures.
  1039. The performance metric for the scavenger is different from the other routines. In
  1040. the other routines the goal is to do as much work as possible once the lock is
  1041. acquired without having to release it. On the other hand for the scavenger the
  1042. goal is to hold the lock for as short a duration as possible because this
  1043. interferes with the regular activity. This is preferred even if it entails
  1044. frequent relaesing/acquisition of locks since it enables higher degrees of
  1045. concurrency.
  1046. --*/
  1047. {
  1048. BOOLEAN AtleastOneEntryScavenged;
  1049. PRX_PREFIX_TABLE pRxNetNameTable = RxDeviceObject->pRxNetNameTable;
  1050. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  1051. PAGED_CODE();
  1052. do {
  1053. AtleastOneEntryScavenged = FALSE;
  1054. RxAcquireScavengerMutex();
  1055. if (pRdbssScavenger->NumberOfDormantFiles > 0) {
  1056. PLIST_ENTRY pListEntry;
  1057. PFOBX pFobxToBePurged;
  1058. // If the number of dormant files exceeds the limit specified for the
  1059. // given server a currently dormant file needs to be picked up for
  1060. // purging.
  1061. pListEntry = pRdbssScavenger->ClosePendingFobxsList.Flink;
  1062. if (pListEntry != &pRdbssScavenger->ClosePendingFobxsList) {
  1063. pFobxToBePurged = (PFOBX)(CONTAINING_RECORD(
  1064. pListEntry,
  1065. FOBX,
  1066. ClosePendingList));
  1067. RemoveEntryList(&pFobxToBePurged->ClosePendingList);
  1068. InitializeListHead(&pFobxToBePurged->ClosePendingList);
  1069. RxReferenceNetFobx(pFobxToBePurged);
  1070. pRdbssScavenger->CurrentScavengerThread = PsGetCurrentThread();
  1071. pRdbssScavenger->CurrentFcbForClosePendingProcessing =
  1072. (PFCB)(pFobxToBePurged->SrvOpen->Fcb);
  1073. pRdbssScavenger->CurrentNetRootForClosePendingProcessing =
  1074. (PNET_ROOT)(pFobxToBePurged->SrvOpen->Fcb->pNetRoot);
  1075. KeResetEvent(
  1076. &(pRdbssScavenger->ClosePendingProcessingSyncEvent));
  1077. } else {
  1078. pFobxToBePurged = NULL;
  1079. }
  1080. if (pFobxToBePurged != NULL) {
  1081. NTSTATUS Status;
  1082. RxReleaseScavengerMutex();
  1083. Status = RxPurgeFobxFromCache(pFobxToBePurged);
  1084. AtleastOneEntryScavenged = (Status == STATUS_SUCCESS);
  1085. RxAcquireScavengerMutex();
  1086. pRdbssScavenger->CurrentScavengerThread = NULL;
  1087. pRdbssScavenger->CurrentFcbForClosePendingProcessing = NULL;
  1088. pRdbssScavenger->CurrentNetRootForClosePendingProcessing = NULL;
  1089. KeSetEvent(
  1090. &(pRdbssScavenger->ClosePendingProcessingSyncEvent),
  1091. 0,
  1092. FALSE);
  1093. }
  1094. }
  1095. if (pRdbssScavenger->FobxsToBeFinalized > 0) {
  1096. PVOID pEntry;
  1097. PFOBX pFobx = NULL;
  1098. PFCB pFcb = NULL;
  1099. RxDbgTrace(0,Dbg,("RDBSS SCAVENGER -- Scavenging Fobxs\n"));
  1100. if (pRdbssScavenger->FobxsToBeFinalized > 0) {
  1101. pEntry = pRdbssScavenger->FobxFinalizationList.Flink;
  1102. pFobx = (PFOBX)
  1103. CONTAINING_RECORD(
  1104. pEntry,
  1105. FOBX,
  1106. ScavengerFinalizationList);
  1107. pFcb = pFobx->SrvOpen->Fcb;
  1108. RxReferenceNetFcb(pFcb);
  1109. pRdbssScavenger->CurrentScavengerThread = PsGetCurrentThread();
  1110. pRdbssScavenger->CurrentFcbForClosePendingProcessing =
  1111. (PFCB)(pFobx->SrvOpen->Fcb);
  1112. pRdbssScavenger->CurrentNetRootForClosePendingProcessing =
  1113. (PNET_ROOT)(pFobx->SrvOpen->Fcb->pNetRoot);
  1114. KeResetEvent(
  1115. &(pRdbssScavenger->ClosePendingProcessingSyncEvent));
  1116. } else {
  1117. pFobx = NULL;
  1118. }
  1119. if (pFobx != NULL) {
  1120. NTSTATUS Status;
  1121. RxReleaseScavengerMutex();
  1122. Status = RxAcquireExclusiveFcb( NULL, pFcb );
  1123. if (Status == (STATUS_SUCCESS)) {
  1124. BOOLEAN ReleaseFcb;
  1125. AtleastOneEntryScavenged = RxScavengeRelatedFobxs(pFcb);
  1126. ReleaseFcb = !RxDereferenceAndFinalizeNetFcb(pFcb,NULL,FALSE,FALSE);
  1127. if (ReleaseFcb) {
  1128. RxReleaseFcb( NULL, pFcb );
  1129. }
  1130. } else {
  1131. RxLog(("Delayed Close Failure FOBX(%lx) FCB(%lx)\n",pFobx,pFcb));
  1132. RxWmiLog(LOG,
  1133. RxScavengerFinalizeEntries,
  1134. LOGPTR(pFobx)
  1135. LOGPTR(pFcb));
  1136. }
  1137. RxAcquireScavengerMutex();
  1138. pRdbssScavenger->CurrentScavengerThread = NULL;
  1139. pRdbssScavenger->CurrentFcbForClosePendingProcessing = NULL;
  1140. pRdbssScavenger->CurrentNetRootForClosePendingProcessing = NULL;
  1141. KeSetEvent(
  1142. &(pRdbssScavenger->ClosePendingProcessingSyncEvent),
  1143. 0,
  1144. FALSE);
  1145. }
  1146. }
  1147. RxReleaseScavengerMutex();
  1148. if (pRdbssScavenger->SrvOpensToBeFinalized > 0) {
  1149. //SRV_OPEN List should not be empty, potential memory leak
  1150. ASSERT(pRdbssScavenger->SrvOpensToBeFinalized == 0);
  1151. }
  1152. if (pRdbssScavenger->FcbsToBeFinalized > 0) {
  1153. //FCB list should be empty , potential memory leak
  1154. ASSERT(pRdbssScavenger->FcbsToBeFinalized == 0);
  1155. }
  1156. if (pRdbssScavenger->VNetRootsToBeFinalized > 0) {
  1157. PVOID pEntry;
  1158. PV_NET_ROOT pVNetRoot;
  1159. RxDbgTrace(0,Dbg,("RDBSS SCAVENGER -- Scavenging VNetRoots\n"));
  1160. RxAcquirePrefixTableLockExclusive(pRxNetNameTable,TRUE);
  1161. RxAcquireScavengerMutex();
  1162. if (pRdbssScavenger->VNetRootsToBeFinalized > 0) {
  1163. pEntry = RemoveHeadList(&pRdbssScavenger->VNetRootFinalizationList);
  1164. pVNetRoot = (PV_NET_ROOT)
  1165. CONTAINING_RECORD(
  1166. pEntry,
  1167. V_NET_ROOT,
  1168. ScavengerFinalizationList);
  1169. RxpUndoScavengerFinalizationMarking(pVNetRoot);
  1170. ASSERT(NodeType(pVNetRoot) == RDBSS_NTC_V_NETROOT);
  1171. } else {
  1172. pVNetRoot = NULL;
  1173. }
  1174. RxReleaseScavengerMutex();
  1175. if (pVNetRoot != NULL) {
  1176. RxFinalizeVNetRoot(pVNetRoot,TRUE,TRUE);
  1177. AtleastOneEntryScavenged = TRUE;
  1178. }
  1179. RxReleasePrefixTableLock(pRxNetNameTable);
  1180. }
  1181. if (pRdbssScavenger->NetRootsToBeFinalized > 0) {
  1182. PVOID pEntry;
  1183. PNET_ROOT pNetRoot;
  1184. RxDbgTrace(0,Dbg,("RDBSS SCAVENGER -- Scavenging NetRoots\n"));
  1185. RxAcquirePrefixTableLockExclusive(pRxNetNameTable,TRUE);
  1186. RxAcquireScavengerMutex();
  1187. if (pRdbssScavenger->NetRootsToBeFinalized > 0) {
  1188. pEntry = RemoveHeadList(&pRdbssScavenger->NetRootFinalizationList);
  1189. pNetRoot = (PNET_ROOT)
  1190. CONTAINING_RECORD(
  1191. pEntry,
  1192. NET_ROOT,
  1193. ScavengerFinalizationList);
  1194. RxpUndoScavengerFinalizationMarking(pNetRoot);
  1195. ASSERT(NodeType(pNetRoot) == RDBSS_NTC_NETROOT);
  1196. } else {
  1197. pNetRoot = NULL;
  1198. }
  1199. RxReleaseScavengerMutex();
  1200. if (pNetRoot != NULL) {
  1201. RxFinalizeNetRoot(pNetRoot,TRUE,TRUE);
  1202. AtleastOneEntryScavenged = TRUE;
  1203. }
  1204. RxReleasePrefixTableLock(pRxNetNameTable);
  1205. }
  1206. if (pRdbssScavenger->SrvCallsToBeFinalized > 0) {
  1207. PVOID pEntry;
  1208. PSRV_CALL pSrvCall;
  1209. RxDbgTrace(0,Dbg,("RDBSS SCAVENGER -- Scavenging SrvCalls\n"));
  1210. RxAcquirePrefixTableLockExclusive(pRxNetNameTable,TRUE);
  1211. RxAcquireScavengerMutex();
  1212. if (pRdbssScavenger->SrvCallsToBeFinalized > 0) {
  1213. pEntry = RemoveHeadList(&pRdbssScavenger->SrvCallFinalizationList);
  1214. pSrvCall = (PSRV_CALL)
  1215. CONTAINING_RECORD(
  1216. pEntry,
  1217. SRV_CALL,
  1218. ScavengerFinalizationList);
  1219. RxpUndoScavengerFinalizationMarking(pSrvCall);
  1220. ASSERT(NodeType(pSrvCall) == RDBSS_NTC_SRVCALL);
  1221. } else {
  1222. pSrvCall = NULL;
  1223. }
  1224. RxReleaseScavengerMutex();
  1225. if (pSrvCall != NULL) {
  1226. RxFinalizeSrvCall(pSrvCall,TRUE,TRUE);
  1227. AtleastOneEntryScavenged = TRUE;
  1228. }
  1229. RxReleasePrefixTableLock(pRxNetNameTable);
  1230. }
  1231. } while (AtleastOneEntryScavenged);
  1232. }
  1233. VOID
  1234. RxScavengerTimerRoutine(
  1235. PVOID pContext)
  1236. /*++
  1237. Routine Description:
  1238. This routine is the timer routine that periodically initiates the finalization of
  1239. entries.
  1240. Arguments:
  1241. pContext -- the context (actually the RxDeviceObject)
  1242. Notes:
  1243. --*/
  1244. {
  1245. PRDBSS_DEVICE_OBJECT RxDeviceObject = (PRDBSS_DEVICE_OBJECT)pContext;
  1246. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  1247. BOOLEAN PostTimerRequest = FALSE;
  1248. PAGED_CODE();
  1249. RxAcquireScavengerMutex();
  1250. KeResetEvent(&pRdbssScavenger->SyncEvent);
  1251. if (pRdbssScavenger->State == RDBSS_SCAVENGER_DORMANT) {
  1252. pRdbssScavenger->State = RDBSS_SCAVENGER_ACTIVE;
  1253. RxReleaseScavengerMutex();
  1254. RxScavengerFinalizeEntries(RxDeviceObject);
  1255. RxAcquireScavengerMutex();
  1256. if (pRdbssScavenger->State == RDBSS_SCAVENGER_ACTIVE) {
  1257. ULONG EntriesToBeFinalized;
  1258. // Check if the scavenger needs to be activated again.
  1259. EntriesToBeFinalized = pRdbssScavenger->SrvCallsToBeFinalized +
  1260. pRdbssScavenger->NetRootsToBeFinalized +
  1261. pRdbssScavenger->VNetRootsToBeFinalized +
  1262. pRdbssScavenger->FcbsToBeFinalized +
  1263. pRdbssScavenger->SrvOpensToBeFinalized +
  1264. pRdbssScavenger->FobxsToBeFinalized +
  1265. pRdbssScavenger->NumberOfDormantFiles;
  1266. if (EntriesToBeFinalized > 0) {
  1267. PostTimerRequest = TRUE;
  1268. pRdbssScavenger->State = RDBSS_SCAVENGER_DORMANT;
  1269. } else {
  1270. pRdbssScavenger->State = RDBSS_SCAVENGER_INACTIVE;
  1271. }
  1272. }
  1273. RxReleaseScavengerMutex();
  1274. if (PostTimerRequest) {
  1275. LARGE_INTEGER TimeInterval;
  1276. TimeInterval.QuadPart = RX_SCAVENGER_FINALIZATION_TIME_INTERVAL;
  1277. RxPostOneShotTimerRequest(
  1278. RxFileSystemDeviceObject,
  1279. &pRdbssScavenger->WorkItem,
  1280. RxScavengerTimerRoutine,
  1281. RxDeviceObject,
  1282. TimeInterval);
  1283. }
  1284. } else {
  1285. RxReleaseScavengerMutex();
  1286. }
  1287. // Trigger the event.
  1288. KeSetEvent(&pRdbssScavenger->SyncEvent,0,FALSE);
  1289. }
  1290. VOID
  1291. RxTerminateScavenging(
  1292. PRX_CONTEXT pRxContext)
  1293. /*++
  1294. Routine Description:
  1295. This routine terminates all scavenging activities in the RDBSS. The termination is
  1296. effected in an orderly fashion after all the entries currently marked for scavenging
  1297. are finalized.
  1298. Arguments:
  1299. pRxContext -- the context
  1300. Notes:
  1301. In implementing the scavenger there are two options. The scavenger can latch onto a
  1302. thread and hold onto it from the moment RDBSS comes into existence till it is unloaded.
  1303. Such an implementation renders a thread useless throughout the lifetime of RDBSS.
  1304. If this is to be avoided the alternative is to trigger the scavenger as and when
  1305. required. While this technique addresses the conservation of system resources it
  1306. poses some tricky synchronization problems having to do with start/stop of the RDR.
  1307. Since the RDR can be started/stopped at any time the Scavenger can be in one of three
  1308. states when the RDR is stopped.
  1309. 1) Scavenger is active.
  1310. 2) Scavenger request is in the timer queue.
  1311. 3) Scavenger is inactive.
  1312. Of these case (3) is the easiest since no special action is required.
  1313. If the scavenger is active then this routine has to synchronize with the timer routine
  1314. that is finzalizing the entries. This is achieved by allowing the termination routine to
  1315. modify the Scavenger state under a mutex and waiting for it to signal the event on
  1316. completion.
  1317. --*/
  1318. {
  1319. NTSTATUS Status = STATUS_SUCCESS;
  1320. PRDBSS_DEVICE_OBJECT RxDeviceObject = pRxContext->RxDeviceObject;
  1321. PRDBSS_SCAVENGER pRdbssScavenger = RxDeviceObject->pRdbssScavenger;
  1322. RDBSS_SCAVENGER_STATE PreviousState;
  1323. PAGED_CODE();
  1324. RxAcquireScavengerMutex();
  1325. PreviousState = pRdbssScavenger->State;
  1326. pRdbssScavenger->State = RDBSS_SCAVENGER_SUSPENDED;
  1327. RxReleaseScavengerMutex();
  1328. if (PreviousState == RDBSS_SCAVENGER_DORMANT) {
  1329. // There is a timer request currently active. cancel it
  1330. Status = RxCancelTimerRequest(
  1331. RxFileSystemDeviceObject,
  1332. RxScavengerTimerRoutine,
  1333. RxDeviceObject);
  1334. }
  1335. if ((PreviousState == RDBSS_SCAVENGER_ACTIVE) ||
  1336. (Status == STATUS_NOT_FOUND)) {
  1337. // Either the timer routine was previously active or it could not be cancelled
  1338. // Wait for it to complete
  1339. KeWaitForSingleObject(&pRdbssScavenger->SyncEvent,Executive,KernelMode,FALSE,NULL);
  1340. }
  1341. RxPurgeAllFobxs(pRxContext->RxDeviceObject);
  1342. RxScavengerFinalizeEntries(pRxContext->RxDeviceObject);
  1343. }