Home | History | Annotate | Download | only in avdt
      1 /******************************************************************************
      2  *
      3  *  Copyright 2002-2012 Broadcom Corporation
      4  *
      5  *  Licensed under the Apache License, Version 2.0 (the "License");
      6  *  you may not use this file except in compliance with the License.
      7  *  You may obtain a copy of the License at:
      8  *
      9  *  http://www.apache.org/licenses/LICENSE-2.0
     10  *
     11  *  Unless required by applicable law or agreed to in writing, software
     12  *  distributed under the License is distributed on an "AS IS" BASIS,
     13  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14  *  See the License for the specific language governing permissions and
     15  *  limitations under the License.
     16  *
     17  ******************************************************************************/
     18 
     19 /******************************************************************************
     20  *
     21  *  This module contains the channel control block state machine and
     22  *  functions which operate on the channel control block.
     23  *
     24  ******************************************************************************/
     25 
     26 #include <string.h>
     27 #include "avdt_api.h"
     28 #include "avdt_int.h"
     29 #include "avdtc_api.h"
     30 #include "bt_common.h"
     31 #include "bt_target.h"
     32 #include "bt_types.h"
     33 #include "bt_utils.h"
     34 #include "btu.h"
     35 #include "osi/include/osi.h"
     36 
     37 /*****************************************************************************
     38  * state machine constants and types
     39  ****************************************************************************/
     40 
     41 /* verbose state strings for trace */
     42 const char* const avdt_ccb_st_str[] = {"CCB_IDLE_ST", "CCB_OPENING_ST",
     43                                        "CCB_OPEN_ST", "CCB_CLOSING_ST"};
     44 
     45 /* verbose event strings for trace */
     46 const char* const avdt_ccb_evt_str[] = {
     47     "API_DISCOVER_REQ_EVT", "API_GETCAP_REQ_EVT",
     48     "API_START_REQ_EVT",    "API_SUSPEND_REQ_EVT",
     49     "API_DISCOVER_RSP_EVT", "API_GETCAP_RSP_EVT",
     50     "API_START_RSP_EVT",    "API_SUSPEND_RSP_EVT",
     51     "API_CONNECT_REQ_EVT",  "API_DISCONNECT_REQ_EVT",
     52     "MSG_DISCOVER_CMD_EVT", "MSG_GETCAP_CMD_EVT",
     53     "MSG_START_CMD_EVT",    "MSG_SUSPEND_CMD_EVT",
     54     "MSG_DISCOVER_RSP_EVT", "MSG_GETCAP_RSP_EVT",
     55     "MSG_START_RSP_EVT",    "MSG_SUSPEND_RSP_EVT",
     56     "RCVRSP_EVT",           "SENDMSG_EVT",
     57     "RET_TOUT_EVT",         "RSP_TOUT_EVT",
     58     "IDLE_TOUT_EVT",        "UL_OPEN_EVT",
     59     "UL_CLOSE_EVT",         "LL_OPEN_EVT",
     60     "LL_CLOSE_EVT",         "LL_CONG_EVT"};
     61 
     62 /* action function list */
     63 const tAVDT_CCB_ACTION avdt_ccb_action[] = {
     64     avdt_ccb_chan_open,        avdt_ccb_chan_close,
     65     avdt_ccb_chk_close,        avdt_ccb_hdl_discover_cmd,
     66     avdt_ccb_hdl_discover_rsp, avdt_ccb_hdl_getcap_cmd,
     67     avdt_ccb_hdl_getcap_rsp,   avdt_ccb_hdl_start_cmd,
     68     avdt_ccb_hdl_start_rsp,    avdt_ccb_hdl_suspend_cmd,
     69     avdt_ccb_hdl_suspend_rsp,  avdt_ccb_snd_discover_cmd,
     70     avdt_ccb_snd_discover_rsp, avdt_ccb_snd_getcap_cmd,
     71     avdt_ccb_snd_getcap_rsp,   avdt_ccb_snd_start_cmd,
     72     avdt_ccb_snd_start_rsp,    avdt_ccb_snd_suspend_cmd,
     73     avdt_ccb_snd_suspend_rsp,  avdt_ccb_clear_cmds,
     74     avdt_ccb_cmd_fail,         avdt_ccb_free_cmd,
     75     avdt_ccb_cong_state,       avdt_ccb_ret_cmd,
     76     avdt_ccb_snd_cmd,          avdt_ccb_snd_msg,
     77     avdt_ccb_set_reconn,       avdt_ccb_clr_reconn,
     78     avdt_ccb_chk_reconn,       avdt_ccb_chk_timer,
     79     avdt_ccb_set_conn,         avdt_ccb_set_disconn,
     80     avdt_ccb_do_disconn,       avdt_ccb_ll_closed,
     81     avdt_ccb_ll_opened,        avdt_ccb_dealloc};
     82 
     83 /* state table information */
     84 #define AVDT_CCB_ACTIONS 2    /* number of actions */
     85 #define AVDT_CCB_NEXT_STATE 2 /* position of next state */
     86 #define AVDT_CCB_NUM_COLS 3   /* number of columns in state tables */
     87 
     88 /* state table for idle state */
     89 const uint8_t avdt_ccb_st_idle[][AVDT_CCB_NUM_COLS] = {
     90     /* Event */
     91     /* Action 1                    Action 2                    Next state */
     92     /* API_DISCOVER_REQ_EVT */
     93     {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
     94     /* API_GETCAP_REQ_EVT */
     95     {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
     96     /* API_START_REQ_EVT */
     97     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
     98     /* API_SUSPEND_REQ_EVT */
     99     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    100     /* API_DISCOVER_RSP_EVT */
    101     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    102     /* API_GETCAP_RSP_EVT */
    103     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    104     /* API_START_RSP_EVT */
    105     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    106     /* API_SUSPEND_RSP_EVT */
    107     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    108     /* API_CONNECT_REQ_EVT */
    109     {AVDT_CCB_SET_CONN, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
    110     /* API_DISCONNECT_REQ_EVT */
    111     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    112     /* MSG_DISCOVER_CMD_EVT */
    113     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    114     /* MSG_GETCAP_CMD_EVT */
    115     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    116     /* MSG_START_CMD_EVT */
    117     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    118     /* MSG_SUSPEND_CMD_EVT */
    119     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    120     /* MSG_DISCOVER_RSP_EVT */
    121     {AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    122     /* MSG_GETCAP_RSP_EVT */
    123     {AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    124     /* MSG_START_RSP_EVT */
    125     {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    126     /* MSG_SUSPEND_RSP_EVT */
    127     {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    128     /* RCVRSP_EVT */
    129     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    130     /* SENDMSG_EVT */
    131     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    132     /* RET_TOUT_EVT */
    133     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    134     /* RSP_TOUT_EVT */
    135     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    136     /* IDLE_TOUT_EVT */
    137     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    138     /* UL_OPEN_EVT */
    139     {AVDT_CCB_CHAN_OPEN, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    140     /* UL_CLOSE_EVT */
    141     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    142     /* LL_OPEN_EVT */
    143     {AVDT_CCB_LL_OPENED, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    144     /* LL_CLOSE_EVT */
    145     {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    146     /* LL_CONG_EVT */
    147     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST}};
    148 
    149 /* state table for opening state */
    150 const uint8_t avdt_ccb_st_opening[][AVDT_CCB_NUM_COLS] = {
    151     /* Event */
    152     /* Action 1                    Action 2                    Next state */
    153     /* API_DISCOVER_REQ_EVT */
    154     {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    155     /* API_GETCAP_REQ_EVT */
    156     {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    157     /* API_START_REQ_EVT */
    158     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    159     /* API_SUSPEND_REQ_EVT */
    160     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    161     /* API_DISCOVER_RSP_EVT */
    162     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    163     /* API_GETCAP_RSP_EVT */
    164     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    165     /* API_START_RSP_EVT */
    166     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    167     /* API_SUSPEND_RSP_EVT */
    168     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    169     /* API_CONNECT_REQ_EVT */
    170     {AVDT_CCB_SET_CONN, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    171     /* API_DISCONNECT_REQ_EVT */
    172     {AVDT_CCB_SET_DISCONN, AVDT_CCB_DO_DISCONN, AVDT_CCB_CLOSING_ST},
    173     /* MSG_DISCOVER_CMD_EVT */
    174     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    175     /* MSG_GETCAP_CMD_EVT */
    176     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    177     /* MSG_START_CMD_EVT */
    178     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    179     /* MSG_SUSPEND_CMD_EVT */
    180     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    181     /* MSG_DISCOVER_RSP_EVT */
    182     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    183     /* MSG_GETCAP_RSP_EVT */
    184     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    185     /* MSG_START_RSP_EVT */
    186     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    187     /* MSG_SUSPEND_RSP_EVT */
    188     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    189     /* RCVRSP_EVT */
    190     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    191     /* SENDMSG_EVT */
    192     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    193     /* RET_TOUT_EVT */
    194     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    195     /* RSP_TOUT_EVT */
    196     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    197     /* IDLE_TOUT_EVT */
    198     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    199     /* UL_OPEN_EVT */
    200     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
    201     /* UL_CLOSE_EVT */
    202     {AVDT_CCB_CLEAR_CMDS, AVDT_CCB_CHAN_CLOSE, AVDT_CCB_CLOSING_ST},
    203     /* LL_OPEN_EVT */
    204     {AVDT_CCB_SND_CMD, AVDT_CCB_LL_OPENED, AVDT_CCB_OPEN_ST},
    205     /* LL_CLOSE_EVT */
    206     {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    207     /* LL_CONG_EVT */
    208     {AVDT_CCB_CONG_STATE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST}};
    209 
    210 /* state table for open state */
    211 const uint8_t avdt_ccb_st_open[][AVDT_CCB_NUM_COLS] = {
    212     /* Event */
    213     /* Action 1                    Action 2                    Next state */
    214     /* API_DISCOVER_REQ_EVT */
    215     {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    216     /* API_GETCAP_REQ_EVT */
    217     {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    218     /* API_START_REQ_EVT */
    219     {AVDT_CCB_SND_START_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    220     /* API_SUSPEND_REQ_EVT */
    221     {AVDT_CCB_SND_SUSPEND_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    222     /* API_DISCOVER_RSP_EVT */
    223     {AVDT_CCB_SND_DISCOVER_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    224     /* API_GETCAP_RSP_EVT */
    225     {AVDT_CCB_SND_GETCAP_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    226     /* API_START_RSP_EVT */
    227     {AVDT_CCB_SND_START_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    228     /* API_SUSPEND_RSP_EVT */
    229     {AVDT_CCB_SND_SUSPEND_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    230     /* API_CONNECT_REQ_EVT */
    231     {AVDT_CCB_SET_CONN, AVDT_CCB_LL_OPENED, AVDT_CCB_OPEN_ST},
    232     /* API_DISCONNECT_REQ_EVT */
    233     {AVDT_CCB_SET_DISCONN, AVDT_CCB_DO_DISCONN, AVDT_CCB_CLOSING_ST},
    234     /* MSG_DISCOVER_CMD_EVT */
    235     {AVDT_CCB_HDL_DISCOVER_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    236     /* MSG_GETCAP_CMD_EVT */
    237     {AVDT_CCB_HDL_GETCAP_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    238     /* MSG_START_CMD_EVT */
    239     {AVDT_CCB_HDL_START_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    240     /* MSG_SUSPEND_CMD_EVT */
    241     {AVDT_CCB_HDL_SUSPEND_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    242     /* MSG_DISCOVER_RSP_EVT */
    243     {AVDT_CCB_CHK_CLOSE, AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_OPEN_ST},
    244     /* MSG_GETCAP_RSP_EVT */
    245     {AVDT_CCB_CHK_CLOSE, AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_OPEN_ST},
    246     /* MSG_START_RSP_EVT */
    247     {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    248     /* MSG_SUSPEND_RSP_EVT */
    249     {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    250     /* RCVRSP_EVT */
    251     {AVDT_CCB_FREE_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    252     /* SENDMSG_EVT */
    253     {AVDT_CCB_SND_MSG, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    254     /* RET_TOUT_EVT */
    255     {AVDT_CCB_RET_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    256     /* RSP_TOUT_EVT */
    257     {AVDT_CCB_CMD_FAIL, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
    258     /* IDLE_TOUT_EVT */
    259     {AVDT_CCB_CLEAR_CMDS, AVDT_CCB_CHAN_CLOSE, AVDT_CCB_CLOSING_ST},
    260     /* UL_OPEN_EVT */
    261     {AVDT_CCB_CHK_TIMER, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    262     /* UL_CLOSE_EVT */
    263     {AVDT_CCB_CHK_CLOSE, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    264     /* LL_OPEN_EVT */
    265     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
    266     /* LL_CLOSE_EVT */
    267     {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    268     /* LL_CONG_EVT */
    269     {AVDT_CCB_CONG_STATE, AVDT_CCB_SND_MSG, AVDT_CCB_OPEN_ST}};
    270 
    271 /* state table for closing state */
    272 const uint8_t avdt_ccb_st_closing[][AVDT_CCB_NUM_COLS] = {
    273     /* Event */
    274     /* Action 1                    Action 2                    Next state */
    275     /* API_DISCOVER_REQ_EVT */
    276     {AVDT_CCB_SET_RECONN, AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_CLOSING_ST},
    277     /* API_GETCAP_REQ_EVT */
    278     {AVDT_CCB_SET_RECONN, AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_CLOSING_ST},
    279     /* API_START_REQ_EVT */
    280     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    281     /* API_SUSPEND_REQ_EVT */
    282     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    283     /* API_DISCOVER_RSP_EVT */
    284     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    285     /* API_GETCAP_RSP_EVT */
    286     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    287     /* API_START_RSP_EVT */
    288     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    289     /* API_SUSPEND_RSP_EVT */
    290     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    291     /* API_CONNECT_REQ_EVT */
    292     {AVDT_CCB_SET_RECONN, AVDT_CCB_SET_CONN, AVDT_CCB_CLOSING_ST},
    293     /* API_DISCONNECT_REQ_EVT */
    294     {AVDT_CCB_CLR_RECONN, AVDT_CCB_SET_DISCONN, AVDT_CCB_CLOSING_ST},
    295     /* MSG_DISCOVER_CMD_EVT */
    296     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    297     /* MSG_GETCAP_CMD_EVT */
    298     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    299     /* MSG_START_CMD_EVT */
    300     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    301     /* MSG_SUSPEND_CMD_EVT */
    302     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    303     /* MSG_DISCOVER_RSP_EVT */
    304     {AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    305     /* MSG_GETCAP_RSP_EVT */
    306     {AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    307     /* MSG_START_RSP_EVT */
    308     {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    309     /* MSG_SUSPEND_RSP_EVT */
    310     {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    311     /* RCVRSP_EVT */
    312     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    313     /* SENDMSG_EVT */
    314     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    315     /* RET_TOUT_EVT */
    316     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    317     /* RSP_TOUT_EVT */
    318     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    319     /* IDLE_TOUT_EVT */
    320     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    321     /* UL_OPEN_EVT */
    322     {AVDT_CCB_SET_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    323     /* UL_CLOSE_EVT */
    324     {AVDT_CCB_CLR_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    325     /* LL_OPEN_EVT */
    326     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
    327     /* LL_CLOSE_EVT */
    328     {AVDT_CCB_CHK_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
    329     /* LL_CONG_EVT */
    330     {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST}};
    331 
    332 /* type for state table */
    333 typedef const uint8_t (*tAVDT_CCB_ST_TBL)[AVDT_CCB_NUM_COLS];
    334 
    335 /* state table */
    336 const tAVDT_CCB_ST_TBL avdt_ccb_st_tbl[] = {
    337     avdt_ccb_st_idle, avdt_ccb_st_opening, avdt_ccb_st_open,
    338     avdt_ccb_st_closing};
    339 
    340 /*******************************************************************************
    341  *
    342  * Function         avdt_ccb_init
    343  *
    344  * Description      Initialize channel control block module.
    345  *
    346  *
    347  * Returns          Nothing.
    348  *
    349  ******************************************************************************/
    350 void avdt_ccb_init(void) {
    351   for (size_t i = 0; i < AVDT_NUM_LINKS; i++) {
    352     avdtp_cb.ccb[i].Reset(i);
    353   }
    354   avdtp_cb.p_ccb_act = avdt_ccb_action;
    355 }
    356 
    357 /*******************************************************************************
    358  *
    359  * Function         avdt_ccb_event
    360  *
    361  * Description      State machine event handling function for ccb
    362  *
    363  *
    364  * Returns          Nothing.
    365  *
    366  ******************************************************************************/
    367 void avdt_ccb_event(AvdtpCcb* p_ccb, uint8_t event, tAVDT_CCB_EVT* p_data) {
    368   tAVDT_CCB_ST_TBL state_table;
    369   uint8_t action;
    370   int i;
    371 
    372 #if (AVDT_DEBUG == TRUE)
    373   AVDT_TRACE_EVENT("%s: CCB ccb=%d event=%s state=%s p_ccb=%p", __func__,
    374                    avdt_ccb_to_idx(p_ccb), avdt_ccb_evt_str[event],
    375                    avdt_ccb_st_str[p_ccb->state], p_ccb);
    376 #endif
    377 
    378   /* look up the state table for the current state */
    379   state_table = avdt_ccb_st_tbl[p_ccb->state];
    380 
    381   /* set next state */
    382   if (p_ccb->state != state_table[event][AVDT_CCB_NEXT_STATE]) {
    383     p_ccb->state = state_table[event][AVDT_CCB_NEXT_STATE];
    384   }
    385 
    386   /* execute action functions */
    387   for (i = 0; i < AVDT_CCB_ACTIONS; i++) {
    388     action = state_table[event][i];
    389     AVDT_TRACE_DEBUG("%s: event=%s state=%s action=%d", __func__,
    390                      avdt_ccb_evt_str[event], avdt_ccb_st_str[p_ccb->state],
    391                      action);
    392     if (action != AVDT_CCB_IGNORE) {
    393       (*avdtp_cb.p_ccb_act[action])(p_ccb, p_data);
    394     } else {
    395       break;
    396     }
    397   }
    398 }
    399 
    400 /*******************************************************************************
    401  *
    402  * Function         avdt_ccb_by_bd
    403  *
    404  * Description      This lookup function finds the ccb for a BD address.
    405  *
    406  *
    407  * Returns          pointer to the ccb, or NULL if none found.
    408  *
    409  ******************************************************************************/
    410 AvdtpCcb* avdt_ccb_by_bd(const RawAddress& bd_addr) {
    411   AvdtpCcb* p_ccb = &avdtp_cb.ccb[0];
    412   int i;
    413 
    414   for (i = 0; i < AVDT_NUM_LINKS; i++, p_ccb++) {
    415     /* if allocated ccb has matching ccb */
    416     if (p_ccb->allocated && p_ccb->peer_addr == bd_addr) {
    417       break;
    418     }
    419   }
    420 
    421   if (i == AVDT_NUM_LINKS) {
    422     /* if no ccb found */
    423     p_ccb = NULL;
    424 
    425     VLOG(1) << "No ccb for addr " << bd_addr;
    426   }
    427   return p_ccb;
    428 }
    429 
    430 /*******************************************************************************
    431  *
    432  * Function         avdt_ccb_alloc
    433  *
    434  * Description      Allocate a channel control block.
    435  *
    436  *
    437  * Returns          pointer to the ccb, or NULL if none could be allocated.
    438  *
    439  ******************************************************************************/
    440 AvdtpCcb* avdt_ccb_alloc(const RawAddress& bd_addr) {
    441   // Find available entry
    442   AvdtpCcb* p_ccb = &avdtp_cb.ccb[0];
    443   for (int i = 0; i < AVDT_NUM_LINKS; i++, p_ccb++) {
    444     if (!p_ccb->allocated) {
    445       p_ccb->Allocate(bd_addr);
    446       AVDT_TRACE_DEBUG("%s: allocated (index %d) for peer %s", __func__, i,
    447                        bd_addr.ToString().c_str());
    448       return p_ccb;
    449     }
    450   }
    451 
    452   AVDT_TRACE_WARNING("%s: out of AvdtpCcb entries", __func__);
    453   return nullptr;
    454 }
    455 
    456 AvdtpCcb* avdt_ccb_alloc_by_channel_index(const RawAddress& bd_addr,
    457                                           uint8_t channel_index) {
    458   // Allocate the entry for the specified channel index
    459   if (channel_index >= AVDT_NUM_LINKS) {
    460     AVDT_TRACE_ERROR("%s: peer %s invalid channel index %d (max %d)", __func__,
    461                      bd_addr.ToString().c_str(), channel_index, AVDT_NUM_LINKS);
    462     return nullptr;
    463   }
    464   AvdtpCcb* p_ccb = &avdtp_cb.ccb[channel_index];
    465   if (p_ccb->allocated) {
    466     AVDT_TRACE_ERROR("%s: peer %s channel index %d already allocated", __func__,
    467                      bd_addr.ToString().c_str(), channel_index);
    468     return nullptr;
    469   }
    470   p_ccb->Allocate(bd_addr);
    471   AVDT_TRACE_DEBUG("%s: allocated (index %d) peer=%s p_ccb=%p", __func__,
    472                    channel_index, p_ccb->peer_addr.ToString().c_str(), p_ccb);
    473   return p_ccb;
    474 }
    475 
    476 void AvdtpCcb::Allocate(const RawAddress& peer_address) {
    477   ResetCcb();
    478   peer_addr = peer_address;
    479   cmd_q = fixed_queue_new(SIZE_MAX);
    480   rsp_q = fixed_queue_new(SIZE_MAX);
    481   idle_ccb_timer = alarm_new("avdtp_ccb.idle_ccb_timer");
    482   ret_ccb_timer = alarm_new("avdtp_ccb.ret_ccb_timer");
    483   rsp_ccb_timer = alarm_new("avdtp_ccb.rsp_ccb_timer");
    484   allocated = true;
    485 }
    486 
    487 /*******************************************************************************
    488  *
    489  * Function         avdt_ccb_dealloc
    490  *
    491  * Description      Deallocate a stream control block.
    492  *
    493  *
    494  * Returns          void.
    495  *
    496  ******************************************************************************/
    497 void avdt_ccb_dealloc(AvdtpCcb* p_ccb, UNUSED_ATTR tAVDT_CCB_EVT* p_data) {
    498   AVDT_TRACE_DEBUG("%s: deallocated (index %d) peer=%s p_ccb=%p", __func__,
    499                    avdt_ccb_to_idx(p_ccb), p_ccb->peer_addr.ToString().c_str(),
    500                    p_ccb);
    501   p_ccb->ResetCcb();
    502 }
    503 
    504 /*******************************************************************************
    505  *
    506  * Function         avdt_ccb_to_idx
    507  *
    508  * Description      Given a pointer to an ccb, return its index.
    509  *
    510  *
    511  * Returns          Index of ccb.
    512  *
    513  ******************************************************************************/
    514 uint8_t avdt_ccb_to_idx(AvdtpCcb* p_ccb) {
    515   /* use array arithmetic to determine index */
    516   return (uint8_t)(p_ccb - avdtp_cb.ccb);
    517 }
    518 
    519 /*******************************************************************************
    520  *
    521  * Function         avdt_ccb_by_idx
    522  *
    523  * Description      Return ccb pointer based on ccb index.
    524  *
    525  *
    526  * Returns          pointer to the ccb, or NULL if none found.
    527  *
    528  ******************************************************************************/
    529 AvdtpCcb* avdt_ccb_by_idx(uint8_t idx) {
    530   AvdtpCcb* p_ccb;
    531 
    532   /* verify index */
    533   if (idx < AVDT_NUM_LINKS) {
    534     p_ccb = &avdtp_cb.ccb[idx];
    535   } else {
    536     p_ccb = NULL;
    537     AVDT_TRACE_WARNING("No ccb for idx %d", idx);
    538   }
    539   return p_ccb;
    540 }
    541