1 /******************************************************************************
2 *
3 * Copyright (C) 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 "bt_types.h"
28 #include "bt_target.h"
29 #include "bt_utils.h"
30 #include "avdt_api.h"
31 #include "avdtc_api.h"
32 #include "avdt_int.h"
33 #include "bt_common.h"
34 #include "btu.h"
35
36 /*****************************************************************************
37 ** state machine constants and types
38 *****************************************************************************/
39 #if AVDT_DEBUG == TRUE
40
41 /* verbose state strings for trace */
42 const char * const avdt_ccb_st_str[] = {
43 "CCB_IDLE_ST",
44 "CCB_OPENING_ST",
45 "CCB_OPEN_ST",
46 "CCB_CLOSING_ST"
47 };
48
49 /* verbose event strings for trace */
50 const char * const avdt_ccb_evt_str[] = {
51 "API_DISCOVER_REQ_EVT",
52 "API_GETCAP_REQ_EVT",
53 "API_START_REQ_EVT",
54 "API_SUSPEND_REQ_EVT",
55 "API_DISCOVER_RSP_EVT",
56 "API_GETCAP_RSP_EVT",
57 "API_START_RSP_EVT",
58 "API_SUSPEND_RSP_EVT",
59 "API_CONNECT_REQ_EVT",
60 "API_DISCONNECT_REQ_EVT",
61 "MSG_DISCOVER_CMD_EVT",
62 "MSG_GETCAP_CMD_EVT",
63 "MSG_START_CMD_EVT",
64 "MSG_SUSPEND_CMD_EVT",
65 "MSG_DISCOVER_RSP_EVT",
66 "MSG_GETCAP_RSP_EVT",
67 "MSG_START_RSP_EVT",
68 "MSG_SUSPEND_RSP_EVT",
69 "RCVRSP_EVT",
70 "SENDMSG_EVT",
71 "RET_TOUT_EVT",
72 "RSP_TOUT_EVT",
73 "IDLE_TOUT_EVT",
74 "UL_OPEN_EVT",
75 "UL_CLOSE_EVT",
76 "LL_OPEN_EVT",
77 "LL_CLOSE_EVT",
78 "LL_CONG_EVT"
79 };
80
81 #endif
82
83
84 /* action function list */
85 const tAVDT_CCB_ACTION avdt_ccb_action[] = {
86 avdt_ccb_chan_open,
87 avdt_ccb_chan_close,
88 avdt_ccb_chk_close,
89 avdt_ccb_hdl_discover_cmd,
90 avdt_ccb_hdl_discover_rsp,
91 avdt_ccb_hdl_getcap_cmd,
92 avdt_ccb_hdl_getcap_rsp,
93 avdt_ccb_hdl_start_cmd,
94 avdt_ccb_hdl_start_rsp,
95 avdt_ccb_hdl_suspend_cmd,
96 avdt_ccb_hdl_suspend_rsp,
97 avdt_ccb_snd_discover_cmd,
98 avdt_ccb_snd_discover_rsp,
99 avdt_ccb_snd_getcap_cmd,
100 avdt_ccb_snd_getcap_rsp,
101 avdt_ccb_snd_start_cmd,
102 avdt_ccb_snd_start_rsp,
103 avdt_ccb_snd_suspend_cmd,
104 avdt_ccb_snd_suspend_rsp,
105 avdt_ccb_clear_cmds,
106 avdt_ccb_cmd_fail,
107 avdt_ccb_free_cmd,
108 avdt_ccb_cong_state,
109 avdt_ccb_ret_cmd,
110 avdt_ccb_snd_cmd,
111 avdt_ccb_snd_msg,
112 avdt_ccb_set_reconn,
113 avdt_ccb_clr_reconn,
114 avdt_ccb_chk_reconn,
115 avdt_ccb_chk_timer,
116 avdt_ccb_set_conn,
117 avdt_ccb_set_disconn,
118 avdt_ccb_do_disconn,
119 avdt_ccb_ll_closed,
120 avdt_ccb_ll_opened,
121 avdt_ccb_dealloc
122 };
123
124 /* state table information */
125 #define AVDT_CCB_ACTIONS 2 /* number of actions */
126 #define AVDT_CCB_NEXT_STATE 2 /* position of next state */
127 #define AVDT_CCB_NUM_COLS 3 /* number of columns in state tables */
128
129 /* state table for idle state */
130 const UINT8 avdt_ccb_st_idle[][AVDT_CCB_NUM_COLS] = {
131 /* Event Action 1 Action 2 Next state */
132 /* API_DISCOVER_REQ_EVT */ {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
133 /* API_GETCAP_REQ_EVT */ {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
134 /* API_START_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
135 /* API_SUSPEND_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
136 /* API_DISCOVER_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
137 /* API_GETCAP_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
138 /* API_START_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
139 /* API_SUSPEND_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
140 /* API_CONNECT_REQ_EVT */ {AVDT_CCB_SET_CONN, AVDT_CCB_CHAN_OPEN, AVDT_CCB_OPENING_ST},
141 /* API_DISCONNECT_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
142 /* MSG_DISCOVER_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
143 /* MSG_GETCAP_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
144 /* MSG_START_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
145 /* MSG_SUSPEND_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
146 /* MSG_DISCOVER_RSP_EVT */ {AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
147 /* MSG_GETCAP_RSP_EVT */ {AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
148 /* MSG_START_RSP_EVT */ {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
149 /* MSG_SUSPEND_RSP_EVT */ {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
150 /* RCVRSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
151 /* SENDMSG_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
152 /* RET_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
153 /* RSP_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
154 /* IDLE_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
155 /* UL_OPEN_EVT */ {AVDT_CCB_CHAN_OPEN, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
156 /* UL_CLOSE_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
157 /* LL_OPEN_EVT */ {AVDT_CCB_LL_OPENED, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
158 /* LL_CLOSE_EVT */ {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
159 /* LL_CONG_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST}
160 };
161
162 /* state table for opening state */
163 const UINT8 avdt_ccb_st_opening[][AVDT_CCB_NUM_COLS] = {
164 /* Event Action 1 Action 2 Next state */
165 /* API_DISCOVER_REQ_EVT */ {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
166 /* API_GETCAP_REQ_EVT */ {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
167 /* API_START_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
168 /* API_SUSPEND_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
169 /* API_DISCOVER_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
170 /* API_GETCAP_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
171 /* API_START_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
172 /* API_SUSPEND_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
173 /* API_CONNECT_REQ_EVT */ {AVDT_CCB_SET_CONN, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
174 /* API_DISCONNECT_REQ_EVT */ {AVDT_CCB_SET_DISCONN, AVDT_CCB_DO_DISCONN, AVDT_CCB_CLOSING_ST},
175 /* MSG_DISCOVER_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
176 /* MSG_GETCAP_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
177 /* MSG_START_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
178 /* MSG_SUSPEND_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
179 /* MSG_DISCOVER_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
180 /* MSG_GETCAP_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
181 /* MSG_START_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
182 /* MSG_SUSPEND_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
183 /* RCVRSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
184 /* SENDMSG_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
185 /* RET_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
186 /* RSP_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
187 /* IDLE_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
188 /* UL_OPEN_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST},
189 /* UL_CLOSE_EVT */ {AVDT_CCB_CLEAR_CMDS, AVDT_CCB_CHAN_CLOSE, AVDT_CCB_CLOSING_ST},
190 /* LL_OPEN_EVT */ {AVDT_CCB_SND_CMD, AVDT_CCB_LL_OPENED, AVDT_CCB_OPEN_ST},
191 /* LL_CLOSE_EVT */ {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
192 /* LL_CONG_EVT */ {AVDT_CCB_CONG_STATE, AVDT_CCB_IGNORE, AVDT_CCB_OPENING_ST}
193 };
194
195 /* state table for open state */
196 const UINT8 avdt_ccb_st_open[][AVDT_CCB_NUM_COLS] = {
197 /* Event Action 1 Action 2 Next state */
198 /* API_DISCOVER_REQ_EVT */ {AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
199 /* API_GETCAP_REQ_EVT */ {AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
200 /* API_START_REQ_EVT */ {AVDT_CCB_SND_START_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
201 /* API_SUSPEND_REQ_EVT */ {AVDT_CCB_SND_SUSPEND_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
202 /* API_DISCOVER_RSP_EVT */ {AVDT_CCB_SND_DISCOVER_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
203 /* API_GETCAP_RSP_EVT */ {AVDT_CCB_SND_GETCAP_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
204 /* API_START_RSP_EVT */ {AVDT_CCB_SND_START_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
205 /* API_SUSPEND_RSP_EVT */ {AVDT_CCB_SND_SUSPEND_RSP, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
206 /* API_CONNECT_REQ_EVT */ {AVDT_CCB_SET_CONN, AVDT_CCB_LL_OPENED, AVDT_CCB_OPEN_ST},
207 /* API_DISCONNECT_REQ_EVT */ {AVDT_CCB_SET_DISCONN, AVDT_CCB_DO_DISCONN, AVDT_CCB_CLOSING_ST},
208 /* MSG_DISCOVER_CMD_EVT */ {AVDT_CCB_HDL_DISCOVER_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
209 /* MSG_GETCAP_CMD_EVT */ {AVDT_CCB_HDL_GETCAP_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
210 /* MSG_START_CMD_EVT */ {AVDT_CCB_HDL_START_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
211 /* MSG_SUSPEND_CMD_EVT */ {AVDT_CCB_HDL_SUSPEND_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
212 /* MSG_DISCOVER_RSP_EVT */ {AVDT_CCB_CHK_CLOSE, AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_OPEN_ST},
213 /* MSG_GETCAP_RSP_EVT */ {AVDT_CCB_CHK_CLOSE, AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_OPEN_ST},
214 /* MSG_START_RSP_EVT */ {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
215 /* MSG_SUSPEND_RSP_EVT */ {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
216 /* RCVRSP_EVT */ {AVDT_CCB_FREE_CMD, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
217 /* SENDMSG_EVT */ {AVDT_CCB_SND_MSG, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
218 /* RET_TOUT_EVT */ {AVDT_CCB_RET_CMD, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
219 /* RSP_TOUT_EVT */ {AVDT_CCB_CMD_FAIL, AVDT_CCB_SND_CMD, AVDT_CCB_OPEN_ST},
220 /* IDLE_TOUT_EVT */ {AVDT_CCB_CLEAR_CMDS, AVDT_CCB_CHAN_CLOSE, AVDT_CCB_CLOSING_ST},
221 /* UL_OPEN_EVT */ {AVDT_CCB_CHK_TIMER, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
222 /* UL_CLOSE_EVT */ {AVDT_CCB_CHK_CLOSE, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
223 /* LL_OPEN_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_OPEN_ST},
224 /* LL_CLOSE_EVT */ {AVDT_CCB_LL_CLOSED, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
225 /* LL_CONG_EVT */ {AVDT_CCB_CONG_STATE, AVDT_CCB_SND_MSG, AVDT_CCB_OPEN_ST}
226 };
227
228 /* state table for closing state */
229 const UINT8 avdt_ccb_st_closing[][AVDT_CCB_NUM_COLS] = {
230 /* Event Action 1 Action 2 Next state */
231 /* API_DISCOVER_REQ_EVT */ {AVDT_CCB_SET_RECONN, AVDT_CCB_SND_DISCOVER_CMD, AVDT_CCB_CLOSING_ST},
232 /* API_GETCAP_REQ_EVT */ {AVDT_CCB_SET_RECONN, AVDT_CCB_SND_GETCAP_CMD, AVDT_CCB_CLOSING_ST},
233 /* API_START_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
234 /* API_SUSPEND_REQ_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
235 /* API_DISCOVER_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
236 /* API_GETCAP_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
237 /* API_START_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
238 /* API_SUSPEND_RSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
239 /* API_CONNECT_REQ_EVT */ {AVDT_CCB_SET_RECONN, AVDT_CCB_SET_CONN, AVDT_CCB_CLOSING_ST},
240 /* API_DISCONNECT_REQ_EVT */ {AVDT_CCB_CLR_RECONN, AVDT_CCB_SET_DISCONN, AVDT_CCB_CLOSING_ST},
241 /* MSG_DISCOVER_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
242 /* MSG_GETCAP_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
243 /* MSG_START_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
244 /* MSG_SUSPEND_CMD_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
245 /* MSG_DISCOVER_RSP_EVT */ {AVDT_CCB_HDL_DISCOVER_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
246 /* MSG_GETCAP_RSP_EVT */ {AVDT_CCB_HDL_GETCAP_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
247 /* MSG_START_RSP_EVT */ {AVDT_CCB_HDL_START_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
248 /* MSG_SUSPEND_RSP_EVT */ {AVDT_CCB_HDL_SUSPEND_RSP, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
249 /* RCVRSP_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
250 /* SENDMSG_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
251 /* RET_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
252 /* RSP_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
253 /* IDLE_TOUT_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
254 /* UL_OPEN_EVT */ {AVDT_CCB_SET_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
255 /* UL_CLOSE_EVT */ {AVDT_CCB_CLR_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
256 /* LL_OPEN_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST},
257 /* LL_CLOSE_EVT */ {AVDT_CCB_CHK_RECONN, AVDT_CCB_IGNORE, AVDT_CCB_IDLE_ST},
258 /* LL_CONG_EVT */ {AVDT_CCB_IGNORE, AVDT_CCB_IGNORE, AVDT_CCB_CLOSING_ST}
259 };
260
261 /* type for state table */
262 typedef const UINT8 (*tAVDT_CCB_ST_TBL)[AVDT_CCB_NUM_COLS];
263
264 /* state table */
265 const tAVDT_CCB_ST_TBL avdt_ccb_st_tbl[] = {
266 avdt_ccb_st_idle,
267 avdt_ccb_st_opening,
268 avdt_ccb_st_open,
269 avdt_ccb_st_closing
270 };
271
272 /*******************************************************************************
273 **
274 ** Function avdt_ccb_init
275 **
276 ** Description Initialize channel control block module.
277 **
278 **
279 ** Returns Nothing.
280 **
281 *******************************************************************************/
avdt_ccb_init(void)282 void avdt_ccb_init(void)
283 {
284 memset(&avdt_cb.ccb[0], 0, sizeof(tAVDT_CCB) * AVDT_NUM_LINKS);
285 avdt_cb.p_ccb_act = (tAVDT_CCB_ACTION *) avdt_ccb_action;
286 }
287
288 /*******************************************************************************
289 **
290 ** Function avdt_ccb_event
291 **
292 ** Description State machine event handling function for ccb
293 **
294 **
295 ** Returns Nothing.
296 **
297 *******************************************************************************/
avdt_ccb_event(tAVDT_CCB * p_ccb,UINT8 event,tAVDT_CCB_EVT * p_data)298 void avdt_ccb_event(tAVDT_CCB *p_ccb, UINT8 event, tAVDT_CCB_EVT *p_data)
299 {
300 tAVDT_CCB_ST_TBL state_table;
301 UINT8 action;
302 int i;
303
304 #if AVDT_DEBUG == TRUE
305 AVDT_TRACE_EVENT("CCB ccb=%d event=%s state=%s", avdt_ccb_to_idx(p_ccb), avdt_ccb_evt_str[event], avdt_ccb_st_str[p_ccb->state]);
306 #endif
307
308 /* look up the state table for the current state */
309 state_table = avdt_ccb_st_tbl[p_ccb->state];
310
311 /* set next state */
312 if (p_ccb->state != state_table[event][AVDT_CCB_NEXT_STATE]) {
313 p_ccb->state = state_table[event][AVDT_CCB_NEXT_STATE];
314 }
315
316 /* execute action functions */
317 for (i = 0; i < AVDT_CCB_ACTIONS; i++)
318 {
319 if ((action = state_table[event][i]) != AVDT_CCB_IGNORE)
320 {
321 (*avdt_cb.p_ccb_act[action])(p_ccb, p_data);
322 }
323 else
324 {
325 break;
326 }
327 }
328 }
329
330
331 /*******************************************************************************
332 **
333 ** Function avdt_ccb_by_bd
334 **
335 ** Description This lookup function finds the ccb for a BD address.
336 **
337 **
338 ** Returns pointer to the ccb, or NULL if none found.
339 **
340 *******************************************************************************/
avdt_ccb_by_bd(BD_ADDR bd_addr)341 tAVDT_CCB *avdt_ccb_by_bd(BD_ADDR bd_addr)
342 {
343 tAVDT_CCB *p_ccb = &avdt_cb.ccb[0];
344 int i;
345
346 for (i = 0; i < AVDT_NUM_LINKS; i++, p_ccb++)
347 {
348 /* if allocated ccb has matching ccb */
349 if (p_ccb->allocated && (!memcmp(p_ccb->peer_addr, bd_addr, BD_ADDR_LEN)))
350 {
351 break;
352 }
353 }
354
355 if (i == AVDT_NUM_LINKS)
356 {
357 /* if no ccb found */
358 p_ccb = NULL;
359
360 AVDT_TRACE_DEBUG("No ccb for addr %02x-%02x-%02x-%02x-%02x-%02x",
361 bd_addr[0], bd_addr[1], bd_addr[2], bd_addr[3], bd_addr[4], bd_addr[5]);
362 }
363 return p_ccb;
364 }
365
366 /*******************************************************************************
367 **
368 ** Function avdt_ccb_alloc
369 **
370 ** Description Allocate a channel control block.
371 **
372 **
373 ** Returns pointer to the ccb, or NULL if none could be allocated.
374 **
375 *******************************************************************************/
avdt_ccb_alloc(BD_ADDR bd_addr)376 tAVDT_CCB *avdt_ccb_alloc(BD_ADDR bd_addr)
377 {
378 tAVDT_CCB *p_ccb = &avdt_cb.ccb[0];
379 int i;
380
381 for (i = 0; i < AVDT_NUM_LINKS; i++, p_ccb++)
382 {
383 if (!p_ccb->allocated)
384 {
385 p_ccb->allocated = TRUE;
386 memcpy(p_ccb->peer_addr, bd_addr, BD_ADDR_LEN);
387 p_ccb->cmd_q = fixed_queue_new(SIZE_MAX);
388 p_ccb->rsp_q = fixed_queue_new(SIZE_MAX);
389 p_ccb->idle_ccb_timer = alarm_new("avdt_ccb.idle_ccb_timer");
390 p_ccb->ret_ccb_timer = alarm_new("avdt_ccb.ret_ccb_timer");
391 p_ccb->rsp_ccb_timer = alarm_new("avdt_ccb.rsp_ccb_timer");
392 AVDT_TRACE_DEBUG("avdt_ccb_alloc %d", i);
393 break;
394 }
395 }
396
397 if (i == AVDT_NUM_LINKS)
398 {
399 /* out of ccbs */
400 p_ccb = NULL;
401 AVDT_TRACE_WARNING("Out of ccbs");
402 }
403 return p_ccb;
404 }
405
406 /*******************************************************************************
407 **
408 ** Function avdt_ccb_dealloc
409 **
410 ** Description Deallocate a stream control block.
411 **
412 **
413 ** Returns void.
414 **
415 *******************************************************************************/
avdt_ccb_dealloc(tAVDT_CCB * p_ccb,tAVDT_CCB_EVT * p_data)416 void avdt_ccb_dealloc(tAVDT_CCB *p_ccb, tAVDT_CCB_EVT *p_data)
417 {
418 UNUSED(p_data);
419
420 AVDT_TRACE_DEBUG("avdt_ccb_dealloc %d", avdt_ccb_to_idx(p_ccb));
421 alarm_free(p_ccb->idle_ccb_timer);
422 alarm_free(p_ccb->ret_ccb_timer);
423 alarm_free(p_ccb->rsp_ccb_timer);
424 fixed_queue_free(p_ccb->cmd_q, NULL);
425 fixed_queue_free(p_ccb->rsp_q, NULL);
426 memset(p_ccb, 0, sizeof(tAVDT_CCB));
427 }
428
429 /*******************************************************************************
430 **
431 ** Function avdt_ccb_to_idx
432 **
433 ** Description Given a pointer to an ccb, return its index.
434 **
435 **
436 ** Returns Index of ccb.
437 **
438 *******************************************************************************/
avdt_ccb_to_idx(tAVDT_CCB * p_ccb)439 UINT8 avdt_ccb_to_idx(tAVDT_CCB *p_ccb)
440 {
441 /* use array arithmetic to determine index */
442 return (UINT8) (p_ccb - avdt_cb.ccb);
443 }
444
445 /*******************************************************************************
446 **
447 ** Function avdt_ccb_by_idx
448 **
449 ** Description Return ccb pointer based on ccb index.
450 **
451 **
452 ** Returns pointer to the ccb, or NULL if none found.
453 **
454 *******************************************************************************/
avdt_ccb_by_idx(UINT8 idx)455 tAVDT_CCB *avdt_ccb_by_idx(UINT8 idx)
456 {
457 tAVDT_CCB *p_ccb;
458
459 /* verify index */
460 if (idx < AVDT_NUM_LINKS)
461 {
462 p_ccb = &avdt_cb.ccb[idx];
463 }
464 else
465 {
466 p_ccb = NULL;
467 AVDT_TRACE_WARNING("No ccb for idx %d", idx);
468 }
469 return p_ccb;
470 }
471
472