blob: 90954b20ec4c13ba58fc8dc3aecd9c663b9045d1 [file] [log] [blame]
Mikhail Naganovf558e022016-11-14 17:45:17 -08001/*
2 * Copyright (C) 2016 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
17#define LOG_TAG "EffectsFactoryHalHidl"
18//#define LOG_NDEBUG 0
19
20#include <cutils/native_handle.h>
Mikhail Naganovf558e022016-11-14 17:45:17 -080021
Mikhail Naganov247b5f92021-01-15 19:16:12 +000022#include <UuidUtils.h>
23#include <util/EffectUtils.h>
24
Mikhail Naganov9f57e3c2016-12-05 12:54:36 -080025#include "ConversionHelperHidl.h"
Kevin Rocard7588ff42018-01-08 11:11:30 -080026#include "EffectBufferHalHidl.h"
Mikhail Naganovf558e022016-11-14 17:45:17 -080027#include "EffectHalHidl.h"
Kevin Rocard00538f12019-06-25 14:26:29 -070028#include "EffectsFactoryHalHidl.h"
Mikhail Naganovf558e022016-11-14 17:45:17 -080029
Mikhail Naganov76956002020-10-30 09:40:21 -070030using ::android::hardware::audio::common::CPP_VERSION::implementation::UuidUtils;
Mikhail Naganov247b5f92021-01-15 19:16:12 +000031using ::android::hardware::audio::effect::CPP_VERSION::implementation::EffectUtils;
Mikhail Naganovf558e022016-11-14 17:45:17 -080032using ::android::hardware::Return;
Mikhail Naganovf558e022016-11-14 17:45:17 -080033
34namespace android {
Mikhail Naganov595caa32018-12-13 11:08:28 -080035namespace effect {
Mikhail Naganovf558e022016-11-14 17:45:17 -080036
Mikhail Naganov9ccaa162018-12-12 10:27:29 -080037using namespace ::android::hardware::audio::common::CPP_VERSION;
Mikhail Naganov595caa32018-12-13 11:08:28 -080038using namespace ::android::hardware::audio::effect::CPP_VERSION;
Mikhail Naganov9ccaa162018-12-12 10:27:29 -080039
Kevin Rocarda8f13932019-06-25 16:08:29 -070040EffectsFactoryHalHidl::EffectsFactoryHalHidl(sp<IEffectsFactory> effectsFactory)
41 : ConversionHelperHidl("EffectsFactory") {
Eric Laurent9289bde2020-08-18 12:49:17 -070042 ALOG_ASSERT(effectsFactory != nullptr, "Provided IEffectsFactory service is NULL");
Kevin Rocarda8f13932019-06-25 16:08:29 -070043 mEffectsFactory = effectsFactory;
Mikhail Naganovf558e022016-11-14 17:45:17 -080044}
45
Mikhail Naganovf558e022016-11-14 17:45:17 -080046status_t EffectsFactoryHalHidl::queryAllDescriptors() {
47 if (mEffectsFactory == 0) return NO_INIT;
48 Result retval = Result::NOT_INITIALIZED;
49 Return<void> ret = mEffectsFactory->getAllDescriptors(
50 [&](Result r, const hidl_vec<EffectDescriptor>& result) {
51 retval = r;
52 if (retval == Result::OK) {
53 mLastDescriptors = result;
54 }
55 });
Steven Morelande83be8a2017-01-06 11:06:33 -080056 if (ret.isOk()) {
Mikhail Naganovf558e022016-11-14 17:45:17 -080057 return retval == Result::OK ? OK : NO_INIT;
58 }
59 mLastDescriptors.resize(0);
Steven Morelande83be8a2017-01-06 11:06:33 -080060 return processReturn(__FUNCTION__, ret);
Mikhail Naganovf558e022016-11-14 17:45:17 -080061}
62
63status_t EffectsFactoryHalHidl::queryNumberEffects(uint32_t *pNumEffects) {
64 status_t queryResult = queryAllDescriptors();
65 if (queryResult == OK) {
66 *pNumEffects = mLastDescriptors.size();
67 }
68 return queryResult;
69}
70
71status_t EffectsFactoryHalHidl::getDescriptor(
72 uint32_t index, effect_descriptor_t *pDescriptor) {
73 // TODO: We need somehow to track the changes on the server side
74 // or figure out how to convert everybody to query all the descriptors at once.
Eric Laurent0ac7f8c2021-07-13 13:37:11 +020075 if (pDescriptor == nullptr) {
76 return BAD_VALUE;
77 }
Mikhail Naganovf558e022016-11-14 17:45:17 -080078 if (mLastDescriptors.size() == 0) {
79 status_t queryResult = queryAllDescriptors();
80 if (queryResult != OK) return queryResult;
81 }
82 if (index >= mLastDescriptors.size()) return NAME_NOT_FOUND;
Mikhail Naganov247b5f92021-01-15 19:16:12 +000083 EffectUtils::effectDescriptorToHal(mLastDescriptors[index], pDescriptor);
Mikhail Naganovf558e022016-11-14 17:45:17 -080084 return OK;
85}
86
87status_t EffectsFactoryHalHidl::getDescriptor(
88 const effect_uuid_t *pEffectUuid, effect_descriptor_t *pDescriptor) {
Eric Laurent0ac7f8c2021-07-13 13:37:11 +020089 if (pDescriptor == nullptr || pEffectUuid == nullptr) {
90 return BAD_VALUE;
91 }
Mikhail Naganovf558e022016-11-14 17:45:17 -080092 if (mEffectsFactory == 0) return NO_INIT;
93 Uuid hidlUuid;
Mikhail Naganov76956002020-10-30 09:40:21 -070094 UuidUtils::uuidFromHal(*pEffectUuid, &hidlUuid);
Mikhail Naganovf558e022016-11-14 17:45:17 -080095 Result retval = Result::NOT_INITIALIZED;
96 Return<void> ret = mEffectsFactory->getDescriptor(hidlUuid,
97 [&](Result r, const EffectDescriptor& result) {
98 retval = r;
99 if (retval == Result::OK) {
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000100 EffectUtils::effectDescriptorToHal(result, pDescriptor);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800101 }
102 });
Steven Morelande83be8a2017-01-06 11:06:33 -0800103 if (ret.isOk()) {
Mikhail Naganovf558e022016-11-14 17:45:17 -0800104 if (retval == Result::OK) return OK;
105 else if (retval == Result::INVALID_ARGUMENTS) return NAME_NOT_FOUND;
106 else return NO_INIT;
107 }
Steven Morelande83be8a2017-01-06 11:06:33 -0800108 return processReturn(__FUNCTION__, ret);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800109}
110
Eric Laurent0ac7f8c2021-07-13 13:37:11 +0200111status_t EffectsFactoryHalHidl::getDescriptors(const effect_uuid_t *pEffectType,
112 std::vector<effect_descriptor_t> *descriptors) {
113 if (pEffectType == nullptr || descriptors == nullptr) {
114 return BAD_VALUE;
115 }
116
117 uint32_t numEffects = 0;
118 status_t status = queryNumberEffects(&numEffects);
119 if (status != NO_ERROR) {
120 ALOGW("%s error %d from FactoryHal queryNumberEffects", __func__, status);
121 return status;
122 }
123
124 for (uint32_t i = 0; i < numEffects; i++) {
125 effect_descriptor_t descriptor;
126 status = getDescriptor(i, &descriptor);
127 if (status != NO_ERROR) {
128 ALOGW("%s error %d from FactoryHal getDescriptor", __func__, status);
129 continue;
130 }
131 if (memcmp(&descriptor.type, pEffectType, sizeof(effect_uuid_t)) == 0) {
132 descriptors->push_back(descriptor);
133 }
134 }
135 return descriptors->empty() ? NAME_NOT_FOUND : NO_ERROR;
136}
137
Mikhail Naganovf558e022016-11-14 17:45:17 -0800138status_t EffectsFactoryHalHidl::createEffect(
139 const effect_uuid_t *pEffectUuid, int32_t sessionId, int32_t ioId,
Eric Laurentd7005292019-11-13 12:57:33 -0800140 int32_t deviceId __unused, sp<EffectHalInterface> *effect) {
Mikhail Naganovf558e022016-11-14 17:45:17 -0800141 if (mEffectsFactory == 0) return NO_INIT;
142 Uuid hidlUuid;
Mikhail Naganov76956002020-10-30 09:40:21 -0700143 UuidUtils::uuidFromHal(*pEffectUuid, &hidlUuid);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800144 Result retval = Result::NOT_INITIALIZED;
Eric Laurentd7005292019-11-13 12:57:33 -0800145 Return<void> ret;
146#if MAJOR_VERSION >= 6
147 ret = mEffectsFactory->createEffect(
148 hidlUuid, sessionId, ioId, deviceId,
149 [&](Result r, const sp<IEffect>& result, uint64_t effectId) {
150 retval = r;
151 if (retval == Result::OK) {
152 *effect = new EffectHalHidl(result, effectId);
153 }
154 });
155#else
156 if (sessionId == AUDIO_SESSION_DEVICE && ioId == AUDIO_IO_HANDLE_NONE) {
157 return INVALID_OPERATION;
158 }
159 ret = mEffectsFactory->createEffect(
Mikhail Naganovf558e022016-11-14 17:45:17 -0800160 hidlUuid, sessionId, ioId,
161 [&](Result r, const sp<IEffect>& result, uint64_t effectId) {
162 retval = r;
163 if (retval == Result::OK) {
164 *effect = new EffectHalHidl(result, effectId);
165 }
166 });
Eric Laurentd7005292019-11-13 12:57:33 -0800167#endif
Steven Morelande83be8a2017-01-06 11:06:33 -0800168 if (ret.isOk()) {
Mikhail Naganovf558e022016-11-14 17:45:17 -0800169 if (retval == Result::OK) return OK;
170 else if (retval == Result::INVALID_ARGUMENTS) return NAME_NOT_FOUND;
171 else return NO_INIT;
172 }
Steven Morelande83be8a2017-01-06 11:06:33 -0800173 return processReturn(__FUNCTION__, ret);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800174}
175
176status_t EffectsFactoryHalHidl::dumpEffects(int fd) {
177 if (mEffectsFactory == 0) return NO_INIT;
178 native_handle_t* hidlHandle = native_handle_create(1, 0);
179 hidlHandle->data[0] = fd;
Kevin Rocarddf9b4202018-05-10 19:56:08 -0700180 Return<void> ret = mEffectsFactory->debug(hidlHandle, {} /* options */);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800181 native_handle_delete(hidlHandle);
Andy Hunge72ff022021-08-16 10:16:15 -0700182
183 // TODO(b/111997867, b/177271958) Workaround - remove when fixed.
184 // A Binder transmitted fd may not close immediately due to a race condition b/111997867
185 // when the remote binder thread removes the last refcount to the fd blocks in the
186 // kernel for binder activity. We send a Binder ping() command to unblock the thread
187 // and complete the fd close / release.
188 //
189 // See DeviceHalHidl::dump(), EffectHalHidl::dump(), StreamHalHidl::dump(),
190 // EffectsFactoryHalHidl::dumpEffects().
191
192 (void)mEffectsFactory->ping(); // synchronous Binder call
193
Steven Morelande83be8a2017-01-06 11:06:33 -0800194 return processReturn(__FUNCTION__, ret);
Mikhail Naganovf558e022016-11-14 17:45:17 -0800195}
196
Kevin Rocard7588ff42018-01-08 11:11:30 -0800197status_t EffectsFactoryHalHidl::allocateBuffer(size_t size, sp<EffectBufferHalInterface>* buffer) {
198 return EffectBufferHalHidl::allocate(size, buffer);
199}
200
201status_t EffectsFactoryHalHidl::mirrorBuffer(void* external, size_t size,
202 sp<EffectBufferHalInterface>* buffer) {
203 return EffectBufferHalHidl::mirror(external, size, buffer);
204}
205
Mikhail Naganov595caa32018-12-13 11:08:28 -0800206} // namespace effect
Kevin Rocard00538f12019-06-25 14:26:29 -0700207
Mikhail Naganovd7b2ff02020-02-07 13:51:04 -0800208extern "C" __attribute__((visibility("default"))) void* createIEffectsFactory() {
Kevin Rocard00538f12019-06-25 14:26:29 -0700209 auto service = hardware::audio::effect::CPP_VERSION::IEffectsFactory::getService();
Mikhail Naganov6718c392022-01-27 22:17:21 +0000210 return service ? new effect::EffectsFactoryHalHidl(service) : nullptr;
Kevin Rocard00538f12019-06-25 14:26:29 -0700211}
212
Mikhail Naganovf558e022016-11-14 17:45:17 -0800213} // namespace android