| /****************************************************************************** |
| * |
| * Copyright (C) 2009-2012 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 implementation file for the MCAP Main Control Block and |
| * Utility functions. |
| * |
| ******************************************************************************/ |
| #include <assert.h> |
| #include <string.h> |
| |
| #include "bt_target.h" |
| #include "bt_common.h" |
| #include "mca_api.h" |
| #include "mca_defs.h" |
| #include "mca_int.h" |
| #include "l2c_api.h" |
| |
| /* Main Control block for MCA */ |
| #if MCA_DYNAMIC_MEMORY == FALSE |
| tMCA_CB mca_cb; |
| #endif |
| |
| /***************************************************************************** |
| ** constants |
| *****************************************************************************/ |
| |
| /* table of standard opcode message size */ |
| const UINT8 mca_std_msg_len[MCA_NUM_STANDARD_OPCODE] = { |
| 4, /* MCA_OP_ERROR_RSP */ |
| 5, /* MCA_OP_MDL_CREATE_REQ */ |
| 5, /* MCA_OP_MDL_CREATE_RSP */ |
| 3, /* MCA_OP_MDL_RECONNECT_REQ */ |
| 4, /* MCA_OP_MDL_RECONNECT_RSP */ |
| 3, /* MCA_OP_MDL_ABORT_REQ */ |
| 4, /* MCA_OP_MDL_ABORT_RSP */ |
| 3, /* MCA_OP_MDL_DELETE_REQ */ |
| 4 /* MCA_OP_MDL_DELETE_RSP */ |
| }; |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_handle_by_cpsm |
| ** |
| ** Description This function returns the handle for the given control |
| ** channel PSM. 0, if not found. |
| ** |
| ** Returns the MCA handle. |
| ** |
| *******************************************************************************/ |
| tMCA_HANDLE mca_handle_by_cpsm(UINT16 psm) |
| { |
| int i; |
| tMCA_HANDLE handle = 0; |
| tMCA_RCB *p_rcb = &mca_cb.rcb[0]; |
| |
| for (i=0; i<MCA_NUM_REGS; i++, p_rcb++) |
| { |
| if (p_rcb->p_cback && p_rcb->reg.ctrl_psm == psm) |
| { |
| handle = i+1; |
| break; |
| } |
| } |
| return handle; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_handle_by_dpsm |
| ** |
| ** Description This function returns the handle for the given data |
| ** channel PSM. 0, if not found. |
| ** |
| ** Returns the MCA handle. |
| ** |
| *******************************************************************************/ |
| tMCA_HANDLE mca_handle_by_dpsm(UINT16 psm) |
| { |
| int i; |
| tMCA_HANDLE handle = 0; |
| tMCA_RCB *p_rcb = &mca_cb.rcb[0]; |
| |
| for (i=0; i<MCA_NUM_REGS; i++, p_rcb++) |
| { |
| if (p_rcb->p_cback && p_rcb->reg.data_psm == psm) |
| { |
| handle = i+1; |
| break; |
| } |
| } |
| return handle; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_tbl_calloc |
| ** |
| ** Description This function allocates a transport table for the given |
| ** control channel. |
| ** |
| ** Returns The tranport table. |
| ** |
| *******************************************************************************/ |
| tMCA_TC_TBL * mca_tc_tbl_calloc(tMCA_CCB *p_ccb) |
| { |
| tMCA_TC_TBL *p_tbl = mca_cb.tc.tc_tbl; |
| int i; |
| |
| /* find next free entry in tc table */ |
| for (i = 0; i < MCA_NUM_TC_TBL; i++, p_tbl++) |
| { |
| if (p_tbl->state == MCA_TC_ST_UNUSED) |
| { |
| break; |
| } |
| } |
| |
| /* sanity check */ |
| assert(i != MCA_NUM_TC_TBL); |
| |
| /* initialize entry */ |
| p_tbl->peer_mtu = L2CAP_DEFAULT_MTU; |
| p_tbl->cfg_flags= 0; |
| p_tbl->cb_idx = mca_ccb_to_hdl(p_ccb); |
| p_tbl->tcid = MCA_CTRL_TCID; |
| p_tbl->my_mtu = MCA_CTRL_MTU; |
| p_tbl->state = MCA_TC_ST_IDLE; |
| p_tbl->lcid = p_ccb->lcid; |
| mca_cb.tc.lcid_tbl[p_ccb->lcid - L2CAP_BASE_APPL_CID] = i; |
| |
| MCA_TRACE_DEBUG("%s() - cb_idx: %d", __func__, p_tbl->cb_idx); |
| return p_tbl; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_tbl_dalloc |
| ** |
| ** Description This function allocates a transport table for the given |
| ** data channel. |
| ** |
| ** Returns The tranport table. |
| ** |
| *******************************************************************************/ |
| tMCA_TC_TBL * mca_tc_tbl_dalloc(tMCA_DCB *p_dcb) |
| { |
| tMCA_TC_TBL *p_tbl = mca_cb.tc.tc_tbl; |
| int i; |
| |
| /* find next free entry in tc table */ |
| for (i = 0; i < MCA_NUM_TC_TBL; i++, p_tbl++) |
| { |
| if (p_tbl->state == MCA_TC_ST_UNUSED) |
| { |
| break; |
| } |
| } |
| |
| /* sanity check */ |
| assert(i != MCA_NUM_TC_TBL); |
| |
| /* initialize entry */ |
| p_tbl->peer_mtu = L2CAP_DEFAULT_MTU; |
| p_tbl->cfg_flags= 0; |
| p_tbl->cb_idx = mca_dcb_to_hdl(p_dcb); |
| p_tbl->tcid = p_dcb->p_cs->type + 1; |
| p_tbl->my_mtu = p_dcb->p_chnl_cfg->data_mtu; |
| p_tbl->state = MCA_TC_ST_IDLE; |
| p_tbl->lcid = p_dcb->lcid; |
| mca_cb.tc.lcid_tbl[p_dcb->lcid - L2CAP_BASE_APPL_CID] = i; |
| |
| MCA_TRACE_DEBUG("%s() - tcid: %d, cb_idx: %d", __func__, p_tbl->tcid, p_tbl->cb_idx); |
| return p_tbl; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_tbl_by_lcid |
| ** |
| ** Description Find the transport channel table entry by LCID. |
| ** |
| ** |
| ** Returns The tranport table. |
| ** |
| *******************************************************************************/ |
| tMCA_TC_TBL *mca_tc_tbl_by_lcid(UINT16 lcid) |
| { |
| UINT8 idx; |
| |
| if (lcid) |
| { |
| idx = mca_cb.tc.lcid_tbl[lcid - L2CAP_BASE_APPL_CID]; |
| |
| if (idx < MCA_NUM_TC_TBL) |
| { |
| return &mca_cb.tc.tc_tbl[idx]; |
| } |
| } |
| return NULL; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_free_tc_tbl_by_lcid |
| ** |
| ** Description Find the transport table entry by LCID |
| ** and free the tc_tbl |
| ** |
| ** Returns void. |
| ** |
| *******************************************************************************/ |
| void mca_free_tc_tbl_by_lcid(UINT16 lcid) |
| { |
| UINT8 idx; |
| |
| if (lcid) |
| { |
| idx = mca_cb.tc.lcid_tbl[lcid - L2CAP_BASE_APPL_CID]; |
| |
| if (idx < MCA_NUM_TC_TBL) |
| { |
| mca_cb.tc.tc_tbl[idx].state = MCA_TC_ST_UNUSED; |
| } |
| } |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_set_cfg_by_tbl |
| ** |
| ** Description Set the L2CAP configuration information |
| ** |
| ** Returns none. |
| ** |
| *******************************************************************************/ |
| void mca_set_cfg_by_tbl(tL2CAP_CFG_INFO *p_cfg, tMCA_TC_TBL *p_tbl) |
| { |
| tMCA_DCB *p_dcb; |
| const tL2CAP_FCR_OPTS *p_opt; |
| tMCA_FCS_OPT fcs = MCA_FCS_NONE; |
| |
| if (p_tbl->tcid == MCA_CTRL_TCID) |
| { |
| p_opt = &mca_l2c_fcr_opts_def; |
| } |
| else |
| { |
| p_dcb = mca_dcb_by_hdl(p_tbl->cb_idx); |
| if (p_dcb) |
| { |
| p_opt = &p_dcb->p_chnl_cfg->fcr_opt; |
| fcs = p_dcb->p_chnl_cfg->fcs; |
| } |
| } |
| memset(p_cfg, 0, sizeof(tL2CAP_CFG_INFO)); |
| p_cfg->mtu_present = TRUE; |
| p_cfg->mtu = p_tbl->my_mtu; |
| p_cfg->fcr_present = TRUE; |
| memcpy(&p_cfg->fcr, p_opt, sizeof (tL2CAP_FCR_OPTS)); |
| if (fcs & MCA_FCS_PRESNT_MASK) |
| { |
| p_cfg->fcs_present = TRUE; |
| p_cfg->fcs = (fcs & MCA_FCS_USE_MASK); |
| } |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_close_ind |
| ** |
| ** Description This function is called by the L2CAP interface when the |
| ** L2CAP channel is closed. It looks up the CCB or DCB for |
| ** the channel and sends it a close event. The reason |
| ** parameter is the same value passed by the L2CAP |
| ** callback function. |
| ** |
| ** Returns Nothing. |
| ** |
| *******************************************************************************/ |
| void mca_tc_close_ind(tMCA_TC_TBL *p_tbl, UINT16 reason) |
| { |
| tMCA_CCB *p_ccb; |
| tMCA_DCB *p_dcb; |
| tMCA_CLOSE close; |
| |
| close.param = MCA_ACP; |
| close.reason = reason; |
| close.lcid = p_tbl->lcid; |
| |
| MCA_TRACE_DEBUG("%s() - tcid: %d, cb_idx:%d, old: %d", __func__, |
| p_tbl->tcid, p_tbl->cb_idx, p_tbl->state); |
| |
| /* Check if the transport channel is in use */ |
| if (p_tbl->state == MCA_TC_ST_UNUSED) |
| return; |
| |
| /* clear mca_tc_tbl entry */ |
| if (p_tbl->cfg_flags&MCA_L2C_CFG_DISCN_INT) |
| close.param = MCA_INT; |
| p_tbl->cfg_flags = 0; |
| p_tbl->peer_mtu = L2CAP_DEFAULT_MTU; |
| |
| /* if control channel, notify ccb that channel close */ |
| if (p_tbl->tcid == MCA_CTRL_TCID) |
| { |
| p_ccb = mca_ccb_by_hdl((tMCA_CL)p_tbl->cb_idx); |
| mca_ccb_event(p_ccb, MCA_CCB_LL_CLOSE_EVT, (tMCA_CCB_EVT *)&close); |
| } |
| /* notify dcb that channel close */ |
| else |
| { |
| /* look up dcb */ |
| p_dcb = mca_dcb_by_hdl(p_tbl->cb_idx); |
| if (p_dcb != NULL) |
| { |
| mca_dcb_event(p_dcb, MCA_DCB_TC_CLOSE_EVT, (tMCA_DCB_EVT *) &close); |
| } |
| } |
| p_tbl->state = MCA_TC_ST_UNUSED; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_open_ind |
| ** |
| ** Description This function is called by the L2CAP interface when |
| ** the L2CAP channel is opened. It looks up the CCB or DCB |
| ** for the channel and sends it an open event. |
| ** |
| ** Returns Nothing. |
| ** |
| *******************************************************************************/ |
| void mca_tc_open_ind(tMCA_TC_TBL *p_tbl) |
| { |
| tMCA_CCB *p_ccb; |
| tMCA_DCB *p_dcb; |
| tMCA_OPEN open; |
| |
| MCA_TRACE_DEBUG("mca_tc_open_ind tcid: %d, cb_idx: %d", p_tbl->tcid, p_tbl->cb_idx); |
| p_tbl->state = MCA_TC_ST_OPEN; |
| |
| open.peer_mtu = p_tbl->peer_mtu; |
| open.lcid = p_tbl->lcid; |
| /* use param to indicate the role of connection. |
| * MCA_ACP, if ACP */ |
| open.param = MCA_INT; |
| if (p_tbl->cfg_flags & MCA_L2C_CFG_CONN_ACP) |
| { |
| open.param = MCA_ACP; |
| } |
| |
| /* if control channel, notify ccb that channel open */ |
| if (p_tbl->tcid == MCA_CTRL_TCID) |
| { |
| p_ccb = mca_ccb_by_hdl((tMCA_CL)p_tbl->cb_idx); |
| |
| mca_ccb_event(p_ccb, MCA_CCB_LL_OPEN_EVT, (tMCA_CCB_EVT *)&open); |
| } |
| /* must be data channel, notify dcb that channel open */ |
| else |
| { |
| /* look up dcb */ |
| p_dcb = mca_dcb_by_hdl(p_tbl->cb_idx); |
| |
| /* put lcid in event data */ |
| if (p_dcb != NULL) |
| { |
| mca_dcb_event(p_dcb, MCA_DCB_TC_OPEN_EVT, (tMCA_DCB_EVT *) &open); |
| } |
| } |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_cong_ind |
| ** |
| ** Description This function is called by the L2CAP interface layer when |
| ** L2CAP calls the congestion callback. It looks up the CCB |
| ** or DCB for the channel and sends it a congestion event. |
| ** The is_congested parameter is the same value passed by |
| ** the L2CAP callback function. |
| ** |
| ** |
| ** Returns Nothing. |
| ** |
| *******************************************************************************/ |
| void mca_tc_cong_ind(tMCA_TC_TBL *p_tbl, BOOLEAN is_congested) |
| { |
| tMCA_CCB *p_ccb; |
| tMCA_DCB *p_dcb; |
| |
| MCA_TRACE_DEBUG("%s() - tcid: %d, cb_idx: %d", __func__, p_tbl->tcid, p_tbl->cb_idx); |
| |
| /* if control channel, notify ccb of congestion */ |
| if (p_tbl->tcid == MCA_CTRL_TCID) |
| { |
| p_ccb = mca_ccb_by_hdl((tMCA_CL)p_tbl->cb_idx); |
| mca_ccb_event(p_ccb, MCA_CCB_LL_CONG_EVT, (tMCA_CCB_EVT *) &is_congested); |
| } |
| /* notify dcb that channel open */ |
| else |
| { |
| /* look up dcb by cb_idx */ |
| p_dcb = mca_dcb_by_hdl(p_tbl->cb_idx); |
| if (p_dcb != NULL) |
| { |
| mca_dcb_event(p_dcb, MCA_DCB_TC_CONG_EVT, (tMCA_DCB_EVT *) &is_congested); |
| } |
| } |
| } |
| |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_tc_data_ind |
| ** |
| ** Description This function is called by the L2CAP interface layer when |
| ** incoming data is received from L2CAP. It looks up the CCB |
| ** or DCB for the channel and routes the data accordingly. |
| ** |
| ** Returns Nothing. |
| ** |
| *******************************************************************************/ |
| void mca_tc_data_ind(tMCA_TC_TBL *p_tbl, BT_HDR *p_buf) |
| { |
| tMCA_CCB *p_ccb; |
| tMCA_DCB *p_dcb; |
| UINT8 event = MCA_CCB_MSG_RSP_EVT; |
| UINT8 *p; |
| UINT8 rej_rsp_code = MCA_RSP_SUCCESS; |
| |
| MCA_TRACE_DEBUG("%s() - tcid: %d, cb_idx: %d", __func__, p_tbl->tcid, p_tbl->cb_idx); |
| |
| /* if control channel, handle control message */ |
| if (p_tbl->tcid == MCA_CTRL_TCID) |
| { |
| p_ccb = mca_ccb_by_hdl((tMCA_CL)p_tbl->cb_idx); |
| if (p_ccb) |
| { |
| p = (UINT8*)(p_buf+1) + p_buf->offset; |
| /* all the request opcode has bit 0 set. response code has bit 0 clear */ |
| if ((*p) & 0x01) |
| event = MCA_CCB_MSG_REQ_EVT; |
| |
| if (*p < MCA_NUM_STANDARD_OPCODE) |
| { |
| if (p_buf->len != mca_std_msg_len[*p]) |
| { |
| MCA_TRACE_ERROR ("$s() - opcode: %d required len: %d, got len: %d" |
| , __func__, *p, mca_std_msg_len[*p], p_buf->len); |
| rej_rsp_code = MCA_RSP_BAD_PARAM; |
| } |
| } |
| else if ((*p >= MCA_FIRST_SYNC_OP) && (*p <= MCA_LAST_SYNC_OP)) |
| { |
| MCA_TRACE_ERROR ("%s() - unsupported SYNC opcode: %d len:%d" |
| , __func__, *p, p_buf->len); |
| /* reject unsupported request */ |
| rej_rsp_code = MCA_RSP_NO_SUPPORT; |
| } |
| else |
| { |
| MCA_TRACE_ERROR ("%s() - bad opcode: %d len:%d", __func__, *p, p_buf->len); |
| /* reject unsupported request */ |
| rej_rsp_code = MCA_RSP_BAD_OPCODE; |
| } |
| |
| p_buf->layer_specific = rej_rsp_code; |
| /* forward the request/response to state machine */ |
| mca_ccb_event(p_ccb, event, (tMCA_CCB_EVT *) p_buf); |
| } /* got a valid ccb */ |
| else |
| osi_free(p_buf); |
| } |
| /* else send event to dcb */ |
| else |
| { |
| p_dcb = mca_dcb_by_hdl(p_tbl->cb_idx); |
| if (p_dcb != NULL) |
| { |
| mca_dcb_event(p_dcb, MCA_DCB_TC_DATA_EVT, (tMCA_DCB_EVT *) p_buf); |
| } |
| else |
| osi_free(p_buf); |
| } |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_rcb_alloc |
| ** |
| ** Description This function allocates a registration control block. |
| ** If no free RCB is available, it returns NULL. |
| ** |
| ** Returns tMCA_RCB * |
| ** |
| *******************************************************************************/ |
| tMCA_RCB * mca_rcb_alloc(tMCA_REG *p_reg) |
| { |
| int i; |
| tMCA_RCB *p_rcb = NULL; |
| |
| for (i=0; i<MCA_NUM_REGS; i++) |
| { |
| if (mca_cb.rcb[i].p_cback == NULL) |
| { |
| p_rcb = &mca_cb.rcb[i]; |
| memcpy (&p_rcb->reg, p_reg, sizeof(tMCA_REG)); |
| break; |
| } |
| } |
| return p_rcb; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_rcb_dealloc |
| ** |
| ** Description This function deallocates the RCB with the given handle. |
| ** |
| ** Returns void. |
| ** |
| *******************************************************************************/ |
| void mca_rcb_dealloc(tMCA_HANDLE handle) |
| { |
| int i; |
| BOOLEAN done = TRUE; |
| tMCA_RCB *p_rcb; |
| tMCA_CCB *p_ccb; |
| |
| if (handle && (handle<=MCA_NUM_REGS)) |
| { |
| handle--; |
| p_rcb = &mca_cb.rcb[handle]; |
| if (p_rcb->p_cback) |
| { |
| p_ccb = &mca_cb.ccb[handle*MCA_NUM_LINKS]; |
| /* check if all associated CCB are disconnected */ |
| for (i=0; i<MCA_NUM_LINKS; i++, p_ccb++) |
| { |
| if (p_ccb->p_rcb) |
| { |
| done = FALSE; |
| mca_ccb_event (p_ccb, MCA_CCB_API_DISCONNECT_EVT, NULL); |
| } |
| } |
| |
| if (done) |
| { |
| memset (p_rcb, 0, sizeof(tMCA_RCB)); |
| MCA_TRACE_DEBUG("%s() - reset MCA_RCB index=%d", __func__, handle); |
| } |
| } |
| } |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_rcb_to_handle |
| ** |
| ** Description This function converts a pointer to an RCB to |
| ** a handle (tMCA_HANDLE). It returns the handle. |
| ** |
| ** Returns void. |
| ** |
| *******************************************************************************/ |
| tMCA_HANDLE mca_rcb_to_handle(tMCA_RCB *p_rcb) |
| { |
| return(UINT8) (p_rcb - mca_cb.rcb + 1); |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_rcb_by_handle |
| ** |
| ** Description This function finds the RCB for a handle (tMCA_HANDLE). |
| ** It returns a pointer to the RCB. If no RCB matches the |
| ** handle it returns NULL. |
| ** |
| ** Returns tMCA_RCB * |
| ** |
| *******************************************************************************/ |
| tMCA_RCB *mca_rcb_by_handle(tMCA_HANDLE handle) |
| { |
| tMCA_RCB *p_rcb = NULL; |
| |
| if (handle && (handle<=MCA_NUM_REGS) && mca_cb.rcb[handle-1].p_cback) |
| { |
| p_rcb = &mca_cb.rcb[handle-1]; |
| } |
| return p_rcb; |
| } |
| |
| /******************************************************************************* |
| ** |
| ** Function mca_is_valid_dep_id |
| ** |
| ** Description This function checks if the given dep_id is valid. |
| ** |
| ** Returns TRUE, if this is a valid local dep_id |
| ** |
| *******************************************************************************/ |
| BOOLEAN mca_is_valid_dep_id(tMCA_RCB *p_rcb, tMCA_DEP dep) |
| { |
| BOOLEAN valid = FALSE; |
| if (dep < MCA_NUM_DEPS && p_rcb->dep[dep].p_data_cback) |
| { |
| valid = TRUE; |
| } |
| return valid; |
| } |