From df19a94770ed0a9709b1ed8db5706cd61bdb321d Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Fr=C3=A9d=C3=A9ric=20Danis?= Date: Fri, 19 Sep 2025 10:23:30 +0200 Subject: [PATCH] shared/hfp: Add Call answer support MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This also manage the +CIEV:,… event to create, remove or update calls. --- src/shared/hfp.c | 79 ++++++++++++++++++++++++++++++++++++++++++++++++ src/shared/hfp.h | 4 +++ 2 files changed, 83 insertions(+) diff --git a/src/shared/hfp.c b/src/shared/hfp.c index 29b467ae3..7e35f239a 100644 --- a/src/shared/hfp.c +++ b/src/shared/hfp.c @@ -1651,9 +1651,36 @@ static void ciev_service_cb(uint8_t val, void *user_data) hfp->callbacks_data); } +static bool update_call_to_active(struct hfp_hf *hfp) +{ + const struct queue_entry *entry; + struct hf_call *call; + + for (entry = queue_get_entries(hfp->calls); entry; + entry = entry->next) { + call = entry->data; + + if (call->status == CALL_STATUS_DIALING || + call->status == CALL_STATUS_ALERTING || + call->status == CALL_STATUS_INCOMING) { + call->status = CALL_STATUS_ACTIVE; + if (hfp->callbacks && + hfp->callbacks->call_status_updated) + hfp->callbacks->call_status_updated( + call->id, + call->status, + hfp->callbacks_data); + return true; + } + } + + return false; +} + static void ciev_call_cb(uint8_t val, void *user_data) { struct hfp_hf *hfp = user_data; + uint id; DBG(hfp, "%u", val); @@ -1662,6 +1689,32 @@ static void ciev_call_cb(uint8_t val, void *user_data) DBG(hfp, "hf: Incorrect call state: %u", val); return; } + + switch (val) { + case CIND_CALL_NONE: + /* Remove all calls */ + queue_remove_all(hfp->calls, NULL, hfp, remove_call_cb); + break; + case CIND_CALL_IN_PROGRESS: + { + /* Find incoming, dialing or alerting call to change + * it to active + */ + if (update_call_to_active(hfp)) + return; + + /* else create new already active call */ + id = next_call_index(hfp); + if (id == 0) { + DBG(hfp, "hf: No new call index available"); + return; + } + call_new(hfp, id, CALL_STATUS_ACTIVE, NULL); + } + break; + default: + DBG(hfp, "hf: Unsupported call state: %u", val); + } } static bool call_outgoing_match(const void *data, const void *match_data) @@ -2367,3 +2420,29 @@ const char *hfp_hf_call_get_number(struct hfp_hf *hfp, uint id) return call->line_id; } + +bool hfp_hf_call_answer(struct hfp_hf *hfp, uint id, + hfp_response_func_t resp_cb, + void *user_data) +{ + struct hf_call *call; + + DBG(hfp, ""); + + if (!hfp) + return false; + + call = queue_find(hfp->calls, call_id_match, UINT_TO_PTR(id)); + if (!call) { + DBG(hfp, "hf: no call with id: %u", id); + return false; + } + + if (call->status != CALL_STATUS_INCOMING) { + DBG(hfp, "hf: %d not in incoming call state: %u", + call->status); + return false; + } + + return hfp_hf_send_command(hfp, resp_cb, user_data, "ATA"); +} diff --git a/src/shared/hfp.h b/src/shared/hfp.h index fec63c150..c623e48e6 100644 --- a/src/shared/hfp.h +++ b/src/shared/hfp.h @@ -236,3 +236,7 @@ bool hfp_hf_session_register(struct hfp_hf *hfp, bool hfp_hf_session(struct hfp_hf *hfp); const char *hfp_hf_call_get_number(struct hfp_hf *hfp, uint id); + +bool hfp_hf_call_answer(struct hfp_hf *hfp, uint id, + hfp_response_func_t resp_cb, + void *user_data); -- 2.47.3