blob: 723e2eb12739b6dd9328563766838c5f2779006f [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 Rocard070e7512018-05-22 09:29:13 -070022#include <android/hardware/audio/2.0/IPrimaryDevice.h>
Kevin Rocard51e076a2018-02-28 14:36:53 -080023#include <android/hardware/audio/4.0/IPrimaryDevice.h>
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080024#include <cutils/native_handle.h>
25#include <hwbinder/IPCThreadState.h>
26#include <utils/Log.h>
27
Kevin Rocardb9cfbf12018-02-23 19:11:06 -080028#include <common/all-versions/VersionUtils.h>
29
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080030#include "DeviceHalHidl.h"
31#include "HidlUtils.h"
32#include "StreamHalHidl.h"
Kevin Rocardb9cfbf12018-02-23 19:11:06 -080033#include "VersionUtils.h"
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080034
Kevin Rocard070e7512018-05-22 09:29:13 -070035using ::android::hardware::audio::common::CPP_VERSION::AudioConfig;
36using ::android::hardware::audio::common::CPP_VERSION::AudioDevice;
37using ::android::hardware::audio::common::CPP_VERSION::AudioInputFlag;
38using ::android::hardware::audio::common::CPP_VERSION::AudioOutputFlag;
39using ::android::hardware::audio::common::CPP_VERSION::AudioPatchHandle;
40using ::android::hardware::audio::common::CPP_VERSION::AudioPort;
41using ::android::hardware::audio::common::CPP_VERSION::AudioPortConfig;
42using ::android::hardware::audio::common::CPP_VERSION::AudioMode;
43using ::android::hardware::audio::common::CPP_VERSION::AudioSource;
44using ::android::hardware::audio::common::CPP_VERSION::HidlUtils;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -080045using ::android::hardware::audio::common::utils::mkEnumConverter;
Kevin Rocard070e7512018-05-22 09:29:13 -070046using ::android::hardware::audio::CPP_VERSION::DeviceAddress;
47using ::android::hardware::audio::CPP_VERSION::IPrimaryDevice;
48using ::android::hardware::audio::CPP_VERSION::ParameterValue;
49using ::android::hardware::audio::CPP_VERSION::Result;
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080050using ::android::hardware::hidl_string;
51using ::android::hardware::hidl_vec;
52
Kevin Rocard070e7512018-05-22 09:29:13 -070053#if MAJOR_VERSION == 4
54using ::android::hardware::audio::CPP_VERSION::SinkMetadata;
55#endif
56
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080057namespace android {
Kevin Rocard070e7512018-05-22 09:29:13 -070058namespace CPP_VERSION {
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080059
60namespace {
61
62status_t deviceAddressFromHal(
63 audio_devices_t device, const char* halAddress, DeviceAddress* address) {
64 address->device = AudioDevice(device);
65
Kevin Rocard5915fa32018-03-29 10:32:44 -070066 if (halAddress == nullptr || strnlen(halAddress, AUDIO_DEVICE_MAX_ADDRESS_LEN) == 0) {
Kevin Rocard4bcd67f2018-02-28 14:33:38 -080067 return OK;
68 }
69 const bool isInput = (device & AUDIO_DEVICE_BIT_IN) != 0;
70 if (isInput) device &= ~AUDIO_DEVICE_BIT_IN;
71 if ((!isInput && (device & AUDIO_DEVICE_OUT_ALL_A2DP) != 0)
72 || (isInput && (device & AUDIO_DEVICE_IN_BLUETOOTH_A2DP) != 0)) {
73 int status = sscanf(halAddress,
74 "%hhX:%hhX:%hhX:%hhX:%hhX:%hhX",
75 &address->address.mac[0], &address->address.mac[1], &address->address.mac[2],
76 &address->address.mac[3], &address->address.mac[4], &address->address.mac[5]);
77 return status == 6 ? OK : BAD_VALUE;
78 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_IP) != 0)
79 || (isInput && (device & AUDIO_DEVICE_IN_IP) != 0)) {
80 int status = sscanf(halAddress,
81 "%hhu.%hhu.%hhu.%hhu",
82 &address->address.ipv4[0], &address->address.ipv4[1],
83 &address->address.ipv4[2], &address->address.ipv4[3]);
84 return status == 4 ? OK : BAD_VALUE;
85 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_ALL_USB)) != 0
86 || (isInput && (device & AUDIO_DEVICE_IN_ALL_USB)) != 0) {
87 int status = sscanf(halAddress,
88 "card=%d;device=%d",
89 &address->address.alsa.card, &address->address.alsa.device);
90 return status == 2 ? OK : BAD_VALUE;
91 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_BUS) != 0)
92 || (isInput && (device & AUDIO_DEVICE_IN_BUS) != 0)) {
93 if (halAddress != NULL) {
94 address->busAddress = halAddress;
95 return OK;
96 }
97 return BAD_VALUE;
98 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_REMOTE_SUBMIX)) != 0
99 || (isInput && (device & AUDIO_DEVICE_IN_REMOTE_SUBMIX) != 0)) {
100 if (halAddress != NULL) {
101 address->rSubmixAddress = halAddress;
102 return OK;
103 }
104 return BAD_VALUE;
105 }
106 return OK;
107}
108
109} // namespace
110
111DeviceHalHidl::DeviceHalHidl(const sp<IDevice>& device)
112 : ConversionHelperHidl("Device"), mDevice(device),
113 mPrimaryDevice(IPrimaryDevice::castFrom(device)) {
114}
115
116DeviceHalHidl::~DeviceHalHidl() {
117 if (mDevice != 0) {
118 mDevice.clear();
119 hardware::IPCThreadState::self()->flushCommands();
120 }
121}
122
123status_t DeviceHalHidl::getSupportedDevices(uint32_t*) {
124 // Obsolete.
125 return INVALID_OPERATION;
126}
127
128status_t DeviceHalHidl::initCheck() {
129 if (mDevice == 0) return NO_INIT;
130 return processReturn("initCheck", mDevice->initCheck());
131}
132
133status_t DeviceHalHidl::setVoiceVolume(float volume) {
134 if (mDevice == 0) return NO_INIT;
135 if (mPrimaryDevice == 0) return INVALID_OPERATION;
136 return processReturn("setVoiceVolume", mPrimaryDevice->setVoiceVolume(volume));
137}
138
139status_t DeviceHalHidl::setMasterVolume(float volume) {
140 if (mDevice == 0) return NO_INIT;
141 if (mPrimaryDevice == 0) return INVALID_OPERATION;
142 return processReturn("setMasterVolume", mPrimaryDevice->setMasterVolume(volume));
143}
144
145status_t DeviceHalHidl::getMasterVolume(float *volume) {
146 if (mDevice == 0) return NO_INIT;
147 if (mPrimaryDevice == 0) return INVALID_OPERATION;
148 Result retval;
149 Return<void> ret = mPrimaryDevice->getMasterVolume(
150 [&](Result r, float v) {
151 retval = r;
152 if (retval == Result::OK) {
153 *volume = v;
154 }
155 });
156 return processReturn("getMasterVolume", ret, retval);
157}
158
159status_t DeviceHalHidl::setMode(audio_mode_t mode) {
160 if (mDevice == 0) return NO_INIT;
161 if (mPrimaryDevice == 0) return INVALID_OPERATION;
162 return processReturn("setMode", mPrimaryDevice->setMode(AudioMode(mode)));
163}
164
165status_t DeviceHalHidl::setMicMute(bool state) {
166 if (mDevice == 0) return NO_INIT;
167 return processReturn("setMicMute", mDevice->setMicMute(state));
168}
169
170status_t DeviceHalHidl::getMicMute(bool *state) {
171 if (mDevice == 0) return NO_INIT;
172 Result retval;
173 Return<void> ret = mDevice->getMicMute(
174 [&](Result r, bool mute) {
175 retval = r;
176 if (retval == Result::OK) {
177 *state = mute;
178 }
179 });
180 return processReturn("getMicMute", ret, retval);
181}
182
183status_t DeviceHalHidl::setMasterMute(bool state) {
184 if (mDevice == 0) return NO_INIT;
185 return processReturn("setMasterMute", mDevice->setMasterMute(state));
186}
187
188status_t DeviceHalHidl::getMasterMute(bool *state) {
189 if (mDevice == 0) return NO_INIT;
190 Result retval;
191 Return<void> ret = mDevice->getMasterMute(
192 [&](Result r, bool mute) {
193 retval = r;
194 if (retval == Result::OK) {
195 *state = mute;
196 }
197 });
198 return processReturn("getMasterMute", ret, retval);
199}
200
201status_t DeviceHalHidl::setParameters(const String8& kvPairs) {
202 if (mDevice == 0) return NO_INIT;
203 hidl_vec<ParameterValue> hidlParams;
204 status_t status = parametersFromHal(kvPairs, &hidlParams);
205 if (status != OK) return status;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800206 // TODO: change the API so that context and kvPairs are separated
207 return processReturn("setParameters",
208 utils::setParameters(mDevice, {} /* context */, hidlParams));
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800209}
210
211status_t DeviceHalHidl::getParameters(const String8& keys, String8 *values) {
212 values->clear();
213 if (mDevice == 0) return NO_INIT;
214 hidl_vec<hidl_string> hidlKeys;
215 status_t status = keysFromHal(keys, &hidlKeys);
216 if (status != OK) return status;
217 Result retval;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800218 Return<void> ret = utils::getParameters(mDevice,
219 {} /* context */,
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800220 hidlKeys,
221 [&](Result r, const hidl_vec<ParameterValue>& parameters) {
222 retval = r;
223 if (retval == Result::OK) {
224 parametersToHal(parameters, values);
225 }
226 });
227 return processReturn("getParameters", ret, retval);
228}
229
230status_t DeviceHalHidl::getInputBufferSize(
231 const struct audio_config *config, size_t *size) {
232 if (mDevice == 0) return NO_INIT;
233 AudioConfig hidlConfig;
234 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
235 Result retval;
236 Return<void> ret = mDevice->getInputBufferSize(
237 hidlConfig,
238 [&](Result r, uint64_t bufferSize) {
239 retval = r;
240 if (retval == Result::OK) {
241 *size = static_cast<size_t>(bufferSize);
242 }
243 });
244 return processReturn("getInputBufferSize", ret, retval);
245}
246
247status_t DeviceHalHidl::openOutputStream(
248 audio_io_handle_t handle,
249 audio_devices_t devices,
250 audio_output_flags_t flags,
251 struct audio_config *config,
252 const char *address,
253 sp<StreamOutHalInterface> *outStream) {
254 if (mDevice == 0) return NO_INIT;
255 DeviceAddress hidlDevice;
256 status_t status = deviceAddressFromHal(devices, address, &hidlDevice);
257 if (status != OK) return status;
258 AudioConfig hidlConfig;
259 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
260 Result retval = Result::NOT_INITIALIZED;
261 Return<void> ret = mDevice->openOutputStream(
262 handle,
263 hidlDevice,
264 hidlConfig,
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800265 mkEnumConverter<AudioOutputFlag>(flags),
Kevin Rocard070e7512018-05-22 09:29:13 -0700266#if MAJOR_VERSION == 4
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800267 {} /* metadata */,
Kevin Rocard070e7512018-05-22 09:29:13 -0700268#endif
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800269 [&](Result r, const sp<IStreamOut>& result, const AudioConfig& suggestedConfig) {
270 retval = r;
271 if (retval == Result::OK) {
272 *outStream = new StreamOutHalHidl(result);
273 }
274 HidlUtils::audioConfigToHal(suggestedConfig, config);
275 });
276 return processReturn("openOutputStream", ret, retval);
277}
278
279status_t DeviceHalHidl::openInputStream(
280 audio_io_handle_t handle,
281 audio_devices_t devices,
282 struct audio_config *config,
283 audio_input_flags_t flags,
284 const char *address,
285 audio_source_t source,
286 sp<StreamInHalInterface> *inStream) {
287 if (mDevice == 0) return NO_INIT;
288 DeviceAddress hidlDevice;
289 status_t status = deviceAddressFromHal(devices, address, &hidlDevice);
290 if (status != OK) return status;
291 AudioConfig hidlConfig;
292 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
293 Result retval = Result::NOT_INITIALIZED;
Kevin Rocard070e7512018-05-22 09:29:13 -0700294#if MAJOR_VERSION == 2
295 auto sourceMetadata = AudioSource(source);
296#elif MAJOR_VERSION == 4
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800297 // TODO: correctly propagate the tracks sources and volume
298 // for now, only send the main source at 1dbfs
Kevin Rocarddf9b4202018-05-10 19:56:08 -0700299 SinkMetadata sourceMetadata = {{{AudioSource(source), 1}}};
Kevin Rocard070e7512018-05-22 09:29:13 -0700300#endif
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800301 Return<void> ret = mDevice->openInputStream(
302 handle,
303 hidlDevice,
304 hidlConfig,
Kevin Rocarddf9b4202018-05-10 19:56:08 -0700305 mkEnumConverter<AudioInputFlag>(flags),
306 sourceMetadata,
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800307 [&](Result r, const sp<IStreamIn>& result, const AudioConfig& suggestedConfig) {
308 retval = r;
309 if (retval == Result::OK) {
310 *inStream = new StreamInHalHidl(result);
311 }
312 HidlUtils::audioConfigToHal(suggestedConfig, config);
313 });
314 return processReturn("openInputStream", ret, retval);
315}
316
317status_t DeviceHalHidl::supportsAudioPatches(bool *supportsPatches) {
318 if (mDevice == 0) return NO_INIT;
319 return processReturn("supportsAudioPatches", mDevice->supportsAudioPatches(), supportsPatches);
320}
321
322status_t DeviceHalHidl::createAudioPatch(
323 unsigned int num_sources,
324 const struct audio_port_config *sources,
325 unsigned int num_sinks,
326 const struct audio_port_config *sinks,
327 audio_patch_handle_t *patch) {
328 if (mDevice == 0) return NO_INIT;
329 hidl_vec<AudioPortConfig> hidlSources, hidlSinks;
330 HidlUtils::audioPortConfigsFromHal(num_sources, sources, &hidlSources);
331 HidlUtils::audioPortConfigsFromHal(num_sinks, sinks, &hidlSinks);
332 Result retval;
333 Return<void> ret = mDevice->createAudioPatch(
334 hidlSources, hidlSinks,
335 [&](Result r, AudioPatchHandle hidlPatch) {
336 retval = r;
337 if (retval == Result::OK) {
338 *patch = static_cast<audio_patch_handle_t>(hidlPatch);
339 }
340 });
341 return processReturn("createAudioPatch", ret, retval);
342}
343
344status_t DeviceHalHidl::releaseAudioPatch(audio_patch_handle_t patch) {
345 if (mDevice == 0) return NO_INIT;
346 return processReturn("releaseAudioPatch", mDevice->releaseAudioPatch(patch));
347}
348
349status_t DeviceHalHidl::getAudioPort(struct audio_port *port) {
350 if (mDevice == 0) return NO_INIT;
351 AudioPort hidlPort;
352 HidlUtils::audioPortFromHal(*port, &hidlPort);
353 Result retval;
354 Return<void> ret = mDevice->getAudioPort(
355 hidlPort,
356 [&](Result r, const AudioPort& p) {
357 retval = r;
358 if (retval == Result::OK) {
359 HidlUtils::audioPortToHal(p, port);
360 }
361 });
362 return processReturn("getAudioPort", ret, retval);
363}
364
365status_t DeviceHalHidl::setAudioPortConfig(const struct audio_port_config *config) {
366 if (mDevice == 0) return NO_INIT;
367 AudioPortConfig hidlConfig;
368 HidlUtils::audioPortConfigFromHal(*config, &hidlConfig);
369 return processReturn("setAudioPortConfig", mDevice->setAudioPortConfig(hidlConfig));
370}
371
Kevin Rocard070e7512018-05-22 09:29:13 -0700372#if MAJOR_VERSION == 2
373status_t DeviceHalHidl::getMicrophones(
374 std::vector<media::MicrophoneInfo> *microphonesInfo __unused) {
375 if (mDevice == 0) return NO_INIT;
376 return INVALID_OPERATION;
377}
378#elif MAJOR_VERSION == 4
jiabin9ff780e2018-03-19 18:19:52 -0700379status_t DeviceHalHidl::getMicrophones(std::vector<media::MicrophoneInfo> *microphonesInfo) {
380 if (mDevice == 0) return NO_INIT;
381 Result retval;
382 Return<void> ret = mDevice->getMicrophones(
383 [&](Result r, hidl_vec<MicrophoneInfo> micArrayHal) {
384 retval = r;
385 for (size_t k = 0; k < micArrayHal.size(); k++) {
386 audio_microphone_characteristic_t dst;
387 //convert
388 microphoneInfoToHal(micArrayHal[k], &dst);
389 media::MicrophoneInfo microphone = media::MicrophoneInfo(dst);
390 microphonesInfo->push_back(microphone);
391 }
392 });
393 return processReturn("getMicrophones", ret, retval);
394}
Kevin Rocard070e7512018-05-22 09:29:13 -0700395#endif
jiabin9ff780e2018-03-19 18:19:52 -0700396
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800397status_t DeviceHalHidl::dump(int fd) {
398 if (mDevice == 0) return NO_INIT;
399 native_handle_t* hidlHandle = native_handle_create(1, 0);
400 hidlHandle->data[0] = fd;
Kevin Rocardb9cfbf12018-02-23 19:11:06 -0800401 Return<void> ret = mDevice->debug(hidlHandle, {} /* options */);
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800402 native_handle_delete(hidlHandle);
403 return processReturn("dump", ret);
404}
405
Kevin Rocard070e7512018-05-22 09:29:13 -0700406} // namespace CPP_VERSION
Kevin Rocard4bcd67f2018-02-28 14:33:38 -0800407} // namespace android