1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        wx/msw/private/fswatcher.h 
   3 // Purpose:     File system watcher impl classes 
   4 // Author:      Bartosz Bekier 
   7 // Copyright:   (c) 2009 Bartosz Bekier <bartosz.bekier@gmail.com> 
   8 // Licence:     wxWindows licence 
   9 ///////////////////////////////////////////////////////////////////////////// 
  11 #ifndef WX_MSW_PRIVATE_FSWATCHER_H_ 
  12 #define WX_MSW_PRIVATE_FSWATCHER_H_ 
  14 #include "wx/filename.h" 
  15 #include "wx/vector.h" 
  16 #include "wx/msw/private.h" 
  18 // ============================================================================ 
  19 // wxFSWatcherEntry implementation & helper declarations 
  20 // ============================================================================ 
  22 class wxFSWatcherImplMSW
; 
  24 class wxFSWatchEntryMSW 
: public wxFSWatchInfo
 
  29         BUFFER_SIZE 
= 4096   // TODO parametrize 
  32     wxFSWatchEntryMSW(const wxFSWatchInfo
& winfo
) : 
  35         // get handle for this path 
  36         m_handle 
= OpenDir(m_path
); 
  37         m_overlapped 
= (OVERLAPPED
*)calloc(1, sizeof(OVERLAPPED
)); 
  38         wxZeroMemory(m_buffer
); 
  41     virtual ~wxFSWatchEntryMSW() 
  43         wxLogTrace(wxTRACE_FSWATCHER
, "Deleting entry '%s'", m_path
); 
  45         if (m_handle 
!= INVALID_HANDLE_VALUE
) 
  47             if (!CloseHandle(m_handle
)) 
  49                 wxLogSysError(_("Unable to close the handle for '%s'"), 
  58         return m_handle 
!= INVALID_HANDLE_VALUE
; 
  61     HANDLE 
GetHandle() const 
  71     OVERLAPPED
* GetOverlapped() const 
  77     // opens dir with all flags, attributes etc. necessary to be later 
  78     // asynchronous watched with ReadDirectoryChangesW 
  79     static HANDLE 
OpenDir(const wxString
& path
) 
  81         HANDLE handle 
= CreateFile(path
.t_str(), 
  88                                    FILE_FLAG_BACKUP_SEMANTICS 
| 
  91         if (handle 
== INVALID_HANDLE_VALUE
) 
  93             wxLogSysError(_("Failed to open directory \"%s\" for monitoring."), 
 100     HANDLE m_handle
;             // handle to opened directory 
 101     char m_buffer
[BUFFER_SIZE
];  // buffer for fs events 
 102     OVERLAPPED
* m_overlapped
; 
 104     wxDECLARE_NO_COPY_CLASS(wxFSWatchEntryMSW
); 
 107 // ============================================================================ 
 108 // wxFSWatcherImplMSW helper classes implementations 
 109 // ============================================================================ 
 115         m_iocp(INVALID_HANDLE_VALUE
) 
 122         if (m_iocp 
!= INVALID_HANDLE_VALUE
) 
 124             if (!CloseHandle(m_iocp
)) 
 126                 wxLogSysError(_("Unable to close I/O completion port handle")); 
 132     // associates a wxFSWatchEntryMSW with completion port 
 133     bool Add(wxSharedPtr
<wxFSWatchEntryMSW
> watch
) 
 135         wxCHECK_MSG( m_iocp 
!= INVALID_HANDLE_VALUE
, false, "IOCP not init" ); 
 136         wxCHECK_MSG( watch
->IsOk(), false, "Invalid watch" ); 
 138         // associate with IOCP 
 139         HANDLE ret 
= CreateIoCompletionPort(watch
->GetHandle(), m_iocp
, 
 140                                             (ULONG_PTR
)watch
.get(), 0); 
 143             wxLogSysError(_("Unable to associate handle with " 
 144                             "I/O completion port")); 
 147         else if (ret 
!= m_iocp
) 
 149             wxFAIL_MSG(_("Unexpectedly new I/O completion port was created")); 
 154         wxFSWatchEntries::value_type 
val(watch
->GetPath(), watch
); 
 155         return m_watches
.insert(val
).second
; 
 158     // Removes a watch we're currently using. Notice that this doesn't happen 
 159     // immediately, CompleteRemoval() must be called later when it's really 
 160     // safe to delete the watch, i.e. after completion of the IO operation 
 162     bool ScheduleForRemoval(const wxSharedPtr
<wxFSWatchEntryMSW
>& watch
) 
 164         wxCHECK_MSG( m_iocp 
!= INVALID_HANDLE_VALUE
, false, "IOCP not init" ); 
 165         wxCHECK_MSG( watch
->IsOk(), false, "Invalid watch" ); 
 167         const wxString path 
= watch
->GetPath(); 
 168         wxFSWatchEntries::iterator it 
= m_watches
.find(path
); 
 169         wxCHECK_MSG( it 
!= m_watches
.end(), false, 
 170                      "Can't remove a watch we don't use" ); 
 172         // We can't just delete the watch here as we can have pending events 
 173         // for it and if we destroyed it now, we could get a dangling (or, 
 174         // worse, reused to point to another object) pointer in ReadEvents() so 
 175         // just remember that this one should be removed when CompleteRemoval() 
 177         m_removedWatches
.insert(wxFSWatchEntries::value_type(path
, watch
)); 
 183     // Really remove the watch previously passed to ScheduleForRemoval(). 
 185     // It's ok to call this for a watch that hadn't been removed before, in 
 186     // this case we'll just return false and do nothing. 
 187     bool CompleteRemoval(wxFSWatchEntryMSW
* watch
) 
 189         wxFSWatchEntries::iterator it 
= m_removedWatches
.find(watch
->GetPath()); 
 190         if ( it 
== m_removedWatches
.end() ) 
 193         // Removing the object from the map will result in deleting the watch 
 194         // itself as it's not referenced from anywhere else now. 
 195         m_removedWatches
.erase(it
); 
 200     // post completion packet 
 201     bool PostEmptyStatus() 
 203         wxCHECK_MSG( m_iocp 
!= INVALID_HANDLE_VALUE
, false, "IOCP not init" ); 
 205         int ret 
= PostQueuedCompletionStatus(m_iocp
, 0, 0, NULL
); 
 208             wxLogSysError(_("Unable to post completion status")); 
 214     // Wait for completion status to arrive. 
 215     // This function can block forever in it's wait for completion status. 
 216     // Use PostEmptyStatus() to wake it up (and end the worker thread) 
 217     bool GetStatus(unsigned long* count
, wxFSWatchEntryMSW
** watch
, 
 218                    OVERLAPPED
** overlapped
) 
 220         wxCHECK_MSG( m_iocp 
!= INVALID_HANDLE_VALUE
, false, "IOCP not init" ); 
 221         wxCHECK_MSG( count 
!= NULL
, false, "Null out parameter 'count'"); 
 222         wxCHECK_MSG( watch 
!= NULL
, false, "Null out parameter 'watch'"); 
 223         wxCHECK_MSG( overlapped 
!= NULL
, false, 
 224                      "Null out parameter 'overlapped'"); 
 226         int ret 
= GetQueuedCompletionStatus(m_iocp
, count
, (ULONG_PTR 
*)watch
, 
 227                                             overlapped
, INFINITE
); 
 230             wxLogSysError(_("Unable to dequeue completion packet")); 
 238         m_iocp 
= CreateIoCompletionPort(INVALID_HANDLE_VALUE
, NULL
, 0, 0); 
 241             wxLogSysError(_("Unable to create I/O completion port")); 
 243         return m_iocp 
!= NULL
; 
 248     // The hash containing all the wxFSWatchEntryMSW objects currently being 
 249     // watched keyed by their paths. 
 250     wxFSWatchEntries m_watches
; 
 252     // Contains the watches which had been removed but are still pending. 
 253     wxFSWatchEntries m_removedWatches
; 
 257 class wxIOCPThread 
: public wxThread
 
 260     wxIOCPThread(wxFSWatcherImplMSW
* service
, wxIOCPService
* iocp
); 
 262     // finishes this thread 
 266     // structure to hold information needed to process one native event 
 267     // this is just a dummy holder, so it doesn't take ownership of it's data 
 268     struct wxEventProcessingData
 
 270         wxEventProcessingData(const FILE_NOTIFY_INFORMATION
* ne
, 
 271                               const wxFSWatchEntryMSW
* watch
) : 
 272             nativeEvent(ne
), watch(watch
) 
 275         const FILE_NOTIFY_INFORMATION
* nativeEvent
; 
 276         const wxFSWatchEntryMSW
* watch
; 
 279     virtual ExitCode 
Entry(); 
 281     // wait for events to occur, read them and send to interested parties 
 282     // returns false it empty status was read, which means we whould exit 
 286     void ProcessNativeEvents(wxVector
<wxEventProcessingData
>& events
); 
 288     void SendEvent(wxFileSystemWatcherEvent
& evt
); 
 290     static int Native2WatcherFlags(int flags
); 
 292     static wxString 
FileNotifyInformationToString( 
 293                                             const FILE_NOTIFY_INFORMATION
& e
); 
 295     static wxFileName 
GetEventPath(const wxFSWatchEntryMSW
& watch
, 
 296                                    const FILE_NOTIFY_INFORMATION
& e
); 
 298     wxFSWatcherImplMSW
* m_service
; 
 299     wxIOCPService
* m_iocp
; 
 302 #endif /* WX_MSW_PRIVATE_FSWATCHER_H_ */