/****************************************************************************** * * Copyright (C) 2010-2014 Broadcom Corporation * * 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. * ******************************************************************************/ /****************************************************************************** * * This is the main implementation file for the NFA HCI. * ******************************************************************************/ #include #include "nfc_api.h" #include "nfa_sys.h" #include "nfa_sys_int.h" #include "nfa_dm_int.h" #include "nfa_hci_api.h" #include "nfa_hci_int.h" #include "nfa_ee_api.h" #include "nfa_ee_int.h" #include "nfa_nv_co.h" #include "nfa_mem_co.h" #include "nfa_hci_defs.h" #include "trace_api.h" /***************************************************************************** ** Global Variables *****************************************************************************/ tNFA_HCI_CB nfa_hci_cb; #ifndef NFA_HCI_NV_READ_TIMEOUT_VAL #define NFA_HCI_NV_READ_TIMEOUT_VAL 1000 #endif #ifndef NFA_HCI_CON_CREATE_TIMEOUT_VAL #define NFA_HCI_CON_CREATE_TIMEOUT_VAL 1000 #endif /***************************************************************************** ** Static Functions *****************************************************************************/ /* event handler function type */ static BOOLEAN nfa_hci_evt_hdlr (BT_HDR *p_msg); static void nfa_hci_sys_enable (void); static void nfa_hci_sys_disable (void); static void nfa_hci_rsp_timeout (tNFA_HCI_EVENT_DATA *p_evt_data); static void nfa_hci_conn_cback (UINT8 conn_id, tNFC_CONN_EVT event, tNFC_CONN *p_data); static void nfa_hci_set_receive_buf (UINT8 pipe); static void nfa_hci_assemble_msg (UINT8 *p_data, UINT16 data_len); static void nfa_hci_handle_nv_read (UINT8 block, tNFA_STATUS status); /***************************************************************************** ** Constants *****************************************************************************/ static const tNFA_SYS_REG nfa_hci_sys_reg = { nfa_hci_sys_enable, nfa_hci_evt_hdlr, nfa_hci_sys_disable, nfa_hci_proc_nfcc_power_mode }; /******************************************************************************* ** ** Function nfa_hci_ee_info_cback ** ** Description Callback function ** ** Returns None ** *******************************************************************************/ void nfa_hci_ee_info_cback (tNFA_EE_DISC_STS status) { UINT8 num_nfcee = 3; tNFA_EE_INFO ee_info[3]; NFA_TRACE_DEBUG1 ("nfa_hci_ee_info_cback (): %d", status); switch (status) { case NFA_EE_DISC_STS_ON: if ( (!nfa_hci_cb.ee_disc_cmplt) &&((nfa_hci_cb.hci_state == NFA_HCI_STATE_STARTUP) || (nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE)) ) { /* NFCEE Discovery is in progress */ nfa_hci_cb.ee_disc_cmplt = TRUE; nfa_hci_cb.num_ee_dis_req_ntf = 0; nfa_hci_cb.num_hot_plug_evts = 0; nfa_hci_cb.conn_id = 0; nfa_hci_startup (); } break; case NFA_EE_DISC_STS_OFF: if (nfa_hci_cb.ee_disable_disc) break; nfa_hci_cb.ee_disable_disc = TRUE; /* Discovery operation is complete, retrieve discovery result */ NFA_EeGetInfo (&num_nfcee, ee_info); nfa_hci_cb.num_nfcee = num_nfcee; if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_WAIT_NETWK_ENABLE) ||(nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE_NETWK_ENABLE) ) { if ( (nfa_hci_cb.num_nfcee <= 1) ||(nfa_hci_cb.num_ee_dis_req_ntf == (nfa_hci_cb.num_nfcee - 1)) ||(nfa_hci_cb.num_hot_plug_evts == (nfa_hci_cb.num_nfcee - 1)) ) { /* No UICC Host is detected or * HOT_PLUG_EVT(s) and or EE DISC REQ Ntf(s) are already received * Get Host list and notify SYS on Initialization complete */ nfa_sys_stop_timer (&nfa_hci_cb.timer); if ( (nfa_hci_cb.num_nfcee > 1) &&(nfa_hci_cb.num_ee_dis_req_ntf != (nfa_hci_cb.num_nfcee - 1)) ) { /* Received HOT PLUG EVT, we will also wait for EE DISC REQ Ntf(s) */ nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, p_nfa_hci_cfg->hci_netwk_enable_timeout); } else { nfa_hci_cb.w4_hci_netwk_init = FALSE; nfa_hciu_send_get_param_cmd (NFA_HCI_ADMIN_PIPE, NFA_HCI_HOST_LIST_INDEX); } } } else if (nfa_hci_cb.num_nfcee <= 1) { /* No UICC Host is detected, HCI NETWORK is enabled */ nfa_hci_cb.w4_hci_netwk_init = FALSE; } break; case NFA_EE_DISC_STS_REQ: nfa_hci_cb.num_ee_dis_req_ntf++; if (nfa_hci_cb.ee_disable_disc) { /* Already received Discovery Ntf */ if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_WAIT_NETWK_ENABLE) ||(nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE_NETWK_ENABLE) ) { /* Received DISC REQ Ntf while waiting for other Host in the network to bootup after DH host bootup is complete */ if (nfa_hci_cb.num_ee_dis_req_ntf == (nfa_hci_cb.num_nfcee - 1)) { /* Received expected number of EE DISC REQ Ntf(s) */ nfa_sys_stop_timer (&nfa_hci_cb.timer); nfa_hci_cb.w4_hci_netwk_init = FALSE; nfa_hciu_send_get_param_cmd (NFA_HCI_ADMIN_PIPE, NFA_HCI_HOST_LIST_INDEX); } } else if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_STARTUP) ||(nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE) ) { /* Received DISC REQ Ntf during DH host bootup */ if (nfa_hci_cb.num_ee_dis_req_ntf == (nfa_hci_cb.num_nfcee - 1)) { /* Received expected number of EE DISC REQ Ntf(s) */ nfa_hci_cb.w4_hci_netwk_init = FALSE; } } } break; } } /******************************************************************************* ** ** Function nfa_hci_init ** ** Description Initialize NFA HCI ** ** Returns None ** *******************************************************************************/ void nfa_hci_init (void) { NFA_TRACE_DEBUG0 ("nfa_hci_init ()"); /* initialize control block */ memset (&nfa_hci_cb, 0, sizeof (tNFA_HCI_CB)); nfa_hci_cb.hci_state = NFA_HCI_STATE_STARTUP; /* register message handler on NFA SYS */ nfa_sys_register (NFA_ID_HCI, &nfa_hci_sys_reg); } /******************************************************************************* ** ** Function nfa_hci_is_valid_cfg ** ** Description Validate hci control block config parameters ** ** Returns None ** *******************************************************************************/ BOOLEAN nfa_hci_is_valid_cfg (void) { UINT8 xx,yy,zz; tNFA_HANDLE reg_app[NFA_HCI_MAX_APP_CB]; UINT8 valid_gate[NFA_HCI_MAX_GATE_CB]; UINT8 app_count = 0; UINT8 gate_count = 0; UINT32 pipe_inx_mask = 0; /* First, see if valid values are stored in app names, send connectivity events flag */ for (xx = 0; xx < NFA_HCI_MAX_APP_CB; xx++) { /* Check if app name is valid with null terminated string */ if (strlen (&nfa_hci_cb.cfg.reg_app_names[xx][0]) > NFA_MAX_HCI_APP_NAME_LEN) return FALSE; /* Send Connectivity event flag can be either TRUE or FALSE */ if ( (nfa_hci_cb.cfg.b_send_conn_evts[xx] != TRUE) &&(nfa_hci_cb.cfg.b_send_conn_evts[xx] != FALSE)) return FALSE; if (nfa_hci_cb.cfg.reg_app_names[xx][0] != 0) { /* Check if the app name is present more than one time in the control block */ for (yy = xx + 1; yy < NFA_HCI_MAX_APP_CB; yy++) { if ( (nfa_hci_cb.cfg.reg_app_names[yy][0] != 0) &&(!strncmp (&nfa_hci_cb.cfg.reg_app_names[xx][0], &nfa_hci_cb.cfg.reg_app_names[yy][0], strlen (nfa_hci_cb.cfg.reg_app_names[xx]))) ) { /* Two app cannot have the same name , NVRAM is corrupted */ NFA_TRACE_EVENT2 ("nfa_hci_is_valid_cfg (%s) Reusing: %u", &nfa_hci_cb.cfg.reg_app_names[xx][0], xx); return FALSE; } } /* Collect list of hci handle */ reg_app[app_count++] = (tNFA_HANDLE) (xx | NFA_HANDLE_GROUP_HCI); } } /* Validate Gate Control block */ for (xx = 0; xx < NFA_HCI_MAX_GATE_CB; xx++) { if (nfa_hci_cb.cfg.dyn_gates[xx].gate_id != 0) { if ( ( (nfa_hci_cb.cfg.dyn_gates[xx].gate_id != NFA_HCI_LOOP_BACK_GATE) &&(nfa_hci_cb.cfg.dyn_gates[xx].gate_id != NFA_HCI_IDENTITY_MANAGEMENT_GATE) &&(nfa_hci_cb.cfg.dyn_gates[xx].gate_id < NFA_HCI_FIRST_HOST_SPECIFIC_GENERIC_GATE)) ||(nfa_hci_cb.cfg.dyn_gates[xx].gate_id > NFA_HCI_LAST_PROP_GATE)) return FALSE; /* Check if the same gate id is present more than once in the control block */ for (yy = xx + 1; yy < NFA_HCI_MAX_GATE_CB; yy++) { if ( (nfa_hci_cb.cfg.dyn_gates[yy].gate_id != 0) &&(nfa_hci_cb.cfg.dyn_gates[xx].gate_id == nfa_hci_cb.cfg.dyn_gates[yy].gate_id) ) { NFA_TRACE_EVENT1 ("nfa_hci_is_valid_cfg Reusing: %u", nfa_hci_cb.cfg.dyn_gates[xx].gate_id); return FALSE; } } if ((nfa_hci_cb.cfg.dyn_gates[xx].gate_owner & (~NFA_HANDLE_GROUP_HCI)) >= NFA_HCI_MAX_APP_CB) { NFA_TRACE_EVENT1 ("nfa_hci_is_valid_cfg Invalid Gate owner: %u", nfa_hci_cb.cfg.dyn_gates[xx].gate_owner); return FALSE; } if (nfa_hci_cb.cfg.dyn_gates[xx].gate_id != NFA_HCI_CONNECTIVITY_GATE) { /* The gate owner should be one of the registered application */ for (zz = 0; zz < app_count; zz++) { if (nfa_hci_cb.cfg.dyn_gates[xx].gate_owner == reg_app[zz]) break; } if (zz == app_count) { NFA_TRACE_EVENT1 ("nfa_hci_is_valid_cfg Invalid Gate owner: %u", nfa_hci_cb.cfg.dyn_gates[xx].gate_owner); return FALSE; } } /* Collect list of allocated gates */ valid_gate[gate_count++] = nfa_hci_cb.cfg.dyn_gates[xx].gate_id; /* No two gates can own a same pipe */ if ((pipe_inx_mask & nfa_hci_cb.cfg.dyn_gates[xx].pipe_inx_mask) != 0) return FALSE; /* Collect the list of pipes on this gate */ pipe_inx_mask |= nfa_hci_cb.cfg.dyn_gates[xx].pipe_inx_mask; } } for (xx = 0; (pipe_inx_mask && (xx < NFA_HCI_MAX_PIPE_CB)); xx++,pipe_inx_mask >>= 1) { /* Every bit set in pipe increment mask indicates a valid pipe */ if (pipe_inx_mask & 1) { /* Check if the pipe is valid one */ if (nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id < NFA_HCI_FIRST_DYNAMIC_PIPE) return FALSE; } } if (xx == NFA_HCI_MAX_PIPE_CB) return FALSE; /* Validate Gate Control block */ for (xx = 0; xx < NFA_HCI_MAX_PIPE_CB; xx++) { if (nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id != 0) { /* Check if pipe id is valid */ if (nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id < NFA_HCI_FIRST_DYNAMIC_PIPE) return FALSE; /* Check if pipe state is valid */ if ( (nfa_hci_cb.cfg.dyn_pipes[xx].pipe_state != NFA_HCI_PIPE_OPENED) &&(nfa_hci_cb.cfg.dyn_pipes[xx].pipe_state != NFA_HCI_PIPE_CLOSED)) return FALSE; /* Check if local gate on which the pipe is created is valid */ if ( (((nfa_hci_cb.cfg.dyn_pipes[xx].local_gate != NFA_HCI_LOOP_BACK_GATE) && (nfa_hci_cb.cfg.dyn_pipes[xx].local_gate != NFA_HCI_IDENTITY_MANAGEMENT_GATE)) && (nfa_hci_cb.cfg.dyn_pipes[xx].local_gate < NFA_HCI_FIRST_HOST_SPECIFIC_GENERIC_GATE)) ||(nfa_hci_cb.cfg.dyn_pipes[xx].local_gate > NFA_HCI_LAST_PROP_GATE)) return FALSE; /* Check if the peer gate on which the pipe is created is valid */ if ( (((nfa_hci_cb.cfg.dyn_pipes[xx].dest_gate != NFA_HCI_LOOP_BACK_GATE) && (nfa_hci_cb.cfg.dyn_pipes[xx].dest_gate != NFA_HCI_IDENTITY_MANAGEMENT_GATE)) && (nfa_hci_cb.cfg.dyn_pipes[xx].dest_gate < NFA_HCI_FIRST_HOST_SPECIFIC_GENERIC_GATE)) ||(nfa_hci_cb.cfg.dyn_pipes[xx].dest_gate > NFA_HCI_LAST_PROP_GATE)) return FALSE; /* Check if the same pipe is present more than once in the control block */ for (yy = xx + 1; yy < NFA_HCI_MAX_PIPE_CB; yy++) { if ( (nfa_hci_cb.cfg.dyn_pipes[yy].pipe_id != 0) &&(nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id == nfa_hci_cb.cfg.dyn_pipes[yy].pipe_id) ) { NFA_TRACE_EVENT1 ("nfa_hci_is_valid_cfg Reusing: %u", nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id); return FALSE; } } /* The local gate should be one of the element in gate control block */ for (zz = 0; zz < gate_count; zz++) { if (nfa_hci_cb.cfg.dyn_pipes[xx].local_gate == valid_gate[zz]) break; } if (zz == gate_count) { NFA_TRACE_EVENT1 ("nfa_hci_is_valid_cfg Invalid Gate: %u", nfa_hci_cb.cfg.dyn_pipes[xx].local_gate); return FALSE; } } } /* Check if admin pipe state is valid */ if ( (nfa_hci_cb.cfg.admin_gate.pipe01_state != NFA_HCI_PIPE_OPENED) &&(nfa_hci_cb.cfg.admin_gate.pipe01_state != NFA_HCI_PIPE_CLOSED)) return FALSE; /* Check if link management pipe state is valid */ if ( (nfa_hci_cb.cfg.link_mgmt_gate.pipe00_state != NFA_HCI_PIPE_OPENED) &&(nfa_hci_cb.cfg.link_mgmt_gate.pipe00_state != NFA_HCI_PIPE_CLOSED)) return FALSE; pipe_inx_mask = nfa_hci_cb.cfg.id_mgmt_gate.pipe_inx_mask; for (xx = 0; (pipe_inx_mask && (xx < NFA_HCI_MAX_PIPE_CB)); xx++,pipe_inx_mask >>= 1) { /* Every bit set in pipe increment mask indicates a valid pipe */ if (pipe_inx_mask & 1) { /* Check if the pipe is valid one */ if (nfa_hci_cb.cfg.dyn_pipes[xx].pipe_id < NFA_HCI_FIRST_DYNAMIC_PIPE) return FALSE; /* Check if the pipe is connected to Identity management gate */ if (nfa_hci_cb.cfg.dyn_pipes[xx].local_gate != NFA_HCI_IDENTITY_MANAGEMENT_GATE) return FALSE; } } if (xx == NFA_HCI_MAX_PIPE_CB) return FALSE; return TRUE; } /******************************************************************************* ** ** Function nfa_hci_cfg_default ** ** Description Configure default values for hci control block ** ** Returns None ** *******************************************************************************/ void nfa_hci_restore_default_config (UINT8 *p_session_id) { memset (&nfa_hci_cb.cfg, 0, sizeof (nfa_hci_cb.cfg)); memcpy (nfa_hci_cb.cfg.admin_gate.session_id, p_session_id, NFA_HCI_SESSION_ID_LEN); nfa_hci_cb.nv_write_needed = TRUE; } /******************************************************************************* ** ** Function nfa_hci_proc_nfcc_power_mode ** ** Description Restore NFA HCI sub-module ** ** Returns None ** *******************************************************************************/ void nfa_hci_proc_nfcc_power_mode (UINT8 nfcc_power_mode) { NFA_TRACE_DEBUG1 ("nfa_hci_proc_nfcc_power_mode () nfcc_power_mode=%d", nfcc_power_mode); /* if NFCC power mode is change to full power */ if (nfcc_power_mode == NFA_DM_PWR_MODE_FULL) { nfa_hci_cb.b_low_power_mode = FALSE; if (nfa_hci_cb.hci_state == NFA_HCI_STATE_IDLE) { nfa_hci_cb.hci_state = NFA_HCI_STATE_RESTORE; nfa_hci_cb.ee_disc_cmplt = FALSE; nfa_hci_cb.ee_disable_disc = TRUE; if (nfa_hci_cb.num_nfcee > 1) nfa_hci_cb.w4_hci_netwk_init = TRUE; else nfa_hci_cb.w4_hci_netwk_init = FALSE; nfa_hci_cb.conn_id = 0; nfa_hci_cb.num_ee_dis_req_ntf = 0; nfa_hci_cb.num_hot_plug_evts = 0; } else { NFA_TRACE_ERROR0 ("nfa_hci_proc_nfcc_power_mode (): Cannot restore now"); nfa_sys_cback_notify_nfcc_power_mode_proc_complete (NFA_ID_HCI); } } else { nfa_hci_cb.hci_state = NFA_HCI_STATE_IDLE; nfa_hci_cb.w4_rsp_evt = FALSE; nfa_hci_cb.conn_id = 0; nfa_sys_stop_timer (&nfa_hci_cb.timer); nfa_hci_cb.b_low_power_mode = TRUE; nfa_sys_cback_notify_nfcc_power_mode_proc_complete (NFA_ID_HCI); } } /******************************************************************************* ** ** Function nfa_hci_dh_startup_complete ** ** Description Initialization of terminal host in HCI Network is completed ** Wait for other host in the network to initialize ** ** Returns None ** *******************************************************************************/ void nfa_hci_dh_startup_complete (void) { if (nfa_hci_cb.w4_hci_netwk_init) { if (nfa_hci_cb.hci_state == NFA_HCI_STATE_STARTUP) { nfa_hci_cb.hci_state = NFA_HCI_STATE_WAIT_NETWK_ENABLE; /* Wait for EE Discovery to complete */ nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, NFA_EE_DISCV_TIMEOUT_VAL); } else if (nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE) { nfa_hci_cb.hci_state = NFA_HCI_STATE_RESTORE_NETWK_ENABLE; /* No HCP packet to DH for a specified period of time indicates all host in the network is initialized */ nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, p_nfa_hci_cfg->hci_netwk_enable_timeout); } } else if ( (nfa_hci_cb.num_nfcee > 1) &&(nfa_hci_cb.num_ee_dis_req_ntf != (nfa_hci_cb.num_nfcee - 1)) ) { if (nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE) nfa_hci_cb.ee_disable_disc = TRUE; /* Received HOT PLUG EVT, we will also wait for EE DISC REQ Ntf(s) */ nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, p_nfa_hci_cfg->hci_netwk_enable_timeout); } else { /* Received EE DISC REQ Ntf(s) */ nfa_hciu_send_get_param_cmd (NFA_HCI_ADMIN_PIPE, NFA_HCI_HOST_LIST_INDEX); } } /******************************************************************************* ** ** Function nfa_hci_startup_complete ** ** Description HCI network initialization is completed ** ** Returns None ** *******************************************************************************/ void nfa_hci_startup_complete (tNFA_STATUS status) { tNFA_HCI_EVT_DATA evt_data; NFA_TRACE_EVENT1 ("nfa_hci_startup_complete (): Status: %u", status); nfa_sys_stop_timer (&nfa_hci_cb.timer); if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE) ||(nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE_NETWK_ENABLE) ) { nfa_ee_proc_hci_info_cback (); nfa_sys_cback_notify_nfcc_power_mode_proc_complete (NFA_ID_HCI); } else { evt_data.hci_init.status = status; nfa_hciu_send_to_all_apps (NFA_HCI_INIT_EVT, &evt_data); nfa_sys_cback_notify_enable_complete (NFA_ID_HCI); } if (status == NFA_STATUS_OK) nfa_hci_cb.hci_state = NFA_HCI_STATE_IDLE; else nfa_hci_cb.hci_state = NFA_HCI_STATE_DISABLED; } /******************************************************************************* ** ** Function nfa_hci_startup ** ** Description Perform HCI startup ** ** Returns None ** *******************************************************************************/ void nfa_hci_startup (void) { tNFA_STATUS status = NFA_STATUS_FAILED; tNFA_EE_INFO ee_info[2]; UINT8 num_nfcee = 2; UINT8 target_handle; UINT8 count = 0; BOOLEAN found = FALSE; if (HCI_LOOPBACK_DEBUG) { /* First step in initialization is to open the admin pipe */ nfa_hciu_send_open_pipe_cmd (NFA_HCI_ADMIN_PIPE); return; } /* We can only start up if NV Ram is read and EE discovery is complete */ if (nfa_hci_cb.nv_read_cmplt && nfa_hci_cb.ee_disc_cmplt && (nfa_hci_cb.conn_id == 0)) { NFA_EeGetInfo (&num_nfcee, ee_info); while ((count < num_nfcee) && (!found)) { target_handle = (UINT8) ee_info[count].ee_handle; if(ee_info[count].ee_interface[0] == NFA_EE_INTERFACE_HCI_ACCESS) { found = TRUE; if (ee_info[count].ee_status == NFA_EE_STATUS_INACTIVE) { NFC_NfceeModeSet (target_handle, NFC_MODE_ACTIVATE); } if ((status = NFC_ConnCreate (NCI_DEST_TYPE_NFCEE, target_handle, NFA_EE_INTERFACE_HCI_ACCESS, nfa_hci_conn_cback)) == NFA_STATUS_OK) nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, NFA_HCI_CON_CREATE_TIMEOUT_VAL); else { nfa_hci_cb.hci_state = NFA_HCI_STATE_DISABLED; NFA_TRACE_ERROR0 ("nfa_hci_startup - Failed to Create Logical connection. HCI Initialization/Restore failed"); nfa_hci_startup_complete (NFA_STATUS_FAILED); } } count++; } if (!found) { NFA_TRACE_ERROR0 ("nfa_hci_startup - HCI ACCESS Interface not discovered. HCI Initialization/Restore failed"); nfa_hci_startup_complete (NFA_STATUS_FAILED); } } } /******************************************************************************* ** ** Function nfa_hci_sys_enable ** ** Description Enable NFA HCI ** ** Returns None ** *******************************************************************************/ static void nfa_hci_sys_enable (void) { NFA_TRACE_DEBUG0 ("nfa_hci_sys_enable ()"); nfa_ee_reg_cback_enable_done (&nfa_hci_ee_info_cback); nfa_nv_co_read ((UINT8 *)&nfa_hci_cb.cfg, sizeof (nfa_hci_cb.cfg),DH_NV_BLOCK); nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, NFA_HCI_NV_READ_TIMEOUT_VAL); } /******************************************************************************* ** ** Function nfa_hci_sys_disable ** ** Description Disable NFA HCI ** ** Returns None ** *******************************************************************************/ static void nfa_hci_sys_disable (void) { tNFA_HCI_EVT_DATA evt_data; nfa_sys_stop_timer (&nfa_hci_cb.timer); if (nfa_hci_cb.conn_id) { if (nfa_sys_is_graceful_disable ()) { /* Tell all applications stack is down */ nfa_hciu_send_to_all_apps (NFA_HCI_EXIT_EVT, &evt_data); NFC_ConnClose (nfa_hci_cb.conn_id); return; } nfa_hci_cb.conn_id = 0; } nfa_hci_cb.hci_state = NFA_HCI_STATE_DISABLED; /* deregister message handler on NFA SYS */ nfa_sys_deregister (NFA_ID_HCI); } /******************************************************************************* ** ** Function nfa_hci_conn_cback ** ** Description This function Process event from NCI ** ** Returns None ** *******************************************************************************/ static void nfa_hci_conn_cback (UINT8 conn_id, tNFC_CONN_EVT event, tNFC_CONN *p_data) { UINT8 *p; BT_HDR *p_pkt = (BT_HDR *) p_data->data.p_data; UINT8 chaining_bit; UINT8 pipe; UINT16 pkt_len; #if (BT_TRACE_VERBOSE == TRUE) char buff[100]; #endif if (event == NFC_CONN_CREATE_CEVT) { nfa_hci_cb.conn_id = conn_id; nfa_hci_cb.buff_size = p_data->conn_create.buff_size; if (nfa_hci_cb.hci_state == NFA_HCI_STATE_STARTUP) { nfa_hci_cb.w4_hci_netwk_init = TRUE; nfa_hciu_alloc_gate (NFA_HCI_CONNECTIVITY_GATE,0); } if (nfa_hci_cb.cfg.admin_gate.pipe01_state == NFA_HCI_PIPE_CLOSED) { /* First step in initialization/restore is to open the admin pipe */ nfa_hciu_send_open_pipe_cmd (NFA_HCI_ADMIN_PIPE); } else { /* Read session id, to know DH session id is correct */ nfa_hciu_send_get_param_cmd (NFA_HCI_ADMIN_PIPE, NFA_HCI_SESSION_IDENTITY_INDEX); } } else if (event == NFC_CONN_CLOSE_CEVT) { nfa_hci_cb.conn_id = 0; nfa_hci_cb.hci_state = NFA_HCI_STATE_DISABLED; /* deregister message handler on NFA SYS */ nfa_sys_deregister (NFA_ID_HCI); } if ((event != NFC_DATA_CEVT) || (p_pkt == NULL)) return; if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_WAIT_NETWK_ENABLE) ||(nfa_hci_cb.hci_state == NFA_HCI_STATE_RESTORE_NETWK_ENABLE) ) { /* Received HCP Packet before timeout, Other Host initialization is not complete */ nfa_sys_stop_timer (&nfa_hci_cb.timer); if (nfa_hci_cb.w4_hci_netwk_init) nfa_sys_start_timer (&nfa_hci_cb.timer, NFA_HCI_RSP_TIMEOUT_EVT, p_nfa_hci_cfg->hci_netwk_enable_timeout); } p = (UINT8 *) (p_pkt + 1) + p_pkt->offset; pkt_len = p_pkt->len; #if (BT_TRACE_PROTOCOL == TRUE) DispHcp (p, pkt_len, TRUE, (BOOLEAN) !nfa_hci_cb.assembling); #endif chaining_bit = ((*p) >> 0x07) & 0x01; pipe = (*p++) & 0x7F; if (pkt_len != 0) pkt_len--; if (nfa_hci_cb.assembling == FALSE) { /* First Segment of a packet */ nfa_hci_cb.type = ((*p) >> 0x06) & 0x03; nfa_hci_cb.inst = (*p++ & 0x3F); if (pkt_len != 0) pkt_len--; nfa_hci_cb.assembly_failed = FALSE; nfa_hci_cb.msg_len = 0; if (chaining_bit == NFA_HCI_MESSAGE_FRAGMENTATION) { nfa_hci_cb.assembling = TRUE; nfa_hci_set_receive_buf (pipe); nfa_hci_assemble_msg (p, pkt_len); } else { if ((pipe >= NFA_HCI_FIRST_DYNAMIC_PIPE) && (nfa_hci_cb.type == NFA_HCI_EVENT_TYPE)) { nfa_hci_set_receive_buf (pipe); nfa_hci_assemble_msg (p, pkt_len); p = nfa_hci_cb.p_msg_data; } } } else { if (nfa_hci_cb.assembly_failed) { /* If Reassembly failed because of insufficient buffer, just drop the new segmented packets */ NFA_TRACE_ERROR1 ("nfa_hci_conn_cback (): Insufficient buffer to Reassemble HCP packet! Dropping :%u bytes", pkt_len); } else { /* Reassemble the packet */ nfa_hci_assemble_msg (p, pkt_len); } if (chaining_bit == NFA_HCI_NO_MESSAGE_FRAGMENTATION) { /* Just added the last segment in the chain. Reset pointers */ nfa_hci_cb.assembling = FALSE; p = nfa_hci_cb.p_msg_data; pkt_len = nfa_hci_cb.msg_len; } } #if (BT_TRACE_VERBOSE == TRUE) NFA_TRACE_EVENT5 ("nfa_hci_conn_cback Recvd data pipe:%d %s chain:%d assmbl:%d len:%d", (UINT8)pipe, nfa_hciu_get_type_inst_names (pipe, nfa_hci_cb.type, nfa_hci_cb.inst, buff), (UINT8)chaining_bit, (UINT8)nfa_hci_cb.assembling, p_pkt->len); #else NFA_TRACE_EVENT6 ("nfa_hci_conn_cback Recvd data pipe:%d Type: %u Inst: %u chain:%d reassm:%d len:%d", pipe, nfa_hci_cb.type, nfa_hci_cb.inst, chaining_bit, nfa_hci_cb.assembling, p_pkt->len); #endif /* If still reassembling fragments, just return */ if (nfa_hci_cb.assembling) { /* if not last packet, release GKI buffer */ GKI_freebuf (p_pkt); return; } /* If we got a response, cancel the response timer. Also, if waiting for */ /* a single response, we can go back to idle state */ if ( (nfa_hci_cb.hci_state == NFA_HCI_STATE_WAIT_RSP) &&((nfa_hci_cb.type == NFA_HCI_RESPONSE_TYPE) || (nfa_hci_cb.w4_rsp_evt && (nfa_hci_cb.type == NFA_HCI_EVENT_TYPE))) ) { nfa_sys_stop_timer (&nfa_hci_cb.timer); nfa_hci_cb.hci_state = NFA_HCI_STATE_IDLE; } switch (pipe) { case NFA_HCI_ADMIN_PIPE: /* Check if data packet is a command, response or event */ if (nfa_hci_cb.type == NFA_HCI_COMMAND_TYPE) { nfa_hci_handle_admin_gate_cmd (p); } else if (nfa_hci_cb.type == NFA_HCI_RESPONSE_TYPE) { nfa_hci_handle_admin_gate_rsp (p, (UINT8) pkt_len); } else if (nfa_hci_cb.type == NFA_HCI_EVENT_TYPE) { nfa_hci_handle_admin_gate_evt (p); } break; case NFA_HCI_LINK_MANAGEMENT_PIPE: /* We don't send Link Management commands, we only get them */ if (nfa_hci_cb.type == NFA_HCI_COMMAND_TYPE) nfa_hci_handle_link_mgm_gate_cmd (p); break; default: if (pipe >= NFA_HCI_FIRST_DYNAMIC_PIPE) nfa_hci_handle_dyn_pipe_pkt (pipe, p, pkt_len); break; } if ((nfa_hci_cb.type == NFA_HCI_RESPONSE_TYPE) || (nfa_hci_cb.w4_rsp_evt && (nfa_hci_cb.type == NFA_HCI_EVENT_TYPE))) { nfa_hci_cb.w4_rsp_evt = FALSE; } /* Send a message to ouselves to check for anything to do */ p_pkt->event = NFA_HCI_CHECK_QUEUE_EVT; p_pkt->len = 0; nfa_sys_sendmsg (p_pkt); } /******************************************************************************* ** ** Function nfa_hci_handle_nv_read ** ** Description handler function for nv read complete event ** ** Returns None ** *******************************************************************************/ void nfa_hci_handle_nv_read (UINT8 block, tNFA_STATUS status) { UINT8 session_id[NFA_HCI_SESSION_ID_LEN]; UINT8 default_session[NFA_HCI_SESSION_ID_LEN] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}; UINT8 reset_session[NFA_HCI_SESSION_ID_LEN] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}; UINT32 os_tick; if (block == DH_NV_BLOCK) { /* Stop timer as NVDATA Read Completed */ nfa_sys_stop_timer (&nfa_hci_cb.timer); nfa_hci_cb.nv_read_cmplt = TRUE; if ( (status != NFA_STATUS_OK) ||(!nfa_hci_is_valid_cfg ()) ||(!(memcmp (nfa_hci_cb.cfg.admin_gate.session_id, default_session, NFA_HCI_SESSION_ID_LEN))) ||(!(memcmp (nfa_hci_cb.cfg.admin_gate.session_id, reset_session, NFA_HCI_SESSION_ID_LEN))) ) { nfa_hci_cb.b_hci_netwk_reset = TRUE; /* Set a new session id so that we clear all pipes later after seeing a difference with the HC Session ID */ memcpy (&session_id[(NFA_HCI_SESSION_ID_LEN / 2)], nfa_hci_cb.cfg.admin_gate.session_id, (NFA_HCI_SESSION_ID_LEN / 2)); os_tick = GKI_get_os_tick_count (); memcpy (session_id, (UINT8 *)&os_tick, (NFA_HCI_SESSION_ID_LEN / 2)); nfa_hci_restore_default_config (session_id); } nfa_hci_startup (); } } /******************************************************************************* ** ** Function nfa_hci_rsp_timeout ** ** Description action function to process timeout ** ** Returns None ** *******************************************************************************/ void nfa_hci_rsp_timeout (tNFA_HCI_EVENT_DATA *p_evt_data) { tNFA_HCI_EVT evt = 0; tNFA_HCI_EVT_DATA evt_data; UINT8 delete_pipe; NFA_TRACE_EVENT2 ("nfa_hci_rsp_timeout () State: %u Cmd: %u", nfa_hci_cb.hci_state, nfa_hci_cb.cmd_sent); evt_data.status = NFA_STATUS_FAILED; switch (nfa_hci_cb.hci_state) { case NFA_HCI_STATE_STARTUP: case NFA_HCI_STATE_RESTORE: NFA_TRACE_ERROR0 ("nfa_hci_rsp_timeout - Initialization failed!"); nfa_hci_startup_complete (NFA_STATUS_TIMEOUT); break; case NFA_HCI_STATE_WAIT_NETWK_ENABLE: case NFA_HCI_STATE_RESTORE_NETWK_ENABLE: if (nfa_hci_cb.w4_hci_netwk_init) { /* HCI Network is enabled */ nfa_hci_cb.w4_hci_netwk_init = FALSE; nfa_hciu_send_get_param_cmd (NFA_HCI_ADMIN_PIPE, NFA_HCI_HOST_LIST_INDEX); } else { nfa_hci_startup_complete (NFA_STATUS_FAILED); } break; case NFA_HCI_STATE_REMOVE_GATE: /* Something wrong, NVRAM data could be corrupt */ if (nfa_hci_cb.cmd_sent == NFA_HCI_ADM_DELETE_PIPE) { nfa_hciu_send_clear_all_pipe_cmd (); } else { nfa_hciu_remove_all_pipes_from_host (0); nfa_hci_api_dealloc_gate (NULL); } break; case NFA_HCI_STATE_APP_DEREGISTER: /* Something wrong, NVRAM data could be corrupt */ if (nfa_hci_cb.cmd_sent == NFA_HCI_ADM_DELETE_PIPE) { nfa_hciu_send_clear_all_pipe_cmd (); } else { nfa_hciu_remove_all_pipes_from_host (0); nfa_hci_api_deregister (NULL); } break; case NFA_HCI_STATE_WAIT_RSP: nfa_hci_cb.hci_state = NFA_HCI_STATE_IDLE; if (nfa_hci_cb.w4_rsp_evt) { nfa_hci_cb.w4_rsp_evt = FALSE; evt = NFA_HCI_EVENT_RCVD_EVT; evt_data.rcvd_evt.pipe = nfa_hci_cb.pipe_in_use; evt_data.rcvd_evt.evt_code = 0; evt_data.rcvd_evt.evt_len = 0; evt_data.rcvd_evt.p_evt_buf = NULL; nfa_hci_cb.rsp_buf_size = 0; nfa_hci_cb.p_rsp_buf = NULL; break; } delete_pipe = 0; switch (nfa_hci_cb.cmd_sent) { case NFA_HCI_ANY_SET_PARAMETER: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already and release the pipe. But still send delete pipe command to be safe. */ delete_pipe = nfa_hci_cb.pipe_in_use; evt_data.registry.pipe = nfa_hci_cb.pipe_in_use; evt_data.registry.data_len = 0; evt_data.registry.index = nfa_hci_cb.param_in_use; evt = NFA_HCI_SET_REG_RSP_EVT; break; case NFA_HCI_ANY_GET_PARAMETER: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already and release the pipe. But still send delete pipe command to be safe. */ delete_pipe = nfa_hci_cb.pipe_in_use; evt_data.registry.pipe = nfa_hci_cb.pipe_in_use; evt_data.registry.data_len = 0; evt_data.registry.index = nfa_hci_cb.param_in_use; evt = NFA_HCI_GET_REG_RSP_EVT; break; case NFA_HCI_ANY_OPEN_PIPE: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already and release the pipe. But still send delete pipe command to be safe. */ delete_pipe = nfa_hci_cb.pipe_in_use; evt_data.opened.pipe = nfa_hci_cb.pipe_in_use; evt = NFA_HCI_OPEN_PIPE_EVT; break; case NFA_HCI_ANY_CLOSE_PIPE: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already and release the pipe. But still send delete pipe command to be safe. */ delete_pipe = nfa_hci_cb.pipe_in_use; evt_data.closed.pipe = nfa_hci_cb.pipe_in_use; evt = NFA_HCI_CLOSE_PIPE_EVT; break; case NFA_HCI_ADM_CREATE_PIPE: evt_data.created.pipe = nfa_hci_cb.pipe_in_use; evt_data.created.source_gate = nfa_hci_cb.local_gate_in_use; evt_data.created.dest_host = nfa_hci_cb.remote_host_in_use; evt_data.created.dest_gate = nfa_hci_cb.remote_gate_in_use; evt = NFA_HCI_CREATE_PIPE_EVT; break; case NFA_HCI_ADM_DELETE_PIPE: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already. Just release the pipe. */ if (nfa_hci_cb.pipe_in_use <= NFA_HCI_LAST_DYNAMIC_PIPE) nfa_hciu_release_pipe (nfa_hci_cb.pipe_in_use); evt_data.deleted.pipe = nfa_hci_cb.pipe_in_use; evt = NFA_HCI_DELETE_PIPE_EVT; break; default: /* * As no response to the command sent on this pipe, we may assume the pipe is * deleted already and release the pipe. But still send delete pipe command to be safe. */ delete_pipe = nfa_hci_cb.pipe_in_use; break; } if (delete_pipe && (delete_pipe <= NFA_HCI_LAST_DYNAMIC_PIPE)) { nfa_hciu_send_delete_pipe_cmd (delete_pipe); nfa_hciu_release_pipe (delete_pipe); } break; case NFA_HCI_STATE_DISABLED: default: NFA_TRACE_DEBUG0 ("nfa_hci_rsp_timeout () Timeout in DISABLED/ Invalid state"); break; } if (evt != 0) nfa_hciu_send_to_app (evt, &evt_data, nfa_hci_cb.app_in_use); } /******************************************************************************* ** ** Function nfa_hci_set_receive_buf ** ** Description Set reassembly buffer for incoming message ** ** Returns status ** *******************************************************************************/ static void nfa_hci_set_receive_buf (UINT8 pipe) { if ( (pipe >= NFA_HCI_FIRST_DYNAMIC_PIPE) &&(nfa_hci_cb.type == NFA_HCI_EVENT_TYPE) ) { if ( (nfa_hci_cb.rsp_buf_size) &&(nfa_hci_cb.p_rsp_buf != NULL) ) { nfa_hci_cb.p_msg_data = nfa_hci_cb.p_rsp_buf; nfa_hci_cb.max_msg_len = nfa_hci_cb.rsp_buf_size; return; } } nfa_hci_cb.p_msg_data = nfa_hci_cb.msg_data; nfa_hci_cb.max_msg_len = NFA_MAX_HCI_EVENT_LEN; } /******************************************************************************* ** ** Function nfa_hci_assemble_msg ** ** Description Reassemble the incoming message ** ** Returns None ** *******************************************************************************/ static void nfa_hci_assemble_msg (UINT8 *p_data, UINT16 data_len) { if ((nfa_hci_cb.msg_len + data_len) > nfa_hci_cb.max_msg_len) { /* Fill the buffer as much it can hold */ memcpy (&nfa_hci_cb.p_msg_data[nfa_hci_cb.msg_len], p_data, (nfa_hci_cb.max_msg_len - nfa_hci_cb.msg_len)); nfa_hci_cb.msg_len = nfa_hci_cb.max_msg_len; /* Set Reassembly failed */ nfa_hci_cb.assembly_failed = TRUE; NFA_TRACE_ERROR1 ("nfa_hci_assemble_msg (): Insufficient buffer to Reassemble HCP packet! Dropping :%u bytes", ((nfa_hci_cb.msg_len + data_len) - nfa_hci_cb.max_msg_len)); } else { memcpy (&nfa_hci_cb.p_msg_data[nfa_hci_cb.msg_len], p_data, data_len); nfa_hci_cb.msg_len += data_len; } } /******************************************************************************* ** ** Function nfa_hci_evt_hdlr ** ** Description Processing all event for NFA HCI ** ** Returns TRUE if p_msg needs to be deallocated ** *******************************************************************************/ static BOOLEAN nfa_hci_evt_hdlr (BT_HDR *p_msg) { tNFA_HCI_EVENT_DATA *p_evt_data = (tNFA_HCI_EVENT_DATA *)p_msg; #if (BT_TRACE_VERBOSE == TRUE) NFA_TRACE_EVENT4 ("nfa_hci_evt_hdlr state: %s (%d) event: %s (0x%04x)", nfa_hciu_get_state_name (nfa_hci_cb.hci_state), nfa_hci_cb.hci_state, nfa_hciu_get_event_name (p_evt_data->hdr.event), p_evt_data->hdr.event); #else NFA_TRACE_EVENT2 ("nfa_hci_evt_hdlr state: %d event: 0x%04x", nfa_hci_cb.hci_state, p_evt_data->hdr.event); #endif /* If this is an API request, queue it up */ if ((p_msg->event >= NFA_HCI_FIRST_API_EVENT) && (p_msg->event <= NFA_HCI_LAST_API_EVENT)) { GKI_enqueue (&nfa_hci_cb.hci_api_q, p_msg); } else { switch (p_msg->event) { case NFA_HCI_RSP_NV_READ_EVT: nfa_hci_handle_nv_read (p_evt_data->nv_read.block, p_evt_data->nv_read.status); break; case NFA_HCI_RSP_NV_WRITE_EVT: /* NV Ram write completed - nothing to do... */ break; case NFA_HCI_RSP_TIMEOUT_EVT: nfa_hci_rsp_timeout ((tNFA_HCI_EVENT_DATA *)p_msg); break; case NFA_HCI_CHECK_QUEUE_EVT: if (HCI_LOOPBACK_DEBUG) { if (p_msg->len != 0) { tNFC_DATA_CEVT xx; xx.p_data = p_msg; nfa_hci_conn_cback (0, NFC_DATA_CEVT, (tNFC_CONN *)&xx); return FALSE; } } break; } } if ((p_msg->event > NFA_HCI_LAST_API_EVENT)) GKI_freebuf (p_msg); nfa_hci_check_api_requests (); if (nfa_hciu_is_no_host_resetting ()) nfa_hci_check_pending_api_requests (); if ((nfa_hci_cb.hci_state == NFA_HCI_STATE_IDLE) && (nfa_hci_cb.nv_write_needed)) { nfa_hci_cb.nv_write_needed = FALSE; nfa_nv_co_write ((UINT8 *)&nfa_hci_cb.cfg, sizeof (nfa_hci_cb.cfg),DH_NV_BLOCK); } return FALSE; }