Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 1 | /* |
| 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 | |
| 17 | #define LOG_TAG "DeviceHalAidl" |
Vlad Popa | 03bd5bc | 2023-01-17 16:16:51 +0100 | [diff] [blame^] | 18 | // #define LOG_NDEBUG 0 |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 19 | |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 20 | #include <aidl/android/hardware/audio/core/StreamDescriptor.h> |
| 21 | #include <error/expected_utils.h> |
| 22 | #include <media/AidlConversionCppNdk.h> |
| 23 | #include <media/AidlConversionUtil.h> |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 24 | #include <mediautils/TimeCheck.h> |
| 25 | #include <utils/Log.h> |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 26 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 27 | #include "DeviceHalAidl.h" |
| 28 | #include "StreamHalAidl.h" |
| 29 | |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 30 | using aidl::android::aidl_utils::statusTFromBinderStatus; |
| 31 | using aidl::android::media::audio::common::AudioMode; |
| 32 | using aidl::android::media::audio::common::Float; |
| 33 | using aidl::android::hardware::audio::core::IModule; |
| 34 | using aidl::android::hardware::audio::core::ITelephony; |
| 35 | using aidl::android::hardware::audio::core::StreamDescriptor; |
Vlad Popa | 03bd5bc | 2023-01-17 16:16:51 +0100 | [diff] [blame^] | 36 | using aidl::android::hardware::audio::core::sounddose::ISoundDose; |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 37 | |
| 38 | namespace android { |
| 39 | |
| 40 | status_t DeviceHalAidl::getSupportedDevices(uint32_t*) { |
| 41 | // Obsolete. |
| 42 | return INVALID_OPERATION; |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 43 | } |
| 44 | |
| 45 | status_t DeviceHalAidl::initCheck() { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 46 | if (mModule == nullptr) return NO_INIT; |
| 47 | // HAL modules are already initialized by the time they are published to the SM. |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 48 | return OK; |
| 49 | } |
| 50 | |
| 51 | status_t DeviceHalAidl::setVoiceVolume(float volume) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 52 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 53 | if (!mModule) return NO_INIT; |
| 54 | std::shared_ptr<ITelephony> telephony; |
| 55 | if (ndk::ScopedAStatus status = mModule->getTelephony(&telephony); |
| 56 | status.isOk() && telephony != nullptr) { |
| 57 | ITelephony::TelecomConfig inConfig{ .voiceVolume = Float{volume} }, outConfig; |
| 58 | RETURN_STATUS_IF_ERROR( |
| 59 | statusTFromBinderStatus(telephony->setTelecomConfig(inConfig, &outConfig))); |
| 60 | ALOGW_IF(outConfig.voiceVolume.has_value() && volume != outConfig.voiceVolume.value().value, |
| 61 | "%s: the resulting voice volume %f is not the same as requested %f", |
| 62 | __func__, outConfig.voiceVolume.value().value, volume); |
| 63 | } |
| 64 | return INVALID_OPERATION; |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 65 | } |
| 66 | |
| 67 | status_t DeviceHalAidl::setMasterVolume(float volume) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 68 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 69 | if (!mModule) return NO_INIT; |
| 70 | return statusTFromBinderStatus(mModule->setMasterVolume(volume)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 71 | } |
| 72 | |
| 73 | status_t DeviceHalAidl::getMasterVolume(float *volume) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 74 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 75 | if (!mModule) return NO_INIT; |
| 76 | return statusTFromBinderStatus(mModule->getMasterVolume(volume)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 77 | } |
| 78 | |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 79 | status_t DeviceHalAidl::setMode(audio_mode_t mode) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 80 | TIME_CHECK(); |
| 81 | if (!mModule) return NO_INIT; |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 82 | AudioMode audioMode = VALUE_OR_FATAL(::aidl::android::legacy2aidl_audio_mode_t_AudioMode(mode)); |
| 83 | std::shared_ptr<ITelephony> telephony; |
| 84 | if (ndk::ScopedAStatus status = mModule->getTelephony(&telephony); |
| 85 | status.isOk() && telephony != nullptr) { |
| 86 | RETURN_STATUS_IF_ERROR(statusTFromBinderStatus(telephony->switchAudioMode(audioMode))); |
| 87 | } |
| 88 | return statusTFromBinderStatus(mModule->updateAudioMode(audioMode)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 89 | } |
| 90 | |
| 91 | status_t DeviceHalAidl::setMicMute(bool state) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 92 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 93 | if (!mModule) return NO_INIT; |
| 94 | return statusTFromBinderStatus(mModule->setMicMute(state)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 95 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 96 | |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 97 | status_t DeviceHalAidl::getMicMute(bool *state) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 98 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 99 | if (!mModule) return NO_INIT; |
| 100 | return statusTFromBinderStatus(mModule->getMicMute(state)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 101 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 102 | |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 103 | status_t DeviceHalAidl::setMasterMute(bool state) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 104 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 105 | if (!mModule) return NO_INIT; |
| 106 | return statusTFromBinderStatus(mModule->setMasterMute(state)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 107 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 108 | |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 109 | status_t DeviceHalAidl::getMasterMute(bool *state) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 110 | TIME_CHECK(); |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 111 | if (!mModule) return NO_INIT; |
| 112 | return statusTFromBinderStatus(mModule->getMasterMute(state)); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 113 | } |
| 114 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 115 | status_t DeviceHalAidl::setParameters(const String8& kvPairs __unused) { |
| 116 | TIME_CHECK(); |
| 117 | if (!mModule) return NO_INIT; |
| 118 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 119 | return OK; |
| 120 | } |
| 121 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 122 | status_t DeviceHalAidl::getParameters(const String8& keys __unused, String8 *values) { |
| 123 | TIME_CHECK(); |
| 124 | values->clear(); |
| 125 | if (!mModule) return NO_INIT; |
| 126 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 127 | return OK; |
| 128 | } |
| 129 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 130 | status_t DeviceHalAidl::getInputBufferSize( |
| 131 | const struct audio_config* config __unused, size_t* size __unused) { |
| 132 | TIME_CHECK(); |
| 133 | if (!mModule) return NO_INIT; |
| 134 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 135 | return OK; |
| 136 | } |
| 137 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 138 | status_t DeviceHalAidl::openOutputStream( |
| 139 | audio_io_handle_t handle __unused, audio_devices_t devices __unused, |
| 140 | audio_output_flags_t flags __unused, struct audio_config* config, |
| 141 | const char* address __unused, |
| 142 | sp<StreamOutHalInterface>* outStream) { |
| 143 | if (!outStream || !config) { |
| 144 | return BAD_VALUE; |
| 145 | } |
| 146 | TIME_CHECK(); |
| 147 | if (!mModule) return NO_INIT; |
| 148 | config->sample_rate = 48000; |
| 149 | config->format = AUDIO_FORMAT_PCM_24_BIT_PACKED; |
| 150 | config->channel_mask = AUDIO_CHANNEL_OUT_STEREO; |
| 151 | StreamDescriptor descriptor; |
| 152 | descriptor.frameSizeBytes = audio_bytes_per_sample(config->format) * |
| 153 | audio_channel_count_from_out_mask(config->channel_mask); |
| 154 | descriptor.bufferSizeFrames = 600; |
| 155 | *outStream = sp<StreamOutHalAidl>::make(descriptor, nullptr); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 156 | return OK; |
| 157 | } |
| 158 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 159 | status_t DeviceHalAidl::openInputStream( |
| 160 | audio_io_handle_t handle __unused, audio_devices_t devices __unused, |
| 161 | struct audio_config* config, audio_input_flags_t flags __unused, |
| 162 | const char* address __unused, audio_source_t source __unused, |
| 163 | audio_devices_t outputDevice __unused, |
| 164 | const char* outputDeviceAddress __unused, |
| 165 | sp<StreamInHalInterface>* inStream) { |
| 166 | if (!inStream || !config) { |
| 167 | return BAD_VALUE; |
| 168 | } |
| 169 | TIME_CHECK(); |
| 170 | if (!mModule) return NO_INIT; |
| 171 | config->sample_rate = 48000; |
| 172 | config->format = AUDIO_FORMAT_PCM_24_BIT_PACKED; |
| 173 | config->channel_mask = AUDIO_CHANNEL_IN_STEREO; |
| 174 | StreamDescriptor descriptor; |
| 175 | descriptor.frameSizeBytes = audio_bytes_per_sample(config->format) * |
| 176 | audio_channel_count_from_out_mask(config->channel_mask); |
| 177 | descriptor.bufferSizeFrames = 600; |
| 178 | *inStream = sp<StreamInHalAidl>::make(descriptor, nullptr); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 179 | return OK; |
| 180 | } |
| 181 | |
| 182 | status_t DeviceHalAidl::supportsAudioPatches(bool* supportsPatches) { |
| 183 | *supportsPatches = true; |
| 184 | return OK; |
| 185 | } |
| 186 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 187 | status_t DeviceHalAidl::createAudioPatch(unsigned int num_sources __unused, |
| 188 | const struct audio_port_config* sources __unused, |
| 189 | unsigned int num_sinks __unused, |
| 190 | const struct audio_port_config* sinks __unused, |
| 191 | audio_patch_handle_t* patch __unused) { |
| 192 | TIME_CHECK(); |
| 193 | if (!mModule) return NO_INIT; |
| 194 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 195 | return OK; |
| 196 | } |
| 197 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 198 | status_t DeviceHalAidl::releaseAudioPatch(audio_patch_handle_t patch __unused) { |
| 199 | TIME_CHECK(); |
| 200 | if (!mModule) return NO_INIT; |
| 201 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 202 | return OK; |
| 203 | } |
| 204 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 205 | status_t DeviceHalAidl::getAudioPort(struct audio_port* port __unused) { |
| 206 | TIME_CHECK(); |
| 207 | ALOGE("%s not implemented yet", __func__); |
| 208 | return INVALID_OPERATION; |
| 209 | } |
| 210 | |
| 211 | status_t DeviceHalAidl::getAudioPort(struct audio_port_v7 *port __unused) { |
| 212 | TIME_CHECK(); |
| 213 | ALOGE("%s not implemented yet", __func__); |
| 214 | return INVALID_OPERATION; |
| 215 | } |
| 216 | |
| 217 | status_t DeviceHalAidl::setAudioPortConfig(const struct audio_port_config* config __unused) { |
| 218 | TIME_CHECK(); |
| 219 | if (!mModule) return NO_INIT; |
| 220 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 221 | return OK; |
| 222 | } |
| 223 | |
| 224 | status_t DeviceHalAidl::getMicrophones( |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 225 | std::vector<audio_microphone_characteristic_t>* microphones __unused) { |
| 226 | TIME_CHECK(); |
| 227 | if (!mModule) return NO_INIT; |
| 228 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 229 | return OK; |
| 230 | } |
| 231 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 232 | status_t DeviceHalAidl::addDeviceEffect(audio_port_handle_t device __unused, |
| 233 | sp<EffectHalInterface> effect) { |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 234 | if (!effect) { |
| 235 | return BAD_VALUE; |
| 236 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 237 | TIME_CHECK(); |
| 238 | if (!mModule) return NO_INIT; |
| 239 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 240 | return OK; |
| 241 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 242 | status_t DeviceHalAidl::removeDeviceEffect(audio_port_handle_t device __unused, |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 243 | sp<EffectHalInterface> effect) { |
| 244 | if (!effect) { |
| 245 | return BAD_VALUE; |
| 246 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 247 | TIME_CHECK(); |
| 248 | if (!mModule) return NO_INIT; |
| 249 | ALOGE("%s not implemented yet", __func__); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 250 | return OK; |
| 251 | } |
| 252 | |
| 253 | status_t DeviceHalAidl::getMmapPolicyInfos( |
| 254 | media::audio::common::AudioMMapPolicyType policyType __unused, |
| 255 | std::vector<media::audio::common::AudioMMapPolicyInfo>* policyInfos __unused) { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 256 | TIME_CHECK(); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 257 | ALOGE("%s not implemented yet", __func__); |
| 258 | return OK; |
| 259 | } |
| 260 | |
| 261 | int32_t DeviceHalAidl::getAAudioMixerBurstCount() { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 262 | TIME_CHECK(); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 263 | ALOGE("%s not implemented yet", __func__); |
| 264 | return OK; |
| 265 | } |
| 266 | |
| 267 | int32_t DeviceHalAidl::getAAudioHardwareBurstMinUsec() { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 268 | TIME_CHECK(); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 269 | ALOGE("%s not implemented yet", __func__); |
| 270 | return OK; |
| 271 | } |
| 272 | |
| 273 | error::Result<audio_hw_sync_t> DeviceHalAidl::getHwAvSync() { |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 274 | TIME_CHECK(); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 275 | ALOGE("%s not implemented yet", __func__); |
| 276 | return base::unexpected(INVALID_OPERATION); |
| 277 | } |
| 278 | |
Mikhail Naganov | fab697c | 2023-01-11 19:33:13 +0000 | [diff] [blame] | 279 | status_t DeviceHalAidl::dump(int fd, const Vector<String16>& args) { |
| 280 | TIME_CHECK(); |
| 281 | if (!mModule) return NO_INIT; |
| 282 | return mModule->dump(fd, Args(args).args(), args.size()); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 283 | }; |
| 284 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 285 | int32_t DeviceHalAidl::supportsBluetoothVariableLatency(bool* supports __unused) { |
| 286 | TIME_CHECK(); |
Shunkai Yao | 5120250 | 2022-12-12 06:11:46 +0000 | [diff] [blame] | 287 | ALOGE("%s not implemented yet", __func__); |
| 288 | return INVALID_OPERATION; |
| 289 | } |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 290 | |
Vlad Popa | 03bd5bc | 2023-01-17 16:16:51 +0100 | [diff] [blame^] | 291 | status_t DeviceHalAidl::getSoundDoseInterface(const std::string& module, |
| 292 | ::ndk::SpAIBinder* soundDoseBinder) { |
| 293 | TIME_CHECK(); |
| 294 | if (!mModule) return NO_INIT; |
| 295 | if (mSoundDose == nullptr) { |
| 296 | ndk::ScopedAStatus status = mModule->getSoundDose(&mSoundDose); |
| 297 | if (!status.isOk()) { |
| 298 | ALOGE("%s failed to return the sound dose interface for module %s: %s", |
| 299 | __func__, |
| 300 | module.c_str(), |
| 301 | status.getDescription().c_str()); |
| 302 | return BAD_VALUE; |
| 303 | } |
| 304 | } |
| 305 | *soundDoseBinder = mSoundDose->asBinder(); |
| 306 | ALOGI("%s using audio AIDL HAL sound dose interface", __func__); |
| 307 | |
| 308 | return OK; |
| 309 | } |
| 310 | |
Mikhail Naganov | 31d4665 | 2023-01-10 18:29:25 +0000 | [diff] [blame] | 311 | } // namespace android |