導(dǎo)言
FMOD Ex API被設(shè)計(jì)得非常直觀和靈活,本教程將簡(jiǎn)要介紹引擎的使用方法,并對(duì)使用中涉及的相關(guān)問題做出解釋。
配置——What to include and what to link.
查閱文檔中的"Platform specific issues",找出指定平臺(tái)上要使用FMOD Ex函數(shù)所需鏈接的文件。
在C/C++中,如果只須使用C接口,則包含"fmod.h",要使用C++接口則須包含"fmod.hpp"。
注意:常量、回調(diào)函數(shù)、宏定義和枚舉類型都在fmod.h中,所以fmod.hpp也包含fmod.h,如果你使用C++的話就必須交叉使用。
對(duì)于Delphi、C#和Visual Basic,都有其對(duì)應(yīng)的頭文件可供在程序中使用。
初始化
要初始化fmod,最簡(jiǎn)單的方法就是調(diào)用System::init函數(shù),FMOD會(huì)使用默認(rèn)參數(shù)來(lái)配置聲卡和其他設(shè)置。
在查看文檔中的System::init函數(shù)時(shí),要記住maxchannels參數(shù)是你在游戲中可以同時(shí)播放聲音的最大數(shù)量,該數(shù)與聲卡或軟件混合器無(wú)關(guān)。
這些聲音是 虛擬聲音(virtual voices)。這意味著你可以同時(shí)播放任意多的聲音,而不用擔(dān)心硬件或軟件資源問題。
你可以在游戲中安全地播放每個(gè)聲音,而無(wú)須擔(dān)心System::playSound超出播放上限或搶占其它聲音。因此只要你喜歡,你可以將maxchannels設(shè)置為任意大的數(shù),如1,100,200,1000。
注意:同時(shí)播放1000個(gè)聲音并不會(huì)音像性能,因?yàn)樗鼈冎械拇蟛糠质锹牪灰姷?(聽不見的聲音被“虛擬化”了)。而FMOD Ex虛擬聲音管理器決定哪些聲音能聽見,哪些聽不見只需要很小的系統(tǒng)開銷。
下面我們看一個(gè)初始化FMOD Ex的例子。
FMOD_RESULT result;
FMOD::System *system;
result =system); // 創(chuàng)建主system對(duì)象
if (result != FMOD_OK)
{
printf("FMOD error! (%d) %s\n", result, FMOD_ErrorString(result));
exit(-1);
}
result = system->init(100, FMOD_INIT_NORMAL, 0); // 初始化FMOD
if (result != FMOD_OK)
{
printf("FMOD error! (%d) %s\n", result, FMOD_ErrorString(result));
exit(-1);
}
這是最基本的初始化FOMD引擎的方法,使用了100個(gè)虛擬聲音。
注意:mod、s3m、xm、it和midi格式的文件播放時(shí)只使用一個(gè)聲音,不要以為在這里增大數(shù)量就可以使用多個(gè)聲音來(lái)播放這些格式的文件。這些格式的都使用它們自己的internal pool voices。
配置選項(xiàng)
如果你不想使用默認(rèn)設(shè)置的話,可以自行設(shè)定輸出硬件、FOMD資源使用和其他配置選項(xiàng)。
這些操作都必須在System::init函數(shù)調(diào)用前進(jìn)行。
常用配置如下:
· System::setOutput – 選擇輸出方式。例如你可以在Windows上選擇DirectSound、WinMM、ASIO、no-sound、wave-writer或其他輸出選項(xiàng),每個(gè)平臺(tái)都有不同的選擇。如果你只是要使用默認(rèn)配置,就不需要調(diào)用它。
· System::setDriver – 選擇用于播放的設(shè)備驅(qū)動(dòng)。當(dāng)你擁有不止一塊聲卡并且不想使用默認(rèn)聲卡的時(shí)候,這個(gè)函數(shù)就很有用了。你需要用System::getNumDrivers函數(shù)來(lái)獲取設(shè)備數(shù),用System::getDriverName函數(shù)來(lái)獲取驅(qū)動(dòng)名稱以供用戶選擇。
· System::setHardwareChannels – 使用這個(gè)函數(shù)來(lái)限制硬件聲音的數(shù)量,或設(shè)定在reverting to 100% software mixed voice support之前的最小硬件聲音數(shù)量。“minimum”選項(xiàng)用于保證同時(shí)至少能聽到的聲音數(shù)量。
· System::setSoftwareChannels – 使用這個(gè)函數(shù)來(lái)設(shè)定FMOD通道使用的軟件混合聲音的數(shù)量。 This will be purely for polyphony reasons or CPU / memory resource usage reasons. 使用mod/s3m/xm/it/midi這些格式時(shí),不要指望使用它來(lái)增加聲音的數(shù)量,它們不使用通道的pool而使用自己的。
· System::setSoftwareFormat – 用于改變FMOD軟件混合器的設(shè)置。包括采樣率、輸出模式(如integer vs float)、輸出通道數(shù)(如multi-output channel ASIO devices)、內(nèi)存使用和混合質(zhì)量。
· System::setDSPBufferSize – 如果在很慢的機(jī)器上或非良好的聲卡驅(qū)動(dòng)上出現(xiàn)聲音抖動(dòng),就需要使用這個(gè)函數(shù)。它可以改變軟件混合器的響應(yīng)時(shí)間,但誤用的話也可能影響性能。一些人可能想讓用戶在“低響應(yīng)時(shí)間”和“高兼容性”兩種模式間做出選擇,可以通過調(diào)整緩沖大小來(lái)達(dá)到犧牲響應(yīng)時(shí)間換取穩(wěn)定性的目的。
· System::setSpeakerMode – 設(shè)置揚(yáng)聲器輸出模式。此函數(shù)只對(duì)FMOD軟件混合引擎起作用,默認(rèn)值為stereo (5.1 on xbox and xbox360 and 7.1 on ps3),可以按需要任意更改。注意:聲道數(shù)越多,占用的內(nèi)存也越多。
下面的例子使用了一些配置項(xiàng)來(lái)初始化FMOD。 要記住這些選項(xiàng)都是可選的,如果不需要就不用設(shè)置,在你沒有弄懂其真正含義前,千萬(wàn)不要僅僅將下面的代碼復(fù)制再粘貼! 例如,如果用戶沒有5.1聲道的系統(tǒng),你就不能僅僅將揚(yáng)聲器模式設(shè)定為5.1。
FMOD_RESULT result;
FMOD::System *system;
result =system); //創(chuàng)建主system對(duì)象
ERRCHECK(result);
// 設(shè)置揚(yáng)聲器模式為5.1聲道
result = system->setSpeakerMode(FMOD_SPEAKERMODE_5POINT1); ERRCHECK(result);
// 允許同時(shí)播放100個(gè)軟件混合聲音
result = system->setSoftwareChannels(100); ERRCHECK(result);
// 要求聲卡至少要有32個(gè)2D和3D硬件聲音,如果聲音數(shù)超過64,就限制為64
result = system->setHardwareChannels(32, 64, 32, 64); ERRCHECK(result);
// 初始化FOMD,使用100個(gè)虛擬聲音
result = system->init(200, FMOD_INIT_NORMAL, 0);
ERRCHECK(result);
加載與播放
要播放聲音,你必須先加載!
使用System::createSound或System::createStream就可以完成這項(xiàng)工作。
默認(rèn)情況下,系統(tǒng)會(huì)嘗試將整個(gè)聲音解壓到內(nèi)存中(如果使用 System::createSound函數(shù)),而和sample不同的是stream(用 System::createStream創(chuàng)建)是在運(yùn)行時(shí)解碼,僅使用很少的內(nèi)存作為緩沖,這就是為什么大文件最好用stream的原因。
更多請(qǐng)參見術(shù)語(yǔ)/基本原理。
下面是一個(gè)加載MP3文件的例子,默認(rèn)條件下System::createSound函數(shù)將整個(gè)MP3解壓成16bit的PCM格式,這就意味著將占用比文件本身大許多倍的內(nèi)存。
FMOD::Sound *sound;
// FMOD_DEFAULT等效于FMOD_LOOP_OFF | FMOD_2D | FMOD_HARDWARE.
result = system->createSound("../media/wave.mp3", FMOD_DEFAULT, 0, &sound);
ERRCHECK(result);
下面是一個(gè)用stream打開MP3文件的例子。System::createStream函數(shù)將打開文件并預(yù)緩沖小部分?jǐn)?shù)據(jù),然后就可以在System::playSound調(diào)用時(shí)直接播放了。
FMOD::Sound *sound;
// FMOD_DEFAULT等效于FMOD_LOOP_OFF | FMOD_2D | FMOD_HARDWARE.
result = system->createStream("../media/wave.mp3", FMOD_DEFAULT, 0, &sound);
ERRCHECK(result);
指定用軟件混合就必須使用FMOD_SOFTWARE標(biāo)記。如果你想要使用如DSP effects、spectrum analysis、getwavedata、point to point looping和其它更多的高級(jí)技術(shù),就必須使用軟件混合。
FMOD::Sound *sound;
// 使用軟件混合
result = system->createSound("../media/wave.mp3", FMOD_SOFTWARE, 0, &sound);
ERRCHECK(result);
下一個(gè)例子是將MP3以sample的形式載入內(nèi)存而不解壓,使用 FMOD_CREATECOMPRESSEDSAMPLE標(biāo)記。此時(shí)如果沒有指定FMOD_HARDWARE或FMOD_SOFTWARE,將默認(rèn)為軟件混合。硬件聲音回放不支持這個(gè)標(biāo)記,除非格式為ADPCM on Xbox、VAG on PS2/PSP或GCADPCM on Gamecube/Wii。 Platforms like PS3 and Xbox 360 are all done one the cpu (usually a different core to the main cpu so it does not affect performance).
FMOD::Sound *sound;
// FMOD_CREATECOMPRESSEDSAMPLE標(biāo)記讓sample先嘗試直接播放(不解壓到內(nèi)存),但僅限于IMA ADPCM、MP2、MP3和XMA格式
result = system->createSound("../media/wave.mp3", FMOD_CREATECOMPRESSEDSAMPLE, 0, &sound);
ERRCHECK(result);
警告! 必須謹(jǐn)慎使用這種模式,它看上去和PCM sample很相似,但它會(huì)在運(yùn)行時(shí)導(dǎo)致巨大CPU開銷。FMOD按照聲音的壓縮格式,在播放時(shí)對(duì)其進(jìn)行解碼。
現(xiàn)在,要播放sound或stream只需簡(jiǎn)單地調(diào)用System::playSound就行了。
FMOD::Channel *channel;
result = system->playSound(FMOD_CHANNEL_FREE, sound, false, &channel);
ERRCHECK(result);
此時(shí),聲音已經(jīng)在后臺(tái)播放了!而你的程序?qū)⒗^續(xù)執(zhí)行。
關(guān)于playSound的注意事項(xiàng):
· 如果不需要的話,可以不必獲取channel句柄,可以將其設(shè)為0或NULL。如果你不需要更改這個(gè)sound實(shí)例,或者這個(gè)聲音很短(不循環(huán)),就可以省去它。例如:
result = system->playSound(FMOD_CHANNEL_FREE, sound, false, 0);
ERRCHECK(result);
· 可以在開始播放時(shí)暫停,這樣就可以更改聲音的屬性而不會(huì)被用戶聽見,這就是“paused”參數(shù)的用處所在。例如,如果你將paused設(shè)為true,再設(shè)置音量為0.5,然后解除暫停,此時(shí)聲音就會(huì)一一般的音量播放。但如果你是將paused設(shè)為false,而其它操作相同的話,你會(huì)聽見聲音開始時(shí)瞬間是全音量播放的,用戶可不希望聽到。
result = system->playSound(FMOD_CHANNEL_FREE, sound, true, &channel);
ERRCHECK(result);
// 暫停時(shí)設(shè)定音量
result = channel->setVolume(0.5f);
ERRCHECK(result);
// 聲音從這里才開始播放
result = channel->setPaused(false);
ERRCHECK(result);
· 一個(gè)“channel”就是一個(gè)聲音的實(shí)例。一個(gè)聲音你可以同時(shí)播放多次,每次播放都會(huì)得到一個(gè)新的channel句柄,但stream除外,它只能同時(shí)播放一次,如果你嘗試多次播放,只會(huì)重復(fù)播放當(dāng)前stream并返回同上次一樣的channel句柄。這是因?yàn)閟tream只有一個(gè)緩沖和一個(gè)文件句柄。要同時(shí)播放兩個(gè)stream就必須打開兩次再播放兩次。
· 始終使用FMOD_CHANNEL_FREE。FMOD會(huì)使用通道管理器自動(dòng)為你選擇一個(gè)未使用的channel。如果希望使用一個(gè)現(xiàn)有的channel來(lái)播放,就使用FMOD_CHANNEL_REUSE標(biāo)記,這樣可以避免每次調(diào)用System::playSound函數(shù)都產(chǎn)生一個(gè)新的實(shí)例。
· 不需要“free”或“release”一個(gè)channel句柄。所有的channel都位于你使用System::init所創(chuàng)建的一個(gè)pool中。當(dāng)聲音停止后,channel可以被復(fù)用;如果所有的channel都處于播放狀態(tài),那么其中一個(gè)優(yōu)先級(jí)最低的會(huì)被搶占。其實(shí)只需增大System::init中的通道數(shù)就可以避免發(fā)生這樣的情況。
· 一個(gè)channel會(huì)在播放結(jié)束時(shí)即刻失效。這意味著你不能再對(duì)其進(jìn)行更改,即使做了也沒有實(shí)際意義,因?yàn)樗豢赡茉俨シ帕恕=^大多數(shù)情況下,引用一個(gè)已失效的channel會(huì)導(dǎo)致一個(gè)FMOD_ERR_INVALID_HANDLE錯(cuò)誤。
Update. (This is important!)
在每一幀中調(diào)用System::update函數(shù)是很重要的,但不需要多次調(diào)用,那樣只會(huì)影響效率。
該函數(shù)用于更新FMOD Ex的以下內(nèi)容:
· Platform specific routines 例如向PS2的IOP發(fā)送一個(gè)frame command packet。在這個(gè)平臺(tái)上,不調(diào)用update的話就聽不到聲音。
· Virtual voice emulation 不調(diào)用update,虛擬聲音就不會(huì)播放。
· 3D voice calculation 如果不調(diào)用update,就算channel或listener已經(jīng)正確設(shè)置,也無(wú)法聽到聲音移動(dòng)的3D音效。
· Geometry engine FMOD的polygon/geometry引擎需要通過update來(lái)啟用。否則用戶定義的occlusion/obstruction特性將無(wú)法呈現(xiàn)。
· Non realtime output FMOD_OUTPUTTYPE_NOSOUND_NRT和FMOD_OUTPUTTYPE_WAVWRITER_NRT標(biāo)記需要此函數(shù)才能更新到輸出(如用FMOD_OUTPUTTYPE_WAVWRITER_NRT寫出到文件)。
· Streaming engine 如果指定了FMOD_INIT_STREAM_FROM_UPDATE標(biāo)記,如果用戶希望在主線程中自己驅(qū)動(dòng)流引擎,就必須有規(guī)律地調(diào)用update,否則會(huì)導(dǎo)致抖動(dòng)和緩沖溢出。
關(guān)閉
調(diào)用System::release函數(shù)來(lái)關(guān)閉輸出設(shè)備并釋放對(duì)象關(guān)聯(lián)的內(nèi)存。
你不必人工關(guān)閉channel和sound,這些都在System::release中自動(dòng)完成。
當(dāng)然,你也可以手動(dòng)關(guān)閉它們,這是個(gè)很好的編程練習(xí)(雖然是多余的)。
如果你要釋放system對(duì)象就不需要調(diào)用System::close函數(shù)了,在System::release中已經(jīng)包含了對(duì)System::close的調(diào)用。
資源使用配制
在程序開發(fā)中,一些開發(fā)人員希望和其它程序一樣用自己的函數(shù)來(lái)訪問所有的磁盤或內(nèi)存。
在FMOD Ex中你可以通過System::setFileSystem函數(shù)來(lái)設(shè)置FMOD文件系統(tǒng)使你自己的文件程序。
要讓FMOD使用你的內(nèi)存系統(tǒng),或?qū)MOD限制在一個(gè)內(nèi)存塊中,使用Memory_Initialize。
注意! 在Xbox和XBox 360中,必須給FMOD提供一個(gè)塊內(nèi)存。Xbox 360上必須使用XPhysicalAlloc來(lái)分配這塊內(nèi)存。更多參見"Platform specific issues"。