AudioEffect.cpp revision 8af901cdea0af7e536579dee6d56e69987035a01
1/*
2**
3** Copyright 2010, The Android Open Source Project
4**
5** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
8**
9**     http://www.apache.org/licenses/LICENSE-2.0
10**
11** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
15** limitations under the License.
16*/
17
18
19//#define LOG_NDEBUG 0
20#define LOG_TAG "AudioEffect"
21
22#include <stdint.h>
23#include <sys/types.h>
24#include <limits.h>
25
26#include <private/media/AudioEffectShared.h>
27#include <media/AudioEffect.h>
28
29#include <utils/Log.h>
30#include <binder/IPCThreadState.h>
31
32
33
34namespace android {
35
36// ---------------------------------------------------------------------------
37
38AudioEffect::AudioEffect()
39    : mStatus(NO_INIT)
40{
41}
42
43
44AudioEffect::AudioEffect(const effect_uuid_t *type,
45                const effect_uuid_t *uuid,
46                int32_t priority,
47                effect_callback_t cbf,
48                void* user,
49                int sessionId,
50                audio_io_handle_t io
51                )
52    : mStatus(NO_INIT)
53{
54    mStatus = set(type, uuid, priority, cbf, user, sessionId, io);
55}
56
57AudioEffect::AudioEffect(const char *typeStr,
58                const char *uuidStr,
59                int32_t priority,
60                effect_callback_t cbf,
61                void* user,
62                int sessionId,
63                audio_io_handle_t io
64                )
65    : mStatus(NO_INIT)
66{
67    effect_uuid_t type;
68    effect_uuid_t *pType = NULL;
69    effect_uuid_t uuid;
70    effect_uuid_t *pUuid = NULL;
71
72    ALOGV("Constructor string\n - type: %s\n - uuid: %s", typeStr, uuidStr);
73
74    if (typeStr != NULL) {
75        if (stringToGuid(typeStr, &type) == NO_ERROR) {
76            pType = &type;
77        }
78    }
79
80    if (uuidStr != NULL) {
81        if (stringToGuid(uuidStr, &uuid) == NO_ERROR) {
82            pUuid = &uuid;
83        }
84    }
85
86    mStatus = set(pType, pUuid, priority, cbf, user, sessionId, io);
87}
88
89status_t AudioEffect::set(const effect_uuid_t *type,
90                const effect_uuid_t *uuid,
91                int32_t priority,
92                effect_callback_t cbf,
93                void* user,
94                int sessionId,
95                audio_io_handle_t io)
96{
97    sp<IEffect> iEffect;
98    sp<IMemory> cblk;
99    int enabled;
100
101    ALOGV("set %p mUserData: %p uuid: %p timeLow %08x", this, user, type, type ? type->timeLow : 0);
102
103    if (mIEffect != 0) {
104        ALOGW("Effect already in use");
105        return INVALID_OPERATION;
106    }
107
108    const sp<IAudioFlinger>& audioFlinger = AudioSystem::get_audio_flinger();
109    if (audioFlinger == 0) {
110        ALOGE("set(): Could not get audioflinger");
111        return NO_INIT;
112    }
113
114    if (type == NULL && uuid == NULL) {
115        ALOGW("Must specify at least type or uuid");
116        return BAD_VALUE;
117    }
118
119    mPriority = priority;
120    mCbf = cbf;
121    mUserData = user;
122    mSessionId = sessionId;
123
124    memset(&mDescriptor, 0, sizeof(effect_descriptor_t));
125    mDescriptor.type = *(type != NULL ? type : EFFECT_UUID_NULL);
126    mDescriptor.uuid = *(uuid != NULL ? uuid : EFFECT_UUID_NULL);
127
128    mIEffectClient = new EffectClient(this);
129
130    iEffect = audioFlinger->createEffect(getpid(), &mDescriptor,
131            mIEffectClient, priority, io, mSessionId, &mStatus, &mId, &enabled);
132
133    if (iEffect == 0 || (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS)) {
134        ALOGE("set(): AudioFlinger could not create effect, status: %d", mStatus);
135        return mStatus;
136    }
137
138    mEnabled = (volatile int32_t)enabled;
139
140    mIEffect = iEffect;
141    cblk = iEffect->getCblk();
142    if (cblk == 0) {
143        mStatus = NO_INIT;
144        ALOGE("Could not get control block");
145        return mStatus;
146    }
147
148    mIEffect = iEffect;
149    mCblkMemory = cblk;
150    mCblk = static_cast<effect_param_cblk_t*>(cblk->pointer());
151    int bufOffset = ((sizeof(effect_param_cblk_t) - 1) / sizeof(int) + 1) * sizeof(int);
152    mCblk->buffer = (uint8_t *)mCblk + bufOffset;
153
154    iEffect->asBinder()->linkToDeath(mIEffectClient);
155    ALOGV("set() %p OK effect: %s id: %d status %d enabled %d", this, mDescriptor.name, mId,
156            mStatus, mEnabled);
157
158    return mStatus;
159}
160
161
162AudioEffect::~AudioEffect()
163{
164    ALOGV("Destructor %p", this);
165
166    if (mStatus == NO_ERROR || mStatus == ALREADY_EXISTS) {
167        if (mIEffect != NULL) {
168            mIEffect->disconnect();
169            mIEffect->asBinder()->unlinkToDeath(mIEffectClient);
170        }
171        IPCThreadState::self()->flushCommands();
172    }
173    mIEffect.clear();
174    mIEffectClient.clear();
175    mCblkMemory.clear();
176}
177
178
179status_t AudioEffect::initCheck() const
180{
181    return mStatus;
182}
183
184// -------------------------------------------------------------------------
185
186effect_descriptor_t AudioEffect::descriptor() const
187{
188    return mDescriptor;
189}
190
191bool AudioEffect::getEnabled() const
192{
193    return (mEnabled != 0);
194}
195
196status_t AudioEffect::setEnabled(bool enabled)
197{
198    if (mStatus != NO_ERROR) {
199        return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
200    }
201
202    status_t status = NO_ERROR;
203
204    AutoMutex lock(mLock);
205    if (enabled != mEnabled) {
206        if (enabled) {
207            ALOGV("enable %p", this);
208            status = mIEffect->enable();
209        } else {
210            ALOGV("disable %p", this);
211            status = mIEffect->disable();
212        }
213        if (status == NO_ERROR) {
214            mEnabled = enabled;
215        }
216    }
217    return status;
218}
219
220status_t AudioEffect::command(uint32_t cmdCode,
221                              uint32_t cmdSize,
222                              void *cmdData,
223                              uint32_t *replySize,
224                              void *replyData)
225{
226    if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
227        ALOGV("command() bad status %d", mStatus);
228        return mStatus;
229    }
230
231    if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
232        if (mEnabled == (cmdCode == EFFECT_CMD_ENABLE)) {
233            return NO_ERROR;
234        }
235        if (replySize == NULL || *replySize != sizeof(status_t) || replyData == NULL) {
236            return BAD_VALUE;
237        }
238        mLock.lock();
239    }
240
241    status_t status = mIEffect->command(cmdCode, cmdSize, cmdData, replySize, replyData);
242
243    if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
244        if (status == NO_ERROR) {
245            status = *(status_t *)replyData;
246        }
247        if (status == NO_ERROR) {
248            mEnabled = (cmdCode == EFFECT_CMD_ENABLE);
249        }
250        mLock.unlock();
251    }
252
253    return status;
254}
255
256
257status_t AudioEffect::setParameter(effect_param_t *param)
258{
259    if (mStatus != NO_ERROR) {
260        return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
261    }
262
263    if (param == NULL || param->psize == 0 || param->vsize == 0) {
264        return BAD_VALUE;
265    }
266
267    uint32_t size = sizeof(int);
268    uint32_t psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
269
270    ALOGV("setParameter: param: %d, param2: %d", *(int *)param->data,
271            (param->psize == 8) ? *((int *)param->data + 1): -1);
272
273    return mIEffect->command(EFFECT_CMD_SET_PARAM, sizeof (effect_param_t) + psize, param, &size,
274            &param->status);
275}
276
277status_t AudioEffect::setParameterDeferred(effect_param_t *param)
278{
279    if (mStatus != NO_ERROR) {
280        return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
281    }
282
283    if (param == NULL || param->psize == 0 || param->vsize == 0) {
284        return BAD_VALUE;
285    }
286
287    Mutex::Autolock _l(mCblk->lock);
288
289    int psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
290    int size = ((sizeof(effect_param_t) + psize - 1) / sizeof(int) + 1) * sizeof(int);
291
292    if (mCblk->clientIndex + size > EFFECT_PARAM_BUFFER_SIZE) {
293        return NO_MEMORY;
294    }
295    int *p = (int *)(mCblk->buffer + mCblk->clientIndex);
296    *p++ = size;
297    memcpy(p, param, sizeof(effect_param_t) + psize);
298    mCblk->clientIndex += size;
299
300    return NO_ERROR;
301}
302
303status_t AudioEffect::setParameterCommit()
304{
305    if (mStatus != NO_ERROR) {
306        return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
307    }
308
309    Mutex::Autolock _l(mCblk->lock);
310    if (mCblk->clientIndex == 0) {
311        return INVALID_OPERATION;
312    }
313    uint32_t size = 0;
314    return mIEffect->command(EFFECT_CMD_SET_PARAM_COMMIT, 0, NULL, &size, NULL);
315}
316
317status_t AudioEffect::getParameter(effect_param_t *param)
318{
319    if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
320        return mStatus;
321    }
322
323    if (param == NULL || param->psize == 0 || param->vsize == 0) {
324        return BAD_VALUE;
325    }
326
327    ALOGV("getParameter: param: %d, param2: %d", *(int *)param->data,
328            (param->psize == 8) ? *((int *)param->data + 1): -1);
329
330    uint32_t psize = sizeof(effect_param_t) + ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) +
331            param->vsize;
332
333    return mIEffect->command(EFFECT_CMD_GET_PARAM, sizeof(effect_param_t) + param->psize, param,
334            &psize, param);
335}
336
337
338// -------------------------------------------------------------------------
339
340void AudioEffect::binderDied()
341{
342    ALOGW("IEffect died");
343    mStatus = DEAD_OBJECT;
344    if (mCbf != NULL) {
345        status_t status = DEAD_OBJECT;
346        mCbf(EVENT_ERROR, mUserData, &status);
347    }
348    mIEffect.clear();
349}
350
351// -------------------------------------------------------------------------
352
353void AudioEffect::controlStatusChanged(bool controlGranted)
354{
355    ALOGV("controlStatusChanged %p control %d callback %p mUserData %p", this, controlGranted, mCbf,
356            mUserData);
357    if (controlGranted) {
358        if (mStatus == ALREADY_EXISTS) {
359            mStatus = NO_ERROR;
360        }
361    } else {
362        if (mStatus == NO_ERROR) {
363            mStatus = ALREADY_EXISTS;
364        }
365    }
366    if (mCbf != NULL) {
367        mCbf(EVENT_CONTROL_STATUS_CHANGED, mUserData, &controlGranted);
368    }
369}
370
371void AudioEffect::enableStatusChanged(bool enabled)
372{
373    ALOGV("enableStatusChanged %p enabled %d mCbf %p", this, enabled, mCbf);
374    if (mStatus == ALREADY_EXISTS) {
375        mEnabled = enabled;
376        if (mCbf != NULL) {
377            mCbf(EVENT_ENABLE_STATUS_CHANGED, mUserData, &enabled);
378        }
379    }
380}
381
382void AudioEffect::commandExecuted(uint32_t cmdCode,
383                                  uint32_t cmdSize,
384                                  void *cmdData,
385                                  uint32_t replySize,
386                                  void *replyData)
387{
388    if (cmdData == NULL || replyData == NULL) {
389        return;
390    }
391
392    if (mCbf != NULL && cmdCode == EFFECT_CMD_SET_PARAM) {
393        effect_param_t *cmd = (effect_param_t *)cmdData;
394        cmd->status = *(int32_t *)replyData;
395        mCbf(EVENT_PARAMETER_CHANGED, mUserData, cmd);
396    }
397}
398
399// -------------------------------------------------------------------------
400
401status_t AudioEffect::queryNumberEffects(uint32_t *numEffects)
402{
403    const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
404    if (af == 0) return PERMISSION_DENIED;
405    return af->queryNumberEffects(numEffects);
406}
407
408status_t AudioEffect::queryEffect(uint32_t index, effect_descriptor_t *descriptor)
409{
410    const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
411    if (af == 0) return PERMISSION_DENIED;
412    return af->queryEffect(index, descriptor);
413}
414
415status_t AudioEffect::getEffectDescriptor(const effect_uuid_t *uuid,
416        effect_descriptor_t *descriptor) /*const*/
417{
418    const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
419    if (af == 0) return PERMISSION_DENIED;
420    return af->getEffectDescriptor(uuid, descriptor);
421}
422
423
424status_t AudioEffect::queryDefaultPreProcessing(int audioSession,
425                                          effect_descriptor_t *descriptors,
426                                          uint32_t *count)
427{
428    const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
429    if (aps == 0) return PERMISSION_DENIED;
430    return aps->queryDefaultPreProcessing(audioSession, descriptors, count);
431}
432// -------------------------------------------------------------------------
433
434status_t AudioEffect::stringToGuid(const char *str, effect_uuid_t *guid)
435{
436    if (str == NULL || guid == NULL) {
437        return BAD_VALUE;
438    }
439
440    int tmp[10];
441
442    if (sscanf(str, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
443            tmp, tmp+1, tmp+2, tmp+3, tmp+4, tmp+5, tmp+6, tmp+7, tmp+8, tmp+9) < 10) {
444        return BAD_VALUE;
445    }
446    guid->timeLow = (uint32_t)tmp[0];
447    guid->timeMid = (uint16_t)tmp[1];
448    guid->timeHiAndVersion = (uint16_t)tmp[2];
449    guid->clockSeq = (uint16_t)tmp[3];
450    guid->node[0] = (uint8_t)tmp[4];
451    guid->node[1] = (uint8_t)tmp[5];
452    guid->node[2] = (uint8_t)tmp[6];
453    guid->node[3] = (uint8_t)tmp[7];
454    guid->node[4] = (uint8_t)tmp[8];
455    guid->node[5] = (uint8_t)tmp[9];
456
457    return NO_ERROR;
458}
459
460status_t AudioEffect::guidToString(const effect_uuid_t *guid, char *str, size_t maxLen)
461{
462    if (guid == NULL || str == NULL) {
463        return BAD_VALUE;
464    }
465
466    snprintf(str, maxLen, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
467            guid->timeLow,
468            guid->timeMid,
469            guid->timeHiAndVersion,
470            guid->clockSeq,
471            guid->node[0],
472            guid->node[1],
473            guid->node[2],
474            guid->node[3],
475            guid->node[4],
476            guid->node[5]);
477
478    return NO_ERROR;
479}
480
481
482}; // namespace android
483