/* * Copyright (C) 2014 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. */ /* Acoustic Echo Cancellation implementation */ #include "sles_allinclusive.h" #include <media/EffectsFactoryApi.h> #include <audio_effects/effect_aec.h> /** * returns true if this interface is not associated with an initialized AEC effect */ static inline bool NO_ECHOCANCEL(IAndroidAcousticEchoCancellation* v) { return (v->mAECEffect == 0); } static SLresult IAndroidAcousticEchoCancellation_SetEnabled( SLAndroidAcousticEchoCancellationItf self, SLboolean enabled) { SL_ENTER_INTERFACE IAndroidAcousticEchoCancellation *thiz = (IAndroidAcousticEchoCancellation *) self; interface_lock_exclusive(thiz); thiz->mEnabled = (SLboolean) enabled; if (NO_ECHOCANCEL(thiz)) { result = SL_RESULT_CONTROL_LOST; } else { android::status_t status = thiz->mAECEffect->setEnabled((bool) thiz->mEnabled); result = android_fx_statusToResult(status); } interface_unlock_exclusive(thiz); SL_LEAVE_INTERFACE } static SLresult IAndroidAcousticEchoCancellation_IsEnabled( SLAndroidAcousticEchoCancellationItf self, SLboolean *pEnabled) { SL_ENTER_INTERFACE if (NULL == pEnabled) { result = SL_RESULT_PARAMETER_INVALID; } else { IAndroidAcousticEchoCancellation *thiz = (IAndroidAcousticEchoCancellation *) self; interface_lock_exclusive(thiz); if (NO_ECHOCANCEL(thiz)) { result = SL_RESULT_CONTROL_LOST; } else { *pEnabled = (SLboolean) thiz->mAECEffect->getEnabled(); result = SL_RESULT_SUCCESS; } interface_unlock_exclusive(thiz); } SL_LEAVE_INTERFACE } SLresult IAndroidAcousticEchoCancellation_IsAvailable(SLAndroidAcousticEchoCancellationItf self, SLboolean *pEnabled) { SL_ENTER_INTERFACE *pEnabled = false; uint32_t numEffects = 0; int ret = EffectQueryNumberEffects(&numEffects); if (ret != 0) { ALOGE("IAndroidAcousticEchoCancellation_IsAvailable() error %d querying number of effects", ret); result = SL_RESULT_FEATURE_UNSUPPORTED; } else { ALOGV("EffectQueryNumberEffects() numEffects=%d", numEffects); effect_descriptor_t fxDesc; for (uint32_t i = 0 ; i < numEffects ; i++) { if (EffectQueryEffect(i, &fxDesc) == 0) { ALOGV("effect %d is called %s", i, fxDesc.name); if (memcmp(&fxDesc.type, SL_IID_ANDROIDACOUSTICECHOCANCELLATION, sizeof(effect_uuid_t)) == 0) { ALOGI("found effect \"%s\" from %s", fxDesc.name, fxDesc.implementor); *pEnabled = true; break; } } } result = SL_RESULT_SUCCESS; } SL_LEAVE_INTERFACE } static const struct SLAndroidAcousticEchoCancellationItf_ IAndroidAcousticEchoCancellation_Itf = { IAndroidAcousticEchoCancellation_SetEnabled, IAndroidAcousticEchoCancellation_IsEnabled, IAndroidAcousticEchoCancellation_IsAvailable }; void IAndroidAcousticEchoCancellation_init(void *self) { IAndroidAcousticEchoCancellation *thiz = (IAndroidAcousticEchoCancellation *) self; thiz->mItf = &IAndroidAcousticEchoCancellation_Itf; thiz->mEnabled = SL_BOOLEAN_FALSE; memset(&thiz->mAECDescriptor, 0, sizeof(effect_descriptor_t)); // placement new (explicit constructor) (void) new (&thiz->mAECEffect) android::sp<android::AudioEffect>(); } void IAndroidAcousticEchoCancellation_deinit(void *self) { IAndroidAcousticEchoCancellation *thiz = (IAndroidAcousticEchoCancellation *) self; // explicit destructor thiz->mAECEffect.~sp(); } bool IAndroidAcousticEchoCancellation_Expose(void *self) { IAndroidAcousticEchoCancellation *thiz = (IAndroidAcousticEchoCancellation *) self; if (!android_fx_initEffectDescriptor(SL_IID_ANDROIDACOUSTICECHOCANCELLATION, &thiz->mAECDescriptor)) { SL_LOGE("Acoustic Echo Cancellation initialization failed."); return false; } return true; }