blob: 424fdb71ee87e1188714802e0af5d2d3395f323d [file] [log] [blame]
Shunkai Yaodca65ce2022-12-02 05:35:41 +00001/*
2 * Copyright (C) 2022 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Shunkai Yao51202502022-12-12 06:11:46 +000017#include <algorithm>
Shunkai Yao5c718342023-02-23 23:49:51 +000018#include <cstddef>
Shunkai Yao51202502022-12-12 06:11:46 +000019#include <cstdint>
Shunkai Yao5c718342023-02-23 23:49:51 +000020#include <iterator>
Shunkai Yao51202502022-12-12 06:11:46 +000021#include <memory>
Shunkai Yaodca65ce2022-12-02 05:35:41 +000022#define LOG_TAG "EffectsFactoryHalAidl"
23//#define LOG_NDEBUG 0
24
Shunkai Yao284bb0d2023-01-10 00:42:36 +000025#include <error/expected_utils.h>
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +000026#include <aidl/android/media/audio/common/AudioStreamType.h>
Shunkai Yaodca65ce2022-12-02 05:35:41 +000027#include <android/binder_manager.h>
Shunkai Yao51202502022-12-12 06:11:46 +000028#include <media/AidlConversionCppNdk.h>
Shunkai Yaoa03533e2023-01-25 06:38:10 +000029#include <media/AidlConversionEffect.h>
Shunkai Yao284bb0d2023-01-10 00:42:36 +000030#include <system/audio.h>
Shunkai Yao007caf02023-05-17 00:52:54 +000031#include <system/audio_aidl_utils.h>
Shunkai Yaodca65ce2022-12-02 05:35:41 +000032#include <utils/Log.h>
33
Shunkai Yao51202502022-12-12 06:11:46 +000034#include "EffectBufferHalAidl.h"
35#include "EffectHalAidl.h"
Shunkai Yao5c718342023-02-23 23:49:51 +000036#include "EffectProxy.h"
Shunkai Yaodca65ce2022-12-02 05:35:41 +000037#include "EffectsFactoryHalAidl.h"
38
Shunkai Yao284bb0d2023-01-10 00:42:36 +000039using ::aidl::android::legacy2aidl_audio_uuid_t_AudioUuid;
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +000040using ::aidl::android::aidl_utils::statusTFromBinderStatus;
41using ::aidl::android::hardware::audio::effect::Descriptor;
42using ::aidl::android::hardware::audio::effect::IFactory;
43using ::aidl::android::hardware::audio::effect::Processing;
Shunkai Yao0a51cf92023-05-10 22:42:56 +000044using ::aidl::android::media::audio::common::AudioSource;
45using ::aidl::android::media::audio::common::AudioStreamType;
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +000046using ::aidl::android::media::audio::common::AudioUuid;
Shunkai Yao007caf02023-05-17 00:52:54 +000047using ::android::audio::utils::toString;
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +000048using ::android::base::unexpected;
49using ::android::detail::AudioHalVersionInfo;
Shunkai Yaodca65ce2022-12-02 05:35:41 +000050
51namespace android {
52namespace effect {
53
Shunkai Yao51202502022-12-12 06:11:46 +000054EffectsFactoryHalAidl::EffectsFactoryHalAidl(std::shared_ptr<IFactory> effectsFactory)
55 : mFactory(effectsFactory),
Shunkai Yao5c718342023-02-23 23:49:51 +000056 mHalVersion(AudioHalVersionInfo(
57 AudioHalVersionInfo::Type::AIDL,
58 [this]() {
59 int32_t majorVersion = 0;
60 return (mFactory && mFactory->getInterfaceVersion(&majorVersion).isOk())
61 ? majorVersion
62 : 0;
63 }())),
64 mHalDescList([this]() {
65 std::vector<Descriptor> list;
66 if (mFactory) {
67 mFactory->queryEffects(std::nullopt, std::nullopt, std::nullopt, &list).isOk();
68 }
69 return list;
70 }()),
Shunkai Yao79f98742023-05-03 23:54:43 +000071 mProxyUuidDescriptorMap([this]() {
72 std::map<AudioUuid, std::vector<Descriptor>> proxyUuidMap;
Shunkai Yao5c718342023-02-23 23:49:51 +000073 for (const auto& desc : mHalDescList) {
Shunkai Yao5c718342023-02-23 23:49:51 +000074 if (desc.common.id.proxy.has_value()) {
75 const auto& uuid = desc.common.id.proxy.value();
Shunkai Yao79f98742023-05-03 23:54:43 +000076 if (proxyUuidMap.count(uuid) == 0) {
77 proxyUuidMap.insert({uuid, {desc}});
78 } else {
79 proxyUuidMap[uuid].emplace_back(desc);
Shunkai Yao5c718342023-02-23 23:49:51 +000080 }
Shunkai Yao5c718342023-02-23 23:49:51 +000081 }
82 }
Shunkai Yao79f98742023-05-03 23:54:43 +000083 return proxyUuidMap;
Shunkai Yao5c718342023-02-23 23:49:51 +000084 }()),
85 mProxyDescList([this]() {
86 std::vector<Descriptor> list;
Shunkai Yao79f98742023-05-03 23:54:43 +000087 for (const auto& proxy : mProxyUuidDescriptorMap) {
88 if (Descriptor desc;
89 EffectProxy::buildDescriptor(proxy.first /* uuid */,
90 proxy.second /* sub-effect descriptor list */,
91 &desc /* proxy descriptor */)
92 .isOk()) {
Shunkai Yao5c718342023-02-23 23:49:51 +000093 list.emplace_back(std::move(desc));
94 }
95 }
96 return list;
97 }()),
98 mNonProxyDescList([this]() {
99 std::vector<Descriptor> list;
100 std::copy_if(mHalDescList.begin(), mHalDescList.end(), std::back_inserter(list),
101 [](const Descriptor& desc) { return !desc.common.id.proxy.has_value(); });
102 return list;
103 }()),
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +0000104 mEffectCount(mNonProxyDescList.size() + mProxyDescList.size()),
Shunkai Yao0a51cf92023-05-10 22:42:56 +0000105 mAidlProcessings([this]() -> std::vector<Processing> {
106 std::vector<Processing> processings;
107 if (!mFactory || !mFactory->queryProcessing(std::nullopt, &processings).isOk()) {
108 ALOGE("%s queryProcessing failed", __func__);
109 }
110 return processings;
111 }()) {
Shunkai Yao5c718342023-02-23 23:49:51 +0000112 ALOG_ASSERT(mFactory != nullptr, "Provided IEffectsFactory service is NULL");
113 ALOGI("%s with %zu nonProxyEffects and %zu proxyEffects", __func__, mNonProxyDescList.size(),
114 mProxyDescList.size());
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000115}
116
117status_t EffectsFactoryHalAidl::queryNumberEffects(uint32_t *pNumEffects) {
118 if (pNumEffects == nullptr) {
119 return BAD_VALUE;
120 }
Shunkai Yao51202502022-12-12 06:11:46 +0000121
Shunkai Yao5c718342023-02-23 23:49:51 +0000122 *pNumEffects = mEffectCount;
Shunkai Yao51202502022-12-12 06:11:46 +0000123 return OK;
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000124}
125
126status_t EffectsFactoryHalAidl::getDescriptor(uint32_t index, effect_descriptor_t* pDescriptor) {
Shunkai Yao51202502022-12-12 06:11:46 +0000127 if (pDescriptor == nullptr) {
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000128 return BAD_VALUE;
129 }
Shunkai Yao51202502022-12-12 06:11:46 +0000130
Shunkai Yao5c718342023-02-23 23:49:51 +0000131 if (index >= mEffectCount) {
132 ALOGE("%s index %d exceed max number %zu", __func__, index, mEffectCount);
Shunkai Yao51202502022-12-12 06:11:46 +0000133 return INVALID_OPERATION;
134 }
135
Shunkai Yao5c718342023-02-23 23:49:51 +0000136 if (index >= mNonProxyDescList.size()) {
137 *pDescriptor =
138 VALUE_OR_RETURN_STATUS(::aidl::android::aidl2legacy_Descriptor_effect_descriptor(
139 mProxyDescList.at(index - mNonProxyDescList.size())));
140 } else {
141 *pDescriptor =
142 VALUE_OR_RETURN_STATUS(::aidl::android::aidl2legacy_Descriptor_effect_descriptor(
143 mNonProxyDescList.at(index)));
144 }
Shunkai Yao51202502022-12-12 06:11:46 +0000145 return OK;
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000146}
147
Shunkai Yao51202502022-12-12 06:11:46 +0000148status_t EffectsFactoryHalAidl::getDescriptor(const effect_uuid_t* halUuid,
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000149 effect_descriptor_t* pDescriptor) {
Shunkai Yao5c718342023-02-23 23:49:51 +0000150 if (halUuid == nullptr) {
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000151 return BAD_VALUE;
152 }
Shunkai Yao51202502022-12-12 06:11:46 +0000153
Shunkai Yao5c718342023-02-23 23:49:51 +0000154 AudioUuid uuid =
155 VALUE_OR_RETURN_STATUS(::aidl::android::legacy2aidl_audio_uuid_t_AudioUuid(*halUuid));
156 return getHalDescriptorWithImplUuid(uuid, pDescriptor);
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000157}
158
Shunkai Yao51202502022-12-12 06:11:46 +0000159status_t EffectsFactoryHalAidl::getDescriptors(const effect_uuid_t* halType,
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000160 std::vector<effect_descriptor_t>* descriptors) {
Shunkai Yao5c718342023-02-23 23:49:51 +0000161 if (halType == nullptr) {
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000162 return BAD_VALUE;
163 }
Shunkai Yao51202502022-12-12 06:11:46 +0000164
Shunkai Yao5c718342023-02-23 23:49:51 +0000165 AudioUuid type =
166 VALUE_OR_RETURN_STATUS(::aidl::android::legacy2aidl_audio_uuid_t_AudioUuid(*halType));
167 return getHalDescriptorWithTypeUuid(type, descriptors);
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000168}
169
Shunkai Yao51202502022-12-12 06:11:46 +0000170status_t EffectsFactoryHalAidl::createEffect(const effect_uuid_t* uuid, int32_t sessionId,
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000171 int32_t ioId, int32_t deviceId __unused,
172 sp<EffectHalInterface>* effect) {
Shunkai Yao51202502022-12-12 06:11:46 +0000173 if (uuid == nullptr || effect == nullptr) {
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000174 return BAD_VALUE;
175 }
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000176 if (sessionId == AUDIO_SESSION_DEVICE && ioId == AUDIO_IO_HANDLE_NONE) {
Shunkai Yao51202502022-12-12 06:11:46 +0000177 return INVALID_OPERATION;
178 }
Shunkai Yaoac61ee92024-03-14 21:57:46 +0000179 ALOGV("%s session %d ioId %d", __func__, sessionId, ioId);
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000180
Shunkai Yao5c718342023-02-23 23:49:51 +0000181 AudioUuid aidlUuid =
182 VALUE_OR_RETURN_STATUS(::aidl::android::legacy2aidl_audio_uuid_t_AudioUuid(*uuid));
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000183 std::shared_ptr<IEffect> aidlEffect;
Shunkai Yao5c718342023-02-23 23:49:51 +0000184 // Use EffectProxy interface instead of IFactory to create
185 const bool isProxy = isProxyEffect(aidlUuid);
186 if (isProxy) {
Shunkai Yao79f98742023-05-03 23:54:43 +0000187 aidlEffect = ndk::SharedRefBase::make<EffectProxy>(
188 aidlUuid, mProxyUuidDescriptorMap.at(aidlUuid) /* sub-effect descriptor list */,
189 mFactory);
190 mProxyList.emplace_back(std::static_pointer_cast<EffectProxy>(aidlEffect));
Shunkai Yao5c718342023-02-23 23:49:51 +0000191 } else {
192 RETURN_STATUS_IF_ERROR(
193 statusTFromBinderStatus(mFactory->createEffect(aidlUuid, &aidlEffect)));
194 }
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000195 if (aidlEffect == nullptr) {
Shunkai Yao007caf02023-05-17 00:52:54 +0000196 ALOGE("%s failed to create effect with UUID: %s", __func__, toString(aidlUuid).c_str());
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000197 return NAME_NOT_FOUND;
198 }
Shunkai Yao5c718342023-02-23 23:49:51 +0000199 Descriptor desc;
Shunkai Yao284bb0d2023-01-10 00:42:36 +0000200 RETURN_STATUS_IF_ERROR(statusTFromBinderStatus(aidlEffect->getDescriptor(&desc)));
201
Mikhail Naganov89c22e42023-06-14 15:49:59 -0700202 *effect = sp<EffectHalAidl>::make(mFactory, aidlEffect, sessionId, ioId, desc, isProxy);
Shunkai Yao51202502022-12-12 06:11:46 +0000203 return OK;
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000204}
205
206status_t EffectsFactoryHalAidl::dumpEffects(int fd) {
Shunkai Yao5c718342023-02-23 23:49:51 +0000207 status_t ret = OK;
208 // record the error ret and continue dump as many effects as possible
Shunkai Yao79f98742023-05-03 23:54:43 +0000209 for (const auto& proxy : mProxyList) {
210 if (status_t temp = BAD_VALUE; proxy && (temp = proxy->dump(fd, nullptr, 0)) != OK) {
211 ret = temp;
Shunkai Yao5c718342023-02-23 23:49:51 +0000212 }
213 }
214 RETURN_STATUS_IF_ERROR(mFactory->dump(fd, nullptr, 0));
215 return ret;
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000216}
217
218status_t EffectsFactoryHalAidl::allocateBuffer(size_t size, sp<EffectBufferHalInterface>* buffer) {
Shunkai Yao51202502022-12-12 06:11:46 +0000219 return EffectBufferHalAidl::allocate(size, buffer);
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000220}
221
222status_t EffectsFactoryHalAidl::mirrorBuffer(void* external, size_t size,
223 sp<EffectBufferHalInterface>* buffer) {
Shunkai Yao51202502022-12-12 06:11:46 +0000224 return EffectBufferHalAidl::mirror(external, size, buffer);
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000225}
226
227AudioHalVersionInfo EffectsFactoryHalAidl::getHalVersion() const {
Shunkai Yao51202502022-12-12 06:11:46 +0000228 return mHalVersion;
229}
230
Shunkai Yao5c718342023-02-23 23:49:51 +0000231status_t EffectsFactoryHalAidl::getHalDescriptorWithImplUuid(const AudioUuid& uuid,
232 effect_descriptor_t* pDescriptor) {
Shunkai Yao51202502022-12-12 06:11:46 +0000233 if (pDescriptor == nullptr) {
234 return BAD_VALUE;
235 }
Shunkai Yao51202502022-12-12 06:11:46 +0000236
Shunkai Yao5c718342023-02-23 23:49:51 +0000237 const auto& list = isProxyEffect(uuid) ? mProxyDescList : mNonProxyDescList;
238 auto matchIt = std::find_if(list.begin(), list.end(),
239 [&](const auto& desc) { return desc.common.id.uuid == uuid; });
240 if (matchIt == list.end()) {
Shunkai Yao007caf02023-05-17 00:52:54 +0000241 ALOGE("%s UUID not found in HAL and proxy list %s", __func__, toString(uuid).c_str());
Shunkai Yao849c56c2023-11-01 02:43:09 +0000242 return NAME_NOT_FOUND;
Shunkai Yao51202502022-12-12 06:11:46 +0000243 }
244
245 *pDescriptor = VALUE_OR_RETURN_STATUS(
246 ::aidl::android::aidl2legacy_Descriptor_effect_descriptor(*matchIt));
247 return OK;
248}
249
Shunkai Yao5c718342023-02-23 23:49:51 +0000250status_t EffectsFactoryHalAidl::getHalDescriptorWithTypeUuid(
Shunkai Yao51202502022-12-12 06:11:46 +0000251 const AudioUuid& type, std::vector<effect_descriptor_t>* descriptors) {
252 if (descriptors == nullptr) {
253 return BAD_VALUE;
254 }
Shunkai Yao5c718342023-02-23 23:49:51 +0000255
Shunkai Yao51202502022-12-12 06:11:46 +0000256 std::vector<Descriptor> result;
Shunkai Yao5c718342023-02-23 23:49:51 +0000257 std::copy_if(mNonProxyDescList.begin(), mNonProxyDescList.end(), std::back_inserter(result),
Shunkai Yao51202502022-12-12 06:11:46 +0000258 [&](auto& desc) { return desc.common.id.type == type; });
Shunkai Yao5c718342023-02-23 23:49:51 +0000259 std::copy_if(mProxyDescList.begin(), mProxyDescList.end(), std::back_inserter(result),
260 [&](auto& desc) { return desc.common.id.type == type; });
261 if (result.empty()) {
Shunkai Yao007caf02023-05-17 00:52:54 +0000262 ALOGW("%s UUID type not found in HAL and proxy list %s", __func__, toString(type).c_str());
Shunkai Yao51202502022-12-12 06:11:46 +0000263 return BAD_VALUE;
264 }
265
266 *descriptors = VALUE_OR_RETURN_STATUS(
267 aidl::android::convertContainer<std::vector<effect_descriptor_t>>(
268 result, ::aidl::android::aidl2legacy_Descriptor_effect_descriptor));
269 return OK;
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000270}
271
Shunkai Yao5c718342023-02-23 23:49:51 +0000272bool EffectsFactoryHalAidl::isProxyEffect(const AudioUuid& uuid) const {
Shunkai Yao79f98742023-05-03 23:54:43 +0000273 return 0 != mProxyUuidDescriptorMap.count(uuid);
Shunkai Yao5c718342023-02-23 23:49:51 +0000274}
275
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +0000276std::shared_ptr<const effectsConfig::Processings> EffectsFactoryHalAidl::getProcessings() const {
Shunkai Yao0a51cf92023-05-10 22:42:56 +0000277
278 auto getConfigEffectWithDescriptor =
279 [](const auto& desc) -> std::shared_ptr<const effectsConfig::Effect> {
280 effectsConfig::Effect effect = {.name = desc.common.name, .isProxy = false};
281 if (const auto uuid =
282 ::aidl::android::aidl2legacy_AudioUuid_audio_uuid_t(desc.common.id.uuid);
283 uuid.ok()) {
Shunkai Yaoe708d972023-06-29 01:39:54 +0000284 static_cast<effectsConfig::EffectImpl&>(effect).uuid = uuid.value();
Shunkai Yao0a51cf92023-05-10 22:42:56 +0000285 return std::make_shared<const effectsConfig::Effect>(effect);
286 } else {
287 return nullptr;
288 }
289 };
290
291 auto getConfigProcessingWithAidlProcessing =
292 [&](const auto& aidlProcess, std::vector<effectsConfig::InputStream>& preprocess,
293 std::vector<effectsConfig::OutputStream>& postprocess) {
294 if (aidlProcess.type.getTag() == Processing::Type::streamType) {
295 AudioStreamType aidlType =
296 aidlProcess.type.template get<Processing::Type::streamType>();
297 const auto type =
298 ::aidl::android::aidl2legacy_AudioStreamType_audio_stream_type_t(
299 aidlType);
300 if (!type.ok()) {
301 return;
302 }
303
304 std::vector<std::shared_ptr<const effectsConfig::Effect>> effects;
305 std::transform(aidlProcess.ids.begin(), aidlProcess.ids.end(),
306 std::back_inserter(effects), getConfigEffectWithDescriptor);
307 effectsConfig::OutputStream stream = {.type = type.value(),
308 .effects = std::move(effects)};
309 postprocess.emplace_back(stream);
310 } else if (aidlProcess.type.getTag() == Processing::Type::source) {
311 AudioSource aidlType =
312 aidlProcess.type.template get<Processing::Type::source>();
313 const auto type =
314 ::aidl::android::aidl2legacy_AudioSource_audio_source_t(aidlType);
315 if (!type.ok()) {
316 return;
317 }
318
319 std::vector<std::shared_ptr<const effectsConfig::Effect>> effects;
320 std::transform(aidlProcess.ids.begin(), aidlProcess.ids.end(),
321 std::back_inserter(effects), getConfigEffectWithDescriptor);
322 effectsConfig::InputStream stream = {.type = type.value(),
323 .effects = std::move(effects)};
324 preprocess.emplace_back(stream);
325 }
326 };
327
328 static std::shared_ptr<const effectsConfig::Processings> processings(
329 [&]() -> std::shared_ptr<const effectsConfig::Processings> {
330 std::vector<effectsConfig::InputStream> preprocess;
331 std::vector<effectsConfig::OutputStream> postprocess;
332 for (const auto& processing : mAidlProcessings) {
333 getConfigProcessingWithAidlProcessing(processing, preprocess, postprocess);
334 }
335
336 if (0 == preprocess.size() && 0 == postprocess.size()) {
337 return nullptr;
338 }
339
340 return std::make_shared<const effectsConfig::Processings>(
341 effectsConfig::Processings({.preprocess = std::move(preprocess),
342 .postprocess = std::move(postprocess)}));
343 }());
344
345 return processings;
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +0000346}
347
Shunkai Yao0a51cf92023-05-10 22:42:56 +0000348// Return 0 for AIDL, as the AIDL interface is not aware of the configuration file.
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +0000349::android::error::Result<size_t> EffectsFactoryHalAidl::getSkippedElements() const {
Shunkai Yao8f6ad0f2023-04-18 23:14:25 +0000350 return 0;
351}
352
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000353} // namespace effect
354
355// When a shared library is built from a static library, even explicit
356// exports from a static library are optimized out unless actually used by
357// the shared library. See EffectsFactoryHalEntry.cpp.
358extern "C" void* createIEffectsFactoryImpl() {
Shunkai Yao51202502022-12-12 06:11:46 +0000359 auto serviceName = std::string(IFactory::descriptor) + "/default";
360 auto service = IFactory::fromBinder(
361 ndk::SpAIBinder(AServiceManager_waitForService(serviceName.c_str())));
362 if (!service) {
363 ALOGE("%s binder service %s not exist", __func__, serviceName.c_str());
364 return nullptr;
365 }
366 return new effect::EffectsFactoryHalAidl(service);
Shunkai Yaodca65ce2022-12-02 05:35:41 +0000367}
368
369} // namespace android