blob: 24f2eeb6edcd767239c0fb497eb6d59899cc00ef [file] [log] [blame]
Kevin Rocardd4de2882018-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 Rocard2390b5d2018-02-28 14:36:53 -080022#include <android/hardware/audio/4.0/IPrimaryDevice.h>
Kevin Rocardd4de2882018-02-28 14:33:38 -080023#include <cutils/native_handle.h>
24#include <hwbinder/IPCThreadState.h>
25#include <utils/Log.h>
26
27#include "DeviceHalHidl.h"
28#include "HidlUtils.h"
29#include "StreamHalHidl.h"
30
Kevin Rocard2390b5d2018-02-28 14:36:53 -080031using ::android::hardware::audio::common::V4_0::AudioConfig;
32using ::android::hardware::audio::common::V4_0::AudioDevice;
33using ::android::hardware::audio::common::V4_0::AudioInputFlag;
34using ::android::hardware::audio::common::V4_0::AudioOutputFlag;
35using ::android::hardware::audio::common::V4_0::AudioPatchHandle;
36using ::android::hardware::audio::common::V4_0::AudioPort;
37using ::android::hardware::audio::common::V4_0::AudioPortConfig;
38using ::android::hardware::audio::common::V4_0::AudioMode;
39using ::android::hardware::audio::common::V4_0::AudioSource;
40using ::android::hardware::audio::common::V4_0::HidlUtils;
41using ::android::hardware::audio::V4_0::DeviceAddress;
42using ::android::hardware::audio::V4_0::IPrimaryDevice;
43using ::android::hardware::audio::V4_0::ParameterValue;
44using ::android::hardware::audio::V4_0::Result;
Kevin Rocardd4de2882018-02-28 14:33:38 -080045using ::android::hardware::hidl_string;
46using ::android::hardware::hidl_vec;
47
48namespace android {
Kevin Rocard2390b5d2018-02-28 14:36:53 -080049namespace V4_0 {
Kevin Rocardd4de2882018-02-28 14:33:38 -080050
51namespace {
52
53status_t deviceAddressFromHal(
54 audio_devices_t device, const char* halAddress, DeviceAddress* address) {
55 address->device = AudioDevice(device);
56
57 if (address == nullptr || strnlen(halAddress, AUDIO_DEVICE_MAX_ADDRESS_LEN) == 0) {
58 return OK;
59 }
60 const bool isInput = (device & AUDIO_DEVICE_BIT_IN) != 0;
61 if (isInput) device &= ~AUDIO_DEVICE_BIT_IN;
62 if ((!isInput && (device & AUDIO_DEVICE_OUT_ALL_A2DP) != 0)
63 || (isInput && (device & AUDIO_DEVICE_IN_BLUETOOTH_A2DP) != 0)) {
64 int status = sscanf(halAddress,
65 "%hhX:%hhX:%hhX:%hhX:%hhX:%hhX",
66 &address->address.mac[0], &address->address.mac[1], &address->address.mac[2],
67 &address->address.mac[3], &address->address.mac[4], &address->address.mac[5]);
68 return status == 6 ? OK : BAD_VALUE;
69 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_IP) != 0)
70 || (isInput && (device & AUDIO_DEVICE_IN_IP) != 0)) {
71 int status = sscanf(halAddress,
72 "%hhu.%hhu.%hhu.%hhu",
73 &address->address.ipv4[0], &address->address.ipv4[1],
74 &address->address.ipv4[2], &address->address.ipv4[3]);
75 return status == 4 ? OK : BAD_VALUE;
76 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_ALL_USB)) != 0
77 || (isInput && (device & AUDIO_DEVICE_IN_ALL_USB)) != 0) {
78 int status = sscanf(halAddress,
79 "card=%d;device=%d",
80 &address->address.alsa.card, &address->address.alsa.device);
81 return status == 2 ? OK : BAD_VALUE;
82 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_BUS) != 0)
83 || (isInput && (device & AUDIO_DEVICE_IN_BUS) != 0)) {
84 if (halAddress != NULL) {
85 address->busAddress = halAddress;
86 return OK;
87 }
88 return BAD_VALUE;
89 } else if ((!isInput && (device & AUDIO_DEVICE_OUT_REMOTE_SUBMIX)) != 0
90 || (isInput && (device & AUDIO_DEVICE_IN_REMOTE_SUBMIX) != 0)) {
91 if (halAddress != NULL) {
92 address->rSubmixAddress = halAddress;
93 return OK;
94 }
95 return BAD_VALUE;
96 }
97 return OK;
98}
99
100} // namespace
101
102DeviceHalHidl::DeviceHalHidl(const sp<IDevice>& device)
103 : ConversionHelperHidl("Device"), mDevice(device),
104 mPrimaryDevice(IPrimaryDevice::castFrom(device)) {
105}
106
107DeviceHalHidl::~DeviceHalHidl() {
108 if (mDevice != 0) {
109 mDevice.clear();
110 hardware::IPCThreadState::self()->flushCommands();
111 }
112}
113
114status_t DeviceHalHidl::getSupportedDevices(uint32_t*) {
115 // Obsolete.
116 return INVALID_OPERATION;
117}
118
119status_t DeviceHalHidl::initCheck() {
120 if (mDevice == 0) return NO_INIT;
121 return processReturn("initCheck", mDevice->initCheck());
122}
123
124status_t DeviceHalHidl::setVoiceVolume(float volume) {
125 if (mDevice == 0) return NO_INIT;
126 if (mPrimaryDevice == 0) return INVALID_OPERATION;
127 return processReturn("setVoiceVolume", mPrimaryDevice->setVoiceVolume(volume));
128}
129
130status_t DeviceHalHidl::setMasterVolume(float volume) {
131 if (mDevice == 0) return NO_INIT;
132 if (mPrimaryDevice == 0) return INVALID_OPERATION;
133 return processReturn("setMasterVolume", mPrimaryDevice->setMasterVolume(volume));
134}
135
136status_t DeviceHalHidl::getMasterVolume(float *volume) {
137 if (mDevice == 0) return NO_INIT;
138 if (mPrimaryDevice == 0) return INVALID_OPERATION;
139 Result retval;
140 Return<void> ret = mPrimaryDevice->getMasterVolume(
141 [&](Result r, float v) {
142 retval = r;
143 if (retval == Result::OK) {
144 *volume = v;
145 }
146 });
147 return processReturn("getMasterVolume", ret, retval);
148}
149
150status_t DeviceHalHidl::setMode(audio_mode_t mode) {
151 if (mDevice == 0) return NO_INIT;
152 if (mPrimaryDevice == 0) return INVALID_OPERATION;
153 return processReturn("setMode", mPrimaryDevice->setMode(AudioMode(mode)));
154}
155
156status_t DeviceHalHidl::setMicMute(bool state) {
157 if (mDevice == 0) return NO_INIT;
158 return processReturn("setMicMute", mDevice->setMicMute(state));
159}
160
161status_t DeviceHalHidl::getMicMute(bool *state) {
162 if (mDevice == 0) return NO_INIT;
163 Result retval;
164 Return<void> ret = mDevice->getMicMute(
165 [&](Result r, bool mute) {
166 retval = r;
167 if (retval == Result::OK) {
168 *state = mute;
169 }
170 });
171 return processReturn("getMicMute", ret, retval);
172}
173
174status_t DeviceHalHidl::setMasterMute(bool state) {
175 if (mDevice == 0) return NO_INIT;
176 return processReturn("setMasterMute", mDevice->setMasterMute(state));
177}
178
179status_t DeviceHalHidl::getMasterMute(bool *state) {
180 if (mDevice == 0) return NO_INIT;
181 Result retval;
182 Return<void> ret = mDevice->getMasterMute(
183 [&](Result r, bool mute) {
184 retval = r;
185 if (retval == Result::OK) {
186 *state = mute;
187 }
188 });
189 return processReturn("getMasterMute", ret, retval);
190}
191
192status_t DeviceHalHidl::setParameters(const String8& kvPairs) {
193 if (mDevice == 0) return NO_INIT;
194 hidl_vec<ParameterValue> hidlParams;
195 status_t status = parametersFromHal(kvPairs, &hidlParams);
196 if (status != OK) return status;
197 return processReturn("setParameters", mDevice->setParameters(hidlParams));
198}
199
200status_t DeviceHalHidl::getParameters(const String8& keys, String8 *values) {
201 values->clear();
202 if (mDevice == 0) return NO_INIT;
203 hidl_vec<hidl_string> hidlKeys;
204 status_t status = keysFromHal(keys, &hidlKeys);
205 if (status != OK) return status;
206 Result retval;
207 Return<void> ret = mDevice->getParameters(
208 hidlKeys,
209 [&](Result r, const hidl_vec<ParameterValue>& parameters) {
210 retval = r;
211 if (retval == Result::OK) {
212 parametersToHal(parameters, values);
213 }
214 });
215 return processReturn("getParameters", ret, retval);
216}
217
218status_t DeviceHalHidl::getInputBufferSize(
219 const struct audio_config *config, size_t *size) {
220 if (mDevice == 0) return NO_INIT;
221 AudioConfig hidlConfig;
222 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
223 Result retval;
224 Return<void> ret = mDevice->getInputBufferSize(
225 hidlConfig,
226 [&](Result r, uint64_t bufferSize) {
227 retval = r;
228 if (retval == Result::OK) {
229 *size = static_cast<size_t>(bufferSize);
230 }
231 });
232 return processReturn("getInputBufferSize", ret, retval);
233}
234
235status_t DeviceHalHidl::openOutputStream(
236 audio_io_handle_t handle,
237 audio_devices_t devices,
238 audio_output_flags_t flags,
239 struct audio_config *config,
240 const char *address,
241 sp<StreamOutHalInterface> *outStream) {
242 if (mDevice == 0) return NO_INIT;
243 DeviceAddress hidlDevice;
244 status_t status = deviceAddressFromHal(devices, address, &hidlDevice);
245 if (status != OK) return status;
246 AudioConfig hidlConfig;
247 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
248 Result retval = Result::NOT_INITIALIZED;
249 Return<void> ret = mDevice->openOutputStream(
250 handle,
251 hidlDevice,
252 hidlConfig,
253 AudioOutputFlag(flags),
254 [&](Result r, const sp<IStreamOut>& result, const AudioConfig& suggestedConfig) {
255 retval = r;
256 if (retval == Result::OK) {
257 *outStream = new StreamOutHalHidl(result);
258 }
259 HidlUtils::audioConfigToHal(suggestedConfig, config);
260 });
261 return processReturn("openOutputStream", ret, retval);
262}
263
264status_t DeviceHalHidl::openInputStream(
265 audio_io_handle_t handle,
266 audio_devices_t devices,
267 struct audio_config *config,
268 audio_input_flags_t flags,
269 const char *address,
270 audio_source_t source,
271 sp<StreamInHalInterface> *inStream) {
272 if (mDevice == 0) return NO_INIT;
273 DeviceAddress hidlDevice;
274 status_t status = deviceAddressFromHal(devices, address, &hidlDevice);
275 if (status != OK) return status;
276 AudioConfig hidlConfig;
277 HidlUtils::audioConfigFromHal(*config, &hidlConfig);
278 Result retval = Result::NOT_INITIALIZED;
279 Return<void> ret = mDevice->openInputStream(
280 handle,
281 hidlDevice,
282 hidlConfig,
283 AudioInputFlag(flags),
284 AudioSource(source),
285 [&](Result r, const sp<IStreamIn>& result, const AudioConfig& suggestedConfig) {
286 retval = r;
287 if (retval == Result::OK) {
288 *inStream = new StreamInHalHidl(result);
289 }
290 HidlUtils::audioConfigToHal(suggestedConfig, config);
291 });
292 return processReturn("openInputStream", ret, retval);
293}
294
295status_t DeviceHalHidl::supportsAudioPatches(bool *supportsPatches) {
296 if (mDevice == 0) return NO_INIT;
297 return processReturn("supportsAudioPatches", mDevice->supportsAudioPatches(), supportsPatches);
298}
299
300status_t DeviceHalHidl::createAudioPatch(
301 unsigned int num_sources,
302 const struct audio_port_config *sources,
303 unsigned int num_sinks,
304 const struct audio_port_config *sinks,
305 audio_patch_handle_t *patch) {
306 if (mDevice == 0) return NO_INIT;
307 hidl_vec<AudioPortConfig> hidlSources, hidlSinks;
308 HidlUtils::audioPortConfigsFromHal(num_sources, sources, &hidlSources);
309 HidlUtils::audioPortConfigsFromHal(num_sinks, sinks, &hidlSinks);
310 Result retval;
311 Return<void> ret = mDevice->createAudioPatch(
312 hidlSources, hidlSinks,
313 [&](Result r, AudioPatchHandle hidlPatch) {
314 retval = r;
315 if (retval == Result::OK) {
316 *patch = static_cast<audio_patch_handle_t>(hidlPatch);
317 }
318 });
319 return processReturn("createAudioPatch", ret, retval);
320}
321
322status_t DeviceHalHidl::releaseAudioPatch(audio_patch_handle_t patch) {
323 if (mDevice == 0) return NO_INIT;
324 return processReturn("releaseAudioPatch", mDevice->releaseAudioPatch(patch));
325}
326
327status_t DeviceHalHidl::getAudioPort(struct audio_port *port) {
328 if (mDevice == 0) return NO_INIT;
329 AudioPort hidlPort;
330 HidlUtils::audioPortFromHal(*port, &hidlPort);
331 Result retval;
332 Return<void> ret = mDevice->getAudioPort(
333 hidlPort,
334 [&](Result r, const AudioPort& p) {
335 retval = r;
336 if (retval == Result::OK) {
337 HidlUtils::audioPortToHal(p, port);
338 }
339 });
340 return processReturn("getAudioPort", ret, retval);
341}
342
343status_t DeviceHalHidl::setAudioPortConfig(const struct audio_port_config *config) {
344 if (mDevice == 0) return NO_INIT;
345 AudioPortConfig hidlConfig;
346 HidlUtils::audioPortConfigFromHal(*config, &hidlConfig);
347 return processReturn("setAudioPortConfig", mDevice->setAudioPortConfig(hidlConfig));
348}
349
350status_t DeviceHalHidl::dump(int fd) {
351 if (mDevice == 0) return NO_INIT;
352 native_handle_t* hidlHandle = native_handle_create(1, 0);
353 hidlHandle->data[0] = fd;
354 Return<void> ret = mDevice->debugDump(hidlHandle);
355 native_handle_delete(hidlHandle);
356 return processReturn("dump", ret);
357}
358
Kevin Rocard2390b5d2018-02-28 14:36:53 -0800359} // namespace V4_0
Kevin Rocardd4de2882018-02-28 14:33:38 -0800360} // namespace android