123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607 |
- /******************************************************************************
- *
- * Copyright 2002-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 module contains the AVDTP adaption layer.
- *
- ******************************************************************************/
- #include <base/logging.h>
- #include <string.h>
- #include "avdt_api.h"
- #include "avdt_int.h"
- #include "avdtc_api.h"
- #include "bt_target.h"
- #include "bt_types.h"
- #include "bt_utils.h"
- #include "l2c_api.h"
- #include "l2cdefs.h"
- #include "osi/include/osi.h"
- AvdtpScb* AvdtpAdaptationLayer::LookupAvdtpScb(
- const AvdtpTransportChannel& tc) {
- if (tc.ccb_idx >= AVDT_NUM_LINKS) {
- AVDT_TRACE_ERROR("%s: AvdtpScb entry not found: invalid ccb_idx:%d",
- __func__, tc.ccb_idx);
- return nullptr;
- }
- if (tc.tcid >= AVDT_NUM_RT_TBL) {
- AVDT_TRACE_ERROR("%s: AvdtpScb entry not found: invalid tcid:%d", __func__,
- tc.tcid);
- return nullptr;
- }
- const AvdtpRoutingEntry& re = rt_tbl[tc.ccb_idx][tc.tcid];
- AVDT_TRACE_DEBUG("%s: ccb_idx:%d tcid:%d scb_hdl:%d", __func__, tc.ccb_idx,
- tc.tcid, re.scb_hdl);
- return avdt_scb_by_hdl(re.scb_hdl);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_type_to_tcid
- *
- * Description Derives the TCID from the channel type and SCB.
- *
- *
- * Returns TCID value.
- *
- ******************************************************************************/
- uint8_t avdt_ad_type_to_tcid(uint8_t type, AvdtpScb* p_scb) {
- if (type == AVDT_CHAN_SIG) {
- return 0;
- }
- // The SCB Handle is unique in the [1, AVDT_NUM_LINKS * AVDT_NUM_SEPS]
- // range. The scb_idx computed here is the SCB index for the corresponding
- // SEP, and it is in the range [0, AVDT_NUM_SEPS) for a particular link.
- uint8_t scb_idx = (avdt_scb_to_hdl(p_scb) - 1) % AVDT_NUM_LINKS;
- // There are AVDT_CHAN_NUM_TYPES channel types per SEP. Here we compute
- // the type index (TCID) from the SEP index and the type itself.
- uint8_t tcid = (scb_idx * (AVDT_CHAN_NUM_TYPES - 1)) + type;
- AVDT_TRACE_DEBUG("%s: type:%d, tcid: %d", __func__, type, tcid);
- return tcid;
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tcid_to_type
- *
- * Description Derives the channel type from the TCID.
- *
- *
- * Returns Channel type value.
- *
- ******************************************************************************/
- static uint8_t avdt_ad_tcid_to_type(uint8_t tcid) {
- uint8_t type;
- if (tcid == 0) {
- type = AVDT_CHAN_SIG;
- } else {
- /* tcid translates to type based on number of channels, as follows:
- ** only media channel : tcid=1,2,3,4,5,6... type=1,1,1,1,1,1...
- ** media and report : tcid=1,2,3,4,5,6... type=1,2,1,2,1,2...
- ** media, report, recov : tcid=1,2,3,4,5,6... type=1,2,3,1,2,3...
- */
- type = ((tcid + AVDT_CHAN_NUM_TYPES - 2) % (AVDT_CHAN_NUM_TYPES - 1)) + 1;
- }
- AVDT_TRACE_DEBUG("tcid: %d, type: %d", tcid, type);
- return type;
- }
- /*******************************************************************************
- *
- * Function avdt_ad_init
- *
- * Description Initialize adaption layer.
- *
- *
- * Returns Nothing.
- *
- ******************************************************************************/
- void avdt_ad_init(void) {
- int i;
- AvdtpTransportChannel* p_tbl = avdtp_cb.ad.tc_tbl;
- avdtp_cb.ad.Reset();
- /* make sure the peer_mtu is a valid value */
- for (i = 0; i < AVDT_NUM_TC_TBL; i++, p_tbl++) {
- p_tbl->peer_mtu = L2CAP_DEFAULT_MTU;
- }
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_tbl_by_st
- *
- * Description Find adaption layer transport channel table entry matching
- * the given state.
- *
- *
- * Returns Pointer to matching entry. For control channel it returns
- * the matching entry. For media or other it returns the
- * first matching entry (there could be more than one).
- *
- ******************************************************************************/
- AvdtpTransportChannel* avdt_ad_tc_tbl_by_st(uint8_t type, AvdtpCcb* p_ccb,
- uint8_t state) {
- int i;
- AvdtpTransportChannel* p_tbl = avdtp_cb.ad.tc_tbl;
- uint8_t ccb_idx;
- if (p_ccb == NULL) {
- /* resending security req */
- for (i = 0; i < AVDT_NUM_TC_TBL; i++, p_tbl++) {
- /* must be AVDT_CHAN_SIG - tcid always zero */
- if ((p_tbl->tcid == 0) && (p_tbl->state == state)) {
- break;
- }
- }
- } else {
- ccb_idx = avdt_ccb_to_idx(p_ccb);
- for (i = 0; i < AVDT_NUM_TC_TBL; i++, p_tbl++) {
- if (type == AVDT_CHAN_SIG) {
- /* if control channel, tcid always zero */
- if ((p_tbl->tcid == 0) && (p_tbl->ccb_idx == ccb_idx) &&
- (p_tbl->state == state)) {
- break;
- }
- } else {
- /* if other channel, tcid is always > zero */
- if ((p_tbl->tcid > 0) && (p_tbl->ccb_idx == ccb_idx) &&
- (p_tbl->state == state)) {
- break;
- }
- }
- }
- }
- /* if nothing found return null */
- if (i == AVDT_NUM_TC_TBL) {
- p_tbl = NULL;
- }
- return p_tbl;
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_tbl_by_lcid
- *
- * Description Find adaption layer transport channel table entry by LCID.
- *
- *
- * Returns Pointer to entry.
- *
- ******************************************************************************/
- AvdtpTransportChannel* avdt_ad_tc_tbl_by_lcid(uint16_t lcid) {
- uint8_t idx;
- idx = avdtp_cb.ad.lcid_tbl[lcid - L2CAP_BASE_APPL_CID];
- if (idx < AVDT_NUM_TC_TBL) {
- return &avdtp_cb.ad.tc_tbl[idx];
- } else {
- return NULL;
- }
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_tbl_by_type
- *
- * Description This function retrieves the transport channel table entry
- * for a particular channel.
- *
- *
- * Returns Pointer to transport channel table entry.
- *
- ******************************************************************************/
- AvdtpTransportChannel* avdt_ad_tc_tbl_by_type(uint8_t type, AvdtpCcb* p_ccb,
- AvdtpScb* p_scb) {
- uint8_t tcid;
- int i;
- AvdtpTransportChannel* p_tbl = avdtp_cb.ad.tc_tbl;
- uint8_t ccb_idx = avdt_ccb_to_idx(p_ccb);
- /* get tcid from type, scb */
- tcid = avdt_ad_type_to_tcid(type, p_scb);
- for (i = 0; i < AVDT_NUM_TC_TBL; i++, p_tbl++) {
- if ((p_tbl->tcid == tcid) && (p_tbl->ccb_idx == ccb_idx)) {
- break;
- }
- }
- CHECK(i != AVDT_NUM_TC_TBL);
- return p_tbl;
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_tbl_alloc
- *
- * Description Allocate an entry in the traffic channel table.
- *
- *
- * Returns Pointer to entry.
- *
- ******************************************************************************/
- AvdtpTransportChannel* avdt_ad_tc_tbl_alloc(AvdtpCcb* p_ccb) {
- int i;
- AvdtpTransportChannel* p_tbl = avdtp_cb.ad.tc_tbl;
- /* find next free entry in tc table */
- for (i = 0; i < AVDT_NUM_TC_TBL; i++, p_tbl++) {
- if (p_tbl->state == AVDT_AD_ST_UNUSED) {
- break;
- }
- }
- /* sanity check */
- CHECK(i != AVDT_NUM_TC_TBL);
- /* initialize entry */
- p_tbl->peer_mtu = L2CAP_DEFAULT_MTU;
- p_tbl->cfg_flags = 0;
- p_tbl->ccb_idx = avdt_ccb_to_idx(p_ccb);
- p_tbl->state = AVDT_AD_ST_IDLE;
- return p_tbl;
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_tbl_to_idx
- *
- * Description Convert a transport channel table entry to an index.
- *
- *
- * Returns Index value.
- *
- ******************************************************************************/
- uint8_t avdt_ad_tc_tbl_to_idx(AvdtpTransportChannel* p_tbl) {
- AVDT_TRACE_DEBUG("avdt_ad_tc_tbl_to_idx: %d", (p_tbl - avdtp_cb.ad.tc_tbl));
- /* use array arithmetic to determine index */
- return (uint8_t)(p_tbl - avdtp_cb.ad.tc_tbl);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_close_ind
- *
- * Description This function is called by the L2CAP interface when the
- * L2CAP channel is closed. It looks up the CCB or SCB 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 avdt_ad_tc_close_ind(AvdtpTransportChannel* p_tbl,
- UNUSED_ATTR uint16_t reason) {
- AvdtpCcb* p_ccb;
- AvdtpScb* p_scb;
- tAVDT_SCB_TC_CLOSE close;
- close.old_tc_state = p_tbl->state;
- /* clear avdt_ad_tc_tbl entry */
- p_tbl->state = AVDT_AD_ST_UNUSED;
- p_tbl->cfg_flags = 0;
- p_tbl->peer_mtu = L2CAP_DEFAULT_MTU;
- AVDT_TRACE_DEBUG("%s: tcid: %d, old: %d", __func__, p_tbl->tcid,
- close.old_tc_state);
- /* if signaling channel, notify ccb that channel open */
- if (p_tbl->tcid == 0) {
- p_ccb = avdt_ccb_by_idx(p_tbl->ccb_idx);
- avdt_ccb_event(p_ccb, AVDT_CCB_LL_CLOSE_EVT, NULL);
- return;
- }
- /* if media or other channel, notify scb that channel close */
- /* look up scb in stream routing table by ccb, tcid */
- p_scb = avdtp_cb.ad.LookupAvdtpScb(*p_tbl);
- if (p_scb == nullptr) {
- AVDT_TRACE_ERROR("%s: Cannot find AvdtScb entry: ccb_idx:%d tcid:%d",
- __func__, p_tbl->ccb_idx, p_tbl->tcid);
- return;
- }
- close.tcid = p_tbl->tcid;
- close.type = avdt_ad_tcid_to_type(p_tbl->tcid);
- tAVDT_SCB_EVT avdt_scb_evt;
- avdt_scb_evt.close = close;
- avdt_scb_event(p_scb, AVDT_SCB_TC_CLOSE_EVT, &avdt_scb_evt);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_tc_open_ind
- *
- * Description This function is called by the L2CAP interface when
- * the L2CAP channel is opened. It looks up the CCB or SCB
- * for the channel and sends it an open event.
- *
- *
- * Returns Nothing.
- *
- ******************************************************************************/
- void avdt_ad_tc_open_ind(AvdtpTransportChannel* p_tbl) {
- AvdtpCcb* p_ccb;
- AvdtpScb* p_scb;
- tAVDT_OPEN open;
- tAVDT_EVT_HDR evt;
- AVDT_TRACE_DEBUG("%s: p_tbl:%p state:%d ccb_idx:%d tcid:%d scb_hdl:%d",
- __func__, p_tbl, p_tbl->state, p_tbl->ccb_idx, p_tbl->tcid,
- avdtp_cb.ad.rt_tbl[p_tbl->ccb_idx][p_tbl->tcid].scb_hdl);
- p_tbl->state = AVDT_AD_ST_OPEN;
- /* if signaling channel, notify ccb that channel open */
- if (p_tbl->tcid == 0) {
- /* set the signal channel to use high priority within the ACL link */
- L2CA_SetTxPriority(avdtp_cb.ad.rt_tbl[p_tbl->ccb_idx][AVDT_CHAN_SIG].lcid,
- L2CAP_CHNL_PRIORITY_HIGH);
- p_ccb = avdt_ccb_by_idx(p_tbl->ccb_idx);
- /* use err_param to indicate the role of connection.
- * AVDT_ACP, if ACP */
- evt.err_param = AVDT_INT;
- if (p_tbl->cfg_flags & AVDT_L2C_CFG_CONN_ACP) {
- evt.err_param = AVDT_ACP;
- }
- tAVDT_CCB_EVT avdt_ccb_evt;
- avdt_ccb_evt.msg.hdr = evt;
- avdt_ccb_event(p_ccb, AVDT_CCB_LL_OPEN_EVT, &avdt_ccb_evt);
- return;
- }
- /* if media or other channel, notify scb that channel open */
- /* look up scb in stream routing table by ccb, tcid */
- p_scb = avdtp_cb.ad.LookupAvdtpScb(*p_tbl);
- if (p_scb == nullptr) {
- AVDT_TRACE_ERROR("%s: Cannot find AvdtScb entry: ccb_idx:%d tcid:%d",
- __func__, p_tbl->ccb_idx, p_tbl->tcid);
- return;
- }
- /* put lcid in event data */
- open.peer_mtu = p_tbl->peer_mtu;
- open.lcid = avdtp_cb.ad.rt_tbl[p_tbl->ccb_idx][p_tbl->tcid].lcid;
- open.hdr.err_code = avdt_ad_tcid_to_type(p_tbl->tcid);
- tAVDT_SCB_EVT avdt_scb_evt;
- avdt_scb_evt.open = open;
- avdt_scb_event(p_scb, AVDT_SCB_TC_OPEN_EVT, &avdt_scb_evt);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_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 SCB 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 avdt_ad_tc_cong_ind(AvdtpTransportChannel* p_tbl, bool is_congested) {
- AvdtpCcb* p_ccb;
- AvdtpScb* p_scb;
- /* if signaling channel, notify ccb of congestion */
- if (p_tbl->tcid == 0) {
- p_ccb = avdt_ccb_by_idx(p_tbl->ccb_idx);
- tAVDT_CCB_EVT avdt_ccb_evt;
- avdt_ccb_evt.llcong = is_congested;
- avdt_ccb_event(p_ccb, AVDT_CCB_LL_CONG_EVT, &avdt_ccb_evt);
- return;
- }
- /* if media or other channel, notify scb that channel open */
- /* look up scb in stream routing table by ccb, tcid */
- p_scb = avdtp_cb.ad.LookupAvdtpScb(*p_tbl);
- if (p_scb == nullptr) {
- AVDT_TRACE_ERROR("%s: Cannot find AvdtScb entry: ccb_idx:%d tcid:%d",
- __func__, p_tbl->ccb_idx, p_tbl->tcid);
- return;
- }
- tAVDT_SCB_EVT avdt_scb_evt;
- avdt_scb_evt.llcong = is_congested;
- avdt_scb_event(p_scb, AVDT_SCB_TC_CONG_EVT, &avdt_scb_evt);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_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 SCB for the channel and routes the data accordingly.
- *
- *
- * Returns Nothing.
- *
- ******************************************************************************/
- void avdt_ad_tc_data_ind(AvdtpTransportChannel* p_tbl, BT_HDR* p_buf) {
- AvdtpCcb* p_ccb;
- AvdtpScb* p_scb;
- /* store type (media, recovery, reporting) */
- p_buf->layer_specific = avdt_ad_tcid_to_type(p_tbl->tcid);
- /* if signaling channel, handle control message */
- if (p_tbl->tcid == 0) {
- p_ccb = avdt_ccb_by_idx(p_tbl->ccb_idx);
- avdt_msg_ind(p_ccb, p_buf);
- return;
- }
- /* if media or other channel, send event to scb */
- p_scb = avdtp_cb.ad.LookupAvdtpScb(*p_tbl);
- if (p_scb == nullptr) {
- AVDT_TRACE_ERROR("%s: Cannot find AvdtScb entry: ccb_idx:%d tcid:%d",
- __func__, p_tbl->ccb_idx, p_tbl->tcid);
- osi_free(p_buf);
- AVDT_TRACE_ERROR("%s: buffer freed", __func__);
- return;
- }
- avdt_scb_event(p_scb, AVDT_SCB_TC_DATA_EVT, (tAVDT_SCB_EVT*)&p_buf);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_write_req
- *
- * Description This function is called by a CCB or SCB to send data to a
- * transport channel. It looks up the LCID of the channel
- * based on the type, CCB, and SCB (if present). Then it
- * passes the data to L2CA_DataWrite().
- *
- *
- * Returns AVDT_AD_SUCCESS, if data accepted
- * AVDT_AD_CONGESTED, if data accepted and the channel is
- * congested
- * AVDT_AD_FAILED, if error
- *
- ******************************************************************************/
- uint8_t avdt_ad_write_req(uint8_t type, AvdtpCcb* p_ccb, AvdtpScb* p_scb,
- BT_HDR* p_buf) {
- uint8_t tcid;
- /* get tcid from type, scb */
- tcid = avdt_ad_type_to_tcid(type, p_scb);
- return L2CA_DataWrite(avdtp_cb.ad.rt_tbl[avdt_ccb_to_idx(p_ccb)][tcid].lcid,
- p_buf);
- }
- /*******************************************************************************
- *
- * Function avdt_ad_open_req
- *
- * Description This function is called by a CCB or SCB to open a transport
- * channel. This function allocates and initializes a
- * transport channel table entry. The channel can be opened
- * in two roles: as an initiator or acceptor. When opened
- * as an initiator the function will start an L2CAP connection.
- * When opened as an acceptor the function simply configures
- * the table entry to listen for an incoming channel.
- *
- *
- * Returns Nothing.
- *
- ******************************************************************************/
- void avdt_ad_open_req(uint8_t type, AvdtpCcb* p_ccb, AvdtpScb* p_scb,
- uint8_t role) {
- AvdtpTransportChannel* p_tbl;
- uint16_t lcid;
- p_tbl = avdt_ad_tc_tbl_alloc(p_ccb);
- if (p_tbl == NULL) {
- AVDT_TRACE_ERROR("avdt_ad_open_req: Cannot allocate p_tbl");
- return;
- }
- p_tbl->tcid = avdt_ad_type_to_tcid(type, p_scb);
- AVDT_TRACE_DEBUG("avdt_ad_open_req: type: %d, role: %d, tcid:%d", type, role,
- p_tbl->tcid);
- if (type == AVDT_CHAN_SIG) {
- /* if signaling, get mtu from registration control block */
- p_tbl->my_mtu = avdtp_cb.rcb.ctrl_mtu;
- p_tbl->my_flush_to = L2CAP_DEFAULT_FLUSH_TO;
- } else {
- /* otherwise get mtu from scb */
- p_tbl->my_mtu = p_scb->stream_config.mtu;
- p_tbl->my_flush_to = p_scb->stream_config.flush_to;
- /* also set scb_hdl in rt_tbl */
- avdtp_cb.ad.rt_tbl[avdt_ccb_to_idx(p_ccb)][p_tbl->tcid].scb_hdl =
- avdt_scb_to_hdl(p_scb);
- AVDT_TRACE_DEBUG("avdtp_cb.ad.rt_tbl[%d][%d].scb_hdl = %d",
- avdt_ccb_to_idx(p_ccb), p_tbl->tcid,
- avdt_scb_to_hdl(p_scb));
- }
- /* if we're acceptor, we're done; just sit back and listen */
- if (role == AVDT_ACP) {
- p_tbl->state = AVDT_AD_ST_ACP;
- }
- /* else we're inititator, start the L2CAP connection */
- else {
- p_tbl->state = AVDT_AD_ST_CONN;
- /* call l2cap connect req */
- lcid = L2CA_ConnectReq(AVDT_PSM, p_ccb->peer_addr);
- if (lcid != 0) {
- /* if connect req ok, store tcid in lcid table */
- avdtp_cb.ad.lcid_tbl[lcid - L2CAP_BASE_APPL_CID] =
- avdt_ad_tc_tbl_to_idx(p_tbl);
- AVDT_TRACE_DEBUG("avdtp_cb.ad.lcid_tbl[%d] = %d",
- (lcid - L2CAP_BASE_APPL_CID),
- avdt_ad_tc_tbl_to_idx(p_tbl));
- avdtp_cb.ad.rt_tbl[avdt_ccb_to_idx(p_ccb)][p_tbl->tcid].lcid = lcid;
- AVDT_TRACE_DEBUG("avdtp_cb.ad.rt_tbl[%d][%d].lcid = 0x%x",
- avdt_ccb_to_idx(p_ccb), p_tbl->tcid, lcid);
- } else {
- /* if connect req failed, call avdt_ad_tc_close_ind() */
- avdt_ad_tc_close_ind(p_tbl, 0);
- }
- }
- }
- /*******************************************************************************
- *
- * Function avdt_ad_close_req
- *
- * Description This function is called by a CCB or SCB to close a
- * transport channel. The function looks up the LCID for the
- * channel and calls L2CA_DisconnectReq().
- *
- *
- * Returns Nothing.
- *
- ******************************************************************************/
- void avdt_ad_close_req(uint8_t type, AvdtpCcb* p_ccb, AvdtpScb* p_scb) {
- uint8_t tcid;
- AvdtpTransportChannel* p_tbl;
- p_tbl = avdt_ad_tc_tbl_by_type(type, p_ccb, p_scb);
- AVDT_TRACE_DEBUG("avdt_ad_close_req state: %d", p_tbl->state);
- switch (p_tbl->state) {
- case AVDT_AD_ST_UNUSED:
- /* probably for reporting */
- break;
- case AVDT_AD_ST_ACP:
- /* if we're listening on this channel, send ourselves a close ind */
- avdt_ad_tc_close_ind(p_tbl, 0);
- break;
- default:
- /* get tcid from type, scb */
- tcid = avdt_ad_type_to_tcid(type, p_scb);
- /* call l2cap disconnect req */
- L2CA_DisconnectReq(avdtp_cb.ad.rt_tbl[avdt_ccb_to_idx(p_ccb)][tcid].lcid);
- }
- }
|