/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* Initial global entry points */ #include "sles_allinclusive.h" #ifdef ANDROID #include <binder/ProcessState.h> #endif /** \brief Internal code shared by slCreateEngine and xaCreateEngine */ LI_API SLresult liCreateEngine(SLObjectItf *pEngine, SLuint32 numOptions, const SLEngineOption *pEngineOptions, SLuint32 numInterfaces, const SLInterfaceID *pInterfaceIds, const SLboolean *pInterfaceRequired, const ClassTable *pCEngine_class) { SLresult result; int ok; ok = pthread_mutex_lock(&theOneTrueMutex); assert(0 == ok); bool needToUnlockTheOneTrueMutex = true; do { if (NULL == pEngine) { result = SL_RESULT_PARAMETER_INVALID; break; } *pEngine = NULL; if ((0 < numOptions) && (NULL == pEngineOptions)) { SL_LOGE("numOptions=%u and pEngineOptions=NULL", numOptions); result = SL_RESULT_PARAMETER_INVALID; break; } // default values SLboolean threadSafe = SL_BOOLEAN_TRUE; SLboolean lossOfControlGlobal = SL_BOOLEAN_FALSE; // process engine options SLuint32 i; const SLEngineOption *option = pEngineOptions; result = SL_RESULT_SUCCESS; for (i = 0; i < numOptions; ++i, ++option) { switch (option->feature) { case SL_ENGINEOPTION_THREADSAFE: threadSafe = SL_BOOLEAN_FALSE != (SLboolean) option->data; // normalize break; case SL_ENGINEOPTION_LOSSOFCONTROL: lossOfControlGlobal = SL_BOOLEAN_FALSE != (SLboolean) option->data; // normalize break; default: SL_LOGE("unknown engine option: feature=%u data=%u", option->feature, option->data); result = SL_RESULT_PARAMETER_INVALID; break; } } if (SL_RESULT_SUCCESS != result) { break; } unsigned exposedMask; assert(NULL != pCEngine_class); result = checkInterfaces(pCEngine_class, numInterfaces, pInterfaceIds, pInterfaceRequired, &exposedMask, NULL); if (SL_RESULT_SUCCESS != result) { break; } // if an engine already exists, then increment its ref count CEngine *thiz = theOneTrueEngine; if (NULL != thiz) { assert(0 < theOneTrueRefCount); ++theOneTrueRefCount; // In order to update the engine object, we need to lock it, // but that would violate the lock order and potentially deadlock. // So we unlock now and note that it should not be unlocked later. ok = pthread_mutex_unlock(&theOneTrueMutex); assert(0 == ok); needToUnlockTheOneTrueMutex = false; object_lock_exclusive(&thiz->mObject); // now expose additional interfaces not requested by the earlier engine create const struct iid_vtable *x = pCEngine_class->mInterfaces; SLuint8 *interfaceStateP = thiz->mObject.mInterfaceStates; SLuint32 index; for (index = 0; index < pCEngine_class->mInterfaceCount; ++index, ++x, exposedMask >>= 1, ++interfaceStateP) { switch (*interfaceStateP) { case INTERFACE_EXPOSED: // previously exposed break; case INTERFACE_INITIALIZED: // not exposed during the earlier create if (exposedMask & 1) { const struct MPH_init *mi = &MPH_init_table[x->mMPH]; BoolHook expose = mi->mExpose; if ((NULL == expose) || (*expose)((char *) thiz + x->mOffset)) { *interfaceStateP = INTERFACE_EXPOSED; } // FIXME log or report to application that expose hook failed } break; case INTERFACE_UNINITIALIZED: // no init hook break; default: // impossible assert(false); break; } } object_unlock_exclusive(&thiz->mObject); // return the shared engine object *pEngine = &thiz->mObject.mItf; break; } // here when creating the first engine reference assert(0 == theOneTrueRefCount); #ifdef ANDROID android::ProcessState::self()->startThreadPool(); #endif thiz = (CEngine *) construct(pCEngine_class, exposedMask, NULL); if (NULL == thiz) { result = SL_RESULT_MEMORY_FAILURE; break; } // initialize fields not associated with an interface // mThreadPool is initialized in CEngine_Realize memset(&thiz->mThreadPool, 0, sizeof(ThreadPool)); memset(&thiz->mSyncThread, 0, sizeof(pthread_t)); #if defined(ANDROID) thiz->mEqNumPresets = 0; thiz->mEqPresetNames = NULL; #endif // initialize fields related to an interface thiz->mObject.mLossOfControlMask = lossOfControlGlobal ? ~0 : 0; thiz->mEngine.mLossOfControlGlobal = lossOfControlGlobal; thiz->mEngineCapabilities.mThreadSafe = threadSafe; IObject_Publish(&thiz->mObject); theOneTrueEngine = thiz; theOneTrueRefCount = 1; // return the new engine object *pEngine = &thiz->mObject.mItf; } while(0); if (needToUnlockTheOneTrueMutex) { ok = pthread_mutex_unlock(&theOneTrueMutex); assert(0 == ok); } return result; } /** Internal function for slQuerySupportedEngineInterfaces and xaQuerySupportedEngineInterfaces */ LI_API SLresult liQueryNumSupportedInterfaces(SLuint32 *pNumSupportedInterfaces, const ClassTable *clazz) { SLresult result; if (NULL == pNumSupportedInterfaces) { result = SL_RESULT_PARAMETER_INVALID; } else { assert(NULL != clazz); SLuint32 count = 0; SLuint32 i; for (i = 0; i < clazz->mInterfaceCount; ++i) { switch (clazz->mInterfaces[i].mInterface) { case INTERFACE_IMPLICIT: case INTERFACE_IMPLICIT_PREREALIZE: case INTERFACE_EXPLICIT: case INTERFACE_EXPLICIT_PREREALIZE: case INTERFACE_DYNAMIC: ++count; break; case INTERFACE_UNAVAILABLE: break; default: assert(false); break; } } *pNumSupportedInterfaces = count; result = SL_RESULT_SUCCESS; } return result; } /** Internal function for slQuerySupportedEngineInterfaces and xaQuerySupportedEngineInterfaces */ LI_API SLresult liQuerySupportedInterfaces(SLuint32 index, SLInterfaceID *pInterfaceId, const ClassTable *clazz) { SLresult result; if (NULL == pInterfaceId) { result = SL_RESULT_PARAMETER_INVALID; } else { *pInterfaceId = NULL; assert(NULL != clazz); result = SL_RESULT_PARAMETER_INVALID; // will be reset later SLuint32 i; for (i = 0; i < clazz->mInterfaceCount; ++i) { switch (clazz->mInterfaces[i].mInterface) { case INTERFACE_IMPLICIT: case INTERFACE_IMPLICIT_PREREALIZE: case INTERFACE_EXPLICIT: case INTERFACE_EXPLICIT_PREREALIZE: case INTERFACE_DYNAMIC: break; case INTERFACE_UNAVAILABLE: continue; default: assert(false); break; } if (index == 0) { *pInterfaceId = &SL_IID_array[clazz->mInterfaces[i].mMPH]; result = SL_RESULT_SUCCESS; break; } --index; } } return result; }