#include <dlfcn.h>
#include "common/OboeDebug.h"
#include "EngineOpenSLES.h"
#include "OpenSLESUtilities.h"
using namespace oboe;
#define LIB_OPENSLES_NAME "libOpenSLES.so"
typedef SLresult (*prototype_slCreateEngine)(
SLObjectItf *pEngine,
SLuint32 numOptions,
const SLEngineOption *pEngineOptions,
SLuint32 numInterfaces,
const SLInterfaceID *pInterfaceIds,
const SLboolean *pInterfaceRequired
);
static prototype_slCreateEngine gFunction_slCreateEngine = nullptr;
static void *gLibOpenSlesLibraryHandle = nullptr;
static bool linkOpenSLES() {
if (gLibOpenSlesLibraryHandle == nullptr && gFunction_slCreateEngine == nullptr) {
gLibOpenSlesLibraryHandle = dlopen(LIB_OPENSLES_NAME, RTLD_NOW);
if (gLibOpenSlesLibraryHandle == nullptr) {
LOGE("linkOpenSLES() could not find " LIB_OPENSLES_NAME);
} else {
gFunction_slCreateEngine = (prototype_slCreateEngine) dlsym(
gLibOpenSlesLibraryHandle,
"slCreateEngine");
LOGD("linkOpenSLES(): dlsym(%s) returned %p", "slCreateEngine",
gFunction_slCreateEngine);
}
}
return gFunction_slCreateEngine != nullptr;
}
EngineOpenSLES &EngineOpenSLES::getInstance() {
static EngineOpenSLES sInstance;
return sInstance;
}
SLresult EngineOpenSLES::open() {
std::lock_guard<std::mutex> lock(mLock);
SLresult result = SL_RESULT_SUCCESS;
if (mOpenCount++ == 0) {
if (!linkOpenSLES()) {
result = SL_RESULT_FEATURE_UNSUPPORTED;
goto error;
};
result = (*gFunction_slCreateEngine)(&mEngineObject, 0, NULL, 0, NULL, NULL);
if (SL_RESULT_SUCCESS != result) {
LOGE("EngineOpenSLES - slCreateEngine() result:%s", getSLErrStr(result));
goto error;
}
result = (*mEngineObject)->Realize(mEngineObject, SL_BOOLEAN_FALSE);
if (SL_RESULT_SUCCESS != result) {
LOGE("EngineOpenSLES - Realize() engine result:%s", getSLErrStr(result));
goto error;
}
result = (*mEngineObject)->GetInterface(mEngineObject, SL_IID_ENGINE, &mEngineInterface);
if (SL_RESULT_SUCCESS != result) {
LOGE("EngineOpenSLES - GetInterface() engine result:%s", getSLErrStr(result));
goto error;
}
}
return result;
error:
close();
return result;
}
void EngineOpenSLES::close() {
std::lock_guard<std::mutex> lock(mLock);
if (--mOpenCount == 0) {
if (mEngineObject != nullptr) {
(*mEngineObject)->Destroy(mEngineObject);
mEngineObject = nullptr;
mEngineInterface = nullptr;
}
}
}
SLresult EngineOpenSLES::createOutputMix(SLObjectItf *objectItf) {
return (*mEngineInterface)->CreateOutputMix(mEngineInterface, objectItf, 0, 0, 0);
}
SLresult EngineOpenSLES::createAudioPlayer(SLObjectItf *objectItf,
SLDataSource *audioSource,
SLDataSink *audioSink) {
const SLInterfaceID ids[] = {SL_IID_BUFFERQUEUE, SL_IID_ANDROIDCONFIGURATION};
const SLboolean reqs[] = {SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE};
return (*mEngineInterface)->CreateAudioPlayer(mEngineInterface, objectItf, audioSource,
audioSink,
sizeof(ids) / sizeof(ids[0]), ids, reqs);
}
SLresult EngineOpenSLES::createAudioRecorder(SLObjectItf *objectItf,
SLDataSource *audioSource,
SLDataSink *audioSink) {
const SLInterfaceID ids[] = {SL_IID_ANDROIDSIMPLEBUFFERQUEUE, SL_IID_ANDROIDCONFIGURATION };
const SLboolean reqs[] = {SL_BOOLEAN_TRUE, SL_BOOLEAN_TRUE};
return (*mEngineInterface)->CreateAudioRecorder(mEngineInterface, objectItf, audioSource,
audioSink,
sizeof(ids) / sizeof(ids[0]), ids, reqs);
}