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.
161 lines
4.9 KiB
161 lines
4.9 KiB
5 years ago
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose:
|
||
|
//
|
||
|
//=============================================================================
|
||
|
|
||
|
#include "stdafx.h"
|
||
|
#include "FileChangeWatcher.h"
|
||
|
#include "tier1/utldict.h"
|
||
|
#include "filesystem_tools.h"
|
||
|
|
||
|
|
||
|
CFileChangeWatcher::CFileChangeWatcher()
|
||
|
{
|
||
|
m_pCallbacks = NULL;
|
||
|
}
|
||
|
|
||
|
CFileChangeWatcher::~CFileChangeWatcher()
|
||
|
{
|
||
|
Term();
|
||
|
}
|
||
|
|
||
|
void CFileChangeWatcher::Init( ICallbacks *pCallbacks )
|
||
|
{
|
||
|
Term();
|
||
|
m_pCallbacks = pCallbacks;
|
||
|
}
|
||
|
|
||
|
bool CFileChangeWatcher::AddDirectory( const char *pSearchPathBase, const char *pDirName, bool bRecursive )
|
||
|
{
|
||
|
char fullDirName[MAX_PATH];
|
||
|
V_ComposeFileName( pSearchPathBase, pDirName, fullDirName, sizeof( fullDirName ) );
|
||
|
|
||
|
HANDLE hDir = CreateFile( fullDirName, GENERIC_READ, FILE_SHARE_READ|FILE_SHARE_WRITE|FILE_SHARE_DELETE, NULL, OPEN_EXISTING, FILE_FLAG_BACKUP_SEMANTICS|FILE_FLAG_OVERLAPPED, NULL );
|
||
|
if ( hDir == INVALID_HANDLE_VALUE )
|
||
|
{
|
||
|
Warning( "CFileChangeWatcher::AddDirectory - can't get a handle to directory %s.\n", pDirName );
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Call this once to start the ball rolling.. Next time we call it, it'll tell us the changes that
|
||
|
// have happened since this call.
|
||
|
CDirWatch *pDirWatch = new CDirWatch;
|
||
|
V_strncpy( pDirWatch->m_SearchPathBase, pSearchPathBase, sizeof( pDirWatch->m_SearchPathBase ) );
|
||
|
V_strncpy( pDirWatch->m_DirName, pDirName, sizeof( pDirWatch->m_DirName ) );
|
||
|
V_strncpy( pDirWatch->m_FullDirName, fullDirName, sizeof( pDirWatch->m_FullDirName ) );
|
||
|
pDirWatch->m_hDir = hDir;
|
||
|
pDirWatch->m_hEvent = CreateEvent( NULL, false, false, NULL );
|
||
|
memset( &pDirWatch->m_Overlapped, 0, sizeof( pDirWatch->m_Overlapped ) );
|
||
|
pDirWatch->m_Overlapped.hEvent = pDirWatch->m_hEvent;
|
||
|
if ( !CallReadDirectoryChanges( pDirWatch ) )
|
||
|
{
|
||
|
CloseHandle( pDirWatch->m_hEvent );
|
||
|
CloseHandle( pDirWatch->m_hDir );
|
||
|
delete pDirWatch;
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
m_DirWatches.AddToTail( pDirWatch );
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void CFileChangeWatcher::Term()
|
||
|
{
|
||
|
for ( int i=0; i < m_DirWatches.Count(); i++ )
|
||
|
{
|
||
|
CloseHandle( m_DirWatches[i]->m_hDir );
|
||
|
CloseHandle( m_DirWatches[i]->m_hEvent );
|
||
|
}
|
||
|
m_DirWatches.PurgeAndDeleteElements();
|
||
|
m_pCallbacks = NULL;
|
||
|
}
|
||
|
|
||
|
int CFileChangeWatcher::Update()
|
||
|
{
|
||
|
CUtlDict< int, int > queuedChanges;
|
||
|
int nTotalChanges = 0;
|
||
|
|
||
|
// Check each CDirWatch.
|
||
|
int i = 0;
|
||
|
while ( i < m_DirWatches.Count() )
|
||
|
{
|
||
|
CDirWatch *pDirWatch = m_DirWatches[i];
|
||
|
|
||
|
DWORD dwBytes = 0;
|
||
|
if ( GetOverlappedResult( pDirWatch->m_hDir, &pDirWatch->m_Overlapped, &dwBytes, FALSE ) )
|
||
|
{
|
||
|
// Read through the notifications.
|
||
|
int nBytesLeft = (int)dwBytes;
|
||
|
char *pCurPos = pDirWatch->m_Buffer;
|
||
|
while ( nBytesLeft >= sizeof( FILE_NOTIFY_INFORMATION ) )
|
||
|
{
|
||
|
FILE_NOTIFY_INFORMATION *pNotify = (FILE_NOTIFY_INFORMATION*)pCurPos;
|
||
|
|
||
|
if ( m_pCallbacks )
|
||
|
{
|
||
|
// Figure out what happened to this file.
|
||
|
WCHAR nullTerminated[2048];
|
||
|
int nBytesToCopy = min( (int)pNotify->FileNameLength, 2047 );
|
||
|
memcpy( nullTerminated, pNotify->FileName, nBytesToCopy );
|
||
|
nullTerminated[nBytesToCopy/2] = 0;
|
||
|
char ansiFilename[1024];
|
||
|
V_UnicodeToUTF8( nullTerminated, ansiFilename, sizeof( ansiFilename ) );
|
||
|
|
||
|
// Now add it to the queue. We use this queue because sometimes Windows will give us multiple
|
||
|
// of the same modified notification back to back, and we want to reduce redundant calls.
|
||
|
int iExisting = queuedChanges.Find( ansiFilename );
|
||
|
if ( iExisting == queuedChanges.InvalidIndex() )
|
||
|
{
|
||
|
iExisting = queuedChanges.Insert( ansiFilename, 0 );
|
||
|
++nTotalChanges;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( pNotify->NextEntryOffset == 0 )
|
||
|
break;
|
||
|
|
||
|
pCurPos += pNotify->NextEntryOffset;
|
||
|
nBytesLeft -= (int)pNotify->NextEntryOffset;
|
||
|
}
|
||
|
|
||
|
CallReadDirectoryChanges( pDirWatch );
|
||
|
continue; // Check again because sometimes it queues up duplicate notifications.
|
||
|
}
|
||
|
|
||
|
// Process all the entries in the queue.
|
||
|
for ( int iQueuedChange=queuedChanges.First(); iQueuedChange != queuedChanges.InvalidIndex(); iQueuedChange=queuedChanges.Next( iQueuedChange ) )
|
||
|
{
|
||
|
SendNotification( pDirWatch, queuedChanges.GetElementName( iQueuedChange ) );
|
||
|
}
|
||
|
queuedChanges.Purge();
|
||
|
++i;
|
||
|
}
|
||
|
|
||
|
return nTotalChanges;
|
||
|
}
|
||
|
|
||
|
void CFileChangeWatcher::SendNotification( CFileChangeWatcher::CDirWatch *pDirWatch, const char *pRelativeFilename )
|
||
|
{
|
||
|
// Use this for full filenames although you don't strictly need it..
|
||
|
char fullFilename[MAX_PATH];
|
||
|
V_ComposeFileName( pDirWatch->m_FullDirName, pRelativeFilename, fullFilename, sizeof( fullFilename ) );
|
||
|
|
||
|
m_pCallbacks->OnFileChange( pRelativeFilename, fullFilename );
|
||
|
}
|
||
|
|
||
|
BOOL CFileChangeWatcher::CallReadDirectoryChanges( CFileChangeWatcher::CDirWatch *pDirWatch )
|
||
|
{
|
||
|
return ReadDirectoryChangesW( pDirWatch->m_hDir,
|
||
|
pDirWatch->m_Buffer,
|
||
|
sizeof( pDirWatch->m_Buffer ),
|
||
|
true,
|
||
|
FILE_NOTIFY_CHANGE_FILE_NAME | FILE_NOTIFY_CHANGE_SIZE | FILE_NOTIFY_CHANGE_LAST_WRITE,
|
||
|
NULL,
|
||
|
&pDirWatch->m_Overlapped,
|
||
|
NULL );
|
||
|
}
|
||
|
|
||
|
|
||
|
|