/****************************************************************************** * * Copyright 2018 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * ******************************************************************************/ #include #include #include #include #include #include #include "audio_hal_interface/hearing_aid_software_encoding.h" #include "audio_hearing_aid_hw/include/audio_hearing_aid_hw.h" #include "bta/include/bta_hearing_aid_api.h" #include "common/repeating_timer.h" #include "common/time_util.h" #include "os/log.h" #include "osi/include/wakelock.h" #include "stack/include/main_thread.h" #include "udrv/include/uipc.h" using base::FilePath; using namespace bluetooth; namespace fmt { template <> struct formatter : enum_formatter {}; template <> struct formatter : enum_formatter {}; template <> struct formatter : enum_formatter {}; } // namespace fmt namespace { #define CASE_RETURN_STR(const) \ case const: \ return #const; const char* audio_ha_hw_dump_ctrl_event(tHEARING_AID_CTRL_CMD event) { switch (event) { CASE_RETURN_STR(HEARING_AID_CTRL_CMD_NONE) CASE_RETURN_STR(HEARING_AID_CTRL_CMD_CHECK_READY) CASE_RETURN_STR(HEARING_AID_CTRL_CMD_START) CASE_RETURN_STR(HEARING_AID_CTRL_CMD_STOP) CASE_RETURN_STR(HEARING_AID_CTRL_CMD_SUSPEND) CASE_RETURN_STR(HEARING_AID_CTRL_GET_INPUT_AUDIO_CONFIG) CASE_RETURN_STR(HEARING_AID_CTRL_GET_OUTPUT_AUDIO_CONFIG) CASE_RETURN_STR(HEARING_AID_CTRL_SET_OUTPUT_AUDIO_CONFIG) CASE_RETURN_STR(HEARING_AID_CTRL_CMD_OFFLOAD_START) default: break; } return "UNKNOWN HEARING_AID_CTRL_CMD"; } int bit_rate = -1; int sample_rate = -1; int data_interval_ms = -1; int num_channels = 2; bluetooth::common::RepeatingTimer audio_timer; HearingAidAudioReceiver* localAudioReceiver = nullptr; std::unique_ptr uipc_hearing_aid = nullptr; struct AudioHalStats { size_t media_read_total_underflow_bytes; size_t media_read_total_underflow_count; uint64_t media_read_last_underflow_us; AudioHalStats() { Reset(); } void Reset() { media_read_total_underflow_bytes = 0; media_read_total_underflow_count = 0; media_read_last_underflow_us = 0; } }; AudioHalStats stats; bool hearing_aid_on_resume_req(bool start_media_task); bool hearing_aid_on_suspend_req(); void send_audio_data() { uint32_t bytes_per_tick = (num_channels * sample_rate * data_interval_ms * (bit_rate / 8)) / 1000; uint8_t p_buf[bytes_per_tick]; uint32_t bytes_read; if (bluetooth::audio::hearing_aid::is_hal_enabled()) { bytes_read = bluetooth::audio::hearing_aid::read(p_buf, bytes_per_tick); } else { bytes_read = UIPC_Read(*uipc_hearing_aid, UIPC_CH_ID_AV_AUDIO, p_buf, bytes_per_tick); } log::debug("bytes_read: {}", bytes_read); if (bytes_read < bytes_per_tick) { stats.media_read_total_underflow_bytes += bytes_per_tick - bytes_read; stats.media_read_total_underflow_count++; stats.media_read_last_underflow_us = bluetooth::common::time_get_os_boottime_us(); } std::vector data(p_buf, p_buf + bytes_read); if (localAudioReceiver != nullptr) { localAudioReceiver->OnAudioDataReady(data); } } void hearing_aid_send_ack(tHEARING_AID_CTRL_ACK status) { uint8_t ack = status; log::debug("Hearing Aid audio ctrl ack: {}", status); UIPC_Send(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, &ack, sizeof(ack)); } void start_audio_ticks() { if (data_interval_ms != HA_INTERVAL_10_MS && data_interval_ms != HA_INTERVAL_20_MS) { log::fatal("Unsupported data interval: {}", data_interval_ms); } wakelock_acquire(); audio_timer.SchedulePeriodic(get_main_thread()->GetWeakPtr(), FROM_HERE, base::BindRepeating(&send_audio_data), std::chrono::milliseconds(data_interval_ms)); log::info("running with data interval: {}", data_interval_ms); } void stop_audio_ticks() { log::info("stopped"); audio_timer.CancelAndWait(); wakelock_release(); } void hearing_aid_data_cb(tUIPC_CH_ID, tUIPC_EVENT event) { log::debug("Hearing Aid audio data event: {}", event); switch (event) { case UIPC_OPEN_EVT: log::info("UIPC_OPEN_EVT"); /* * Read directly from media task from here on (keep callback for * connection events. */ UIPC_Ioctl(*uipc_hearing_aid, UIPC_CH_ID_AV_AUDIO, UIPC_REG_REMOVE_ACTIVE_READSET, NULL); UIPC_Ioctl(*uipc_hearing_aid, UIPC_CH_ID_AV_AUDIO, UIPC_SET_READ_POLL_TMO, reinterpret_cast(0)); do_in_main_thread(FROM_HERE, base::BindOnce(start_audio_ticks)); break; case UIPC_CLOSE_EVT: log::info("UIPC_CLOSE_EVT"); hearing_aid_send_ack(HEARING_AID_CTRL_ACK_SUCCESS); do_in_main_thread(FROM_HERE, base::BindOnce(stop_audio_ticks)); break; default: log::error("Hearing Aid audio data event not recognized: {}", event); } } void hearing_aid_recv_ctrl_data() { tHEARING_AID_CTRL_CMD cmd = HEARING_AID_CTRL_CMD_NONE; int n; uint8_t read_cmd = 0; /* The read command size is one octet */ n = UIPC_Read(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, &read_cmd, 1); cmd = static_cast(read_cmd); /* detach on ctrl channel means audioflinger process was terminated */ if (n == 0) { log::warn("CTRL CH DETACHED"); UIPC_Close(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL); return; } log::info("{}", audio_ha_hw_dump_ctrl_event(cmd)); // a2dp_cmd_pending = cmd; tHEARING_AID_CTRL_ACK ctrl_ack_status; switch (cmd) { case HEARING_AID_CTRL_CMD_CHECK_READY: hearing_aid_send_ack(HEARING_AID_CTRL_ACK_SUCCESS); break; case HEARING_AID_CTRL_CMD_START: ctrl_ack_status = HEARING_AID_CTRL_ACK_SUCCESS; // timer is restarted in UIPC_Open if (!hearing_aid_on_resume_req(false)) { ctrl_ack_status = HEARING_AID_CTRL_ACK_FAILURE; } else { UIPC_Open(*uipc_hearing_aid, UIPC_CH_ID_AV_AUDIO, hearing_aid_data_cb, HEARING_AID_DATA_PATH); } hearing_aid_send_ack(ctrl_ack_status); break; case HEARING_AID_CTRL_CMD_STOP: if (!hearing_aid_on_suspend_req()) { log::info( "HEARING_AID_CTRL_CMD_STOP: hearing_aid_on_suspend_req() errs, but " "ignored."); } hearing_aid_send_ack(HEARING_AID_CTRL_ACK_SUCCESS); break; case HEARING_AID_CTRL_CMD_SUSPEND: ctrl_ack_status = HEARING_AID_CTRL_ACK_SUCCESS; if (!hearing_aid_on_suspend_req()) { ctrl_ack_status = HEARING_AID_CTRL_ACK_FAILURE; } hearing_aid_send_ack(ctrl_ack_status); break; case HEARING_AID_CTRL_GET_OUTPUT_AUDIO_CONFIG: { btav_a2dp_codec_config_t codec_config; btav_a2dp_codec_config_t codec_capability; if (sample_rate == 16000) { codec_config.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_16000; codec_capability.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_16000; } else if (sample_rate == 24000) { codec_config.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_24000; codec_capability.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_24000; } else { log::fatal("unsupported sample rate: {}", sample_rate); } codec_config.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16; codec_capability.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_16; codec_config.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO; codec_capability.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_STEREO; hearing_aid_send_ack(HEARING_AID_CTRL_ACK_SUCCESS); // Send the current codec config UIPC_Send(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_config.sample_rate), sizeof(btav_a2dp_codec_sample_rate_t)); UIPC_Send(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_config.bits_per_sample), sizeof(btav_a2dp_codec_bits_per_sample_t)); UIPC_Send(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_config.channel_mode), sizeof(btav_a2dp_codec_channel_mode_t)); // Send the current codec capability UIPC_Send(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_capability.sample_rate), sizeof(btav_a2dp_codec_sample_rate_t)); UIPC_Send( *uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_capability.bits_per_sample), sizeof(btav_a2dp_codec_bits_per_sample_t)); UIPC_Send( *uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, 0, reinterpret_cast(&codec_capability.channel_mode), sizeof(btav_a2dp_codec_channel_mode_t)); break; } case HEARING_AID_CTRL_SET_OUTPUT_AUDIO_CONFIG: { // TODO: we only support one config for now! btav_a2dp_codec_config_t codec_config; codec_config.sample_rate = BTAV_A2DP_CODEC_SAMPLE_RATE_NONE; codec_config.bits_per_sample = BTAV_A2DP_CODEC_BITS_PER_SAMPLE_NONE; codec_config.channel_mode = BTAV_A2DP_CODEC_CHANNEL_MODE_NONE; hearing_aid_send_ack(HEARING_AID_CTRL_ACK_SUCCESS); // Send the current codec config if (UIPC_Read(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, reinterpret_cast(&codec_config.sample_rate), sizeof(btav_a2dp_codec_sample_rate_t)) != sizeof(btav_a2dp_codec_sample_rate_t)) { log::error("Error reading sample rate from audio HAL"); break; } if (UIPC_Read(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, reinterpret_cast(&codec_config.bits_per_sample), sizeof(btav_a2dp_codec_bits_per_sample_t)) != sizeof(btav_a2dp_codec_bits_per_sample_t)) { log::error("Error reading bits per sample from audio HAL"); break; } if (UIPC_Read(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, reinterpret_cast(&codec_config.channel_mode), sizeof(btav_a2dp_codec_channel_mode_t)) != sizeof(btav_a2dp_codec_channel_mode_t)) { log::error("Error reading channel mode from audio HAL"); break; } log::info( "HEARING_AID_CTRL_SET_OUTPUT_AUDIO_CONFIG: sample_rate={}, " "bits_per_sample={},channel_mode={}", codec_config.sample_rate, codec_config.bits_per_sample, codec_config.channel_mode); break; } default: log::error("UNSUPPORTED CMD: {}", cmd); hearing_aid_send_ack(HEARING_AID_CTRL_ACK_FAILURE); break; } log::info("a2dp-ctrl-cmd : {} DONE", audio_ha_hw_dump_ctrl_event(cmd)); } void hearing_aid_ctrl_cb(tUIPC_CH_ID, tUIPC_EVENT event) { log::debug("Hearing Aid audio ctrl event: {}", event); switch (event) { case UIPC_OPEN_EVT: break; case UIPC_CLOSE_EVT: /* restart ctrl server unless we are shutting down */ if (HearingAid::IsHearingAidRunning()) { UIPC_Open(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, hearing_aid_ctrl_cb, HEARING_AID_CTRL_PATH); } break; case UIPC_RX_DATA_READY_EVT: hearing_aid_recv_ctrl_data(); break; default: log::error("Hearing Aid audio ctrl unrecognized event: {}", event); } } bool hearing_aid_on_resume_req(bool start_media_task) { if (localAudioReceiver == nullptr) { log::error("HEARING_AID_CTRL_CMD_START: audio receiver not started"); return false; } bt_status_t status; if (start_media_task) { status = do_in_main_thread( FROM_HERE, base::BindOnce(&HearingAidAudioReceiver::OnAudioResume, base::Unretained(localAudioReceiver), start_audio_ticks)); } else { auto start_dummy_ticks = []() { log::info("start_audio_ticks: waiting for data path opened"); }; status = do_in_main_thread( FROM_HERE, base::BindOnce(&HearingAidAudioReceiver::OnAudioResume, base::Unretained(localAudioReceiver), start_dummy_ticks)); } if (status != BT_STATUS_SUCCESS) { log::error("HEARING_AID_CTRL_CMD_START: do_in_main_thread err={}", status); return false; } return true; } bool hearing_aid_on_suspend_req() { if (localAudioReceiver == nullptr) { log::error("HEARING_AID_CTRL_CMD_SUSPEND: audio receiver not started"); return false; } bt_status_t status = do_in_main_thread( FROM_HERE, base::BindOnce(&HearingAidAudioReceiver::OnAudioSuspend, base::Unretained(localAudioReceiver), stop_audio_ticks)); if (status != BT_STATUS_SUCCESS) { log::error("HEARING_AID_CTRL_CMD_SUSPEND: do_in_main_thread err={}", status); return false; } return true; } } // namespace void HearingAidAudioSource::Start(const CodecConfiguration& codecConfiguration, HearingAidAudioReceiver* audioReceiver, uint16_t remote_delay_ms) { log::info("Hearing Aid Source Open"); bit_rate = codecConfiguration.bit_rate; sample_rate = codecConfiguration.sample_rate; data_interval_ms = codecConfiguration.data_interval_ms; stats.Reset(); if (bluetooth::audio::hearing_aid::is_hal_enabled()) { bluetooth::audio::hearing_aid::start_session(); bluetooth::audio::hearing_aid::set_remote_delay(remote_delay_ms); } localAudioReceiver = audioReceiver; } void HearingAidAudioSource::Stop() { log::info("Hearing Aid Source Close"); localAudioReceiver = nullptr; if (bluetooth::audio::hearing_aid::is_hal_enabled()) { bluetooth::audio::hearing_aid::end_session(); } stop_audio_ticks(); } void HearingAidAudioSource::Initialize() { auto stream_cb = bluetooth::audio::hearing_aid::StreamCallbacks{ .on_resume_ = hearing_aid_on_resume_req, .on_suspend_ = hearing_aid_on_suspend_req, }; if (!bluetooth::audio::hearing_aid::init(stream_cb, get_main_thread())) { log::warn("Using legacy HAL"); uipc_hearing_aid = UIPC_Init(); UIPC_Open(*uipc_hearing_aid, UIPC_CH_ID_AV_CTRL, hearing_aid_ctrl_cb, HEARING_AID_CTRL_PATH); } } void HearingAidAudioSource::CleanUp() { if (bluetooth::audio::hearing_aid::is_hal_enabled()) { bluetooth::audio::hearing_aid::cleanup(); } else { UIPC_Close(*uipc_hearing_aid, UIPC_CH_ID_ALL); uipc_hearing_aid = nullptr; } } void HearingAidAudioSource::DebugDump(int fd) { uint64_t now_us = bluetooth::common::time_get_os_boottime_us(); std::stringstream stream; stream << " Hearing Aid Audio HAL:" << "\n Counts (underflow) : " << stats.media_read_total_underflow_count << "\n Bytes (underflow) : " << stats.media_read_total_underflow_bytes << "\n Last update time ago in ms (underflow) : " << (stats.media_read_last_underflow_us > 0 ? (unsigned long long)(now_us - stats.media_read_last_underflow_us) / 1000 : 0) << std::endl; dprintf(fd, "%s", stream.str().c_str()); }