1 /******************************************************************************
2 *
3 * Copyright (C) 2010-2013 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 *
22 * This file contains the LLCP utilities
23 *
24 ******************************************************************************/
25
26 #include <string.h>
27 #include "gki.h"
28 #include "nfc_target.h"
29 #include "bt_types.h"
30 #include "trace_api.h"
31 #include "llcp_int.h"
32 #include "llcp_defs.h"
33 #include "nfc_int.h"
34
35 /*******************************************************************************
36 **
37 ** Function llcp_util_parse_link_params
38 **
39 ** Description Parse LLCP Link parameters
40 **
41 ** Returns TRUE if success
42 **
43 *******************************************************************************/
llcp_util_parse_link_params(UINT16 length,UINT8 * p_bytes)44 BOOLEAN llcp_util_parse_link_params (UINT16 length, UINT8 *p_bytes)
45 {
46 UINT8 param_type, param_len, *p = p_bytes;
47
48 while (length)
49 {
50 BE_STREAM_TO_UINT8 (param_type, p);
51 length--;
52
53 switch (param_type)
54 {
55 case LLCP_VERSION_TYPE:
56 BE_STREAM_TO_UINT8 (param_len, p);
57 BE_STREAM_TO_UINT8 (llcp_cb.lcb.peer_version, p);
58 LLCP_TRACE_DEBUG1 ("Peer Version - 0x%02X", llcp_cb.lcb.peer_version);
59 break;
60
61 case LLCP_MIUX_TYPE:
62 BE_STREAM_TO_UINT8 (param_len, p);
63 BE_STREAM_TO_UINT16 (llcp_cb.lcb.peer_miu, p);
64 llcp_cb.lcb.peer_miu &= LLCP_MIUX_MASK;
65 llcp_cb.lcb.peer_miu += LLCP_DEFAULT_MIU;
66 LLCP_TRACE_DEBUG1 ("Peer MIU - %d bytes", llcp_cb.lcb.peer_miu);
67 break;
68
69 case LLCP_WKS_TYPE:
70 BE_STREAM_TO_UINT8 (param_len, p);
71 BE_STREAM_TO_UINT16 (llcp_cb.lcb.peer_wks, p);
72 LLCP_TRACE_DEBUG1 ("Peer WKS - 0x%04X", llcp_cb.lcb.peer_wks);
73 break;
74
75 case LLCP_LTO_TYPE:
76 BE_STREAM_TO_UINT8 (param_len, p);
77 BE_STREAM_TO_UINT8 (llcp_cb.lcb.peer_lto, p);
78 llcp_cb.lcb.peer_lto *= LLCP_LTO_UNIT; /* 10ms unit */
79 LLCP_TRACE_DEBUG1 ("Peer LTO - %d ms", llcp_cb.lcb.peer_lto);
80 break;
81
82 case LLCP_OPT_TYPE:
83 BE_STREAM_TO_UINT8 (param_len, p);
84 BE_STREAM_TO_UINT8 (llcp_cb.lcb.peer_opt, p);
85 LLCP_TRACE_DEBUG1 ("Peer OPT - 0x%02X", llcp_cb.lcb.peer_opt);
86 break;
87
88 default:
89 LLCP_TRACE_ERROR1 ("llcp_util_parse_link_params (): Unexpected type 0x%x", param_type);
90 BE_STREAM_TO_UINT8 (param_len, p);
91 p += param_len;
92 break;
93 }
94
95 if (length >= param_len + 1)
96 length -= param_len + 1;
97 else
98 {
99 LLCP_TRACE_ERROR0 ("llcp_util_parse_link_params (): Bad LTV's");
100 return (FALSE);
101 }
102 }
103 return (TRUE);
104 }
105
106 /*******************************************************************************
107 **
108 ** Function llcp_util_adjust_ll_congestion
109 **
110 ** Description adjust tx/rx congestion thresholds on logical link
111 **
112 ** Returns void
113 **
114 *******************************************************************************/
llcp_util_adjust_ll_congestion(void)115 void llcp_util_adjust_ll_congestion (void)
116 {
117 /* buffer quota is allocated equally for each logical data link */
118 if (llcp_cb.num_logical_data_link)
119 {
120 llcp_cb.ll_tx_congest_start = llcp_cb.max_num_ll_tx_buff / llcp_cb.num_logical_data_link;
121 llcp_cb.ll_rx_congest_start = llcp_cb.max_num_ll_rx_buff / llcp_cb.num_logical_data_link;
122 }
123 else
124 {
125 llcp_cb.ll_tx_congest_start = llcp_cb.max_num_ll_tx_buff;
126 llcp_cb.ll_rx_congest_start = llcp_cb.max_num_ll_rx_buff;
127 }
128
129 /* at least one for each logical data link */
130 if (llcp_cb.ll_tx_congest_start == 0)
131 {
132 llcp_cb.ll_tx_congest_start = 1;
133 }
134 if (llcp_cb.ll_rx_congest_start == 0)
135 {
136 llcp_cb.ll_rx_congest_start = 1;
137 }
138
139 if (llcp_cb.ll_tx_congest_start > 1)
140 {
141 llcp_cb.ll_tx_congest_end = 1;
142 }
143 else
144 {
145 llcp_cb.ll_tx_congest_end = 0;
146 }
147
148 LLCP_TRACE_DEBUG4 ("num_logical_data_link=%d, ll_tx_congest_start=%d, ll_tx_congest_end=%d, ll_rx_congest_start=%d",
149 llcp_cb.num_logical_data_link,
150 llcp_cb.ll_tx_congest_start,
151 llcp_cb.ll_tx_congest_end,
152 llcp_cb.ll_rx_congest_start);
153 }
154
155 /*******************************************************************************
156 **
157 ** Function llcp_util_adjust_dl_rx_congestion
158 **
159 ** Description adjust rx congestion thresholds on data link
160 **
161 ** Returns void
162 **
163 *******************************************************************************/
llcp_util_adjust_dl_rx_congestion(void)164 void llcp_util_adjust_dl_rx_congestion (void)
165 {
166 UINT8 idx, rx_congest_start;
167
168 if (llcp_cb.num_data_link_connection)
169 {
170 rx_congest_start = llcp_cb.num_rx_buff / llcp_cb.num_data_link_connection;
171
172 for (idx = 0; idx < LLCP_MAX_DATA_LINK; idx++)
173 {
174 if (llcp_cb.dlcb[idx].state == LLCP_DLC_STATE_CONNECTED)
175 {
176 if (rx_congest_start > llcp_cb.dlcb[idx].local_rw)
177 {
178 /*
179 ** set rx congestion threshold LLCP_DL_MIN_RX_CONGEST at least
180 ** so, we don't need to flow off too often.
181 */
182 if (llcp_cb.dlcb[idx].local_rw + 1 > LLCP_DL_MIN_RX_CONGEST)
183 llcp_cb.dlcb[idx].rx_congest_threshold = llcp_cb.dlcb[idx].local_rw + 1;
184 else
185 llcp_cb.dlcb[idx].rx_congest_threshold = LLCP_DL_MIN_RX_CONGEST;
186 }
187 else
188 {
189 llcp_cb.dlcb[idx].rx_congest_threshold = LLCP_DL_MIN_RX_CONGEST;
190 }
191
192 LLCP_TRACE_DEBUG3 ("DLC[%d], local_rw=%d, rx_congest_threshold=%d",
193 idx,
194 llcp_cb.dlcb[idx].local_rw,
195 llcp_cb.dlcb[idx].rx_congest_threshold);
196 }
197 }
198 }
199
200 }
201
202 /*******************************************************************************
203 **
204 ** Function llcp_util_check_rx_congested_status
205 **
206 ** Description Update rx congested status
207 **
208 ** Returns void
209 **
210 *******************************************************************************/
llcp_util_check_rx_congested_status(void)211 void llcp_util_check_rx_congested_status (void)
212 {
213 UINT8 idx;
214
215 if (llcp_cb.overall_rx_congested)
216 {
217 /* check if rx congestion clear */
218 if (llcp_cb.total_rx_ui_pdu + llcp_cb.total_rx_i_pdu <= llcp_cb.overall_rx_congest_end)
219 {
220 LLCP_TRACE_DEBUG3 ("llcp_util_check_rx_congested_status (): rx link is uncongested, %d+%d <= %d",
221 llcp_cb.total_rx_ui_pdu, llcp_cb.total_rx_i_pdu,
222 llcp_cb.overall_rx_congest_end);
223
224 llcp_cb.overall_rx_congested = FALSE;
225
226 for (idx = 0; idx < LLCP_MAX_DATA_LINK; idx++)
227 {
228 /* set flag to clear local busy status on data link connections */
229 if ( (llcp_cb.dlcb[idx].state == LLCP_DLC_STATE_CONNECTED)
230 &&(llcp_cb.dlcb[idx].is_rx_congested == FALSE) )
231 {
232 llcp_cb.dlcb[idx].flags |= LLCP_DATA_LINK_FLAG_PENDING_RR_RNR;
233 }
234 }
235 }
236 }
237 else
238 {
239 /* check if rx link is congested */
240 if (llcp_cb.total_rx_ui_pdu + llcp_cb.total_rx_i_pdu >= llcp_cb.overall_rx_congest_start)
241 {
242 LLCP_TRACE_WARNING3 ("llcp_util_check_rx_congested_status (): rx link is congested, %d+%d >= %d",
243 llcp_cb.total_rx_ui_pdu, llcp_cb.total_rx_i_pdu,
244 llcp_cb.overall_rx_congest_start);
245
246 llcp_cb.overall_rx_congested = TRUE;
247
248 /* rx link congestion is started, send RNR to remote end point */
249 for (idx = 0; idx < LLCP_MAX_DATA_LINK; idx++)
250 {
251 if ( (llcp_cb.dlcb[idx].state == LLCP_DLC_STATE_CONNECTED)
252 &&(llcp_cb.dlcb[idx].is_rx_congested == FALSE) )
253 {
254 llcp_cb.dlcb[idx].flags |= LLCP_DATA_LINK_FLAG_PENDING_RR_RNR;
255 }
256 }
257 }
258 }
259 }
260
261 /*******************************************************************************
262 **
263 ** Function llcp_util_send_ui
264 **
265 ** Description Send UI PDU
266 **
267 ** Returns tLLCP_STATUS
268 **
269 *******************************************************************************/
llcp_util_send_ui(UINT8 ssap,UINT8 dsap,tLLCP_APP_CB * p_app_cb,BT_HDR * p_msg)270 tLLCP_STATUS llcp_util_send_ui (UINT8 ssap, UINT8 dsap, tLLCP_APP_CB *p_app_cb, BT_HDR *p_msg)
271 {
272 UINT8 *p;
273 tLLCP_STATUS status = LLCP_STATUS_SUCCESS;
274
275 p_msg->offset -= LLCP_PDU_HEADER_SIZE;
276 p_msg->len += LLCP_PDU_HEADER_SIZE;
277
278 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
279 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (dsap, LLCP_PDU_UI_TYPE, ssap));
280
281 GKI_enqueue (&p_app_cb->ui_xmit_q, p_msg);
282 llcp_cb.total_tx_ui_pdu++;
283
284 llcp_link_check_send_data ();
285
286 if ( (p_app_cb->is_ui_tx_congested)
287 ||(p_app_cb->ui_xmit_q.count >= llcp_cb.ll_tx_congest_start)
288 ||(llcp_cb.overall_tx_congested)
289 ||(llcp_cb.total_tx_ui_pdu >= llcp_cb.max_num_ll_tx_buff) )
290 {
291 /* set congested here so overall congestion check routine will not report event again, */
292 /* or notify uncongestion later */
293 p_app_cb->is_ui_tx_congested = TRUE;
294
295 LLCP_TRACE_WARNING2 ("Logical link (SAP=0x%X) congested: ui_xmit_q.count=%d",
296 ssap, p_app_cb->ui_xmit_q.count);
297
298 status = LLCP_STATUS_CONGESTED;
299 }
300
301 return status;
302 }
303
304 /*******************************************************************************
305 **
306 ** Function llcp_util_send_disc
307 **
308 ** Description Send DISC PDU
309 **
310 ** Returns void
311 **
312 *******************************************************************************/
llcp_util_send_disc(UINT8 dsap,UINT8 ssap)313 void llcp_util_send_disc (UINT8 dsap, UINT8 ssap)
314 {
315 BT_HDR *p_msg;
316 UINT8 *p;
317
318 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
319
320 if (p_msg)
321 {
322 p_msg->len = LLCP_PDU_DISC_SIZE;
323 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
324
325 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
326 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (dsap, LLCP_PDU_DISC_TYPE, ssap));
327
328 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
329 llcp_link_check_send_data ();
330 }
331 }
332
333 /*******************************************************************************
334 **
335 ** Function llcp_util_allocate_data_link
336 **
337 ** Description Allocate tLLCP_DLCB for data link connection
338 **
339 ** Returns tLLCP_DLCB *
340 **
341 ******************************************************************************/
llcp_util_allocate_data_link(UINT8 reg_sap,UINT8 remote_sap)342 tLLCP_DLCB *llcp_util_allocate_data_link (UINT8 reg_sap, UINT8 remote_sap)
343 {
344 tLLCP_DLCB *p_dlcb = NULL;
345 int idx;
346
347 LLCP_TRACE_DEBUG2 ("llcp_util_allocate_data_link (): reg_sap = 0x%x, remote_sap = 0x%x",
348 reg_sap, remote_sap);
349
350 for (idx = 0; idx < LLCP_MAX_DATA_LINK; idx++)
351 {
352 if (llcp_cb.dlcb[idx].state == LLCP_DLC_STATE_IDLE)
353 {
354 p_dlcb = &(llcp_cb.dlcb[idx]);
355
356 memset (p_dlcb, 0, sizeof (tLLCP_DLCB));
357 break;
358 }
359 }
360
361 if (!p_dlcb)
362 {
363 LLCP_TRACE_ERROR0 ("llcp_util_allocate_data_link (): Out of DLCB");
364 }
365 else
366 {
367 p_dlcb->p_app_cb = llcp_util_get_app_cb (reg_sap);
368 p_dlcb->local_sap = reg_sap;
369 p_dlcb->remote_sap = remote_sap;
370 p_dlcb->timer.param = (TIMER_PARAM_TYPE) p_dlcb;
371
372 /* this is for inactivity timer and congestion control. */
373 llcp_cb.num_data_link_connection++;
374
375 LLCP_TRACE_DEBUG3 ("llcp_util_allocate_data_link (): local_sap = 0x%x, remote_sap = 0x%x, num_data_link_connection = %d",
376 p_dlcb->local_sap, p_dlcb->remote_sap, llcp_cb.num_data_link_connection);
377 }
378 return p_dlcb;
379 }
380
381 /*******************************************************************************
382 **
383 ** Function llcp_util_deallocate_data_link
384 **
385 ** Description Deallocate tLLCP_DLCB
386 **
387 ** Returns void
388 **
389 ******************************************************************************/
llcp_util_deallocate_data_link(tLLCP_DLCB * p_dlcb)390 void llcp_util_deallocate_data_link (tLLCP_DLCB *p_dlcb)
391 {
392 if (p_dlcb)
393 {
394 LLCP_TRACE_DEBUG1 ("llcp_util_deallocate_data_link (): local_sap = 0x%x", p_dlcb->local_sap);
395
396 if (p_dlcb->state != LLCP_DLC_STATE_IDLE)
397 {
398 nfc_stop_quick_timer (&p_dlcb->timer);
399 llcp_dlc_flush_q (p_dlcb);
400
401 p_dlcb->state = LLCP_DLC_STATE_IDLE;
402
403 if (llcp_cb.num_data_link_connection > 0)
404 {
405 llcp_cb.num_data_link_connection--;
406 }
407
408 LLCP_TRACE_DEBUG1 ("llcp_util_deallocate_data_link (): num_data_link_connection = %d", llcp_cb.num_data_link_connection);
409 }
410 }
411 }
412
413 /*******************************************************************************
414 **
415 ** Function llcp_util_send_connect
416 **
417 ** Description Send CONNECT PDU
418 **
419 ** Returns tLLCP_STATUS
420 **
421 ******************************************************************************/
llcp_util_send_connect(tLLCP_DLCB * p_dlcb,tLLCP_CONNECTION_PARAMS * p_params)422 tLLCP_STATUS llcp_util_send_connect (tLLCP_DLCB *p_dlcb, tLLCP_CONNECTION_PARAMS *p_params)
423 {
424 BT_HDR *p_msg;
425 UINT8 *p;
426 UINT16 miu_len = 0, rw_len = 0, sn_len = 0;
427
428 if (p_params->miu != LLCP_DEFAULT_MIU)
429 {
430 miu_len = 4; /* TYPE, LEN, 2 bytes MIU */
431 }
432 if (p_params->rw != LLCP_DEFAULT_RW)
433 {
434 rw_len = 3; /* TYPE, LEN, 1 byte RW */
435 p_params->rw &= 0x0F; /* only 4 bits */
436 }
437 if ((strlen (p_params->sn)) && (p_dlcb->remote_sap == LLCP_SAP_SDP))
438 {
439 sn_len = (UINT16) (2 + strlen (p_params->sn)); /* TYPE, LEN, SN */
440 }
441
442 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
443
444 if (p_msg)
445 {
446 p_msg->len = LLCP_PDU_HEADER_SIZE + miu_len + rw_len + sn_len;
447 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
448
449 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
450
451 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (p_dlcb->remote_sap, LLCP_PDU_CONNECT_TYPE, p_dlcb->local_sap));
452
453 if (miu_len)
454 {
455 UINT8_TO_BE_STREAM (p, LLCP_MIUX_TYPE);
456 UINT8_TO_BE_STREAM (p, LLCP_MIUX_LEN);
457 UINT16_TO_BE_STREAM (p, p_params->miu - LLCP_DEFAULT_MIU);
458 }
459
460 if (rw_len)
461 {
462 UINT8_TO_BE_STREAM (p, LLCP_RW_TYPE);
463 UINT8_TO_BE_STREAM (p, LLCP_RW_LEN);
464 UINT8_TO_BE_STREAM (p, p_params->rw);
465 }
466
467 if (sn_len)
468 {
469 UINT8_TO_BE_STREAM (p, LLCP_SN_TYPE);
470 UINT8_TO_BE_STREAM (p, sn_len - 2);
471 memcpy (p, p_params->sn, sn_len - 2);
472 }
473
474 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
475 llcp_link_check_send_data ();
476
477 return LLCP_STATUS_SUCCESS;
478 }
479
480 return LLCP_STATUS_FAIL;
481 }
482
483 /*******************************************************************************
484 **
485 ** Function llcp_util_parse_connect
486 **
487 ** Description Parse CONNECT PDU
488 **
489 ** Returns tLLCP_STATUS
490 **
491 *******************************************************************************/
llcp_util_parse_connect(UINT8 * p_bytes,UINT16 length,tLLCP_CONNECTION_PARAMS * p_params)492 tLLCP_STATUS llcp_util_parse_connect (UINT8 *p_bytes, UINT16 length, tLLCP_CONNECTION_PARAMS *p_params)
493 {
494 UINT8 param_type, param_len, *p = p_bytes;
495
496 p_params->miu = LLCP_DEFAULT_MIU;
497 p_params->rw = LLCP_DEFAULT_RW;
498 p_params->sn[0] = 0;
499 p_params->sn[1] = 0;
500
501 while (length)
502 {
503 BE_STREAM_TO_UINT8 (param_type, p);
504 length--;
505
506 switch (param_type)
507 {
508 case LLCP_MIUX_TYPE:
509 BE_STREAM_TO_UINT8 (param_len, p);
510 BE_STREAM_TO_UINT16 (p_params->miu, p);
511 p_params->miu &= LLCP_MIUX_MASK;
512 p_params->miu += LLCP_DEFAULT_MIU;
513
514 LLCP_TRACE_DEBUG1 ("llcp_util_parse_connect (): LLCP_MIUX_TYPE:%d", p_params->miu);
515 break;
516
517 case LLCP_RW_TYPE:
518 BE_STREAM_TO_UINT8 (param_len, p);
519 BE_STREAM_TO_UINT8 (p_params->rw, p);
520 p_params->rw &= 0x0F;
521
522 LLCP_TRACE_DEBUG1 ("llcp_util_parse_connect (): LLCP_RW_TYPE:%d", p_params->rw);
523 break;
524
525 case LLCP_SN_TYPE:
526 BE_STREAM_TO_UINT8 (param_len, p);
527
528 if (param_len == 0)
529 {
530 /* indicate that SN type is included without SN */
531 p_params->sn[1] = LLCP_SN_TYPE;
532 }
533 else if (param_len <= LLCP_MAX_SN_LEN)
534 {
535 memcpy (p_params->sn, p, param_len);
536 p_params->sn[param_len] = 0;
537 }
538 else
539 {
540 memcpy (p_params->sn, p, LLCP_MAX_SN_LEN);
541 p_params->sn[LLCP_MAX_SN_LEN] = 0;
542 }
543 p += param_len;
544
545 LLCP_TRACE_DEBUG1 ("llcp_util_parse_connect (): LLCP_SN_TYPE:<%s>", p_params->sn);
546 break;
547
548 default:
549 LLCP_TRACE_ERROR1 ("llcp_util_parse_connect (): Unexpected type 0x%x", param_type);
550 BE_STREAM_TO_UINT8 (param_len, p);
551 p += param_len;
552 break;
553 }
554
555 /* check remaining lengh */
556 if (length >= param_len + 1)
557 {
558 length -= param_len + 1;
559 }
560 else
561 {
562 LLCP_TRACE_ERROR0 ("llcp_util_parse_connect (): Bad LTV's");
563 return LLCP_STATUS_FAIL;
564 }
565 }
566 return LLCP_STATUS_SUCCESS;
567 }
568
569 /*******************************************************************************
570 **
571 ** Function llcp_util_send_cc
572 **
573 ** Description Send CC PDU
574 **
575 ** Returns tLLCP_STATUS
576 **
577 ******************************************************************************/
llcp_util_send_cc(tLLCP_DLCB * p_dlcb,tLLCP_CONNECTION_PARAMS * p_params)578 tLLCP_STATUS llcp_util_send_cc (tLLCP_DLCB *p_dlcb, tLLCP_CONNECTION_PARAMS *p_params)
579 {
580 BT_HDR *p_msg;
581 UINT8 *p;
582 UINT16 miu_len = 0, rw_len = 0;
583
584 if (p_params->miu != LLCP_DEFAULT_MIU)
585 {
586 miu_len = 4;
587 }
588 if (p_params->rw != LLCP_DEFAULT_RW)
589 {
590 rw_len = 3;
591 p_params->rw &= 0x0F;
592 }
593
594 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
595
596 if (p_msg)
597 {
598 p_msg->len = LLCP_PDU_HEADER_SIZE + miu_len + rw_len;
599 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
600
601 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
602
603 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (p_dlcb->remote_sap, LLCP_PDU_CC_TYPE, p_dlcb->local_sap));
604
605 if (miu_len)
606 {
607 UINT8_TO_BE_STREAM (p, LLCP_MIUX_TYPE);
608 UINT8_TO_BE_STREAM (p, LLCP_MIUX_LEN);
609 UINT16_TO_BE_STREAM (p, p_params->miu - LLCP_DEFAULT_MIU);
610 }
611
612 if (rw_len)
613 {
614 UINT8_TO_BE_STREAM (p, LLCP_RW_TYPE);
615 UINT8_TO_BE_STREAM (p, LLCP_RW_LEN);
616 UINT8_TO_BE_STREAM (p, p_params->rw);
617 }
618
619 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
620 llcp_link_check_send_data ();
621
622 return LLCP_STATUS_SUCCESS;
623 }
624
625 return LLCP_STATUS_FAIL;
626 }
627
628 /*******************************************************************************
629 **
630 ** Function llcp_util_parse_cc
631 **
632 ** Description Parse CC PDU
633 **
634 ** Returns tLLCP_STATUS
635 **
636 *******************************************************************************/
llcp_util_parse_cc(UINT8 * p_bytes,UINT16 length,UINT16 * p_miu,UINT8 * p_rw)637 tLLCP_STATUS llcp_util_parse_cc (UINT8 *p_bytes, UINT16 length, UINT16 *p_miu, UINT8 *p_rw)
638 {
639 UINT8 param_type, param_len, *p = p_bytes;
640
641 *p_miu = LLCP_DEFAULT_MIU;
642 *p_rw = LLCP_DEFAULT_RW;
643
644 while (length)
645 {
646 BE_STREAM_TO_UINT8 (param_type, p);
647 length--;
648
649 switch (param_type)
650 {
651 case LLCP_MIUX_TYPE:
652 BE_STREAM_TO_UINT8 (param_len, p);
653 BE_STREAM_TO_UINT16 ((*p_miu), p);
654 (*p_miu) &= LLCP_MIUX_MASK;
655 (*p_miu) += LLCP_DEFAULT_MIU;
656
657 LLCP_TRACE_DEBUG1 ("llcp_util_parse_cc (): LLCP_MIUX_TYPE:%d", *p_miu);
658 break;
659
660 case LLCP_RW_TYPE:
661 BE_STREAM_TO_UINT8 (param_len, p);
662 BE_STREAM_TO_UINT8 ((*p_rw), p);
663 (*p_rw) &= 0x0F;
664
665 LLCP_TRACE_DEBUG1 ("llcp_util_parse_cc (): LLCP_RW_TYPE:%d", *p_rw);
666 break;
667
668 default:
669 LLCP_TRACE_ERROR1 ("llcp_util_parse_cc (): Unexpected type 0x%x", param_type);
670 BE_STREAM_TO_UINT8 (param_len, p);
671 p += param_len;
672 break;
673 }
674
675 if (length >= param_len + 1)
676 length -= param_len + 1;
677 else
678 {
679 LLCP_TRACE_ERROR0 ("llcp_util_parse_cc (): Bad LTV's");
680 return LLCP_STATUS_FAIL;
681 }
682 }
683 return LLCP_STATUS_SUCCESS;
684 }
685
686 /*******************************************************************************
687 **
688 ** Function llcp_util_send_dm
689 **
690 ** Description Send DM PDU
691 **
692 ** Returns void
693 **
694 *******************************************************************************/
llcp_util_send_dm(UINT8 dsap,UINT8 ssap,UINT8 reason)695 void llcp_util_send_dm (UINT8 dsap, UINT8 ssap, UINT8 reason)
696 {
697 BT_HDR *p_msg;
698 UINT8 *p;
699
700 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
701
702 if (p_msg)
703 {
704 p_msg->len = LLCP_PDU_DM_SIZE;
705 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
706
707 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
708 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (dsap, LLCP_PDU_DM_TYPE, ssap));
709 UINT8_TO_BE_STREAM (p, reason);
710
711 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
712 llcp_link_check_send_data ();
713 }
714 }
715
716 /*******************************************************************************
717 **
718 ** Function llcp_util_build_info_pdu
719 **
720 ** Description Add DSAP, PTYPE, SSAP and sequence numbers and update local ack
721 ** sequence
722 **
723 ** Returns void
724 **
725 *******************************************************************************/
llcp_util_build_info_pdu(tLLCP_DLCB * p_dlcb,BT_HDR * p_msg)726 void llcp_util_build_info_pdu (tLLCP_DLCB *p_dlcb, BT_HDR *p_msg)
727 {
728 UINT8 *p;
729 UINT8 rcv_seq;
730
731 p_msg->offset -= LLCP_PDU_HEADER_SIZE + LLCP_SEQUENCE_SIZE;
732 p_msg->len += LLCP_PDU_HEADER_SIZE + LLCP_SEQUENCE_SIZE;
733 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
734
735 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (p_dlcb->remote_sap, LLCP_PDU_I_TYPE, p_dlcb->local_sap));
736
737 /* if local_busy or rx congested then do not update receive sequence number to flow off */
738 if ( (p_dlcb->local_busy)
739 ||(p_dlcb->is_rx_congested)
740 ||(llcp_cb.overall_rx_congested) )
741 {
742 rcv_seq = p_dlcb->sent_ack_seq;
743 }
744 else
745 {
746 p_dlcb->sent_ack_seq = p_dlcb->next_rx_seq;
747 rcv_seq = p_dlcb->sent_ack_seq;
748 }
749 UINT8_TO_BE_STREAM (p, LLCP_GET_SEQUENCE (p_dlcb->next_tx_seq, rcv_seq));
750 }
751
752 /*******************************************************************************
753 **
754 ** Function llcp_util_send_frmr
755 **
756 ** Description Send FRMR PDU
757 **
758 ** Returns tLLCP_STATUS
759 **
760 *******************************************************************************/
llcp_util_send_frmr(tLLCP_DLCB * p_dlcb,UINT8 flags,UINT8 ptype,UINT8 sequence)761 tLLCP_STATUS llcp_util_send_frmr (tLLCP_DLCB *p_dlcb, UINT8 flags, UINT8 ptype, UINT8 sequence)
762 {
763 BT_HDR *p_msg;
764 UINT8 *p;
765
766 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
767
768 if (p_msg)
769 {
770 p_msg->len = LLCP_PDU_FRMR_SIZE;
771 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
772
773 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
774
775 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (p_dlcb->remote_sap, LLCP_PDU_FRMR_TYPE, p_dlcb->local_sap));
776 UINT8_TO_BE_STREAM (p, (flags << 4) | ptype);
777 UINT8_TO_BE_STREAM (p, sequence);
778 UINT8_TO_BE_STREAM (p, (p_dlcb->next_tx_seq << 4) | p_dlcb->next_rx_seq);
779 UINT8_TO_BE_STREAM (p, (p_dlcb->rcvd_ack_seq << 4) | p_dlcb->sent_ack_seq);
780
781 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
782 llcp_link_check_send_data ();
783
784 return LLCP_STATUS_SUCCESS;
785 }
786 else
787 {
788 LLCP_TRACE_ERROR0 ("llcp_util_send_frmr (): Out of resource");
789 return LLCP_STATUS_FAIL;
790 }
791 }
792
793 /*******************************************************************************
794 **
795 ** Function llcp_util_send_rr_rnr
796 **
797 ** Description Send RR or RNR PDU
798 **
799 ** Returns void
800 **
801 *******************************************************************************/
llcp_util_send_rr_rnr(tLLCP_DLCB * p_dlcb)802 void llcp_util_send_rr_rnr (tLLCP_DLCB *p_dlcb)
803 {
804 BT_HDR *p_msg;
805 UINT8 *p;
806 UINT8 pdu_type;
807 UINT8 pdu_size;
808 UINT8 rcv_seq;
809
810 /* if no indication of change in local busy or rx congestion */
811 if ((p_dlcb->flags & LLCP_DATA_LINK_FLAG_PENDING_RR_RNR) == 0)
812 {
813 /* if all ack is sent */
814 if (p_dlcb->sent_ack_seq == p_dlcb->next_rx_seq)
815 {
816 /* we don't need to send RR/RNR */
817 return;
818 }
819 else
820 {
821 /* if rx flow off because of local busy or congestion */
822 if ( (p_dlcb->local_busy)
823 ||(p_dlcb->is_rx_congested)
824 ||(llcp_cb.overall_rx_congested) )
825 {
826 /* don't send RR/RNR */
827 return;
828 }
829 }
830 }
831
832 if ( (p_dlcb->local_busy)
833 ||(p_dlcb->is_rx_congested)
834 ||(llcp_cb.overall_rx_congested) )
835 {
836 LLCP_TRACE_DEBUG3 ("llcp_util_send_rr_rnr (): local_busy=%d,is_rx_congested=%d,overall_rx_congested=%d",
837 p_dlcb->local_busy, p_dlcb->is_rx_congested, llcp_cb.overall_rx_congested);
838
839 /* if local_busy or rx congested then do not update receive sequence number to flow off */
840 pdu_type = LLCP_PDU_RNR_TYPE;
841 pdu_size = LLCP_PDU_RNR_SIZE;
842 rcv_seq = p_dlcb->sent_ack_seq;
843 }
844 else
845 {
846 pdu_type = LLCP_PDU_RR_TYPE;
847 pdu_size = LLCP_PDU_RR_SIZE;
848
849 p_dlcb->sent_ack_seq = p_dlcb->next_rx_seq;
850 rcv_seq = p_dlcb->sent_ack_seq;
851 }
852
853 p_msg = (BT_HDR*) GKI_getpoolbuf (LLCP_POOL_ID);
854
855 if (p_msg)
856 {
857 p_dlcb->flags &= ~LLCP_DATA_LINK_FLAG_PENDING_RR_RNR;
858
859 p_msg->len = pdu_size;
860 p_msg->offset = NCI_MSG_OFFSET_SIZE + NCI_DATA_HDR_SIZE;
861
862 p = (UINT8 *) (p_msg + 1) + p_msg->offset;
863
864 UINT16_TO_BE_STREAM (p, LLCP_GET_PDU_HEADER (p_dlcb->remote_sap, pdu_type, p_dlcb->local_sap));
865
866 UINT8_TO_BE_STREAM (p, rcv_seq);
867
868 #if (BT_TRACE_VERBOSE == TRUE)
869 LLCP_TRACE_DEBUG5 ("LLCP TX - N(S,R):(NA,%d) V(S,SA,R,RA):(%d,%d,%d,%d)",
870 p_dlcb->next_rx_seq,
871 p_dlcb->next_tx_seq, p_dlcb->rcvd_ack_seq,
872 p_dlcb->next_rx_seq, p_dlcb->sent_ack_seq);
873 #endif
874 GKI_enqueue (&llcp_cb.lcb.sig_xmit_q, p_msg);
875 llcp_link_check_send_data ();
876 }
877 else
878 {
879 LLCP_TRACE_ERROR0 ("llcp_util_send_rr_rnr (): Out of resource");
880 }
881 }
882
883 /*******************************************************************************
884 **
885 ** Function llcp_util_get_app_cb
886 **
887 ** Description get pointer of application registered control block by SAP
888 **
889 ** Returns tLLCP_APP_CB *
890 **
891 *******************************************************************************/
llcp_util_get_app_cb(UINT8 local_sap)892 tLLCP_APP_CB *llcp_util_get_app_cb (UINT8 local_sap)
893 {
894 tLLCP_APP_CB *p_app_cb = NULL;
895
896 if (local_sap <= LLCP_UPPER_BOUND_WK_SAP)
897 {
898 if ((local_sap != LLCP_SAP_LM) && (local_sap < LLCP_MAX_WKS))
899 {
900 p_app_cb = &llcp_cb.wks_cb[local_sap];
901 }
902 }
903 else if (local_sap <= LLCP_UPPER_BOUND_SDP_SAP)
904 {
905 if (local_sap - LLCP_LOWER_BOUND_SDP_SAP < LLCP_MAX_SERVER)
906 {
907 p_app_cb = &llcp_cb.server_cb[local_sap - LLCP_LOWER_BOUND_SDP_SAP];
908 }
909 }
910 else if (local_sap <= LLCP_UPPER_BOUND_LOCAL_SAP)
911 {
912 if (local_sap - LLCP_LOWER_BOUND_LOCAL_SAP < LLCP_MAX_CLIENT)
913 {
914 p_app_cb = &llcp_cb.client_cb[local_sap - LLCP_LOWER_BOUND_LOCAL_SAP];
915 }
916 }
917
918 return (p_app_cb);
919 }
920