blob: cd83171d3d2b8a815bf23bb72eec8cfcaa567c01 [file] [log] [blame]
Kevin Rocard4bcd67f2018-02-28 14:33:38 -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#include <stdio.h>
18
19#define LOG_TAG "DeviceHalHidl"
20//#define LOG_NDEBUG 0
21
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080022#include <cutils/native_handle.h>
Jiabin Huangebe64102021-09-07 20:01:07 +000023#include <cutils/properties.h>
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080024#include <hwbinder/IPCThreadState.h>
jiabindaf49952019-11-22 14:10:57 -080025#include <media/AudioContainers.h>
Andy Hung224f82f2022-03-22 00:00:49 -070026#include <mediautils/TimeCheck.h>
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080027#include <utils/Log.h>
28
Mikhail Naganov247b5f92021-01-15 19:16:12 +000029#include PATH(android/hardware/audio/FILE_VERSION/IPrimaryDevice.h)
30#include <HidlUtils.h>
Kevin Rocardb9cfbf12018-02-23 19:11:06 -080031#include <common/all-versions/VersionUtils.h>
Mikhail Naganov247b5f92021-01-15 19:16:12 +000032#include <util/CoreUtils.h>
Kevin Rocardb9cfbf12018-02-23 19:11:06 -080033
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080034#include "DeviceHalHidl.h"
Mikhail Naganov247b5f92021-01-15 19:16:12 +000035#include "ParameterUtils.h"
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080036#include "StreamHalHidl.h"
37
Mikhail Naganov6718c392022-01-27 22:17:21 +000038using ::android::hardware::audio::common::COMMON_TYPES_CPP_VERSION::implementation::HidlUtils;
Kevin Rocard7a9f05a2018-11-28 16:52:25 -080039using ::android::hardware::audio::common::utils::EnumBitfield;
Mikhail Naganov6718c392022-01-27 22:17:21 +000040using ::android::hardware::audio::CORE_TYPES_CPP_VERSION::implementation::CoreUtils;
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080041using ::android::hardware::hidl_string;
42using ::android::hardware::hidl_vec;
43
44namespace android {
45
Mikhail Naganov6718c392022-01-27 22:17:21 +000046using namespace ::android::hardware::audio::common::COMMON_TYPES_CPP_VERSION;
47using namespace ::android::hardware::audio::CORE_TYPES_CPP_VERSION;
Mikhail Naganov9ccaa162018-12-12 10:27:29 -080048
Mikhail Naganov6718c392022-01-27 22:17:21 +000049DeviceHalHidl::DeviceHalHidl(const sp<::android::hardware::audio::CPP_VERSION::IDevice>& device)
Mikhail Naganov288a3432022-03-25 00:29:56 +000050 : CoreConversionHelperHidl("DeviceHalHidl"), mDevice(device) {
Mikhail Naganov6718c392022-01-27 22:17:21 +000051}
Eric Laurentb82e6b72019-11-22 17:25:04 -080052
Mikhail Naganov6718c392022-01-27 22:17:21 +000053DeviceHalHidl::DeviceHalHidl(
54 const sp<::android::hardware::audio::CPP_VERSION::IPrimaryDevice>& device)
Mikhail Naganov288a3432022-03-25 00:29:56 +000055 : CoreConversionHelperHidl("DeviceHalHidl"),
Mikhail Naganov6718c392022-01-27 22:17:21 +000056#if MAJOR_VERSION <= 6 || (MAJOR_VERSION == 7 && MINOR_VERSION == 0)
57 mDevice(device),
58#endif
59 mPrimaryDevice(device) {
60#if MAJOR_VERSION == 7 && MINOR_VERSION == 1
61 auto getDeviceRet = mPrimaryDevice->getDevice();
62 if (getDeviceRet.isOk()) {
63 mDevice = getDeviceRet;
64 } else {
65 ALOGE("Call to IPrimaryDevice.getDevice has failed: %s",
66 getDeviceRet.description().c_str());
67 }
68#endif
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080069}
70
71DeviceHalHidl::~DeviceHalHidl() {
72 if (mDevice != 0) {
Mikhail Naganov3355e442019-11-20 14:20:01 -080073#if MAJOR_VERSION <= 5
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080074 mDevice.clear();
75 hardware::IPCThreadState::self()->flushCommands();
Mikhail Naganov3355e442019-11-20 14:20:01 -080076#elif MAJOR_VERSION >= 6
77 mDevice->close();
78#endif
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080079 }
80}
81
Mikhail Naganovf83b9742023-04-24 13:06:04 -070082status_t DeviceHalHidl::getAudioPorts(
83 std::vector<media::audio::common::AudioPort> *ports __unused) {
84 return INVALID_OPERATION;
85}
86
87status_t DeviceHalHidl::getAudioRoutes(std::vector<media::AudioRoute> *routes __unused) {
88 return INVALID_OPERATION;
89}
90
Mikhail Naganov1fba38c2023-05-03 17:45:36 -070091status_t DeviceHalHidl::getSupportedModes(
92 std::vector<media::audio::common::AudioMode> *modes __unused) {
93 return INVALID_OPERATION;
94}
95
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080096status_t DeviceHalHidl::getSupportedDevices(uint32_t*) {
97 // Obsolete.
98 return INVALID_OPERATION;
99}
100
101status_t DeviceHalHidl::initCheck() {
Andy Hung224f82f2022-03-22 00:00:49 -0700102 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800103 if (mDevice == 0) return NO_INIT;
104 return processReturn("initCheck", mDevice->initCheck());
105}
106
107status_t DeviceHalHidl::setVoiceVolume(float volume) {
Andy Hung224f82f2022-03-22 00:00:49 -0700108 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800109 if (mDevice == 0) return NO_INIT;
110 if (mPrimaryDevice == 0) return INVALID_OPERATION;
111 return processReturn("setVoiceVolume", mPrimaryDevice->setVoiceVolume(volume));
112}
113
114status_t DeviceHalHidl::setMasterVolume(float volume) {
Andy Hung224f82f2022-03-22 00:00:49 -0700115 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800116 if (mDevice == 0) return NO_INIT;
Mikhail Naganovae1f6622019-02-21 15:20:05 -0800117 return processReturn("setMasterVolume", mDevice->setMasterVolume(volume));
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800118}
119
120status_t DeviceHalHidl::getMasterVolume(float *volume) {
Andy Hung224f82f2022-03-22 00:00:49 -0700121 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800122 if (mDevice == 0) return NO_INIT;
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800123 Result retval;
Mikhail Naganovae1f6622019-02-21 15:20:05 -0800124 Return<void> ret = mDevice->getMasterVolume(
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800125 [&](Result r, float v) {
126 retval = r;
127 if (retval == Result::OK) {
128 *volume = v;
129 }
130 });
131 return processReturn("getMasterVolume", ret, retval);
132}
133
134status_t DeviceHalHidl::setMode(audio_mode_t mode) {
Andy Hung224f82f2022-03-22 00:00:49 -0700135 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800136 if (mDevice == 0) return NO_INIT;
137 if (mPrimaryDevice == 0) return INVALID_OPERATION;
138 return processReturn("setMode", mPrimaryDevice->setMode(AudioMode(mode)));
139}
140
141status_t DeviceHalHidl::setMicMute(bool state) {
Andy Hung224f82f2022-03-22 00:00:49 -0700142 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800143 if (mDevice == 0) return NO_INIT;
144 return processReturn("setMicMute", mDevice->setMicMute(state));
145}
146
147status_t DeviceHalHidl::getMicMute(bool *state) {
Andy Hung224f82f2022-03-22 00:00:49 -0700148 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800149 if (mDevice == 0) return NO_INIT;
150 Result retval;
151 Return<void> ret = mDevice->getMicMute(
152 [&](Result r, bool mute) {
153 retval = r;
154 if (retval == Result::OK) {
155 *state = mute;
156 }
157 });
158 return processReturn("getMicMute", ret, retval);
159}
160
161status_t DeviceHalHidl::setMasterMute(bool state) {
Andy Hung224f82f2022-03-22 00:00:49 -0700162 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800163 if (mDevice == 0) return NO_INIT;
164 return processReturn("setMasterMute", mDevice->setMasterMute(state));
165}
166
167status_t DeviceHalHidl::getMasterMute(bool *state) {
Andy Hung224f82f2022-03-22 00:00:49 -0700168 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800169 if (mDevice == 0) return NO_INIT;
170 Result retval;
171 Return<void> ret = mDevice->getMasterMute(
172 [&](Result r, bool mute) {
173 retval = r;
174 if (retval == Result::OK) {
175 *state = mute;
176 }
177 });
178 return processReturn("getMasterMute", ret, retval);
179}
180
181status_t DeviceHalHidl::setParameters(const String8& kvPairs) {
Andy Hung224f82f2022-03-22 00:00:49 -0700182 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800183 if (mDevice == 0) return NO_INIT;
184 hidl_vec<ParameterValue> hidlParams;
185 status_t status = parametersFromHal(kvPairs, &hidlParams);
186 if (status != OK) return status;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800187 // TODO: change the API so that context and kvPairs are separated
188 return processReturn("setParameters",
189 utils::setParameters(mDevice, {} /* context */, hidlParams));
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800190}
191
192status_t DeviceHalHidl::getParameters(const String8& keys, String8 *values) {
Andy Hung224f82f2022-03-22 00:00:49 -0700193 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800194 values->clear();
195 if (mDevice == 0) return NO_INIT;
196 hidl_vec<hidl_string> hidlKeys;
197 status_t status = keysFromHal(keys, &hidlKeys);
198 if (status != OK) return status;
199 Result retval;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800200 Return<void> ret = utils::getParameters(mDevice,
201 {} /* context */,
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800202 hidlKeys,
203 [&](Result r, const hidl_vec<ParameterValue>& parameters) {
204 retval = r;
205 if (retval == Result::OK) {
206 parametersToHal(parameters, values);
207 }
208 });
209 return processReturn("getParameters", ret, retval);
210}
211
212status_t DeviceHalHidl::getInputBufferSize(
213 const struct audio_config *config, size_t *size) {
Andy Hung224f82f2022-03-22 00:00:49 -0700214 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800215 if (mDevice == 0) return NO_INIT;
216 AudioConfig hidlConfig;
Mikhail Naganov05e03192020-12-14 23:19:54 +0000217 HidlUtils::audioConfigFromHal(*config, true /*isInput*/, &hidlConfig);
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800218 Result retval;
219 Return<void> ret = mDevice->getInputBufferSize(
220 hidlConfig,
221 [&](Result r, uint64_t bufferSize) {
222 retval = r;
223 if (retval == Result::OK) {
224 *size = static_cast<size_t>(bufferSize);
225 }
226 });
227 return processReturn("getInputBufferSize", ret, retval);
228}
229
230status_t DeviceHalHidl::openOutputStream(
231 audio_io_handle_t handle,
jiabin43810402019-10-24 14:58:31 -0700232 audio_devices_t deviceType,
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800233 audio_output_flags_t flags,
234 struct audio_config *config,
235 const char *address,
236 sp<StreamOutHalInterface> *outStream) {
Andy Hung224f82f2022-03-22 00:00:49 -0700237 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800238 if (mDevice == 0) return NO_INIT;
239 DeviceAddress hidlDevice;
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000240 if (status_t status = CoreUtils::deviceAddressFromHal(deviceType, address, &hidlDevice);
241 status != OK) {
242 return status;
243 }
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800244 AudioConfig hidlConfig;
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000245 if (status_t status = HidlUtils::audioConfigFromHal(*config, false /*isInput*/, &hidlConfig);
246 status != OK) {
247 return status;
248 }
Eric Laurente28c66d2022-01-21 13:40:41 +0100249
250#if !(MAJOR_VERSION == 7 && MINOR_VERSION == 1)
251 //TODO: b/193496180 use spatializer flag at audio HAL when available
252 if ((flags & AUDIO_OUTPUT_FLAG_SPATIALIZER) != 0) {
253 flags = (audio_output_flags_t)(flags & ~AUDIO_OUTPUT_FLAG_SPATIALIZER);
254 flags = (audio_output_flags_t)
255 (flags | AUDIO_OUTPUT_FLAG_FAST | AUDIO_OUTPUT_FLAG_DEEP_BUFFER);
256 }
257#endif
258
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000259 CoreUtils::AudioOutputFlags hidlFlags;
260 if (status_t status = CoreUtils::audioOutputFlagsFromHal(flags, &hidlFlags); status != OK) {
261 return status;
262 }
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800263 Result retval = Result::NOT_INITIALIZED;
Mikhail Naganov6718c392022-01-27 22:17:21 +0000264#if MAJOR_VERSION == 7 && MINOR_VERSION == 1
265 Return<void> ret = mDevice->openOutputStream_7_1(
266#else
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800267 Return<void> ret = mDevice->openOutputStream(
Mikhail Naganov6718c392022-01-27 22:17:21 +0000268#endif
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000269 handle, hidlDevice, hidlConfig, hidlFlags,
Kevin Rocard3d48dce2018-11-08 17:16:57 -0800270#if MAJOR_VERSION >= 4
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800271 {} /* metadata */,
Kevin Rocard070e7512018-05-22 09:29:13 -0700272#endif
Mikhail Naganov6718c392022-01-27 22:17:21 +0000273 [&](Result r, const sp<::android::hardware::audio::CPP_VERSION::IStreamOut>& result,
274 const AudioConfig& suggestedConfig) {
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800275 retval = r;
276 if (retval == Result::OK) {
277 *outStream = new StreamOutHalHidl(result);
278 }
279 HidlUtils::audioConfigToHal(suggestedConfig, config);
280 });
281 return processReturn("openOutputStream", ret, retval);
282}
283
284status_t DeviceHalHidl::openInputStream(
285 audio_io_handle_t handle,
286 audio_devices_t devices,
287 struct audio_config *config,
288 audio_input_flags_t flags,
289 const char *address,
290 audio_source_t source,
Mikhail Naganovb4e037e2019-01-14 15:56:33 -0800291 audio_devices_t outputDevice,
292 const char *outputDeviceAddress,
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800293 sp<StreamInHalInterface> *inStream) {
Andy Hung224f82f2022-03-22 00:00:49 -0700294 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800295 if (mDevice == 0) return NO_INIT;
296 DeviceAddress hidlDevice;
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000297 if (status_t status = CoreUtils::deviceAddressFromHal(devices, address, &hidlDevice);
298 status != OK) {
299 return status;
300 }
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800301 AudioConfig hidlConfig;
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000302 if (status_t status = HidlUtils::audioConfigFromHal(*config, true /*isInput*/, &hidlConfig);
303 status != OK) {
304 return status;
305 }
306 CoreUtils::AudioInputFlags hidlFlags;
Dean Wheatleyea186342021-03-12 16:49:47 +1100307#if MAJOR_VERSION <= 5
308 // Some flags were specific to framework and must not leak to the HAL.
309 flags = static_cast<audio_input_flags_t>(flags & ~AUDIO_INPUT_FLAG_DIRECT);
310#endif
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000311 if (status_t status = CoreUtils::audioInputFlagsFromHal(flags, &hidlFlags); status != OK) {
312 return status;
313 }
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800314 Result retval = Result::NOT_INITIALIZED;
Kevin Rocard070e7512018-05-22 09:29:13 -0700315#if MAJOR_VERSION == 2
Mikhail Naganovd9499eb2018-12-17 16:23:22 -0800316 auto sinkMetadata = AudioSource(source);
Kevin Rocard3d48dce2018-11-08 17:16:57 -0800317#elif MAJOR_VERSION >= 4
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800318 // TODO: correctly propagate the tracks sources and volume
319 // for now, only send the main source at 1dbfs
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000320 AudioSource hidlSource;
321 if (status_t status = HidlUtils::audioSourceFromHal(source, &hidlSource); status != OK) {
322 return status;
323 }
324 SinkMetadata sinkMetadata = {{{ .source = std::move(hidlSource), .gain = 1 }}};
Kevin Rocard070e7512018-05-22 09:29:13 -0700325#endif
Mikhail Naganovb4e037e2019-01-14 15:56:33 -0800326#if MAJOR_VERSION < 5
327 (void)outputDevice;
328 (void)outputDeviceAddress;
329#else
Mikhail Naganovc4258802021-02-08 17:14:17 -0800330#if MAJOR_VERSION >= 7
331 (void)HidlUtils::audioChannelMaskFromHal(
332 AUDIO_CHANNEL_NONE, true /*isInput*/, &sinkMetadata.tracks[0].channelMask);
333#endif
Mikhail Naganovb4e037e2019-01-14 15:56:33 -0800334 if (outputDevice != AUDIO_DEVICE_NONE) {
335 DeviceAddress hidlOutputDevice;
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000336 if (status_t status = CoreUtils::deviceAddressFromHal(
337 outputDevice, outputDeviceAddress, &hidlOutputDevice); status != OK) {
338 return status;
339 }
Mikhail Naganovb4e037e2019-01-14 15:56:33 -0800340 sinkMetadata.tracks[0].destination.device(std::move(hidlOutputDevice));
341 }
342#endif
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800343 Return<void> ret = mDevice->openInputStream(
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000344 handle, hidlDevice, hidlConfig, hidlFlags, sinkMetadata,
Mikhail Naganovaccbe8a2022-02-03 23:45:36 +0000345 [&](Result r,
346 const sp<::android::hardware::audio::CORE_TYPES_CPP_VERSION::IStreamIn>& result,
Mikhail Naganov6718c392022-01-27 22:17:21 +0000347 const AudioConfig& suggestedConfig) {
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800348 retval = r;
349 if (retval == Result::OK) {
350 *inStream = new StreamInHalHidl(result);
351 }
352 HidlUtils::audioConfigToHal(suggestedConfig, config);
353 });
354 return processReturn("openInputStream", ret, retval);
355}
356
357status_t DeviceHalHidl::supportsAudioPatches(bool *supportsPatches) {
Andy Hung224f82f2022-03-22 00:00:49 -0700358 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800359 if (mDevice == 0) return NO_INIT;
360 return processReturn("supportsAudioPatches", mDevice->supportsAudioPatches(), supportsPatches);
361}
362
363status_t DeviceHalHidl::createAudioPatch(
364 unsigned int num_sources,
365 const struct audio_port_config *sources,
366 unsigned int num_sinks,
367 const struct audio_port_config *sinks,
368 audio_patch_handle_t *patch) {
Andy Hung224f82f2022-03-22 00:00:49 -0700369 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800370 if (mDevice == 0) return NO_INIT;
Eric Laurent8711cfd2019-06-10 18:06:33 -0700371 if (patch == nullptr) return BAD_VALUE;
372
Mikhail Naganov73bdf572019-12-11 12:34:15 -0800373#if MAJOR_VERSION < 6
Eric Laurent8711cfd2019-06-10 18:06:33 -0700374 if (*patch != AUDIO_PATCH_HANDLE_NONE) {
375 status_t status = releaseAudioPatch(*patch);
376 ALOGW_IF(status != NO_ERROR, "%s error %d releasing patch handle %d",
377 __func__, status, *patch);
Mikhail Naganov73bdf572019-12-11 12:34:15 -0800378 *patch = AUDIO_PATCH_HANDLE_NONE;
Eric Laurent8711cfd2019-06-10 18:06:33 -0700379 }
Mikhail Naganov73bdf572019-12-11 12:34:15 -0800380#endif
Eric Laurent8711cfd2019-06-10 18:06:33 -0700381
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800382 hidl_vec<AudioPortConfig> hidlSources, hidlSinks;
383 HidlUtils::audioPortConfigsFromHal(num_sources, sources, &hidlSources);
384 HidlUtils::audioPortConfigsFromHal(num_sinks, sinks, &hidlSinks);
Mikhail Naganov73bdf572019-12-11 12:34:15 -0800385 Result retval = Result::OK;
386 Return<void> ret;
387 std::string methodName = "createAudioPatch";
388 if (*patch == AUDIO_PATCH_HANDLE_NONE) { // always true for MAJOR_VERSION < 6
389 ret = mDevice->createAudioPatch(
390 hidlSources, hidlSinks,
391 [&](Result r, AudioPatchHandle hidlPatch) {
392 retval = r;
393 if (retval == Result::OK) {
394 *patch = static_cast<audio_patch_handle_t>(hidlPatch);
395 }
396 });
397 } else {
398#if MAJOR_VERSION >= 6
399 ret = mDevice->updateAudioPatch(
400 *patch,
401 hidlSources, hidlSinks,
402 [&](Result r, AudioPatchHandle hidlPatch) {
403 retval = r;
404 if (retval == Result::OK) {
405 *patch = static_cast<audio_patch_handle_t>(hidlPatch);
406 }
407 });
408 methodName = "updateAudioPatch";
409#endif
410 }
411 return processReturn(methodName.c_str(), ret, retval);
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800412}
413
414status_t DeviceHalHidl::releaseAudioPatch(audio_patch_handle_t patch) {
Andy Hung224f82f2022-03-22 00:00:49 -0700415 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800416 if (mDevice == 0) return NO_INIT;
417 return processReturn("releaseAudioPatch", mDevice->releaseAudioPatch(patch));
418}
419
Mikhail Naganov720cc432021-03-24 20:42:49 -0700420template <typename HalPort>
421status_t DeviceHalHidl::getAudioPortImpl(HalPort *port) {
Mikhail Naganovf83b9742023-04-24 13:06:04 -0700422 using ::android::hardware::audio::common::COMMON_TYPES_CPP_VERSION::AudioPort;
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800423 if (mDevice == 0) return NO_INIT;
424 AudioPort hidlPort;
425 HidlUtils::audioPortFromHal(*port, &hidlPort);
426 Result retval;
427 Return<void> ret = mDevice->getAudioPort(
428 hidlPort,
429 [&](Result r, const AudioPort& p) {
430 retval = r;
431 if (retval == Result::OK) {
432 HidlUtils::audioPortToHal(p, port);
433 }
434 });
435 return processReturn("getAudioPort", ret, retval);
436}
437
Mikhail Naganov720cc432021-03-24 20:42:49 -0700438status_t DeviceHalHidl::getAudioPort(struct audio_port *port) {
Andy Hung224f82f2022-03-22 00:00:49 -0700439 TIME_CHECK();
Mikhail Naganov720cc432021-03-24 20:42:49 -0700440 return getAudioPortImpl(port);
441}
442
jiabinb4fed192020-09-22 14:45:40 -0700443status_t DeviceHalHidl::getAudioPort(struct audio_port_v7 *port) {
Andy Hung224f82f2022-03-22 00:00:49 -0700444 TIME_CHECK();
jiabinb4fed192020-09-22 14:45:40 -0700445#if MAJOR_VERSION >= 7
Mikhail Naganov720cc432021-03-24 20:42:49 -0700446 return getAudioPortImpl(port);
jiabinb4fed192020-09-22 14:45:40 -0700447#else
448 struct audio_port audioPort = {};
Mikhail Naganov720cc432021-03-24 20:42:49 -0700449 status_t result = NO_ERROR;
450 if (!audio_populate_audio_port(port, &audioPort)) {
451 ALOGE("Failed to populate legacy audio port from audio_port_v7");
452 result = BAD_VALUE;
453 }
454 status_t status = getAudioPort(&audioPort);
jiabinb4fed192020-09-22 14:45:40 -0700455 if (status == NO_ERROR) {
456 audio_populate_audio_port_v7(&audioPort, port);
Mikhail Naganov720cc432021-03-24 20:42:49 -0700457 } else {
458 result = status;
jiabinb4fed192020-09-22 14:45:40 -0700459 }
Mikhail Naganov720cc432021-03-24 20:42:49 -0700460 return result;
jiabinb4fed192020-09-22 14:45:40 -0700461#endif
jiabinb4fed192020-09-22 14:45:40 -0700462}
463
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800464status_t DeviceHalHidl::setAudioPortConfig(const struct audio_port_config *config) {
Mikhail Naganovf83b9742023-04-24 13:06:04 -0700465 using ::android::hardware::audio::common::COMMON_TYPES_CPP_VERSION::AudioPortConfig;
Andy Hung224f82f2022-03-22 00:00:49 -0700466 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800467 if (mDevice == 0) return NO_INIT;
468 AudioPortConfig hidlConfig;
469 HidlUtils::audioPortConfigFromHal(*config, &hidlConfig);
470 return processReturn("setAudioPortConfig", mDevice->setAudioPortConfig(hidlConfig));
471}
472
Kevin Rocard070e7512018-05-22 09:29:13 -0700473#if MAJOR_VERSION == 2
474status_t DeviceHalHidl::getMicrophones(
Shunkai Yao51202502022-12-12 06:11:46 +0000475 std::vector<audio_microphone_characteristic_t> *microphonesInfo __unused) {
Kevin Rocard070e7512018-05-22 09:29:13 -0700476 if (mDevice == 0) return NO_INIT;
477 return INVALID_OPERATION;
478}
Kevin Rocard3d48dce2018-11-08 17:16:57 -0800479#elif MAJOR_VERSION >= 4
Shunkai Yao51202502022-12-12 06:11:46 +0000480status_t DeviceHalHidl::getMicrophones(
481 std::vector<audio_microphone_characteristic_t> *microphonesInfo) {
Andy Hung224f82f2022-03-22 00:00:49 -0700482 TIME_CHECK();
jiabin9ff780e2018-03-19 18:19:52 -0700483 if (mDevice == 0) return NO_INIT;
484 Result retval;
485 Return<void> ret = mDevice->getMicrophones(
486 [&](Result r, hidl_vec<MicrophoneInfo> micArrayHal) {
487 retval = r;
488 for (size_t k = 0; k < micArrayHal.size(); k++) {
489 audio_microphone_characteristic_t dst;
490 //convert
Mikhail Naganov247b5f92021-01-15 19:16:12 +0000491 (void)CoreUtils::microphoneInfoToHal(micArrayHal[k], &dst);
Shunkai Yao51202502022-12-12 06:11:46 +0000492 microphonesInfo->push_back(dst);
jiabin9ff780e2018-03-19 18:19:52 -0700493 }
494 });
495 return processReturn("getMicrophones", ret, retval);
496}
Kevin Rocard070e7512018-05-22 09:29:13 -0700497#endif
jiabin9ff780e2018-03-19 18:19:52 -0700498
Eric Laurentb82e6b72019-11-22 17:25:04 -0800499#if MAJOR_VERSION >= 6
500status_t DeviceHalHidl::addDeviceEffect(
501 audio_port_handle_t device, sp<EffectHalInterface> effect) {
Andy Hung224f82f2022-03-22 00:00:49 -0700502 TIME_CHECK();
Eric Laurentb82e6b72019-11-22 17:25:04 -0800503 if (mDevice == 0) return NO_INIT;
504 return processReturn("addDeviceEffect", mDevice->addDeviceEffect(
Mikhail Naganov6718c392022-01-27 22:17:21 +0000505 static_cast<AudioPortHandle>(device), effect->effectId()));
Eric Laurentb82e6b72019-11-22 17:25:04 -0800506}
507#else
508status_t DeviceHalHidl::addDeviceEffect(
509 audio_port_handle_t device __unused, sp<EffectHalInterface> effect __unused) {
510 return INVALID_OPERATION;
511}
512#endif
513
514#if MAJOR_VERSION >= 6
515status_t DeviceHalHidl::removeDeviceEffect(
516 audio_port_handle_t device, sp<EffectHalInterface> effect) {
Andy Hung224f82f2022-03-22 00:00:49 -0700517 TIME_CHECK();
Eric Laurentb82e6b72019-11-22 17:25:04 -0800518 if (mDevice == 0) return NO_INIT;
519 return processReturn("removeDeviceEffect", mDevice->removeDeviceEffect(
Mikhail Naganov6718c392022-01-27 22:17:21 +0000520 static_cast<AudioPortHandle>(device), effect->effectId()));
Eric Laurentb82e6b72019-11-22 17:25:04 -0800521}
522#else
523status_t DeviceHalHidl::removeDeviceEffect(
524 audio_port_handle_t device __unused, sp<EffectHalInterface> effect __unused) {
525 return INVALID_OPERATION;
526}
527#endif
528
jiabin872de702023-04-27 22:04:31 +0000529status_t DeviceHalHidl::prepareToDisconnectExternalDevice(const struct audio_port_v7* port) {
530 // For HIDL HAL, there is not API to call notify the HAL to prepare for device connected
531 // state changed. Call `setConnectedState` directly.
532 const status_t status = setConnectedState(port, false /*connected*/);
533 if (status == NO_ERROR) {
534 // Cache the port id so that it won't disconnect twice.
535 mDeviceDisconnectionNotified.insert(port->id);
536 }
537 return status;
538}
539
Mikhail Naganov516d3982022-02-01 23:53:59 +0000540status_t DeviceHalHidl::setConnectedState(const struct audio_port_v7 *port, bool connected) {
Mikhail Naganovf83b9742023-04-24 13:06:04 -0700541 using ::android::hardware::audio::common::COMMON_TYPES_CPP_VERSION::AudioPort;
Andy Hung224f82f2022-03-22 00:00:49 -0700542 TIME_CHECK();
Mikhail Naganov516d3982022-02-01 23:53:59 +0000543 if (mDevice == 0) return NO_INIT;
jiabin872de702023-04-27 22:04:31 +0000544 if (!connected && mDeviceDisconnectionNotified.erase(port->id) > 0) {
545 // For device disconnection, APM will first call `prepareToDisconnectExternalDevice` and
546 // then call `setConnectedState`. However, in HIDL HAL, there is no API for
547 // `prepareToDisconnectExternalDevice`. In that case, HIDL HAL will call `setConnectedState`
548 // when calling `prepareToDisconnectExternalDevice`. Do not call to the HAL if previous
549 // call is successful. Also remove the cache here to avoid a large cache after a long run.
550 return NO_ERROR;
551 }
Mikhail Naganov516d3982022-02-01 23:53:59 +0000552#if MAJOR_VERSION == 7 && MINOR_VERSION == 1
553 if (supportsSetConnectedState7_1) {
554 AudioPort hidlPort;
555 if (status_t result = HidlUtils::audioPortFromHal(*port, &hidlPort); result != NO_ERROR) {
556 return result;
557 }
558 Return<Result> ret = mDevice->setConnectedState_7_1(hidlPort, connected);
559 if (!ret.isOk() || ret != Result::NOT_SUPPORTED) {
560 return processReturn("setConnectedState_7_1", ret);
561 } else if (ret == Result::OK) {
562 return NO_ERROR;
563 }
564 supportsSetConnectedState7_1 = false;
565 }
566#endif
567 DeviceAddress hidlAddress;
568 if (status_t result = CoreUtils::deviceAddressFromHal(
569 port->ext.device.type, port->ext.device.address, &hidlAddress);
570 result != NO_ERROR) {
571 return result;
572 }
573 return processReturn("setConnectedState", mDevice->setConnectedState(hidlAddress, connected));
574}
575
Ytai Ben-Tsvi48287b52021-12-01 15:07:11 -0800576error::Result<audio_hw_sync_t> DeviceHalHidl::getHwAvSync() {
Andy Hung224f82f2022-03-22 00:00:49 -0700577 TIME_CHECK();
Ytai Ben-Tsvi48287b52021-12-01 15:07:11 -0800578 if (mDevice == 0) return NO_INIT;
579 audio_hw_sync_t value;
580 Result result;
581 Return<void> ret = mDevice->getHwAvSync([&value, &result](Result r, audio_hw_sync_t v) {
582 value = v;
583 result = r;
584 });
585 RETURN_IF_ERROR(processReturn("getHwAvSync", ret, result));
586 return value;
587}
588
Andy Hung61589a42021-06-16 09:37:53 -0700589status_t DeviceHalHidl::dump(int fd, const Vector<String16>& args) {
Andy Hung224f82f2022-03-22 00:00:49 -0700590 TIME_CHECK();
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800591 if (mDevice == 0) return NO_INIT;
592 native_handle_t* hidlHandle = native_handle_create(1, 0);
593 hidlHandle->data[0] = fd;
Andy Hung61589a42021-06-16 09:37:53 -0700594 hidl_vec<hidl_string> hidlArgs;
595 argsFromHal(args, &hidlArgs);
596 Return<void> ret = mDevice->debug(hidlHandle, hidlArgs);
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800597 native_handle_delete(hidlHandle);
Andy Hunge72ff022021-08-16 10:16:15 -0700598
599 // TODO(b/111997867, b/177271958) Workaround - remove when fixed.
600 // A Binder transmitted fd may not close immediately due to a race condition b/111997867
601 // when the remote binder thread removes the last refcount to the fd blocks in the
602 // kernel for binder activity. We send a Binder ping() command to unblock the thread
603 // and complete the fd close / release.
604 //
605 // See DeviceHalHidl::dump(), EffectHalHidl::dump(), StreamHalHidl::dump(),
606 // EffectsFactoryHalHidl::dumpEffects().
607
608 (void)mDevice->ping(); // synchronous Binder call
609
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800610 return processReturn("dump", ret);
611}
612
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800613} // namespace android