422 lines
9.3 KiB
C++
422 lines
9.3 KiB
C++
|
//========= Copyright Valve Corporation, All rights reserved. ============//
|
||
|
//
|
||
|
// Purpose:
|
||
|
//
|
||
|
// $NoKeywords: $
|
||
|
//===========================================================================//
|
||
|
#include "cbase.h"
|
||
|
#include "sentence.h"
|
||
|
#include "wavefile.h"
|
||
|
#include "tier2/riff.h"
|
||
|
#include "filesystem.h"
|
||
|
#include <io.h>
|
||
|
#include <fcntl.h>
|
||
|
#include <sys/types.h>
|
||
|
#include "IFileLoader.h"
|
||
|
|
||
|
bool SceneManager_LoadSentenceFromWavFileUsingIO( char const *wavfile, CSentence& sentence, IFileReadBinary& io );
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Implements the RIFF i/o interface on stdio
|
||
|
//-----------------------------------------------------------------------------
|
||
|
class ThreadIOReadBinary : public IFileReadBinary
|
||
|
{
|
||
|
public:
|
||
|
int open( const char *pFileName )
|
||
|
{
|
||
|
char filename[ 512 ];
|
||
|
// POSSIBLE BUG: THIS MIGHT NOT BE THREAD SAFE!!!
|
||
|
filesystem->RelativePathToFullPath( pFileName, "GAME", filename, sizeof( filename ) );
|
||
|
return (int)_open( filename, _O_BINARY | _O_RDONLY );
|
||
|
}
|
||
|
|
||
|
int read( void *pOutput, int size, int file )
|
||
|
{
|
||
|
if ( !file )
|
||
|
return 0;
|
||
|
|
||
|
return _read( file, pOutput, size );
|
||
|
}
|
||
|
|
||
|
void seek( int file, int pos )
|
||
|
{
|
||
|
if ( !file )
|
||
|
return;
|
||
|
|
||
|
_lseek( file, pos, SEEK_SET );
|
||
|
}
|
||
|
|
||
|
unsigned int tell( int file )
|
||
|
{
|
||
|
if ( !file )
|
||
|
return 0;
|
||
|
|
||
|
return _tell( file );
|
||
|
}
|
||
|
|
||
|
unsigned int size( int file )
|
||
|
{
|
||
|
if ( !file )
|
||
|
return 0;
|
||
|
|
||
|
long curpos = tell( file );
|
||
|
_lseek( file, 0, SEEK_END );
|
||
|
int s = tell( file );
|
||
|
_lseek( file, curpos, SEEK_SET );
|
||
|
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
void close( int file )
|
||
|
{
|
||
|
if ( !file )
|
||
|
return;
|
||
|
|
||
|
_close( file );
|
||
|
}
|
||
|
};
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: All wavefile I/O occurs on a thread
|
||
|
//-----------------------------------------------------------------------------
|
||
|
class CFileLoaderThread : public IFileLoader
|
||
|
{
|
||
|
public:
|
||
|
struct SentenceRequest
|
||
|
{
|
||
|
SentenceRequest()
|
||
|
{
|
||
|
filename[ 0 ] = 0;
|
||
|
sentence.Reset();
|
||
|
wavefile = NULL;
|
||
|
valid = false;
|
||
|
}
|
||
|
|
||
|
bool valid;
|
||
|
char filename[ 256 ];
|
||
|
CSentence sentence;
|
||
|
|
||
|
CWaveFile *wavefile;
|
||
|
};
|
||
|
|
||
|
// Construction
|
||
|
CFileLoaderThread( void );
|
||
|
virtual ~CFileLoaderThread( void );
|
||
|
|
||
|
// Sockets add/remove themselves via their constructor
|
||
|
virtual void AddWaveFilesToThread( CUtlVector< CWaveFile * >& wavefiles );
|
||
|
|
||
|
// Lock changes to wavefile list, etc.
|
||
|
virtual void Lock( void );
|
||
|
// Unlock wavefile list, etc.
|
||
|
virtual void Unlock( void );
|
||
|
|
||
|
// Retrieve handle to shutdown event
|
||
|
virtual HANDLE GetShutdownHandle( void );
|
||
|
|
||
|
// Caller should call lock before accessing any of these methods and unlock afterwards!!!
|
||
|
virtual int ProcessCompleted();
|
||
|
|
||
|
int DoThreadWork();
|
||
|
|
||
|
virtual void Start();
|
||
|
|
||
|
virtual int GetPendingLoadCount();
|
||
|
private:
|
||
|
// Critical section used for synchronizing access to wavefile list
|
||
|
CRITICAL_SECTION cs;
|
||
|
CRITICAL_SECTION m_CountCS;
|
||
|
|
||
|
// List of wavefiles we are listening on
|
||
|
CUtlVector< SentenceRequest * > m_FileList;
|
||
|
|
||
|
CUtlVector< SentenceRequest * > m_Pending;
|
||
|
CUtlVector< SentenceRequest * > m_Completed;
|
||
|
// Thread handle
|
||
|
HANDLE m_hThread;
|
||
|
// Thread id
|
||
|
DWORD m_nThreadId;
|
||
|
// Event to set when we want to tell the thread to shut itself down
|
||
|
HANDLE m_hShutdown;
|
||
|
|
||
|
ThreadIOReadBinary m_ThreadIO;
|
||
|
bool m_bLocked;
|
||
|
|
||
|
int m_nTotalAdds;
|
||
|
int m_nTotalPending;
|
||
|
int m_nTotalProcessed;
|
||
|
int m_nTotalCompleted;
|
||
|
|
||
|
HANDLE m_hNewItems;
|
||
|
};
|
||
|
|
||
|
// Singleton handler
|
||
|
static CFileLoaderThread g_WaveLoader;
|
||
|
extern IFileLoader *fileloader = &g_WaveLoader;
|
||
|
|
||
|
int CFileLoaderThread::DoThreadWork()
|
||
|
{
|
||
|
int i;
|
||
|
// Check for shutdown event
|
||
|
if ( WAIT_OBJECT_0 == WaitForSingleObject( GetShutdownHandle(), 0 ) )
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
// No changes to list right now
|
||
|
Lock();
|
||
|
// Move new items to work list
|
||
|
int newItems = m_FileList.Count();
|
||
|
for ( i = 0; i < newItems; i++ )
|
||
|
{
|
||
|
// Move to pending and issue async i/o calls
|
||
|
m_Pending.AddToHead( m_FileList[ i ] );
|
||
|
|
||
|
EnterCriticalSection( &m_CountCS );
|
||
|
m_nTotalPending++;
|
||
|
LeaveCriticalSection( &m_CountCS );
|
||
|
}
|
||
|
m_FileList.RemoveAll();
|
||
|
// Done adding new work items
|
||
|
Unlock();
|
||
|
|
||
|
int remaining = m_Pending.Count();
|
||
|
if ( !remaining )
|
||
|
return 1;
|
||
|
|
||
|
int workitems = remaining; // min( remaining, 1000 );
|
||
|
|
||
|
CUtlVector< SentenceRequest * > transfer;
|
||
|
|
||
|
for ( i = 0; i < workitems; i++ )
|
||
|
{
|
||
|
SentenceRequest *r = m_Pending[ 0 ];
|
||
|
m_Pending.Remove( 0 );
|
||
|
|
||
|
transfer.AddToTail( r );
|
||
|
|
||
|
// Do the work
|
||
|
EnterCriticalSection( &m_CountCS );
|
||
|
m_nTotalProcessed++;
|
||
|
LeaveCriticalSection( &m_CountCS );
|
||
|
|
||
|
Lock();
|
||
|
bool load = !r->wavefile->HasLoadedSentenceInfo();
|
||
|
Unlock();
|
||
|
|
||
|
if ( load )
|
||
|
{
|
||
|
r->valid = SceneManager_LoadSentenceFromWavFileUsingIO( r->filename, r->sentence, m_ThreadIO );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
r->valid = true;
|
||
|
}
|
||
|
|
||
|
if ( WaitForSingleObject( m_hNewItems, 0 ) == WAIT_OBJECT_0 )
|
||
|
{
|
||
|
ResetEvent( m_hNewItems );
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Now move to completed list
|
||
|
Lock();
|
||
|
int c = transfer.Count();
|
||
|
|
||
|
for ( i = 0; i < c; ++i )
|
||
|
{
|
||
|
SentenceRequest *r = transfer[ i ];
|
||
|
if ( r->valid )
|
||
|
{
|
||
|
|
||
|
m_nTotalCompleted++;
|
||
|
|
||
|
|
||
|
m_Completed.AddToTail( r );
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
delete r;
|
||
|
}
|
||
|
}
|
||
|
Unlock();
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int CFileLoaderThread::ProcessCompleted()
|
||
|
{
|
||
|
Lock();
|
||
|
int c = m_Completed.Count();
|
||
|
for ( int i = c - 1; i >= 0 ; i-- )
|
||
|
{
|
||
|
SentenceRequest *r = m_Completed[ i ];
|
||
|
|
||
|
if ( !r->wavefile->HasLoadedSentenceInfo() )
|
||
|
{
|
||
|
r->wavefile->SetThreadLoadedSentence( r->sentence );
|
||
|
}
|
||
|
|
||
|
delete r;
|
||
|
}
|
||
|
m_Completed.RemoveAll();
|
||
|
Unlock();
|
||
|
return c;
|
||
|
}
|
||
|
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Main winsock processing thread
|
||
|
// Input : threadobject -
|
||
|
// Output : static DWORD WINAPI
|
||
|
//-----------------------------------------------------------------------------
|
||
|
static DWORD WINAPI FileLoaderThreadFunc( LPVOID threadobject )
|
||
|
{
|
||
|
// Get pointer to CFileLoaderThread object
|
||
|
CFileLoaderThread *wavefilethread = ( CFileLoaderThread * )threadobject;
|
||
|
Assert( wavefilethread );
|
||
|
if ( !wavefilethread )
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
// Keep looking for data until shutdown event is triggered
|
||
|
while ( 1 )
|
||
|
{
|
||
|
if( !wavefilethread->DoThreadWork() )
|
||
|
break;
|
||
|
|
||
|
// Yield a small bit of time to main app
|
||
|
Sleep( 10 );
|
||
|
}
|
||
|
|
||
|
ExitThread( 0 );
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Construction
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CFileLoaderThread::CFileLoaderThread( void )
|
||
|
{
|
||
|
m_nTotalAdds = 0;
|
||
|
m_nTotalProcessed = 0;
|
||
|
m_nTotalCompleted = 0;
|
||
|
m_nTotalPending = 0;
|
||
|
|
||
|
m_bLocked = false;
|
||
|
|
||
|
InitializeCriticalSection( &cs );
|
||
|
InitializeCriticalSection( &m_CountCS );
|
||
|
|
||
|
m_hShutdown = CreateEvent( NULL, TRUE, FALSE, NULL );
|
||
|
Assert( m_hShutdown );
|
||
|
|
||
|
m_hThread = NULL;
|
||
|
|
||
|
m_hNewItems = CreateEvent( NULL, TRUE, FALSE, NULL );
|
||
|
|
||
|
Start();
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CFileLoaderThread::Start()
|
||
|
{
|
||
|
m_hThread = CreateThread( NULL, 0, FileLoaderThreadFunc, (void *)this, 0, &m_nThreadId );
|
||
|
Assert( m_hThread );
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
CFileLoaderThread::~CFileLoaderThread( void )
|
||
|
{
|
||
|
Lock();
|
||
|
{
|
||
|
SetEvent( m_hShutdown );
|
||
|
Sleep( 2 );
|
||
|
TerminateThread( m_hThread, 0 );
|
||
|
}
|
||
|
Unlock();
|
||
|
|
||
|
// Kill the wavefile
|
||
|
//!! need to validate this line
|
||
|
// Assert( !m_FileList );
|
||
|
|
||
|
CloseHandle( m_hThread );
|
||
|
|
||
|
CloseHandle( m_hShutdown );
|
||
|
|
||
|
DeleteCriticalSection( &cs );
|
||
|
DeleteCriticalSection( &m_CountCS );
|
||
|
|
||
|
CloseHandle( m_hNewItems );
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Returns handle of shutdown event
|
||
|
// Output : HANDLE
|
||
|
//-----------------------------------------------------------------------------
|
||
|
HANDLE CFileLoaderThread::GetShutdownHandle( void )
|
||
|
{
|
||
|
return m_hShutdown;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose: Locks object and adds wavefile to thread
|
||
|
// Input : *wavefile -
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CFileLoaderThread::AddWaveFilesToThread( CUtlVector< CWaveFile * >& wavefiles )
|
||
|
{
|
||
|
Lock();
|
||
|
int c = wavefiles.Count();
|
||
|
for ( int i = 0; i < c; i++ )
|
||
|
{
|
||
|
SentenceRequest *request = new SentenceRequest;
|
||
|
request->wavefile = wavefiles[ i ];
|
||
|
Q_strncpy( request->filename, request->wavefile->GetFileName(), sizeof( request->filename ) );
|
||
|
|
||
|
m_FileList.AddToTail( request );
|
||
|
|
||
|
m_nTotalAdds++;
|
||
|
}
|
||
|
|
||
|
SetEvent( m_hNewItems );
|
||
|
Unlock();
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CFileLoaderThread::Lock( void )
|
||
|
{
|
||
|
EnterCriticalSection( &cs );
|
||
|
Assert( !m_bLocked );
|
||
|
m_bLocked = true;
|
||
|
}
|
||
|
|
||
|
//-----------------------------------------------------------------------------
|
||
|
// Purpose:
|
||
|
//-----------------------------------------------------------------------------
|
||
|
void CFileLoaderThread::Unlock( void )
|
||
|
{
|
||
|
Assert( m_bLocked );
|
||
|
m_bLocked = false;
|
||
|
LeaveCriticalSection( &cs );
|
||
|
}
|
||
|
|
||
|
int CFileLoaderThread::GetPendingLoadCount()
|
||
|
{
|
||
|
int iret = 0;
|
||
|
|
||
|
EnterCriticalSection( &m_CountCS );
|
||
|
|
||
|
iret = m_nTotalPending - m_nTotalProcessed;
|
||
|
|
||
|
LeaveCriticalSection( &m_CountCS );
|
||
|
|
||
|
return iret;
|
||
|
}
|