/****************************************************************************** * * 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 device manager. * ******************************************************************************/ #include <string.h> #include "nfa_api.h" #include "nfa_sys.h" #include "nfa_dm_int.h" #include "nfa_sys_int.h" /***************************************************************************** ** Constants and types *****************************************************************************/ static const tNFA_SYS_REG nfa_dm_sys_reg = { nfa_dm_sys_enable, nfa_dm_evt_hdlr, nfa_dm_sys_disable, nfa_dm_proc_nfcc_power_mode }; tNFA_DM_CB nfa_dm_cb = {FALSE}; #define NFA_DM_NUM_ACTIONS (NFA_DM_MAX_EVT & 0x00ff) /* type for action functions */ typedef BOOLEAN (*tNFA_DM_ACTION) (tNFA_DM_MSG *p_data); /* action function list */ const tNFA_DM_ACTION nfa_dm_action[] = { /* device manager local device API events */ nfa_dm_enable, /* NFA_DM_API_ENABLE_EVT */ nfa_dm_disable, /* NFA_DM_API_DISABLE_EVT */ nfa_dm_set_config, /* NFA_DM_API_SET_CONFIG_EVT */ nfa_dm_get_config, /* NFA_DM_API_GET_CONFIG_EVT */ nfa_dm_act_request_excl_rf_ctrl, /* NFA_DM_API_REQUEST_EXCL_RF_CTRL_EVT */ nfa_dm_act_release_excl_rf_ctrl, /* NFA_DM_API_RELEASE_EXCL_RF_CTRL_EVT */ nfa_dm_act_enable_polling, /* NFA_DM_API_ENABLE_POLLING_EVT */ nfa_dm_act_disable_polling, /* NFA_DM_API_DISABLE_POLLING_EVT */ nfa_dm_act_enable_listening, /* NFA_DM_API_ENABLE_LISTENING_EVT */ nfa_dm_act_disable_listening, /* NFA_DM_API_DISABLE_LISTENING_EVT */ nfa_dm_act_pause_p2p, /* NFA_DM_API_PAUSE_P2P_EVT */ nfa_dm_act_resume_p2p, /* NFA_DM_API_RESUME_P2P_EVT */ nfa_dm_act_send_raw_frame, /* NFA_DM_API_RAW_FRAME_EVT */ nfa_dm_set_p2p_listen_tech, /* NFA_DM_API_SET_P2P_LISTEN_TECH_EVT */ nfa_dm_act_start_rf_discovery, /* NFA_DM_API_START_RF_DISCOVERY_EVT */ nfa_dm_act_stop_rf_discovery, /* NFA_DM_API_STOP_RF_DISCOVERY_EVT */ nfa_dm_act_set_rf_disc_duration, /* NFA_DM_API_SET_RF_DISC_DURATION_EVT */ nfa_dm_act_select, /* NFA_DM_API_SELECT_EVT */ nfa_dm_act_update_rf_params, /* NFA_DM_API_UPDATE_RF_PARAMS_EVT */ nfa_dm_act_deactivate, /* NFA_DM_API_DEACTIVATE_EVT */ nfa_dm_act_power_off_sleep, /* NFA_DM_API_POWER_OFF_SLEEP_EVT */ nfa_dm_ndef_reg_hdlr, /* NFA_DM_API_REG_NDEF_HDLR_EVT */ nfa_dm_ndef_dereg_hdlr, /* NFA_DM_API_DEREG_NDEF_HDLR_EVT */ nfa_dm_act_reg_vsc, /* NFA_DM_API_REG_VSC_EVT */ nfa_dm_act_send_vsc, /* NFA_DM_API_SEND_VSC_EVT */ nfa_dm_act_disable_timeout /* NFA_DM_TIMEOUT_DISABLE_EVT */ }; /***************************************************************************** ** Local function prototypes *****************************************************************************/ #if (BT_TRACE_VERBOSE == TRUE) static char *nfa_dm_evt_2_str (UINT16 event); #endif /******************************************************************************* ** ** Function nfa_dm_init ** ** Description Initialises the NFC device manager ** ** Returns void ** *******************************************************************************/ void nfa_dm_init (void) { NFA_TRACE_DEBUG0 ("nfa_dm_init ()"); memset (&nfa_dm_cb, 0, sizeof (tNFA_DM_CB)); nfa_dm_cb.poll_disc_handle = NFA_HANDLE_INVALID; nfa_dm_cb.disc_cb.disc_duration = NFA_DM_DISC_DURATION_POLL; nfa_dm_cb.nfcc_pwr_mode = NFA_DM_PWR_MODE_FULL; /* register message handler on NFA SYS */ nfa_sys_register (NFA_ID_DM, &nfa_dm_sys_reg); } /******************************************************************************* ** ** Function nfa_dm_evt_hdlr ** ** Description Event handling function for DM ** ** ** Returns void ** *******************************************************************************/ BOOLEAN nfa_dm_evt_hdlr (BT_HDR *p_msg) { BOOLEAN freebuf = TRUE; UINT16 event = p_msg->event & 0x00ff; #if (BT_TRACE_VERBOSE == TRUE) NFA_TRACE_EVENT2 ("nfa_dm_evt_hdlr event: %s (0x%02x)", nfa_dm_evt_2_str (event), event); #else NFA_TRACE_EVENT1 ("nfa_dm_evt_hdlr event: 0x%x", event); #endif /* execute action functions */ if (event < NFA_DM_NUM_ACTIONS) { freebuf = (*nfa_dm_action[event]) ((tNFA_DM_MSG*) p_msg); } return freebuf; } /******************************************************************************* ** ** Function nfa_dm_sys_disable ** ** Description This function is called after all subsystems have been disabled. ** ** Returns void ** *******************************************************************************/ void nfa_dm_sys_disable (void) { /* Disable the DM sub-system */ /* If discovery state is not IDLE or DEACTIVATED and graceful disable, */ /* then we need to deactivate link or stop discovery */ if (nfa_sys_is_graceful_disable ()) { if ( (nfa_dm_cb.disc_cb.disc_state == NFA_DM_RFST_IDLE) &&((nfa_dm_cb.disc_cb.disc_flags & NFA_DM_DISC_FLAGS_DISABLING) == 0) ) { /* discovery is not started */ nfa_dm_disable_complete (); } else { /* probably waiting to be disabled */ NFA_TRACE_WARNING2 ("DM disc_state state = %d disc_flags:0x%x", nfa_dm_cb.disc_cb.disc_state, nfa_dm_cb.disc_cb.disc_flags); } } else { nfa_dm_disable_complete (); } } /******************************************************************************* ** ** Function nfa_dm_is_protocol_supported ** ** Description Check if protocol is supported by RW module ** ** Returns TRUE if protocol is supported by NFA ** *******************************************************************************/ BOOLEAN nfa_dm_is_protocol_supported (tNFC_PROTOCOL protocol, UINT8 sel_res) { return ( (protocol == NFC_PROTOCOL_T1T) ||((protocol == NFC_PROTOCOL_T2T) && (sel_res == NFC_SEL_RES_NFC_FORUM_T2T)) ||(protocol == NFC_PROTOCOL_T3T) ||(protocol == NFC_PROTOCOL_ISO_DEP) ||(protocol == NFC_PROTOCOL_NFC_DEP) ||(protocol == NFC_PROTOCOL_15693) ); } /******************************************************************************* ** ** Function nfa_dm_is_active ** ** Description check if all modules of NFA is done with enable process and ** NFA is not restoring NFCC. ** ** Returns TRUE, if NFA_DM_ENABLE_EVT is reported and it is not restoring NFCC ** *******************************************************************************/ BOOLEAN nfa_dm_is_active (void) { NFA_TRACE_DEBUG1 ("nfa_dm_is_active () flags:0x%x", nfa_dm_cb.flags); if ( (nfa_dm_cb.flags & NFA_DM_FLAGS_DM_IS_ACTIVE) &&((nfa_dm_cb.flags & (NFA_DM_FLAGS_ENABLE_EVT_PEND | NFA_DM_FLAGS_NFCC_IS_RESTORING | NFA_DM_FLAGS_POWER_OFF_SLEEP)) == 0) ) { return TRUE; } else return FALSE; } /******************************************************************************* ** ** Function nfa_dm_check_set_config ** ** Description Update config parameters only if it's different from NFCC ** ** ** Returns tNFA_STATUS ** *******************************************************************************/ tNFA_STATUS nfa_dm_check_set_config (UINT8 tlv_list_len, UINT8 *p_tlv_list, BOOLEAN app_init) { UINT8 type, len, *p_value, *p_stored, max_len; UINT8 xx = 0, updated_len = 0, *p_cur_len; BOOLEAN update; tNFC_STATUS nfc_status; UINT32 cur_bit; NFA_TRACE_DEBUG0 ("nfa_dm_check_set_config ()"); /* We only allow 32 pending SET_CONFIGs */ if (nfa_dm_cb.setcfg_pending_num >= NFA_DM_SETCONFIG_PENDING_MAX) { NFA_TRACE_ERROR0 ("nfa_dm_check_set_config () error: pending number of SET_CONFIG exceeded"); return NFA_STATUS_FAILED; } while (tlv_list_len - xx >= 2) /* at least type and len */ { update = FALSE; type = *(p_tlv_list + xx); len = *(p_tlv_list + xx + 1); p_value = p_tlv_list + xx + 2; p_cur_len = NULL; switch (type) { /* ** Poll F Configuration */ case NFC_PMID_PF_RC: p_stored = nfa_dm_cb.params.pf_rc; max_len = NCI_PARAM_LEN_PF_RC; break; case NFC_PMID_TOTAL_DURATION: p_stored = nfa_dm_cb.params.total_duration; max_len = NCI_PARAM_LEN_TOTAL_DURATION; break; /* ** Listen A Configuration */ case NFC_PMID_LA_BIT_FRAME_SDD: p_stored = nfa_dm_cb.params.la_bit_frame_sdd; max_len = NCI_PARAM_LEN_LA_BIT_FRAME_SDD; p_cur_len = &nfa_dm_cb.params.la_bit_frame_sdd_len; break; case NFC_PMID_LA_PLATFORM_CONFIG: p_stored = nfa_dm_cb.params.la_platform_config; max_len = NCI_PARAM_LEN_LA_PLATFORM_CONFIG; p_cur_len = &nfa_dm_cb.params.la_platform_config_len; break; case NFC_PMID_LA_SEL_INFO: p_stored = nfa_dm_cb.params.la_sel_info; max_len = NCI_PARAM_LEN_LA_SEL_INFO; p_cur_len = &nfa_dm_cb.params.la_sel_info_len; break; case NFC_PMID_LA_NFCID1: p_stored = nfa_dm_cb.params.la_nfcid1; max_len = NCI_NFCID1_MAX_LEN; p_cur_len = &nfa_dm_cb.params.la_nfcid1_len; break; case NFC_PMID_LA_HIST_BY: p_stored = nfa_dm_cb.params.la_hist_by; max_len = NCI_MAX_HIS_BYTES_LEN; p_cur_len = &nfa_dm_cb.params.la_hist_by_len; break; /* ** Listen B Configuration */ case NFC_PMID_LB_SENSB_INFO: p_stored = nfa_dm_cb.params.lb_sensb_info; max_len = NCI_PARAM_LEN_LB_SENSB_INFO; p_cur_len = &nfa_dm_cb.params.lb_sensb_info_len; break; case NFC_PMID_LB_NFCID0: p_stored = nfa_dm_cb.params.lb_nfcid0; max_len = NCI_PARAM_LEN_LB_NFCID0; p_cur_len = &nfa_dm_cb.params.lb_nfcid0_len; break; case NFC_PMID_LB_APPDATA: p_stored = nfa_dm_cb.params.lb_appdata; max_len = NCI_PARAM_LEN_LB_APPDATA; p_cur_len = &nfa_dm_cb.params.lb_appdata_len; break; case NFC_PMID_LB_ADC_FO: p_stored = nfa_dm_cb.params.lb_adc_fo; max_len = NCI_PARAM_LEN_LB_ADC_FO; p_cur_len = &nfa_dm_cb.params.lb_adc_fo_len; break; case NFC_PMID_LB_H_INFO: p_stored = nfa_dm_cb.params.lb_h_info; max_len = NCI_MAX_ATTRIB_LEN; p_cur_len = &nfa_dm_cb.params.lb_h_info_len; break; /* ** Listen F Configuration */ case NFC_PMID_LF_PROTOCOL: p_stored = nfa_dm_cb.params.lf_protocol; max_len = NCI_PARAM_LEN_LF_PROTOCOL; p_cur_len = &nfa_dm_cb.params.lf_protocol_len; break; case NFC_PMID_LF_T3T_FLAGS2: p_stored = nfa_dm_cb.params.lf_t3t_flags2; max_len = NCI_PARAM_LEN_LF_T3T_FLAGS2; p_cur_len = &nfa_dm_cb.params.lf_t3t_flags2_len; break; case NFC_PMID_LF_T3T_PMM: p_stored = nfa_dm_cb.params.lf_t3t_pmm; max_len = NCI_PARAM_LEN_LF_T3T_PMM; break; /* ** ISO-DEP and NFC-DEP Configuration */ case NFC_PMID_FWI: p_stored = nfa_dm_cb.params.fwi; max_len = NCI_PARAM_LEN_FWI; break; case NFC_PMID_WT: p_stored = nfa_dm_cb.params.wt; max_len = NCI_PARAM_LEN_WT; break; case NFC_PMID_ATR_REQ_GEN_BYTES: p_stored = nfa_dm_cb.params.atr_req_gen_bytes; max_len = NCI_MAX_GEN_BYTES_LEN; p_cur_len = &nfa_dm_cb.params.atr_req_gen_bytes_len; break; case NFC_PMID_ATR_RES_GEN_BYTES: p_stored = nfa_dm_cb.params.atr_res_gen_bytes; max_len = NCI_MAX_GEN_BYTES_LEN; p_cur_len = &nfa_dm_cb.params.atr_res_gen_bytes_len; break; default: /* ** Listen F Configuration */ if ((type >= NFC_PMID_LF_T3T_ID1) && (type < NFC_PMID_LF_T3T_ID1 + NFA_CE_LISTEN_INFO_MAX)) { p_stored = nfa_dm_cb.params.lf_t3t_id[type - NFC_PMID_LF_T3T_ID1]; max_len = NCI_PARAM_LEN_LF_T3T_ID; } else { /* we don't stored this config items */ update = TRUE; p_stored = NULL; } break; } if ((p_stored)&&(len <= max_len)) { if (p_cur_len) { if (*p_cur_len != len) { *p_cur_len = len; update = TRUE; } else if (memcmp (p_value, p_stored, len)) { update = TRUE; } } else if (len == max_len) /* fixed length */ { if (memcmp (p_value, p_stored, len)) { update = TRUE; } } } if (update) { /* we don't store this type */ if (p_stored) { memcpy (p_stored, p_value, len); } /* If need to change TLV in the original list. (Do not modify list if app_init) */ if ((updated_len != xx) && (!app_init)) { memcpy (p_tlv_list + updated_len, p_tlv_list + xx, (len + 2)); } updated_len += (len + 2); } xx += len + 2; /* move to next TLV */ } /* If any TVLs to update, or if the SetConfig was initiated by the application, then send the SET_CONFIG command */ if (updated_len || app_init) { if ((nfc_status = NFC_SetConfig (updated_len, p_tlv_list)) == NFC_STATUS_OK) { /* Keep track of whether we will need to notify NFA_DM_SET_CONFIG_EVT on NFC_SET_CONFIG_REVT */ /* Get the next available bit offset for this setconfig (based on how many SetConfigs are outstanding) */ cur_bit = (UINT32) (1 << nfa_dm_cb.setcfg_pending_num); /* If setconfig is due to NFA_SetConfig: then set the bit (NFA_DM_SET_CONFIG_EVT needed on NFC_SET_CONFIG_REVT) */ if (app_init) { nfa_dm_cb.setcfg_pending_mask |= cur_bit; } /* Otherwise setconfig is internal: clear the bit (NFA_DM_SET_CONFIG_EVT not needed on NFC_SET_CONFIG_REVT) */ else { nfa_dm_cb.setcfg_pending_mask &= ~cur_bit; } /* Increment setcfg_pending counter */ nfa_dm_cb.setcfg_pending_num++; } return (nfc_status); } else { return NFA_STATUS_OK; } } #if (BT_TRACE_VERBOSE == TRUE) /******************************************************************************* ** ** Function nfa_dm_nfc_revt_2_str ** ** Description convert nfc revt to string ** *******************************************************************************/ static char *nfa_dm_evt_2_str (UINT16 event) { switch (NFA_SYS_EVT_START (NFA_ID_DM) | event) { case NFA_DM_API_ENABLE_EVT: return "NFA_DM_API_ENABLE_EVT"; case NFA_DM_API_DISABLE_EVT: return "NFA_DM_API_DISABLE_EVT"; case NFA_DM_API_SET_CONFIG_EVT: return "NFA_DM_API_SET_CONFIG_EVT"; case NFA_DM_API_GET_CONFIG_EVT: return "NFA_DM_API_GET_CONFIG_EVT"; case NFA_DM_API_REQUEST_EXCL_RF_CTRL_EVT: return "NFA_DM_API_REQUEST_EXCL_RF_CTRL_EVT"; case NFA_DM_API_RELEASE_EXCL_RF_CTRL_EVT: return "NFA_DM_API_RELEASE_EXCL_RF_CTRL_EVT"; case NFA_DM_API_ENABLE_POLLING_EVT: return "NFA_DM_API_ENABLE_POLLING_EVT"; case NFA_DM_API_DISABLE_POLLING_EVT: return "NFA_DM_API_DISABLE_POLLING_EVT"; case NFA_DM_API_ENABLE_LISTENING_EVT: return "NFA_DM_API_ENABLE_LISTENING_EVT"; case NFA_DM_API_DISABLE_LISTENING_EVT: return "NFA_DM_API_DISABLE_LISTENING_EVT"; case NFA_DM_API_PAUSE_P2P_EVT: return "NFA_DM_API_PAUSE_P2P_EVT"; case NFA_DM_API_RESUME_P2P_EVT: return "NFA_DM_API_RESUME_P2P_EVT"; case NFA_DM_API_RAW_FRAME_EVT: return "NFA_DM_API_RAW_FRAME_EVT"; case NFA_DM_API_SET_P2P_LISTEN_TECH_EVT: return "NFA_DM_API_SET_P2P_LISTEN_TECH_EVT"; case NFA_DM_API_START_RF_DISCOVERY_EVT: return "NFA_DM_API_START_RF_DISCOVERY_EVT"; case NFA_DM_API_STOP_RF_DISCOVERY_EVT: return "NFA_DM_API_STOP_RF_DISCOVERY_EVT"; case NFA_DM_API_SET_RF_DISC_DURATION_EVT: return "NFA_DM_API_SET_RF_DISC_DURATION_EVT"; case NFA_DM_API_SELECT_EVT: return "NFA_DM_API_SELECT_EVT"; case NFA_DM_API_UPDATE_RF_PARAMS_EVT: return "NFA_DM_API_UPDATE_RF_PARAMS_EVT"; case NFA_DM_API_DEACTIVATE_EVT: return "NFA_DM_API_DEACTIVATE_EVT"; case NFA_DM_API_POWER_OFF_SLEEP_EVT: return "NFA_DM_API_POWER_OFF_SLEEP_EVT"; case NFA_DM_API_REG_NDEF_HDLR_EVT: return "NFA_DM_API_REG_NDEF_HDLR_EVT"; case NFA_DM_API_DEREG_NDEF_HDLR_EVT: return "NFA_DM_API_DEREG_NDEF_HDLR_EVT"; case NFA_DM_TIMEOUT_DISABLE_EVT: return "NFA_DM_TIMEOUT_DISABLE_EVT"; } return "Unknown or Vendor Specific"; } #endif /* BT_TRACE_VERBOSE */