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.

982 lines
21 KiB

  1. /*++
  2. Copyright (c) 1995-1996 Microsoft Corporation
  3. Module Name :
  4. dirmon.cxx
  5. Abstract:
  6. This module includes definitions of functions and variables
  7. for CDirMonitor and CDirMonitorEntry object
  8. Author:
  9. Charles Grant ( cgrant ) April-1997
  10. Revision History:
  11. --*/
  12. /************************************************************
  13. * Include Headers
  14. ************************************************************/
  15. #include <iis.h>
  16. #include "dbgutil.h"
  17. #include "dirmon.h"
  18. //
  19. // CDirMonitorEntry
  20. //
  21. #define DEFAULT_BUFFER_SIZE 512
  22. VOID
  23. WINAPI
  24. DirMonOverlappedCompletionRoutine(
  25. DWORD dwErrorCode,
  26. DWORD dwNumberOfBytesTransfered,
  27. LPOVERLAPPED lpOverlapped
  28. )
  29. {
  30. CDirMonitor::OverlappedCompletionRoutine( dwErrorCode,
  31. dwNumberOfBytesTransfered,
  32. lpOverlapped );
  33. }
  34. VOID
  35. CDirMonitor::OverlappedCompletionRoutine(
  36. DWORD dwErrorCode,
  37. DWORD dwNumberOfBytesTransfered,
  38. LPOVERLAPPED lpOverlapped
  39. )
  40. {
  41. PVOID pvContext;
  42. pvContext = CONTAINING_RECORD( lpOverlapped,
  43. CDirMonitorEntry,
  44. m_ovr );
  45. DBG_ASSERT( pvContext != NULL );
  46. CDirMonitor::DirMonitorCompletionFunction( pvContext,
  47. dwNumberOfBytesTransfered,
  48. dwErrorCode,
  49. lpOverlapped );
  50. }
  51. CDirMonitorEntry::CDirMonitorEntry() :
  52. m_cDirRefCount(0),
  53. m_cIORefCount(0),
  54. m_hDir(INVALID_HANDLE_VALUE),
  55. m_dwNotificationFlags(0),
  56. m_pszPath(NULL),
  57. m_cPathLength(0),
  58. m_pDirMonitor(NULL),
  59. m_cBufferSize(0),
  60. m_pbBuffer(NULL),
  61. m_fInCleanup(FALSE),
  62. m_fWatchSubdirectories(FALSE)
  63. /*++
  64. Routine Description:
  65. CDirMonitorEntry constructor
  66. Arguments:
  67. None
  68. Return Value:
  69. Nothing
  70. --*/
  71. {
  72. }
  73. CDirMonitorEntry::~CDirMonitorEntry(
  74. VOID
  75. )
  76. /*++
  77. Routine Description:
  78. CDirMonitorEntry destructor
  79. Arguments:
  80. None
  81. Return Value:
  82. Nothing
  83. --*/
  84. {
  85. HANDLE hDir;
  86. IF_DEBUG( DIRMON )
  87. {
  88. DBGPRINTF((DBG_CONTEXT, "[CDirMonitorEntry] Destructor\n"));
  89. }
  90. // We should only be destroyed when
  91. // our ref counts have gone to 0
  92. DBG_ASSERT(m_cDirRefCount == 0);
  93. DBG_ASSERT(m_cIORefCount == 0);
  94. //
  95. // We really ought to have closed the handle by now
  96. //
  97. if (m_hDir != INVALID_HANDLE_VALUE) {
  98. DBGPRINTF(( DBG_CONTEXT, "~CDirMonitorEntry: open handle %p\n",
  99. m_hDir ));
  100. hDir = m_hDir;
  101. m_hDir = INVALID_HANDLE_VALUE;
  102. CloseHandle( hDir );
  103. }
  104. if (m_pDirMonitor != NULL)
  105. {
  106. m_pDirMonitor->RemoveEntry(this);
  107. m_pDirMonitor = NULL;
  108. }
  109. m_cPathLength = 0;
  110. if ( m_pszPath != NULL )
  111. {
  112. LocalFree( m_pszPath );
  113. m_pszPath = NULL;
  114. }
  115. if (m_pbBuffer != NULL)
  116. {
  117. LocalFree(m_pbBuffer);
  118. m_cBufferSize = 0;
  119. }
  120. }
  121. BOOL
  122. CDirMonitorEntry::Init(
  123. DWORD cBufferSize = DEFAULT_BUFFER_SIZE
  124. )
  125. /*++
  126. Routine Description:
  127. Initialize the dir montior entry.
  128. Arguments:
  129. cBufferSize - Initial size of buffer used to store change notifications
  130. Return Value:
  131. TRUE if success, otherwise FALSE
  132. --*/
  133. {
  134. // Don't allow a 0 length buffer
  135. if (cBufferSize == 0)
  136. {
  137. return FALSE;
  138. }
  139. DBG_ASSERT( m_pbBuffer == NULL );
  140. m_pbBuffer = (BYTE *) LocalAlloc( LPTR, cBufferSize );
  141. if (m_pbBuffer != NULL)
  142. {
  143. m_cBufferSize = cBufferSize;
  144. return TRUE;
  145. }
  146. else
  147. {
  148. // Unable to allocate buffer
  149. return FALSE;
  150. }
  151. }
  152. BOOL
  153. CDirMonitorEntry::RequestNotification(
  154. VOID
  155. )
  156. /*++
  157. Routine Description:
  158. Request ATQ to monitor directory changes for the directory handle
  159. associated with this entry
  160. Arguments:
  161. None
  162. Return Value:
  163. TRUE if success, otherwise FALSE
  164. --*/
  165. {
  166. BOOL fResult = FALSE;
  167. DWORD cbRead = 0;
  168. IF_DEBUG( DIRMON )
  169. {
  170. DBGPRINTF((DBG_CONTEXT, "[CDirMonitorEntry] Request change notification\n"));
  171. }
  172. DBG_ASSERT(m_pDirMonitor);
  173. // Reset the overlapped io structure
  174. memset(&m_ovr, 0, sizeof(m_ovr));
  175. // Increase the ref count in advance
  176. IOAddRef();
  177. // Request notification of directory changes
  178. fResult = ReadDirectoryChangesW( m_hDir,
  179. m_pbBuffer,
  180. m_cBufferSize,
  181. m_fWatchSubdirectories,
  182. m_dwNotificationFlags,
  183. &cbRead,
  184. &m_ovr,
  185. NULL );
  186. if (!fResult)
  187. {
  188. // ReadDirChanges failed so
  189. // release the ref count we did in advance
  190. // Might cause IO ref count to go to 0
  191. IORelease();
  192. }
  193. return fResult;
  194. }
  195. BOOL
  196. CDirMonitorEntry::Cleanup(
  197. VOID
  198. )
  199. /*++
  200. Routine Description:
  201. Cleans up resource and determines if the caller need to delete
  202. the Directory Monitor Entry instance.
  203. Arguments:
  204. None
  205. Return Value:
  206. TRUE if the caller is responsible for deleting the object
  207. This will be the case if there are no pending Asynch IO requests
  208. --*/
  209. {
  210. BOOL fDeleteNeeded = FALSE;
  211. BOOL fHandleClosed = FALSE;
  212. HANDLE hDir = INVALID_HANDLE_VALUE;
  213. DBG_ASSERT(m_cDirRefCount == 0);
  214. BOOL fInCleanup = (BOOL) InterlockedExchange((long *) &m_fInCleanup, TRUE);
  215. DBG_ASSERT(fInCleanup == FALSE);
  216. // Get the IO ref count BEFORE we close the handle
  217. DWORD cIORefCount = m_cIORefCount;
  218. if (m_hDir != INVALID_HANDLE_VALUE)
  219. {
  220. // If we have a pending AtqReadDirectoryChanges,
  221. // closing the directory handle will cause a call back from ATQ.
  222. // The call back should relase the final refcount on the object
  223. // which should result in its deletion
  224. hDir = m_hDir;
  225. m_hDir = INVALID_HANDLE_VALUE;
  226. fHandleClosed = CloseHandle( hDir );
  227. }
  228. // If there were no pending Asynch IO operations or if we failed
  229. // to close the handle, then the caller will be responsible for
  230. // deleting this object.
  231. if (cIORefCount == 0 || fHandleClosed == FALSE)
  232. {
  233. fDeleteNeeded = TRUE;
  234. }
  235. return fDeleteNeeded;
  236. }
  237. BOOL
  238. CDirMonitorEntry::ResetDirectoryHandle(
  239. VOID
  240. )
  241. /*++
  242. Routine Description:
  243. Opens a new directory handle for the path,
  244. and closes the old ones. We want to be able to do this so we
  245. can change the size of the buffer passed in ReadDirectoryChangesW.
  246. Arguments:
  247. None
  248. Return Value:
  249. TRUE if the handles were succesfully reopened
  250. FALSE otherwise
  251. --*/
  252. {
  253. // BUGBUG - Beta2 HACK
  254. return FALSE;
  255. // We'd better have a directory path available to try this
  256. if (m_pszPath == NULL)
  257. {
  258. return FALSE;
  259. }
  260. // Get a new handle to the directory
  261. HANDLE hDir = CreateFileW( m_pszPath,
  262. FILE_LIST_DIRECTORY,
  263. FILE_SHARE_READ |
  264. FILE_SHARE_WRITE |
  265. FILE_SHARE_DELETE,
  266. NULL,
  267. OPEN_EXISTING,
  268. FILE_FLAG_BACKUP_SEMANTICS |
  269. FILE_FLAG_OVERLAPPED,
  270. NULL );
  271. if ( hDir == INVALID_HANDLE_VALUE )
  272. {
  273. // We couldn't open another handle on the directory,
  274. // leave the current handle and ATQ context alone
  275. return FALSE;
  276. }
  277. //
  278. // Associate handle with thread pool's completion port
  279. //
  280. if ( !ThreadPoolBindIoCompletionCallback( hDir,
  281. DirMonOverlappedCompletionRoutine,
  282. 0 ) )
  283. {
  284. // We couldn't get a new ATQ context. Close our new handle.
  285. // We leave the objects current handle and ATQ context alone
  286. CloseHandle(hDir);
  287. return FALSE;
  288. }
  289. // We have the new handle and ATQ context so we close
  290. // and replace the old ones.
  291. if ( m_hDir != INVALID_HANDLE_VALUE )
  292. {
  293. CloseHandle( m_hDir );
  294. }
  295. m_hDir = hDir;
  296. return TRUE;
  297. }
  298. BOOL
  299. CDirMonitorEntry::SetBufferSize(
  300. DWORD cBufferSize
  301. )
  302. /*++
  303. Routine Description:
  304. Sets the size of the buffer used for storing change notification records
  305. Arguments:
  306. cBufferSize new size for the buffer.
  307. Return Value:
  308. TRUE if the size of the buffer was succesfully set
  309. FALSE otherwise
  310. Note
  311. When a call to ReadDirectoryChangesW is made, the size of the buffer is set in
  312. the data associated with the directory handle and is not changed on subsequent
  313. calls to ReadDirectoryChangesW. To make use of the new buffer size the directory
  314. handle must be closed and a new handle opened (see ResetDirectoryHandle())
  315. --*/
  316. {
  317. // We should never be called if the buffer doesn't already exist
  318. DBG_ASSERT(m_pbBuffer);
  319. // Don't allow the buffer to be set to 0
  320. if (cBufferSize == 0)
  321. {
  322. return FALSE;
  323. }
  324. VOID *pbBuffer = LocalReAlloc( m_pbBuffer,
  325. cBufferSize,
  326. LMEM_MOVEABLE );
  327. if (pbBuffer == NULL)
  328. {
  329. // Re-allocation failed, stuck with the same size buffer
  330. return FALSE;
  331. }
  332. else
  333. {
  334. // Re-allocation succeded, update the member variables
  335. m_pbBuffer = (BYTE *) pbBuffer;
  336. m_cBufferSize = cBufferSize;
  337. return TRUE;
  338. }
  339. }
  340. //
  341. // CDirMonitor
  342. //
  343. CDirMonitor::CDirMonitor()
  344. : CTypedHashTable<CDirMonitor, CDirMonitorEntry, const WCHAR*>("DirMon")
  345. /*++
  346. Routine Description:
  347. CDirMonitor constructor
  348. Arguments:
  349. None
  350. Return Value:
  351. Nothing
  352. --*/
  353. {
  354. ThreadPoolInitialize();
  355. INITIALIZE_CRITICAL_SECTION( &m_csSerialComplLock );
  356. m_cRefs = 1;
  357. m_fShutdown = FALSE;
  358. }
  359. CDirMonitor::~CDirMonitor()
  360. /*++
  361. Routine Description:
  362. CDirMonitor destructor
  363. Arguments:
  364. None
  365. Return Value:
  366. Nothing
  367. --*/
  368. {
  369. DeleteCriticalSection(&m_csSerialComplLock);
  370. ThreadPoolTerminate();
  371. }
  372. BOOL
  373. CDirMonitor::Monitor(
  374. CDirMonitorEntry * pDME,
  375. LPWSTR pszDirectory,
  376. BOOL fWatchSubDirectories,
  377. DWORD dwNotificationFlags
  378. )
  379. /*++
  380. Routine Description:
  381. Create a monitor entry for the specified path
  382. Arguments:
  383. pszDirectory - directory to monitor
  384. pCtxt - Context of path is being monitored
  385. pszDirectory - name of directory to monitor
  386. fWatchSubDirectories - whether to get notifications for subdirectories
  387. dwNotificationFlags - which activities to be notified of
  388. Return Value:
  389. TRUE if success, otherwise FALSE
  390. Remarks:
  391. Caller should have a lock on the CDirMonitor
  392. Not compatible with WIN95
  393. --*/
  394. {
  395. LIST_ENTRY *pEntry;
  396. HANDLE hDirectoryFile = INVALID_HANDLE_VALUE;
  397. BOOL fRet = TRUE;
  398. DWORD dwDirLength = 0;
  399. LK_RETCODE lkrc;
  400. IF_DEBUG( DIRMON )
  401. {
  402. DBGPRINTF((DBG_CONTEXT, "[CDirMonitor] Monitoring new CDirMonitorEntry\n"));
  403. }
  404. // Must have a directory monitor entry and a string
  405. // containing the directory path
  406. if (!pDME || !pszDirectory)
  407. {
  408. SetLastError(ERROR_INVALID_PARAMETER);
  409. return FALSE;
  410. }
  411. // Make copy of pszDirectory for the entry to hang on to
  412. pDME->m_cPathLength = wcslen(pszDirectory);
  413. if ( !(pDME->m_pszPath = (LPWSTR)LocalAlloc( LPTR,
  414. ( pDME->m_cPathLength + 1 ) * sizeof( WCHAR ) )) )
  415. {
  416. pDME->m_cPathLength = 0;
  417. return FALSE;
  418. }
  419. memcpy( pDME->m_pszPath,
  420. pszDirectory,
  421. ( pDME->m_cPathLength + 1 ) * sizeof( WCHAR ) );
  422. pDME->Init();
  423. // Open the directory handle
  424. hDirectoryFile = CreateFileW(
  425. pszDirectory,
  426. FILE_LIST_DIRECTORY,
  427. FILE_SHARE_READ |
  428. FILE_SHARE_WRITE |
  429. FILE_SHARE_DELETE,
  430. NULL,
  431. OPEN_EXISTING,
  432. FILE_FLAG_BACKUP_SEMANTICS |
  433. FILE_FLAG_OVERLAPPED,
  434. NULL );
  435. if ( hDirectoryFile == INVALID_HANDLE_VALUE )
  436. {
  437. // Cleanup
  438. LocalFree(pDME->m_pszPath);
  439. pDME->m_pszPath = NULL;
  440. pDME->m_cPathLength = 0;
  441. return FALSE;
  442. }
  443. else
  444. {
  445. // Store the handle so we can close it on cleanup
  446. pDME->m_hDir = hDirectoryFile;
  447. // Set the flags for the type of notifications we want
  448. // and if we should watch subdirectories or just the root
  449. pDME->m_dwNotificationFlags = dwNotificationFlags;
  450. pDME->m_fWatchSubdirectories = fWatchSubDirectories;
  451. // Get an ATQ context for this handle
  452. // and register our completion call back function
  453. if ( ThreadPoolBindIoCompletionCallback(
  454. hDirectoryFile,
  455. DirMonOverlappedCompletionRoutine,
  456. 0 ) )
  457. {
  458. // Insert this entry into the list of active entries
  459. lkrc = InsertEntry(pDME);
  460. if (lkrc == LK_SUCCESS)
  461. {
  462. // Ask for notification if this directory has changes
  463. if (!pDME->RequestNotification())
  464. {
  465. // Couldn't register for change notification
  466. // Clean up resources
  467. RemoveEntry(pDME);
  468. pDME->m_hDir = INVALID_HANDLE_VALUE;
  469. CloseHandle( hDirectoryFile );
  470. LocalFree(pDME->m_pszPath);
  471. pDME->m_pszPath = NULL;
  472. pDME->m_cPathLength = 0;
  473. return FALSE;
  474. }
  475. }
  476. else
  477. {
  478. //
  479. // Could not add to hash table. Cleanup
  480. //
  481. CloseHandle(hDirectoryFile);
  482. pDME->m_hDir = INVALID_HANDLE_VALUE;
  483. LocalFree(pDME->m_pszPath);
  484. pDME->m_pszPath = NULL;
  485. pDME->m_cPathLength = 0;
  486. //
  487. // If it has been added from underneath us, indicate so
  488. //
  489. if ( lkrc == LK_KEY_EXISTS )
  490. {
  491. SetLastError( ERROR_ALREADY_EXISTS );
  492. }
  493. return FALSE;
  494. }
  495. }
  496. else
  497. {
  498. // Failed to add handle to ATQ, clean up
  499. CloseHandle(hDirectoryFile);
  500. pDME->m_hDir = INVALID_HANDLE_VALUE;
  501. LocalFree(pDME->m_pszPath);
  502. pDME->m_pszPath = NULL;
  503. pDME->m_cPathLength = 0;
  504. return FALSE;
  505. }
  506. }
  507. return TRUE;
  508. }
  509. VOID
  510. CDirMonitor::DirMonitorCompletionFunction(
  511. PVOID pCtxt,
  512. DWORD dwBytesWritten,
  513. DWORD dwCompletionStatus,
  514. OVERLAPPED * pOvr
  515. )
  516. /*++
  517. Routine Description:
  518. Static member function called by ATQ to signal directory changes
  519. Arguments:
  520. pCtxt - CDirMonitorEntry*
  521. dwBytesWritten - # bytes returned by ReadDirectoryChanges
  522. dwCompletionStatus - status of request to ReadDirectoryChanges
  523. pOvr - OVERLAPPED as specified in call to ReadDirectoryChanges
  524. Return Value:
  525. Nothing
  526. --*/
  527. {
  528. IF_DEBUG( DIRMON )
  529. {
  530. DBGPRINTF((DBG_CONTEXT, "[CDirMonitor] Notification call-back begining. Status %d\n", dwCompletionStatus));
  531. }
  532. CDirMonitorEntry* pDirMonitorEntry = reinterpret_cast<CDirMonitorEntry*>(pCtxt);
  533. DBG_ASSERT(pDirMonitorEntry);
  534. // Safety add ref, this should guarentee that the DME is not deleted
  535. // while we are still processing the callback
  536. pDirMonitorEntry->IOAddRef();
  537. // Release for the current Asynch operation
  538. // Should not send IO ref count to 0
  539. DBG_REQUIRE(pDirMonitorEntry->IORelease());
  540. BOOL fRequestNotification = FALSE;
  541. // There has been a change in the directory we were monitoring
  542. // carry out whatever work we need to do.
  543. if (!pDirMonitorEntry->m_fInCleanup)
  544. {
  545. pDirMonitorEntry->m_pDirMonitor->SerialComplLock();
  546. // BUGBUG Under stress ActOnNotification has been initiating a chain
  547. // of events leading to an AV. For Beta 3 we think we can ignore
  548. // these AV. For the final product we need to rework the critical
  549. // sections for the template manager and
  550. // the include file table.
  551. __try
  552. {
  553. fRequestNotification = pDirMonitorEntry->ActOnNotification(dwCompletionStatus, dwBytesWritten);
  554. }
  555. __except( EXCEPTION_EXECUTE_HANDLER )
  556. {
  557. // We should never get here
  558. DBG_ASSERT(FALSE);
  559. }
  560. // If ActOnNotification returned TRUE, then make another Asynch
  561. // notification request.
  562. if (fRequestNotification)
  563. {
  564. fRequestNotification = pDirMonitorEntry->RequestNotification();
  565. }
  566. pDirMonitorEntry->m_pDirMonitor->SerialComplUnlock();
  567. }
  568. // Remove safety ref count, may cause IO ref count to go to 0
  569. pDirMonitorEntry->IORelease();
  570. IF_DEBUG( DIRMON )
  571. {
  572. DBGPRINTF((DBG_CONTEXT, "[CDirMonitor] Notification call-back ending\n"));
  573. }
  574. }
  575. CDirMonitorEntry *
  576. CDirMonitor::FindEntry(
  577. LPWSTR pszPath
  578. )
  579. /*++
  580. Routine Description:
  581. Searches the list of entries for the specified path
  582. Arguments:
  583. pszPath - file path, including file name
  584. Return Value:
  585. pointer to the entry, allready addref'd
  586. --*/
  587. {
  588. DBG_ASSERT(pszPath);
  589. CDirMonitorEntry *pDME = NULL;
  590. //
  591. // Lock to prevent aga finding an entry that could be deleted by someone
  592. //
  593. ReadLock();
  594. FindKey(pszPath, &pDME);
  595. if (pDME != NULL)
  596. {
  597. if (pDME->m_fInCleanup)
  598. {
  599. // Don't hand back a DME that is being shutdown
  600. pDME = NULL;
  601. }
  602. else
  603. {
  604. // We found a valid DME which we are going to hand to the caller
  605. pDME->AddRef();
  606. }
  607. }
  608. ReadUnlock();
  609. return pDME;
  610. }
  611. LK_RETCODE
  612. CDirMonitor::InsertEntry(
  613. CDirMonitorEntry *pDME
  614. )
  615. /*++
  616. Routine Description:
  617. Insert an entry into the list of entries for the monitor
  618. Arguments:
  619. pDME - entry to insert
  620. Return Value:
  621. LK returncode
  622. --*/
  623. {
  624. DBG_ASSERT(pDME);
  625. LK_RETCODE lkResult;
  626. //
  627. // If we're cleaning up the table, don't let anyone else add stuff to
  628. // it
  629. //
  630. if ( m_fShutdown )
  631. {
  632. return LK_NOT_INITIALIZED;
  633. }
  634. pDME->m_pDirMonitor = this;
  635. lkResult = InsertRecord(pDME, false);
  636. if (lkResult == LK_SUCCESS)
  637. {
  638. DBGPRINTF((DBG_CONTEXT, "[CDirMonitor] Inserting directory (DME %08x) %ws\n", pDME, pDME->m_pszPath));
  639. AddRef();
  640. }
  641. else
  642. {
  643. pDME->m_pDirMonitor = NULL;
  644. }
  645. return lkResult;
  646. }
  647. LK_RETCODE
  648. CDirMonitor::RemoveEntry(
  649. CDirMonitorEntry *pDME
  650. )
  651. /*++
  652. Routine Description:
  653. Deletes an entry from the list of entries for the monitor
  654. Arguments:
  655. pDME - entry to delete
  656. Return Value:
  657. None
  658. --*/
  659. {
  660. DBG_ASSERT(pDME);
  661. //
  662. // Delete the entry from the hash table
  663. //
  664. LK_RETCODE lkResult = DeleteRecord(pDME);
  665. pDME->m_pDirMonitor = NULL;
  666. DBGPRINTF((DBG_CONTEXT, "[CDirMonitor] Removed DME(%08x), directory %ws\n", pDME, pDME->m_pszPath));
  667. //
  668. // Release the DME's reference on the DirMonitor object.
  669. //
  670. // It seems obvious, but we must have the Release() occur AFTER the
  671. // removal of the hash-table entry. This is because the loop check
  672. // in CDirMonitor::Cleanup() can complete while DeleteKey() is still
  673. // doing stuff to the table. I mention this since DirMon cleanup is an
  674. // often-butchered code path.
  675. //
  676. Release();
  677. return lkResult;
  678. }
  679. VOID
  680. CDirMonitor::Cleanup(
  681. VOID
  682. )
  683. /*++
  684. Routine Description:
  685. Pauses while all entries are cleaned up
  686. Arguments:
  687. None
  688. Return Value:
  689. None
  690. --*/
  691. {
  692. //
  693. // Don't let anyone else try to insert into the hashtable
  694. //
  695. m_fShutdown = TRUE;
  696. //
  697. // Sleep until the hash table is empty and all contained
  698. // CDirMonitorEntry's are destroyed
  699. //
  700. while (Size() > 0 || m_cRefs != 1)
  701. {
  702. //
  703. // At least one DME is still active, sleep and try again
  704. //
  705. Sleep(200);
  706. }
  707. }