1878ed226SJulian Elischer /* 2878ed226SJulian Elischer * ng_hci_evnt.c 3c398230bSWarner Losh */ 4c398230bSWarner Losh 5c398230bSWarner Losh /*- 64d846d26SWarner Losh * SPDX-License-Identifier: BSD-2-Clause 7fe267a55SPedro F. Giffuni * 8878ed226SJulian Elischer * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com> 9878ed226SJulian Elischer * All rights reserved. 10878ed226SJulian Elischer * 11878ed226SJulian Elischer * Redistribution and use in source and binary forms, with or without 12878ed226SJulian Elischer * modification, are permitted provided that the following conditions 13878ed226SJulian Elischer * are met: 14878ed226SJulian Elischer * 1. Redistributions of source code must retain the above copyright 15878ed226SJulian Elischer * notice, this list of conditions and the following disclaimer. 16878ed226SJulian Elischer * 2. Redistributions in binary form must reproduce the above copyright 17878ed226SJulian Elischer * notice, this list of conditions and the following disclaimer in the 18878ed226SJulian Elischer * documentation and/or other materials provided with the distribution. 19878ed226SJulian Elischer * 20878ed226SJulian Elischer * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 21878ed226SJulian Elischer * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 22878ed226SJulian Elischer * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 23878ed226SJulian Elischer * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 24878ed226SJulian Elischer * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 25878ed226SJulian Elischer * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 26878ed226SJulian Elischer * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 27878ed226SJulian Elischer * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 28878ed226SJulian Elischer * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 29878ed226SJulian Elischer * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 30878ed226SJulian Elischer * SUCH DAMAGE. 31878ed226SJulian Elischer * 320986ab12SMaksim Yevmenkin * $Id: ng_hci_evnt.c,v 1.6 2003/09/08 18:57:51 max Exp $ 33878ed226SJulian Elischer */ 34878ed226SJulian Elischer 35878ed226SJulian Elischer #include <sys/param.h> 36878ed226SJulian Elischer #include <sys/systm.h> 37878ed226SJulian Elischer #include <sys/kernel.h> 38878ed226SJulian Elischer #include <sys/endian.h> 39878ed226SJulian Elischer #include <sys/malloc.h> 40878ed226SJulian Elischer #include <sys/mbuf.h> 41878ed226SJulian Elischer #include <sys/queue.h> 42878ed226SJulian Elischer #include <netgraph/ng_message.h> 43878ed226SJulian Elischer #include <netgraph/netgraph.h> 44b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_bluetooth.h> 45b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_hci.h> 46b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_var.h> 47b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_cmds.h> 48b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_evnt.h> 49b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_ulpi.h> 50b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_misc.h> 51878ed226SJulian Elischer 52878ed226SJulian Elischer /****************************************************************************** 53878ed226SJulian Elischer ****************************************************************************** 54878ed226SJulian Elischer ** HCI event processing module 55878ed226SJulian Elischer ****************************************************************************** 56878ed226SJulian Elischer ******************************************************************************/ 57878ed226SJulian Elischer 58878ed226SJulian Elischer /* 59878ed226SJulian Elischer * Event processing routines 60878ed226SJulian Elischer */ 61878ed226SJulian Elischer 62878ed226SJulian Elischer static int inquiry_result (ng_hci_unit_p, struct mbuf *); 63878ed226SJulian Elischer static int con_compl (ng_hci_unit_p, struct mbuf *); 64878ed226SJulian Elischer static int con_req (ng_hci_unit_p, struct mbuf *); 65878ed226SJulian Elischer static int discon_compl (ng_hci_unit_p, struct mbuf *); 66f2bb1caeSJulian Elischer static int encryption_change (ng_hci_unit_p, struct mbuf *); 67878ed226SJulian Elischer static int read_remote_features_compl (ng_hci_unit_p, struct mbuf *); 68878ed226SJulian Elischer static int qos_setup_compl (ng_hci_unit_p, struct mbuf *); 69878ed226SJulian Elischer static int hardware_error (ng_hci_unit_p, struct mbuf *); 70878ed226SJulian Elischer static int role_change (ng_hci_unit_p, struct mbuf *); 71878ed226SJulian Elischer static int num_compl_pkts (ng_hci_unit_p, struct mbuf *); 72878ed226SJulian Elischer static int mode_change (ng_hci_unit_p, struct mbuf *); 73878ed226SJulian Elischer static int data_buffer_overflow (ng_hci_unit_p, struct mbuf *); 74878ed226SJulian Elischer static int read_clock_offset_compl (ng_hci_unit_p, struct mbuf *); 75878ed226SJulian Elischer static int qos_violation (ng_hci_unit_p, struct mbuf *); 76878ed226SJulian Elischer static int page_scan_mode_change (ng_hci_unit_p, struct mbuf *); 77878ed226SJulian Elischer static int page_scan_rep_mode_change (ng_hci_unit_p, struct mbuf *); 78878ed226SJulian Elischer static int sync_con_queue (ng_hci_unit_p, ng_hci_unit_con_p, int); 79878ed226SJulian Elischer static int send_data_packets (ng_hci_unit_p, int, int); 80fbc48c2bSTakanori Watanabe static int le_event (ng_hci_unit_p, struct mbuf *); 81878ed226SJulian Elischer 82878ed226SJulian Elischer /* 83878ed226SJulian Elischer * Process HCI event packet 84878ed226SJulian Elischer */ 85878ed226SJulian Elischer 86878ed226SJulian Elischer int 87878ed226SJulian Elischer ng_hci_process_event(ng_hci_unit_p unit, struct mbuf *event) 88878ed226SJulian Elischer { 89878ed226SJulian Elischer ng_hci_event_pkt_t *hdr = NULL; 90878ed226SJulian Elischer int error = 0; 91878ed226SJulian Elischer 92878ed226SJulian Elischer /* Get event packet header */ 93878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*hdr)); 94878ed226SJulian Elischer if (event == NULL) 95878ed226SJulian Elischer return (ENOBUFS); 96878ed226SJulian Elischer 97878ed226SJulian Elischer hdr = mtod(event, ng_hci_event_pkt_t *); 98878ed226SJulian Elischer 99878ed226SJulian Elischer NG_HCI_INFO( 100878ed226SJulian Elischer "%s: %s - got HCI event=%#x, length=%d\n", 101878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), hdr->event, hdr->length); 102878ed226SJulian Elischer 103878ed226SJulian Elischer /* Get rid of event header and process event */ 104878ed226SJulian Elischer m_adj(event, sizeof(*hdr)); 105878ed226SJulian Elischer 106878ed226SJulian Elischer switch (hdr->event) { 107878ed226SJulian Elischer case NG_HCI_EVENT_INQUIRY_COMPL: 108878ed226SJulian Elischer case NG_HCI_EVENT_RETURN_LINK_KEYS: 109878ed226SJulian Elischer case NG_HCI_EVENT_PIN_CODE_REQ: 110878ed226SJulian Elischer case NG_HCI_EVENT_LINK_KEY_REQ: 111878ed226SJulian Elischer case NG_HCI_EVENT_LINK_KEY_NOTIFICATION: 112878ed226SJulian Elischer case NG_HCI_EVENT_LOOPBACK_COMMAND: 113878ed226SJulian Elischer case NG_HCI_EVENT_AUTH_COMPL: 114878ed226SJulian Elischer case NG_HCI_EVENT_CHANGE_CON_LINK_KEY_COMPL: 115878ed226SJulian Elischer case NG_HCI_EVENT_MASTER_LINK_KEY_COMPL: 116878ed226SJulian Elischer case NG_HCI_EVENT_FLUSH_OCCUR: /* XXX Do we have to handle it? */ 117878ed226SJulian Elischer case NG_HCI_EVENT_MAX_SLOT_CHANGE: 118878ed226SJulian Elischer case NG_HCI_EVENT_CON_PKT_TYPE_CHANGED: 119878ed226SJulian Elischer case NG_HCI_EVENT_BT_LOGO: 120878ed226SJulian Elischer case NG_HCI_EVENT_VENDOR: 121878ed226SJulian Elischer case NG_HCI_EVENT_REMOTE_NAME_REQ_COMPL: 122878ed226SJulian Elischer case NG_HCI_EVENT_READ_REMOTE_VER_INFO_COMPL: 123*4ae0fa8aSAndreas Kempe case NG_HCI_EVENT_IO_CAPABILITY_REQUEST: 124*4ae0fa8aSAndreas Kempe case NG_HCI_EVENT_SIMPLE_PAIRING_COMPLETE: 125878ed226SJulian Elischer /* These do not need post processing */ 126878ed226SJulian Elischer NG_FREE_M(event); 127878ed226SJulian Elischer break; 128fbc48c2bSTakanori Watanabe case NG_HCI_EVENT_LE: 129fbc48c2bSTakanori Watanabe error = le_event(unit, event); 130fbc48c2bSTakanori Watanabe break; 131878ed226SJulian Elischer 132878ed226SJulian Elischer case NG_HCI_EVENT_INQUIRY_RESULT: 133878ed226SJulian Elischer error = inquiry_result(unit, event); 134878ed226SJulian Elischer break; 135878ed226SJulian Elischer 136878ed226SJulian Elischer case NG_HCI_EVENT_CON_COMPL: 137878ed226SJulian Elischer error = con_compl(unit, event); 138878ed226SJulian Elischer break; 139878ed226SJulian Elischer 140878ed226SJulian Elischer case NG_HCI_EVENT_CON_REQ: 141878ed226SJulian Elischer error = con_req(unit, event); 142878ed226SJulian Elischer break; 143878ed226SJulian Elischer 144878ed226SJulian Elischer case NG_HCI_EVENT_DISCON_COMPL: 145878ed226SJulian Elischer error = discon_compl(unit, event); 146878ed226SJulian Elischer break; 147878ed226SJulian Elischer 148f2bb1caeSJulian Elischer case NG_HCI_EVENT_ENCRYPTION_CHANGE: 149f2bb1caeSJulian Elischer error = encryption_change(unit, event); 150f2bb1caeSJulian Elischer break; 151f2bb1caeSJulian Elischer 152878ed226SJulian Elischer case NG_HCI_EVENT_READ_REMOTE_FEATURES_COMPL: 153878ed226SJulian Elischer error = read_remote_features_compl(unit, event); 154878ed226SJulian Elischer break; 155878ed226SJulian Elischer 156878ed226SJulian Elischer case NG_HCI_EVENT_QOS_SETUP_COMPL: 157878ed226SJulian Elischer error = qos_setup_compl(unit, event); 158878ed226SJulian Elischer break; 159878ed226SJulian Elischer 160878ed226SJulian Elischer case NG_HCI_EVENT_COMMAND_COMPL: 161878ed226SJulian Elischer error = ng_hci_process_command_complete(unit, event); 162878ed226SJulian Elischer break; 163878ed226SJulian Elischer 164878ed226SJulian Elischer case NG_HCI_EVENT_COMMAND_STATUS: 165878ed226SJulian Elischer error = ng_hci_process_command_status(unit, event); 166878ed226SJulian Elischer break; 167878ed226SJulian Elischer 168878ed226SJulian Elischer case NG_HCI_EVENT_HARDWARE_ERROR: 169878ed226SJulian Elischer error = hardware_error(unit, event); 170878ed226SJulian Elischer break; 171878ed226SJulian Elischer 172878ed226SJulian Elischer case NG_HCI_EVENT_ROLE_CHANGE: 173878ed226SJulian Elischer error = role_change(unit, event); 174878ed226SJulian Elischer break; 175878ed226SJulian Elischer 176878ed226SJulian Elischer case NG_HCI_EVENT_NUM_COMPL_PKTS: 177878ed226SJulian Elischer error = num_compl_pkts(unit, event); 178878ed226SJulian Elischer break; 179878ed226SJulian Elischer 180878ed226SJulian Elischer case NG_HCI_EVENT_MODE_CHANGE: 181878ed226SJulian Elischer error = mode_change(unit, event); 182878ed226SJulian Elischer break; 183878ed226SJulian Elischer 184878ed226SJulian Elischer case NG_HCI_EVENT_DATA_BUFFER_OVERFLOW: 185878ed226SJulian Elischer error = data_buffer_overflow(unit, event); 186878ed226SJulian Elischer break; 187878ed226SJulian Elischer 188878ed226SJulian Elischer case NG_HCI_EVENT_READ_CLOCK_OFFSET_COMPL: 189878ed226SJulian Elischer error = read_clock_offset_compl(unit, event); 190878ed226SJulian Elischer break; 191878ed226SJulian Elischer 192878ed226SJulian Elischer case NG_HCI_EVENT_QOS_VIOLATION: 193878ed226SJulian Elischer error = qos_violation(unit, event); 194878ed226SJulian Elischer break; 195878ed226SJulian Elischer 196878ed226SJulian Elischer case NG_HCI_EVENT_PAGE_SCAN_MODE_CHANGE: 197878ed226SJulian Elischer error = page_scan_mode_change(unit, event); 198878ed226SJulian Elischer break; 199878ed226SJulian Elischer 200878ed226SJulian Elischer case NG_HCI_EVENT_PAGE_SCAN_REP_MODE_CHANGE: 201878ed226SJulian Elischer error = page_scan_rep_mode_change(unit, event); 202878ed226SJulian Elischer break; 203878ed226SJulian Elischer 204878ed226SJulian Elischer default: 205878ed226SJulian Elischer NG_FREE_M(event); 206878ed226SJulian Elischer error = EINVAL; 207878ed226SJulian Elischer break; 208878ed226SJulian Elischer } 209878ed226SJulian Elischer 210878ed226SJulian Elischer return (error); 211878ed226SJulian Elischer } /* ng_hci_process_event */ 212878ed226SJulian Elischer 213878ed226SJulian Elischer /* 214878ed226SJulian Elischer * Send ACL and/or SCO data to the unit driver 215878ed226SJulian Elischer */ 216878ed226SJulian Elischer 217878ed226SJulian Elischer void 218878ed226SJulian Elischer ng_hci_send_data(ng_hci_unit_p unit) 219878ed226SJulian Elischer { 220878ed226SJulian Elischer int count; 221878ed226SJulian Elischer 222878ed226SJulian Elischer /* Send ACL data */ 223878ed226SJulian Elischer NG_HCI_BUFF_ACL_AVAIL(unit->buffer, count); 224878ed226SJulian Elischer 225878ed226SJulian Elischer NG_HCI_INFO( 226878ed226SJulian Elischer "%s: %s - sending ACL data packets, count=%d\n", 227878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), count); 228878ed226SJulian Elischer 229878ed226SJulian Elischer if (count > 0) { 230878ed226SJulian Elischer count = send_data_packets(unit, NG_HCI_LINK_ACL, count); 231878ed226SJulian Elischer NG_HCI_STAT_ACL_SENT(unit->stat, count); 232878ed226SJulian Elischer NG_HCI_BUFF_ACL_USE(unit->buffer, count); 233878ed226SJulian Elischer } 234878ed226SJulian Elischer 235878ed226SJulian Elischer /* Send SCO data */ 236878ed226SJulian Elischer NG_HCI_BUFF_SCO_AVAIL(unit->buffer, count); 237878ed226SJulian Elischer 238878ed226SJulian Elischer NG_HCI_INFO( 239878ed226SJulian Elischer "%s: %s - sending SCO data packets, count=%d\n", 240878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), count); 241878ed226SJulian Elischer 242878ed226SJulian Elischer if (count > 0) { 243878ed226SJulian Elischer count = send_data_packets(unit, NG_HCI_LINK_SCO, count); 244878ed226SJulian Elischer NG_HCI_STAT_SCO_SENT(unit->stat, count); 245878ed226SJulian Elischer NG_HCI_BUFF_SCO_USE(unit->buffer, count); 246878ed226SJulian Elischer } 247878ed226SJulian Elischer } /* ng_hci_send_data */ 248878ed226SJulian Elischer 249878ed226SJulian Elischer /* 250878ed226SJulian Elischer * Send data packets to the lower layer. 251878ed226SJulian Elischer */ 252878ed226SJulian Elischer 253878ed226SJulian Elischer static int 254878ed226SJulian Elischer send_data_packets(ng_hci_unit_p unit, int link_type, int limit) 255878ed226SJulian Elischer { 256878ed226SJulian Elischer ng_hci_unit_con_p con = NULL, winner = NULL; 257fbc48c2bSTakanori Watanabe int reallink_type; 258878ed226SJulian Elischer item_p item = NULL; 259878ed226SJulian Elischer int min_pending, total_sent, sent, error, v; 260878ed226SJulian Elischer 261878ed226SJulian Elischer for (total_sent = 0; limit > 0; ) { 262878ed226SJulian Elischer min_pending = 0x0fffffff; 263878ed226SJulian Elischer winner = NULL; 264878ed226SJulian Elischer 265878ed226SJulian Elischer /* 266878ed226SJulian Elischer * Find the connection that has has data to send 267878ed226SJulian Elischer * and the smallest number of pending packets 268878ed226SJulian Elischer */ 269878ed226SJulian Elischer 270878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) { 271fbc48c2bSTakanori Watanabe reallink_type = (con->link_type == NG_HCI_LINK_SCO)? 272fbc48c2bSTakanori Watanabe NG_HCI_LINK_SCO: NG_HCI_LINK_ACL; 273fbc48c2bSTakanori Watanabe if (reallink_type != link_type){ 274878ed226SJulian Elischer continue; 275fbc48c2bSTakanori Watanabe } 276878ed226SJulian Elischer if (NG_BT_ITEMQ_LEN(&con->conq) == 0) 277878ed226SJulian Elischer continue; 278878ed226SJulian Elischer 279878ed226SJulian Elischer if (con->pending < min_pending) { 280878ed226SJulian Elischer winner = con; 281878ed226SJulian Elischer min_pending = con->pending; 282878ed226SJulian Elischer } 283878ed226SJulian Elischer } 284878ed226SJulian Elischer 285878ed226SJulian Elischer if (winner == NULL) 286878ed226SJulian Elischer break; 287878ed226SJulian Elischer 288878ed226SJulian Elischer /* 289878ed226SJulian Elischer * OK, we have a winner now send as much packets as we can 290878ed226SJulian Elischer * Count the number of packets we have sent and then sync 291878ed226SJulian Elischer * winner connection queue. 292878ed226SJulian Elischer */ 293878ed226SJulian Elischer 294878ed226SJulian Elischer for (sent = 0; limit > 0; limit --, total_sent ++, sent ++) { 295878ed226SJulian Elischer NG_BT_ITEMQ_DEQUEUE(&winner->conq, item); 296878ed226SJulian Elischer if (item == NULL) 297878ed226SJulian Elischer break; 298878ed226SJulian Elischer 299878ed226SJulian Elischer NG_HCI_INFO( 300878ed226SJulian Elischer "%s: %s - sending data packet, handle=%d, len=%d\n", 301878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 302878ed226SJulian Elischer winner->con_handle, NGI_M(item)->m_pkthdr.len); 303878ed226SJulian Elischer 304878ed226SJulian Elischer /* Check if driver hook still there */ 305878ed226SJulian Elischer v = (unit->drv != NULL && NG_HOOK_IS_VALID(unit->drv)); 306878ed226SJulian Elischer if (!v || (unit->state & NG_HCI_UNIT_READY) != 307878ed226SJulian Elischer NG_HCI_UNIT_READY) { 308878ed226SJulian Elischer NG_HCI_ERR( 309878ed226SJulian Elischer "%s: %s - could not send data. Hook \"%s\" is %svalid, state=%#x\n", 310878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 311878ed226SJulian Elischer NG_HCI_HOOK_DRV, ((v)? "" : "not "), 312878ed226SJulian Elischer unit->state); 313878ed226SJulian Elischer 314878ed226SJulian Elischer NG_FREE_ITEM(item); 315878ed226SJulian Elischer error = ENOTCONN; 316878ed226SJulian Elischer } else { 317878ed226SJulian Elischer v = NGI_M(item)->m_pkthdr.len; 318878ed226SJulian Elischer 319878ed226SJulian Elischer /* Give packet to raw hook */ 320878ed226SJulian Elischer ng_hci_mtap(unit, NGI_M(item)); 321878ed226SJulian Elischer 322878ed226SJulian Elischer /* ... and forward item to the driver */ 323878ed226SJulian Elischer NG_FWD_ITEM_HOOK(error, item, unit->drv); 324878ed226SJulian Elischer } 325878ed226SJulian Elischer 326878ed226SJulian Elischer if (error != 0) { 327878ed226SJulian Elischer NG_HCI_ERR( 328878ed226SJulian Elischer "%s: %s - could not send data packet, handle=%d, error=%d\n", 329878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 330878ed226SJulian Elischer winner->con_handle, error); 331878ed226SJulian Elischer break; 332878ed226SJulian Elischer } 333878ed226SJulian Elischer 334878ed226SJulian Elischer winner->pending ++; 335878ed226SJulian Elischer NG_HCI_STAT_BYTES_SENT(unit->stat, v); 336878ed226SJulian Elischer } 337878ed226SJulian Elischer 338878ed226SJulian Elischer /* 339878ed226SJulian Elischer * Sync connection queue for the winner 340878ed226SJulian Elischer */ 341878ed226SJulian Elischer sync_con_queue(unit, winner, sent); 342878ed226SJulian Elischer } 343878ed226SJulian Elischer 344878ed226SJulian Elischer return (total_sent); 345878ed226SJulian Elischer } /* send_data_packets */ 346878ed226SJulian Elischer 347878ed226SJulian Elischer /* 348878ed226SJulian Elischer * Send flow control messages to the upper layer 349878ed226SJulian Elischer */ 350878ed226SJulian Elischer 351878ed226SJulian Elischer static int 352878ed226SJulian Elischer sync_con_queue(ng_hci_unit_p unit, ng_hci_unit_con_p con, int completed) 353878ed226SJulian Elischer { 354878ed226SJulian Elischer hook_p hook = NULL; 355878ed226SJulian Elischer struct ng_mesg *msg = NULL; 356878ed226SJulian Elischer ng_hci_sync_con_queue_ep *state = NULL; 357878ed226SJulian Elischer int error; 358878ed226SJulian Elischer 359fbc48c2bSTakanori Watanabe hook = (con->link_type != NG_HCI_LINK_SCO)? unit->acl : unit->sco; 360878ed226SJulian Elischer if (hook == NULL || NG_HOOK_NOT_VALID(hook)) 361878ed226SJulian Elischer return (ENOTCONN); 362878ed226SJulian Elischer 363878ed226SJulian Elischer NG_MKMESSAGE(msg, NGM_HCI_COOKIE, NGM_HCI_SYNC_CON_QUEUE, 364878ed226SJulian Elischer sizeof(*state), M_NOWAIT); 365878ed226SJulian Elischer if (msg == NULL) 366878ed226SJulian Elischer return (ENOMEM); 367878ed226SJulian Elischer 368878ed226SJulian Elischer state = (ng_hci_sync_con_queue_ep *)(msg->data); 369878ed226SJulian Elischer state->con_handle = con->con_handle; 370878ed226SJulian Elischer state->completed = completed; 371878ed226SJulian Elischer 3724ae439a3SMaksim Yevmenkin NG_SEND_MSG_HOOK(error, unit->node, msg, hook, 0); 373878ed226SJulian Elischer 374878ed226SJulian Elischer return (error); 375878ed226SJulian Elischer } /* sync_con_queue */ 376fbc48c2bSTakanori Watanabe /* le meta event */ 377fbc48c2bSTakanori Watanabe /* Inquiry result event */ 378fbc48c2bSTakanori Watanabe static int 379fbc48c2bSTakanori Watanabe le_advertizing_report(ng_hci_unit_p unit, struct mbuf *event) 380fbc48c2bSTakanori Watanabe { 381fbc48c2bSTakanori Watanabe ng_hci_le_advertising_report_ep *ep = NULL; 382fbc48c2bSTakanori Watanabe ng_hci_neighbor_p n = NULL; 383fbc48c2bSTakanori Watanabe bdaddr_t bdaddr; 384fbc48c2bSTakanori Watanabe int error = 0; 3853cba89e6STakanori Watanabe int num_reports = 0; 386fbc48c2bSTakanori Watanabe u_int8_t addr_type; 387fbc48c2bSTakanori Watanabe 388fbc48c2bSTakanori Watanabe NG_HCI_M_PULLUP(event, sizeof(*ep)); 389fbc48c2bSTakanori Watanabe if (event == NULL) 390fbc48c2bSTakanori Watanabe return (ENOBUFS); 391fbc48c2bSTakanori Watanabe 392fbc48c2bSTakanori Watanabe ep = mtod(event, ng_hci_le_advertising_report_ep *); 3933cba89e6STakanori Watanabe num_reports = ep->num_reports; 394fbc48c2bSTakanori Watanabe m_adj(event, sizeof(*ep)); 3953cba89e6STakanori Watanabe ep = NULL; 396fbc48c2bSTakanori Watanabe 3973cba89e6STakanori Watanabe for (; num_reports > 0; num_reports --) { 3989f566782SJohn Baldwin /* event_type */ 399fbc48c2bSTakanori Watanabe m_adj(event, sizeof(u_int8_t)); 4009f566782SJohn Baldwin 4019f566782SJohn Baldwin /* Get remote unit address */ 402fbc48c2bSTakanori Watanabe NG_HCI_M_PULLUP(event, sizeof(u_int8_t)); 40305c0c99eSEd Maste if (event == NULL) { 40405c0c99eSEd Maste error = ENOBUFS; 40505c0c99eSEd Maste goto out; 40605c0c99eSEd Maste } 407fbc48c2bSTakanori Watanabe addr_type = *mtod(event, u_int8_t *); 408fbc48c2bSTakanori Watanabe m_adj(event, sizeof(u_int8_t)); 409fbc48c2bSTakanori Watanabe 410fbc48c2bSTakanori Watanabe m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr); 411fbc48c2bSTakanori Watanabe m_adj(event, sizeof(bdaddr)); 412fbc48c2bSTakanori Watanabe 413fbc48c2bSTakanori Watanabe /* Lookup entry in the cache */ 414fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &bdaddr, (addr_type) ? NG_HCI_LINK_LE_RANDOM:NG_HCI_LINK_LE_PUBLIC); 415fbc48c2bSTakanori Watanabe if (n == NULL) { 416fbc48c2bSTakanori Watanabe /* Create new entry */ 417fbc48c2bSTakanori Watanabe n = ng_hci_new_neighbor(unit); 418fbc48c2bSTakanori Watanabe if (n == NULL) { 419fbc48c2bSTakanori Watanabe error = ENOMEM; 420fbc48c2bSTakanori Watanabe break; 421fbc48c2bSTakanori Watanabe } 422fbc48c2bSTakanori Watanabe bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 423fbc48c2bSTakanori Watanabe n->addrtype = (addr_type)? NG_HCI_LINK_LE_RANDOM : 424fbc48c2bSTakanori Watanabe NG_HCI_LINK_LE_PUBLIC; 425fbc48c2bSTakanori Watanabe 426fbc48c2bSTakanori Watanabe } else 427fbc48c2bSTakanori Watanabe getmicrotime(&n->updated); 428fbc48c2bSTakanori Watanabe 429fbc48c2bSTakanori Watanabe { 430fbc48c2bSTakanori Watanabe /* 431fbc48c2bSTakanori Watanabe * TODO: Make these information 432fbc48c2bSTakanori Watanabe * Available from userland. 433fbc48c2bSTakanori Watanabe */ 434fbc48c2bSTakanori Watanabe u_int8_t length_data; 435fbc48c2bSTakanori Watanabe 4364aa92fe2STakanori Watanabe event = m_pullup(event, sizeof(u_int8_t)); 4374aa92fe2STakanori Watanabe if(event == NULL){ 4384aa92fe2STakanori Watanabe NG_HCI_WARN("%s: Event datasize Pullup Failed\n", __func__); 4394aa92fe2STakanori Watanabe goto out; 4404aa92fe2STakanori Watanabe } 441fbc48c2bSTakanori Watanabe length_data = *mtod(event, u_int8_t *); 442fbc48c2bSTakanori Watanabe m_adj(event, sizeof(u_int8_t)); 4434aa92fe2STakanori Watanabe n->extinq_size = (length_data < NG_HCI_EXTINQ_MAX)? 4444aa92fe2STakanori Watanabe length_data : NG_HCI_EXTINQ_MAX; 4454aa92fe2STakanori Watanabe 446fbc48c2bSTakanori Watanabe /*Advertizement data*/ 4474aa92fe2STakanori Watanabe event = m_pullup(event, n->extinq_size); 4484aa92fe2STakanori Watanabe if(event == NULL){ 4494aa92fe2STakanori Watanabe NG_HCI_WARN("%s: Event data pullup Failed\n", __func__); 4504aa92fe2STakanori Watanabe goto out; 4514aa92fe2STakanori Watanabe } 4524aa92fe2STakanori Watanabe m_copydata(event, 0, n->extinq_size, n->extinq_data); 4534aa92fe2STakanori Watanabe m_adj(event, n->extinq_size); 4544aa92fe2STakanori Watanabe event = m_pullup(event, sizeof(char )); 455fbc48c2bSTakanori Watanabe /*Get RSSI*/ 4564aa92fe2STakanori Watanabe if(event == NULL){ 4574aa92fe2STakanori Watanabe NG_HCI_WARN("%s: Event rssi pull up Failed\n", __func__); 4584aa92fe2STakanori Watanabe 4594aa92fe2STakanori Watanabe goto out; 4604aa92fe2STakanori Watanabe } 4614aa92fe2STakanori Watanabe n->page_scan_mode = *mtod(event, char *); 462fbc48c2bSTakanori Watanabe m_adj(event, sizeof(u_int8_t)); 463fbc48c2bSTakanori Watanabe } 464fbc48c2bSTakanori Watanabe } 4654aa92fe2STakanori Watanabe out: 466fbc48c2bSTakanori Watanabe NG_FREE_M(event); 467fbc48c2bSTakanori Watanabe 468fbc48c2bSTakanori Watanabe return (error); 469fbc48c2bSTakanori Watanabe } /* inquiry_result */ 470fbc48c2bSTakanori Watanabe 471fbc48c2bSTakanori Watanabe static int le_connection_complete(ng_hci_unit_p unit, struct mbuf *event) 472fbc48c2bSTakanori Watanabe { 473fbc48c2bSTakanori Watanabe int error = 0; 474fbc48c2bSTakanori Watanabe 475fbc48c2bSTakanori Watanabe ng_hci_le_connection_complete_ep *ep = NULL; 476fbc48c2bSTakanori Watanabe ng_hci_unit_con_p con = NULL; 477fbc48c2bSTakanori Watanabe int link_type; 478fbc48c2bSTakanori Watanabe uint8_t uclass[3] = {0,0,0};//dummy uclass 479fbc48c2bSTakanori Watanabe 480fbc48c2bSTakanori Watanabe NG_HCI_M_PULLUP(event, sizeof(*ep)); 481fbc48c2bSTakanori Watanabe if (event == NULL) 482fbc48c2bSTakanori Watanabe return (ENOBUFS); 483fbc48c2bSTakanori Watanabe 484fbc48c2bSTakanori Watanabe ep = mtod(event, ng_hci_le_connection_complete_ep *); 485fbc48c2bSTakanori Watanabe link_type = (ep->address_type)? NG_HCI_LINK_LE_RANDOM : 486fbc48c2bSTakanori Watanabe NG_HCI_LINK_LE_PUBLIC; 487fbc48c2bSTakanori Watanabe /* 488fbc48c2bSTakanori Watanabe * Find the first connection descriptor that matches the following: 489fbc48c2bSTakanori Watanabe * 490fbc48c2bSTakanori Watanabe * 1) con->link_type == link_type 491fbc48c2bSTakanori Watanabe * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE 492fbc48c2bSTakanori Watanabe * 3) con->bdaddr == ep->address 493fbc48c2bSTakanori Watanabe */ 494fbc48c2bSTakanori Watanabe LIST_FOREACH(con, &unit->con_list, next) 495fbc48c2bSTakanori Watanabe if (con->link_type == link_type && 496fbc48c2bSTakanori Watanabe con->state == NG_HCI_CON_W4_CONN_COMPLETE && 497fbc48c2bSTakanori Watanabe bcmp(&con->bdaddr, &ep->address, sizeof(bdaddr_t)) == 0) 498fbc48c2bSTakanori Watanabe break; 499fbc48c2bSTakanori Watanabe 500fbc48c2bSTakanori Watanabe /* 501fbc48c2bSTakanori Watanabe * Two possible cases: 502fbc48c2bSTakanori Watanabe * 503fbc48c2bSTakanori Watanabe * 1) We have found connection descriptor. That means upper layer has 504fbc48c2bSTakanori Watanabe * requested this connection via LP_CON_REQ message. In this case 505fbc48c2bSTakanori Watanabe * connection must have timeout set. If ng_hci_con_untimeout() fails 506fbc48c2bSTakanori Watanabe * then timeout message already went into node's queue. In this case 507fbc48c2bSTakanori Watanabe * ignore Connection_Complete event and let timeout deal with it. 508fbc48c2bSTakanori Watanabe * 509fbc48c2bSTakanori Watanabe * 2) We do not have connection descriptor. That means upper layer 510fbc48c2bSTakanori Watanabe * nas not requested this connection , (less likely) we gave up 511fbc48c2bSTakanori Watanabe * on this connection (timeout) or as node act as slave role. 512fbc48c2bSTakanori Watanabe * The most likely scenario is that 513fbc48c2bSTakanori Watanabe * we have received LE_Create_Connection command 514fbc48c2bSTakanori Watanabe * from the RAW hook 515fbc48c2bSTakanori Watanabe */ 516fbc48c2bSTakanori Watanabe 517fbc48c2bSTakanori Watanabe if (con == NULL) { 518fbc48c2bSTakanori Watanabe if (ep->status != 0) 519fbc48c2bSTakanori Watanabe goto out; 520fbc48c2bSTakanori Watanabe 521fbc48c2bSTakanori Watanabe con = ng_hci_new_con(unit, link_type); 522fbc48c2bSTakanori Watanabe if (con == NULL) { 523fbc48c2bSTakanori Watanabe error = ENOMEM; 524fbc48c2bSTakanori Watanabe goto out; 525fbc48c2bSTakanori Watanabe } 526fbc48c2bSTakanori Watanabe 527fbc48c2bSTakanori Watanabe con->state = NG_HCI_CON_W4_LP_CON_RSP; 528fbc48c2bSTakanori Watanabe ng_hci_con_timeout(con); 529fbc48c2bSTakanori Watanabe 530fbc48c2bSTakanori Watanabe bcopy(&ep->address, &con->bdaddr, sizeof(con->bdaddr)); 531fbc48c2bSTakanori Watanabe error = ng_hci_lp_con_ind(con, uclass); 532fbc48c2bSTakanori Watanabe if (error != 0) { 533fbc48c2bSTakanori Watanabe ng_hci_con_untimeout(con); 534fbc48c2bSTakanori Watanabe ng_hci_free_con(con); 53514803ec8STakanori Watanabe goto out; 536fbc48c2bSTakanori Watanabe } 537fbc48c2bSTakanori Watanabe 538fbc48c2bSTakanori Watanabe } else if ((error = ng_hci_con_untimeout(con)) != 0) 539fbc48c2bSTakanori Watanabe goto out; 540fbc48c2bSTakanori Watanabe 541fbc48c2bSTakanori Watanabe /* 542fbc48c2bSTakanori Watanabe * Update connection descriptor and send notification 543fbc48c2bSTakanori Watanabe * to the upper layers. 544fbc48c2bSTakanori Watanabe */ 545fbc48c2bSTakanori Watanabe 546fbc48c2bSTakanori Watanabe con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->handle)); 547fbc48c2bSTakanori Watanabe con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE; 548fbc48c2bSTakanori Watanabe 549fbc48c2bSTakanori Watanabe ng_hci_lp_con_cfm(con, ep->status); 550fbc48c2bSTakanori Watanabe 551fbc48c2bSTakanori Watanabe /* Adjust connection state */ 552fbc48c2bSTakanori Watanabe if (ep->status != 0) 553fbc48c2bSTakanori Watanabe ng_hci_free_con(con); 554fbc48c2bSTakanori Watanabe else { 555fbc48c2bSTakanori Watanabe con->state = NG_HCI_CON_OPEN; 556fbc48c2bSTakanori Watanabe 557fbc48c2bSTakanori Watanabe /* 558fbc48c2bSTakanori Watanabe * Change link policy for the ACL connections. Enable all 559fbc48c2bSTakanori Watanabe * supported link modes. Enable Role switch as well if 560fbc48c2bSTakanori Watanabe * device supports it. 561fbc48c2bSTakanori Watanabe */ 562fbc48c2bSTakanori Watanabe } 563fbc48c2bSTakanori Watanabe 564fbc48c2bSTakanori Watanabe out: 565fbc48c2bSTakanori Watanabe NG_FREE_M(event); 566fbc48c2bSTakanori Watanabe 567fbc48c2bSTakanori Watanabe return (error); 568fbc48c2bSTakanori Watanabe 569fbc48c2bSTakanori Watanabe } 570fbc48c2bSTakanori Watanabe 571fbc48c2bSTakanori Watanabe static int le_connection_update(ng_hci_unit_p unit, struct mbuf *event) 572fbc48c2bSTakanori Watanabe { 573fbc48c2bSTakanori Watanabe int error = 0; 574fbc48c2bSTakanori Watanabe /*TBD*/ 575fbc48c2bSTakanori Watanabe 576fbc48c2bSTakanori Watanabe NG_FREE_M(event); 577fbc48c2bSTakanori Watanabe return error; 578fbc48c2bSTakanori Watanabe 579fbc48c2bSTakanori Watanabe } 580fbc48c2bSTakanori Watanabe static int 581fbc48c2bSTakanori Watanabe le_event(ng_hci_unit_p unit, struct mbuf *event) 582fbc48c2bSTakanori Watanabe { 583fbc48c2bSTakanori Watanabe int error = 0; 584fbc48c2bSTakanori Watanabe ng_hci_le_ep *lep; 585fbc48c2bSTakanori Watanabe 586fbc48c2bSTakanori Watanabe NG_HCI_M_PULLUP(event, sizeof(*lep)); 587fbc48c2bSTakanori Watanabe if(event ==NULL){ 588fbc48c2bSTakanori Watanabe return ENOBUFS; 589fbc48c2bSTakanori Watanabe } 590fbc48c2bSTakanori Watanabe lep = mtod(event, ng_hci_le_ep *); 591fbc48c2bSTakanori Watanabe m_adj(event, sizeof(*lep)); 592fbc48c2bSTakanori Watanabe switch(lep->subevent_code){ 593fbc48c2bSTakanori Watanabe case NG_HCI_LEEV_CON_COMPL: 594fbc48c2bSTakanori Watanabe le_connection_complete(unit, event); 595fbc48c2bSTakanori Watanabe break; 596fbc48c2bSTakanori Watanabe case NG_HCI_LEEV_ADVREP: 597fbc48c2bSTakanori Watanabe le_advertizing_report(unit, event); 598fbc48c2bSTakanori Watanabe break; 599fbc48c2bSTakanori Watanabe case NG_HCI_LEEV_CON_UPDATE_COMPL: 600fbc48c2bSTakanori Watanabe le_connection_update(unit, event); 601fbc48c2bSTakanori Watanabe break; 602fbc48c2bSTakanori Watanabe case NG_HCI_LEEV_READ_REMOTE_FEATURES_COMPL: 603fbc48c2bSTakanori Watanabe //TBD 604fbc48c2bSTakanori Watanabe /*FALLTHROUGH*/ 605fbc48c2bSTakanori Watanabe case NG_HCI_LEEV_LONG_TERM_KEY_REQUEST: 606fbc48c2bSTakanori Watanabe //TBD 607fbc48c2bSTakanori Watanabe /*FALLTHROUGH*/ 608fbc48c2bSTakanori Watanabe default: 609fbc48c2bSTakanori Watanabe NG_FREE_M(event); 610fbc48c2bSTakanori Watanabe } 611fbc48c2bSTakanori Watanabe return error; 612fbc48c2bSTakanori Watanabe } 613878ed226SJulian Elischer 614878ed226SJulian Elischer /* Inquiry result event */ 615878ed226SJulian Elischer static int 616878ed226SJulian Elischer inquiry_result(ng_hci_unit_p unit, struct mbuf *event) 617878ed226SJulian Elischer { 618878ed226SJulian Elischer ng_hci_inquiry_result_ep *ep = NULL; 619878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 620878ed226SJulian Elischer bdaddr_t bdaddr; 621878ed226SJulian Elischer int error = 0; 622878ed226SJulian Elischer 623878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 624878ed226SJulian Elischer if (event == NULL) 625878ed226SJulian Elischer return (ENOBUFS); 626878ed226SJulian Elischer 627878ed226SJulian Elischer ep = mtod(event, ng_hci_inquiry_result_ep *); 628878ed226SJulian Elischer m_adj(event, sizeof(*ep)); 629878ed226SJulian Elischer 630878ed226SJulian Elischer for (; ep->num_responses > 0; ep->num_responses --) { 631878ed226SJulian Elischer /* Get remote unit address */ 632878ed226SJulian Elischer m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr); 633878ed226SJulian Elischer m_adj(event, sizeof(bdaddr)); 634878ed226SJulian Elischer 635878ed226SJulian Elischer /* Lookup entry in the cache */ 636fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &bdaddr, NG_HCI_LINK_ACL); 637878ed226SJulian Elischer if (n == NULL) { 638878ed226SJulian Elischer /* Create new entry */ 639878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 640878ed226SJulian Elischer if (n == NULL) { 641878ed226SJulian Elischer error = ENOMEM; 642878ed226SJulian Elischer break; 643878ed226SJulian Elischer } 644878ed226SJulian Elischer } else 645878ed226SJulian Elischer getmicrotime(&n->updated); 646878ed226SJulian Elischer 647878ed226SJulian Elischer bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 648fbc48c2bSTakanori Watanabe n->addrtype = NG_HCI_LINK_ACL; 649878ed226SJulian Elischer 650878ed226SJulian Elischer /* XXX call m_pullup here? */ 651878ed226SJulian Elischer 652878ed226SJulian Elischer n->page_scan_rep_mode = *mtod(event, u_int8_t *); 653878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 654878ed226SJulian Elischer 655878ed226SJulian Elischer /* page_scan_period_mode */ 656878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 657878ed226SJulian Elischer 658878ed226SJulian Elischer n->page_scan_mode = *mtod(event, u_int8_t *); 659878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 660878ed226SJulian Elischer 661878ed226SJulian Elischer /* class */ 662878ed226SJulian Elischer m_adj(event, NG_HCI_CLASS_SIZE); 663878ed226SJulian Elischer 664878ed226SJulian Elischer /* clock offset */ 665878ed226SJulian Elischer m_copydata(event, 0, sizeof(n->clock_offset), 666878ed226SJulian Elischer (caddr_t) &n->clock_offset); 667878ed226SJulian Elischer n->clock_offset = le16toh(n->clock_offset); 668878ed226SJulian Elischer } 669878ed226SJulian Elischer 670878ed226SJulian Elischer NG_FREE_M(event); 671878ed226SJulian Elischer 672878ed226SJulian Elischer return (error); 673878ed226SJulian Elischer } /* inquiry_result */ 674878ed226SJulian Elischer 675878ed226SJulian Elischer /* Connection complete event */ 676878ed226SJulian Elischer static int 677878ed226SJulian Elischer con_compl(ng_hci_unit_p unit, struct mbuf *event) 678878ed226SJulian Elischer { 679878ed226SJulian Elischer ng_hci_con_compl_ep *ep = NULL; 680878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 681878ed226SJulian Elischer int error = 0; 682878ed226SJulian Elischer 683878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 684878ed226SJulian Elischer if (event == NULL) 685878ed226SJulian Elischer return (ENOBUFS); 686878ed226SJulian Elischer 687878ed226SJulian Elischer ep = mtod(event, ng_hci_con_compl_ep *); 688878ed226SJulian Elischer 689878ed226SJulian Elischer /* 690878ed226SJulian Elischer * Find the first connection descriptor that matches the following: 691878ed226SJulian Elischer * 692878ed226SJulian Elischer * 1) con->link_type == ep->link_type 693878ed226SJulian Elischer * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE 694878ed226SJulian Elischer * 3) con->bdaddr == ep->bdaddr 695878ed226SJulian Elischer */ 696878ed226SJulian Elischer 697878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 698878ed226SJulian Elischer if (con->link_type == ep->link_type && 699878ed226SJulian Elischer con->state == NG_HCI_CON_W4_CONN_COMPLETE && 700878ed226SJulian Elischer bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0) 701878ed226SJulian Elischer break; 702878ed226SJulian Elischer 703878ed226SJulian Elischer /* 704878ed226SJulian Elischer * Two possible cases: 705878ed226SJulian Elischer * 706878ed226SJulian Elischer * 1) We have found connection descriptor. That means upper layer has 7070986ab12SMaksim Yevmenkin * requested this connection via LP_CON_REQ message. In this case 7080986ab12SMaksim Yevmenkin * connection must have timeout set. If ng_hci_con_untimeout() fails 7090986ab12SMaksim Yevmenkin * then timeout message already went into node's queue. In this case 7100986ab12SMaksim Yevmenkin * ignore Connection_Complete event and let timeout deal with it. 711878ed226SJulian Elischer * 712878ed226SJulian Elischer * 2) We do not have connection descriptor. That means upper layer 713878ed226SJulian Elischer * nas not requested this connection or (less likely) we gave up 714878ed226SJulian Elischer * on this connection (timeout). The most likely scenario is that 715878ed226SJulian Elischer * we have received Create_Connection/Add_SCO_Connection command 716878ed226SJulian Elischer * from the RAW hook 717878ed226SJulian Elischer */ 718878ed226SJulian Elischer 719878ed226SJulian Elischer if (con == NULL) { 720878ed226SJulian Elischer if (ep->status != 0) 721878ed226SJulian Elischer goto out; 722878ed226SJulian Elischer 723878ed226SJulian Elischer con = ng_hci_new_con(unit, ep->link_type); 724878ed226SJulian Elischer if (con == NULL) { 725878ed226SJulian Elischer error = ENOMEM; 726878ed226SJulian Elischer goto out; 727878ed226SJulian Elischer } 728878ed226SJulian Elischer 729878ed226SJulian Elischer bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr)); 7300986ab12SMaksim Yevmenkin } else if ((error = ng_hci_con_untimeout(con)) != 0) 7310986ab12SMaksim Yevmenkin goto out; 732878ed226SJulian Elischer 733878ed226SJulian Elischer /* 734878ed226SJulian Elischer * Update connection descriptor and send notification 735878ed226SJulian Elischer * to the upper layers. 736878ed226SJulian Elischer */ 737878ed226SJulian Elischer 738878ed226SJulian Elischer con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 739878ed226SJulian Elischer con->encryption_mode = ep->encryption_mode; 740878ed226SJulian Elischer 741878ed226SJulian Elischer ng_hci_lp_con_cfm(con, ep->status); 742878ed226SJulian Elischer 743878ed226SJulian Elischer /* Adjust connection state */ 744878ed226SJulian Elischer if (ep->status != 0) 745878ed226SJulian Elischer ng_hci_free_con(con); 746878ed226SJulian Elischer else { 747878ed226SJulian Elischer con->state = NG_HCI_CON_OPEN; 748878ed226SJulian Elischer 749878ed226SJulian Elischer /* 750878ed226SJulian Elischer * Change link policy for the ACL connections. Enable all 751878ed226SJulian Elischer * supported link modes. Enable Role switch as well if 752878ed226SJulian Elischer * device supports it. 753878ed226SJulian Elischer */ 754878ed226SJulian Elischer 755878ed226SJulian Elischer if (ep->link_type == NG_HCI_LINK_ACL) { 756878ed226SJulian Elischer struct __link_policy { 757878ed226SJulian Elischer ng_hci_cmd_pkt_t hdr; 758878ed226SJulian Elischer ng_hci_write_link_policy_settings_cp cp; 759878ed226SJulian Elischer } __attribute__ ((packed)) *lp; 760878ed226SJulian Elischer struct mbuf *m; 761878ed226SJulian Elischer 762eb1b1807SGleb Smirnoff MGETHDR(m, M_NOWAIT, MT_DATA); 763878ed226SJulian Elischer if (m != NULL) { 764878ed226SJulian Elischer m->m_pkthdr.len = m->m_len = sizeof(*lp); 765878ed226SJulian Elischer lp = mtod(m, struct __link_policy *); 766878ed226SJulian Elischer 767878ed226SJulian Elischer lp->hdr.type = NG_HCI_CMD_PKT; 768878ed226SJulian Elischer lp->hdr.opcode = htole16(NG_HCI_OPCODE( 769878ed226SJulian Elischer NG_HCI_OGF_LINK_POLICY, 770878ed226SJulian Elischer NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS)); 771878ed226SJulian Elischer lp->hdr.length = sizeof(lp->cp); 772878ed226SJulian Elischer 773878ed226SJulian Elischer lp->cp.con_handle = ep->con_handle; 774878ed226SJulian Elischer 775878ed226SJulian Elischer lp->cp.settings = 0; 776f2bb1caeSJulian Elischer if ((unit->features[0] & NG_HCI_LMP_SWITCH) && 777f2bb1caeSJulian Elischer unit->role_switch) 778878ed226SJulian Elischer lp->cp.settings |= 0x1; 779878ed226SJulian Elischer if (unit->features[0] & NG_HCI_LMP_HOLD_MODE) 780878ed226SJulian Elischer lp->cp.settings |= 0x2; 781878ed226SJulian Elischer if (unit->features[0] & NG_HCI_LMP_SNIFF_MODE) 782878ed226SJulian Elischer lp->cp.settings |= 0x4; 783878ed226SJulian Elischer if (unit->features[1] & NG_HCI_LMP_PARK_MODE) 784878ed226SJulian Elischer lp->cp.settings |= 0x8; 785878ed226SJulian Elischer 786878ed226SJulian Elischer lp->cp.settings &= unit->link_policy_mask; 787878ed226SJulian Elischer lp->cp.settings = htole16(lp->cp.settings); 788878ed226SJulian Elischer 789878ed226SJulian Elischer NG_BT_MBUFQ_ENQUEUE(&unit->cmdq, m); 790878ed226SJulian Elischer if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) 791878ed226SJulian Elischer ng_hci_send_command(unit); 792878ed226SJulian Elischer } 793878ed226SJulian Elischer } 794878ed226SJulian Elischer } 795878ed226SJulian Elischer out: 796878ed226SJulian Elischer NG_FREE_M(event); 797878ed226SJulian Elischer 798878ed226SJulian Elischer return (error); 7990986ab12SMaksim Yevmenkin } /* con_compl */ 800878ed226SJulian Elischer 801878ed226SJulian Elischer /* Connection request event */ 802878ed226SJulian Elischer static int 803878ed226SJulian Elischer con_req(ng_hci_unit_p unit, struct mbuf *event) 804878ed226SJulian Elischer { 805878ed226SJulian Elischer ng_hci_con_req_ep *ep = NULL; 806878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 807878ed226SJulian Elischer int error = 0; 808878ed226SJulian Elischer 809878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 810878ed226SJulian Elischer if (event == NULL) 811878ed226SJulian Elischer return (ENOBUFS); 812878ed226SJulian Elischer 813878ed226SJulian Elischer ep = mtod(event, ng_hci_con_req_ep *); 814878ed226SJulian Elischer 815878ed226SJulian Elischer /* 816878ed226SJulian Elischer * Find the first connection descriptor that matches the following: 817878ed226SJulian Elischer * 818878ed226SJulian Elischer * 1) con->link_type == ep->link_type 819878ed226SJulian Elischer * 820878ed226SJulian Elischer * 2) con->state == NG_HCI_CON_W4_LP_CON_RSP || 821878ed226SJulian Elischer * con->state == NG_HCI_CON_W4_CONN_COMPL 822878ed226SJulian Elischer * 823878ed226SJulian Elischer * 3) con->bdaddr == ep->bdaddr 824878ed226SJulian Elischer * 825878ed226SJulian Elischer * Possible cases: 826878ed226SJulian Elischer * 827878ed226SJulian Elischer * 1) We do not have connection descriptor. This is simple. Create 828878ed226SJulian Elischer * new fresh connection descriptor and send notification to the 829878ed226SJulian Elischer * appropriate upstream hook (based on link_type). 830878ed226SJulian Elischer * 831878ed226SJulian Elischer * 2) We found connection handle. This is more complicated. 832878ed226SJulian Elischer * 833878ed226SJulian Elischer * 2.1) ACL links 834878ed226SJulian Elischer * 835878ed226SJulian Elischer * Since only one ACL link can exist between each pair of 836878ed226SJulian Elischer * units then we have a race. Our upper layer has requested 837878ed226SJulian Elischer * an ACL connection to the remote unit, but we did not send 838878ed226SJulian Elischer * command yet. At the same time the remote unit has requested 839878ed226SJulian Elischer * an ACL connection from us. In this case we will ignore 840878ed226SJulian Elischer * Connection_Request event. This probably will cause connect 841878ed226SJulian Elischer * failure on both units. 842878ed226SJulian Elischer * 843878ed226SJulian Elischer * 2.2) SCO links 844878ed226SJulian Elischer * 845878ed226SJulian Elischer * The spec on page 45 says : 846878ed226SJulian Elischer * 847878ed226SJulian Elischer * "The master can support up to three SCO links to the same 848878ed226SJulian Elischer * slave or to different slaves. A slave can support up to 849878ed226SJulian Elischer * three SCO links from the same master, or two SCO links if 850878ed226SJulian Elischer * the links originate from different masters." 851878ed226SJulian Elischer * 852878ed226SJulian Elischer * The only problem is how to handle multiple SCO links between 853878ed226SJulian Elischer * matster and slave. For now we will assume that multiple SCO 854878ed226SJulian Elischer * links MUST be opened one after another. 855878ed226SJulian Elischer */ 856878ed226SJulian Elischer 857878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 858878ed226SJulian Elischer if (con->link_type == ep->link_type && 859878ed226SJulian Elischer (con->state == NG_HCI_CON_W4_LP_CON_RSP || 860878ed226SJulian Elischer con->state == NG_HCI_CON_W4_CONN_COMPLETE) && 861878ed226SJulian Elischer bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0) 862878ed226SJulian Elischer break; 863878ed226SJulian Elischer 864878ed226SJulian Elischer if (con == NULL) { 865878ed226SJulian Elischer con = ng_hci_new_con(unit, ep->link_type); 866878ed226SJulian Elischer if (con != NULL) { 867878ed226SJulian Elischer bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr)); 868878ed226SJulian Elischer 869878ed226SJulian Elischer con->state = NG_HCI_CON_W4_LP_CON_RSP; 870878ed226SJulian Elischer ng_hci_con_timeout(con); 871878ed226SJulian Elischer 872878ed226SJulian Elischer error = ng_hci_lp_con_ind(con, ep->uclass); 8730986ab12SMaksim Yevmenkin if (error != 0) { 8740986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 875878ed226SJulian Elischer ng_hci_free_con(con); 8760986ab12SMaksim Yevmenkin } 877878ed226SJulian Elischer } else 878878ed226SJulian Elischer error = ENOMEM; 879878ed226SJulian Elischer } 880878ed226SJulian Elischer 881878ed226SJulian Elischer NG_FREE_M(event); 882878ed226SJulian Elischer 883878ed226SJulian Elischer return (error); 884878ed226SJulian Elischer } /* con_req */ 885878ed226SJulian Elischer 886878ed226SJulian Elischer /* Disconnect complete event */ 887878ed226SJulian Elischer static int 888878ed226SJulian Elischer discon_compl(ng_hci_unit_p unit, struct mbuf *event) 889878ed226SJulian Elischer { 890878ed226SJulian Elischer ng_hci_discon_compl_ep *ep = NULL; 891878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 892878ed226SJulian Elischer int error = 0; 893878ed226SJulian Elischer u_int16_t h; 894878ed226SJulian Elischer 895878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 896878ed226SJulian Elischer if (event == NULL) 897878ed226SJulian Elischer return (ENOBUFS); 898878ed226SJulian Elischer 899878ed226SJulian Elischer ep = mtod(event, ng_hci_discon_compl_ep *); 900878ed226SJulian Elischer 901878ed226SJulian Elischer /* 902878ed226SJulian Elischer * XXX 903878ed226SJulian Elischer * Do we have to send notification if ep->status != 0? 904878ed226SJulian Elischer * For now we will send notification for both ACL and SCO connections 905878ed226SJulian Elischer * ONLY if ep->status == 0. 906878ed226SJulian Elischer */ 907878ed226SJulian Elischer 908878ed226SJulian Elischer if (ep->status == 0) { 909878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 910878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 911878ed226SJulian Elischer if (con != NULL) { 912878ed226SJulian Elischer error = ng_hci_lp_discon_ind(con, ep->reason); 9130986ab12SMaksim Yevmenkin 9140986ab12SMaksim Yevmenkin /* Remove all timeouts (if any) */ 9150986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 9160986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 9170986ab12SMaksim Yevmenkin 918878ed226SJulian Elischer ng_hci_free_con(con); 919878ed226SJulian Elischer } else { 920878ed226SJulian Elischer NG_HCI_ALERT( 921878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 922878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 923878ed226SJulian Elischer error = ENOENT; 924878ed226SJulian Elischer } 925878ed226SJulian Elischer } 926878ed226SJulian Elischer 927878ed226SJulian Elischer NG_FREE_M(event); 928878ed226SJulian Elischer 929878ed226SJulian Elischer return (error); 930878ed226SJulian Elischer } /* discon_compl */ 931878ed226SJulian Elischer 932f2bb1caeSJulian Elischer /* Encryption change event */ 933f2bb1caeSJulian Elischer static int 934f2bb1caeSJulian Elischer encryption_change(ng_hci_unit_p unit, struct mbuf *event) 935f2bb1caeSJulian Elischer { 936f2bb1caeSJulian Elischer ng_hci_encryption_change_ep *ep = NULL; 937f2bb1caeSJulian Elischer ng_hci_unit_con_p con = NULL; 938f2bb1caeSJulian Elischer int error = 0; 939483ed395STakanori Watanabe u_int16_t h; 940f2bb1caeSJulian Elischer 941f2bb1caeSJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 942f2bb1caeSJulian Elischer if (event == NULL) 943f2bb1caeSJulian Elischer return (ENOBUFS); 944f2bb1caeSJulian Elischer 945f2bb1caeSJulian Elischer ep = mtod(event, ng_hci_encryption_change_ep *); 946483ed395STakanori Watanabe h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 947483ed395STakanori Watanabe con = ng_hci_con_by_handle(unit, h); 948f2bb1caeSJulian Elischer 949f2bb1caeSJulian Elischer if (ep->status == 0) { 950f2bb1caeSJulian Elischer if (con == NULL) { 951f2bb1caeSJulian Elischer NG_HCI_ALERT( 952f2bb1caeSJulian Elischer "%s: %s - invalid connection handle=%d\n", 953f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 954f2bb1caeSJulian Elischer error = ENOENT; 9553a601a23STakanori Watanabe } else if (con->link_type == NG_HCI_LINK_SCO) { 956f2bb1caeSJulian Elischer NG_HCI_ALERT( 957f2bb1caeSJulian Elischer "%s: %s - invalid link type=%d\n", 958f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), 959f2bb1caeSJulian Elischer con->link_type); 960f2bb1caeSJulian Elischer error = EINVAL; 961f2bb1caeSJulian Elischer } else if (ep->encryption_enable) 962f2bb1caeSJulian Elischer /* XXX is that true? */ 963f2bb1caeSJulian Elischer con->encryption_mode = NG_HCI_ENCRYPTION_MODE_P2P; 964f2bb1caeSJulian Elischer else 965f2bb1caeSJulian Elischer con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE; 966f2bb1caeSJulian Elischer } else 967f2bb1caeSJulian Elischer NG_HCI_ERR( 968f2bb1caeSJulian Elischer "%s: %s - failed to change encryption mode, status=%d\n", 969f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 970f2bb1caeSJulian Elischer 971483ed395STakanori Watanabe /*Anyway, propagete encryption status to upper layer*/ 972483ed395STakanori Watanabe ng_hci_lp_enc_change(con, con->encryption_mode); 973483ed395STakanori Watanabe 974f2bb1caeSJulian Elischer NG_FREE_M(event); 975f2bb1caeSJulian Elischer 976f2bb1caeSJulian Elischer return (error); 977f2bb1caeSJulian Elischer } /* encryption_change */ 978f2bb1caeSJulian Elischer 979878ed226SJulian Elischer /* Read remote feature complete event */ 980878ed226SJulian Elischer static int 981878ed226SJulian Elischer read_remote_features_compl(ng_hci_unit_p unit, struct mbuf *event) 982878ed226SJulian Elischer { 983878ed226SJulian Elischer ng_hci_read_remote_features_compl_ep *ep = NULL; 984878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 985878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 986878ed226SJulian Elischer u_int16_t h; 987878ed226SJulian Elischer int error = 0; 988878ed226SJulian Elischer 989878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 990878ed226SJulian Elischer if (event == NULL) 991878ed226SJulian Elischer return (ENOBUFS); 992878ed226SJulian Elischer 993878ed226SJulian Elischer ep = mtod(event, ng_hci_read_remote_features_compl_ep *); 994878ed226SJulian Elischer 995878ed226SJulian Elischer if (ep->status == 0) { 996878ed226SJulian Elischer /* Check if we have this connection handle */ 997878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 998878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 999878ed226SJulian Elischer if (con == NULL) { 1000878ed226SJulian Elischer NG_HCI_ALERT( 1001878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1002878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1003878ed226SJulian Elischer error = ENOENT; 1004878ed226SJulian Elischer goto out; 1005878ed226SJulian Elischer } 1006878ed226SJulian Elischer 1007878ed226SJulian Elischer /* Update cache entry */ 1008fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL); 1009878ed226SJulian Elischer if (n == NULL) { 1010878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1011878ed226SJulian Elischer if (n == NULL) { 1012878ed226SJulian Elischer error = ENOMEM; 1013878ed226SJulian Elischer goto out; 1014878ed226SJulian Elischer } 1015878ed226SJulian Elischer 1016878ed226SJulian Elischer bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1017fbc48c2bSTakanori Watanabe n->addrtype = NG_HCI_LINK_ACL; 1018878ed226SJulian Elischer } else 1019878ed226SJulian Elischer getmicrotime(&n->updated); 1020878ed226SJulian Elischer 1021878ed226SJulian Elischer bcopy(ep->features, n->features, sizeof(n->features)); 1022878ed226SJulian Elischer } else 1023878ed226SJulian Elischer NG_HCI_ERR( 1024878ed226SJulian Elischer "%s: %s - failed to read remote unit features, status=%d\n", 1025878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 1026878ed226SJulian Elischer out: 1027878ed226SJulian Elischer NG_FREE_M(event); 1028878ed226SJulian Elischer 1029878ed226SJulian Elischer return (error); 1030878ed226SJulian Elischer } /* read_remote_features_compl */ 1031878ed226SJulian Elischer 1032878ed226SJulian Elischer /* QoS setup complete event */ 1033878ed226SJulian Elischer static int 1034878ed226SJulian Elischer qos_setup_compl(ng_hci_unit_p unit, struct mbuf *event) 1035878ed226SJulian Elischer { 1036878ed226SJulian Elischer ng_hci_qos_setup_compl_ep *ep = NULL; 1037878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1038878ed226SJulian Elischer u_int16_t h; 1039878ed226SJulian Elischer int error = 0; 1040878ed226SJulian Elischer 1041878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1042878ed226SJulian Elischer if (event == NULL) 1043878ed226SJulian Elischer return (ENOBUFS); 1044878ed226SJulian Elischer 1045878ed226SJulian Elischer ep = mtod(event, ng_hci_qos_setup_compl_ep *); 1046878ed226SJulian Elischer 1047878ed226SJulian Elischer /* Check if we have this connection handle */ 1048878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1049878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1050878ed226SJulian Elischer if (con == NULL) { 1051878ed226SJulian Elischer NG_HCI_ALERT( 1052878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1053878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1054878ed226SJulian Elischer error = ENOENT; 1055878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 1056878ed226SJulian Elischer NG_HCI_ALERT( 1057878ed226SJulian Elischer "%s: %s - invalid link type=%d, handle=%d\n", 1058878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->link_type, h); 1059878ed226SJulian Elischer error = EINVAL; 1060878ed226SJulian Elischer } else if (con->state != NG_HCI_CON_OPEN) { 1061878ed226SJulian Elischer NG_HCI_ALERT( 1062878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n", 1063878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 1064878ed226SJulian Elischer con->state, h); 1065878ed226SJulian Elischer error = EINVAL; 1066878ed226SJulian Elischer } else /* Notify upper layer */ 1067878ed226SJulian Elischer error = ng_hci_lp_qos_cfm(con, ep->status); 1068878ed226SJulian Elischer 1069878ed226SJulian Elischer NG_FREE_M(event); 1070878ed226SJulian Elischer 1071878ed226SJulian Elischer return (error); 1072878ed226SJulian Elischer } /* qos_setup_compl */ 1073878ed226SJulian Elischer 1074878ed226SJulian Elischer /* Hardware error event */ 1075878ed226SJulian Elischer static int 1076878ed226SJulian Elischer hardware_error(ng_hci_unit_p unit, struct mbuf *event) 1077878ed226SJulian Elischer { 1078878ed226SJulian Elischer NG_HCI_ALERT( 1079878ed226SJulian Elischer "%s: %s - hardware error %#x\n", 1080878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), *mtod(event, u_int8_t *)); 1081878ed226SJulian Elischer 1082878ed226SJulian Elischer NG_FREE_M(event); 1083878ed226SJulian Elischer 1084878ed226SJulian Elischer return (0); 1085878ed226SJulian Elischer } /* hardware_error */ 1086878ed226SJulian Elischer 1087878ed226SJulian Elischer /* Role change event */ 1088878ed226SJulian Elischer static int 1089878ed226SJulian Elischer role_change(ng_hci_unit_p unit, struct mbuf *event) 1090878ed226SJulian Elischer { 1091878ed226SJulian Elischer ng_hci_role_change_ep *ep = NULL; 1092878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1093878ed226SJulian Elischer 1094878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1095878ed226SJulian Elischer if (event == NULL) 1096878ed226SJulian Elischer return (ENOBUFS); 1097878ed226SJulian Elischer 1098878ed226SJulian Elischer ep = mtod(event, ng_hci_role_change_ep *); 1099878ed226SJulian Elischer 1100878ed226SJulian Elischer if (ep->status == 0) { 1101878ed226SJulian Elischer /* XXX shoud we also change "role" for SCO connections? */ 1102878ed226SJulian Elischer con = ng_hci_con_by_bdaddr(unit, &ep->bdaddr, NG_HCI_LINK_ACL); 1103878ed226SJulian Elischer if (con != NULL) 1104878ed226SJulian Elischer con->role = ep->role; 1105878ed226SJulian Elischer else 1106878ed226SJulian Elischer NG_HCI_ALERT( 1107878ed226SJulian Elischer "%s: %s - ACL connection does not exist, bdaddr=%x:%x:%x:%x:%x:%x\n", 1108878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 1109878ed226SJulian Elischer ep->bdaddr.b[5], ep->bdaddr.b[4], 1110878ed226SJulian Elischer ep->bdaddr.b[3], ep->bdaddr.b[2], 1111878ed226SJulian Elischer ep->bdaddr.b[1], ep->bdaddr.b[0]); 1112878ed226SJulian Elischer } else 1113878ed226SJulian Elischer NG_HCI_ERR( 1114878ed226SJulian Elischer "%s: %s - failed to change role, status=%d, bdaddr=%x:%x:%x:%x:%x:%x\n", 1115878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status, 1116878ed226SJulian Elischer ep->bdaddr.b[5], ep->bdaddr.b[4], ep->bdaddr.b[3], 1117878ed226SJulian Elischer ep->bdaddr.b[2], ep->bdaddr.b[1], ep->bdaddr.b[0]); 1118878ed226SJulian Elischer 1119878ed226SJulian Elischer NG_FREE_M(event); 1120878ed226SJulian Elischer 1121878ed226SJulian Elischer return (0); 1122878ed226SJulian Elischer } /* role_change */ 1123878ed226SJulian Elischer 1124878ed226SJulian Elischer /* Number of completed packets event */ 1125878ed226SJulian Elischer static int 1126878ed226SJulian Elischer num_compl_pkts(ng_hci_unit_p unit, struct mbuf *event) 1127878ed226SJulian Elischer { 1128878ed226SJulian Elischer ng_hci_num_compl_pkts_ep *ep = NULL; 1129878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1130878ed226SJulian Elischer u_int16_t h, p; 1131878ed226SJulian Elischer 1132878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1133878ed226SJulian Elischer if (event == NULL) 1134878ed226SJulian Elischer return (ENOBUFS); 1135878ed226SJulian Elischer 1136878ed226SJulian Elischer ep = mtod(event, ng_hci_num_compl_pkts_ep *); 1137878ed226SJulian Elischer m_adj(event, sizeof(*ep)); 1138878ed226SJulian Elischer 1139878ed226SJulian Elischer for (; ep->num_con_handles > 0; ep->num_con_handles --) { 1140878ed226SJulian Elischer /* Get connection handle */ 1141878ed226SJulian Elischer m_copydata(event, 0, sizeof(h), (caddr_t) &h); 1142878ed226SJulian Elischer m_adj(event, sizeof(h)); 1143878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(h)); 1144878ed226SJulian Elischer 1145878ed226SJulian Elischer /* Get number of completed packets */ 1146878ed226SJulian Elischer m_copydata(event, 0, sizeof(p), (caddr_t) &p); 1147878ed226SJulian Elischer m_adj(event, sizeof(p)); 1148878ed226SJulian Elischer p = le16toh(p); 1149878ed226SJulian Elischer 1150878ed226SJulian Elischer /* Check if we have this connection handle */ 1151878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1152878ed226SJulian Elischer if (con != NULL) { 1153878ed226SJulian Elischer con->pending -= p; 1154878ed226SJulian Elischer if (con->pending < 0) { 1155878ed226SJulian Elischer NG_HCI_WARN( 1156878ed226SJulian Elischer "%s: %s - pending packet counter is out of sync! " \ 1157878ed226SJulian Elischer "handle=%d, pending=%d, ncp=%d\n", __func__, NG_NODE_NAME(unit->node), 1158878ed226SJulian Elischer con->con_handle, con->pending, p); 1159878ed226SJulian Elischer 1160878ed226SJulian Elischer con->pending = 0; 1161878ed226SJulian Elischer } 1162878ed226SJulian Elischer 1163878ed226SJulian Elischer /* Update buffer descriptor */ 1164fbc48c2bSTakanori Watanabe if (con->link_type != NG_HCI_LINK_SCO) 1165878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(unit->buffer, p); 1166878ed226SJulian Elischer else 1167878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(unit->buffer, p); 1168878ed226SJulian Elischer } else 1169878ed226SJulian Elischer NG_HCI_ALERT( 1170878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1171878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1172878ed226SJulian Elischer } 1173878ed226SJulian Elischer 1174878ed226SJulian Elischer NG_FREE_M(event); 1175878ed226SJulian Elischer 1176878ed226SJulian Elischer /* Send more data */ 1177878ed226SJulian Elischer ng_hci_send_data(unit); 1178878ed226SJulian Elischer 1179878ed226SJulian Elischer return (0); 1180878ed226SJulian Elischer } /* num_compl_pkts */ 1181878ed226SJulian Elischer 1182878ed226SJulian Elischer /* Mode change event */ 1183878ed226SJulian Elischer static int 1184878ed226SJulian Elischer mode_change(ng_hci_unit_p unit, struct mbuf *event) 1185878ed226SJulian Elischer { 1186878ed226SJulian Elischer ng_hci_mode_change_ep *ep = NULL; 1187878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1188878ed226SJulian Elischer int error = 0; 1189878ed226SJulian Elischer 1190878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1191878ed226SJulian Elischer if (event == NULL) 1192878ed226SJulian Elischer return (ENOBUFS); 1193878ed226SJulian Elischer 1194878ed226SJulian Elischer ep = mtod(event, ng_hci_mode_change_ep *); 1195878ed226SJulian Elischer 1196878ed226SJulian Elischer if (ep->status == 0) { 1197878ed226SJulian Elischer u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1198878ed226SJulian Elischer 1199878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1200878ed226SJulian Elischer if (con == NULL) { 1201878ed226SJulian Elischer NG_HCI_ALERT( 1202878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1203878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1204878ed226SJulian Elischer error = ENOENT; 1205878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 1206878ed226SJulian Elischer NG_HCI_ALERT( 1207878ed226SJulian Elischer "%s: %s - invalid link type=%d\n", 1208878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 1209878ed226SJulian Elischer con->link_type); 1210878ed226SJulian Elischer error = EINVAL; 1211878ed226SJulian Elischer } else 1212878ed226SJulian Elischer con->mode = ep->unit_mode; 1213878ed226SJulian Elischer } else 1214878ed226SJulian Elischer NG_HCI_ERR( 1215878ed226SJulian Elischer "%s: %s - failed to change mode, status=%d\n", 1216878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 1217878ed226SJulian Elischer 1218878ed226SJulian Elischer NG_FREE_M(event); 1219878ed226SJulian Elischer 1220878ed226SJulian Elischer return (error); 1221878ed226SJulian Elischer } /* mode_change */ 1222878ed226SJulian Elischer 1223878ed226SJulian Elischer /* Data buffer overflow event */ 1224878ed226SJulian Elischer static int 1225878ed226SJulian Elischer data_buffer_overflow(ng_hci_unit_p unit, struct mbuf *event) 1226878ed226SJulian Elischer { 1227878ed226SJulian Elischer NG_HCI_ALERT( 1228878ed226SJulian Elischer "%s: %s - %s data buffer overflow\n", 1229878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 1230878ed226SJulian Elischer (*mtod(event, u_int8_t *) == NG_HCI_LINK_ACL)? "ACL" : "SCO"); 1231878ed226SJulian Elischer 1232878ed226SJulian Elischer NG_FREE_M(event); 1233878ed226SJulian Elischer 1234878ed226SJulian Elischer return (0); 1235878ed226SJulian Elischer } /* data_buffer_overflow */ 1236878ed226SJulian Elischer 1237878ed226SJulian Elischer /* Read clock offset complete event */ 1238878ed226SJulian Elischer static int 1239878ed226SJulian Elischer read_clock_offset_compl(ng_hci_unit_p unit, struct mbuf *event) 1240878ed226SJulian Elischer { 1241878ed226SJulian Elischer ng_hci_read_clock_offset_compl_ep *ep = NULL; 1242878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1243878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 1244878ed226SJulian Elischer int error = 0; 1245878ed226SJulian Elischer 1246878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1247878ed226SJulian Elischer if (event == NULL) 1248878ed226SJulian Elischer return (ENOBUFS); 1249878ed226SJulian Elischer 1250878ed226SJulian Elischer ep = mtod(event, ng_hci_read_clock_offset_compl_ep *); 1251878ed226SJulian Elischer 1252878ed226SJulian Elischer if (ep->status == 0) { 1253878ed226SJulian Elischer u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1254878ed226SJulian Elischer 1255878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1256878ed226SJulian Elischer if (con == NULL) { 1257878ed226SJulian Elischer NG_HCI_ALERT( 1258878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1259878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1260878ed226SJulian Elischer error = ENOENT; 1261878ed226SJulian Elischer goto out; 1262878ed226SJulian Elischer } 1263878ed226SJulian Elischer 1264878ed226SJulian Elischer /* Update cache entry */ 1265fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &con->bdaddr, NG_HCI_LINK_ACL); 1266878ed226SJulian Elischer if (n == NULL) { 1267878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1268878ed226SJulian Elischer if (n == NULL) { 1269878ed226SJulian Elischer error = ENOMEM; 1270878ed226SJulian Elischer goto out; 1271878ed226SJulian Elischer } 1272878ed226SJulian Elischer 1273878ed226SJulian Elischer bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1274fbc48c2bSTakanori Watanabe n->addrtype = NG_HCI_LINK_ACL; 1275878ed226SJulian Elischer } else 1276878ed226SJulian Elischer getmicrotime(&n->updated); 1277878ed226SJulian Elischer 1278878ed226SJulian Elischer n->clock_offset = le16toh(ep->clock_offset); 1279878ed226SJulian Elischer } else 1280878ed226SJulian Elischer NG_HCI_ERR( 1281878ed226SJulian Elischer "%s: %s - failed to Read Remote Clock Offset, status=%d\n", 1282878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 1283878ed226SJulian Elischer out: 1284878ed226SJulian Elischer NG_FREE_M(event); 1285878ed226SJulian Elischer 1286878ed226SJulian Elischer return (error); 1287878ed226SJulian Elischer } /* read_clock_offset_compl */ 1288878ed226SJulian Elischer 1289878ed226SJulian Elischer /* QoS violation event */ 1290878ed226SJulian Elischer static int 1291878ed226SJulian Elischer qos_violation(ng_hci_unit_p unit, struct mbuf *event) 1292878ed226SJulian Elischer { 1293878ed226SJulian Elischer ng_hci_qos_violation_ep *ep = NULL; 1294878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1295878ed226SJulian Elischer u_int16_t h; 1296878ed226SJulian Elischer int error = 0; 1297878ed226SJulian Elischer 1298878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1299878ed226SJulian Elischer if (event == NULL) 1300878ed226SJulian Elischer return (ENOBUFS); 1301878ed226SJulian Elischer 1302878ed226SJulian Elischer ep = mtod(event, ng_hci_qos_violation_ep *); 1303878ed226SJulian Elischer 1304878ed226SJulian Elischer /* Check if we have this connection handle */ 1305878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1306878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1307878ed226SJulian Elischer if (con == NULL) { 1308878ed226SJulian Elischer NG_HCI_ALERT( 1309878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1310878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1311878ed226SJulian Elischer error = ENOENT; 1312878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 1313878ed226SJulian Elischer NG_HCI_ALERT( 1314878ed226SJulian Elischer "%s: %s - invalid link type=%d\n", 1315878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->link_type); 1316878ed226SJulian Elischer error = EINVAL; 1317878ed226SJulian Elischer } else if (con->state != NG_HCI_CON_OPEN) { 1318878ed226SJulian Elischer NG_HCI_ALERT( 1319878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n", 1320878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->state, h); 1321878ed226SJulian Elischer error = EINVAL; 1322878ed226SJulian Elischer } else /* Notify upper layer */ 1323878ed226SJulian Elischer error = ng_hci_lp_qos_ind(con); 1324878ed226SJulian Elischer 1325878ed226SJulian Elischer NG_FREE_M(event); 1326878ed226SJulian Elischer 1327878ed226SJulian Elischer return (error); 1328878ed226SJulian Elischer } /* qos_violation */ 1329878ed226SJulian Elischer 1330878ed226SJulian Elischer /* Page scan mode change event */ 1331878ed226SJulian Elischer static int 1332878ed226SJulian Elischer page_scan_mode_change(ng_hci_unit_p unit, struct mbuf *event) 1333878ed226SJulian Elischer { 1334878ed226SJulian Elischer ng_hci_page_scan_mode_change_ep *ep = NULL; 1335878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 1336878ed226SJulian Elischer int error = 0; 1337878ed226SJulian Elischer 1338878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1339878ed226SJulian Elischer if (event == NULL) 1340878ed226SJulian Elischer return (ENOBUFS); 1341878ed226SJulian Elischer 1342878ed226SJulian Elischer ep = mtod(event, ng_hci_page_scan_mode_change_ep *); 1343878ed226SJulian Elischer 1344878ed226SJulian Elischer /* Update cache entry */ 1345fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL); 1346878ed226SJulian Elischer if (n == NULL) { 1347878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1348878ed226SJulian Elischer if (n == NULL) { 1349878ed226SJulian Elischer error = ENOMEM; 1350878ed226SJulian Elischer goto out; 1351878ed226SJulian Elischer } 1352878ed226SJulian Elischer 1353878ed226SJulian Elischer bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1354fbc48c2bSTakanori Watanabe n->addrtype = NG_HCI_LINK_ACL; 1355878ed226SJulian Elischer } else 1356878ed226SJulian Elischer getmicrotime(&n->updated); 1357878ed226SJulian Elischer 1358878ed226SJulian Elischer n->page_scan_mode = ep->page_scan_mode; 1359878ed226SJulian Elischer out: 1360878ed226SJulian Elischer NG_FREE_M(event); 1361878ed226SJulian Elischer 1362878ed226SJulian Elischer return (error); 1363878ed226SJulian Elischer } /* page_scan_mode_change */ 1364878ed226SJulian Elischer 1365878ed226SJulian Elischer /* Page scan repetition mode change event */ 1366878ed226SJulian Elischer static int 1367878ed226SJulian Elischer page_scan_rep_mode_change(ng_hci_unit_p unit, struct mbuf *event) 1368878ed226SJulian Elischer { 1369878ed226SJulian Elischer ng_hci_page_scan_rep_mode_change_ep *ep = NULL; 1370878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 1371878ed226SJulian Elischer int error = 0; 1372878ed226SJulian Elischer 1373878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1374878ed226SJulian Elischer if (event == NULL) 1375878ed226SJulian Elischer return (ENOBUFS); 1376878ed226SJulian Elischer 1377878ed226SJulian Elischer ep = mtod(event, ng_hci_page_scan_rep_mode_change_ep *); 1378878ed226SJulian Elischer 1379878ed226SJulian Elischer /* Update cache entry */ 1380fbc48c2bSTakanori Watanabe n = ng_hci_get_neighbor(unit, &ep->bdaddr, NG_HCI_LINK_ACL); 1381878ed226SJulian Elischer if (n == NULL) { 1382878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1383878ed226SJulian Elischer if (n == NULL) { 1384878ed226SJulian Elischer error = ENOMEM; 1385878ed226SJulian Elischer goto out; 1386878ed226SJulian Elischer } 1387878ed226SJulian Elischer 1388878ed226SJulian Elischer bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1389fbc48c2bSTakanori Watanabe n->addrtype = NG_HCI_LINK_ACL; 1390878ed226SJulian Elischer } else 1391878ed226SJulian Elischer getmicrotime(&n->updated); 1392878ed226SJulian Elischer 1393878ed226SJulian Elischer n->page_scan_rep_mode = ep->page_scan_rep_mode; 1394878ed226SJulian Elischer out: 1395878ed226SJulian Elischer NG_FREE_M(event); 1396878ed226SJulian Elischer 1397878ed226SJulian Elischer return (error); 1398878ed226SJulian Elischer } /* page_scan_rep_mode_change */ 1399