diff --git a/android/handsfree.c b/android/handsfree.c
index 76c05f8..00da445 100644
--- a/android/handsfree.c
+++ b/android/handsfree.c
bool ccwa_enabled;
bool indicators_enabled;
struct indicator inds[IND_COUNT];
+ int num_active;
+ int num_held;
+ int setup_state;
uint8_t negotiated_codec;
uint8_t proposed_codec;
return TRUE;
}
-static void phone_state_active(int num_active, int num_held)
+static void phone_state_dialing(int num_active, int num_held)
{
+ update_indicator(IND_CALLSETUP, 2);
+ if (num_active == 0 && num_held > 0)
+ update_indicator(IND_CALLHELD, 2);
}
-static void phone_state_held(int num_active, int num_held)
+static void phone_state_alerting(int num_active, int num_held)
{
-
+ update_indicator(IND_CALLSETUP, 3);
}
-static void phone_state_dialing(int num_active, int num_held)
+static void phone_state_waiting(int num_active, int num_held, uint8_t type,
+ const uint8_t *number, int number_len)
{
- update_indicator(IND_CALLSETUP, 2);
-}
+ char *num;
-static void phone_state_alerting(int num_active, int num_held)
-{
- update_indicator(IND_CALLSETUP, 3);
+ if (!device.ccwa_enabled)
+ return;
+
+ num = number_len ? (char *) number : "";
+
+ if (type == HAL_HANDSFREE_CALL_ADDRTYPE_INTERNATIONAL && num[0] != '+')
+ hfp_gw_send_info(device.gw, "+CCWA: \"+%s\",%u", num, type );
+ else
+ hfp_gw_send_info(device.gw, "+CCWA: \"%s\",%u", num, type );
+
+ update_indicator(IND_CALLSETUP, 1);
}
static void phone_state_incoming(int num_active, int num_held, uint8_t type,
- const char *number)
+ const uint8_t *number, int number_len)
{
- char *clip = NULL;
+ char *clip, *num;
+
+ if (device.setup_state == HAL_HANDSFREE_CALL_STATE_INCOMING)
+ return;
+
+ if (num_active > 0 || num_held > 0) {
+ phone_state_waiting(num_active, num_held, type, number,
+ number_len);
+ return;
+ }
update_indicator(IND_CALLSETUP, 1);
- if (type == HAL_HANDSFREE_CALL_ADDRTYPE_INTERNATIONAL &&
- number[0] != '+')
- clip = g_strdup_printf("+CLIP: \"+%s\",%u", number, type );
+ num = number_len ? (char *) number : "";
+
+ if (type == HAL_HANDSFREE_CALL_ADDRTYPE_INTERNATIONAL && num[0] != '+')
+ clip = g_strdup_printf("+CLIP: \"+%s\",%u", num, type );
else
- clip = g_strdup_printf("+CLIP: \"%s\",%u", number, type );
+ clip = g_strdup_printf("+CLIP: \"%s\",%u", num, type );
/* send first RING */
ring_cb(clip);
g_free(clip);
}
-static void phone_state_waiting(int num_active, int num_held, uint8_t type,
- const char *number)
-{
-
-}
-
static void phone_state_idle(int num_active, int num_held)
{
-
- update_indicator(IND_CALL, !!num_active);
-
if (device.ring) {
g_source_remove(device.ring);
device.ring = 0;
}
- update_indicator(IND_CALLSETUP, 0);
+ switch (device.setup_state) {
+ case HAL_HANDSFREE_CALL_STATE_INCOMING:
+ if (num_active > device.num_active)
+ update_indicator(IND_CALL, 1);
+
+ if (num_held > device.num_held)
+ update_indicator(IND_CALLHELD, 1);
+
+ update_indicator(IND_CALLSETUP, 0);
+ break;
+ case HAL_HANDSFREE_CALL_STATE_DIALING:
+ case HAL_HANDSFREE_CALL_STATE_ALERTING:
+ if (num_active > device.num_active)
+ update_indicator(IND_CALL, 1);
+
+ update_indicator(IND_CALLSETUP, 0);
+ break;
+ case HAL_HANDSFREE_CALL_STATE_IDLE:
+ /* check if calls swapped */
+ if (num_held != 0 && num_active != 0 &&
+ device.num_active == num_held &&
+ device.num_held == num_active) {
+ /* TODO better way for forcing indicator */
+ device.inds[IND_CALLHELD].val = 0;
+ update_indicator(IND_CALLHELD, 1);
+ } else {
+ update_indicator(IND_CALLHELD,
+ num_held ? (num_active ? 1 : 2) : 0);
+ }
+
+ update_indicator(IND_CALL, !!(num_active + num_held));
+ update_indicator(IND_CALLSETUP, 0);
+
+ break;
+ default:
+ DBG("unhandled state %u", device.setup_state);
+ break;
+ }
}
static void handle_phone_state_change(const void *buf, uint16_t len)
{
const struct hal_cmd_handsfree_phone_state_change *cmd = buf;
- const char *number;
uint8_t status;
if (len != sizeof(*cmd) + cmd->number_len || (cmd->number_len != 0 &&
DBG("active=%u hold=%u state=%u", cmd->num_active, cmd->num_held,
cmd->state);
- number = cmd->number_len ? (char *) cmd->number : "";
-
switch (cmd->state) {
- case HAL_HANDSFREE_CALL_STATE_ACTIVE:
- phone_state_active(cmd->num_active, cmd->num_held);
- break;
- case HAL_HANDSFREE_CALL_STATE_HELD:
- phone_state_held(cmd->num_active, cmd->num_held);
- break;
case HAL_HANDSFREE_CALL_STATE_DIALING:
phone_state_dialing(cmd->num_active, cmd->num_held);
break;
break;
case HAL_HANDSFREE_CALL_STATE_INCOMING:
phone_state_incoming(cmd->num_active, cmd->num_held, cmd->type,
- number);
- break;
- case HAL_HANDSFREE_CALL_STATE_WAITING:
- phone_state_waiting(cmd->num_active, cmd->num_held, cmd->type,
- number);
+ cmd->number, cmd->number_len);
break;
case HAL_HANDSFREE_CALL_STATE_IDLE:
phone_state_idle(cmd->num_active, cmd->num_held);
break;
default:
+ DBG("unhandled new state %u (current state %u)", cmd->state,
+ device.setup_state);
+
status = HAL_STATUS_FAILED;
goto failed;
}
+ device.num_active = cmd->num_active;
+ device.num_held = cmd->num_held;
+ device.setup_state = cmd->state;
+
status = HAL_STATUS_SUCCESS;
failed: