6 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
8 * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com>
11 * Redistribution and use in source and binary forms, with or without
12 * modification, are permitted provided that the following conditions
14 * 1. Redistributions of source code must retain the above copyright
15 * notice, this list of conditions and the following disclaimer.
16 * 2. Redistributions in binary form must reproduce the above copyright
17 * notice, this list of conditions and the following disclaimer in the
18 * documentation and/or other materials provided with the distribution.
20 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
21 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
23 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
24 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
25 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
26 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
27 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
28 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
29 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
32 * $Id: ng_hci_evnt.c,v 1.6 2003/09/08 18:57:51 max Exp $
36 #include <sys/param.h>
37 #include <sys/systm.h>
38 #include <sys/kernel.h>
39 #include <sys/endian.h>
40 #include <sys/malloc.h>
42 #include <sys/queue.h>
43 #include <netgraph/ng_message.h>
44 #include <netgraph/netgraph.h>
45 #include <netgraph/bluetooth/include/ng_bluetooth.h>
46 #include <netgraph/bluetooth/include/ng_hci.h>
47 #include <netgraph/bluetooth/hci/ng_hci_var.h>
48 #include <netgraph/bluetooth/hci/ng_hci_cmds.h>
49 #include <netgraph/bluetooth/hci/ng_hci_evnt.h>
50 #include <netgraph/bluetooth/hci/ng_hci_ulpi.h>
51 #include <netgraph/bluetooth/hci/ng_hci_misc.h>
53 /******************************************************************************
54 ******************************************************************************
55 ** HCI event processing module
56 ******************************************************************************
57 ******************************************************************************/
60 * Event processing routines
63 static int inquiry_result (ng_hci_unit_p, struct mbuf *);
64 static int con_compl (ng_hci_unit_p, struct mbuf *);
65 static int con_req (ng_hci_unit_p, struct mbuf *);
66 static int discon_compl (ng_hci_unit_p, struct mbuf *);
67 static int encryption_change (ng_hci_unit_p, struct mbuf *);
68 static int read_remote_features_compl (ng_hci_unit_p, struct mbuf *);
69 static int qos_setup_compl (ng_hci_unit_p, struct mbuf *);
70 static int hardware_error (ng_hci_unit_p, struct mbuf *);
71 static int role_change (ng_hci_unit_p, struct mbuf *);
72 static int num_compl_pkts (ng_hci_unit_p, struct mbuf *);
73 static int mode_change (ng_hci_unit_p, struct mbuf *);
74 static int data_buffer_overflow (ng_hci_unit_p, struct mbuf *);
75 static int read_clock_offset_compl (ng_hci_unit_p, struct mbuf *);
76 static int qos_violation (ng_hci_unit_p, struct mbuf *);
77 static int page_scan_mode_change (ng_hci_unit_p, struct mbuf *);
78 static int page_scan_rep_mode_change (ng_hci_unit_p, struct mbuf *);
79 static int sync_con_queue (ng_hci_unit_p, ng_hci_unit_con_p, int);
80 static int send_data_packets (ng_hci_unit_p, int, int);
81 static int le_event (ng_hci_unit_p, struct mbuf *);
84 * Process HCI event packet
88 ng_hci_process_event(ng_hci_unit_p unit, struct mbuf *event)
90 ng_hci_event_pkt_t *hdr = NULL;
93 /* Get event packet header */
94 NG_HCI_M_PULLUP(event, sizeof(*hdr));
98 hdr = mtod(event, ng_hci_event_pkt_t *);
101 "%s: %s - got HCI event=%#x, length=%d\n",
102 __func__, NG_NODE_NAME(unit->node), hdr->event, hdr->length);
104 /* Get rid of event header and process event */
105 m_adj(event, sizeof(*hdr));
107 switch (hdr->event) {
108 case NG_HCI_EVENT_INQUIRY_COMPL:
109 case NG_HCI_EVENT_RETURN_LINK_KEYS:
110 case NG_HCI_EVENT_PIN_CODE_REQ:
111 case NG_HCI_EVENT_LINK_KEY_REQ:
112 case NG_HCI_EVENT_LINK_KEY_NOTIFICATION:
113 case NG_HCI_EVENT_LOOPBACK_COMMAND:
114 case NG_HCI_EVENT_AUTH_COMPL:
115 case NG_HCI_EVENT_CHANGE_CON_LINK_KEY_COMPL:
116 case NG_HCI_EVENT_MASTER_LINK_KEY_COMPL:
117 case NG_HCI_EVENT_FLUSH_OCCUR: /* XXX Do we have to handle it? */
118 case NG_HCI_EVENT_MAX_SLOT_CHANGE:
119 case NG_HCI_EVENT_CON_PKT_TYPE_CHANGED:
120 case NG_HCI_EVENT_BT_LOGO:
121 case NG_HCI_EVENT_VENDOR:
122 case NG_HCI_EVENT_REMOTE_NAME_REQ_COMPL:
123 case NG_HCI_EVENT_READ_REMOTE_VER_INFO_COMPL:
124 /* These do not need post processing */
127 case NG_HCI_EVENT_LE:
128 error = le_event(unit, event);
131 case NG_HCI_EVENT_INQUIRY_RESULT:
132 error = inquiry_result(unit, event);
135 case NG_HCI_EVENT_CON_COMPL:
136 error = con_compl(unit, event);
139 case NG_HCI_EVENT_CON_REQ:
140 error = con_req(unit, event);
143 case NG_HCI_EVENT_DISCON_COMPL:
144 error = discon_compl(unit, event);
147 case NG_HCI_EVENT_ENCRYPTION_CHANGE:
148 error = encryption_change(unit, event);
151 case NG_HCI_EVENT_READ_REMOTE_FEATURES_COMPL:
152 error = read_remote_features_compl(unit, event);
155 case NG_HCI_EVENT_QOS_SETUP_COMPL:
156 error = qos_setup_compl(unit, event);
159 case NG_HCI_EVENT_COMMAND_COMPL:
160 error = ng_hci_process_command_complete(unit, event);
163 case NG_HCI_EVENT_COMMAND_STATUS:
164 error = ng_hci_process_command_status(unit, event);
167 case NG_HCI_EVENT_HARDWARE_ERROR:
168 error = hardware_error(unit, event);
171 case NG_HCI_EVENT_ROLE_CHANGE:
172 error = role_change(unit, event);
175 case NG_HCI_EVENT_NUM_COMPL_PKTS:
176 error = num_compl_pkts(unit, event);
179 case NG_HCI_EVENT_MODE_CHANGE:
180 error = mode_change(unit, event);
183 case NG_HCI_EVENT_DATA_BUFFER_OVERFLOW:
184 error = data_buffer_overflow(unit, event);
187 case NG_HCI_EVENT_READ_CLOCK_OFFSET_COMPL:
188 error = read_clock_offset_compl(unit, event);
191 case NG_HCI_EVENT_QOS_VIOLATION:
192 error = qos_violation(unit, event);
195 case NG_HCI_EVENT_PAGE_SCAN_MODE_CHANGE:
196 error = page_scan_mode_change(unit, event);
199 case NG_HCI_EVENT_PAGE_SCAN_REP_MODE_CHANGE:
200 error = page_scan_rep_mode_change(unit, event);
210 } /* ng_hci_process_event */
213 * Send ACL and/or SCO data to the unit driver
217 ng_hci_send_data(ng_hci_unit_p unit)
222 NG_HCI_BUFF_ACL_AVAIL(unit->buffer, count);
225 "%s: %s - sending ACL data packets, count=%d\n",
226 __func__, NG_NODE_NAME(unit->node), count);
229 count = send_data_packets(unit, NG_HCI_LINK_ACL, count);
230 NG_HCI_STAT_ACL_SENT(unit->stat, count);
231 NG_HCI_BUFF_ACL_USE(unit->buffer, count);
235 NG_HCI_BUFF_SCO_AVAIL(unit->buffer, count);
238 "%s: %s - sending SCO data packets, count=%d\n",
239 __func__, NG_NODE_NAME(unit->node), count);
242 count = send_data_packets(unit, NG_HCI_LINK_SCO, count);
243 NG_HCI_STAT_SCO_SENT(unit->stat, count);
244 NG_HCI_BUFF_SCO_USE(unit->buffer, count);
246 } /* ng_hci_send_data */
249 * Send data packets to the lower layer.
253 send_data_packets(ng_hci_unit_p unit, int link_type, int limit)
255 ng_hci_unit_con_p con = NULL, winner = NULL;
258 int min_pending, total_sent, sent, error, v;
260 for (total_sent = 0; limit > 0; ) {
261 min_pending = 0x0fffffff;
265 * Find the connection that has has data to send
266 * and the smallest number of pending packets
269 LIST_FOREACH(con, &unit->con_list, next) {
270 reallink_type = (con->link_type == NG_HCI_LINK_SCO)?
271 NG_HCI_LINK_SCO: NG_HCI_LINK_ACL;
272 if (reallink_type != link_type){
275 if (NG_BT_ITEMQ_LEN(&con->conq) == 0)
278 if (con->pending < min_pending) {
280 min_pending = con->pending;
288 * OK, we have a winner now send as much packets as we can
289 * Count the number of packets we have sent and then sync
290 * winner connection queue.
293 for (sent = 0; limit > 0; limit --, total_sent ++, sent ++) {
294 NG_BT_ITEMQ_DEQUEUE(&winner->conq, item);
299 "%s: %s - sending data packet, handle=%d, len=%d\n",
300 __func__, NG_NODE_NAME(unit->node),
301 winner->con_handle, NGI_M(item)->m_pkthdr.len);
303 /* Check if driver hook still there */
304 v = (unit->drv != NULL && NG_HOOK_IS_VALID(unit->drv));
305 if (!v || (unit->state & NG_HCI_UNIT_READY) !=
308 "%s: %s - could not send data. Hook \"%s\" is %svalid, state=%#x\n",
309 __func__, NG_NODE_NAME(unit->node),
310 NG_HCI_HOOK_DRV, ((v)? "" : "not "),
316 v = NGI_M(item)->m_pkthdr.len;
318 /* Give packet to raw hook */
319 ng_hci_mtap(unit, NGI_M(item));
321 /* ... and forward item to the driver */
322 NG_FWD_ITEM_HOOK(error, item, unit->drv);
327 "%s: %s - could not send data packet, handle=%d, error=%d\n",
328 __func__, NG_NODE_NAME(unit->node),
329 winner->con_handle, error);
334 NG_HCI_STAT_BYTES_SENT(unit->stat, v);
338 * Sync connection queue for the winner
340 sync_con_queue(unit, winner, sent);
344 } /* send_data_packets */
347 * Send flow control messages to the upper layer
351 sync_con_queue(ng_hci_unit_p unit, ng_hci_unit_con_p con, int completed)
354 struct ng_mesg *msg = NULL;
355 ng_hci_sync_con_queue_ep *state = NULL;
358 hook = (con->link_type != NG_HCI_LINK_SCO)? unit->acl : unit->sco;
359 if (hook == NULL || NG_HOOK_NOT_VALID(hook))
362 NG_MKMESSAGE(msg, NGM_HCI_COOKIE, NGM_HCI_SYNC_CON_QUEUE,
363 sizeof(*state), M_NOWAIT);
367 state = (ng_hci_sync_con_queue_ep *)(msg->data);
368 state->con_handle = con->con_handle;
369 state->completed = completed;
371 NG_SEND_MSG_HOOK(error, unit->node, msg, hook, 0);
374 } /* sync_con_queue */
376 /* Inquiry result event */
378 le_advertizing_report(ng_hci_unit_p unit, struct mbuf *event)
380 ng_hci_le_advertising_report_ep *ep = NULL;
381 ng_hci_neighbor_p n = NULL;
388 NG_HCI_M_PULLUP(event, sizeof(*ep));
392 ep = mtod(event, ng_hci_le_advertising_report_ep *);
393 num_reports = ep->num_reports;
394 m_adj(event, sizeof(*ep));
397 for (; num_reports > 0; num_reports --) {
398 /* Get remote unit address */
399 NG_HCI_M_PULLUP(event, sizeof(u_int8_t));
400 event_type = *mtod(event, u_int8_t *);
401 m_adj(event, sizeof(u_int8_t));
402 NG_HCI_M_PULLUP(event, sizeof(u_int8_t));
403 addr_type = *mtod(event, u_int8_t *);
404 m_adj(event, sizeof(u_int8_t));
406 m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr);
407 m_adj(event, sizeof(bdaddr));
409 /* Lookup entry in the cache */
410 n = ng_hci_get_neighbor(unit, &bdaddr, (addr_type) ? NG_HCI_LINK_LE_RANDOM:NG_HCI_LINK_LE_PUBLIC);
412 /* Create new entry */
413 n = ng_hci_new_neighbor(unit);
418 bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr));
419 n->addrtype = (addr_type)? NG_HCI_LINK_LE_RANDOM :
420 NG_HCI_LINK_LE_PUBLIC;
423 getmicrotime(&n->updated);
427 * TODO: Make these information
428 * Available from userland.
430 u_int8_t length_data;
432 event = m_pullup(event, sizeof(u_int8_t));
434 NG_HCI_WARN("%s: Event datasize Pullup Failed\n", __func__);
437 length_data = *mtod(event, u_int8_t *);
438 m_adj(event, sizeof(u_int8_t));
439 n->extinq_size = (length_data < NG_HCI_EXTINQ_MAX)?
440 length_data : NG_HCI_EXTINQ_MAX;
442 /*Advertizement data*/
443 event = m_pullup(event, n->extinq_size);
445 NG_HCI_WARN("%s: Event data pullup Failed\n", __func__);
448 m_copydata(event, 0, n->extinq_size, n->extinq_data);
449 m_adj(event, n->extinq_size);
450 event = m_pullup(event, sizeof(char ));
453 NG_HCI_WARN("%s: Event rssi pull up Failed\n", __func__);
457 n->page_scan_mode = *mtod(event, char *);
458 m_adj(event, sizeof(u_int8_t));
465 } /* inquiry_result */
467 static int le_connection_complete(ng_hci_unit_p unit, struct mbuf *event)
471 ng_hci_le_connection_complete_ep *ep = NULL;
472 ng_hci_unit_con_p con = NULL;
474 uint8_t uclass[3] = {0,0,0};//dummy uclass
476 NG_HCI_M_PULLUP(event, sizeof(*ep));
480 ep = mtod(event, ng_hci_le_connection_complete_ep *);
481 link_type = (ep->address_type)? NG_HCI_LINK_LE_RANDOM :
482 NG_HCI_LINK_LE_PUBLIC;
484 * Find the first connection descriptor that matches the following:
486 * 1) con->link_type == link_type
487 * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE
488 * 3) con->bdaddr == ep->address
490 LIST_FOREACH(con, &unit->con_list, next)
491 if (con->link_type == link_type &&
492 con->state == NG_HCI_CON_W4_CONN_COMPLETE &&
493 bcmp(&con->bdaddr, &ep->address, sizeof(bdaddr_t)) == 0)
497 * Two possible cases:
499 * 1) We have found connection descriptor. That means upper layer has
500 * requested this connection via LP_CON_REQ message. In this case
501 * connection must have timeout set. If ng_hci_con_untimeout() fails
502 * then timeout message already went into node's queue. In this case
503 * ignore Connection_Complete event and let timeout deal with it.
505 * 2) We do not have connection descriptor. That means upper layer
506 * nas not requested this connection , (less likely) we gave up
507 * on this connection (timeout) or as node act as slave role.
508 * The most likely scenario is that
509 * we have received LE_Create_Connection command
517 con = ng_hci_new_con(unit, link_type);
523 con->state = NG_HCI_CON_W4_LP_CON_RSP;
524 ng_hci_con_timeout(con);
526 bcopy(&ep->address, &con->bdaddr, sizeof(con->bdaddr));
527 error = ng_hci_lp_con_ind(con, uclass);
529 ng_hci_con_untimeout(con);
530 ng_hci_free_con(con);
534 } else if ((error = ng_hci_con_untimeout(con)) != 0)
538 * Update connection descriptor and send notification
539 * to the upper layers.
542 con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->handle));
543 con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE;
545 ng_hci_lp_con_cfm(con, ep->status);
547 /* Adjust connection state */
549 ng_hci_free_con(con);
551 con->state = NG_HCI_CON_OPEN;
554 * Change link policy for the ACL connections. Enable all
555 * supported link modes. Enable Role switch as well if
556 * device supports it.
567 static int le_connection_update(ng_hci_unit_p unit, struct mbuf *event)
577 le_event(ng_hci_unit_p unit, struct mbuf *event)
582 NG_HCI_M_PULLUP(event, sizeof(*lep));
586 lep = mtod(event, ng_hci_le_ep *);
587 m_adj(event, sizeof(*lep));
588 switch(lep->subevent_code){
589 case NG_HCI_LEEV_CON_COMPL:
590 le_connection_complete(unit, event);
592 case NG_HCI_LEEV_ADVREP:
593 le_advertizing_report(unit, event);
595 case NG_HCI_LEEV_CON_UPDATE_COMPL:
596 le_connection_update(unit, event);
598 case NG_HCI_LEEV_READ_REMOTE_FEATURES_COMPL:
601 case NG_HCI_LEEV_LONG_TERM_KEY_REQUEST:
610 /* Inquiry result event */
612 inquiry_result(ng_hci_unit_p unit, struct mbuf *event)
614 ng_hci_inquiry_result_ep *ep = NULL;
615 ng_hci_neighbor_p n = NULL;
619 NG_HCI_M_PULLUP(event, sizeof(*ep));
623 ep = mtod(event, ng_hci_inquiry_result_ep *);
624 m_adj(event, sizeof(*ep));
626 for (; ep->num_responses > 0; ep->num_responses --) {
627 /* Get remote unit address */
628 m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr);
629 m_adj(event, sizeof(bdaddr));
631 /* Lookup entry in the cache */
632 n = ng_hci_get_neighbor(unit, &bdaddr, NG_HCI_LINK_ACL);
634 /* Create new entry */
635 n = ng_hci_new_neighbor(unit);
641 getmicrotime(&n->updated);
643 bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr));
644 n->addrtype = NG_HCI_LINK_ACL;
646 /* XXX call m_pullup here? */
648 n->page_scan_rep_mode = *mtod(event, u_int8_t *);
649 m_adj(event, sizeof(u_int8_t));
651 /* page_scan_period_mode */
652 m_adj(event, sizeof(u_int8_t));
654 n->page_scan_mode = *mtod(event, u_int8_t *);
655 m_adj(event, sizeof(u_int8_t));
658 m_adj(event, NG_HCI_CLASS_SIZE);
661 m_copydata(event, 0, sizeof(n->clock_offset),
662 (caddr_t) &n->clock_offset);
663 n->clock_offset = le16toh(n->clock_offset);
669 } /* inquiry_result */
671 /* Connection complete event */
673 con_compl(ng_hci_unit_p unit, struct mbuf *event)
675 ng_hci_con_compl_ep *ep = NULL;
676 ng_hci_unit_con_p con = NULL;
679 NG_HCI_M_PULLUP(event, sizeof(*ep));
683 ep = mtod(event, ng_hci_con_compl_ep *);
686 * Find the first connection descriptor that matches the following:
688 * 1) con->link_type == ep->link_type
689 * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE
690 * 3) con->bdaddr == ep->bdaddr
693 LIST_FOREACH(con, &unit->con_list, next)
694 if (con->link_type == ep->link_type &&
695 con->state == NG_HCI_CON_W4_CONN_COMPLETE &&
696 bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
700 * Two possible cases:
702 * 1) We have found connection descriptor. That means upper layer has
703 * requested this connection via LP_CON_REQ message. In this case
704 * connection must have timeout set. If ng_hci_con_untimeout() fails
705 * then timeout message already went into node's queue. In this case
706 * ignore Connection_Complete event and let timeout deal with it.
708 * 2) We do not have connection descriptor. That means upper layer
709 * nas not requested this connection or (less likely) we gave up
710 * on this connection (timeout). The most likely scenario is that
711 * we have received Create_Connection/Add_SCO_Connection command
719 con = ng_hci_new_con(unit, ep->link_type);
725 bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
726 } else if ((error = ng_hci_con_untimeout(con)) != 0)
730 * Update connection descriptor and send notification
731 * to the upper layers.
734 con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
735 con->encryption_mode = ep->encryption_mode;
737 ng_hci_lp_con_cfm(con, ep->status);
739 /* Adjust connection state */
741 ng_hci_free_con(con);
743 con->state = NG_HCI_CON_OPEN;
746 * Change link policy for the ACL connections. Enable all
747 * supported link modes. Enable Role switch as well if
748 * device supports it.
751 if (ep->link_type == NG_HCI_LINK_ACL) {
752 struct __link_policy {
753 ng_hci_cmd_pkt_t hdr;
754 ng_hci_write_link_policy_settings_cp cp;
755 } __attribute__ ((packed)) *lp;
758 MGETHDR(m, M_NOWAIT, MT_DATA);
760 m->m_pkthdr.len = m->m_len = sizeof(*lp);
761 lp = mtod(m, struct __link_policy *);
763 lp->hdr.type = NG_HCI_CMD_PKT;
764 lp->hdr.opcode = htole16(NG_HCI_OPCODE(
765 NG_HCI_OGF_LINK_POLICY,
766 NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS));
767 lp->hdr.length = sizeof(lp->cp);
769 lp->cp.con_handle = ep->con_handle;
772 if ((unit->features[0] & NG_HCI_LMP_SWITCH) &&
774 lp->cp.settings |= 0x1;
775 if (unit->features[0] & NG_HCI_LMP_HOLD_MODE)
776 lp->cp.settings |= 0x2;
777 if (unit->features[0] & NG_HCI_LMP_SNIFF_MODE)
778 lp->cp.settings |= 0x4;
779 if (unit->features[1] & NG_HCI_LMP_PARK_MODE)
780 lp->cp.settings |= 0x8;
782 lp->cp.settings &= unit->link_policy_mask;
783 lp->cp.settings = htole16(lp->cp.settings);
785 NG_BT_MBUFQ_ENQUEUE(&unit->cmdq, m);
786 if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING))
787 ng_hci_send_command(unit);
797 /* Connection request event */
799 con_req(ng_hci_unit_p unit, struct mbuf *event)
801 ng_hci_con_req_ep *ep = NULL;
802 ng_hci_unit_con_p con = NULL;
805 NG_HCI_M_PULLUP(event, sizeof(*ep));
809 ep = mtod(event, ng_hci_con_req_ep *);
812 * Find the first connection descriptor that matches the following:
814 * 1) con->link_type == ep->link_type
816 * 2) con->state == NG_HCI_CON_W4_LP_CON_RSP ||
817 * con->state == NG_HCI_CON_W4_CONN_COMPL
819 * 3) con->bdaddr == ep->bdaddr
823 * 1) We do not have connection descriptor. This is simple. Create
824 * new fresh connection descriptor and send notification to the
825 * appropriate upstream hook (based on link_type).
827 * 2) We found connection handle. This is more complicated.
831 * Since only one ACL link can exist between each pair of
832 * units then we have a race. Our upper layer has requested
833 * an ACL connection to the remote unit, but we did not send
834 * command yet. At the same time the remote unit has requested
835 * an ACL connection from us. In this case we will ignore
836 * Connection_Request event. This probably will cause connect
837 * failure on both units.
841 * The spec on page 45 says :
843 * "The master can support up to three SCO links to the same
844 * slave or to different slaves. A slave can support up to
845 * three SCO links from the same master, or two SCO links if
846 * the links originate from different masters."
848 * The only problem is how to handle multiple SCO links between
849 * matster and slave. For now we will assume that multiple SCO
850 * links MUST be opened one after another.
853 LIST_FOREACH(con, &unit->con_list, next)
854 if (con->link_type == ep->link_type &&
855 (con->state == NG_HCI_CON_W4_LP_CON_RSP ||
856 con->state == NG_HCI_CON_W4_CONN_COMPLETE) &&
857 bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0)
861 con = ng_hci_new_con(unit, ep->link_type);
863 bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr));
865 con->state = NG_HCI_CON_W4_LP_CON_RSP;
866 ng_hci_con_timeout(con);
868 error = ng_hci_lp_con_ind(con, ep->uclass);
870 ng_hci_con_untimeout(con);
871 ng_hci_free_con(con);
882 /* Disconnect complete event */
884 discon_compl(ng_hci_unit_p unit, struct mbuf *event)
886 ng_hci_discon_compl_ep *ep = NULL;
887 ng_hci_unit_con_p con = NULL;
891 NG_HCI_M_PULLUP(event, sizeof(*ep));
895 ep = mtod(event, ng_hci_discon_compl_ep *);
899 * Do we have to send notification if ep->status != 0?
900 * For now we will send notification for both ACL and SCO connections
901 * ONLY if ep->status == 0.
904 if (ep->status == 0) {
905 h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
906 con = ng_hci_con_by_handle(unit, h);
908 error = ng_hci_lp_discon_ind(con, ep->reason);
910 /* Remove all timeouts (if any) */
911 if (con->flags & NG_HCI_CON_TIMEOUT_PENDING)
912 ng_hci_con_untimeout(con);
914 ng_hci_free_con(con);
917 "%s: %s - invalid connection handle=%d\n",
918 __func__, NG_NODE_NAME(unit->node), h);
928 /* Encryption change event */
930 encryption_change(ng_hci_unit_p unit, struct mbuf *event)
932 ng_hci_encryption_change_ep *ep = NULL;
933 ng_hci_unit_con_p con = NULL;
937 NG_HCI_M_PULLUP(event, sizeof(*ep));
941 ep = mtod(event, ng_hci_encryption_change_ep *);
942 h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
943 con = ng_hci_con_by_handle(unit, h);
945 if (ep->status == 0) {
948 "%s: %s - invalid connection handle=%d\n",
949 __func__, NG_NODE_NAME(unit->node), h);
951 } else if (con->link_type == NG_HCI_LINK_SCO) {
953 "%s: %s - invalid link type=%d\n",
954 __func__, NG_NODE_NAME(unit->node),
957 } else if (ep->encryption_enable)
958 /* XXX is that true? */
959 con->encryption_mode = NG_HCI_ENCRYPTION_MODE_P2P;
961 con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE;
964 "%s: %s - failed to change encryption mode, status=%d\n",
965 __func__, NG_NODE_NAME(unit->node), ep->status);
967 /*Anyway, propagete encryption status to upper layer*/
968 ng_hci_lp_enc_change(con, con->encryption_mode);
973 } /* encryption_change */
975 /* Read remote feature complete event */
977 read_remote_features_compl(ng_hci_unit_p unit, struct mbuf *event)
979 ng_hci_read_remote_features_compl_ep *ep = NULL;
980 ng_hci_unit_con_p con = NULL;
981 ng_hci_neighbor_p n = NULL;
985 NG_HCI_M_PULLUP(event, sizeof(*ep));
989 ep = mtod(event, ng_hci_read_remote_features_compl_ep *);
991 if (ep->status == 0) {
992 /* Check if we have this connection handle */
993 h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
994 con = ng_hci_con_by_handle(unit, h);
997 "%s: %s - invalid connection handle=%d\n",
998 __func__, NG_NODE_NAME(unit->node), h);
1003 /* Update cache entry */
1004 n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL);
1006 n = ng_hci_new_neighbor(unit);
1012 bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1013 n->addrtype = NG_HCI_LINK_ACL;
1015 getmicrotime(&n->updated);
1017 bcopy(ep->features, n->features, sizeof(n->features));
1020 "%s: %s - failed to read remote unit features, status=%d\n",
1021 __func__, NG_NODE_NAME(unit->node), ep->status);
1026 } /* read_remote_features_compl */
1028 /* QoS setup complete event */
1030 qos_setup_compl(ng_hci_unit_p unit, struct mbuf *event)
1032 ng_hci_qos_setup_compl_ep *ep = NULL;
1033 ng_hci_unit_con_p con = NULL;
1037 NG_HCI_M_PULLUP(event, sizeof(*ep));
1041 ep = mtod(event, ng_hci_qos_setup_compl_ep *);
1043 /* Check if we have this connection handle */
1044 h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1045 con = ng_hci_con_by_handle(unit, h);
1048 "%s: %s - invalid connection handle=%d\n",
1049 __func__, NG_NODE_NAME(unit->node), h);
1051 } else if (con->link_type != NG_HCI_LINK_ACL) {
1053 "%s: %s - invalid link type=%d, handle=%d\n",
1054 __func__, NG_NODE_NAME(unit->node), con->link_type, h);
1056 } else if (con->state != NG_HCI_CON_OPEN) {
1058 "%s: %s - invalid connection state=%d, handle=%d\n",
1059 __func__, NG_NODE_NAME(unit->node),
1062 } else /* Notify upper layer */
1063 error = ng_hci_lp_qos_cfm(con, ep->status);
1068 } /* qos_setup_compl */
1070 /* Hardware error event */
1072 hardware_error(ng_hci_unit_p unit, struct mbuf *event)
1075 "%s: %s - hardware error %#x\n",
1076 __func__, NG_NODE_NAME(unit->node), *mtod(event, u_int8_t *));
1081 } /* hardware_error */
1083 /* Role change event */
1085 role_change(ng_hci_unit_p unit, struct mbuf *event)
1087 ng_hci_role_change_ep *ep = NULL;
1088 ng_hci_unit_con_p con = NULL;
1090 NG_HCI_M_PULLUP(event, sizeof(*ep));
1094 ep = mtod(event, ng_hci_role_change_ep *);
1096 if (ep->status == 0) {
1097 /* XXX shoud we also change "role" for SCO connections? */
1098 con = ng_hci_con_by_bdaddr(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1100 con->role = ep->role;
1103 "%s: %s - ACL connection does not exist, bdaddr=%x:%x:%x:%x:%x:%x\n",
1104 __func__, NG_NODE_NAME(unit->node),
1105 ep->bdaddr.b[5], ep->bdaddr.b[4],
1106 ep->bdaddr.b[3], ep->bdaddr.b[2],
1107 ep->bdaddr.b[1], ep->bdaddr.b[0]);
1110 "%s: %s - failed to change role, status=%d, bdaddr=%x:%x:%x:%x:%x:%x\n",
1111 __func__, NG_NODE_NAME(unit->node), ep->status,
1112 ep->bdaddr.b[5], ep->bdaddr.b[4], ep->bdaddr.b[3],
1113 ep->bdaddr.b[2], ep->bdaddr.b[1], ep->bdaddr.b[0]);
1120 /* Number of completed packets event */
1122 num_compl_pkts(ng_hci_unit_p unit, struct mbuf *event)
1124 ng_hci_num_compl_pkts_ep *ep = NULL;
1125 ng_hci_unit_con_p con = NULL;
1128 NG_HCI_M_PULLUP(event, sizeof(*ep));
1132 ep = mtod(event, ng_hci_num_compl_pkts_ep *);
1133 m_adj(event, sizeof(*ep));
1135 for (; ep->num_con_handles > 0; ep->num_con_handles --) {
1136 /* Get connection handle */
1137 m_copydata(event, 0, sizeof(h), (caddr_t) &h);
1138 m_adj(event, sizeof(h));
1139 h = NG_HCI_CON_HANDLE(le16toh(h));
1141 /* Get number of completed packets */
1142 m_copydata(event, 0, sizeof(p), (caddr_t) &p);
1143 m_adj(event, sizeof(p));
1146 /* Check if we have this connection handle */
1147 con = ng_hci_con_by_handle(unit, h);
1150 if (con->pending < 0) {
1152 "%s: %s - pending packet counter is out of sync! " \
1153 "handle=%d, pending=%d, ncp=%d\n", __func__, NG_NODE_NAME(unit->node),
1154 con->con_handle, con->pending, p);
1159 /* Update buffer descriptor */
1160 if (con->link_type != NG_HCI_LINK_SCO)
1161 NG_HCI_BUFF_ACL_FREE(unit->buffer, p);
1163 NG_HCI_BUFF_SCO_FREE(unit->buffer, p);
1166 "%s: %s - invalid connection handle=%d\n",
1167 __func__, NG_NODE_NAME(unit->node), h);
1172 /* Send more data */
1173 ng_hci_send_data(unit);
1176 } /* num_compl_pkts */
1178 /* Mode change event */
1180 mode_change(ng_hci_unit_p unit, struct mbuf *event)
1182 ng_hci_mode_change_ep *ep = NULL;
1183 ng_hci_unit_con_p con = NULL;
1186 NG_HCI_M_PULLUP(event, sizeof(*ep));
1190 ep = mtod(event, ng_hci_mode_change_ep *);
1192 if (ep->status == 0) {
1193 u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1195 con = ng_hci_con_by_handle(unit, h);
1198 "%s: %s - invalid connection handle=%d\n",
1199 __func__, NG_NODE_NAME(unit->node), h);
1201 } else if (con->link_type != NG_HCI_LINK_ACL) {
1203 "%s: %s - invalid link type=%d\n",
1204 __func__, NG_NODE_NAME(unit->node),
1208 con->mode = ep->unit_mode;
1211 "%s: %s - failed to change mode, status=%d\n",
1212 __func__, NG_NODE_NAME(unit->node), ep->status);
1219 /* Data buffer overflow event */
1221 data_buffer_overflow(ng_hci_unit_p unit, struct mbuf *event)
1224 "%s: %s - %s data buffer overflow\n",
1225 __func__, NG_NODE_NAME(unit->node),
1226 (*mtod(event, u_int8_t *) == NG_HCI_LINK_ACL)? "ACL" : "SCO");
1231 } /* data_buffer_overflow */
1233 /* Read clock offset complete event */
1235 read_clock_offset_compl(ng_hci_unit_p unit, struct mbuf *event)
1237 ng_hci_read_clock_offset_compl_ep *ep = NULL;
1238 ng_hci_unit_con_p con = NULL;
1239 ng_hci_neighbor_p n = NULL;
1242 NG_HCI_M_PULLUP(event, sizeof(*ep));
1246 ep = mtod(event, ng_hci_read_clock_offset_compl_ep *);
1248 if (ep->status == 0) {
1249 u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1251 con = ng_hci_con_by_handle(unit, h);
1254 "%s: %s - invalid connection handle=%d\n",
1255 __func__, NG_NODE_NAME(unit->node), h);
1260 /* Update cache entry */
1261 n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL);
1263 n = ng_hci_new_neighbor(unit);
1269 bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1270 n->addrtype = NG_HCI_LINK_ACL;
1272 getmicrotime(&n->updated);
1274 n->clock_offset = le16toh(ep->clock_offset);
1277 "%s: %s - failed to Read Remote Clock Offset, status=%d\n",
1278 __func__, NG_NODE_NAME(unit->node), ep->status);
1283 } /* read_clock_offset_compl */
1285 /* QoS violation event */
1287 qos_violation(ng_hci_unit_p unit, struct mbuf *event)
1289 ng_hci_qos_violation_ep *ep = NULL;
1290 ng_hci_unit_con_p con = NULL;
1294 NG_HCI_M_PULLUP(event, sizeof(*ep));
1298 ep = mtod(event, ng_hci_qos_violation_ep *);
1300 /* Check if we have this connection handle */
1301 h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle));
1302 con = ng_hci_con_by_handle(unit, h);
1305 "%s: %s - invalid connection handle=%d\n",
1306 __func__, NG_NODE_NAME(unit->node), h);
1308 } else if (con->link_type != NG_HCI_LINK_ACL) {
1310 "%s: %s - invalid link type=%d\n",
1311 __func__, NG_NODE_NAME(unit->node), con->link_type);
1313 } else if (con->state != NG_HCI_CON_OPEN) {
1315 "%s: %s - invalid connection state=%d, handle=%d\n",
1316 __func__, NG_NODE_NAME(unit->node), con->state, h);
1318 } else /* Notify upper layer */
1319 error = ng_hci_lp_qos_ind(con);
1324 } /* qos_violation */
1326 /* Page scan mode change event */
1328 page_scan_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1330 ng_hci_page_scan_mode_change_ep *ep = NULL;
1331 ng_hci_neighbor_p n = NULL;
1334 NG_HCI_M_PULLUP(event, sizeof(*ep));
1338 ep = mtod(event, ng_hci_page_scan_mode_change_ep *);
1340 /* Update cache entry */
1341 n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1343 n = ng_hci_new_neighbor(unit);
1349 bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1350 n->addrtype = NG_HCI_LINK_ACL;
1352 getmicrotime(&n->updated);
1354 n->page_scan_mode = ep->page_scan_mode;
1359 } /* page_scan_mode_change */
1361 /* Page scan repetition mode change event */
1363 page_scan_rep_mode_change(ng_hci_unit_p unit, struct mbuf *event)
1365 ng_hci_page_scan_rep_mode_change_ep *ep = NULL;
1366 ng_hci_neighbor_p n = NULL;
1369 NG_HCI_M_PULLUP(event, sizeof(*ep));
1373 ep = mtod(event, ng_hci_page_scan_rep_mode_change_ep *);
1375 /* Update cache entry */
1376 n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL);
1378 n = ng_hci_new_neighbor(unit);
1384 bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr));
1385 n->addrtype = NG_HCI_LINK_ACL;
1387 getmicrotime(&n->updated);
1389 n->page_scan_rep_mode = ep->page_scan_rep_mode;
1394 } /* page_scan_rep_mode_change */