|
|
|
/*==LICENSE==*
|
|
|
|
|
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools
|
|
|
|
Copyright (C) 2011 Cyan Worlds, Inc.
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com
|
|
|
|
or by snail mail at:
|
|
|
|
Cyan Worlds, Inc.
|
|
|
|
14617 N Newport Hwy
|
|
|
|
Mead, WA 99021
|
|
|
|
|
|
|
|
*==LICENSE==*/
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
// //
|
|
|
|
// plAudioCaps - Utility class to query and detect available audio options.//
|
|
|
|
// //
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
#include "HeadSpin.h"
|
|
|
|
#include <al.h>
|
|
|
|
#include <alc.h>
|
|
|
|
#include "plEAXEffects.h"
|
|
|
|
|
|
|
|
#include "plAudioCaps.h"
|
|
|
|
#ifdef EAX_SDK_AVAILABLE
|
|
|
|
#include <eax.h>
|
|
|
|
#include <eaxlegacy.h>
|
|
|
|
#endif
|
|
|
|
#include <DShow.h>
|
|
|
|
|
|
|
|
#include "plStatusLog/plStatusLog.h"
|
|
|
|
|
|
|
|
#define MAX_NUM_SOURCES 128
|
|
|
|
#define kLogMe if( fLog != nil ) fLog->AddLineF(
|
|
|
|
#define MAX_AUDIOCARD_NAME 256
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
//// Detector Functions //////////////////////////////////////////////////////
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
plAudioCaps plAudioCapsDetector::fCaps;
|
|
|
|
hsBool plAudioCapsDetector::fGotCaps = false;
|
|
|
|
|
|
|
|
plStatusLog *plAudioCapsDetector::fLog = nil;
|
|
|
|
|
|
|
|
plAudioCapsDetector::plAudioCapsDetector()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
plAudioCapsDetector::~plAudioCapsDetector()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
//// Detect //////////////////////////////////////////////////////////////////
|
|
|
|
// Our big function that does all of our work
|
|
|
|
|
|
|
|
plAudioCaps &plAudioCapsDetector::Detect( hsBool logIt, hsBool init )
|
|
|
|
{
|
|
|
|
// If we already have the device capabilities, just return them
|
|
|
|
if(fGotCaps) return fCaps;
|
|
|
|
fCaps.fIsAvailable = true;
|
|
|
|
|
|
|
|
ALCdevice * device;
|
|
|
|
ALCcontext * context;
|
|
|
|
|
|
|
|
if(init)
|
|
|
|
{
|
|
|
|
device = alcOpenDevice(0);
|
|
|
|
if(!device)
|
|
|
|
{
|
|
|
|
fCaps.fIsAvailable = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
context = alcCreateContext(device, 0);
|
|
|
|
if(alGetError() != AL_NO_ERROR)
|
|
|
|
{
|
|
|
|
fCaps.fIsAvailable = false;
|
|
|
|
}
|
|
|
|
alcMakeContextCurrent(context);
|
|
|
|
if(alGetError() != AL_NO_ERROR)
|
|
|
|
{
|
|
|
|
fCaps.fIsAvailable = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
EnumerateAudioDevices();
|
|
|
|
|
|
|
|
if( logIt )
|
|
|
|
fLog = plStatusLogMgr::GetInstance().CreateStatusLog( 30, "audioCaps.log" );
|
|
|
|
else
|
|
|
|
fLog = nil;
|
|
|
|
|
|
|
|
kLogMe 0xff00ff00, "Starting audio caps detection..." );
|
|
|
|
|
|
|
|
// find the max number of sources
|
|
|
|
ALuint sources[MAX_NUM_SOURCES];
|
|
|
|
ALuint i = 0;
|
|
|
|
for(; i < MAX_NUM_SOURCES; i++)
|
|
|
|
{
|
|
|
|
alGenSources(1, &sources[i]);
|
|
|
|
if(alGetError() != AL_NO_ERROR)
|
|
|
|
break;
|
|
|
|
fCaps.fMaxNumSources++;
|
|
|
|
}
|
|
|
|
alDeleteSources(i, sources);
|
|
|
|
kLogMe 0xffffffff, "Max Number of sources: %d", i);
|
|
|
|
plStatusLog::AddLineS("audio.log", "Max Number of sources: %d", i);
|
|
|
|
|
|
|
|
// Detect EAX support
|
|
|
|
kLogMe 0xff00ff00, "Attempting to detect EAX support..." );
|
|
|
|
fCaps.fEAXAvailable = IDetectEAX( );
|
|
|
|
|
|
|
|
kLogMe 0xff00ff00, "Audio caps detection COMPLETE." );
|
|
|
|
delete fLog;
|
|
|
|
|
|
|
|
fGotCaps = true; // We've got the device capabilities
|
|
|
|
|
|
|
|
if(init)
|
|
|
|
{
|
|
|
|
alcMakeContextCurrent(nil);
|
|
|
|
alcDestroyContext(context);
|
|
|
|
alcCloseDevice(device);
|
|
|
|
}
|
|
|
|
return fCaps;
|
|
|
|
}
|
|
|
|
|
|
|
|
void plAudioCapsDetector::EnumerateAudioDevices()
|
|
|
|
{
|
|
|
|
ICreateDevEnum *pDevEnum;
|
|
|
|
IEnumMoniker *pEnumMon;
|
|
|
|
IMoniker *pMoniker;
|
|
|
|
ULONG cFetched;
|
|
|
|
HRESULT hr;
|
|
|
|
char audioCardName[MAX_AUDIOCARD_NAME];
|
|
|
|
short *pShort;
|
|
|
|
|
|
|
|
// Enumerate audio devices
|
|
|
|
hr = CoCreateInstance(CLSID_SystemDeviceEnum, NULL, CLSCTX_INPROC_SERVER, IID_ICreateDevEnum, (void **)&pDevEnum);
|
|
|
|
if(SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
hr = pDevEnum->CreateClassEnumerator(CLSID_AudioRendererCategory, &pEnumMon, 0);
|
|
|
|
if(SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
while(pEnumMon->Next(1, &pMoniker, &cFetched) == S_OK)
|
|
|
|
{
|
|
|
|
if(pMoniker)
|
|
|
|
{
|
|
|
|
IPropertyBag *pPropBag;
|
|
|
|
hr = pMoniker->BindToStorage(0, 0, IID_IPropertyBag, (void **)&pPropBag);
|
|
|
|
if(SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
VARIANT varName;
|
|
|
|
VariantInit(&varName);
|
|
|
|
hr = pPropBag->Read(L"FriendlyName", &varName, 0);
|
|
|
|
memset(audioCardName, 0, MAX_AUDIOCARD_NAME);
|
|
|
|
pShort = varName.piVal;
|
|
|
|
|
|
|
|
// copy from wide character array to char array
|
|
|
|
for(int i = 0; *pShort != 0 && i < MAX_AUDIOCARD_NAME; pShort++, i++)
|
|
|
|
{
|
|
|
|
audioCardName[i] = (char)*pShort;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
plStatusLog::AddLineS("audiocaps.log", audioCardName );
|
|
|
|
}
|
|
|
|
VariantClear(&varName);
|
|
|
|
pPropBag->Release();
|
|
|
|
}
|
|
|
|
pMoniker->Release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pEnumMon->Release();
|
|
|
|
}
|
|
|
|
pDevEnum->Release();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//// IDetectEAX //////////////////////////////////////////////////////////////
|
|
|
|
// Attempt to actually init the EAX listener.Note that we can potentially do
|
|
|
|
// this even if we didn't load the EAX Unified driver (we could just be
|
|
|
|
// running EAX 3.0 native), so you can really just think of the EAX Unified
|
|
|
|
// init code above as a way of trying to make sure this line here will
|
|
|
|
// succeed as often as possible.
|
|
|
|
|
|
|
|
hsBool plAudioCapsDetector::IDetectEAX( )
|
|
|
|
{
|
|
|
|
#ifdef EAX_SDK_AVAILABLE
|
|
|
|
hsBool gotSupport = true;
|
|
|
|
|
|
|
|
if(!alIsExtensionPresent((ALchar *)"EAX4.0")) // is eax 4 supported
|
|
|
|
{
|
|
|
|
if(!alIsExtensionPresent((ALchar *) "EAX4.0Emulated")) // is an earlier version of eax supported
|
|
|
|
{
|
|
|
|
kLogMe 0xff00ff00, "EAX not supported");
|
|
|
|
gotSupport = false;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fCaps.fEAXUnified = true;
|
|
|
|
kLogMe 0xff00ff00, "EAX 4 Emulated supported");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
kLogMe 0xff00ff00, "EAX 4 available");
|
|
|
|
}
|
|
|
|
return gotSupport;
|
|
|
|
#else
|
|
|
|
kLogMe 0xff00ff00, "EAX disabled in this build");
|
|
|
|
return false;
|
|
|
|
#endif
|
|
|
|
}
|