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