blob: a043329fa3f088cf4e484b0d79a6951abbe9b02c [file] [log] [blame]
Eric Laurent948235c2010-06-09 00:17:29 -07001/*
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>
Eric Laurent948235c2010-06-09 00:17:29 -070030#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 output
51 )
52 : mStatus(NO_INIT)
53{
Jean-Michel Trivi4cb15cf2010-07-09 12:11:49 -070054 mStatus = set(type, uuid, priority, cbf, user, sessionId, output);
Eric Laurent948235c2010-06-09 00:17:29 -070055}
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 output
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 LOGV("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
Jean-Michel Trivi4cb15cf2010-07-09 12:11:49 -070086 mStatus = set(pType, pUuid, priority, cbf, user, sessionId, output);
Eric Laurent948235c2010-06-09 00:17:29 -070087}
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 output)
96{
97 sp<IEffect> iEffect;
98 sp<IMemory> cblk;
99 int enabled;
100
101 LOGV("set %p mUserData: %p", this, user);
102
103 if (mIEffect != 0) {
104 LOGW("Effect already in use");
105 return INVALID_OPERATION;
106 }
107
108 const sp<IAudioFlinger>& audioFlinger = AudioSystem::get_audio_flinger();
109 if (audioFlinger == 0) {
110 LOGE("set(): Could not get audioflinger");
111 return NO_INIT;
112 }
113
114 if (type == NULL && uuid == NULL) {
115 LOGW("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 memcpy(&mDescriptor.type, EFFECT_UUID_NULL, sizeof(effect_uuid_t));
126 memcpy(&mDescriptor.uuid, EFFECT_UUID_NULL, sizeof(effect_uuid_t));
127
128 if (type != NULL) {
129 memcpy(&mDescriptor.type, type, sizeof(effect_uuid_t));
130 }
131 if (uuid != NULL) {
132 memcpy(&mDescriptor.uuid, uuid, sizeof(effect_uuid_t));
133 }
134
135 mIEffectClient = new EffectClient(this);
136
137 iEffect = audioFlinger->createEffect(getpid(), (effect_descriptor_t *)&mDescriptor,
138 mIEffectClient, priority, output, mSessionId, &mStatus, &mId, &enabled);
139
140 if (iEffect == 0 || (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS)) {
141 LOGE("set(): AudioFlinger could not create effect, status: %d", mStatus);
142 return mStatus;
143 }
144
145 mEnabled = (volatile int32_t)enabled;
146
147 mIEffect = iEffect;
148 cblk = iEffect->getCblk();
149 if (cblk == 0) {
150 mStatus = NO_INIT;
151 LOGE("Could not get control block");
152 return mStatus;
153 }
154
155 mIEffect = iEffect;
156 mCblkMemory = cblk;
157 mCblk = static_cast<effect_param_cblk_t*>(cblk->pointer());
158 int bufOffset = ((sizeof(effect_param_cblk_t) - 1) / sizeof(int) + 1) * sizeof(int);
159 mCblk->buffer = (uint8_t *)mCblk + bufOffset;
160
161 iEffect->asBinder()->linkToDeath(mIEffectClient);
162 LOGV("set() %p OK effect: %s id: %d status %d enabled %d, ", this, mDescriptor.name, mId, mStatus, mEnabled);
163
164 return mStatus;
165}
166
167
168AudioEffect::~AudioEffect()
169{
170 LOGV("Destructor %p", this);
171
172 if (mStatus == NO_ERROR || mStatus == ALREADY_EXISTS) {
Eric Laurent948235c2010-06-09 00:17:29 -0700173 if (mIEffect != NULL) {
174 mIEffect->disconnect();
175 mIEffect->asBinder()->unlinkToDeath(mIEffectClient);
176 }
177 IPCThreadState::self()->flushCommands();
178 }
179 mIEffect.clear();
180 mIEffectClient.clear();
181 mCblkMemory.clear();
182}
183
184
185status_t AudioEffect::initCheck() const
186{
187 return mStatus;
188}
189
190// -------------------------------------------------------------------------
191
192effect_descriptor_t AudioEffect::descriptor() const
193{
194 return mDescriptor;
195}
196
Eric Laurentdf9b81c2010-07-02 08:12:41 -0700197bool AudioEffect::getEnabled() const
Eric Laurent948235c2010-06-09 00:17:29 -0700198{
199 return (mEnabled != 0);
200}
201
Eric Laurentdf9b81c2010-07-02 08:12:41 -0700202status_t AudioEffect::setEnabled(bool enabled)
Eric Laurent948235c2010-06-09 00:17:29 -0700203{
204 if (mStatus != NO_ERROR) {
205 return INVALID_OPERATION;
206 }
Eric Laurent948235c2010-06-09 00:17:29 -0700207
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800208 status_t status = NO_ERROR;
209
210 AutoMutex lock(mLock);
211 if (enabled != mEnabled) {
212 if (enabled) {
213 LOGV("enable %p", this);
214 status = mIEffect->enable();
215 } else {
216 LOGV("disable %p", this);
217 status = mIEffect->disable();
Eric Laurentdf9b81c2010-07-02 08:12:41 -0700218 }
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800219 if (status == NO_ERROR) {
220 mEnabled = enabled;
Eric Laurentdf9b81c2010-07-02 08:12:41 -0700221 }
Eric Laurent948235c2010-06-09 00:17:29 -0700222 }
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800223 return status;
Eric Laurent948235c2010-06-09 00:17:29 -0700224}
225
Eric Laurenta4c72ac2010-07-28 05:40:18 -0700226status_t AudioEffect::command(uint32_t cmdCode,
227 uint32_t cmdSize,
228 void *cmdData,
229 uint32_t *replySize,
230 void *replyData)
Eric Laurent948235c2010-06-09 00:17:29 -0700231{
232 if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
Eric Laurent4d3fb502010-09-24 11:52:04 -0700233 LOGV("command() bad status %d", mStatus);
Eric Laurent948235c2010-06-09 00:17:29 -0700234 return INVALID_OPERATION;
235 }
236
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800237 if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
238 if (mEnabled == (cmdCode == EFFECT_CMD_ENABLE)) {
239 return NO_ERROR;
240 }
241 if (replySize == NULL || *replySize != sizeof(status_t) || replyData == NULL) {
242 return BAD_VALUE;
243 }
244 mLock.lock();
Eric Laurent4d3fb502010-09-24 11:52:04 -0700245 }
246
Eric Laurent2a6b80b2010-07-29 23:43:43 -0700247 status_t status = mIEffect->command(cmdCode, cmdSize, cmdData, replySize, replyData);
Eric Laurent4d3fb502010-09-24 11:52:04 -0700248
249 if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800250 if (status == NO_ERROR) {
251 status = *(status_t *)replyData;
Eric Laurent4d3fb502010-09-24 11:52:04 -0700252 }
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800253 if (status == NO_ERROR) {
254 mEnabled = (cmdCode == EFFECT_CMD_ENABLE);
Eric Laurent4d3fb502010-09-24 11:52:04 -0700255 }
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800256 mLock.unlock();
Eric Laurent2a6b80b2010-07-29 23:43:43 -0700257 }
258
Eric Laurent2a6b80b2010-07-29 23:43:43 -0700259 return status;
Eric Laurent948235c2010-06-09 00:17:29 -0700260}
261
262
263status_t AudioEffect::setParameter(effect_param_t *param)
264{
265 if (mStatus != NO_ERROR) {
266 return INVALID_OPERATION;
267 }
268
269 if (param == NULL || param->psize == 0 || param->vsize == 0) {
270 return BAD_VALUE;
271 }
272
Eric Laurenta4c72ac2010-07-28 05:40:18 -0700273 uint32_t size = sizeof(int);
274 uint32_t psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
Eric Laurent948235c2010-06-09 00:17:29 -0700275
276 LOGV("setParameter: param: %d, param2: %d", *(int *)param->data, (param->psize == 8) ? *((int *)param->data + 1): -1);
277
278 return mIEffect->command(EFFECT_CMD_SET_PARAM, sizeof (effect_param_t) + psize, param, &size, &param->status);
279}
280
281status_t AudioEffect::setParameterDeferred(effect_param_t *param)
282{
283 if (mStatus != NO_ERROR) {
284 return INVALID_OPERATION;
285 }
286
287 if (param == NULL || param->psize == 0 || param->vsize == 0) {
288 return BAD_VALUE;
289 }
290
291 Mutex::Autolock _l(mCblk->lock);
292
293 int psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
294 int size = ((sizeof(effect_param_t) + psize - 1) / sizeof(int) + 1) * sizeof(int);
295
296 if (mCblk->clientIndex + size > EFFECT_PARAM_BUFFER_SIZE) {
297 return NO_MEMORY;
298 }
299 int *p = (int *)(mCblk->buffer + mCblk->clientIndex);
300 *p++ = size;
301 memcpy(p, param, sizeof(effect_param_t) + psize);
302 mCblk->clientIndex += size;
303
304 return NO_ERROR;
305}
306
307status_t AudioEffect::setParameterCommit()
308{
309 if (mStatus != NO_ERROR) {
310 return INVALID_OPERATION;
311 }
312
313 Mutex::Autolock _l(mCblk->lock);
314 if (mCblk->clientIndex == 0) {
315 return INVALID_OPERATION;
316 }
Eric Laurenta4c72ac2010-07-28 05:40:18 -0700317 uint32_t size = 0;
Eric Laurent948235c2010-06-09 00:17:29 -0700318 return mIEffect->command(EFFECT_CMD_SET_PARAM_COMMIT, 0, NULL, &size, NULL);
319}
320
321status_t AudioEffect::getParameter(effect_param_t *param)
322{
323 if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
324 return INVALID_OPERATION;
325 }
326
327 if (param == NULL || param->psize == 0 || param->vsize == 0) {
328 return BAD_VALUE;
329 }
330
331 LOGV("getParameter: param: %d, param2: %d", *(int *)param->data, (param->psize == 8) ? *((int *)param->data + 1): -1);
332
Eric Laurenta4c72ac2010-07-28 05:40:18 -0700333 uint32_t psize = sizeof(effect_param_t) + ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
Eric Laurent948235c2010-06-09 00:17:29 -0700334
335 return mIEffect->command(EFFECT_CMD_GET_PARAM, sizeof(effect_param_t) + param->psize, param, &psize, param);
336}
337
338
339// -------------------------------------------------------------------------
340
341void AudioEffect::binderDied()
342{
343 LOGW("IEffect died");
344 mStatus = NO_INIT;
345 if (mCbf) {
346 status_t status = DEAD_OBJECT;
347 mCbf(EVENT_ERROR, mUserData, &status);
348 }
349 mIEffect.clear();
350}
351
352// -------------------------------------------------------------------------
353
354void AudioEffect::controlStatusChanged(bool controlGranted)
355{
356 LOGV("controlStatusChanged %p control %d callback %p mUserData %p", this, controlGranted, mCbf, 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) {
367 mCbf(EVENT_CONTROL_STATUS_CHANGED, mUserData, &controlGranted);
368 }
369}
370
371void AudioEffect::enableStatusChanged(bool enabled)
372{
Eric Laurentdf9b81c2010-07-02 08:12:41 -0700373 LOGV("enableStatusChanged %p enabled %d mCbf %p", this, enabled, mCbf);
Eric Laurent948235c2010-06-09 00:17:29 -0700374 if (mStatus == ALREADY_EXISTS) {
Eric Laurentf3d6dd02010-11-18 08:40:16 -0800375 mEnabled = enabled;
Eric Laurent948235c2010-06-09 00:17:29 -0700376 if (mCbf) {
377 mCbf(EVENT_ENABLE_STATUS_CHANGED, mUserData, &enabled);
378 }
379 }
380}
381
Eric Laurenta4c72ac2010-07-28 05:40:18 -0700382void AudioEffect::commandExecuted(uint32_t cmdCode,
383 uint32_t cmdSize,
384 void *cmdData,
385 uint32_t replySize,
386 void *replyData)
Eric Laurent948235c2010-06-09 00:17:29 -0700387{
388 if (cmdData == NULL || replyData == NULL) {
389 return;
390 }
391
392 if (mCbf && 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::loadEffectLibrary(const char *libPath, int *handle)
402{
403 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
404 if (af == 0) return PERMISSION_DENIED;
405 return af->loadEffectLibrary(libPath, handle);
406}
407
408status_t AudioEffect::unloadEffectLibrary(int handle)
409{
410 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
411 if (af == 0) return PERMISSION_DENIED;
412 return af->unloadEffectLibrary(handle);
413}
414
415status_t AudioEffect::queryNumberEffects(uint32_t *numEffects)
416{
417 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
418 if (af == 0) return PERMISSION_DENIED;
419 return af->queryNumberEffects(numEffects);
420}
421
Eric Laurent53334cd2010-06-23 17:38:20 -0700422status_t AudioEffect::queryEffect(uint32_t index, effect_descriptor_t *descriptor)
Eric Laurent948235c2010-06-09 00:17:29 -0700423{
424 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
425 if (af == 0) return PERMISSION_DENIED;
Eric Laurent53334cd2010-06-23 17:38:20 -0700426 return af->queryEffect(index, descriptor);
Eric Laurent948235c2010-06-09 00:17:29 -0700427}
428
429status_t AudioEffect::getEffectDescriptor(effect_uuid_t *uuid, effect_descriptor_t *descriptor)
430{
431 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
432 if (af == 0) return PERMISSION_DENIED;
433 return af->getEffectDescriptor(uuid, descriptor);
434}
435
436// -------------------------------------------------------------------------
437
438status_t AudioEffect::stringToGuid(const char *str, effect_uuid_t *guid)
439{
440 if (str == NULL || guid == NULL) {
441 return BAD_VALUE;
442 }
443
444 int tmp[10];
445
446 if (sscanf(str, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
447 tmp, tmp+1, tmp+2, tmp+3, tmp+4, tmp+5, tmp+6, tmp+7, tmp+8, tmp+9) < 10) {
448 return BAD_VALUE;
449 }
450 guid->timeLow = (uint32_t)tmp[0];
451 guid->timeMid = (uint16_t)tmp[1];
452 guid->timeHiAndVersion = (uint16_t)tmp[2];
453 guid->clockSeq = (uint16_t)tmp[3];
454 guid->node[0] = (uint8_t)tmp[4];
455 guid->node[1] = (uint8_t)tmp[5];
456 guid->node[2] = (uint8_t)tmp[6];
457 guid->node[3] = (uint8_t)tmp[7];
458 guid->node[4] = (uint8_t)tmp[8];
459 guid->node[5] = (uint8_t)tmp[9];
460
461 return NO_ERROR;
462}
463
464status_t AudioEffect::guidToString(const effect_uuid_t *guid, char *str, size_t maxLen)
465{
466 if (guid == NULL || str == NULL) {
467 return BAD_VALUE;
468 }
469
470 snprintf(str, maxLen, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
471 guid->timeLow,
472 guid->timeMid,
473 guid->timeHiAndVersion,
474 guid->clockSeq,
475 guid->node[0],
476 guid->node[1],
477 guid->node[2],
478 guid->node[3],
479 guid->node[4],
480 guid->node[5]);
481
482 return NO_ERROR;
483}
484
485
486}; // namespace android
487