//========= Copyright Valve Corporation, All rights reserved. ============// // // Purpose: Sound management functions. Exposes a list of available sounds. // // $NoKeywords: $ //=============================================================================// #include "stdafx.h" #include "soundsystem.h" #include "mmsystem.h" #include "filesystem.h" #include "KeyValues.h" #include "hammer.h" #include "HammerScene.h" #include "ScenePreviewDlg.h" #include "soundchars.h" // memdbgon must be the last include file in a .cpp file!!! #include <tier0/memdbgon.h> // FIXME: Put gamesounds parsing into shared code somewhere #define MANIFEST_FILE "scripts/game_sounds_manifest.txt" #define SOUNDGENDER_MACRO "$gender" #define SOUNDGENDER_MACRO_LENGTH 7 // Length of above including $ // Sounds we're playing are loaded into here for Windows to access while playing them. CUtlVector<char> g_SoundPlayData; //----------------------------------------------------------------------------- // Singleton sound system //----------------------------------------------------------------------------- CSoundSystem g_Sounds; //----------------------------------------------------------------------------- // Constructor, destructor //----------------------------------------------------------------------------- CSoundSystem::CSoundSystem() { } CSoundSystem::~CSoundSystem() { ShutDown(); } //----------------------------------------------------------------------------- // Initialization, shutdown //----------------------------------------------------------------------------- bool CSoundSystem::Initialize( ) { for ( int i = 0; i < SOUND_TYPE_COUNT; ++i ) { m_SoundList[i].m_Sounds.EnsureCapacity( 1024 ); m_SoundList[i].m_pStrings = NULL; if (!BuildSoundList( (SoundType_t)i ) ) return false; } return true; } void CSoundSystem::ShutDown(void) { for ( int i = 0; i < SOUND_TYPE_COUNT; ++i ) { CleanupSoundList( (SoundType_t)i ); } } //----------------------------------------------------------------------------- // Build the list of sounds //----------------------------------------------------------------------------- bool CSoundSystem::BuildSoundList( SoundType_t type ) { CleanupSoundList( type ); switch( type ) { case SOUND_TYPE_RAW: return RecurseIntoDirectories( "sound", &CSoundSystem::ProcessDirectory_RawFileList ); case SOUND_TYPE_GAMESOUND: return BuildGameSoundList(); case SOUND_TYPE_SCENE: return RecurseIntoDirectories( "scenes", &CSoundSystem::ProcessDirectory_SceneFileList ); } return false; } //----------------------------------------------------------------------------- // Cleans up the sound list //----------------------------------------------------------------------------- void CSoundSystem::CleanupSoundList( SoundType_t type ) { m_SoundList[type].m_Sounds.RemoveAll(); DestroyStringCache( m_SoundList[type].m_pStrings ); m_SoundList[type].m_pStrings = NULL; } //----------------------------------------------------------------------------- // Allocate, deallocate a string cache //----------------------------------------------------------------------------- CSoundSystem::StringCache_t *CSoundSystem::CreateStringCache( CSoundSystem::StringCache_t* pPrevious ) { StringCache_t *pCache = new StringCache_t; pCache->m_nTailIndex = 0; pCache->m_pNext = pPrevious; return pCache; } void CSoundSystem::DestroyStringCache( CSoundSystem::StringCache_t *pCache ) { if ( pCache ) { DestroyStringCache( pCache->m_pNext ); delete pCache; } } //----------------------------------------------------------------------------- // Adds a string to the string cache //----------------------------------------------------------------------------- char *CSoundSystem::AddStringToCache( SoundType_t type, const char *pString ) { int copyLen = V_strlen( pString ) + 1; StringCache_t *pCache = m_SoundList[type].m_pStrings; if ( (!pCache) || ( copyLen + pCache->m_nTailIndex > StringCache_t::STRING_CACHE_SIZE ) ) { m_SoundList[type].m_pStrings = CreateStringCache( pCache ); pCache = m_SoundList[type].m_pStrings; } char fixedString[MAX_PATH]; V_strncpy( fixedString, pString, sizeof( fixedString ) ); V_FixSlashes( fixedString ); copyLen = V_strlen( fixedString ) + 1; char *pDest = &pCache->m_pBuf[ pCache->m_nTailIndex ]; memcpy( pDest, fixedString, copyLen ); pCache->m_nTailIndex += copyLen; return pDest; } //----------------------------------------------------------------------------- // Adds a sound to a sound list //----------------------------------------------------------------------------- void CSoundSystem::AddSoundToList( SoundType_t type, const char *pSoundName, const char *pActualFile, const char *pSourceFile ) { // FIXME: Optimize the allocation pattern? int i = m_SoundList[type].m_Sounds.AddToTail(); SoundInfo_t &info = m_SoundList[type].m_Sounds[i]; info.m_pSoundName = AddStringToCache( type, pSoundName ); if ( type == SOUND_TYPE_RAW ) { info.m_pSoundFile = info.m_pSoundName; info.m_pSourceFile = info.m_pSoundName; } else { info.m_pSoundFile = AddStringToCache( type, pActualFile ); info.m_pSourceFile = pSourceFile; } } //----------------------------------------------------------------------------- // Add all sounds that lie within a single directory //----------------------------------------------------------------------------- void CSoundSystem::BuildFileListInDirectory( char const* pDirectoryName, const char *pExt, SoundType_t soundType ) { Assert( Q_strlen( pExt ) <= 3 ); int nDirectoryNameLen = V_strlen( pDirectoryName ); char *pWildCard = ( char * )stackalloc( nDirectoryNameLen + 7 ); Q_snprintf( pWildCard, nDirectoryNameLen + 7, "%s/*.%s", pDirectoryName, pExt ); FileFindHandle_t findHandle; const char *pFileName = g_pFullFileSystem->FindFirst( pWildCard, &findHandle ); for ( ; pFileName; pFileName = g_pFullFileSystem->FindNext( findHandle ) ) { if( g_pFullFileSystem->FindIsDirectory( findHandle ) ) continue; // Strip off the 'sound/' part of the sound name. int nAllocSize = nDirectoryNameLen + Q_strlen(pFileName) + 2; char *pFileNameWithPath = (char *)stackalloc( nAllocSize ); const char *pStartPos = max( strchr( pDirectoryName, '/' ), strchr( pDirectoryName, '\\' ) ); if ( pStartPos ) Q_snprintf( pFileNameWithPath, nAllocSize, "%s%c%s", pStartPos+1, CORRECT_PATH_SEPARATOR, pFileName ); else V_strncpy( pFileNameWithPath, pFileName, nAllocSize ); Q_strnlwr( pFileNameWithPath, nAllocSize ); AddSoundToList( soundType, pFileNameWithPath, pFileNameWithPath, NULL ); } g_pFullFileSystem->FindClose( findHandle ); } //----------------------------------------------------------------------------- // Populate the list of .WAV files //----------------------------------------------------------------------------- bool CSoundSystem::RecurseIntoDirectories( char const* pDirectoryName, pDirCallbackFn fn ) { // Have the callback process the directory. if ( !(this->*fn)( pDirectoryName ) ) return false; int nDirectoryNameLen = Q_strlen( pDirectoryName ); char *pWildCard = ( char * )stackalloc( nDirectoryNameLen + 5 ); strcpy(pWildCard, pDirectoryName); strcat(pWildCard, "/*.*"); int nPathStrLen = nDirectoryNameLen + 1; FileFindHandle_t findHandle; const char *pFileName = g_pFullFileSystem->FindFirst( pWildCard, &findHandle ); for ( ; pFileName; pFileName = g_pFullFileSystem->FindNext( findHandle ) ) { if ((pFileName[0] != '.') || (pFileName[1] != '.' && pFileName[1] != 0)) { if( !g_pFullFileSystem->FindIsDirectory( findHandle ) ) continue; int fileNameStrLen = Q_strlen( pFileName ); char *pFileNameWithPath = ( char * )stackalloc( nPathStrLen + fileNameStrLen + 1 ); memcpy( pFileNameWithPath, pWildCard, nPathStrLen ); pFileNameWithPath[nPathStrLen] = '\0'; Q_strncat( pFileNameWithPath, pFileName, nPathStrLen + fileNameStrLen + 1 ); if (!RecurseIntoDirectories( pFileNameWithPath, fn )) return false; } } return true; } //----------------------------------------------------------------------------- // Populate the list of .WAV files //----------------------------------------------------------------------------- bool CSoundSystem::ProcessDirectory_RawFileList( char const* pDirectoryName ) { if ( !g_pFileSystem ) return false; Assert( Q_strnicmp( pDirectoryName, "sound", 5 ) == 0 ); // Get all sound files out of this directory BuildFileListInDirectory( pDirectoryName, "wav", SOUND_TYPE_RAW ); BuildFileListInDirectory( pDirectoryName, "mp3", SOUND_TYPE_RAW ); return true; } //----------------------------------------------------------------------------- // Populate the list of .VCD files //----------------------------------------------------------------------------- bool CSoundSystem::ProcessDirectory_SceneFileList( char const* pDirectoryName ) { if ( !g_pFileSystem ) return false; // Get all sound files out of this directory BuildFileListInDirectory( pDirectoryName, "vcd", SOUND_TYPE_SCENE ); return true; } //----------------------------------------------------------------------------- // Splits a name into 2 //----------------------------------------------------------------------------- static void SplitName( char const *input, int splitchar, int splitlen, char *before, int beforelen, char *after, int afterlen ) { char const *in = input; char *out = before; int c = 0; int l = 0; int maxl = beforelen; while ( *in ) { if ( c == splitchar ) { while ( --splitlen >= 0 ) { in++; } *out = 0; out = after; maxl = afterlen; c++; continue; } if ( l >= maxl ) { in++; c++; continue; } *out++ = *in++; l++; c++; } *out = 0; } //----------------------------------------------------------------------------- // Gamesounds may have macros embedded in them //----------------------------------------------------------------------------- void CSoundSystem::AddGameSoundToList( const char *pGameSound, char const *pFileName, const char *pSourceFile ) { char const *p = Q_stristr( pFileName, SOUNDGENDER_MACRO ); if ( !p ) { AddSoundToList( SOUND_TYPE_GAMESOUND, pGameSound, pFileName, pSourceFile ); return; } int offset = p - pFileName; Assert( offset >= 0 ); int duration = SOUNDGENDER_MACRO_LENGTH; // Create a "male" version of the sound only for browsing char before[ 256 ], after[ 256 ]; Q_memset( before, 0, sizeof( before ) ); Q_memset( after, 0, sizeof( after ) ); SplitName( pFileName, offset, duration, before, sizeof( before ), after, sizeof( after ) ); char temp[ 256 ]; Q_snprintf( temp, sizeof( temp ), "%s%s%s", before, "male", after ); AddSoundToList( SOUND_TYPE_GAMESOUND, pGameSound, temp, pSourceFile ); } // memdbgon must be the last include file in a .cpp file!!! #include <tier0/memdbgoff.h> //----------------------------------------------------------------------------- // Load all game sounds from a particular file //----------------------------------------------------------------------------- void CSoundSystem::AddGameSoundsFromFile( const char *pFileName ) { KeyValues *kv = new KeyValues( pFileName ); if ( !kv->LoadFromFile( g_pFileSystem, pFileName, "GAME" ) ) { kv->deleteThis(); return; } const char *pSourceFile = AddStringToCache( SOUND_TYPE_GAMESOUND, pFileName ); // parse out all of the top level sections and save their names for ( KeyValues *pKeys = kv; pKeys; pKeys = pKeys->GetNextKey() ) { if ( !pKeys->GetFirstSubKey() ) continue; const char *pRawFile = pKeys->GetString( "wave", NULL ); if ( pRawFile ) { AddGameSoundToList( pKeys->GetName(), pRawFile, pSourceFile ); } else { KeyValues *pRndWave = pKeys->FindKey( "rndwave" ); if ( pRndWave ) { KeyValues *pFirstFile = pRndWave->GetFirstSubKey(); if ( pFirstFile ) { AddGameSoundToList( pKeys->GetName(), pFirstFile->GetString(), pSourceFile ); } } } } if ( kv ) { kv->deleteThis(); } } //----------------------------------------------------------------------------- // Populate the list of game sounds //----------------------------------------------------------------------------- bool CSoundSystem::BuildGameSoundList() { KeyValues *manifest = new KeyValues( MANIFEST_FILE ); if ( !manifest->LoadFromFile( g_pFileSystem, MANIFEST_FILE, "GAME" ) ) { manifest->deleteThis(); return false; } for ( KeyValues *sub = manifest->GetFirstSubKey(); sub != NULL; sub = sub->GetNextKey() ) { if ( !Q_stricmp( sub->GetName(), "precache_file" ) || !Q_stricmp( sub->GetName(), "declare_file" ) || !Q_stricmp( sub->GetName(), "preload_file" ) ) { // Add and always precache AddGameSoundsFromFile( sub->GetString() ); } } manifest->deleteThis(); return true; } //----------------------------------------------------------------------------- // Plays a sound //----------------------------------------------------------------------------- bool CSoundSystem::FindSoundByName( const char *pFilename, SoundType_t *type, int *nIndex ) { char searchStr[MAX_PATH]; V_strncpy( searchStr, pFilename, sizeof( searchStr ) ); V_FixSlashes( searchStr ); for ( int i = SOUND_TYPE_COUNT; --i >= 0; ) { for ( int j = SoundCount( (SoundType_t)i ); --j >= 0; ) { if ( Q_stristr( searchStr, SoundName( (SoundType_t)i, j ) ) ) { *type = (SoundType_t)i; *nIndex = j; return true; } } } return false; } bool CSoundSystem::PlayScene( const char *pFileName ) { char fullFilename[MAX_PATH]; V_snprintf( fullFilename, sizeof( fullFilename ), "scenes%c%s", CORRECT_PATH_SEPARATOR, pFileName ); CChoreoScene *pScene = HammerLoadScene( fullFilename ); if ( !pScene ) return false; CScenePreviewDlg dlg( pScene, pFileName ); dlg.DoModal(); return true; } //----------------------------------------------------------------------------- // Plays a sound //----------------------------------------------------------------------------- bool CSoundSystem::Play( SoundType_t type, int nIndex ) { const char *pFileName = SoundFile( type, nIndex ); if ( !pFileName ) return false; // If it's a scene, get the first sound in the scene. if ( type == SOUND_TYPE_SCENE ) { return PlayScene( pFileName ); } // Voiceover files have this. pFileName = PSkipSoundChars( pFileName ); char pRelativePath[MAX_PATH]; Q_snprintf( pRelativePath, MAX_PATH, "sound/%s", pFileName ); // Stop any previously-playing sound. StopSound(); // We used to use GetLocalPath, but that doesn't work under Steam. FileHandle_t fp = g_pFileSystem->Open( pRelativePath, "rb" ); if ( fp ) { g_SoundPlayData.SetSize( g_pFileSystem->Size( fp ) ); if ( g_pFileSystem->Read( g_SoundPlayData.Base(), g_SoundPlayData.Count(), fp ) == g_SoundPlayData.Count() ) { return (PlaySound( g_SoundPlayData.Base(), NULL, SND_ASYNC | SND_MEMORY ) != FALSE); } g_pFileSystem->Close( fp ); } return false; } //----------------------------------------------------------------------------- // Stops any playing sound. //----------------------------------------------------------------------------- void CSoundSystem::StopSound() { PlaySound( NULL, NULL, SND_ASYNC | SND_MEMORY ); } //----------------------------------------------------------------------------- // Opens the source file associated with a sound //----------------------------------------------------------------------------- void CSoundSystem::OpenSource( SoundType_t type, int nIndex ) { if ( type == SOUND_TYPE_RAW ) return; const char *pFileName = SoundSourceFile( type, nIndex ); if ( pFileName ) { char pRelativePath[MAX_PATH]; Q_snprintf( pRelativePath, MAX_PATH, "%s", pFileName ); char pFullPath[MAX_PATH]; if ( g_pFullFileSystem->GetLocalPath( pRelativePath, pFullPath, MAX_PATH ) ) { ShellExecute( NULL, "open", pFullPath, NULL, NULL, SW_SHOWNORMAL ); } } }