/* * Copyright (c) 2006, Creative Labs Inc. * All rights reserved. * * Redistribution and use in source and binary forms, with or without modification, are permitted provided * that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, this list of conditions and * the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, this list of conditions * and the following disclaimer in the documentation and/or other materials provided with the distribution. * * Neither the name of Creative Labs Inc. nor the names of its contributors may be used to endorse or * promote products derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A * PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE * POSSIBILITY OF SUCH DAMAGE. */ #include "stdneb.h" #include "aldlist.h" //#include #include "OpenAL/alc.h" #ifdef __OSX__ #define _stricmp strcmp #endif /* * Init call */ ALDeviceList::ALDeviceList() { ALDEVICEINFO ALDeviceInfo; char *devices; int index; const char *defaultDeviceName; const char *actualDeviceName; // DeviceInfo vector stores, for each enumerated device, it's device name, selection status, spec version #, and extension support vDeviceInfo.Clear(); vDeviceInfo.Reserve(10); defaultDeviceIndex = 0; // grab function pointers for 1.0-API functions, and if successful proceed to enumerate all devices #define TRUE 1 if (LoadOAL10Library(NULL, &ALFunction) == TRUE) { if (ALFunction.alcIsExtensionPresent(NULL, "ALC_ENUMERATION_EXT")) { devices = (char *)ALFunction.alcGetString(NULL, ALC_DEVICE_SPECIFIER); defaultDeviceName = (char *)ALFunction.alcGetString(NULL, ALC_DEFAULT_DEVICE_SPECIFIER); index = 0; // go through device list (each device terminated with a single NULL, list terminated with double NULL) while (*devices != NULL) { if (strcmp(defaultDeviceName, devices) == 0) { defaultDeviceIndex = index; } ALCdevice *device = ALFunction.alcOpenDevice(devices); if (device) { ALCcontext *context = ALFunction.alcCreateContext(device, NULL); if (context) { ALFunction.alcMakeContextCurrent(context); // if new actual device name isn't already in the list, then add it... actualDeviceName = ALFunction.alcGetString(device, ALC_DEVICE_SPECIFIER); bool bNewName = true; for (int i = 0; i < GetNumDevices(); i++) { if (strcmp(GetDeviceName(i), actualDeviceName) == 0) { bNewName = false; } } if ((bNewName) && (actualDeviceName != NULL) && (strlen(actualDeviceName) > 0)) { memset(&ALDeviceInfo, 0, sizeof(ALDEVICEINFO)); ALDeviceInfo.bSelected = true; ALDeviceInfo.strDeviceName = actualDeviceName; ALFunction.alcGetIntegerv(device, ALC_MAJOR_VERSION, sizeof(int), &ALDeviceInfo.iMajorVersion); ALFunction.alcGetIntegerv(device, ALC_MINOR_VERSION, sizeof(int), &ALDeviceInfo.iMinorVersion); ALDeviceInfo.pvstrExtensions = new Util::Array(); // Check for ALC Extensions if (ALFunction.alcIsExtensionPresent(device, "ALC_EXT_CAPTURE") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("ALC_EXT_CAPTURE"); if (ALFunction.alcIsExtensionPresent(device, "ALC_EXT_EFX") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("ALC_EXT_EFX"); // Check for AL Extensions if (ALFunction.alIsExtensionPresent("AL_EXT_OFFSET") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("AL_EXT_OFFSET"); if (ALFunction.alIsExtensionPresent("AL_EXT_LINEAR_DISTANCE") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("AL_EXT_LINEAR_DISTANCE"); if (ALFunction.alIsExtensionPresent("AL_EXT_EXPONENT_DISTANCE") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("AL_EXT_EXPONENT_DISTANCE"); if (ALFunction.alIsExtensionPresent("EAX2.0") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("EAX2.0"); if (ALFunction.alIsExtensionPresent("EAX3.0") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("EAX3.0"); if (ALFunction.alIsExtensionPresent("EAX4.0") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("EAX4.0"); if (ALFunction.alIsExtensionPresent("EAX5.0") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("EAX5.0"); if (ALFunction.alIsExtensionPresent("EAX-RAM") == AL_TRUE) ALDeviceInfo.pvstrExtensions->Append("EAX-RAM"); // Get Source Count ALDeviceInfo.uiSourceCount = GetMaxNumSources(); vDeviceInfo.Append(ALDeviceInfo); } ALFunction.alcMakeContextCurrent(NULL); ALFunction.alcDestroyContext(context); } ALFunction.alcCloseDevice(device); } devices += strlen(devices) + 1; index += 1; } } } ResetFilters(); } /* * Exit call */ ALDeviceList::~ALDeviceList() { for (SizeT i = 0; i < vDeviceInfo.Size(); i++) { if (vDeviceInfo[i].pvstrExtensions) { vDeviceInfo[i].pvstrExtensions->Clear(); delete vDeviceInfo[i].pvstrExtensions; } } vDeviceInfo.Clear(); UnloadOAL10Library(); } /* * Returns the number of devices in the complete device list */ int ALDeviceList::GetNumDevices() { return (int)vDeviceInfo.Size(); } /* * Returns the device name at an index in the complete device list */ char * ALDeviceList::GetDeviceName(int index) { if (index < GetNumDevices()) return (char *)vDeviceInfo[index].strDeviceName.AsCharPtr(); else return NULL; } /* * Returns the major and minor version numbers for a device at a specified index in the complete list */ void ALDeviceList::GetDeviceVersion(int index, int *major, int *minor) { if (index < GetNumDevices()) { if (major) *major = vDeviceInfo[index].iMajorVersion; if (minor) *minor = vDeviceInfo[index].iMinorVersion; } return; } /* * Returns the maximum number of Sources that can be generate on the given device */ unsigned int ALDeviceList::GetMaxNumSources(int index) { if (index < GetNumDevices()) return vDeviceInfo[index].uiSourceCount; else return 0; } /* * Checks if the extension is supported on the given device */ bool ALDeviceList::IsExtensionSupported(int index, char *szExtName) { bool bReturn = false; if (index < GetNumDevices()) { for (SizeT i = 0; i < vDeviceInfo[index].pvstrExtensions->Size(); i++) { if (! n_stricmp (vDeviceInfo[index].pvstrExtensions->operator[](i).AsCharPtr(), szExtName)) { bReturn = true; break; } } } return bReturn; } /* * returns the index of the default device in the complete device list */ int ALDeviceList::GetDefaultDevice() { return defaultDeviceIndex; } /* * Deselects devices which don't have the specified minimum version */ void ALDeviceList::FilterDevicesMinVer(int major, int minor) { int dMajor, dMinor; for (SizeT i = 0; i < vDeviceInfo.Size(); i++) { GetDeviceVersion(i, &dMajor, &dMinor); if ((dMajor < major) || ((dMajor == major) && (dMinor < minor))) { vDeviceInfo[i].bSelected = false; } } } /* * Deselects devices which don't have the specified maximum version */ void ALDeviceList::FilterDevicesMaxVer(int major, int minor) { int dMajor, dMinor; for (SizeT i = 0; i < vDeviceInfo.Size(); i++) { GetDeviceVersion(i, &dMajor, &dMinor); if ((dMajor > major) || ((dMajor == major) && (dMinor > minor))) { vDeviceInfo[i].bSelected = false; } } } /* * Deselects device which don't support the given extension name */ void ALDeviceList::FilterDevicesExtension(char *szExtName) { bool bFound; for (SizeT i = 0; i < vDeviceInfo.Size(); i++) { bFound = false; for (SizeT j = 0; j < vDeviceInfo[i].pvstrExtensions->Size(); j++) { if (!n_stricmp(vDeviceInfo[i].pvstrExtensions->operator[](i).AsCharPtr(), szExtName)) { bFound = true; break; } } if (!bFound) vDeviceInfo[i].bSelected = false; } } /* * Resets all filtering, such that all devices are in the list */ void ALDeviceList::ResetFilters() { for (int i = 0; i < GetNumDevices(); i++) { vDeviceInfo[i].bSelected = true; } filterIndex = 0; } /* * Gets index of first filtered device */ int ALDeviceList::GetFirstFilteredDevice() { int i; for (i = 0; i < GetNumDevices(); i++) { if (vDeviceInfo[i].bSelected == true) { break; } } filterIndex = i + 1; return i; } /* * Gets index of next filtered device */ int ALDeviceList::GetNextFilteredDevice() { int i; for (i = filterIndex; i < GetNumDevices(); i++) { if (vDeviceInfo[i].bSelected == true) { break; } } filterIndex = i + 1; return i; } /* * Internal function to detemine max number of Sources that can be generated */ unsigned int ALDeviceList::GetMaxNumSources() { ALuint uiSources[256]; unsigned int iSourceCount = 0; // Clear AL Error Code ALFunction.alGetError(); // Generate up to 256 Sources, checking for any errors for (iSourceCount = 0; iSourceCount < 256; iSourceCount++) { ALFunction.alGenSources(1, &uiSources[iSourceCount]); if (ALFunction.alGetError() != AL_NO_ERROR) break; } // Release the Sources ALFunction.alDeleteSources(iSourceCount, uiSources); if (ALFunction.alGetError() != AL_NO_ERROR) { for (unsigned int i = 0; i < 256; i++) { ALFunction.alDeleteSources(1, &uiSources[i]); } } return iSourceCount; }