| /* |
| * 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. |
| */ |
| |
| |
| #include "sles_allinclusive.h" |
| #include "math.h" |
| #include "utils/RefBase.h" |
| #include "utils/String16.h" |
| |
| #include <system/audio_effects/effect_bassboost.h> |
| #include <system/audio_effects/effect_equalizer.h> |
| #include <system/audio_effects/effect_environmentalreverb.h> |
| #include <system/audio_effects/effect_presetreverb.h> |
| #include <system/audio_effects/effect_virtualizer.h> |
| |
| #include <system/audio_effects/effect_aec.h> |
| #include <system/audio_effects/effect_agc.h> |
| #include <system/audio_effects/effect_ns.h> |
| |
| #include <system/audio.h> |
| |
| static const int EQUALIZER_PARAM_SIZE_MAX = sizeof(effect_param_t) + 2 * sizeof(int32_t) |
| + EFFECT_STRING_LEN_MAX; |
| |
| static const int BASSBOOST_PARAM_SIZE_MAX = sizeof(effect_param_t) + 2 * sizeof(int32_t); |
| |
| static const int VIRTUALIZER_PARAM_SIZE_MAX = sizeof(effect_param_t) + 2 * sizeof(int32_t); |
| |
| static const int ENVREVERB_PARAM_SIZE_MAX_SINGLE = sizeof(effect_param_t) + 2 * sizeof(int32_t); |
| |
| static const int ENVREVERB_PARAM_SIZE_MAX_ALL = sizeof(effect_param_t) + sizeof(int32_t) |
| + sizeof(s_reverb_settings); |
| |
| static const int PRESETREVERB_PARAM_SIZE_MAX = sizeof(effect_param_t) + 2 * sizeof(int32_t); |
| |
| static inline SLuint32 KEY_FROM_GUID(SLInterfaceID pUuid) { |
| return pUuid->time_low; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| static |
| uint32_t eq_paramSize(int32_t param) { |
| uint32_t size; |
| |
| switch (param) { |
| case EQ_PARAM_NUM_BANDS: |
| case EQ_PARAM_LEVEL_RANGE: |
| case EQ_PARAM_CUR_PRESET: |
| case EQ_PARAM_GET_NUM_OF_PRESETS: |
| size = sizeof(int32_t); |
| break; |
| case EQ_PARAM_BAND_LEVEL: |
| case EQ_PARAM_CENTER_FREQ: |
| case EQ_PARAM_BAND_FREQ_RANGE: |
| case EQ_PARAM_GET_BAND: |
| case EQ_PARAM_GET_PRESET_NAME: |
| size = 2 * sizeof(int32_t); |
| break; |
| default: |
| size = 2 * sizeof(int32_t); |
| SL_LOGE("Trying to use an unknown EQ parameter %d", param); |
| break; |
| } |
| return size; |
| } |
| |
| static |
| uint32_t eq_valueSize(int32_t param) { |
| uint32_t size; |
| |
| switch (param) { |
| case EQ_PARAM_NUM_BANDS: |
| case EQ_PARAM_CUR_PRESET: |
| case EQ_PARAM_GET_NUM_OF_PRESETS: |
| case EQ_PARAM_BAND_LEVEL: |
| case EQ_PARAM_GET_BAND: |
| size = sizeof(int16_t); |
| break; |
| case EQ_PARAM_LEVEL_RANGE: |
| size = 2 * sizeof(int16_t); |
| break; |
| case EQ_PARAM_CENTER_FREQ: |
| size = sizeof(int32_t); |
| break; |
| case EQ_PARAM_BAND_FREQ_RANGE: |
| size = 2 * sizeof(int32_t); |
| break; |
| case EQ_PARAM_GET_PRESET_NAME: |
| size = EFFECT_STRING_LEN_MAX; |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown EQ parameter %d", param); |
| break; |
| } |
| return size; |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * returns the size in bytes of the value of each bass boost parameter |
| */ |
| static |
| uint32_t bb_valueSize(int32_t param) { |
| uint32_t size; |
| |
| switch (param) { |
| case BASSBOOST_PARAM_STRENGTH_SUPPORTED: |
| size = sizeof(int32_t); |
| break; |
| case BASSBOOST_PARAM_STRENGTH: |
| size = sizeof(int16_t); |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown BassBoost parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * returns the size in bytes of the value of each virtualizer parameter |
| */ |
| static |
| uint32_t virt_valueSize(int32_t param) { |
| uint32_t size; |
| |
| switch (param) { |
| case VIRTUALIZER_PARAM_STRENGTH_SUPPORTED: |
| size = sizeof(int32_t); |
| break; |
| case VIRTUALIZER_PARAM_STRENGTH: |
| size = sizeof(int16_t); |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown Virtualizer parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * returns the size in bytes of the value of each environmental reverb parameter |
| */ |
| static |
| uint32_t erev_valueSize(int32_t param) { |
| uint32_t size; |
| |
| switch (param) { |
| case REVERB_PARAM_ROOM_LEVEL: |
| case REVERB_PARAM_ROOM_HF_LEVEL: |
| case REVERB_PARAM_REFLECTIONS_LEVEL: |
| case REVERB_PARAM_REVERB_LEVEL: |
| size = sizeof(int16_t); // millibel |
| break; |
| case REVERB_PARAM_DECAY_TIME: |
| case REVERB_PARAM_REFLECTIONS_DELAY: |
| case REVERB_PARAM_REVERB_DELAY: |
| size = sizeof(uint32_t); // milliseconds |
| break; |
| case REVERB_PARAM_DECAY_HF_RATIO: |
| case REVERB_PARAM_DIFFUSION: |
| case REVERB_PARAM_DENSITY: |
| size = sizeof(int16_t); // permille |
| break; |
| case REVERB_PARAM_PROPERTIES: |
| size = sizeof(s_reverb_settings); // struct of all reverb properties |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown Environmental Reverb parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_eq_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, int32_t param2, void *pValue) |
| { |
| android::status_t status; |
| uint32_t buf32[(EQUALIZER_PARAM_SIZE_MAX - 1) / sizeof(uint32_t) + 1]; |
| effect_param_t *p = (effect_param_t *)buf32; |
| |
| p->psize = eq_paramSize(param); |
| *(int32_t *)p->data = param; |
| if (p->psize == 2 * sizeof(int32_t)) { |
| *((int32_t *)p->data + 1) = param2; |
| } |
| p->vsize = eq_valueSize(param); |
| status = pFx->getParameter(p); |
| if (android::NO_ERROR == status) { |
| status = p->status; |
| if (android::NO_ERROR == status) { |
| memcpy(pValue, p->data + p->psize, p->vsize); |
| } |
| } |
| |
| return status; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_eq_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, int32_t param2, void *pValue) |
| { |
| android::status_t status; |
| uint32_t buf32[(EQUALIZER_PARAM_SIZE_MAX - 1) / sizeof(uint32_t) + 1]; |
| effect_param_t *p = (effect_param_t *)buf32; |
| |
| p->psize = eq_paramSize(param); |
| *(int32_t *)p->data = param; |
| if (p->psize == 2 * sizeof(int32_t)) { |
| *((int32_t *)p->data + 1) = param2; |
| } |
| p->vsize = eq_valueSize(param); |
| memcpy(p->data + p->psize, pValue, p->vsize); |
| status = pFx->setParameter(p); |
| if (android::NO_ERROR == status) { |
| status = p->status; |
| } |
| |
| return status; |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_bb_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| return android_fx_setParam(pFx, param, BASSBOOST_PARAM_SIZE_MAX, |
| pValue, bb_valueSize(param)); |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_bb_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| return android_fx_getParam(pFx, param, BASSBOOST_PARAM_SIZE_MAX, |
| pValue, bb_valueSize(param)); |
| } |
| |
| //----------------------------------------------------------------------------- |
| void android_bb_init(audio_session_t sessionId, IBassBoost* ibb) { |
| SL_LOGV("session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, ibb->mBassBoostEffect, |
| &ibb->mBassBoostDescriptor.type)) |
| { |
| SL_LOGE("BassBoost effect initialization failed"); |
| return; |
| } |
| |
| // initialize strength |
| int16_t strength; |
| if (android::NO_ERROR == android_bb_getParam(ibb->mBassBoostEffect, |
| BASSBOOST_PARAM_STRENGTH, &strength)) { |
| ibb->mStrength = (SLpermille) strength; |
| } |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| void android_eq_init(audio_session_t sessionId, IEqualizer* ieq) { |
| SL_LOGV("android_eq_init on session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, ieq->mEqEffect, &ieq->mEqDescriptor.type)) { |
| SL_LOGE("Equalizer effect initialization failed"); |
| return; |
| } |
| |
| // initialize number of bands, band level range, and number of presets |
| uint16_t num = 0; |
| if (android::NO_ERROR == android_eq_getParam(ieq->mEqEffect, EQ_PARAM_NUM_BANDS, 0, &num)) { |
| ieq->mNumBands = num; |
| } |
| int16_t range[2] = {0, 0}; |
| if (android::NO_ERROR == android_eq_getParam(ieq->mEqEffect, EQ_PARAM_LEVEL_RANGE, 0, range)) { |
| ieq->mBandLevelRangeMin = range[0]; |
| ieq->mBandLevelRangeMax = range[1]; |
| } |
| |
| SL_LOGV(" EQ init: num bands = %u, band range=[%d %d]mB", num, range[0], range[1]); |
| |
| // FIXME don't store presets names, they can be queried each time they're needed |
| // initialize preset number and names, store in IEngine |
| uint16_t numPresets = 0; |
| if (android::NO_ERROR == android_eq_getParam(ieq->mEqEffect, |
| EQ_PARAM_GET_NUM_OF_PRESETS, 0, &numPresets)) { |
| ieq->mThis->mEngine->mEqNumPresets = numPresets; |
| ieq->mNumPresets = numPresets; |
| } |
| |
| object_lock_exclusive(&ieq->mThis->mEngine->mObject); |
| char name[EFFECT_STRING_LEN_MAX]; |
| if ((0 < numPresets) && (NULL == ieq->mThis->mEngine->mEqPresetNames)) { |
| ieq->mThis->mEngine->mEqPresetNames = (char **)new char *[numPresets]; |
| for(uint32_t i = 0 ; i < numPresets ; i++) { |
| if (android::NO_ERROR == android_eq_getParam(ieq->mEqEffect, |
| EQ_PARAM_GET_PRESET_NAME, i, name)) { |
| ieq->mThis->mEngine->mEqPresetNames[i] = new char[strlen(name) + 1]; |
| strcpy(ieq->mThis->mEngine->mEqPresetNames[i], name); |
| SL_LOGV(" EQ init: presets = %u is %s", i, ieq->mThis->mEngine->mEqPresetNames[i]); |
| } |
| } |
| } |
| object_unlock_exclusive(&ieq->mThis->mEngine->mObject); |
| |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| void android_virt_init(audio_session_t sessionId, IVirtualizer* ivi) { |
| SL_LOGV("android_virt_init on session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, ivi->mVirtualizerEffect, |
| &ivi->mVirtualizerDescriptor.type)) { |
| SL_LOGE("Virtualizer effect initialization failed"); |
| return; |
| } |
| |
| // initialize strength |
| int16_t strength; |
| if (android::NO_ERROR == android_virt_getParam(ivi->mVirtualizerEffect, |
| VIRTUALIZER_PARAM_STRENGTH, &strength)) { |
| ivi->mStrength = (SLpermille) strength; |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_virt_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| return android_fx_setParam(pFx, param, VIRTUALIZER_PARAM_SIZE_MAX, |
| pValue, virt_valueSize(param)); |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_virt_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| return android_fx_getParam(pFx, param, VIRTUALIZER_PARAM_SIZE_MAX, |
| pValue, virt_valueSize(param)); |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| void android_prev_init(IPresetReverb* ipr) { |
| SL_LOGV("session is implicitly %d (aux effect)", AUDIO_SESSION_OUTPUT_MIX); |
| |
| if (!android_fx_initEffectObj(AUDIO_SESSION_OUTPUT_MIX /*sessionId*/, |
| ipr->mPresetReverbEffect, &ipr->mPresetReverbDescriptor.type)) { |
| SL_LOGE("PresetReverb effect initialization failed"); |
| return; |
| } |
| |
| // initialize preset |
| uint16_t preset; |
| if (android::NO_ERROR == android_prev_getPreset(ipr->mPresetReverbEffect, &preset)) { |
| ipr->mPreset = preset; |
| // enable the effect if it has a preset loaded |
| ipr->mPresetReverbEffect->setEnabled(SL_REVERBPRESET_NONE != preset); |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_prev_setPreset(const android::sp<android::AudioEffect>& pFx, |
| uint16_t preset) { |
| android::status_t status = android_fx_setParam(pFx, REVERB_PARAM_PRESET, |
| PRESETREVERB_PARAM_SIZE_MAX, &preset, sizeof(uint16_t)); |
| // enable the effect if the preset is different from SL_REVERBPRESET_NONE |
| pFx->setEnabled(SL_REVERBPRESET_NONE != preset); |
| return status; |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_prev_getPreset(const android::sp<android::AudioEffect>& pFx, |
| uint16_t* preset) { |
| return android_fx_getParam(pFx, REVERB_PARAM_PRESET, PRESETREVERB_PARAM_SIZE_MAX, preset, |
| sizeof(uint16_t)); |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| void android_erev_init(IEnvironmentalReverb* ier) { |
| SL_LOGV("session is implicitly %d (aux effect)", AUDIO_SESSION_OUTPUT_MIX); |
| |
| if (!android_fx_initEffectObj(AUDIO_SESSION_OUTPUT_MIX /*sessionId*/, |
| ier->mEnvironmentalReverbEffect, &ier->mEnvironmentalReverbDescriptor.type)) { |
| SL_LOGE("EnvironmentalReverb effect initialization failed"); |
| return; |
| } |
| |
| // enable env reverb: other SL ES effects have an explicit SetEnabled() function, and the |
| // preset reverb state depends on the selected preset. |
| ier->mEnvironmentalReverbEffect->setEnabled(true); |
| |
| // initialize reverb properties |
| SLEnvironmentalReverbSettings properties; |
| if (android::NO_ERROR == android_erev_getParam(ier->mEnvironmentalReverbEffect, |
| REVERB_PARAM_PROPERTIES, &properties)) { |
| ier->mProperties = properties; |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_erev_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| // given the size difference between a single reverb property and the whole set of reverb |
| // properties, select which max size to pass to avoid allocating too much memory |
| if (param == REVERB_PARAM_PROPERTIES) { |
| return android_fx_setParam(pFx, param, ENVREVERB_PARAM_SIZE_MAX_ALL, |
| pValue, erev_valueSize(param)); |
| } else { |
| return android_fx_setParam(pFx, param, ENVREVERB_PARAM_SIZE_MAX_SINGLE, |
| pValue, erev_valueSize(param)); |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_erev_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| |
| // given the size difference between a single reverb property and the whole set of reverb |
| // properties, select which max size to pass to avoid allocating too much memory |
| if (param == REVERB_PARAM_PROPERTIES) { |
| return android_fx_getParam(pFx, param, ENVREVERB_PARAM_SIZE_MAX_ALL, |
| pValue, erev_valueSize(param)); |
| } else { |
| return android_fx_getParam(pFx, param, ENVREVERB_PARAM_SIZE_MAX_SINGLE, |
| pValue, erev_valueSize(param)); |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| void android_aec_init(audio_session_t sessionId, IAndroidAcousticEchoCancellation* iaec) { |
| SL_LOGV("android_aec_init on session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, iaec->mAECEffect, |
| &iaec->mAECDescriptor.type)) { |
| SL_LOGE("AEC effect initialization failed"); |
| return; |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| void android_agc_init(audio_session_t sessionId, IAndroidAutomaticGainControl* iagc) { |
| SL_LOGV("android_agc_init on session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, iagc->mAGCEffect, |
| &iagc->mAGCDescriptor.type)) { |
| SL_LOGE("AGC effect initialization failed"); |
| return; |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| void android_ns_init(audio_session_t sessionId, IAndroidNoiseSuppression* ins) { |
| SL_LOGV("android_ns_init on session %d", sessionId); |
| |
| if (!android_fx_initEffectObj(sessionId, ins->mNSEffect, |
| &ins->mNSDescriptor.type)) { |
| SL_LOGE("NS effect initialization failed"); |
| return; |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * pre-condition: |
| * ap != NULL |
| * for media players: |
| * ap->mAPlayer != 0 |
| * ap->mTrackPlayer->mAudioTrack == 0 |
| * for buffer queue players: |
| * ap->mAPlayer == 0 |
| * ap->mTrackPlayer->mAudioTrack != 0 is optional; if no track yet then the setting is deferred |
| */ |
| android::status_t android_fxSend_attach(CAudioPlayer* ap, bool attach, |
| const android::sp<android::AudioEffect>& pFx, SLmillibel sendLevel) { |
| |
| if (pFx == 0) { |
| return android::INVALID_OPERATION; |
| } |
| |
| // There are 3 cases: |
| // mAPlayer != 0 && mAudioTrack == 0 means playing decoded audio |
| // mAPlayer == 0 && mAudioTrack != 0 means playing PCM audio |
| // mAPlayer == 0 && mAudioTrack == 0 means player not fully configured yet |
| // The asserts document and verify this. |
| if (ap->mAPlayer != 0) { |
| assert(ap->mTrackPlayer->mAudioTrack == 0); |
| if (attach) { |
| ap->mAPlayer->attachAuxEffect(pFx->id()); |
| ap->mAPlayer->setAuxEffectSendLevel( sles_to_android_amplification(sendLevel) ); |
| } else { |
| ap->mAPlayer->attachAuxEffect(0); |
| } |
| return android::NO_ERROR; |
| } |
| |
| if (ap->mTrackPlayer->mAudioTrack == 0) { |
| // the player doesn't have an AudioTrack at the moment, so store this info to use it |
| // when the AudioTrack becomes available |
| if (attach) { |
| ap->mAuxEffect = pFx; |
| } else { |
| ap->mAuxEffect.clear(); |
| } |
| // we keep track of the send level, independently of the current audio player level |
| ap->mAuxSendLevel = sendLevel - ap->mVolume.mLevel; |
| return android::NO_ERROR; |
| } |
| |
| if (attach) { |
| android::status_t status = ap->mTrackPlayer->mAudioTrack->attachAuxEffect(pFx->id()); |
| //SL_LOGV("attachAuxEffect(%d) returned %d", pFx->id(), status); |
| if (android::NO_ERROR == status) { |
| status = |
| ap->mTrackPlayer->mAudioTrack->setAuxEffectSendLevel( |
| sles_to_android_amplification(sendLevel) ); |
| } |
| return status; |
| } else { |
| return ap->mTrackPlayer->mAudioTrack->attachAuxEffect(0); |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * pre-condition: |
| * ap != NULL |
| * ap->mOutputMix != NULL |
| */ |
| SLresult android_fxSend_attachToAux(CAudioPlayer* ap, SLInterfaceID pUuid, SLboolean attach, |
| SLmillibel sendLevel) { |
| COutputMix *outputMix = CAudioPlayer_GetOutputMix(ap); |
| ssize_t index = outputMix->mAndroidEffect.mEffects->indexOfKey(KEY_FROM_GUID(pUuid)); |
| |
| if (0 > index) { |
| SL_LOGE("invalid effect ID: no such effect attached to the OutputMix"); |
| return SL_RESULT_PARAMETER_INVALID; |
| } |
| |
| android::sp<android::AudioEffect> pFx = |
| outputMix->mAndroidEffect.mEffects->valueAt(index); |
| if (pFx == 0) { |
| return SL_RESULT_RESOURCE_ERROR; |
| } |
| if (android::NO_ERROR == android_fxSend_attach( ap, (bool) attach, pFx, sendLevel) ) { |
| return SL_RESULT_SUCCESS; |
| } else { |
| return SL_RESULT_RESOURCE_ERROR; |
| } |
| |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * pre-condition: |
| * ap != NULL |
| * for media players: |
| * ap->mAPlayer != 0 |
| * ap->mTrackPlayer->mAudioTrack == 0 |
| * for buffer queue players: |
| * ap->mAPlayer == 0 |
| * ap->mTrackPlayer->mAudioTrack != 0 is optional; if no track yet then the setting is deferred |
| */ |
| android::status_t android_fxSend_setSendLevel(CAudioPlayer* ap, SLmillibel sendLevel) { |
| // we keep track of the send level, independently of the current audio player level |
| ap->mAuxSendLevel = sendLevel - ap->mVolume.mLevel; |
| |
| if (ap->mAPlayer != 0) { |
| assert(ap->mTrackPlayer->mAudioTrack == 0); |
| ap->mAPlayer->setAuxEffectSendLevel( sles_to_android_amplification(sendLevel) ); |
| return android::NO_ERROR; |
| } |
| |
| if (ap->mTrackPlayer->mAudioTrack == 0) { |
| return android::NO_ERROR; |
| } |
| |
| return ap->mTrackPlayer->mAudioTrack->setAuxEffectSendLevel( |
| sles_to_android_amplification(sendLevel) ); |
| } |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_fx_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, uint32_t paramSizeMax, void *pValue, uint32_t valueSize) |
| { |
| |
| android::status_t status; |
| uint32_t buf32[(paramSizeMax - 1) / sizeof(uint32_t) + 1]; |
| effect_param_t *p = (effect_param_t *)buf32; |
| |
| p->psize = sizeof(int32_t); |
| *(int32_t *)p->data = param; |
| p->vsize = valueSize; |
| memcpy(p->data + p->psize, pValue, p->vsize); |
| status = pFx->setParameter(p); |
| if (android::NO_ERROR == status) { |
| status = p->status; |
| } |
| return status; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| android::status_t android_fx_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, uint32_t paramSizeMax, void *pValue, uint32_t valueSize) |
| { |
| android::status_t status; |
| uint32_t buf32[(paramSizeMax - 1) / sizeof(uint32_t) + 1]; |
| effect_param_t *p = (effect_param_t *)buf32; |
| |
| p->psize = sizeof(int32_t); |
| *(int32_t *)p->data = param; |
| p->vsize = valueSize; |
| status = pFx->getParameter(p); |
| if (android::NO_ERROR == status) { |
| status = p->status; |
| if (android::NO_ERROR == status) { |
| memcpy(pValue, p->data + p->psize, p->vsize); |
| } |
| } |
| |
| return status; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_fx_statusToResult(android::status_t status) { |
| |
| if ((android::INVALID_OPERATION == status) || (android::DEAD_OBJECT == status)) { |
| return SL_RESULT_CONTROL_LOST; |
| } else { |
| return SL_RESULT_SUCCESS; |
| } |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| bool android_fx_initEffectObj(audio_session_t sessionId, android::sp<android::AudioEffect>& effect, |
| const effect_uuid_t *type) { |
| //SL_LOGV("android_fx_initEffectObj on session %d", sessionId); |
| |
| effect = new android::AudioEffect(type, android::String16(), EFFECT_UUID_NULL, |
| 0,// priority |
| 0,// effect callback |
| 0,// callback data |
| sessionId,// session ID |
| 0 );// output |
| |
| android::status_t status = effect->initCheck(); |
| if (android::NO_ERROR != status) { |
| effect.clear(); |
| SL_LOGE("Effect initCheck() returned %d", status); |
| return false; |
| } |
| |
| return true; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| bool android_fx_initEffectDescriptor(const SLInterfaceID effectId, |
| effect_descriptor_t* fxDescrLoc) { |
| uint32_t numEffects = 0; |
| effect_descriptor_t descriptor; |
| bool foundEffect = false; |
| |
| // any effects? |
| android::status_t res = android::AudioEffect::queryNumberEffects(&numEffects); |
| if (android::NO_ERROR != res) { |
| SL_LOGE("unable to find any effects."); |
| goto effectError; |
| } |
| |
| // request effect in the effects? |
| for (uint32_t i=0 ; i < numEffects ; i++) { |
| res = android::AudioEffect::queryEffect(i, &descriptor); |
| if ((android::NO_ERROR == res) && |
| (0 == memcmp(effectId, &descriptor.type, sizeof(effect_uuid_t)))) { |
| SL_LOGV("found effect %d %s", i, descriptor.name); |
| foundEffect = true; |
| break; |
| } |
| } |
| if (foundEffect) { |
| memcpy(fxDescrLoc, &descriptor, sizeof(effect_descriptor_t)); |
| } else { |
| SL_LOGE("unable to find an implementation for the requested effect."); |
| goto effectError; |
| } |
| |
| return true; |
| |
| effectError: |
| // the requested effect wasn't found |
| memset(fxDescrLoc, 0, sizeof(effect_descriptor_t)); |
| |
| return false; |
| } |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_queryNumEffects(SLuint32 *pNumSupportedAudioEffects) { |
| |
| if (NULL == pNumSupportedAudioEffects) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } |
| |
| android::status_t status = |
| android::AudioEffect::queryNumberEffects((uint32_t*)pNumSupportedAudioEffects); |
| |
| SLresult result = SL_RESULT_SUCCESS; |
| switch (status) { |
| case android::NO_ERROR: |
| result = SL_RESULT_SUCCESS; |
| break; |
| case android::PERMISSION_DENIED: |
| result = SL_RESULT_PERMISSION_DENIED; |
| break; |
| case android::NO_INIT: |
| result = SL_RESULT_RESOURCE_ERROR; |
| break; |
| case android::BAD_VALUE: |
| result = SL_RESULT_PARAMETER_INVALID; |
| break; |
| default: |
| result = SL_RESULT_INTERNAL_ERROR; |
| SL_LOGE("received invalid status %d from AudioEffect::queryNumberEffects()", status); |
| break; |
| } |
| return result; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_queryEffect(SLuint32 index, effect_descriptor_t* pDescriptor) { |
| |
| if (NULL == pDescriptor) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } |
| |
| android::status_t status = |
| android::AudioEffect::queryEffect(index, pDescriptor); |
| |
| SLresult result = SL_RESULT_SUCCESS; |
| if (android::NO_ERROR != status) { |
| switch (status) { |
| case android::PERMISSION_DENIED: |
| result = SL_RESULT_PERMISSION_DENIED; |
| break; |
| case android::NO_INIT: |
| case android::INVALID_OPERATION: |
| result = SL_RESULT_RESOURCE_ERROR; |
| break; |
| case android::BAD_VALUE: |
| result = SL_RESULT_PARAMETER_INVALID; |
| break; |
| default: |
| result = SL_RESULT_INTERNAL_ERROR; |
| SL_LOGE("received invalid status %d from AudioEffect::queryNumberEffects()", status); |
| break; |
| } |
| // an error occurred, reset the effect descriptor |
| memset(pDescriptor, 0, sizeof(effect_descriptor_t)); |
| } |
| |
| return result; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_createEffect(IAndroidEffect* iae, SLInterfaceID pUuid, |
| audio_session_t sessionId) |
| { |
| |
| SLresult result = SL_RESULT_SUCCESS; |
| |
| // does this effect already exist? |
| if (0 <= iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid))) { |
| return result; |
| } |
| |
| // create new effect |
| android::sp<android::AudioEffect> pFx = new android::AudioEffect( |
| NULL, // not using type to create effect |
| android::String16(), |
| (const effect_uuid_t*)pUuid, |
| 0,// priority |
| 0,// effect callback |
| 0,// callback data |
| sessionId, |
| 0 );// output |
| |
| // verify effect was successfully created before storing it |
| android::status_t status = pFx->initCheck(); |
| if (android::NO_ERROR != status) { |
| SL_LOGE("AudioEffect initCheck() returned %d, effect will not be stored", status); |
| result = SL_RESULT_RESOURCE_ERROR; |
| } else { |
| SL_LOGV("AudioEffect successfully created on session %d", sessionId); |
| iae->mEffects->add(KEY_FROM_GUID(pUuid), pFx); |
| } |
| |
| return result; |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_releaseEffect(IAndroidEffect* iae, SLInterfaceID pUuid) { |
| |
| ssize_t index = iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid)); |
| |
| if (0 > index) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } else { |
| iae->mEffects->removeItem(index); |
| return SL_RESULT_SUCCESS; |
| } |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_setEnabled(IAndroidEffect* iae, SLInterfaceID pUuid, SLboolean enabled) { |
| |
| ssize_t index = iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid)); |
| |
| if (0 > index) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } else { |
| android::sp<android::AudioEffect> pFx = iae->mEffects->valueAt(index); |
| android::status_t status = pFx->setEnabled(SL_BOOLEAN_TRUE == enabled); |
| return android_fx_statusToResult(status); |
| } |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_isEnabled(IAndroidEffect* iae, SLInterfaceID pUuid, SLboolean *pEnabled) |
| { |
| ssize_t index = iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid)); |
| |
| if (0 > index) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } else { |
| android::sp<android::AudioEffect> pFx = iae->mEffects->valueAt(index); |
| *pEnabled = (SLboolean) pFx->getEnabled(); |
| return SL_RESULT_SUCCESS; |
| } |
| } |
| |
| |
| //----------------------------------------------------------------------------- |
| SLresult android_genericFx_sendCommand(IAndroidEffect* iae, SLInterfaceID pUuid, |
| SLuint32 command, SLuint32 commandSize, void* pCommandData, |
| SLuint32 *replySize, void *pReplyData) { |
| |
| ssize_t index = iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid)); |
| |
| if (0 > index) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } else { |
| android::sp<android::AudioEffect> pFx = iae->mEffects->valueAt(index); |
| android::status_t status = pFx->command( |
| (uint32_t) command, |
| (uint32_t) commandSize, |
| pCommandData, |
| (uint32_t*)replySize, |
| pReplyData); |
| if (android::BAD_VALUE == status) { |
| return SL_RESULT_PARAMETER_INVALID; |
| } else { |
| return SL_RESULT_SUCCESS; |
| } |
| } |
| } |
| |
| //----------------------------------------------------------------------------- |
| /** |
| * returns true if the given effect id is present in the AndroidEffect interface |
| */ |
| bool android_genericFx_hasEffect(IAndroidEffect* iae, SLInterfaceID pUuid) { |
| return( 0 <= iae->mEffects->indexOfKey(KEY_FROM_GUID(pUuid))); |
| } |
| |
| //----------------------------------------------------------------------------- |
| static const int AEC_PARAM_SIZE_MAX = sizeof(effect_param_t) + (2 * sizeof(int32_t)); |
| /** |
| * returns the size in bytes of the value of each acoustic echo cancellation parameter |
| */ |
| uint32_t aec_valueSize(int32_t param) { |
| uint32_t size; |
| switch (param) { |
| case AEC_PARAM_ECHO_DELAY: |
| size = sizeof(int32_t); |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown Acoustic Echo Cancellation parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| android::status_t android_aec_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| return android_fx_setParam(pFx, param, AEC_PARAM_SIZE_MAX, |
| pValue, aec_valueSize(param)); |
| } |
| |
| android::status_t android_aec_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| return android_fx_getParam(pFx, param, AEC_PARAM_SIZE_MAX, |
| pValue, aec_valueSize(param)); |
| } |
| |
| //----------------------------------------------------------------------------- |
| static const int AGC_PARAM_SIZE_MAX = sizeof(effect_param_t) + (2 * sizeof(int16_t)) + sizeof(bool); |
| /** |
| * returns the size in bytes of the value of each automatic gain control parameter |
| */ |
| uint32_t agc_valueSize(int32_t param) { |
| uint32_t size; |
| switch (param) { |
| case AGC_PARAM_TARGET_LEVEL: |
| case AGC_PARAM_COMP_GAIN: |
| size = sizeof(int16_t); |
| break; |
| case AGC_PARAM_LIMITER_ENA: |
| size = sizeof(bool); |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown Automatic Gain Control parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| android::status_t android_agc_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| return android_fx_setParam(pFx, param, AGC_PARAM_SIZE_MAX, |
| pValue, agc_valueSize(param)); |
| } |
| |
| android::status_t android_agc_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) { |
| return android_fx_getParam(pFx, param, AGC_PARAM_SIZE_MAX, |
| pValue, agc_valueSize(param)); |
| } |
| |
| //----------------------------------------------------------------------------- |
| static const int NS_PARAM_SIZE_MAX = sizeof(effect_param_t) + 2 * sizeof(int32_t); |
| /** |
| * returns the size in bytes of the value of each noise suppression parameter |
| */ |
| uint32_t ns_valueSize(int32_t param) { |
| uint32_t size; |
| switch (param) { |
| case NS_PARAM_LEVEL: |
| size = sizeof(int32_t); |
| break; |
| default: |
| size = sizeof(int32_t); |
| SL_LOGE("Trying to access an unknown Noise suppression parameter %d", param); |
| break; |
| } |
| |
| return size; |
| } |
| |
| android::status_t android_ns_setParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) |
| { |
| return android_fx_setParam(pFx, param, NS_PARAM_SIZE_MAX, |
| pValue, ns_valueSize(param)); |
| } |
| |
| android::status_t android_ns_getParam(const android::sp<android::AudioEffect>& pFx, |
| int32_t param, void *pValue) |
| { |
| return android_fx_getParam(pFx, param, NS_PARAM_SIZE_MAX, |
| pValue, ns_valueSize(param)); |
| } |