• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /******************************************************************************
2  *
3  *  Copyright 2009-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 #include <base/strings/stringprintf.h>
20 #include <string.h>
21 #include "bt_target.h"
22 #include "bt_utils.h"
23 #include "btm_int.h"
24 #include "btu.h"
25 #include "device/include/controller.h"
26 #include "gap_api.h"
27 #include "l2c_int.h"
28 #include "l2cdefs.h"
29 #include "osi/include/fixed_queue.h"
30 #include "osi/include/mutex.h"
31 
32 using base::StringPrintf;
33 
34 /* Define the GAP Connection Control Block */
35 typedef struct {
36 #define GAP_CCB_STATE_IDLE 0
37 #define GAP_CCB_STATE_LISTENING 1
38 #define GAP_CCB_STATE_CONN_SETUP 2
39 #define GAP_CCB_STATE_CFG_SETUP 3
40 #define GAP_CCB_STATE_WAIT_SEC 4
41 #define GAP_CCB_STATE_CONNECTED 5
42   uint8_t con_state;
43 
44 #define GAP_CCB_FLAGS_IS_ORIG 0x01
45 #define GAP_CCB_FLAGS_HIS_CFG_DONE 0x02
46 #define GAP_CCB_FLAGS_MY_CFG_DONE 0x04
47 #define GAP_CCB_FLAGS_SEC_DONE 0x08
48 #define GAP_CCB_FLAGS_CONN_DONE 0x0E
49   uint8_t con_flags;
50 
51   uint8_t service_id;     /* Used by BTM */
52   uint16_t gap_handle;    /* GAP handle */
53   uint16_t connection_id; /* L2CAP CID */
54   bool rem_addr_specified;
55   uint8_t chan_mode_mask; /* Supported channel modes (FCR) */
56   RawAddress rem_dev_address;
57   uint16_t psm;
58   uint16_t rem_mtu_size;
59 
60   bool is_congested;
61   fixed_queue_t* tx_queue; /* Queue of buffers waiting to be sent */
62   fixed_queue_t* rx_queue; /* Queue of buffers waiting to be read */
63 
64   uint32_t rx_queue_size; /* Total data count in rx_queue */
65 
66   tGAP_CONN_CALLBACK* p_callback; /* Users callback function */
67 
68   tL2CAP_CFG_INFO cfg;              /* Configuration */
69   tL2CAP_ERTM_INFO ertm_info;       /* Pools and modes for ertm */
70   tBT_TRANSPORT transport;          /* Transport channel BR/EDR or BLE */
71   tL2CAP_LE_CFG_INFO local_coc_cfg; /* local configuration for LE Coc */
72   tL2CAP_LE_CFG_INFO peer_coc_cfg;  /* local configuration for LE Coc */
73 } tGAP_CCB;
74 
75 typedef struct {
76   tL2CAP_APPL_INFO reg_info; /* L2CAP Registration info */
77   tGAP_CCB ccb_pool[GAP_MAX_CONNECTIONS];
78 } tGAP_CONN;
79 
80 namespace {
81 tGAP_CONN conn;
82 }
83 
84 /******************************************************************************/
85 /*            L O C A L    F U N C T I O N     P R O T O T Y P E S            */
86 /******************************************************************************/
87 static void gap_connect_ind(const RawAddress& bd_addr, uint16_t l2cap_cid,
88                             uint16_t psm, uint8_t l2cap_id);
89 static void gap_connect_cfm(uint16_t l2cap_cid, uint16_t result);
90 static void gap_config_ind(uint16_t l2cap_cid, tL2CAP_CFG_INFO* p_cfg);
91 static void gap_config_cfm(uint16_t l2cap_cid, tL2CAP_CFG_INFO* p_cfg);
92 static void gap_disconnect_ind(uint16_t l2cap_cid, bool ack_needed);
93 static void gap_data_ind(uint16_t l2cap_cid, BT_HDR* p_msg);
94 static void gap_congestion_ind(uint16_t lcid, bool is_congested);
95 static void gap_tx_complete_ind(uint16_t l2cap_cid, uint16_t sdu_sent);
96 static void gap_credits_received_cb(uint16_t l2cap_cid,
97                                     uint16_t credits_received,
98                                     uint16_t credit_count);
99 
100 static tGAP_CCB* gap_find_ccb_by_cid(uint16_t cid);
101 static tGAP_CCB* gap_find_ccb_by_handle(uint16_t handle);
102 static tGAP_CCB* gap_allocate_ccb(void);
103 static void gap_release_ccb(tGAP_CCB* p_ccb);
104 static void gap_checks_con_flags(tGAP_CCB* p_ccb);
105 
106 /*******************************************************************************
107  *
108  * Function         gap_conn_init
109  *
110  * Description      This function is called to initialize GAP connection
111  *                  management
112  *
113  * Returns          void
114  *
115  ******************************************************************************/
gap_conn_init(void)116 void gap_conn_init(void) {
117   memset(&conn, 0, sizeof(tGAP_CONN));
118   conn.reg_info.pL2CA_ConnectInd_Cb = gap_connect_ind;
119   conn.reg_info.pL2CA_ConnectCfm_Cb = gap_connect_cfm;
120   conn.reg_info.pL2CA_ConnectPnd_Cb = NULL;
121   conn.reg_info.pL2CA_ConfigInd_Cb = gap_config_ind;
122   conn.reg_info.pL2CA_ConfigCfm_Cb = gap_config_cfm;
123   conn.reg_info.pL2CA_DisconnectInd_Cb = gap_disconnect_ind;
124   conn.reg_info.pL2CA_DisconnectCfm_Cb = NULL;
125   conn.reg_info.pL2CA_QoSViolationInd_Cb = NULL;
126   conn.reg_info.pL2CA_DataInd_Cb = gap_data_ind;
127   conn.reg_info.pL2CA_CongestionStatus_Cb = gap_congestion_ind;
128   conn.reg_info.pL2CA_TxComplete_Cb = gap_tx_complete_ind;
129   conn.reg_info.pL2CA_CreditsReceived_Cb = gap_credits_received_cb;
130 }
131 
132 /*******************************************************************************
133  *
134  * Function         GAP_ConnOpen
135  *
136  * Description      This function is called to open an L2CAP connection.
137  *
138  * Parameters:      is_server   - If true, the connection is not created
139  *                                but put into a "listen" mode waiting for
140  *                                the remote side to connect.
141  *
142  *                  service_id  - Unique service ID from
143  *                                BTM_SEC_SERVICE_FIRST_EMPTY (6)
144  *                                to BTM_SEC_MAX_SERVICE_RECORDS (32)
145  *
146  *                  p_rem_bda   - Pointer to remote BD Address.
147  *                                If a server, and we don't care about the
148  *                                remote BD Address, then NULL should be passed.
149  *
150  *                  psm         - the PSM used for the connection
151  *                  le_mps      - Maximum PDU Size for LE CoC
152  *
153  *                  p_config    - Optional pointer to configuration structure.
154  *                                If NULL, the default GAP configuration will
155  *                                be used.
156  *
157  *                  security    - security flags
158  *                  chan_mode_mask - (GAP_FCR_CHAN_OPT_BASIC,
159  *                                    GAP_FCR_CHAN_OPT_ERTM,
160  *                                    GAP_FCR_CHAN_OPT_STREAM)
161  *
162  *                  p_cb        - Pointer to callback function for events.
163  *
164  * Returns          handle of the connection if successful, else
165  *                            GAP_INVALID_HANDLE
166  *
167  ******************************************************************************/
GAP_ConnOpen(const char * p_serv_name,uint8_t service_id,bool is_server,const RawAddress * p_rem_bda,uint16_t psm,uint16_t le_mps,tL2CAP_CFG_INFO * p_cfg,tL2CAP_ERTM_INFO * ertm_info,uint16_t security,uint8_t chan_mode_mask,tGAP_CONN_CALLBACK * p_cb,tBT_TRANSPORT transport)168 uint16_t GAP_ConnOpen(const char* p_serv_name, uint8_t service_id,
169                       bool is_server, const RawAddress* p_rem_bda, uint16_t psm,
170                       uint16_t le_mps, tL2CAP_CFG_INFO* p_cfg,
171                       tL2CAP_ERTM_INFO* ertm_info, uint16_t security,
172                       uint8_t chan_mode_mask, tGAP_CONN_CALLBACK* p_cb,
173                       tBT_TRANSPORT transport) {
174   tGAP_CCB* p_ccb;
175   uint16_t cid;
176 
177   DVLOG(1) << "GAP_CONN - Open Request";
178 
179   /* Allocate a new CCB. Return if none available. */
180   p_ccb = gap_allocate_ccb();
181   if (p_ccb == NULL) return (GAP_INVALID_HANDLE);
182 
183   /* update the transport */
184   p_ccb->transport = transport;
185 
186   /* The service_id must be set before calling gap_release_ccb(). */
187   p_ccb->service_id = service_id;
188 
189   /* If caller specified a BD address, save it */
190   if (p_rem_bda) {
191     /* the bd addr is not RawAddress::kAny, then a bd address was specified */
192     if (*p_rem_bda != RawAddress::kAny) p_ccb->rem_addr_specified = true;
193 
194     p_ccb->rem_dev_address = *p_rem_bda;
195   } else if (!is_server) {
196     /* remore addr is not specified and is not a server -> bad */
197     return (GAP_INVALID_HANDLE);
198   }
199 
200   /* A client MUST have specified a bd addr to connect with */
201   if (!p_ccb->rem_addr_specified && !is_server) {
202     gap_release_ccb(p_ccb);
203     LOG(ERROR)
204         << "GAP ERROR: Client must specify a remote BD ADDR to connect to!";
205     return (GAP_INVALID_HANDLE);
206   }
207 
208   /* Check if configuration was specified */
209   if (p_cfg) p_ccb->cfg = *p_cfg;
210 
211   /* Configure L2CAP COC, if transport is LE */
212   if (transport == BT_TRANSPORT_LE) {
213     p_ccb->local_coc_cfg.credits = L2CAP_LE_CREDIT_DEFAULT;
214     p_ccb->local_coc_cfg.mtu = p_cfg->mtu;
215 
216     uint16_t max_mps = controller_get_interface()->get_acl_data_size_ble();
217     if (le_mps > max_mps) {
218       LOG(INFO) << "Limiting MPS to one buffer size - " << max_mps;
219       le_mps = max_mps;
220     }
221     p_ccb->local_coc_cfg.mps = le_mps;
222 
223     VLOG(2) << __func__ << ": credits=" << p_ccb->local_coc_cfg.credits
224             << ", mps=" << p_ccb->local_coc_cfg.mps
225             << ", mtu=" << p_ccb->local_coc_cfg.mtu;
226   }
227 
228   p_ccb->p_callback = p_cb;
229 
230   /* If originator, use a dynamic PSM */
231   if (!is_server)
232     conn.reg_info.pL2CA_ConnectInd_Cb = NULL;
233   else
234     conn.reg_info.pL2CA_ConnectInd_Cb = gap_connect_ind;
235 
236   /* Fill in eL2CAP parameter data */
237   if (p_ccb->cfg.fcr_present) {
238     if (ertm_info == NULL) {
239       p_ccb->ertm_info.preferred_mode = p_ccb->cfg.fcr.mode;
240       p_ccb->ertm_info.user_rx_buf_size = GAP_DATA_BUF_SIZE;
241       p_ccb->ertm_info.user_tx_buf_size = GAP_DATA_BUF_SIZE;
242       p_ccb->ertm_info.fcr_rx_buf_size = L2CAP_INVALID_ERM_BUF_SIZE;
243       p_ccb->ertm_info.fcr_tx_buf_size = L2CAP_INVALID_ERM_BUF_SIZE;
244     } else {
245       p_ccb->ertm_info = *ertm_info;
246     }
247   }
248 
249   /* Register the PSM with L2CAP */
250   if (transport == BT_TRANSPORT_BR_EDR) {
251     p_ccb->psm = L2CA_Register(psm, &conn.reg_info, false /* enable_snoop */,
252                                &p_ccb->ertm_info);
253     if (p_ccb->psm == 0) {
254       LOG(ERROR) << StringPrintf("%s: Failure registering PSM 0x%04x", __func__,
255                                  psm);
256       gap_release_ccb(p_ccb);
257       return (GAP_INVALID_HANDLE);
258     }
259   }
260 
261   if (transport == BT_TRANSPORT_LE) {
262     p_ccb->psm = L2CA_RegisterLECoc(psm, (tL2CAP_APPL_INFO*)&conn.reg_info);
263     if (p_ccb->psm == 0) {
264       LOG(ERROR) << StringPrintf("%s: Failure registering PSM 0x%04x", __func__,
265                                  psm);
266       gap_release_ccb(p_ccb);
267       return (GAP_INVALID_HANDLE);
268     }
269   }
270 
271   /* Register with Security Manager for the specific security level */
272   if (!BTM_SetSecurityLevel((uint8_t)!is_server, p_serv_name, p_ccb->service_id,
273                             security, p_ccb->psm, 0, 0)) {
274     LOG(ERROR) << "GAP_CONN - Security Error";
275     gap_release_ccb(p_ccb);
276     return (GAP_INVALID_HANDLE);
277   }
278 
279   /* optional FCR channel modes */
280   if (ertm_info != NULL) {
281     p_ccb->ertm_info.allowed_modes =
282         (chan_mode_mask) ? chan_mode_mask : (uint8_t)L2CAP_FCR_CHAN_OPT_BASIC;
283   }
284 
285   if (is_server) {
286     p_ccb->con_flags |=
287         GAP_CCB_FLAGS_SEC_DONE; /* assume btm/l2cap would handle it */
288     p_ccb->con_state = GAP_CCB_STATE_LISTENING;
289     return (p_ccb->gap_handle);
290   } else {
291     /* We are the originator of this connection */
292     p_ccb->con_flags = GAP_CCB_FLAGS_IS_ORIG;
293 
294     /* Transition to the next appropriate state, waiting for connection confirm.
295      */
296     p_ccb->con_state = GAP_CCB_STATE_CONN_SETUP;
297 
298     /* mark security done flag, when security is not required */
299     if ((security & (BTM_SEC_OUT_AUTHORIZE | BTM_SEC_OUT_AUTHENTICATE |
300                      BTM_SEC_OUT_ENCRYPT)) == 0)
301       p_ccb->con_flags |= GAP_CCB_FLAGS_SEC_DONE;
302 
303     /* Check if L2CAP started the connection process */
304     if (p_rem_bda && (transport == BT_TRANSPORT_BR_EDR)) {
305       cid = L2CA_ErtmConnectReq(p_ccb->psm, *p_rem_bda, &p_ccb->ertm_info);
306       if (cid != 0) {
307         p_ccb->connection_id = cid;
308         return (p_ccb->gap_handle);
309       }
310     }
311 
312     if (p_rem_bda && (transport == BT_TRANSPORT_LE)) {
313       cid = L2CA_ConnectLECocReq(p_ccb->psm, *p_rem_bda, &p_ccb->local_coc_cfg);
314       if (cid != 0) {
315         p_ccb->connection_id = cid;
316         return (p_ccb->gap_handle);
317       }
318     }
319 
320     gap_release_ccb(p_ccb);
321     return (GAP_INVALID_HANDLE);
322   }
323 }
324 
325 /*******************************************************************************
326  *
327  * Function         GAP_ConnClose
328  *
329  * Description      This function is called to close a connection.
330  *
331  * Parameters:      handle - Handle of the connection returned by GAP_ConnOpen
332  *
333  * Returns          BT_PASS             - closed OK
334  *                  GAP_ERR_BAD_HANDLE  - invalid handle
335  *
336  ******************************************************************************/
GAP_ConnClose(uint16_t gap_handle)337 uint16_t GAP_ConnClose(uint16_t gap_handle) {
338   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
339 
340   DVLOG(1) << StringPrintf("GAP_CONN - close  handle: 0x%x", gap_handle);
341 
342   if (p_ccb) {
343     /* Check if we have a connection ID */
344     if (p_ccb->con_state != GAP_CCB_STATE_LISTENING)
345       L2CA_DisconnectReq(p_ccb->connection_id);
346 
347     gap_release_ccb(p_ccb);
348 
349     return (BT_PASS);
350   }
351 
352   return (GAP_ERR_BAD_HANDLE);
353 }
354 
355 /*******************************************************************************
356  *
357  * Function         GAP_ConnReadData
358  *
359  * Description      Normally not GKI aware application will call this function
360  *                  after receiving GAP_EVT_RXDATA event.
361  *
362  * Parameters:      handle      - Handle of the connection returned in the Open
363  *                  p_data      - Data area
364  *                  max_len     - Byte count requested
365  *                  p_len       - Byte count received
366  *
367  * Returns          BT_PASS             - data read
368  *                  GAP_ERR_BAD_HANDLE  - invalid handle
369  *                  GAP_NO_DATA_AVAIL   - no data available
370  *
371  ******************************************************************************/
GAP_ConnReadData(uint16_t gap_handle,uint8_t * p_data,uint16_t max_len,uint16_t * p_len)372 uint16_t GAP_ConnReadData(uint16_t gap_handle, uint8_t* p_data,
373                           uint16_t max_len, uint16_t* p_len) {
374   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
375   uint16_t copy_len;
376 
377   if (!p_ccb) return (GAP_ERR_BAD_HANDLE);
378 
379   *p_len = 0;
380 
381   if (fixed_queue_is_empty(p_ccb->rx_queue)) return (GAP_NO_DATA_AVAIL);
382 
383   mutex_global_lock();
384 
385   while (max_len) {
386     BT_HDR* p_buf =
387         static_cast<BT_HDR*>(fixed_queue_try_peek_first(p_ccb->rx_queue));
388     if (p_buf == NULL) break;
389 
390     copy_len = (p_buf->len > max_len) ? max_len : p_buf->len;
391     max_len -= copy_len;
392     *p_len += copy_len;
393     if (p_data) {
394       memcpy(p_data, (uint8_t*)(p_buf + 1) + p_buf->offset, copy_len);
395       p_data += copy_len;
396     }
397 
398     if (p_buf->len > copy_len) {
399       p_buf->offset += copy_len;
400       p_buf->len -= copy_len;
401       break;
402     }
403     osi_free(fixed_queue_try_dequeue(p_ccb->rx_queue));
404   }
405 
406   p_ccb->rx_queue_size -= *p_len;
407 
408   mutex_global_unlock();
409 
410   DVLOG(1) << StringPrintf(
411       "GAP_ConnReadData - rx_queue_size left=%d, *p_len=%d",
412       p_ccb->rx_queue_size, *p_len);
413 
414   return (BT_PASS);
415 }
416 
417 /*******************************************************************************
418  *
419  * Function         GAP_GetRxQueueCnt
420  *
421  * Description      This function return number of bytes on the rx queue.
422  *
423  * Parameters:      handle     - Handle returned in the GAP_ConnOpen
424  *                  p_rx_queue_count - Pointer to return queue count in.
425  *
426  *
427  ******************************************************************************/
GAP_GetRxQueueCnt(uint16_t handle,uint32_t * p_rx_queue_count)428 int GAP_GetRxQueueCnt(uint16_t handle, uint32_t* p_rx_queue_count) {
429   tGAP_CCB* p_ccb;
430   int rc = BT_PASS;
431 
432   /* Check that handle is valid */
433   if (handle < GAP_MAX_CONNECTIONS) {
434     p_ccb = &conn.ccb_pool[handle];
435 
436     if (p_ccb->con_state == GAP_CCB_STATE_CONNECTED) {
437       *p_rx_queue_count = p_ccb->rx_queue_size;
438     } else
439       rc = GAP_INVALID_HANDLE;
440   } else
441     rc = GAP_INVALID_HANDLE;
442 
443   DVLOG(1) << StringPrintf("GAP_GetRxQueueCnt - rc = 0x%04x, rx_queue_count=%d",
444                            rc, *p_rx_queue_count);
445 
446   return (rc);
447 }
448 
449 /*******************************************************************************
450  *
451  * Function         GAP_ConnBTRead
452  *
453  * Description      Bluetooth-aware applications will call this function after
454  *                  receiving GAP_EVT_RXDATA event.
455  *
456  * Parameters:      handle      - Handle of the connection returned in the Open
457  *                  pp_buf      - pointer to address of buffer with data,
458  *
459  * Returns          BT_PASS             - data read
460  *                  GAP_ERR_BAD_HANDLE  - invalid handle
461  *                  GAP_NO_DATA_AVAIL   - no data available
462  *
463  ******************************************************************************/
GAP_ConnBTRead(uint16_t gap_handle,BT_HDR ** pp_buf)464 uint16_t GAP_ConnBTRead(uint16_t gap_handle, BT_HDR** pp_buf) {
465   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
466   BT_HDR* p_buf;
467 
468   if (!p_ccb) return (GAP_ERR_BAD_HANDLE);
469 
470   p_buf = (BT_HDR*)fixed_queue_try_dequeue(p_ccb->rx_queue);
471 
472   if (p_buf) {
473     *pp_buf = p_buf;
474 
475     p_ccb->rx_queue_size -= p_buf->len;
476     return (BT_PASS);
477   } else {
478     *pp_buf = NULL;
479     return (GAP_NO_DATA_AVAIL);
480   }
481 }
482 
483 /* Try to write the queued data to l2ca. Return true on success, or if queue is
484  * congested. False if error occured when writing. */
gap_try_write_queued_data(tGAP_CCB * p_ccb)485 static bool gap_try_write_queued_data(tGAP_CCB* p_ccb) {
486   if (p_ccb->is_congested) return true;
487 
488   /* Send the buffer through L2CAP */
489   BT_HDR* p_buf;
490   while ((p_buf = (BT_HDR*)fixed_queue_try_dequeue(p_ccb->tx_queue)) != NULL) {
491     uint8_t status = L2CA_DataWrite(p_ccb->connection_id, p_buf);
492 
493     if (status == L2CAP_DW_CONGESTED) {
494       p_ccb->is_congested = true;
495       return true;
496     } else if (status != L2CAP_DW_SUCCESS)
497       return false;
498   }
499   return true;
500 }
501 
502 /*******************************************************************************
503  *
504  * Function         GAP_ConnWriteData
505  *
506  * Description      Normally not GKI aware application will call this function
507  *                  to send data to the connection.
508  *
509  * Parameters:      handle      - Handle of the connection returned in the Open
510  *                  msg         - pointer to single SDU to send. This function
511  *                                will take ownership of it.
512  *
513  * Returns          BT_PASS                 - data read
514  *                  GAP_ERR_BAD_HANDLE      - invalid handle
515  *                  GAP_ERR_BAD_STATE       - connection not established
516  *                  GAP_CONGESTION          - system is congested
517  *
518  ******************************************************************************/
GAP_ConnWriteData(uint16_t gap_handle,BT_HDR * msg)519 uint16_t GAP_ConnWriteData(uint16_t gap_handle, BT_HDR* msg) {
520   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
521 
522   if (!p_ccb) {
523     osi_free(msg);
524     return GAP_ERR_BAD_HANDLE;
525   }
526 
527   if (p_ccb->con_state != GAP_CCB_STATE_CONNECTED) {
528     osi_free(msg);
529     return GAP_ERR_BAD_STATE;
530   }
531 
532   if (msg->len > p_ccb->rem_mtu_size) {
533     osi_free(msg);
534     return GAP_ERR_ILL_PARM;
535   }
536 
537   DVLOG(1) << StringPrintf("GAP_WriteData %d bytes", msg->len);
538 
539   fixed_queue_enqueue(p_ccb->tx_queue, msg);
540 
541   if (!gap_try_write_queued_data(p_ccb)) return GAP_ERR_BAD_STATE;
542 
543   return (BT_PASS);
544 }
545 
546 /*******************************************************************************
547  *
548  * Function         GAP_ConnReconfig
549  *
550  * Description      Applications can call this function to reconfigure the
551  *                  connection.
552  *
553  * Parameters:      handle      - Handle of the connection
554  *                  p_cfg       - Pointer to new configuration
555  *
556  * Returns          BT_PASS                 - config process started
557  *                  GAP_ERR_BAD_HANDLE      - invalid handle
558  *
559  ******************************************************************************/
GAP_ConnReconfig(uint16_t gap_handle,tL2CAP_CFG_INFO * p_cfg)560 uint16_t GAP_ConnReconfig(uint16_t gap_handle, tL2CAP_CFG_INFO* p_cfg) {
561   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
562 
563   if (!p_ccb) return (GAP_ERR_BAD_HANDLE);
564 
565   p_ccb->cfg = *p_cfg;
566 
567   if (p_ccb->con_state == GAP_CCB_STATE_CONNECTED)
568     L2CA_ConfigReq(p_ccb->connection_id, p_cfg);
569 
570   return (BT_PASS);
571 }
572 
573 /*******************************************************************************
574  *
575  * Function         GAP_ConnSetIdleTimeout
576  *
577  * Description      Higher layers call this function to set the idle timeout for
578  *                  a connection, or for all future connections. The "idle
579  *                  timeout" is the amount of time that a connection can remain
580  *                  up with no L2CAP channels on it. A timeout of zero means
581  *                  that the connection will be torn down immediately when the
582  *                  last channel is removed. A timeout of 0xFFFF means no
583  *                  timeout. Values are in seconds.
584  *
585  * Parameters:      handle      - Handle of the connection
586  *                  timeout     - in secs
587  *                                0 = immediate disconnect when last channel is
588  *                                    removed
589  *                                0xFFFF = no idle timeout
590  *
591  * Returns          BT_PASS                 - config process started
592  *                  GAP_ERR_BAD_HANDLE      - invalid handle
593  *
594  ******************************************************************************/
GAP_ConnSetIdleTimeout(uint16_t gap_handle,uint16_t timeout)595 uint16_t GAP_ConnSetIdleTimeout(uint16_t gap_handle, uint16_t timeout) {
596   tGAP_CCB* p_ccb;
597 
598   p_ccb = gap_find_ccb_by_handle(gap_handle);
599   if (p_ccb == NULL) return (GAP_ERR_BAD_HANDLE);
600 
601   if (L2CA_SetIdleTimeout(p_ccb->connection_id, timeout, false))
602     return (BT_PASS);
603   else
604     return (GAP_ERR_BAD_HANDLE);
605 }
606 
607 /*******************************************************************************
608  *
609  * Function         GAP_ConnGetRemoteAddr
610  *
611  * Description      This function is called to get the remote BD address
612  *                  of a connection.
613  *
614  * Parameters:      handle - Handle of the connection returned by GAP_ConnOpen
615  *
616  * Returns          BT_PASS             - closed OK
617  *                  GAP_ERR_BAD_HANDLE  - invalid handle
618  *
619  ******************************************************************************/
GAP_ConnGetRemoteAddr(uint16_t gap_handle)620 const RawAddress* GAP_ConnGetRemoteAddr(uint16_t gap_handle) {
621   tGAP_CCB* p_ccb = gap_find_ccb_by_handle(gap_handle);
622 
623   DVLOG(1) << __func__ << " gap_handle = " << gap_handle;
624 
625   if ((p_ccb) && (p_ccb->con_state > GAP_CCB_STATE_LISTENING)) {
626     DVLOG(1) << __func__ << " BDA: " << p_ccb->rem_dev_address;
627     return &p_ccb->rem_dev_address;
628   } else {
629     DVLOG(1) << __func__ << " return Error ";
630     return nullptr;
631   }
632 }
633 
634 /*******************************************************************************
635  *
636  * Function         GAP_ConnGetRemMtuSize
637  *
638  * Description      Returns the remote device's MTU size
639  *
640  * Parameters:      handle      - Handle of the connection
641  *
642  * Returns          uint16_t    - maximum size buffer that can be transmitted to
643  *                                the peer
644  *
645  ******************************************************************************/
GAP_ConnGetRemMtuSize(uint16_t gap_handle)646 uint16_t GAP_ConnGetRemMtuSize(uint16_t gap_handle) {
647   tGAP_CCB* p_ccb;
648 
649   p_ccb = gap_find_ccb_by_handle(gap_handle);
650   if (p_ccb == NULL) return (0);
651 
652   return (p_ccb->rem_mtu_size);
653 }
654 
655 /*******************************************************************************
656  *
657  * Function         GAP_ConnGetL2CAPCid
658  *
659  * Description      Returns the L2CAP channel id
660  *
661  * Parameters:      handle      - Handle of the connection
662  *
663  * Returns          uint16_t    - The L2CAP channel id
664  *                  0, if error
665  *
666  ******************************************************************************/
GAP_ConnGetL2CAPCid(uint16_t gap_handle)667 uint16_t GAP_ConnGetL2CAPCid(uint16_t gap_handle) {
668   tGAP_CCB* p_ccb;
669 
670   p_ccb = gap_find_ccb_by_handle(gap_handle);
671   if (p_ccb == NULL) return (0);
672 
673   return (p_ccb->connection_id);
674 }
675 
676 /*******************************************************************************
677  *
678  * Function         gap_tx_connect_ind
679  *
680  * Description      Sends out GAP_EVT_TX_EMPTY when transmission has been
681  *                  completed.
682  *
683  * Returns          void
684  *
685  ******************************************************************************/
gap_tx_complete_ind(uint16_t l2cap_cid,uint16_t sdu_sent)686 void gap_tx_complete_ind(uint16_t l2cap_cid, uint16_t sdu_sent) {
687   tGAP_CCB* p_ccb = gap_find_ccb_by_cid(l2cap_cid);
688   if (p_ccb == NULL) return;
689 
690   if ((p_ccb->con_state == GAP_CCB_STATE_CONNECTED) && (sdu_sent == 0xFFFF)) {
691     DVLOG(1) << StringPrintf("%s: GAP_EVT_TX_EMPTY", __func__);
692     p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_TX_EMPTY, nullptr);
693   }
694 }
695 
gap_credits_received_cb(uint16_t l2cap_cid,uint16_t credits_received,uint16_t credit_count)696 void gap_credits_received_cb(uint16_t l2cap_cid, uint16_t credits_received,
697                              uint16_t credit_count) {
698   tGAP_CCB* p_ccb = gap_find_ccb_by_cid(l2cap_cid);
699   if (!p_ccb) return;
700 
701   tGAP_CB_DATA data{.coc_credits = {.credits_received = credits_received,
702                                     .credit_count = credit_count}};
703   p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_LE_COC_CREDITS, &data);
704 }
705 
706 /*******************************************************************************
707  *
708  * Function         gap_connect_ind
709  *
710  * Description      This function handles an inbound connection indication
711  *                  from L2CAP. This is the case where we are acting as a
712  *                  server.
713  *
714  * Returns          void
715  *
716  ******************************************************************************/
gap_connect_ind(const RawAddress & bd_addr,uint16_t l2cap_cid,uint16_t psm,uint8_t l2cap_id)717 static void gap_connect_ind(const RawAddress& bd_addr, uint16_t l2cap_cid,
718                             uint16_t psm, uint8_t l2cap_id) {
719   uint16_t xx;
720   tGAP_CCB* p_ccb;
721 
722   /* See if we have a CCB listening for the connection */
723   for (xx = 0, p_ccb = conn.ccb_pool; xx < GAP_MAX_CONNECTIONS; xx++, p_ccb++) {
724     if ((p_ccb->con_state == GAP_CCB_STATE_LISTENING) && (p_ccb->psm == psm) &&
725         (!p_ccb->rem_addr_specified || (bd_addr == p_ccb->rem_dev_address)))
726       break;
727   }
728 
729   if (xx == GAP_MAX_CONNECTIONS) {
730     LOG(WARNING) << "*******";
731     LOG(WARNING) << "WARNING: GAP Conn Indication for Unexpected Bd "
732                     "Addr...Disconnecting";
733     LOG(WARNING) << "*******";
734 
735     /* Disconnect because it is an unexpected connection */
736     L2CA_DisconnectReq(l2cap_cid);
737     return;
738   }
739 
740   /* Transition to the next appropriate state, waiting for config setup. */
741   if (p_ccb->transport == BT_TRANSPORT_BR_EDR)
742     p_ccb->con_state = GAP_CCB_STATE_CFG_SETUP;
743 
744   /* Save the BD Address and Channel ID. */
745   p_ccb->rem_dev_address = bd_addr;
746   p_ccb->connection_id = l2cap_cid;
747 
748   /* Send response to the L2CAP layer. */
749   if (p_ccb->transport == BT_TRANSPORT_BR_EDR)
750     L2CA_ErtmConnectRsp(bd_addr, l2cap_id, l2cap_cid, L2CAP_CONN_OK,
751                         L2CAP_CONN_OK, &p_ccb->ertm_info);
752 
753   if (p_ccb->transport == BT_TRANSPORT_LE) {
754     L2CA_ConnectLECocRsp(bd_addr, l2cap_id, l2cap_cid, L2CAP_CONN_OK,
755                          L2CAP_CONN_OK, &p_ccb->local_coc_cfg);
756 
757     /* get the remote coc configuration */
758     L2CA_GetPeerLECocConfig(l2cap_cid, &p_ccb->peer_coc_cfg);
759     p_ccb->rem_mtu_size = p_ccb->peer_coc_cfg.mtu;
760 
761     /* configuration is not required for LE COC */
762     p_ccb->con_flags |= GAP_CCB_FLAGS_HIS_CFG_DONE;
763     p_ccb->con_flags |= GAP_CCB_FLAGS_MY_CFG_DONE;
764     gap_checks_con_flags(p_ccb);
765   }
766 
767   DVLOG(1) << StringPrintf("GAP_CONN - Rcvd L2CAP conn ind, CID: 0x%x",
768                            p_ccb->connection_id);
769 
770   /* Send a Configuration Request. */
771   if (p_ccb->transport == BT_TRANSPORT_BR_EDR)
772     L2CA_ConfigReq(l2cap_cid, &p_ccb->cfg);
773 }
774 
775 /*******************************************************************************
776  *
777  * Function         gap_checks_con_flags
778  *
779  * Description      This function processes the L2CAP configuration indication
780  *                  event.
781  *
782  * Returns          void
783  *
784  ******************************************************************************/
gap_checks_con_flags(tGAP_CCB * p_ccb)785 static void gap_checks_con_flags(tGAP_CCB* p_ccb) {
786   DVLOG(1) << __func__ << " conn_flags:0x" << +p_ccb->con_flags;
787   /* if all the required con_flags are set, report the OPEN event now */
788   if ((p_ccb->con_flags & GAP_CCB_FLAGS_CONN_DONE) == GAP_CCB_FLAGS_CONN_DONE) {
789     p_ccb->con_state = GAP_CCB_STATE_CONNECTED;
790 
791     p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_CONN_OPENED, nullptr);
792   }
793 }
794 
795 /*******************************************************************************
796  *
797  * Function         gap_sec_check_complete
798  *
799  * Description      The function called when Security Manager finishes
800  *                  verification of the service side connection
801  *
802  * Returns          void
803  *
804  ******************************************************************************/
gap_sec_check_complete(const RawAddress *,tBT_TRANSPORT,void * p_ref_data,uint8_t res)805 static void gap_sec_check_complete(const RawAddress*, tBT_TRANSPORT,
806                                    void* p_ref_data, uint8_t res) {
807   tGAP_CCB* p_ccb = (tGAP_CCB*)p_ref_data;
808 
809   DVLOG(1) << StringPrintf(
810       "gap_sec_check_complete conn_state:%d, conn_flags:0x%x, status:%d",
811       p_ccb->con_state, p_ccb->con_flags, res);
812   if (p_ccb->con_state == GAP_CCB_STATE_IDLE) return;
813 
814   if (res == BTM_SUCCESS) {
815     p_ccb->con_flags |= GAP_CCB_FLAGS_SEC_DONE;
816     gap_checks_con_flags(p_ccb);
817   } else {
818     /* security failed - disconnect the channel */
819     L2CA_DisconnectReq(p_ccb->connection_id);
820   }
821 }
822 
823 /*******************************************************************************
824  *
825  * Function         gap_connect_cfm
826  *
827  * Description      This function handles the connect confirm events
828  *                  from L2CAP. This is the case when we are acting as a
829  *                  client and have sent a connect request.
830  *
831  * Returns          void
832  *
833  ******************************************************************************/
gap_connect_cfm(uint16_t l2cap_cid,uint16_t result)834 static void gap_connect_cfm(uint16_t l2cap_cid, uint16_t result) {
835   tGAP_CCB* p_ccb;
836 
837   /* Find CCB based on CID */
838   p_ccb = gap_find_ccb_by_cid(l2cap_cid);
839   if (p_ccb == NULL) return;
840 
841   /* initiate security process, if needed */
842   if ((p_ccb->con_flags & GAP_CCB_FLAGS_SEC_DONE) == 0 &&
843       p_ccb->transport != BT_TRANSPORT_LE) {
844     btm_sec_mx_access_request(p_ccb->rem_dev_address, p_ccb->psm, true, 0, 0,
845                               &gap_sec_check_complete, p_ccb);
846   }
847 
848   /* If the connection response contains success status, then */
849   /* Transition to the next state and startup the timer.      */
850   if ((result == L2CAP_CONN_OK) &&
851       (p_ccb->con_state == GAP_CCB_STATE_CONN_SETUP)) {
852     if (p_ccb->transport == BT_TRANSPORT_BR_EDR) {
853       p_ccb->con_state = GAP_CCB_STATE_CFG_SETUP;
854 
855       /* Send a Configuration Request. */
856       L2CA_ConfigReq(l2cap_cid, &p_ccb->cfg);
857     }
858 
859     if (p_ccb->transport == BT_TRANSPORT_LE) {
860       /* get the remote coc configuration */
861       L2CA_GetPeerLECocConfig(l2cap_cid, &p_ccb->peer_coc_cfg);
862       p_ccb->rem_mtu_size = p_ccb->peer_coc_cfg.mtu;
863 
864       /* configuration is not required for LE COC */
865       p_ccb->con_flags |= GAP_CCB_FLAGS_HIS_CFG_DONE;
866       p_ccb->con_flags |= GAP_CCB_FLAGS_MY_CFG_DONE;
867       p_ccb->con_flags |= GAP_CCB_FLAGS_SEC_DONE;
868       gap_checks_con_flags(p_ccb);
869     }
870   } else {
871     /* Tell the user if he has a callback */
872     if (p_ccb->p_callback)
873       (*p_ccb->p_callback)(p_ccb->gap_handle, GAP_EVT_CONN_CLOSED, nullptr);
874 
875     gap_release_ccb(p_ccb);
876   }
877 }
878 
879 /*******************************************************************************
880  *
881  * Function         gap_config_ind
882  *
883  * Description      This function processes the L2CAP configuration indication
884  *                  event.
885  *
886  * Returns          void
887  *
888  ******************************************************************************/
gap_config_ind(uint16_t l2cap_cid,tL2CAP_CFG_INFO * p_cfg)889 static void gap_config_ind(uint16_t l2cap_cid, tL2CAP_CFG_INFO* p_cfg) {
890   tGAP_CCB* p_ccb;
891   uint16_t local_mtu_size;
892 
893   /* Find CCB based on CID */
894   p_ccb = gap_find_ccb_by_cid(l2cap_cid);
895   if (p_ccb == NULL) return;
896 
897   /* Remember the remote MTU size */
898 
899   if (p_ccb->cfg.fcr.mode == L2CAP_FCR_ERTM_MODE) {
900     local_mtu_size =
901         p_ccb->ertm_info.user_tx_buf_size - sizeof(BT_HDR) - L2CAP_MIN_OFFSET;
902   } else
903     local_mtu_size = L2CAP_MTU_SIZE;
904 
905   if ((!p_cfg->mtu_present) || (p_cfg->mtu > local_mtu_size)) {
906     p_ccb->rem_mtu_size = local_mtu_size;
907   } else
908     p_ccb->rem_mtu_size = p_cfg->mtu;
909 
910   /* For now, always accept configuration from the other side */
911   p_cfg->flush_to_present = false;
912   p_cfg->mtu_present = false;
913   p_cfg->result = L2CAP_CFG_OK;
914   p_cfg->fcs_present = false;
915 
916   L2CA_ConfigRsp(l2cap_cid, p_cfg);
917 
918   p_ccb->con_flags |= GAP_CCB_FLAGS_HIS_CFG_DONE;
919 
920   gap_checks_con_flags(p_ccb);
921 }
922 
923 /*******************************************************************************
924  *
925  * Function         gap_config_cfm
926  *
927  * Description      This function processes the L2CAP configuration confirmation
928  *                  event.
929  *
930  * Returns          void
931  *
932  ******************************************************************************/
gap_config_cfm(uint16_t l2cap_cid,tL2CAP_CFG_INFO * p_cfg)933 static void gap_config_cfm(uint16_t l2cap_cid, tL2CAP_CFG_INFO* p_cfg) {
934   tGAP_CCB* p_ccb;
935 
936   /* Find CCB based on CID */
937   p_ccb = gap_find_ccb_by_cid(l2cap_cid);
938   if (p_ccb == NULL) return;
939 
940   if (p_cfg->result == L2CAP_CFG_OK) {
941     p_ccb->con_flags |= GAP_CCB_FLAGS_MY_CFG_DONE;
942 
943     if (p_ccb->cfg.fcr_present)
944       p_ccb->cfg.fcr.mode = p_cfg->fcr.mode;
945     else
946       p_ccb->cfg.fcr.mode = L2CAP_FCR_BASIC_MODE;
947 
948     gap_checks_con_flags(p_ccb);
949   } else {
950     p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_CONN_CLOSED, nullptr);
951     gap_release_ccb(p_ccb);
952   }
953 }
954 
955 /*******************************************************************************
956  *
957  * Function         gap_disconnect_ind
958  *
959  * Description      This function handles a disconnect event from L2CAP. If
960  *                  requested to, we ack the disconnect before dropping the CCB
961  *
962  * Returns          void
963  *
964  ******************************************************************************/
gap_disconnect_ind(uint16_t l2cap_cid,bool ack_needed)965 static void gap_disconnect_ind(uint16_t l2cap_cid, bool ack_needed) {
966   tGAP_CCB* p_ccb;
967 
968   DVLOG(1) << StringPrintf("GAP_CONN - Rcvd L2CAP disc, CID: 0x%x", l2cap_cid);
969 
970   /* Find CCB based on CID */
971   p_ccb = gap_find_ccb_by_cid(l2cap_cid);
972   if (p_ccb == NULL) return;
973 
974   if (ack_needed) L2CA_DisconnectRsp(l2cap_cid);
975 
976   p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_CONN_CLOSED, nullptr);
977   gap_release_ccb(p_ccb);
978 }
979 
980 /*******************************************************************************
981  *
982  * Function         gap_data_ind
983  *
984  * Description      This function is called when data is received from L2CAP.
985  *
986  * Returns          void
987  *
988  ******************************************************************************/
gap_data_ind(uint16_t l2cap_cid,BT_HDR * p_msg)989 static void gap_data_ind(uint16_t l2cap_cid, BT_HDR* p_msg) {
990   tGAP_CCB* p_ccb;
991 
992   /* Find CCB based on CID */
993   p_ccb = gap_find_ccb_by_cid(l2cap_cid);
994   if (p_ccb == NULL) {
995     osi_free(p_msg);
996     return;
997   }
998 
999   if (p_ccb->con_state == GAP_CCB_STATE_CONNECTED) {
1000     fixed_queue_enqueue(p_ccb->rx_queue, p_msg);
1001 
1002     p_ccb->rx_queue_size += p_msg->len;
1003     /*
1004     DVLOG(1) << StringPrintf ("gap_data_ind - rx_queue_size=%d, msg len=%d",
1005                                    p_ccb->rx_queue_size, p_msg->len);
1006      */
1007 
1008     p_ccb->p_callback(p_ccb->gap_handle, GAP_EVT_CONN_DATA_AVAIL, nullptr);
1009   } else {
1010     osi_free(p_msg);
1011   }
1012 }
1013 
1014 /*******************************************************************************
1015  *
1016  * Function         gap_congestion_ind
1017  *
1018  * Description      This is a callback function called by L2CAP when
1019  *                  data L2CAP congestion status changes
1020  *
1021  ******************************************************************************/
gap_congestion_ind(uint16_t lcid,bool is_congested)1022 static void gap_congestion_ind(uint16_t lcid, bool is_congested) {
1023   DVLOG(1) << StringPrintf("GAP_CONN - Rcvd L2CAP Is Congested (%d), CID: 0x%x",
1024                            is_congested, lcid);
1025 
1026   tGAP_CCB* p_ccb = gap_find_ccb_by_cid(lcid); /* Find CCB based on CID */
1027   if (!p_ccb) return;
1028 
1029   p_ccb->is_congested = is_congested;
1030 
1031   p_ccb->p_callback(
1032       p_ccb->gap_handle,
1033       (is_congested) ? GAP_EVT_CONN_CONGESTED : GAP_EVT_CONN_UNCONGESTED,
1034       nullptr);
1035 
1036   gap_try_write_queued_data(p_ccb);
1037 }
1038 
1039 /*******************************************************************************
1040  *
1041  * Function         gap_find_ccb_by_cid
1042  *
1043  * Description      This function searches the CCB table for an entry with the
1044  *                  passed CID.
1045  *
1046  * Returns          the CCB address, or NULL if not found.
1047  *
1048  ******************************************************************************/
gap_find_ccb_by_cid(uint16_t cid)1049 static tGAP_CCB* gap_find_ccb_by_cid(uint16_t cid) {
1050   uint16_t xx;
1051   tGAP_CCB* p_ccb;
1052 
1053   /* Look through each connection control block */
1054   for (xx = 0, p_ccb = conn.ccb_pool; xx < GAP_MAX_CONNECTIONS; xx++, p_ccb++) {
1055     if ((p_ccb->con_state != GAP_CCB_STATE_IDLE) &&
1056         (p_ccb->connection_id == cid))
1057       return (p_ccb);
1058   }
1059 
1060   /* If here, not found */
1061   return (NULL);
1062 }
1063 
1064 /*******************************************************************************
1065  *
1066  * Function         gap_find_ccb_by_handle
1067  *
1068  * Description      This function searches the CCB table for an entry with the
1069  *                  passed handle.
1070  *
1071  * Returns          the CCB address, or NULL if not found.
1072  *
1073  ******************************************************************************/
gap_find_ccb_by_handle(uint16_t handle)1074 static tGAP_CCB* gap_find_ccb_by_handle(uint16_t handle) {
1075   tGAP_CCB* p_ccb;
1076 
1077   /* Check that handle is valid */
1078   if (handle < GAP_MAX_CONNECTIONS) {
1079     p_ccb = &conn.ccb_pool[handle];
1080 
1081     if (p_ccb->con_state != GAP_CCB_STATE_IDLE) return (p_ccb);
1082   }
1083 
1084   /* If here, handle points to invalid connection */
1085   return (NULL);
1086 }
1087 
1088 /*******************************************************************************
1089  *
1090  * Function         gap_allocate_ccb
1091  *
1092  * Description      This function allocates a new CCB.
1093  *
1094  * Returns          CCB address, or NULL if none available.
1095  *
1096  ******************************************************************************/
gap_allocate_ccb(void)1097 static tGAP_CCB* gap_allocate_ccb(void) {
1098   uint16_t xx;
1099   tGAP_CCB* p_ccb;
1100 
1101   /* Look through each connection control block for a free one */
1102   for (xx = 0, p_ccb = conn.ccb_pool; xx < GAP_MAX_CONNECTIONS; xx++, p_ccb++) {
1103     if (p_ccb->con_state == GAP_CCB_STATE_IDLE) {
1104       memset(p_ccb, 0, sizeof(tGAP_CCB));
1105       p_ccb->tx_queue = fixed_queue_new(SIZE_MAX);
1106       p_ccb->rx_queue = fixed_queue_new(SIZE_MAX);
1107 
1108       p_ccb->gap_handle = xx;
1109       p_ccb->rem_mtu_size = L2CAP_MTU_SIZE;
1110 
1111       return (p_ccb);
1112     }
1113   }
1114 
1115   /* If here, no free CCB found */
1116   return (NULL);
1117 }
1118 
1119 /*******************************************************************************
1120  *
1121  * Function         gap_release_ccb
1122  *
1123  * Description      This function releases a CCB.
1124  *
1125  * Returns          void
1126  *
1127  ******************************************************************************/
gap_release_ccb(tGAP_CCB * p_ccb)1128 static void gap_release_ccb(tGAP_CCB* p_ccb) {
1129   /* Drop any buffers we may be holding */
1130   p_ccb->rx_queue_size = 0;
1131 
1132   while (!fixed_queue_is_empty(p_ccb->rx_queue))
1133     osi_free(fixed_queue_try_dequeue(p_ccb->rx_queue));
1134   fixed_queue_free(p_ccb->rx_queue, NULL);
1135   p_ccb->rx_queue = NULL;
1136 
1137   while (!fixed_queue_is_empty(p_ccb->tx_queue))
1138     osi_free(fixed_queue_try_dequeue(p_ccb->tx_queue));
1139   fixed_queue_free(p_ccb->tx_queue, NULL);
1140   p_ccb->tx_queue = NULL;
1141 
1142   p_ccb->con_state = GAP_CCB_STATE_IDLE;
1143 
1144   /* If no-one else is using the PSM, deregister from L2CAP */
1145   tGAP_CCB* p_ccb_local = conn.ccb_pool;
1146   for (uint16_t i = 0; i < GAP_MAX_CONNECTIONS; i++, p_ccb_local++) {
1147     if ((p_ccb_local->con_state != GAP_CCB_STATE_IDLE) &&
1148         (p_ccb_local->psm == p_ccb->psm)) {
1149       DVLOG(1) << __func__ << " : " << +p_ccb_local->psm
1150                << " PSM is still in use, do not deregister";
1151       return;
1152     }
1153   }
1154 
1155   /* Free the security record for this PSM */
1156   BTM_SecClrService(p_ccb->service_id);
1157   if (p_ccb->transport == BT_TRANSPORT_BR_EDR) L2CA_Deregister(p_ccb->psm);
1158   if (p_ccb->transport == BT_TRANSPORT_LE) L2CA_DeregisterLECoc(p_ccb->psm);
1159 }
1160 
1161 extern void gap_attr_db_init(void);
1162 
1163 /*
1164  * This routine should not be called except once per stack invocation.
1165  */
GAP_Init(void)1166 void GAP_Init(void) {
1167   gap_conn_init();
1168   gap_attr_db_init();
1169 }
1170