• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /******************************************************************************
2  *
3  *  Copyright 2016 The Android Open Source Project
4  *  Copyright 2002-2012 Broadcom Corporation
5  *
6  *  Licensed under the Apache License, Version 2.0 (the "License");
7  *  you may not use this file except in compliance with the License.
8  *  You may obtain a copy of the License at:
9  *
10  *  http://www.apache.org/licenses/LICENSE-2.0
11  *
12  *  Unless required by applicable law or agreed to in writing, software
13  *  distributed under the License is distributed on an "AS IS" BASIS,
14  *  WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15  *  See the License for the specific language governing permissions and
16  *  limitations under the License.
17  *
18  ******************************************************************************/
19 
20 /******************************************************************************
21  *
22  *  this file contains the connection interface functions
23  *
24  ******************************************************************************/
25 
26 #include <stdio.h>
27 #include <stdlib.h>
28 #include <string.h>
29 
30 #include "bt_types.h"
31 
32 #include "l2c_api.h"
33 #include "l2cdefs.h"
34 
35 #include "btm_api.h"
36 #include "btm_int.h"
37 #include "btu.h"
38 
39 #include "hiddefs.h"
40 
41 #include "bt_utils.h"
42 #include "hidd_api.h"
43 #include "hidd_int.h"
44 
45 #include "osi/include/osi.h"
46 
47 static void hidd_l2cif_connect_ind(const RawAddress& bd_addr, uint16_t cid,
48                                    uint16_t psm, uint8_t id);
49 static void hidd_l2cif_connect_cfm(uint16_t cid, uint16_t result);
50 static void hidd_l2cif_config_ind(uint16_t cid, tL2CAP_CFG_INFO* p_cfg);
51 static void hidd_l2cif_config_cfm(uint16_t cid, tL2CAP_CFG_INFO* p_cfg);
52 static void hidd_l2cif_disconnect_ind(uint16_t cid, bool ack_needed);
53 static void hidd_l2cif_disconnect_cfm(uint16_t cid, uint16_t result);
54 static void hidd_l2cif_data_ind(uint16_t cid, BT_HDR* p_msg);
55 static void hidd_l2cif_cong_ind(uint16_t cid, bool congested);
56 
57 static const tL2CAP_APPL_INFO dev_reg_info = {
58     hidd_l2cif_connect_ind,
59     hidd_l2cif_connect_cfm,
60     NULL,
61     hidd_l2cif_config_ind,
62     hidd_l2cif_config_cfm,
63     hidd_l2cif_disconnect_ind,
64     hidd_l2cif_disconnect_cfm,
65     NULL,
66     hidd_l2cif_data_ind,
67     hidd_l2cif_cong_ind,
68     NULL,
69     NULL /* tL2CA_CREDITS_RECEIVED_CB */};
70 
71 /*******************************************************************************
72  *
73  * Function         hidd_check_config_done
74  *
75  * Description      Checks if connection is configured and callback can be fired
76  *
77  * Returns          void
78  *
79  ******************************************************************************/
hidd_check_config_done()80 static void hidd_check_config_done() {
81   tHID_CONN* p_hcon;
82 
83   p_hcon = &hd_cb.device.conn;
84 
85   if (((p_hcon->conn_flags & HID_CONN_FLAGS_ALL_CONFIGURED) ==
86        HID_CONN_FLAGS_ALL_CONFIGURED) &&
87       (p_hcon->conn_state == HID_CONN_STATE_CONFIG)) {
88     p_hcon->conn_state = HID_CONN_STATE_CONNECTED;
89 
90     hd_cb.device.state = HIDD_DEV_CONNECTED;
91 
92     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_OPEN, 0, NULL);
93 
94     // send outstanding data on intr
95     if (hd_cb.pending_data) {
96       L2CA_DataWrite(p_hcon->intr_cid, hd_cb.pending_data);
97       hd_cb.pending_data = NULL;
98     }
99   }
100 }
101 
102 /*******************************************************************************
103  *
104  * Function         hidh_sec_check_complete_term
105  *
106  * Description      HID security check complete callback function.
107  *
108  * Returns          Send L2CA_ConnectRsp OK if secutiry check succeed; otherwise
109  *                  send security block L2C connection response.
110  *
111  ******************************************************************************/
hidd_sec_check_complete(UNUSED_ATTR const RawAddress * bd_addr,UNUSED_ATTR tBT_TRANSPORT transport,void * p_ref_data,uint8_t res)112 static void hidd_sec_check_complete(UNUSED_ATTR const RawAddress* bd_addr,
113                                     UNUSED_ATTR tBT_TRANSPORT transport,
114                                     void* p_ref_data, uint8_t res) {
115   tHID_DEV_DEV_CTB* p_dev = (tHID_DEV_DEV_CTB*)p_ref_data;
116 
117   if (res == BTM_SUCCESS && p_dev->conn.conn_state == HID_CONN_STATE_SECURITY) {
118     p_dev->conn.disc_reason = HID_SUCCESS;
119     p_dev->conn.conn_state = HID_CONN_STATE_CONNECTING_INTR;
120 
121     L2CA_ConnectRsp(p_dev->addr, p_dev->conn.ctrl_id, p_dev->conn.ctrl_cid,
122                     L2CAP_CONN_OK, L2CAP_CONN_OK);
123     L2CA_ConfigReq(p_dev->conn.ctrl_cid, &hd_cb.l2cap_cfg);
124   } else if (res != BTM_SUCCESS) {
125     HIDD_TRACE_WARNING("%s: connection rejected by security", __func__);
126 
127     p_dev->conn.disc_reason = HID_ERR_AUTH_FAILED;
128     p_dev->conn.conn_state = HID_CONN_STATE_UNUSED;
129     L2CA_ConnectRsp(p_dev->addr, p_dev->conn.ctrl_id, p_dev->conn.ctrl_cid,
130                     L2CAP_CONN_SECURITY_BLOCK, L2CAP_CONN_OK);
131     return;
132   }
133 }
134 
135 /*******************************************************************************
136  *
137  * Function         hidd_sec_check_complete_orig
138  *
139  * Description      HID security check complete callback function (device
140 *originated)
141  *
142  * Returns          void
143  *
144  ******************************************************************************/
hidd_sec_check_complete_orig(UNUSED_ATTR const RawAddress * bd_addr,UNUSED_ATTR tBT_TRANSPORT transport,void * p_ref_data,uint8_t res)145 void hidd_sec_check_complete_orig(UNUSED_ATTR const RawAddress* bd_addr,
146                                   UNUSED_ATTR tBT_TRANSPORT transport,
147                                   void* p_ref_data, uint8_t res) {
148   tHID_DEV_DEV_CTB* p_dev = (tHID_DEV_DEV_CTB*)p_ref_data;
149 
150   if (p_dev->conn.conn_state != HID_CONN_STATE_SECURITY) {
151     HIDD_TRACE_WARNING("%s: invalid state (%02x)", __func__,
152                        p_dev->conn.conn_state);
153     return;
154   }
155 
156   if (res == BTM_SUCCESS) {
157     HIDD_TRACE_EVENT("%s: security ok", __func__);
158     p_dev->conn.disc_reason = HID_SUCCESS;
159 
160     p_dev->conn.conn_state = HID_CONN_STATE_CONFIG;
161     L2CA_ConfigReq(p_dev->conn.ctrl_cid, &hd_cb.l2cap_cfg);
162   } else {
163     HIDD_TRACE_WARNING("%s: security check failed (%02x)", __func__, res);
164     p_dev->conn.disc_reason = HID_ERR_AUTH_FAILED;
165     hidd_conn_disconnect();
166   }
167 }
168 
169 /*******************************************************************************
170  *
171  * Function         hidd_l2cif_connect_ind
172  *
173  * Description      Handles incoming L2CAP connection (we act as server)
174  *
175  * Returns          void
176  *
177  ******************************************************************************/
hidd_l2cif_connect_ind(const RawAddress & bd_addr,uint16_t cid,uint16_t psm,uint8_t id)178 static void hidd_l2cif_connect_ind(const RawAddress& bd_addr, uint16_t cid,
179                                    uint16_t psm, uint8_t id) {
180   tHID_CONN* p_hcon;
181   tHID_DEV_DEV_CTB* p_dev;
182   bool accept = TRUE;  // accept by default
183 
184   HIDD_TRACE_EVENT("%s: psm=%04x cid=%04x id=%02x", __func__, psm, cid, id);
185 
186   p_dev = &hd_cb.device;
187 
188   if (!hd_cb.allow_incoming) {
189     HIDD_TRACE_WARNING("%s: incoming connections not allowed, rejecting",
190                        __func__);
191     L2CA_ConnectRsp(bd_addr, id, cid, L2CAP_CONN_NO_RESOURCES, 0);
192     return;
193   }
194 
195   p_hcon = &hd_cb.device.conn;
196 
197   switch (psm) {
198     case HID_PSM_INTERRUPT:
199       if (p_hcon->ctrl_cid == 0) {
200         accept = FALSE;
201         HIDD_TRACE_WARNING("%s: incoming INTR without CTRL, rejecting",
202                            __func__);
203       }
204 
205       if (p_hcon->conn_state != HID_CONN_STATE_CONNECTING_INTR) {
206         accept = FALSE;
207         HIDD_TRACE_WARNING("%s: incoming INTR in invalid state (%d), rejecting",
208                            __func__, p_hcon->conn_state);
209       }
210 
211       break;
212 
213     case HID_PSM_CONTROL:
214       if (p_hcon->conn_state != HID_CONN_STATE_UNUSED) {
215         accept = FALSE;
216         HIDD_TRACE_WARNING("%s: incoming CTRL in invalid state (%d), rejecting",
217                            __func__, p_hcon->conn_state);
218       }
219 
220       break;
221 
222     default:
223       accept = FALSE;
224       HIDD_TRACE_ERROR("%s: received invalid PSM, rejecting", __func__);
225       break;
226   }
227 
228   if (!accept) {
229     L2CA_ConnectRsp(bd_addr, id, cid, L2CAP_CONN_NO_RESOURCES, 0);
230     return;
231   }
232 
233   // for CTRL we need to go through security and we reply in callback from there
234   if (psm == HID_PSM_CONTROL) {
235     // We are ready to accept connection from this device, since we aren't
236     // connected to anything and are in the correct state.
237     p_dev->in_use = TRUE;
238     p_dev->addr = bd_addr;
239     p_dev->state = HIDD_DEV_NO_CONN;
240 
241     p_hcon->conn_flags = 0;
242     p_hcon->ctrl_cid = cid;
243     p_hcon->ctrl_id = id;
244     p_hcon->disc_reason = HID_L2CAP_CONN_FAIL;
245 
246     p_hcon->conn_state = HID_CONN_STATE_SECURITY;
247     if (btm_sec_mx_access_request(p_dev->addr, HID_PSM_CONTROL, FALSE,
248                                   BTM_SEC_PROTO_HID, HIDD_NOSEC_CHN,
249                                   &hidd_sec_check_complete,
250                                   p_dev) == BTM_CMD_STARTED) {
251       L2CA_ConnectRsp(bd_addr, id, cid, L2CAP_CONN_PENDING, L2CAP_CONN_OK);
252     }
253 
254     return;
255   }
256 
257   // for INTR we go directly to config state
258   p_hcon->conn_state = HID_CONN_STATE_CONFIG;
259   p_hcon->intr_cid = cid;
260 
261   L2CA_ConnectRsp(bd_addr, id, cid, L2CAP_CONN_OK, L2CAP_CONN_OK);
262   L2CA_ConfigReq(cid, &hd_cb.l2cap_intr_cfg);
263 }
264 
265 /*******************************************************************************
266  *
267  * Function         hidd_l2cif_connect_cfm
268  *
269  * Description      Handles L2CAP connection response (we act as client)
270  *
271  * Returns          void
272  *
273  ******************************************************************************/
hidd_l2cif_connect_cfm(uint16_t cid,uint16_t result)274 static void hidd_l2cif_connect_cfm(uint16_t cid, uint16_t result) {
275   tHID_DEV_DEV_CTB* p_dev = &hd_cb.device;
276   tHID_CONN* p_hcon = &hd_cb.device.conn;
277 
278   HIDD_TRACE_EVENT("%s: cid=%04x result=%d", __func__, cid, result);
279 
280   if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
281     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
282     return;
283   }
284 
285   if (!(p_hcon->conn_flags & HID_CONN_FLAGS_IS_ORIG) ||
286       ((cid == p_hcon->ctrl_cid) &&
287        (p_hcon->conn_state != HID_CONN_STATE_CONNECTING_CTRL)) ||
288       ((cid == p_hcon->intr_cid) &&
289        (p_hcon->conn_state != HID_CONN_STATE_CONNECTING_INTR))) {
290     HIDD_TRACE_WARNING("%s: unexpected", __func__);
291     return;
292   }
293 
294   if (result != L2CAP_CONN_OK) {
295     HIDD_TRACE_WARNING("%s: connection failed, now disconnect", __func__);
296 
297     if (cid == p_hcon->ctrl_cid)
298       p_hcon->ctrl_cid = 0;
299     else
300       p_hcon->intr_cid = 0;
301 
302     hidd_conn_disconnect();
303 
304     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
305                    HID_L2CAP_CONN_FAIL | (uint32_t)result, NULL);
306     return;
307   }
308 
309   /* CTRL connect conf */
310   if (cid == p_hcon->ctrl_cid) {
311     p_hcon->conn_state = HID_CONN_STATE_SECURITY;
312     p_hcon->disc_reason =
313         HID_L2CAP_CONN_FAIL; /* in case disconnected before sec completed */
314 
315     btm_sec_mx_access_request(p_dev->addr, HID_PSM_CONTROL, TRUE,
316                               BTM_SEC_PROTO_HID, HIDD_SEC_CHN,
317                               &hidd_sec_check_complete_orig, p_dev);
318   } else {
319     p_hcon->conn_state = HID_CONN_STATE_CONFIG;
320     L2CA_ConfigReq(cid, &hd_cb.l2cap_intr_cfg);
321   }
322 
323   return;
324 }
325 
326 /*******************************************************************************
327  *
328  * Function         hidd_l2cif_config_ind
329  *
330  * Description      Handles incoming L2CAP configuration request
331  *
332  * Returns          void
333  *
334  ******************************************************************************/
hidd_l2cif_config_ind(uint16_t cid,tL2CAP_CFG_INFO * p_cfg)335 static void hidd_l2cif_config_ind(uint16_t cid, tL2CAP_CFG_INFO* p_cfg) {
336   tHID_CONN* p_hcon;
337 
338   HIDD_TRACE_EVENT("%s: cid=%04x", __func__, cid);
339 
340   p_hcon = &hd_cb.device.conn;
341 
342   if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
343     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
344     return;
345   }
346 
347   if ((!p_cfg->mtu_present) || (p_cfg->mtu > HID_DEV_MTU_SIZE))
348     p_hcon->rem_mtu_size = HID_DEV_MTU_SIZE;
349   else
350     p_hcon->rem_mtu_size = p_cfg->mtu;
351 
352   // accept without changes
353   p_cfg->flush_to_present = FALSE;
354   p_cfg->mtu_present = FALSE;
355   p_cfg->result = L2CAP_CFG_OK;
356 
357   if (cid == p_hcon->intr_cid && hd_cb.use_in_qos && !p_cfg->qos_present) {
358     p_cfg->qos_present = TRUE;
359     memcpy(&p_cfg->qos, &hd_cb.in_qos, sizeof(FLOW_SPEC));
360   }
361 
362   L2CA_ConfigRsp(cid, p_cfg);
363 
364   // update flags
365   if (cid == p_hcon->ctrl_cid) {
366     p_hcon->conn_flags |= HID_CONN_FLAGS_HIS_CTRL_CFG_DONE;
367 
368     if ((p_hcon->conn_flags & HID_CONN_FLAGS_IS_ORIG) &&
369         (p_hcon->conn_flags & HID_CONN_FLAGS_MY_CTRL_CFG_DONE)) {
370       p_hcon->disc_reason = HID_L2CAP_CONN_FAIL;
371       if ((p_hcon->intr_cid =
372                L2CA_ConnectReq(HID_PSM_INTERRUPT, hd_cb.device.addr)) == 0) {
373         hidd_conn_disconnect();
374         p_hcon->conn_state = HID_CONN_STATE_UNUSED;
375 
376         HIDD_TRACE_WARNING("%s: could not start L2CAP connection for INTR",
377                            __func__);
378         hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
379                        HID_ERR_L2CAP_FAILED, NULL);
380         return;
381       } else {
382         p_hcon->conn_state = HID_CONN_STATE_CONNECTING_INTR;
383       }
384     }
385   } else {
386     p_hcon->conn_flags |= HID_CONN_FLAGS_HIS_INTR_CFG_DONE;
387   }
388 
389   hidd_check_config_done();
390 }
391 
392 /*******************************************************************************
393  *
394  * Function         hidd_l2cif_config_cfm
395  *
396  * Description      Handles incoming L2CAP configuration response
397  *
398  * Returns          void
399  *
400  ******************************************************************************/
hidd_l2cif_config_cfm(uint16_t cid,tL2CAP_CFG_INFO * p_cfg)401 static void hidd_l2cif_config_cfm(uint16_t cid, tL2CAP_CFG_INFO* p_cfg) {
402   tHID_CONN* p_hcon;
403   uint32_t reason;
404 
405   HIDD_TRACE_EVENT("%s: cid=%04x pcfg->result=%d", __func__, cid,
406                    p_cfg->result);
407 
408   p_hcon = &hd_cb.device.conn;
409 
410   if (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid) {
411     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
412     return;
413   }
414 
415   if (p_hcon->intr_cid == cid &&
416       p_cfg->result == L2CAP_CFG_UNACCEPTABLE_PARAMS && p_cfg->qos_present) {
417     tL2CAP_CFG_INFO new_qos;
418 
419     // QoS parameters not accepted for intr, try again with host proposal
420 
421     memcpy(&new_qos, &hd_cb.l2cap_intr_cfg, sizeof(new_qos));
422     memcpy(&new_qos.qos, &p_cfg->qos, sizeof(FLOW_SPEC));
423     new_qos.qos_present = TRUE;
424 
425     HIDD_TRACE_WARNING("%s: config failed, retry", __func__);
426 
427     L2CA_ConfigReq(cid, &new_qos);
428     return;
429   } else if (p_hcon->intr_cid == cid &&
430              p_cfg->result == L2CAP_CFG_UNKNOWN_OPTIONS) {
431     // QoS not understood by remote device, try configuring without QoS
432 
433     HIDD_TRACE_WARNING("%s: config failed, retry without QoS", __func__);
434 
435     L2CA_ConfigReq(cid, &hd_cb.l2cap_cfg);
436     return;
437   } else if (p_cfg->result != L2CAP_CFG_OK) {
438     HIDD_TRACE_WARNING("%s: config failed, disconnecting", __func__);
439 
440     hidd_conn_disconnect();
441     reason = HID_L2CAP_CFG_FAIL | (uint32_t)p_cfg->result;
442 
443     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE, reason, NULL);
444     return;
445   }
446 
447   // update flags
448   if (cid == p_hcon->ctrl_cid) {
449     p_hcon->conn_flags |= HID_CONN_FLAGS_MY_CTRL_CFG_DONE;
450 
451     if ((p_hcon->conn_flags & HID_CONN_FLAGS_IS_ORIG) &&
452         (p_hcon->conn_flags & HID_CONN_FLAGS_HIS_CTRL_CFG_DONE)) {
453       p_hcon->disc_reason = HID_L2CAP_CONN_FAIL;
454       if ((p_hcon->intr_cid =
455                L2CA_ConnectReq(HID_PSM_INTERRUPT, hd_cb.device.addr)) == 0) {
456         hidd_conn_disconnect();
457         p_hcon->conn_state = HID_CONN_STATE_UNUSED;
458 
459         HIDD_TRACE_WARNING("%s: could not start L2CAP connection for INTR",
460                            __func__);
461         hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
462                        HID_ERR_L2CAP_FAILED, NULL);
463         return;
464       } else {
465         p_hcon->conn_state = HID_CONN_STATE_CONNECTING_INTR;
466       }
467     }
468   } else {
469     p_hcon->conn_flags |= HID_CONN_FLAGS_MY_INTR_CFG_DONE;
470   }
471 
472   hidd_check_config_done();
473 }
474 
475 /*******************************************************************************
476  *
477  * Function         hidd_l2cif_disconnect_ind
478  *
479  * Description      Handler incoming L2CAP disconnection request
480  *
481  * Returns          void
482  *
483  ******************************************************************************/
hidd_l2cif_disconnect_ind(uint16_t cid,bool ack_needed)484 static void hidd_l2cif_disconnect_ind(uint16_t cid, bool ack_needed) {
485   tHID_CONN* p_hcon;
486 
487   HIDD_TRACE_EVENT("%s: cid=%04x ack_needed=%d", __func__, cid, ack_needed);
488 
489   p_hcon = &hd_cb.device.conn;
490 
491   if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
492       (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
493     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
494     return;
495   }
496 
497   if (ack_needed) L2CA_DisconnectRsp(cid);
498 
499   p_hcon->conn_state = HID_CONN_STATE_DISCONNECTING;
500 
501   if (cid == p_hcon->ctrl_cid)
502     p_hcon->ctrl_cid = 0;
503   else
504     p_hcon->intr_cid = 0;
505 
506   if ((p_hcon->ctrl_cid == 0) && (p_hcon->intr_cid == 0)) {
507     HIDD_TRACE_EVENT("%s: INTR and CTRL disconnected", __func__);
508 
509     // clean any outstanding data on intr
510     if (hd_cb.pending_data) {
511       osi_free(hd_cb.pending_data);
512       hd_cb.pending_data = NULL;
513     }
514 
515     hd_cb.device.state = HIDD_DEV_NO_CONN;
516     p_hcon->conn_state = HID_CONN_STATE_UNUSED;
517 
518     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE, p_hcon->disc_reason,
519                    NULL);
520   }
521 }
522 
523 /*******************************************************************************
524  *
525  * Function         hidd_l2cif_disconnect_cfm
526  *
527  * Description      Handles L2CAP disconection response
528  *
529  * Returns          void
530  *
531  ******************************************************************************/
hidd_l2cif_disconnect_cfm(uint16_t cid,uint16_t result)532 static void hidd_l2cif_disconnect_cfm(uint16_t cid, uint16_t result) {
533   tHID_CONN* p_hcon;
534 
535   HIDD_TRACE_EVENT("%s: cid=%04x result=%d", __func__, cid, result);
536 
537   p_hcon = &hd_cb.device.conn;
538 
539   if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
540       (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
541     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
542     return;
543   }
544 
545   if (cid == p_hcon->ctrl_cid) {
546     p_hcon->ctrl_cid = 0;
547   } else {
548     p_hcon->intr_cid = 0;
549 
550     // now disconnect CTRL
551     L2CA_DisconnectReq(p_hcon->ctrl_cid);
552   }
553 
554   if ((p_hcon->ctrl_cid == 0) && (p_hcon->intr_cid == 0)) {
555     HIDD_TRACE_EVENT("%s: INTR and CTRL disconnected", __func__);
556 
557     hd_cb.device.state = HIDD_DEV_NO_CONN;
558     p_hcon->conn_state = HID_CONN_STATE_UNUSED;
559 
560     if (hd_cb.pending_vc_unplug) {
561       hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_VC_UNPLUG,
562                      p_hcon->disc_reason, NULL);
563       hd_cb.pending_vc_unplug = FALSE;
564     } else {
565       hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE,
566                      p_hcon->disc_reason, NULL);
567     }
568   }
569 }
570 
571 /*******************************************************************************
572  *
573  * Function         hidd_l2cif_cong_ind
574  *
575  * Description      Handles L2CAP congestion status event
576  *
577  * Returns          void
578  *
579  ******************************************************************************/
hidd_l2cif_cong_ind(uint16_t cid,bool congested)580 static void hidd_l2cif_cong_ind(uint16_t cid, bool congested) {
581   tHID_CONN* p_hcon;
582 
583   HIDD_TRACE_EVENT("%s: cid=%04x congested=%d", __func__, cid, congested);
584 
585   p_hcon = &hd_cb.device.conn;
586 
587   if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
588       (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
589     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
590     return;
591   }
592 
593   if (congested) {
594     p_hcon->conn_flags |= HID_CONN_FLAGS_CONGESTED;
595   } else {
596     p_hcon->conn_flags &= ~HID_CONN_FLAGS_CONGESTED;
597   }
598 }
599 
600 /*******************************************************************************
601  *
602  * Function         hidd_l2cif_data_ind
603  *
604  * Description      Handler incoming data on L2CAP channel
605  *
606  * Returns          void
607  *
608  ******************************************************************************/
hidd_l2cif_data_ind(uint16_t cid,BT_HDR * p_msg)609 static void hidd_l2cif_data_ind(uint16_t cid, BT_HDR* p_msg) {
610   tHID_CONN* p_hcon;
611   uint8_t* p_data = (uint8_t*)(p_msg + 1) + p_msg->offset;
612   uint8_t msg_type, param;
613   bool err = FALSE;
614 
615   HIDD_TRACE_EVENT("%s: cid=%04x", __func__, cid);
616 
617   if (p_msg->len < 1) {
618     HIDD_TRACE_ERROR("Invalid data length, ignore");
619     osi_free(p_msg);
620     return;
621   }
622 
623   p_hcon = &hd_cb.device.conn;
624 
625   if (p_hcon->conn_state == HID_CONN_STATE_UNUSED ||
626       (p_hcon->ctrl_cid != cid && p_hcon->intr_cid != cid)) {
627     HIDD_TRACE_WARNING("%s: unknown cid", __func__);
628     osi_free(p_msg);
629     return;
630   }
631 
632   msg_type = HID_GET_TRANS_FROM_HDR(*p_data);
633   param = HID_GET_PARAM_FROM_HDR(*p_data);
634 
635   if (msg_type == HID_TRANS_DATA && cid == p_hcon->intr_cid) {
636     // skip HID header
637     p_msg->offset++;
638     p_msg->len--;
639 
640     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_INTR_DATA, 0, p_msg);
641     return;
642   }
643 
644   switch (msg_type) {
645     case HID_TRANS_GET_REPORT:
646       // at this stage we don't know if Report Id shall be included in request
647       // so we pass complete packet in callback and let other code analyze this
648       hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_GET_REPORT,
649                      !!(param & HID_PAR_GET_REP_BUFSIZE_FOLLOWS), p_msg);
650       break;
651 
652     case HID_TRANS_SET_REPORT:
653       // as above
654       hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SET_REPORT, 0, p_msg);
655       break;
656 
657     case HID_TRANS_GET_IDLE:
658       hidd_conn_send_data(HID_CHANNEL_CTRL, HID_TRANS_DATA,
659                           HID_PAR_REP_TYPE_OTHER, hd_cb.device.idle_time, 0,
660                           NULL);
661       osi_free(p_msg);
662       break;
663 
664     case HID_TRANS_SET_IDLE:
665       if (p_msg->len != 2) {
666         HIDD_TRACE_ERROR("%s: invalid len (%d) set idle request received",
667                          __func__, p_msg->len);
668         err = TRUE;
669       } else {
670         hd_cb.device.idle_time = p_data[1];
671         HIDD_TRACE_DEBUG("%s: idle_time = %d", __func__,
672                          hd_cb.device.idle_time);
673         if (hd_cb.device.idle_time) {
674           HIDD_TRACE_WARNING(
675               "%s: idle_time of %d ms not supported by HID Device", __func__,
676               (hd_cb.device.idle_time * 4));
677           err = TRUE;
678         }
679       }
680       if (!err) {
681         hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
682                             HID_PAR_HANDSHAKE_RSP_SUCCESS, 0, 0, NULL);
683       } else {
684         hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
685                             HID_PAR_HANDSHAKE_RSP_ERR_INVALID_PARAM, 0, 0,
686                             NULL);
687       }
688       osi_free(p_msg);
689       break;
690 
691     case HID_TRANS_GET_PROTOCOL:
692       hidd_conn_send_data(HID_CHANNEL_CTRL, HID_TRANS_DATA,
693                           HID_PAR_REP_TYPE_OTHER, !hd_cb.device.boot_mode, 0,
694                           NULL);
695       osi_free(p_msg);
696       break;
697 
698     case HID_TRANS_SET_PROTOCOL:
699       hd_cb.device.boot_mode = !(param & HID_PAR_PROTOCOL_MASK);
700       hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SET_PROTOCOL,
701                      param & HID_PAR_PROTOCOL_MASK, NULL);
702       hidd_conn_send_data(0, HID_TRANS_HANDSHAKE, HID_PAR_HANDSHAKE_RSP_SUCCESS,
703                           0, 0, NULL);
704       osi_free(p_msg);
705       break;
706 
707     case HID_TRANS_CONTROL:
708       switch (param) {
709         case HID_PAR_CONTROL_SUSPEND:
710           hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_SUSPEND, 0, NULL);
711           break;
712 
713         case HID_PAR_CONTROL_EXIT_SUSPEND:
714           hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_EXIT_SUSPEND, 0,
715                          NULL);
716           break;
717 
718         case HID_PAR_CONTROL_VIRTUAL_CABLE_UNPLUG:
719           hidd_conn_disconnect();
720 
721           // set flag so we can notify properly when disconnected
722           hd_cb.pending_vc_unplug = TRUE;
723           break;
724       }
725 
726       osi_free(p_msg);
727       break;
728 
729     case HID_TRANS_DATA:
730     default:
731       HIDD_TRACE_WARNING("%s: got unsupported msg (%d)", __func__, msg_type);
732       hidd_conn_send_data(0, HID_TRANS_HANDSHAKE,
733                           HID_PAR_HANDSHAKE_RSP_ERR_UNSUPPORTED_REQ, 0, 0,
734                           NULL);
735       osi_free(p_msg);
736       break;
737   }
738 }
739 
740 /*******************************************************************************
741  *
742  * Function         hidd_conn_reg
743  *
744  * Description      Registers L2CAP channels
745  *
746  * Returns          void
747  *
748  ******************************************************************************/
hidd_conn_reg(void)749 tHID_STATUS hidd_conn_reg(void) {
750   HIDD_TRACE_API("%s", __func__);
751 
752   memset(&hd_cb.l2cap_cfg, 0, sizeof(tL2CAP_CFG_INFO));
753 
754   hd_cb.l2cap_cfg.mtu_present = TRUE;
755   hd_cb.l2cap_cfg.mtu = HID_DEV_MTU_SIZE;
756   hd_cb.l2cap_cfg.flush_to_present = TRUE;
757   hd_cb.l2cap_cfg.flush_to = HID_DEV_FLUSH_TO;
758 
759   memset(&hd_cb.l2cap_intr_cfg, 0, sizeof(tL2CAP_CFG_INFO));
760   hd_cb.l2cap_intr_cfg.mtu_present = TRUE;
761   hd_cb.l2cap_intr_cfg.mtu = HID_DEV_MTU_SIZE;
762   hd_cb.l2cap_intr_cfg.flush_to_present = TRUE;
763   hd_cb.l2cap_intr_cfg.flush_to = HID_DEV_FLUSH_TO;
764 
765   if (!L2CA_Register(HID_PSM_CONTROL, (tL2CAP_APPL_INFO*)&dev_reg_info,
766                      false /* enable_snoop */, nullptr)) {
767     HIDD_TRACE_ERROR("HID Control (device) registration failed");
768     return (HID_ERR_L2CAP_FAILED);
769   }
770 
771   if (!L2CA_Register(HID_PSM_INTERRUPT, (tL2CAP_APPL_INFO*)&dev_reg_info,
772                      false /* enable_snoop */, nullptr)) {
773     L2CA_Deregister(HID_PSM_CONTROL);
774     HIDD_TRACE_ERROR("HID Interrupt (device) registration failed");
775     return (HID_ERR_L2CAP_FAILED);
776   }
777 
778   return (HID_SUCCESS);
779 }
780 
781 /*******************************************************************************
782  *
783  * Function         hidd_conn_dereg
784  *
785  * Description      Deregisters L2CAP channels
786  *
787  * Returns          void
788  *
789  ******************************************************************************/
hidd_conn_dereg(void)790 void hidd_conn_dereg(void) {
791   HIDD_TRACE_API("%s", __func__);
792 
793   L2CA_Deregister(HID_PSM_CONTROL);
794   L2CA_Deregister(HID_PSM_INTERRUPT);
795 }
796 
797 /*******************************************************************************
798  *
799  * Function         hidd_conn_initiate
800  *
801  * Description      Initiates HID connection to plugged device
802  *
803  * Returns          HID_SUCCESS
804  *
805  ******************************************************************************/
hidd_conn_initiate(void)806 tHID_STATUS hidd_conn_initiate(void) {
807   tHID_DEV_DEV_CTB* p_dev = &hd_cb.device;
808 
809   HIDD_TRACE_API("%s", __func__);
810 
811   if (!p_dev->in_use) {
812     HIDD_TRACE_WARNING("%s: no virtual cable established", __func__);
813     return (HID_ERR_NOT_REGISTERED);
814   }
815 
816   if (p_dev->conn.conn_state != HID_CONN_STATE_UNUSED) {
817     HIDD_TRACE_WARNING("%s: connection already in progress", __func__);
818     return (HID_ERR_CONN_IN_PROCESS);
819   }
820 
821   p_dev->conn.ctrl_cid = 0;
822   p_dev->conn.intr_cid = 0;
823   p_dev->conn.disc_reason = HID_L2CAP_CONN_FAIL;
824 
825   p_dev->conn.conn_flags = HID_CONN_FLAGS_IS_ORIG;
826 
827   BTM_SetOutService(p_dev->addr, BTM_SEC_SERVICE_HIDD_SEC_CTRL, HIDD_SEC_CHN);
828 
829   /* Check if L2CAP started the connection process */
830   if ((p_dev->conn.ctrl_cid = L2CA_ConnectReq(HID_PSM_CONTROL, p_dev->addr)) ==
831       0) {
832     HIDD_TRACE_WARNING("%s: could not start L2CAP connection", __func__);
833     hd_cb.callback(hd_cb.device.addr, HID_DHOST_EVT_CLOSE, HID_ERR_L2CAP_FAILED,
834                    NULL);
835   } else {
836     p_dev->conn.conn_state = HID_CONN_STATE_CONNECTING_CTRL;
837   }
838 
839   return (HID_SUCCESS);
840 }
841 
842 /*******************************************************************************
843  *
844  * Function         hidd_conn_disconnect
845  *
846  * Description      Disconnects existing HID connection
847  *
848  * Returns          HID_SUCCESS
849  *
850  ******************************************************************************/
hidd_conn_disconnect(void)851 tHID_STATUS hidd_conn_disconnect(void) {
852   tHID_CONN* p_hcon;
853 
854   HIDD_TRACE_API("%s", __func__);
855 
856   // clean any outstanding data on intr
857   if (hd_cb.pending_data) {
858     osi_free(hd_cb.pending_data);
859     hd_cb.pending_data = NULL;
860   }
861 
862   p_hcon = &hd_cb.device.conn;
863 
864   if ((p_hcon->ctrl_cid != 0) || (p_hcon->intr_cid != 0)) {
865     p_hcon->conn_state = HID_CONN_STATE_DISCONNECTING;
866 
867     /* Set l2cap idle timeout to 0 (so ACL link is disconnected
868      * immediately after last channel is closed) */
869     L2CA_SetIdleTimeoutByBdAddr(hd_cb.device.addr, 0, BT_TRANSPORT_BR_EDR);
870 
871     if (p_hcon->intr_cid) {
872       L2CA_DisconnectReq(p_hcon->intr_cid);
873     } else if (p_hcon->ctrl_cid) {
874       L2CA_DisconnectReq(p_hcon->ctrl_cid);
875     }
876   } else {
877     HIDD_TRACE_WARNING("%s: already disconnected", __func__);
878     p_hcon->conn_state = HID_CONN_STATE_UNUSED;
879   }
880 
881   return (HID_SUCCESS);
882 }
883 
884 /*******************************************************************************
885  *
886  * Function         hidd_conn_send_data
887  *
888  * Description      Sends data to host
889  *
890  * Returns          tHID_STATUS
891  *
892  ******************************************************************************/
hidd_conn_send_data(uint8_t channel,uint8_t msg_type,uint8_t param,uint8_t data,uint16_t len,uint8_t * p_data)893 tHID_STATUS hidd_conn_send_data(uint8_t channel, uint8_t msg_type,
894                                 uint8_t param, uint8_t data, uint16_t len,
895                                 uint8_t* p_data) {
896   tHID_CONN* p_hcon;
897   BT_HDR* p_buf;
898   uint8_t* p_out;
899   uint16_t cid;
900   uint16_t buf_size;
901 
902   HIDD_TRACE_VERBOSE("%s: channel(%d), msg_type(%d), len(%d)", __func__,
903                      channel, msg_type, len);
904 
905   p_hcon = &hd_cb.device.conn;
906 
907   if (p_hcon->conn_flags & HID_CONN_FLAGS_CONGESTED) {
908     return HID_ERR_CONGESTED;
909   }
910 
911   switch (msg_type) {
912     case HID_TRANS_HANDSHAKE:
913     case HID_TRANS_CONTROL:
914       cid = p_hcon->ctrl_cid;
915       buf_size = HID_CONTROL_BUF_SIZE;
916       break;
917     case HID_TRANS_DATA:
918       if (channel == HID_CHANNEL_CTRL) {
919         cid = p_hcon->ctrl_cid;
920         buf_size = HID_CONTROL_BUF_SIZE;
921       } else {
922         cid = p_hcon->intr_cid;
923         buf_size = HID_INTERRUPT_BUF_SIZE;
924       }
925       break;
926     default:
927       return (HID_ERR_INVALID_PARAM);
928   }
929 
930   p_buf = (BT_HDR*)osi_malloc(buf_size);
931   if (p_buf == NULL) return (HID_ERR_NO_RESOURCES);
932 
933   p_buf->offset = L2CAP_MIN_OFFSET;
934 
935   p_out = (uint8_t*)(p_buf + 1) + p_buf->offset;
936 
937   *p_out = HID_BUILD_HDR(msg_type, param);
938   p_out++;
939 
940   p_buf->len = 1;  // start with header only
941 
942   // add report id prefix only if non-zero (which is reserved)
943   if (msg_type == HID_TRANS_DATA && (data || param == HID_PAR_REP_TYPE_OTHER)) {
944     *p_out = data;  // report_id
945     p_out++;
946     p_buf->len++;
947   }
948 
949   if (len > 0 && p_data != NULL) {
950     memcpy(p_out, p_data, len);
951     p_buf->len += len;
952   }
953 
954   // check if connected
955   if (hd_cb.device.state != HIDD_DEV_CONNECTED) {
956     // for DATA on intr we hold transfer and try to reconnect
957     if (msg_type == HID_TRANS_DATA && cid == p_hcon->intr_cid) {
958       // drop previous data, we do not queue it for now
959       if (hd_cb.pending_data) {
960         osi_free(hd_cb.pending_data);
961       }
962 
963       hd_cb.pending_data = p_buf;
964 
965       if (hd_cb.device.conn.conn_state == HID_CONN_STATE_UNUSED) {
966         hidd_conn_initiate();
967       }
968 
969       return HID_SUCCESS;
970     }
971 
972     return HID_ERR_NO_CONNECTION;
973   }
974 
975 #ifdef REPORT_TRANSFER_TIMESTAMP
976   if (report_transfer) {
977     HIDD_TRACE_ERROR("%s: report sent", __func__);
978   }
979 #endif
980   HIDD_TRACE_VERBOSE("%s: report sent", __func__);
981 
982   if (!L2CA_DataWrite(cid, p_buf)) return (HID_ERR_CONGESTED);
983 
984   return (HID_SUCCESS);
985 }
986