blob: 62f863ddaeb8b87b331742cbe03eabc18bcf53cd [file] [log] [blame]
Eric Laurent801a1182010-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
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -080026#include <android/media/IAudioPolicyService.h>
Eric Laurent801a1182010-06-09 00:17:29 -070027#include <binder/IPCThreadState.h>
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -080028#include <media/AidlConversion.h>
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -070029#include <media/AudioEffect.h>
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -080030#include <media/PolicyAidlConversion.h>
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -070031#include <media/ShmemCompat.h>
32#include <private/media/AudioEffectShared.h>
33#include <utils/Log.h>
Eric Laurent801a1182010-06-09 00:17:29 -070034
35namespace android {
Andy Hung1131b6e2020-12-08 20:47:45 -080036using aidl_utils::statusTFromBinderStatus;
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -070037using binder::Status;
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -080038using media::IAudioPolicyService;
Mikhail Naganovddceecc2021-09-03 13:58:56 -070039using media::audio::common::AudioSource;
40using media::audio::common::AudioUuid;
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -070041
42namespace {
43
44// Copy from a raw pointer + size into a vector of bytes.
45void appendToBuffer(const void* data,
46 size_t size,
47 std::vector<uint8_t>* buffer) {
48 const uint8_t* p = reinterpret_cast<const uint8_t*>(data);
49 buffer->insert(buffer->end(), p, p + size);
50}
51
52} // namespace
53
Eric Laurent801a1182010-06-09 00:17:29 -070054// ---------------------------------------------------------------------------
55
Svet Ganov33761132021-05-13 22:51:08 +000056AudioEffect::AudioEffect(const android::content::AttributionSourceState& attributionSource)
57 : mClientAttributionSource(attributionSource)
Eric Laurent801a1182010-06-09 00:17:29 -070058{
59}
60
Eric Laurent801a1182010-06-09 00:17:29 -070061status_t AudioEffect::set(const effect_uuid_t *type,
62 const effect_uuid_t *uuid,
63 int32_t priority,
64 effect_callback_t cbf,
65 void* user,
Glenn Kastend848eb42016-03-08 13:42:11 -080066 audio_session_t sessionId,
Eric Laurent94876032019-11-13 12:45:28 -080067 audio_io_handle_t io,
Eric Laurent2fe0acd2020-03-13 14:30:46 -070068 const AudioDeviceTypeAddr& device,
Eric Laurentde8caf42021-08-11 17:19:25 +020069 bool probe,
70 bool notifyFramesProcessed)
Eric Laurent801a1182010-06-09 00:17:29 -070071{
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -070072 sp<media::IEffect> iEffect;
Eric Laurent801a1182010-06-09 00:17:29 -070073 sp<IMemory> cblk;
74 int enabled;
75
Steve Block3856b092011-10-20 11:56:00 +010076 ALOGV("set %p mUserData: %p uuid: %p timeLow %08x", this, user, type, type ? type->timeLow : 0);
Eric Laurent801a1182010-06-09 00:17:29 -070077
78 if (mIEffect != 0) {
Steve Block5ff1dd52012-01-05 23:22:43 +000079 ALOGW("Effect already in use");
Eric Laurent801a1182010-06-09 00:17:29 -070080 return INVALID_OPERATION;
81 }
82
Eric Laurent94876032019-11-13 12:45:28 -080083 if (sessionId == AUDIO_SESSION_DEVICE && io != AUDIO_IO_HANDLE_NONE) {
84 ALOGW("IO handle should not be specified for device effect");
85 return BAD_VALUE;
86 }
Eric Laurent801a1182010-06-09 00:17:29 -070087 const sp<IAudioFlinger>& audioFlinger = AudioSystem::get_audio_flinger();
88 if (audioFlinger == 0) {
Steve Block29357bc2012-01-06 19:20:56 +000089 ALOGE("set(): Could not get audioflinger");
Eric Laurent801a1182010-06-09 00:17:29 -070090 return NO_INIT;
91 }
92
93 if (type == NULL && uuid == NULL) {
Steve Block5ff1dd52012-01-05 23:22:43 +000094 ALOGW("Must specify at least type or uuid");
Eric Laurent801a1182010-06-09 00:17:29 -070095 return BAD_VALUE;
96 }
Eric Laurent2fe0acd2020-03-13 14:30:46 -070097 mProbe = probe;
Eric Laurent801a1182010-06-09 00:17:29 -070098 mPriority = priority;
99 mCbf = cbf;
100 mUserData = user;
101 mSessionId = sessionId;
102
103 memset(&mDescriptor, 0, sizeof(effect_descriptor_t));
Glenn Kastena189a682012-02-20 12:16:30 -0800104 mDescriptor.type = *(type != NULL ? type : EFFECT_UUID_NULL);
105 mDescriptor.uuid = *(uuid != NULL ? uuid : EFFECT_UUID_NULL);
Eric Laurent801a1182010-06-09 00:17:29 -0700106
Svet Ganov33761132021-05-13 22:51:08 +0000107 // TODO b/182392769: use attribution source util
Eric Laurent801a1182010-06-09 00:17:29 -0700108 mIEffectClient = new EffectClient(this);
Philip P. Moltmannbda45752020-07-17 16:41:18 -0700109 pid_t pid = IPCThreadState::self()->getCallingPid();
Svet Ganov33761132021-05-13 22:51:08 +0000110 mClientAttributionSource.pid = VALUE_OR_RETURN_STATUS(legacy2aidl_pid_t_int32_t(pid));
Philip P. Moltmannbda45752020-07-17 16:41:18 -0700111 pid_t uid = IPCThreadState::self()->getCallingUid();
Svet Ganov33761132021-05-13 22:51:08 +0000112 mClientAttributionSource.uid = VALUE_OR_RETURN_STATUS(legacy2aidl_uid_t_int32_t(uid));
Eric Laurent801a1182010-06-09 00:17:29 -0700113
Ytai Ben-Tsvice182942020-11-04 14:48:01 -0800114 media::CreateEffectRequest request;
115 request.desc = VALUE_OR_RETURN_STATUS(
116 legacy2aidl_effect_descriptor_t_EffectDescriptor(mDescriptor));
117 request.client = mIEffectClient;
118 request.priority = priority;
119 request.output = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_io_handle_t_int32_t(io));
120 request.sessionId = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_session_t_int32_t(mSessionId));
121 request.device = VALUE_OR_RETURN_STATUS(legacy2aidl_AudioDeviceTypeAddress(device));
Svet Ganov33761132021-05-13 22:51:08 +0000122 request.attributionSource = mClientAttributionSource;
Ytai Ben-Tsvice182942020-11-04 14:48:01 -0800123 request.probe = probe;
Eric Laurentde8caf42021-08-11 17:19:25 +0200124 request.notifyFramesProcessed = notifyFramesProcessed;
Ytai Ben-Tsvice182942020-11-04 14:48:01 -0800125
126 media::CreateEffectResponse response;
127
128 mStatus = audioFlinger->createEffect(request, &response);
129
130 if (mStatus == OK) {
131 mId = response.id;
132 enabled = response.enabled;
133 iEffect = response.effect;
134 mDescriptor = VALUE_OR_RETURN_STATUS(
135 aidl2legacy_EffectDescriptor_effect_descriptor_t(response.desc));
136 }
Eric Laurent801a1182010-06-09 00:17:29 -0700137
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700138 // In probe mode, we stop here and return the status: the IEffect interface to
139 // audio flinger will not be retained. initCheck() will return the creation status
140 // but all other APIs will return invalid operation.
141 if (probe || iEffect == 0 || (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS)) {
Mikhail Naganovabd6e9d2020-03-23 22:25:56 +0000142 char typeBuffer[64] = {}, uuidBuffer[64] = {};
Mikhail Naganov424c4f52017-07-19 17:54:29 -0700143 guidToString(type, typeBuffer, sizeof(typeBuffer));
144 guidToString(uuid, uuidBuffer, sizeof(uuidBuffer));
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700145 ALOGE_IF(!probe, "set(): AudioFlinger could not create effect %s / %s, status: %d",
Mikhail Naganovabd6e9d2020-03-23 22:25:56 +0000146 type != nullptr ? typeBuffer : "NULL",
147 uuid != nullptr ? uuidBuffer : "NULL",
148 mStatus);
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700149 if (!probe && iEffect == 0) {
Eric Laurenteecd7652015-06-04 16:20:16 -0700150 mStatus = NO_INIT;
151 }
Eric Laurent801a1182010-06-09 00:17:29 -0700152 return mStatus;
153 }
154
155 mEnabled = (volatile int32_t)enabled;
156
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700157 if (media::SharedFileRegion shmem;
158 !iEffect->getCblk(&shmem).isOk()
159 || !convertSharedFileRegionToIMemory(shmem, &cblk)
160 || cblk == 0) {
Eric Laurent801a1182010-06-09 00:17:29 -0700161 mStatus = NO_INIT;
Steve Block29357bc2012-01-06 19:20:56 +0000162 ALOGE("Could not get control block");
Eric Laurent801a1182010-06-09 00:17:29 -0700163 return mStatus;
164 }
165
Eric Laurenteecd7652015-06-04 16:20:16 -0700166 mIEffect = iEffect;
Eric Laurent801a1182010-06-09 00:17:29 -0700167 mCblkMemory = cblk;
Ytai Ben-Tsvi7dd39722019-09-05 15:14:30 -0700168 // TODO: Using unsecurePointer() has some associated security pitfalls
169 // (see declaration for details).
170 // Either document why it is safe in this case or address the
171 // issue (e.g. by copying).
172 mCblk = static_cast<effect_param_cblk_t*>(cblk->unsecurePointer());
Eric Laurent801a1182010-06-09 00:17:29 -0700173 int bufOffset = ((sizeof(effect_param_cblk_t) - 1) / sizeof(int) + 1) * sizeof(int);
174 mCblk->buffer = (uint8_t *)mCblk + bufOffset;
175
Marco Nelissen06b46062014-11-14 07:58:25 -0800176 IInterface::asBinder(iEffect)->linkToDeath(mIEffectClient);
Jean-Michel Trivia0fd9ca2014-09-18 14:07:18 -0700177 ALOGV("set() %p OK effect: %s id: %d status %d enabled %d pid %d", this, mDescriptor.name, mId,
Svet Ganov33761132021-05-13 22:51:08 +0000178 mStatus, mEnabled, mClientAttributionSource.pid);
Jean-Michel Trivia0fd9ca2014-09-18 14:07:18 -0700179
Eric Laurent3f75a5b2019-11-12 15:55:51 -0800180 if (!audio_is_global_session(mSessionId)) {
Philip P. Moltmannbda45752020-07-17 16:41:18 -0700181 AudioSystem::acquireAudioSessionId(mSessionId, pid, uid);
Jean-Michel Trivia0fd9ca2014-09-18 14:07:18 -0700182 }
Eric Laurent801a1182010-06-09 00:17:29 -0700183
184 return mStatus;
185}
186
Mikhail Naganov416fffe2020-07-31 17:36:08 -0700187status_t AudioEffect::set(const char *typeStr,
188 const char *uuidStr,
189 int32_t priority,
190 effect_callback_t cbf,
191 void* user,
192 audio_session_t sessionId,
193 audio_io_handle_t io,
194 const AudioDeviceTypeAddr& device,
Eric Laurentde8caf42021-08-11 17:19:25 +0200195 bool probe,
196 bool notifyFramesProcessed)
Mikhail Naganov416fffe2020-07-31 17:36:08 -0700197{
198 effect_uuid_t type;
199 effect_uuid_t *pType = nullptr;
200 effect_uuid_t uuid;
201 effect_uuid_t *pUuid = nullptr;
202
203 ALOGV("AudioEffect::set string\n - type: %s\n - uuid: %s",
204 typeStr ? typeStr : "nullptr", uuidStr ? uuidStr : "nullptr");
205
206 if (stringToGuid(typeStr, &type) == NO_ERROR) {
207 pType = &type;
208 }
209 if (stringToGuid(uuidStr, &uuid) == NO_ERROR) {
210 pUuid = &uuid;
211 }
212
Eric Laurentde8caf42021-08-11 17:19:25 +0200213 return set(pType, pUuid, priority, cbf, user, sessionId, io,
214 device, probe, notifyFramesProcessed);
Mikhail Naganov416fffe2020-07-31 17:36:08 -0700215}
216
Eric Laurent801a1182010-06-09 00:17:29 -0700217
218AudioEffect::~AudioEffect()
219{
Steve Block3856b092011-10-20 11:56:00 +0100220 ALOGV("Destructor %p", this);
Eric Laurent801a1182010-06-09 00:17:29 -0700221
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700222 if (!mProbe && (mStatus == NO_ERROR || mStatus == ALREADY_EXISTS)) {
Eric Laurent3f75a5b2019-11-12 15:55:51 -0800223 if (!audio_is_global_session(mSessionId)) {
Philip P. Moltmannbda45752020-07-17 16:41:18 -0700224 AudioSystem::releaseAudioSessionId(mSessionId,
Svet Ganov33761132021-05-13 22:51:08 +0000225 VALUE_OR_FATAL(aidl2legacy_int32_t_pid_t(mClientAttributionSource.pid)));
Jean-Michel Trivia0fd9ca2014-09-18 14:07:18 -0700226 }
Eric Laurent801a1182010-06-09 00:17:29 -0700227 if (mIEffect != NULL) {
228 mIEffect->disconnect();
Marco Nelissen06b46062014-11-14 07:58:25 -0800229 IInterface::asBinder(mIEffect)->unlinkToDeath(mIEffectClient);
Eric Laurent801a1182010-06-09 00:17:29 -0700230 }
Eric Laurenteecd7652015-06-04 16:20:16 -0700231 mIEffect.clear();
232 mCblkMemory.clear();
Eric Laurent801a1182010-06-09 00:17:29 -0700233 }
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700234 mIEffectClient.clear();
235 IPCThreadState::self()->flushCommands();
Eric Laurent801a1182010-06-09 00:17:29 -0700236}
237
238
239status_t AudioEffect::initCheck() const
240{
241 return mStatus;
242}
243
244// -------------------------------------------------------------------------
245
246effect_descriptor_t AudioEffect::descriptor() const
247{
248 return mDescriptor;
249}
250
Eric Laurentda7581b2010-07-02 08:12:41 -0700251bool AudioEffect::getEnabled() const
Eric Laurent801a1182010-06-09 00:17:29 -0700252{
253 return (mEnabled != 0);
254}
255
Eric Laurentda7581b2010-07-02 08:12:41 -0700256status_t AudioEffect::setEnabled(bool enabled)
Eric Laurent801a1182010-06-09 00:17:29 -0700257{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700258 if (mProbe) {
259 return INVALID_OPERATION;
260 }
Eric Laurent801a1182010-06-09 00:17:29 -0700261 if (mStatus != NO_ERROR) {
Glenn Kastenf063b492012-02-17 16:24:10 -0800262 return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700263 }
Eric Laurent801a1182010-06-09 00:17:29 -0700264
Eric Laurentf5aafb22010-11-18 08:40:16 -0800265 status_t status = NO_ERROR;
Eric Laurentf5aafb22010-11-18 08:40:16 -0800266 AutoMutex lock(mLock);
267 if (enabled != mEnabled) {
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700268 Status bs;
269
Eric Laurentf5aafb22010-11-18 08:40:16 -0800270 if (enabled) {
Steve Block3856b092011-10-20 11:56:00 +0100271 ALOGV("enable %p", this);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700272 bs = mIEffect->enable(&status);
Eric Laurentf5aafb22010-11-18 08:40:16 -0800273 } else {
Steve Block3856b092011-10-20 11:56:00 +0100274 ALOGV("disable %p", this);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700275 bs = mIEffect->disable(&status);
276 }
277 if (!bs.isOk()) {
Andy Hung1131b6e2020-12-08 20:47:45 -0800278 status = statusTFromBinderStatus(bs);
Eric Laurentda7581b2010-07-02 08:12:41 -0700279 }
Eric Laurentf5aafb22010-11-18 08:40:16 -0800280 if (status == NO_ERROR) {
281 mEnabled = enabled;
Eric Laurentda7581b2010-07-02 08:12:41 -0700282 }
Eric Laurent801a1182010-06-09 00:17:29 -0700283 }
Eric Laurentf5aafb22010-11-18 08:40:16 -0800284 return status;
Eric Laurent801a1182010-06-09 00:17:29 -0700285}
286
Eric Laurent25f43952010-07-28 05:40:18 -0700287status_t AudioEffect::command(uint32_t cmdCode,
288 uint32_t cmdSize,
289 void *cmdData,
290 uint32_t *replySize,
291 void *replyData)
Eric Laurent801a1182010-06-09 00:17:29 -0700292{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700293 if (mProbe) {
294 return INVALID_OPERATION;
295 }
Eric Laurent801a1182010-06-09 00:17:29 -0700296 if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
Steve Block3856b092011-10-20 11:56:00 +0100297 ALOGV("command() bad status %d", mStatus);
John Grossmanaf7d8182012-01-11 12:23:42 -0800298 return mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700299 }
300
Eric Laurentf5aafb22010-11-18 08:40:16 -0800301 if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
302 if (mEnabled == (cmdCode == EFFECT_CMD_ENABLE)) {
303 return NO_ERROR;
304 }
305 if (replySize == NULL || *replySize != sizeof(status_t) || replyData == NULL) {
306 return BAD_VALUE;
307 }
308 mLock.lock();
Eric Laurent0fa449c2010-09-24 11:52:04 -0700309 }
310
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700311 std::vector<uint8_t> data;
312 appendToBuffer(cmdData, cmdSize, &data);
313
314 status_t status;
315 std::vector<uint8_t> response;
316
317 Status bs = mIEffect->command(cmdCode, data, *replySize, &response, &status);
318 if (!bs.isOk()) {
Andy Hung1131b6e2020-12-08 20:47:45 -0800319 status = statusTFromBinderStatus(bs);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700320 }
321 if (status == NO_ERROR) {
322 memcpy(replyData, response.data(), response.size());
323 *replySize = response.size();
324 }
Eric Laurent0fa449c2010-09-24 11:52:04 -0700325
326 if (cmdCode == EFFECT_CMD_ENABLE || cmdCode == EFFECT_CMD_DISABLE) {
Eric Laurentf5aafb22010-11-18 08:40:16 -0800327 if (status == NO_ERROR) {
328 status = *(status_t *)replyData;
Eric Laurent0fa449c2010-09-24 11:52:04 -0700329 }
Eric Laurentf5aafb22010-11-18 08:40:16 -0800330 if (status == NO_ERROR) {
331 mEnabled = (cmdCode == EFFECT_CMD_ENABLE);
Eric Laurent0fa449c2010-09-24 11:52:04 -0700332 }
Eric Laurentf5aafb22010-11-18 08:40:16 -0800333 mLock.unlock();
Eric Laurent8569f0d2010-07-29 23:43:43 -0700334 }
335
Eric Laurent8569f0d2010-07-29 23:43:43 -0700336 return status;
Eric Laurent801a1182010-06-09 00:17:29 -0700337}
338
Eric Laurent801a1182010-06-09 00:17:29 -0700339status_t AudioEffect::setParameter(effect_param_t *param)
340{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700341 if (mProbe) {
342 return INVALID_OPERATION;
343 }
Eric Laurent801a1182010-06-09 00:17:29 -0700344 if (mStatus != NO_ERROR) {
Glenn Kastenf063b492012-02-17 16:24:10 -0800345 return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700346 }
347
348 if (param == NULL || param->psize == 0 || param->vsize == 0) {
349 return BAD_VALUE;
350 }
351
Eric Laurent25f43952010-07-28 05:40:18 -0700352 uint32_t psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
Eric Laurent801a1182010-06-09 00:17:29 -0700353
Glenn Kasten85ab62c2012-11-01 11:11:38 -0700354 ALOGV("setParameter: param: %d, param2: %d", *(int *)param->data,
355 (param->psize == 8) ? *((int *)param->data + 1): -1);
Eric Laurent801a1182010-06-09 00:17:29 -0700356
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700357 std::vector<uint8_t> cmd;
358 appendToBuffer(param, sizeof(effect_param_t) + psize, &cmd);
359 std::vector<uint8_t> response;
360 status_t status;
361 Status bs = mIEffect->command(EFFECT_CMD_SET_PARAM,
362 cmd,
363 sizeof(int),
364 &response,
365 &status);
366 if (!bs.isOk()) {
Andy Hung1131b6e2020-12-08 20:47:45 -0800367 status = statusTFromBinderStatus(bs);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700368 return status;
369 }
370 assert(response.size() == sizeof(int));
371 memcpy(&param->status, response.data(), response.size());
372 return status;
Eric Laurent801a1182010-06-09 00:17:29 -0700373}
374
375status_t AudioEffect::setParameterDeferred(effect_param_t *param)
376{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700377 if (mProbe) {
378 return INVALID_OPERATION;
379 }
Eric Laurent801a1182010-06-09 00:17:29 -0700380 if (mStatus != NO_ERROR) {
Glenn Kastenf063b492012-02-17 16:24:10 -0800381 return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700382 }
383
384 if (param == NULL || param->psize == 0 || param->vsize == 0) {
385 return BAD_VALUE;
386 }
387
388 Mutex::Autolock _l(mCblk->lock);
389
390 int psize = ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) + param->vsize;
391 int size = ((sizeof(effect_param_t) + psize - 1) / sizeof(int) + 1) * sizeof(int);
392
393 if (mCblk->clientIndex + size > EFFECT_PARAM_BUFFER_SIZE) {
394 return NO_MEMORY;
395 }
396 int *p = (int *)(mCblk->buffer + mCblk->clientIndex);
397 *p++ = size;
398 memcpy(p, param, sizeof(effect_param_t) + psize);
399 mCblk->clientIndex += size;
400
401 return NO_ERROR;
402}
403
404status_t AudioEffect::setParameterCommit()
405{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700406 if (mProbe) {
407 return INVALID_OPERATION;
408 }
Eric Laurent801a1182010-06-09 00:17:29 -0700409 if (mStatus != NO_ERROR) {
Glenn Kastenf063b492012-02-17 16:24:10 -0800410 return (mStatus == ALREADY_EXISTS) ? (status_t) INVALID_OPERATION : mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700411 }
412
413 Mutex::Autolock _l(mCblk->lock);
414 if (mCblk->clientIndex == 0) {
415 return INVALID_OPERATION;
416 }
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700417 std::vector<uint8_t> cmd;
418 std::vector<uint8_t> response;
419 status_t status;
420 Status bs = mIEffect->command(EFFECT_CMD_SET_PARAM_COMMIT,
421 cmd,
422 0,
423 &response,
424 &status);
425 if (!bs.isOk()) {
Andy Hung1131b6e2020-12-08 20:47:45 -0800426 status = statusTFromBinderStatus(bs);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700427 }
428 return status;
Eric Laurent801a1182010-06-09 00:17:29 -0700429}
430
431status_t AudioEffect::getParameter(effect_param_t *param)
432{
Eric Laurent2fe0acd2020-03-13 14:30:46 -0700433 if (mProbe) {
434 return INVALID_OPERATION;
435 }
Eric Laurent801a1182010-06-09 00:17:29 -0700436 if (mStatus != NO_ERROR && mStatus != ALREADY_EXISTS) {
John Grossmanaf7d8182012-01-11 12:23:42 -0800437 return mStatus;
Eric Laurent801a1182010-06-09 00:17:29 -0700438 }
439
440 if (param == NULL || param->psize == 0 || param->vsize == 0) {
441 return BAD_VALUE;
442 }
443
Glenn Kasten85ab62c2012-11-01 11:11:38 -0700444 ALOGV("getParameter: param: %d, param2: %d", *(int *)param->data,
445 (param->psize == 8) ? *((int *)param->data + 1): -1);
Eric Laurent801a1182010-06-09 00:17:29 -0700446
Glenn Kasten85ab62c2012-11-01 11:11:38 -0700447 uint32_t psize = sizeof(effect_param_t) + ((param->psize - 1) / sizeof(int) + 1) * sizeof(int) +
448 param->vsize;
Eric Laurent801a1182010-06-09 00:17:29 -0700449
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700450 status_t status;
451 std::vector<uint8_t> cmd;
452 std::vector<uint8_t> response;
453 appendToBuffer(param, sizeof(effect_param_t) + param->psize, &cmd);
454
455 Status bs = mIEffect->command(EFFECT_CMD_GET_PARAM, cmd, psize, &response, &status);
456 if (!bs.isOk()) {
Andy Hung1131b6e2020-12-08 20:47:45 -0800457 status = statusTFromBinderStatus(bs);
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700458 return status;
459 }
460 memcpy(param, response.data(), response.size());
461 return status;
Eric Laurent801a1182010-06-09 00:17:29 -0700462}
463
464
465// -------------------------------------------------------------------------
466
467void AudioEffect::binderDied()
468{
Steve Block5ff1dd52012-01-05 23:22:43 +0000469 ALOGW("IEffect died");
John Grossmanaf7d8182012-01-11 12:23:42 -0800470 mStatus = DEAD_OBJECT;
Glenn Kastena0d68332012-01-27 16:47:15 -0800471 if (mCbf != NULL) {
Eric Laurent801a1182010-06-09 00:17:29 -0700472 status_t status = DEAD_OBJECT;
473 mCbf(EVENT_ERROR, mUserData, &status);
474 }
475 mIEffect.clear();
476}
477
478// -------------------------------------------------------------------------
479
480void AudioEffect::controlStatusChanged(bool controlGranted)
481{
Glenn Kasten85ab62c2012-11-01 11:11:38 -0700482 ALOGV("controlStatusChanged %p control %d callback %p mUserData %p", this, controlGranted, mCbf,
483 mUserData);
Eric Laurent801a1182010-06-09 00:17:29 -0700484 if (controlGranted) {
485 if (mStatus == ALREADY_EXISTS) {
486 mStatus = NO_ERROR;
487 }
488 } else {
489 if (mStatus == NO_ERROR) {
490 mStatus = ALREADY_EXISTS;
491 }
492 }
Glenn Kastena0d68332012-01-27 16:47:15 -0800493 if (mCbf != NULL) {
Eric Laurent801a1182010-06-09 00:17:29 -0700494 mCbf(EVENT_CONTROL_STATUS_CHANGED, mUserData, &controlGranted);
495 }
496}
497
498void AudioEffect::enableStatusChanged(bool enabled)
499{
Steve Block3856b092011-10-20 11:56:00 +0100500 ALOGV("enableStatusChanged %p enabled %d mCbf %p", this, enabled, mCbf);
Eric Laurent801a1182010-06-09 00:17:29 -0700501 if (mStatus == ALREADY_EXISTS) {
Eric Laurentf5aafb22010-11-18 08:40:16 -0800502 mEnabled = enabled;
Glenn Kastena0d68332012-01-27 16:47:15 -0800503 if (mCbf != NULL) {
Eric Laurent801a1182010-06-09 00:17:29 -0700504 mCbf(EVENT_ENABLE_STATUS_CHANGED, mUserData, &enabled);
505 }
506 }
507}
508
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700509void AudioEffect::commandExecuted(int32_t cmdCode,
510 const std::vector<uint8_t>& cmdData,
511 const std::vector<uint8_t>& replyData)
Eric Laurent801a1182010-06-09 00:17:29 -0700512{
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700513 if (cmdData.empty() || replyData.empty()) {
Eric Laurent801a1182010-06-09 00:17:29 -0700514 return;
515 }
516
Glenn Kastena0d68332012-01-27 16:47:15 -0800517 if (mCbf != NULL && cmdCode == EFFECT_CMD_SET_PARAM) {
Ytai Ben-Tsvi9cd89812020-07-01 17:12:06 -0700518 std::vector<uint8_t> cmdDataCopy(cmdData);
519 effect_param_t* cmd = reinterpret_cast<effect_param_t *>(cmdDataCopy.data());
520 cmd->status = *reinterpret_cast<const int32_t *>(replyData.data());
Eric Laurent801a1182010-06-09 00:17:29 -0700521 mCbf(EVENT_PARAMETER_CHANGED, mUserData, cmd);
522 }
523}
524
Eric Laurentde8caf42021-08-11 17:19:25 +0200525void AudioEffect::framesProcessed(int32_t frames)
526{
527 if (mCbf != NULL) {
528 mCbf(EVENT_FRAMES_PROCESSED, mUserData, &frames);
529 }
530}
531
Eric Laurent801a1182010-06-09 00:17:29 -0700532// -------------------------------------------------------------------------
533
Eric Laurent801a1182010-06-09 00:17:29 -0700534status_t AudioEffect::queryNumberEffects(uint32_t *numEffects)
535{
536 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
537 if (af == 0) return PERMISSION_DENIED;
538 return af->queryNumberEffects(numEffects);
539}
540
Eric Laurentffe9c252010-06-23 17:38:20 -0700541status_t AudioEffect::queryEffect(uint32_t index, effect_descriptor_t *descriptor)
Eric Laurent801a1182010-06-09 00:17:29 -0700542{
543 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
544 if (af == 0) return PERMISSION_DENIED;
Eric Laurentffe9c252010-06-23 17:38:20 -0700545 return af->queryEffect(index, descriptor);
Eric Laurent801a1182010-06-09 00:17:29 -0700546}
547
Glenn Kasten5e92a782012-01-30 07:40:52 -0800548status_t AudioEffect::getEffectDescriptor(const effect_uuid_t *uuid,
Ari Hausman-Cohen2046ec72018-04-24 14:00:55 -0700549 const effect_uuid_t *type,
550 uint32_t preferredTypeFlag,
551 effect_descriptor_t *descriptor)
Eric Laurent801a1182010-06-09 00:17:29 -0700552{
553 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
554 if (af == 0) return PERMISSION_DENIED;
Ari Hausman-Cohen2046ec72018-04-24 14:00:55 -0700555 return af->getEffectDescriptor(uuid, type, preferredTypeFlag, descriptor);
Eric Laurent801a1182010-06-09 00:17:29 -0700556}
557
Glenn Kastend848eb42016-03-08 13:42:11 -0800558status_t AudioEffect::queryDefaultPreProcessing(audio_session_t audioSession,
Eric Laurent57dae992011-07-24 13:36:09 -0700559 effect_descriptor_t *descriptors,
560 uint32_t *count)
561{
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800562 if (descriptors == nullptr || count == nullptr) {
563 return BAD_VALUE;
564 }
Eric Laurent57dae992011-07-24 13:36:09 -0700565 const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
566 if (aps == 0) return PERMISSION_DENIED;
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800567
568 int32_t audioSessionAidl = VALUE_OR_RETURN_STATUS(
569 legacy2aidl_audio_session_t_int32_t(audioSession));
Mikhail Naganov0078ee52021-09-30 23:06:20 +0000570 media::audio::common::Int countAidl;
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800571 countAidl.value = VALUE_OR_RETURN_STATUS(convertIntegral<int32_t>(*count));
572 std::vector<media::EffectDescriptor> retAidl;
573 RETURN_STATUS_IF_ERROR(statusTFromBinderStatus(
574 aps->queryDefaultPreProcessing(audioSessionAidl, &countAidl, &retAidl)));
575 *count = VALUE_OR_RETURN_STATUS(convertIntegral<uint32_t>(countAidl.value));
576 RETURN_STATUS_IF_ERROR(convertRange(retAidl.begin(), retAidl.end(), descriptors,
577 aidl2legacy_EffectDescriptor_effect_descriptor_t));
578 return OK;
Eric Laurent57dae992011-07-24 13:36:09 -0700579}
Ari Hausman-Cohen433722e2018-04-24 14:25:22 -0700580
581status_t AudioEffect::newEffectUniqueId(audio_unique_id_t* id)
582{
583 const sp<IAudioFlinger>& af = AudioSystem::get_audio_flinger();
584 if (af == 0) return PERMISSION_DENIED;
585 *id = af->newAudioUniqueId(AUDIO_UNIQUE_ID_USE_EFFECT);
586 return NO_ERROR;
587}
588
Ari Hausman-Cohen24628312018-08-13 15:01:09 -0700589status_t AudioEffect::addSourceDefaultEffect(const char *typeStr,
590 const String16& opPackageName,
591 const char *uuidStr,
592 int32_t priority,
593 audio_source_t source,
594 audio_unique_id_t *id)
595{
596 const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
597 if (aps == 0) return PERMISSION_DENIED;
598
599 if (typeStr == NULL && uuidStr == NULL) return BAD_VALUE;
600
601 // Convert type & uuid from string to effect_uuid_t.
602 effect_uuid_t type;
603 if (typeStr != NULL) {
604 status_t res = stringToGuid(typeStr, &type);
605 if (res != OK) return res;
606 } else {
607 type = *EFFECT_UUID_NULL;
608 }
609
610 effect_uuid_t uuid;
611 if (uuidStr != NULL) {
612 status_t res = stringToGuid(uuidStr, &uuid);
613 if (res != OK) return res;
614 } else {
615 uuid = *EFFECT_UUID_NULL;
616 }
617
Mikhail Naganovddceecc2021-09-03 13:58:56 -0700618 AudioUuid typeAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_uuid_t_AudioUuid(type));
619 AudioUuid uuidAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_uuid_t_AudioUuid(uuid));
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800620 std::string opPackageNameAidl = VALUE_OR_RETURN_STATUS(
621 legacy2aidl_String16_string(opPackageName));
Mikhail Naganovddceecc2021-09-03 13:58:56 -0700622 AudioSource sourceAidl = VALUE_OR_RETURN_STATUS(
623 legacy2aidl_audio_source_t_AudioSource(source));
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800624 int32_t retAidl;
625 RETURN_STATUS_IF_ERROR(statusTFromBinderStatus(
626 aps->addSourceDefaultEffect(typeAidl, opPackageNameAidl, uuidAidl, priority, sourceAidl,
627 &retAidl)));
628 *id = VALUE_OR_RETURN_STATUS(aidl2legacy_int32_t_audio_unique_id_t(retAidl));
629 return OK;
Ari Hausman-Cohen24628312018-08-13 15:01:09 -0700630}
631
Ari Hausman-Cohen433722e2018-04-24 14:25:22 -0700632status_t AudioEffect::addStreamDefaultEffect(const char *typeStr,
633 const String16& opPackageName,
634 const char *uuidStr,
635 int32_t priority,
636 audio_usage_t usage,
637 audio_unique_id_t *id)
638{
639 const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
640 if (aps == 0) return PERMISSION_DENIED;
641
642 if (typeStr == NULL && uuidStr == NULL) return BAD_VALUE;
643
644 // Convert type & uuid from string to effect_uuid_t.
645 effect_uuid_t type;
646 if (typeStr != NULL) {
647 status_t res = stringToGuid(typeStr, &type);
648 if (res != OK) return res;
649 } else {
650 type = *EFFECT_UUID_NULL;
651 }
652
653 effect_uuid_t uuid;
654 if (uuidStr != NULL) {
655 status_t res = stringToGuid(uuidStr, &uuid);
656 if (res != OK) return res;
657 } else {
658 uuid = *EFFECT_UUID_NULL;
659 }
660
Mikhail Naganovddceecc2021-09-03 13:58:56 -0700661 AudioUuid typeAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_uuid_t_AudioUuid(type));
662 AudioUuid uuidAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_uuid_t_AudioUuid(uuid));
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800663 std::string opPackageNameAidl = VALUE_OR_RETURN_STATUS(
664 legacy2aidl_String16_string(opPackageName));
Mikhail Naganovdbf03642021-08-25 18:15:32 -0700665 media::audio::common::AudioUsage usageAidl = VALUE_OR_RETURN_STATUS(
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800666 legacy2aidl_audio_usage_t_AudioUsage(usage));
667 int32_t retAidl;
668 RETURN_STATUS_IF_ERROR(statusTFromBinderStatus(
669 aps->addStreamDefaultEffect(typeAidl, opPackageNameAidl, uuidAidl, priority, usageAidl,
670 &retAidl)));
671 *id = VALUE_OR_RETURN_STATUS(aidl2legacy_int32_t_audio_unique_id_t(retAidl));
672 return OK;
Ari Hausman-Cohen433722e2018-04-24 14:25:22 -0700673}
674
Ari Hausman-Cohen24628312018-08-13 15:01:09 -0700675status_t AudioEffect::removeSourceDefaultEffect(audio_unique_id_t id)
676{
677 const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
678 if (aps == 0) return PERMISSION_DENIED;
679
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800680 int32_t idAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_unique_id_t_int32_t(id));
681 return statusTFromBinderStatus(aps->removeSourceDefaultEffect(idAidl));
Ari Hausman-Cohen24628312018-08-13 15:01:09 -0700682}
683
Ari Hausman-Cohen433722e2018-04-24 14:25:22 -0700684status_t AudioEffect::removeStreamDefaultEffect(audio_unique_id_t id)
685{
686 const sp<IAudioPolicyService>& aps = AudioSystem::get_audio_policy_service();
687 if (aps == 0) return PERMISSION_DENIED;
688
Ytai Ben-Tsvi0a4904a2021-01-06 12:57:05 -0800689 int32_t idAidl = VALUE_OR_RETURN_STATUS(legacy2aidl_audio_unique_id_t_int32_t(id));
690 return statusTFromBinderStatus(aps->removeStreamDefaultEffect(idAidl));
Ari Hausman-Cohen433722e2018-04-24 14:25:22 -0700691}
692
Eric Laurent801a1182010-06-09 00:17:29 -0700693// -------------------------------------------------------------------------
694
695status_t AudioEffect::stringToGuid(const char *str, effect_uuid_t *guid)
696{
697 if (str == NULL || guid == NULL) {
698 return BAD_VALUE;
699 }
700
701 int tmp[10];
702
703 if (sscanf(str, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
704 tmp, tmp+1, tmp+2, tmp+3, tmp+4, tmp+5, tmp+6, tmp+7, tmp+8, tmp+9) < 10) {
705 return BAD_VALUE;
706 }
707 guid->timeLow = (uint32_t)tmp[0];
708 guid->timeMid = (uint16_t)tmp[1];
709 guid->timeHiAndVersion = (uint16_t)tmp[2];
710 guid->clockSeq = (uint16_t)tmp[3];
711 guid->node[0] = (uint8_t)tmp[4];
712 guid->node[1] = (uint8_t)tmp[5];
713 guid->node[2] = (uint8_t)tmp[6];
714 guid->node[3] = (uint8_t)tmp[7];
715 guid->node[4] = (uint8_t)tmp[8];
716 guid->node[5] = (uint8_t)tmp[9];
717
718 return NO_ERROR;
719}
720
721status_t AudioEffect::guidToString(const effect_uuid_t *guid, char *str, size_t maxLen)
722{
723 if (guid == NULL || str == NULL) {
724 return BAD_VALUE;
725 }
726
727 snprintf(str, maxLen, "%08x-%04x-%04x-%04x-%02x%02x%02x%02x%02x%02x",
728 guid->timeLow,
729 guid->timeMid,
730 guid->timeHiAndVersion,
731 guid->clockSeq,
732 guid->node[0],
733 guid->node[1],
734 guid->node[2],
735 guid->node[3],
736 guid->node[4],
737 guid->node[5]);
738
739 return NO_ERROR;
740}
741
742
Glenn Kasten40bc9062015-03-20 09:09:33 -0700743} // namespace android