CMAudio/src/OpenAL.cpp

739 lines
22 KiB
C++
Raw Normal View History

2020-06-18 03:38:48 +08:00
#include<hgl/log/LogInfo.h>
#include<hgl/filesystem/FileSystem.h>
#include<hgl/audio/OpenAL.h>
#include<hgl/type/Pair.h>
#include<hgl/type/Stack.h>
#include<hgl/type/String.h>
2020-06-18 03:38:48 +08:00
#include<hgl/type/StringList.h>
#include<hgl/platform/ExternalModule.h>
using namespace hgl;
namespace openal
{
static ALCchar AudioDeviceName[AL_DEVICE_NAME_MAX_LEN]={0};
static ExternalModule *AudioEM =nullptr; //OpenAL动态链接库指针
static ALCdevice *AudioDevice =nullptr; //OpenAL设备
static ALCcontext *AudioContext =nullptr; //OpenAL上下文
static UTF8StringList OpenALExt_List;
static UTF8StringList OpenALContextExt_List;
static bool AudioFloat32 =false; //是否支持float 32数据
static bool AudioEFX =false; //EFX是否可用
static bool AudioXRAM =false; //X-RAM是否可用
bool LoadALCFunc(ExternalModule *);
bool LoadALFunc(ExternalModule *);
bool CheckXRAM(ALCdevice_struct *);
bool CheckEFX(ALCdevice_struct *);
void ClearAL();
void ClearALC();
void ClearXRAM();
void ClearEFX();
void PutOpenALInfo();
//--------------------------------------------------------------------------------------------------
bool OnFindedOpenALDynamicLinkLibrary(const OSString &filename,void *user_data,bool exist)
{
if(!exist)return(true);
AudioEM=LoadExternalModule(filename);
if(!AudioEM)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Finded a OpenAL dynamic link library, but it can't load. filename: ")+OSString(filename));
return(true);
}
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Loading of openAL dynamic link library successfully, filename: ")+OSString(filename));
return(false);
}
//--------------------------------------------------------------------------------------------------
2020-06-18 03:38:48 +08:00
/**
* OpenAL
* @return OpenAL是否成功
*/
bool LoadOpenAL(const os_char *filename)
{
OSStringList oalpn;
OSStringList oalfn=
2020-06-18 03:38:48 +08:00
{
#if HGL_OS == HGL_OS_Windows
OS_TEXT("OpenAL32.dll"),
OS_TEXT("ct_oal.dll"),
OS_TEXT("nvOpenAL.dll"),
OS_TEXT("soft_oal.dll"),
OS_TEXT("wrap_oal.dll"),
2020-06-18 03:38:48 +08:00
#elif (HGL_OS == HGL_OS_Linux)||(HGL_OS == HGL_OS_FreeBSD)||(HGL_OS == HGL_OS_NetBSD)||(HGL_OS == HGL_OS_OpenBSD)
OS_TEXT("libopenal.so"),
OS_TEXT("libopenal.so.1"),
2020-06-18 03:38:48 +08:00
#elif HGL_OS == HGL_OS_MacOS
OS_TEXT("libopenal.dylib"),
2020-06-18 03:38:48 +08:00
#endif//HGL_OS == HGL_OS_Windows
};
int count=0;
os_char *final_filename=nullptr;
{
OSString pn;
2020-06-18 03:38:48 +08:00
filesystem::GetCurrentPath(pn);
oalpn.Add(pn);
2020-06-18 03:38:48 +08:00
pn=filesystem::MergeFilename(pn,OS_TEXT("Plug-Ins"));
if(filesystem::IsDirectory(pn))
oalpn.Add(pn);
2020-06-18 03:38:48 +08:00
filesystem::GetOSLibararyPath(pn);
oalpn.Add(pn);
2020-06-18 03:38:48 +08:00
}
CloseOpenAL();
2020-06-18 03:38:48 +08:00
if(filename)
{
filesystem::FindFileOnPaths(filename,oalpn,nullptr,OnFindedOpenALDynamicLinkLibrary);
}
else
{
filesystem::FindFileOnPaths(oalfn,oalpn,nullptr,OnFindedOpenALDynamicLinkLibrary);
2020-06-18 03:38:48 +08:00
}
if(AudioEM)
{
return true;
2020-06-18 03:38:48 +08:00
}
else
{
2022-03-31 19:00:28 +08:00
LOG_ERROR( OS_TEXT("The OpenAL Dynamic link library was not found, perhaps the loading failed..\n")
OS_TEXT("you can download it on \"http://www.openal.org\" or \"https://openal-soft.org\",\n")
OS_TEXT("Downloading the driver driver installation from the official website of the sound card is also a good choice."));
return false;
2020-06-18 03:38:48 +08:00
}
}
const char *alcGetDeviceNameList()
{
if(!alcIsExtensionPresent)return(nullptr);
if(alcIsExtensionPresent(nullptr,"ALC_ENUMERATE_ALL_EXT"))
return (char *)alcGetString(nullptr,ALC_ALL_DEVICES_SPECIFIER);
else
if(alcIsExtensionPresent(nullptr,"ALC_ENUMERATION_EXT"))
return (char *)alcGetString(nullptr,ALC_DEVICE_SPECIFIER);
else
return(nullptr);
}
bool alcGetDefaultDeviceName(char *name)
{
if(alcIsExtensionPresent)
{
const char *result=nullptr;
if(alcIsExtensionPresent(nullptr,"ALC_ENUMERATE_ALL_EXT"))
result=(char *)alcGetString(nullptr,ALC_DEFAULT_ALL_DEVICES_SPECIFIER);
else
if(alcIsExtensionPresent(nullptr,"ALC_ENUMERATION_EXT"))
result=(char *)alcGetString(nullptr,ALC_DEFAULT_DEVICE_SPECIFIER);
if(result&&(*result))
{
hgl::strcpy(name,AL_DEVICE_NAME_MAX_LEN,result);
return(true);
}
}
*name=0;
return(false);
}
bool alcGetDeviceInfo(OpenALDevice &dev,const char *device_name)
{
ALCdevice *device=alcOpenDevice(device_name);
if(!device)
return(false);
hgl::strcpy(dev.name, AL_DEVICE_NAME_MAX_LEN,device_name);
hgl::strcpy(dev.specifier, AL_DEVICE_NAME_MAX_LEN,alcGetString(device,ALC_DEVICE_SPECIFIER));
alcGetIntegerv(device,ALC_MAJOR_VERSION,sizeof(int),&dev.major);
if(dev.major)
{
alcGetIntegerv(device,ALC_MINOR_VERSION,sizeof(int),&dev.minor);
}
else
{
dev.major=1;
dev.minor=0;
}
alcCloseDevice(device);
return(true);
}
bool alcGetDefaultDevice(OpenALDevice &dev)
{
if(!alcGetDefaultDeviceName(dev.name))
return(false);
return alcGetDeviceInfo(dev,dev.name);
}
int alcGetDeviceList(List<OpenALDevice> &device_list)
{
const char *devices=alcGetDeviceNameList();
if(!devices)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Can't get list of OpenAL Devices."));
2020-06-18 03:38:48 +08:00
return(-1);
}
while(*devices)
{
OpenALDevice dev;
if(alcGetDeviceInfo(dev,devices))
device_list.Add(dev);
devices+=::strlen(devices)+1;
}
return device_list.GetCount();
}
void EnumOpenALDevice()
{
const char *devices=alcGetDeviceNameList();
const char *actual_devicename;
if(!devices)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("can't get list of OpenAL Devices."));
2020-06-18 03:38:48 +08:00
return;
}
while(*devices)
{
ALCdevice *device=alcOpenDevice(devices);
if(device)
{
actual_devicename=alcGetString(device,ALC_DEVICE_SPECIFIER);
int major=0,minor=0;
alcGetIntegerv(device,ALC_MAJOR_VERSION,sizeof(int),&major);
if(major)
{
alcGetIntegerv(device,ALC_MINOR_VERSION,sizeof(int),&minor);
2022-03-31 19:00:28 +08:00
LOG_INFO(u8"OpenAL device: "+UTF8String(devices)+u8" Specifier:"+UTF8String(actual_devicename)+u8", version: "+UTF8String::valueOf(major)+"."+UTF8String::valueOf(minor));
2020-06-18 03:38:48 +08:00
}
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(u8"OpenAL device: "+UTF8String(devices)+u8" Specifier:"+UTF8String(actual_devicename)+u8",can't get version, used OpenAL 1.0.");
2020-06-18 03:38:48 +08:00
}
alcCloseDevice(device);
}
devices+=::strlen(devices)+1;
}
}
void alcSetDefaultContext()
{
alcMakeContextCurrent(AudioContext);
}
void GetOpenALExt(UTF8StringList &ext_list,const char *ext_str)
{
int len=hgl::strlen(ext_str);
SplitToStringListBySpace(ext_list,ext_str,len); //space指所有不可打印字符
}
void InitOpenALExt()
{
GetOpenALExt(OpenALExt_List,alGetString(AL_EXTENSIONS));
GetOpenALExt(OpenALContextExt_List,alcGetString(AudioDevice,ALC_EXTENSIONS));
if(OpenALExt_List.Find("AL_EXT_FLOAT32")!=-1)
AudioFloat32=true;
}
/**
* OpenAL驱动
* @param driver_name ,
* @return
*/
bool InitOpenALDriver(const os_char *driver_name)
{
if (!AudioEM)
if (!LoadOpenAL(driver_name))RETURN_FALSE;
RETURN_BOOL(LoadALCFunc(AudioEM));
}
bool InitOpenALDeviceByName(const char *device_name)
{
bool default_device=false;
if(device_name)
hgl::strcpy(AudioDeviceName,AL_DEVICE_NAME_MAX_LEN,device_name);
else
{
alcGetDefaultDeviceName(AudioDeviceName);
if(*AudioDeviceName)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(u8"Select the default OpenAL device: "+UTF8String(AudioDeviceName));
2020-06-18 03:38:48 +08:00
}
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Select the default OpenAL device."));
2020-06-18 03:38:48 +08:00
}
default_device=true;
}
AudioDevice=alcOpenDevice(AudioDeviceName);
if(AudioDevice==nullptr)
{
2022-03-31 19:00:28 +08:00
LOG_ERROR(u8"Failed to OpenAL Device: "+UTF8String(AudioDeviceName));
2020-06-18 03:38:48 +08:00
if(default_device)
{
CloseOpenAL();
return(AL_FALSE);
}
//使用缺省设备
{
alcGetDefaultDeviceName(AudioDeviceName);
AudioDevice=alcOpenDevice(AudioDeviceName);
if(!AudioDevice)
{
2022-03-31 19:00:28 +08:00
LOG_ERROR(OS_TEXT("The OpenAL device cannot be initialized properly using the specified device and the default device!"));
2020-06-18 03:38:48 +08:00
CloseOpenAL();
return(AL_FALSE);
}
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Select the default OpenAL device because the specified device is failing."));
2020-06-18 03:38:48 +08:00
}
}
2022-03-31 19:00:28 +08:00
LOG_INFO(U8_TEXT("Opened OpenAL Device")+UTF8String(AudioDeviceName));
2020-06-18 03:38:48 +08:00
AudioContext=alcCreateContext(AudioDevice,0);
if(AudioContext==nullptr)
{
2022-03-31 19:00:28 +08:00
LOG_ERROR(OS_TEXT("Create OpenAL Context OK."));
2020-06-18 03:38:48 +08:00
CloseOpenAL();
return (AL_FALSE);
}
#ifdef _DEBUG
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Failed to Create OpenAL Context."));
2020-06-18 03:38:48 +08:00
}
#endif//_DEBUG
if(!alcMakeContextCurrent(AudioContext))
{
2022-03-31 19:00:28 +08:00
LOG_ERROR(OS_TEXT("Failed to Make OpenAL Context"));
2020-06-18 03:38:48 +08:00
CloseOpenAL();
RETURN_FALSE;
}
#ifdef _DEBUG
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Make OpenAL Context OK."));
2020-06-18 03:38:48 +08:00
}
#endif//_DEBUG
hgl::strcpy(AudioDeviceName,AL_DEVICE_NAME_MAX_LEN,alcGetString(AudioDevice,ALC_DEVICE_SPECIFIER));
2022-03-31 19:00:28 +08:00
LOG_INFO(u8"Initing OpenAL Device: "+UTF8String(AudioDeviceName));
2020-06-18 03:38:48 +08:00
if (!LoadALFunc(AudioEM))
{
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Failed to load OpenAL functions"));
2020-06-18 03:38:48 +08:00
CloseOpenAL();
RETURN_FALSE;
}
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Loaded OpenAL functions."));
2020-06-18 03:38:48 +08:00
if(alcIsExtensionPresent)
{
AudioXRAM=CheckXRAM(AudioDevice);
AudioEFX=CheckEFX(AudioDevice);
}
else
{
AudioXRAM=false;
AudioEFX=false;
}
alGetError();
InitOpenALExt();
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Inited OpenAL."));
2020-06-18 03:38:48 +08:00
return(AL_TRUE);
}
bool InitOpenALDevice(const OpenALDevice *device)
{
if(device)
return InitOpenALDeviceByName(device->name);
else
return InitOpenALDeviceByName(nullptr);
}
//--------------------------------------------------------------------------------------------------
/**
* OpenAL
*/
void CloseOpenAL()
{
if(AudioEM)
{
alcMakeContextCurrent(0);
if(AudioContext) alcDestroyContext(AudioContext);
if(AudioDevice) alcCloseDevice(AudioDevice);
SAFE_CLEAR(AudioEM);
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("Close OpenAL."));
2020-06-18 03:38:48 +08:00
}
ClearAL();
ClearALC();
ClearXRAM();
ClearEFX();
}
/**
* ()
*/
bool SetDistanceModel(ALenum distance_model)
{
if (!AudioEM)return(false);
if (!alDistanceModel)return(false);
if (distance_model<AL_INVERSE_DISTANCE
||distance_model>AL_EXPONENT_DISTANCE_CLAMPED)
return(false);
alDistanceModel(distance_model);
return(true);
}
bool SetSpeedOfSound(const float ss)
{
if (!AudioEM)return(false);
if (!alSpeedOfSound)return(false);
alSpeedOfSound(ss);
return(true);
}
double SetSpeedOfSound(const double height,const double temperature,const double humidity)
{
if (!AudioEM)return(0);
if (!alSpeedOfSound)return(0);
double v=HGL_SPEED_OF_SOUND; //海拔0米0摄氏度
v*=sqrt(1.0f+temperature/(-HGL_ABSOLUTE_ZERO)); //先计算温度
////11000米高度温度-56.5音速295m/s。需根据大气密度来计算
//另需计算温度影响在0.1-0.6之间。
alSpeedOfSound(v/HGL_SPEED_OF_SOUND);
return(v);
}
bool SetDopplerFactor(const float df)
{
if (!AudioEM)return(false);
if (!alDopplerFactor)return(false);
alDopplerFactor(df);
return(true);
}
bool SetDopplerVelocity(const float dv)
{
if (!AudioEM)return(false);
if (!alDopplerVelocity)return(false);
alDopplerVelocity(dv);
return(true);
}
//--------------------------------------------------------------------------------------------------
// void *alcGetCurrentDevice() { return (AudioDevice); }
// char *alcGetCurrentDeviceName() { return(AudioDeviceName); }
const struct ALFormatBytes
{
ALenum format;
uint bytes;
}
al_format_bytes[]=
{
{AL_FORMAT_MONO8, 1},
{AL_FORMAT_MONO16, 2},
{AL_FORMAT_STEREO8, 2},
{AL_FORMAT_STEREO16, 4},
// {AL_FORMAT_QUAD16, 8},
// {AL_FORMAT_51CHN16, 12},
// {AL_FORMAT_61CHN16, 14},
// {AL_FORMAT_71CHN16, 16},
//
{AL_FORMAT_MONO_FLOAT32, 4},
{AL_FORMAT_STEREO_FLOAT32, 8},
//
// {AL_FORMAT_QUAD8, 4},
// {AL_FORMAT_QUAD16, 8},
// {AL_FORMAT_QUAD32, 16},
// {AL_FORMAT_REAR8, 4},
// {AL_FORMAT_REAR16, 8},
// {AL_FORMAT_REAR32, 16},
// {AL_FORMAT_51CHN8, 6},
// {AL_FORMAT_51CHN16, 12},
// {AL_FORMAT_51CHN32, 24},
// {AL_FORMAT_61CHN8, 7},
// {AL_FORMAT_61CHN16, 14},
// {AL_FORMAT_61CHN32, 28},
// {AL_FORMAT_71CHN8, 8},
// {AL_FORMAT_71CHN16, 16},
// {AL_FORMAT_71CHN32, 32},
{0,0}
};
const int al_get_format_byte(ALenum format)
{
const ALFormatBytes *p=al_format_bytes;
while(p->format)
{
if(p->format==format)
return p->bytes;
++p;
}
return(0);
}
/**
*
* @param size
* @param format
* @param freq
* @return ()
*/
double AudioDataTime(ALuint size,ALenum format,ALsizei freq)
{
if(size==0||freq==0)return(0);
const uint byte=al_get_format_byte(format);
if(byte==0)
return(0);
return (double(size)/double(freq)/double(byte));
}
/**
*
* @param format
* @param freq
* @return ()
*/
unsigned int AudioTime(ALenum format,ALsizei freq)
{
const uint byte=al_get_format_byte(format);
if(byte==0)
return(0);
return byte*freq;
}
unsigned int GetMaxNumSources()
{
ALuint source;
Stack<ALuint> source_stack;
// Clear AL Error Code
alGetError();
for(;;)
{
alGenSources(1, &source);
if (alGetError() != AL_NO_ERROR)
break;
source_stack.Push(source);
}
const int count=source_stack.GetCount();
const ALuint *p=source_stack.GetData();
alDeleteSources(count,p);
if (alGetError() != AL_NO_ERROR)
{
for(int i=0;i<count;i++)
alDeleteSources(1, p++);
}
return count;
}
/**
*
*/
bool IsSupportFloatAudioData()
{
return AudioFloat32;
}
//--------------------------------------------------------------------------------------------------
const char *alGetErrorInfo(const char *filename,const int line)
{
if(!alGetError)return(U8_TEXT("OpenALEE/OpenAL 未初始化!"));
const char *result=nullptr;
const char al_error_invalid []=U8_TEXT("invalid");
const char al_error_invalid_name []=U8_TEXT("invalid name");
const char al_error_invalid_enum []=U8_TEXT("invalid enum");
const char al_error_invalid_value []=U8_TEXT("invalid value");
const char al_error_invalid_operation []=U8_TEXT("invalid operation");
const char al_error_out_of_memory []=U8_TEXT("out of memory");
const char al_error_unknown_error []=U8_TEXT("unknown error");
const ALenum error=alGetError();
if(error==AL_NO_ERROR )result=nullptr;else
if(error==AL_INVALID )result=al_error_invalid;else
if(error==AL_INVALID_NAME )result=al_error_invalid_name;else
if(error==AL_INVALID_ENUM )result=al_error_invalid_enum;else
if(error==AL_INVALID_VALUE )result=al_error_invalid_value;else
if(error==AL_INVALID_OPERATION )result=al_error_invalid_operation;else
if(error==AL_OUT_OF_MEMORY )result=al_error_out_of_memory;else
result=al_error_unknown_error;
if(result)
{
2022-03-31 19:00:28 +08:00
LOG_ERROR(U8_TEXT("OpenAL error,source file:\"")+UTF8String(filename)+u8"\",line:"+UTF8String::valueOf(line));
2020-06-18 03:38:48 +08:00
LOG_ERROR(U8_TEXT("OpenAL ErrorNo:")+UTF8String(result));
}
return(result);
}
void PutOpenALInfo()
{
if(!alGetString)return;
2022-03-31 19:00:28 +08:00
LOG_INFO(UTF8String(u8" OpenAL Vendor: ")+alGetString(AL_VENDOR ));
LOG_INFO(UTF8String(u8" OpenAL Version: ")+alGetString(AL_VERSION ));
LOG_INFO(UTF8String(u8" OpenAL Renderer: ")+alGetString(AL_RENDERER ));
LOG_INFO(OS_TEXT( " Max audio sources: ")+OSString::valueOf(GetMaxNumSources()));
LOG_INFO(AudioFloat32?OS_TEXT( " Supported float data: Yes")
:OS_TEXT( " Supported float data: No"));
2020-06-18 03:38:48 +08:00
2022-03-31 19:00:28 +08:00
LOG_INFO(AudioEFX?OS_TEXT( " EFX: Supported")
:OS_TEXT( " EFX: No"));
LOG_INFO(AudioXRAM?OS_TEXT( " X-RAM: Supported")
:OS_TEXT( " X-RAM: No"));
2020-06-18 03:38:48 +08:00
if(AudioXRAM)
{
int size;
alcGetIntegerv(AudioDevice,eXRAMSize,sizeof(int),&size);
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT( " X-RAM Volume: ")+OSString::valueOf(size));
2020-06-18 03:38:48 +08:00
}
for(int i=0;i<OpenALExt_List.GetCount();i++)
if(i==0)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(UTF8String(u8" OpenAL Extentsion: ")+OpenALExt_List[i]);
2020-06-18 03:38:48 +08:00
}
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(UTF8String(u8" ")+OpenALExt_List[i]);
2020-06-18 03:38:48 +08:00
}
for(int i=0;i<OpenALContextExt_List.GetCount();i++)
if(i==0)
{
2022-03-31 19:00:28 +08:00
LOG_INFO(UTF8String(u8"OpenAL Device Extension: ")+OpenALContextExt_List[i]);
2020-06-18 03:38:48 +08:00
}
else
{
2022-03-31 19:00:28 +08:00
LOG_INFO(UTF8String(u8" ")+OpenALContextExt_List[i]);
2020-06-18 03:38:48 +08:00
}
#ifdef _DEBUG
2022-03-31 19:00:28 +08:00
LOG_INFO(OS_TEXT("OpenAL Infomation finished."));
2020-06-18 03:38:48 +08:00
#endif//
}
/**
* OpenAL
* @param driver_name ,
* @param device_name ,
* @param out_info OpenAL设备信息
* @param enum_device
* @return
*/
bool InitOpenAL(const os_char *driver_name,const char *device_name,bool out_info,bool enum_device)
{
if (!InitOpenALDriver(driver_name))
return(false);
if (enum_device)
EnumOpenALDevice();
if (!InitOpenALDeviceByName(device_name))
{
CloseOpenAL();
return(false);
}
InitOpenALExt();
if(out_info)
PutOpenALInfo();
return(true);
}
}//namespace openal