1878ed226SJulian Elischer /* 2878ed226SJulian Elischer * ng_hci_evnt.c 3878ed226SJulian Elischer * 4878ed226SJulian Elischer * Copyright (c) Maksim Yevmenkin <m_evmenkin@yahoo.com> 5878ed226SJulian Elischer * All rights reserved. 6878ed226SJulian Elischer * 7878ed226SJulian Elischer * Redistribution and use in source and binary forms, with or without 8878ed226SJulian Elischer * modification, are permitted provided that the following conditions 9878ed226SJulian Elischer * are met: 10878ed226SJulian Elischer * 1. Redistributions of source code must retain the above copyright 11878ed226SJulian Elischer * notice, this list of conditions and the following disclaimer. 12878ed226SJulian Elischer * 2. Redistributions in binary form must reproduce the above copyright 13878ed226SJulian Elischer * notice, this list of conditions and the following disclaimer in the 14878ed226SJulian Elischer * documentation and/or other materials provided with the distribution. 15878ed226SJulian Elischer * 16878ed226SJulian Elischer * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17878ed226SJulian Elischer * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18878ed226SJulian Elischer * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19878ed226SJulian Elischer * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20878ed226SJulian Elischer * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21878ed226SJulian Elischer * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22878ed226SJulian Elischer * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23878ed226SJulian Elischer * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24878ed226SJulian Elischer * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25878ed226SJulian Elischer * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26878ed226SJulian Elischer * SUCH DAMAGE. 27878ed226SJulian Elischer * 280986ab12SMaksim Yevmenkin * $Id: ng_hci_evnt.c,v 1.6 2003/09/08 18:57:51 max Exp $ 29878ed226SJulian Elischer * $FreeBSD$ 30878ed226SJulian Elischer */ 31878ed226SJulian Elischer 32878ed226SJulian Elischer #include <sys/param.h> 33878ed226SJulian Elischer #include <sys/systm.h> 34878ed226SJulian Elischer #include <sys/kernel.h> 35878ed226SJulian Elischer #include <sys/endian.h> 36878ed226SJulian Elischer #include <sys/malloc.h> 37878ed226SJulian Elischer #include <sys/mbuf.h> 38878ed226SJulian Elischer #include <sys/queue.h> 39878ed226SJulian Elischer #include <netgraph/ng_message.h> 40878ed226SJulian Elischer #include <netgraph/netgraph.h> 41b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_bluetooth.h> 42b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_hci.h> 43b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_var.h> 44b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_cmds.h> 45b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_evnt.h> 46b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_ulpi.h> 47b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_misc.h> 48878ed226SJulian Elischer 49878ed226SJulian Elischer /****************************************************************************** 50878ed226SJulian Elischer ****************************************************************************** 51878ed226SJulian Elischer ** HCI event processing module 52878ed226SJulian Elischer ****************************************************************************** 53878ed226SJulian Elischer ******************************************************************************/ 54878ed226SJulian Elischer 55878ed226SJulian Elischer /* 56878ed226SJulian Elischer * Event processing routines 57878ed226SJulian Elischer */ 58878ed226SJulian Elischer 59878ed226SJulian Elischer static int inquiry_result (ng_hci_unit_p, struct mbuf *); 60878ed226SJulian Elischer static int con_compl (ng_hci_unit_p, struct mbuf *); 61878ed226SJulian Elischer static int con_req (ng_hci_unit_p, struct mbuf *); 62878ed226SJulian Elischer static int discon_compl (ng_hci_unit_p, struct mbuf *); 63f2bb1caeSJulian Elischer static int encryption_change (ng_hci_unit_p, struct mbuf *); 64878ed226SJulian Elischer static int read_remote_features_compl (ng_hci_unit_p, struct mbuf *); 65878ed226SJulian Elischer static int qos_setup_compl (ng_hci_unit_p, struct mbuf *); 66878ed226SJulian Elischer static int hardware_error (ng_hci_unit_p, struct mbuf *); 67878ed226SJulian Elischer static int role_change (ng_hci_unit_p, struct mbuf *); 68878ed226SJulian Elischer static int num_compl_pkts (ng_hci_unit_p, struct mbuf *); 69878ed226SJulian Elischer static int mode_change (ng_hci_unit_p, struct mbuf *); 70878ed226SJulian Elischer static int data_buffer_overflow (ng_hci_unit_p, struct mbuf *); 71878ed226SJulian Elischer static int read_clock_offset_compl (ng_hci_unit_p, struct mbuf *); 72878ed226SJulian Elischer static int qos_violation (ng_hci_unit_p, struct mbuf *); 73878ed226SJulian Elischer static int page_scan_mode_change (ng_hci_unit_p, struct mbuf *); 74878ed226SJulian Elischer static int page_scan_rep_mode_change (ng_hci_unit_p, struct mbuf *); 75878ed226SJulian Elischer static int sync_con_queue (ng_hci_unit_p, ng_hci_unit_con_p, int); 76878ed226SJulian Elischer static int send_data_packets (ng_hci_unit_p, int, int); 77878ed226SJulian Elischer 78878ed226SJulian Elischer /* 79878ed226SJulian Elischer * Process HCI event packet 80878ed226SJulian Elischer */ 81878ed226SJulian Elischer 82878ed226SJulian Elischer int 83878ed226SJulian Elischer ng_hci_process_event(ng_hci_unit_p unit, struct mbuf *event) 84878ed226SJulian Elischer { 85878ed226SJulian Elischer ng_hci_event_pkt_t *hdr = NULL; 86878ed226SJulian Elischer int error = 0; 87878ed226SJulian Elischer 88878ed226SJulian Elischer /* Get event packet header */ 89878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*hdr)); 90878ed226SJulian Elischer if (event == NULL) 91878ed226SJulian Elischer return (ENOBUFS); 92878ed226SJulian Elischer 93878ed226SJulian Elischer hdr = mtod(event, ng_hci_event_pkt_t *); 94878ed226SJulian Elischer 95878ed226SJulian Elischer NG_HCI_INFO( 96878ed226SJulian Elischer "%s: %s - got HCI event=%#x, length=%d\n", 97878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), hdr->event, hdr->length); 98878ed226SJulian Elischer 99878ed226SJulian Elischer /* Get rid of event header and process event */ 100878ed226SJulian Elischer m_adj(event, sizeof(*hdr)); 101878ed226SJulian Elischer 102878ed226SJulian Elischer switch (hdr->event) { 103878ed226SJulian Elischer case NG_HCI_EVENT_INQUIRY_COMPL: 104878ed226SJulian Elischer case NG_HCI_EVENT_RETURN_LINK_KEYS: 105878ed226SJulian Elischer case NG_HCI_EVENT_PIN_CODE_REQ: 106878ed226SJulian Elischer case NG_HCI_EVENT_LINK_KEY_REQ: 107878ed226SJulian Elischer case NG_HCI_EVENT_LINK_KEY_NOTIFICATION: 108878ed226SJulian Elischer case NG_HCI_EVENT_LOOPBACK_COMMAND: 109878ed226SJulian Elischer case NG_HCI_EVENT_AUTH_COMPL: 110878ed226SJulian Elischer case NG_HCI_EVENT_CHANGE_CON_LINK_KEY_COMPL: 111878ed226SJulian Elischer case NG_HCI_EVENT_MASTER_LINK_KEY_COMPL: 112878ed226SJulian Elischer case NG_HCI_EVENT_FLUSH_OCCUR: /* XXX Do we have to handle it? */ 113878ed226SJulian Elischer case NG_HCI_EVENT_MAX_SLOT_CHANGE: 114878ed226SJulian Elischer case NG_HCI_EVENT_CON_PKT_TYPE_CHANGED: 115878ed226SJulian Elischer case NG_HCI_EVENT_BT_LOGO: 116878ed226SJulian Elischer case NG_HCI_EVENT_VENDOR: 117878ed226SJulian Elischer case NG_HCI_EVENT_REMOTE_NAME_REQ_COMPL: 118878ed226SJulian Elischer case NG_HCI_EVENT_READ_REMOTE_VER_INFO_COMPL: 119878ed226SJulian Elischer /* These do not need post processing */ 120878ed226SJulian Elischer NG_FREE_M(event); 121878ed226SJulian Elischer break; 122878ed226SJulian Elischer 123878ed226SJulian Elischer case NG_HCI_EVENT_INQUIRY_RESULT: 124878ed226SJulian Elischer error = inquiry_result(unit, event); 125878ed226SJulian Elischer break; 126878ed226SJulian Elischer 127878ed226SJulian Elischer case NG_HCI_EVENT_CON_COMPL: 128878ed226SJulian Elischer error = con_compl(unit, event); 129878ed226SJulian Elischer break; 130878ed226SJulian Elischer 131878ed226SJulian Elischer case NG_HCI_EVENT_CON_REQ: 132878ed226SJulian Elischer error = con_req(unit, event); 133878ed226SJulian Elischer break; 134878ed226SJulian Elischer 135878ed226SJulian Elischer case NG_HCI_EVENT_DISCON_COMPL: 136878ed226SJulian Elischer error = discon_compl(unit, event); 137878ed226SJulian Elischer break; 138878ed226SJulian Elischer 139f2bb1caeSJulian Elischer case NG_HCI_EVENT_ENCRYPTION_CHANGE: 140f2bb1caeSJulian Elischer error = encryption_change(unit, event); 141f2bb1caeSJulian Elischer break; 142f2bb1caeSJulian Elischer 143878ed226SJulian Elischer case NG_HCI_EVENT_READ_REMOTE_FEATURES_COMPL: 144878ed226SJulian Elischer error = read_remote_features_compl(unit, event); 145878ed226SJulian Elischer break; 146878ed226SJulian Elischer 147878ed226SJulian Elischer case NG_HCI_EVENT_QOS_SETUP_COMPL: 148878ed226SJulian Elischer error = qos_setup_compl(unit, event); 149878ed226SJulian Elischer break; 150878ed226SJulian Elischer 151878ed226SJulian Elischer case NG_HCI_EVENT_COMMAND_COMPL: 152878ed226SJulian Elischer error = ng_hci_process_command_complete(unit, event); 153878ed226SJulian Elischer break; 154878ed226SJulian Elischer 155878ed226SJulian Elischer case NG_HCI_EVENT_COMMAND_STATUS: 156878ed226SJulian Elischer error = ng_hci_process_command_status(unit, event); 157878ed226SJulian Elischer break; 158878ed226SJulian Elischer 159878ed226SJulian Elischer case NG_HCI_EVENT_HARDWARE_ERROR: 160878ed226SJulian Elischer error = hardware_error(unit, event); 161878ed226SJulian Elischer break; 162878ed226SJulian Elischer 163878ed226SJulian Elischer case NG_HCI_EVENT_ROLE_CHANGE: 164878ed226SJulian Elischer error = role_change(unit, event); 165878ed226SJulian Elischer break; 166878ed226SJulian Elischer 167878ed226SJulian Elischer case NG_HCI_EVENT_NUM_COMPL_PKTS: 168878ed226SJulian Elischer error = num_compl_pkts(unit, event); 169878ed226SJulian Elischer break; 170878ed226SJulian Elischer 171878ed226SJulian Elischer case NG_HCI_EVENT_MODE_CHANGE: 172878ed226SJulian Elischer error = mode_change(unit, event); 173878ed226SJulian Elischer break; 174878ed226SJulian Elischer 175878ed226SJulian Elischer case NG_HCI_EVENT_DATA_BUFFER_OVERFLOW: 176878ed226SJulian Elischer error = data_buffer_overflow(unit, event); 177878ed226SJulian Elischer break; 178878ed226SJulian Elischer 179878ed226SJulian Elischer case NG_HCI_EVENT_READ_CLOCK_OFFSET_COMPL: 180878ed226SJulian Elischer error = read_clock_offset_compl(unit, event); 181878ed226SJulian Elischer break; 182878ed226SJulian Elischer 183878ed226SJulian Elischer case NG_HCI_EVENT_QOS_VIOLATION: 184878ed226SJulian Elischer error = qos_violation(unit, event); 185878ed226SJulian Elischer break; 186878ed226SJulian Elischer 187878ed226SJulian Elischer case NG_HCI_EVENT_PAGE_SCAN_MODE_CHANGE: 188878ed226SJulian Elischer error = page_scan_mode_change(unit, event); 189878ed226SJulian Elischer break; 190878ed226SJulian Elischer 191878ed226SJulian Elischer case NG_HCI_EVENT_PAGE_SCAN_REP_MODE_CHANGE: 192878ed226SJulian Elischer error = page_scan_rep_mode_change(unit, event); 193878ed226SJulian Elischer break; 194878ed226SJulian Elischer 195878ed226SJulian Elischer default: 196878ed226SJulian Elischer NG_FREE_M(event); 197878ed226SJulian Elischer error = EINVAL; 198878ed226SJulian Elischer break; 199878ed226SJulian Elischer } 200878ed226SJulian Elischer 201878ed226SJulian Elischer return (error); 202878ed226SJulian Elischer } /* ng_hci_process_event */ 203878ed226SJulian Elischer 204878ed226SJulian Elischer /* 205878ed226SJulian Elischer * Send ACL and/or SCO data to the unit driver 206878ed226SJulian Elischer */ 207878ed226SJulian Elischer 208878ed226SJulian Elischer void 209878ed226SJulian Elischer ng_hci_send_data(ng_hci_unit_p unit) 210878ed226SJulian Elischer { 211878ed226SJulian Elischer int count; 212878ed226SJulian Elischer 213878ed226SJulian Elischer /* Send ACL data */ 214878ed226SJulian Elischer NG_HCI_BUFF_ACL_AVAIL(unit->buffer, count); 215878ed226SJulian Elischer 216878ed226SJulian Elischer NG_HCI_INFO( 217878ed226SJulian Elischer "%s: %s - sending ACL data packets, count=%d\n", 218878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), count); 219878ed226SJulian Elischer 220878ed226SJulian Elischer if (count > 0) { 221878ed226SJulian Elischer count = send_data_packets(unit, NG_HCI_LINK_ACL, count); 222878ed226SJulian Elischer NG_HCI_STAT_ACL_SENT(unit->stat, count); 223878ed226SJulian Elischer NG_HCI_BUFF_ACL_USE(unit->buffer, count); 224878ed226SJulian Elischer } 225878ed226SJulian Elischer 226878ed226SJulian Elischer /* Send SCO data */ 227878ed226SJulian Elischer NG_HCI_BUFF_SCO_AVAIL(unit->buffer, count); 228878ed226SJulian Elischer 229878ed226SJulian Elischer NG_HCI_INFO( 230878ed226SJulian Elischer "%s: %s - sending SCO data packets, count=%d\n", 231878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), count); 232878ed226SJulian Elischer 233878ed226SJulian Elischer if (count > 0) { 234878ed226SJulian Elischer count = send_data_packets(unit, NG_HCI_LINK_SCO, count); 235878ed226SJulian Elischer NG_HCI_STAT_SCO_SENT(unit->stat, count); 236878ed226SJulian Elischer NG_HCI_BUFF_SCO_USE(unit->buffer, count); 237878ed226SJulian Elischer } 238878ed226SJulian Elischer } /* ng_hci_send_data */ 239878ed226SJulian Elischer 240878ed226SJulian Elischer /* 241878ed226SJulian Elischer * Send data packets to the lower layer. 242878ed226SJulian Elischer */ 243878ed226SJulian Elischer 244878ed226SJulian Elischer static int 245878ed226SJulian Elischer send_data_packets(ng_hci_unit_p unit, int link_type, int limit) 246878ed226SJulian Elischer { 247878ed226SJulian Elischer ng_hci_unit_con_p con = NULL, winner = NULL; 248878ed226SJulian Elischer item_p item = NULL; 249878ed226SJulian Elischer int min_pending, total_sent, sent, error, v; 250878ed226SJulian Elischer 251878ed226SJulian Elischer for (total_sent = 0; limit > 0; ) { 252878ed226SJulian Elischer min_pending = 0x0fffffff; 253878ed226SJulian Elischer winner = NULL; 254878ed226SJulian Elischer 255878ed226SJulian Elischer /* 256878ed226SJulian Elischer * Find the connection that has has data to send 257878ed226SJulian Elischer * and the smallest number of pending packets 258878ed226SJulian Elischer */ 259878ed226SJulian Elischer 260878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) { 261878ed226SJulian Elischer if (con->link_type != link_type) 262878ed226SJulian Elischer continue; 263878ed226SJulian Elischer if (NG_BT_ITEMQ_LEN(&con->conq) == 0) 264878ed226SJulian Elischer continue; 265878ed226SJulian Elischer 266878ed226SJulian Elischer if (con->pending < min_pending) { 267878ed226SJulian Elischer winner = con; 268878ed226SJulian Elischer min_pending = con->pending; 269878ed226SJulian Elischer } 270878ed226SJulian Elischer } 271878ed226SJulian Elischer 272878ed226SJulian Elischer if (winner == NULL) 273878ed226SJulian Elischer break; 274878ed226SJulian Elischer 275878ed226SJulian Elischer /* 276878ed226SJulian Elischer * OK, we have a winner now send as much packets as we can 277878ed226SJulian Elischer * Count the number of packets we have sent and then sync 278878ed226SJulian Elischer * winner connection queue. 279878ed226SJulian Elischer */ 280878ed226SJulian Elischer 281878ed226SJulian Elischer for (sent = 0; limit > 0; limit --, total_sent ++, sent ++) { 282878ed226SJulian Elischer NG_BT_ITEMQ_DEQUEUE(&winner->conq, item); 283878ed226SJulian Elischer if (item == NULL) 284878ed226SJulian Elischer break; 285878ed226SJulian Elischer 286878ed226SJulian Elischer NG_HCI_INFO( 287878ed226SJulian Elischer "%s: %s - sending data packet, handle=%d, len=%d\n", 288878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 289878ed226SJulian Elischer winner->con_handle, NGI_M(item)->m_pkthdr.len); 290878ed226SJulian Elischer 291878ed226SJulian Elischer /* Check if driver hook still there */ 292878ed226SJulian Elischer v = (unit->drv != NULL && NG_HOOK_IS_VALID(unit->drv)); 293878ed226SJulian Elischer if (!v || (unit->state & NG_HCI_UNIT_READY) != 294878ed226SJulian Elischer NG_HCI_UNIT_READY) { 295878ed226SJulian Elischer NG_HCI_ERR( 296878ed226SJulian Elischer "%s: %s - could not send data. Hook \"%s\" is %svalid, state=%#x\n", 297878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 298878ed226SJulian Elischer NG_HCI_HOOK_DRV, ((v)? "" : "not "), 299878ed226SJulian Elischer unit->state); 300878ed226SJulian Elischer 301878ed226SJulian Elischer NG_FREE_ITEM(item); 302878ed226SJulian Elischer error = ENOTCONN; 303878ed226SJulian Elischer } else { 304878ed226SJulian Elischer v = NGI_M(item)->m_pkthdr.len; 305878ed226SJulian Elischer 306878ed226SJulian Elischer /* Give packet to raw hook */ 307878ed226SJulian Elischer ng_hci_mtap(unit, NGI_M(item)); 308878ed226SJulian Elischer 309878ed226SJulian Elischer /* ... and forward item to the driver */ 310878ed226SJulian Elischer NG_FWD_ITEM_HOOK(error, item, unit->drv); 311878ed226SJulian Elischer } 312878ed226SJulian Elischer 313878ed226SJulian Elischer if (error != 0) { 314878ed226SJulian Elischer NG_HCI_ERR( 315878ed226SJulian Elischer "%s: %s - could not send data packet, handle=%d, error=%d\n", 316878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 317878ed226SJulian Elischer winner->con_handle, error); 318878ed226SJulian Elischer break; 319878ed226SJulian Elischer } 320878ed226SJulian Elischer 321878ed226SJulian Elischer winner->pending ++; 322878ed226SJulian Elischer NG_HCI_STAT_BYTES_SENT(unit->stat, v); 323878ed226SJulian Elischer } 324878ed226SJulian Elischer 325878ed226SJulian Elischer /* 326878ed226SJulian Elischer * Sync connection queue for the winner 327878ed226SJulian Elischer */ 328878ed226SJulian Elischer 329878ed226SJulian Elischer sync_con_queue(unit, winner, sent); 330878ed226SJulian Elischer } 331878ed226SJulian Elischer 332878ed226SJulian Elischer return (total_sent); 333878ed226SJulian Elischer } /* send_data_packets */ 334878ed226SJulian Elischer 335878ed226SJulian Elischer /* 336878ed226SJulian Elischer * Send flow control messages to the upper layer 337878ed226SJulian Elischer */ 338878ed226SJulian Elischer 339878ed226SJulian Elischer static int 340878ed226SJulian Elischer sync_con_queue(ng_hci_unit_p unit, ng_hci_unit_con_p con, int completed) 341878ed226SJulian Elischer { 342878ed226SJulian Elischer hook_p hook = NULL; 343878ed226SJulian Elischer struct ng_mesg *msg = NULL; 344878ed226SJulian Elischer ng_hci_sync_con_queue_ep *state = NULL; 345878ed226SJulian Elischer int error; 346878ed226SJulian Elischer 347878ed226SJulian Elischer hook = (con->link_type == NG_HCI_LINK_ACL)? unit->acl : unit->sco; 348878ed226SJulian Elischer if (hook == NULL || NG_HOOK_NOT_VALID(hook)) 349878ed226SJulian Elischer return (ENOTCONN); 350878ed226SJulian Elischer 351878ed226SJulian Elischer NG_MKMESSAGE(msg, NGM_HCI_COOKIE, NGM_HCI_SYNC_CON_QUEUE, 352878ed226SJulian Elischer sizeof(*state), M_NOWAIT); 353878ed226SJulian Elischer if (msg == NULL) 354878ed226SJulian Elischer return (ENOMEM); 355878ed226SJulian Elischer 356878ed226SJulian Elischer state = (ng_hci_sync_con_queue_ep *)(msg->data); 357878ed226SJulian Elischer state->con_handle = con->con_handle; 358878ed226SJulian Elischer state->completed = completed; 359878ed226SJulian Elischer 3604ae439a3SMaksim Yevmenkin NG_SEND_MSG_HOOK(error, unit->node, msg, hook, 0); 361878ed226SJulian Elischer 362878ed226SJulian Elischer return (error); 363878ed226SJulian Elischer } /* sync_con_queue */ 364878ed226SJulian Elischer 365878ed226SJulian Elischer /* Inquiry result event */ 366878ed226SJulian Elischer static int 367878ed226SJulian Elischer inquiry_result(ng_hci_unit_p unit, struct mbuf *event) 368878ed226SJulian Elischer { 369878ed226SJulian Elischer ng_hci_inquiry_result_ep *ep = NULL; 370878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 371878ed226SJulian Elischer bdaddr_t bdaddr; 372878ed226SJulian Elischer int error = 0; 373878ed226SJulian Elischer 374878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 375878ed226SJulian Elischer if (event == NULL) 376878ed226SJulian Elischer return (ENOBUFS); 377878ed226SJulian Elischer 378878ed226SJulian Elischer ep = mtod(event, ng_hci_inquiry_result_ep *); 379878ed226SJulian Elischer m_adj(event, sizeof(*ep)); 380878ed226SJulian Elischer 381878ed226SJulian Elischer for (; ep->num_responses > 0; ep->num_responses --) { 382878ed226SJulian Elischer /* Get remote unit address */ 383878ed226SJulian Elischer m_copydata(event, 0, sizeof(bdaddr), (caddr_t) &bdaddr); 384878ed226SJulian Elischer m_adj(event, sizeof(bdaddr)); 385878ed226SJulian Elischer 386878ed226SJulian Elischer /* Lookup entry in the cache */ 387878ed226SJulian Elischer n = ng_hci_get_neighbor(unit, &bdaddr); 388878ed226SJulian Elischer if (n == NULL) { 389878ed226SJulian Elischer /* Create new entry */ 390878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 391878ed226SJulian Elischer if (n == NULL) { 392878ed226SJulian Elischer error = ENOMEM; 393878ed226SJulian Elischer break; 394878ed226SJulian Elischer } 395878ed226SJulian Elischer } else 396878ed226SJulian Elischer getmicrotime(&n->updated); 397878ed226SJulian Elischer 398878ed226SJulian Elischer bcopy(&bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 399878ed226SJulian Elischer 400878ed226SJulian Elischer /* XXX call m_pullup here? */ 401878ed226SJulian Elischer 402878ed226SJulian Elischer n->page_scan_rep_mode = *mtod(event, u_int8_t *); 403878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 404878ed226SJulian Elischer 405878ed226SJulian Elischer /* page_scan_period_mode */ 406878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 407878ed226SJulian Elischer 408878ed226SJulian Elischer n->page_scan_mode = *mtod(event, u_int8_t *); 409878ed226SJulian Elischer m_adj(event, sizeof(u_int8_t)); 410878ed226SJulian Elischer 411878ed226SJulian Elischer /* class */ 412878ed226SJulian Elischer m_adj(event, NG_HCI_CLASS_SIZE); 413878ed226SJulian Elischer 414878ed226SJulian Elischer /* clock offset */ 415878ed226SJulian Elischer m_copydata(event, 0, sizeof(n->clock_offset), 416878ed226SJulian Elischer (caddr_t) &n->clock_offset); 417878ed226SJulian Elischer n->clock_offset = le16toh(n->clock_offset); 418878ed226SJulian Elischer } 419878ed226SJulian Elischer 420878ed226SJulian Elischer NG_FREE_M(event); 421878ed226SJulian Elischer 422878ed226SJulian Elischer return (error); 423878ed226SJulian Elischer } /* inquiry_result */ 424878ed226SJulian Elischer 425878ed226SJulian Elischer /* Connection complete event */ 426878ed226SJulian Elischer static int 427878ed226SJulian Elischer con_compl(ng_hci_unit_p unit, struct mbuf *event) 428878ed226SJulian Elischer { 429878ed226SJulian Elischer ng_hci_con_compl_ep *ep = NULL; 430878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 431878ed226SJulian Elischer int error = 0; 432878ed226SJulian Elischer 433878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 434878ed226SJulian Elischer if (event == NULL) 435878ed226SJulian Elischer return (ENOBUFS); 436878ed226SJulian Elischer 437878ed226SJulian Elischer ep = mtod(event, ng_hci_con_compl_ep *); 438878ed226SJulian Elischer 439878ed226SJulian Elischer /* 440878ed226SJulian Elischer * Find the first connection descriptor that matches the following: 441878ed226SJulian Elischer * 442878ed226SJulian Elischer * 1) con->link_type == ep->link_type 443878ed226SJulian Elischer * 2) con->state == NG_HCI_CON_W4_CONN_COMPLETE 444878ed226SJulian Elischer * 3) con->bdaddr == ep->bdaddr 445878ed226SJulian Elischer */ 446878ed226SJulian Elischer 447878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 448878ed226SJulian Elischer if (con->link_type == ep->link_type && 449878ed226SJulian Elischer con->state == NG_HCI_CON_W4_CONN_COMPLETE && 450878ed226SJulian Elischer bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0) 451878ed226SJulian Elischer break; 452878ed226SJulian Elischer 453878ed226SJulian Elischer /* 454878ed226SJulian Elischer * Two possible cases: 455878ed226SJulian Elischer * 456878ed226SJulian Elischer * 1) We have found connection descriptor. That means upper layer has 4570986ab12SMaksim Yevmenkin * requested this connection via LP_CON_REQ message. In this case 4580986ab12SMaksim Yevmenkin * connection must have timeout set. If ng_hci_con_untimeout() fails 4590986ab12SMaksim Yevmenkin * then timeout message already went into node's queue. In this case 4600986ab12SMaksim Yevmenkin * ignore Connection_Complete event and let timeout deal with it. 461878ed226SJulian Elischer * 462878ed226SJulian Elischer * 2) We do not have connection descriptor. That means upper layer 463878ed226SJulian Elischer * nas not requested this connection or (less likely) we gave up 464878ed226SJulian Elischer * on this connection (timeout). The most likely scenario is that 465878ed226SJulian Elischer * we have received Create_Connection/Add_SCO_Connection command 466878ed226SJulian Elischer * from the RAW hook 467878ed226SJulian Elischer */ 468878ed226SJulian Elischer 469878ed226SJulian Elischer if (con == NULL) { 470878ed226SJulian Elischer if (ep->status != 0) 471878ed226SJulian Elischer goto out; 472878ed226SJulian Elischer 473878ed226SJulian Elischer con = ng_hci_new_con(unit, ep->link_type); 474878ed226SJulian Elischer if (con == NULL) { 475878ed226SJulian Elischer error = ENOMEM; 476878ed226SJulian Elischer goto out; 477878ed226SJulian Elischer } 478878ed226SJulian Elischer 479878ed226SJulian Elischer bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr)); 4800986ab12SMaksim Yevmenkin } else if ((error = ng_hci_con_untimeout(con)) != 0) 4810986ab12SMaksim Yevmenkin goto out; 482878ed226SJulian Elischer 483878ed226SJulian Elischer /* 484878ed226SJulian Elischer * Update connection descriptor and send notification 485878ed226SJulian Elischer * to the upper layers. 486878ed226SJulian Elischer */ 487878ed226SJulian Elischer 488878ed226SJulian Elischer con->con_handle = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 489878ed226SJulian Elischer con->encryption_mode = ep->encryption_mode; 490878ed226SJulian Elischer 491878ed226SJulian Elischer ng_hci_lp_con_cfm(con, ep->status); 492878ed226SJulian Elischer 493878ed226SJulian Elischer /* Adjust connection state */ 494878ed226SJulian Elischer if (ep->status != 0) 495878ed226SJulian Elischer ng_hci_free_con(con); 496878ed226SJulian Elischer else { 497878ed226SJulian Elischer con->state = NG_HCI_CON_OPEN; 498878ed226SJulian Elischer 499878ed226SJulian Elischer /* 500878ed226SJulian Elischer * Change link policy for the ACL connections. Enable all 501878ed226SJulian Elischer * supported link modes. Enable Role switch as well if 502878ed226SJulian Elischer * device supports it. 503878ed226SJulian Elischer */ 504878ed226SJulian Elischer 505878ed226SJulian Elischer if (ep->link_type == NG_HCI_LINK_ACL) { 506878ed226SJulian Elischer struct __link_policy { 507878ed226SJulian Elischer ng_hci_cmd_pkt_t hdr; 508878ed226SJulian Elischer ng_hci_write_link_policy_settings_cp cp; 509878ed226SJulian Elischer } __attribute__ ((packed)) *lp; 510878ed226SJulian Elischer struct mbuf *m; 511878ed226SJulian Elischer 512a163d034SWarner Losh MGETHDR(m, M_DONTWAIT, MT_DATA); 513878ed226SJulian Elischer if (m != NULL) { 514878ed226SJulian Elischer m->m_pkthdr.len = m->m_len = sizeof(*lp); 515878ed226SJulian Elischer lp = mtod(m, struct __link_policy *); 516878ed226SJulian Elischer 517878ed226SJulian Elischer lp->hdr.type = NG_HCI_CMD_PKT; 518878ed226SJulian Elischer lp->hdr.opcode = htole16(NG_HCI_OPCODE( 519878ed226SJulian Elischer NG_HCI_OGF_LINK_POLICY, 520878ed226SJulian Elischer NG_HCI_OCF_WRITE_LINK_POLICY_SETTINGS)); 521878ed226SJulian Elischer lp->hdr.length = sizeof(lp->cp); 522878ed226SJulian Elischer 523878ed226SJulian Elischer lp->cp.con_handle = ep->con_handle; 524878ed226SJulian Elischer 525878ed226SJulian Elischer lp->cp.settings = 0; 526f2bb1caeSJulian Elischer if ((unit->features[0] & NG_HCI_LMP_SWITCH) && 527f2bb1caeSJulian Elischer unit->role_switch) 528878ed226SJulian Elischer lp->cp.settings |= 0x1; 529878ed226SJulian Elischer if (unit->features[0] & NG_HCI_LMP_HOLD_MODE) 530878ed226SJulian Elischer lp->cp.settings |= 0x2; 531878ed226SJulian Elischer if (unit->features[0] & NG_HCI_LMP_SNIFF_MODE) 532878ed226SJulian Elischer lp->cp.settings |= 0x4; 533878ed226SJulian Elischer if (unit->features[1] & NG_HCI_LMP_PARK_MODE) 534878ed226SJulian Elischer lp->cp.settings |= 0x8; 535878ed226SJulian Elischer 536878ed226SJulian Elischer lp->cp.settings &= unit->link_policy_mask; 537878ed226SJulian Elischer lp->cp.settings = htole16(lp->cp.settings); 538878ed226SJulian Elischer 539878ed226SJulian Elischer NG_BT_MBUFQ_ENQUEUE(&unit->cmdq, m); 540878ed226SJulian Elischer if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) 541878ed226SJulian Elischer ng_hci_send_command(unit); 542878ed226SJulian Elischer } 543878ed226SJulian Elischer } 544878ed226SJulian Elischer } 545878ed226SJulian Elischer out: 546878ed226SJulian Elischer NG_FREE_M(event); 547878ed226SJulian Elischer 548878ed226SJulian Elischer return (error); 5490986ab12SMaksim Yevmenkin } /* con_compl */ 550878ed226SJulian Elischer 551878ed226SJulian Elischer /* Connection request event */ 552878ed226SJulian Elischer static int 553878ed226SJulian Elischer con_req(ng_hci_unit_p unit, struct mbuf *event) 554878ed226SJulian Elischer { 555878ed226SJulian Elischer ng_hci_con_req_ep *ep = NULL; 556878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 557878ed226SJulian Elischer int error = 0; 558878ed226SJulian Elischer 559878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 560878ed226SJulian Elischer if (event == NULL) 561878ed226SJulian Elischer return (ENOBUFS); 562878ed226SJulian Elischer 563878ed226SJulian Elischer ep = mtod(event, ng_hci_con_req_ep *); 564878ed226SJulian Elischer 565878ed226SJulian Elischer /* 566878ed226SJulian Elischer * Find the first connection descriptor that matches the following: 567878ed226SJulian Elischer * 568878ed226SJulian Elischer * 1) con->link_type == ep->link_type 569878ed226SJulian Elischer * 570878ed226SJulian Elischer * 2) con->state == NG_HCI_CON_W4_LP_CON_RSP || 571878ed226SJulian Elischer * con->state == NG_HCI_CON_W4_CONN_COMPL 572878ed226SJulian Elischer * 573878ed226SJulian Elischer * 3) con->bdaddr == ep->bdaddr 574878ed226SJulian Elischer * 575878ed226SJulian Elischer * Possible cases: 576878ed226SJulian Elischer * 577878ed226SJulian Elischer * 1) We do not have connection descriptor. This is simple. Create 578878ed226SJulian Elischer * new fresh connection descriptor and send notification to the 579878ed226SJulian Elischer * appropriate upstream hook (based on link_type). 580878ed226SJulian Elischer * 581878ed226SJulian Elischer * 2) We found connection handle. This is more complicated. 582878ed226SJulian Elischer * 583878ed226SJulian Elischer * 2.1) ACL links 584878ed226SJulian Elischer * 585878ed226SJulian Elischer * Since only one ACL link can exist between each pair of 586878ed226SJulian Elischer * units then we have a race. Our upper layer has requested 587878ed226SJulian Elischer * an ACL connection to the remote unit, but we did not send 588878ed226SJulian Elischer * command yet. At the same time the remote unit has requested 589878ed226SJulian Elischer * an ACL connection from us. In this case we will ignore 590878ed226SJulian Elischer * Connection_Request event. This probably will cause connect 591878ed226SJulian Elischer * failure on both units. 592878ed226SJulian Elischer * 593878ed226SJulian Elischer * 2.2) SCO links 594878ed226SJulian Elischer * 595878ed226SJulian Elischer * The spec on page 45 says : 596878ed226SJulian Elischer * 597878ed226SJulian Elischer * "The master can support up to three SCO links to the same 598878ed226SJulian Elischer * slave or to different slaves. A slave can support up to 599878ed226SJulian Elischer * three SCO links from the same master, or two SCO links if 600878ed226SJulian Elischer * the links originate from different masters." 601878ed226SJulian Elischer * 602878ed226SJulian Elischer * The only problem is how to handle multiple SCO links between 603878ed226SJulian Elischer * matster and slave. For now we will assume that multiple SCO 604878ed226SJulian Elischer * links MUST be opened one after another. 605878ed226SJulian Elischer */ 606878ed226SJulian Elischer 607878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 608878ed226SJulian Elischer if (con->link_type == ep->link_type && 609878ed226SJulian Elischer (con->state == NG_HCI_CON_W4_LP_CON_RSP || 610878ed226SJulian Elischer con->state == NG_HCI_CON_W4_CONN_COMPLETE) && 611878ed226SJulian Elischer bcmp(&con->bdaddr, &ep->bdaddr, sizeof(bdaddr_t)) == 0) 612878ed226SJulian Elischer break; 613878ed226SJulian Elischer 614878ed226SJulian Elischer if (con == NULL) { 615878ed226SJulian Elischer con = ng_hci_new_con(unit, ep->link_type); 616878ed226SJulian Elischer if (con != NULL) { 617878ed226SJulian Elischer bcopy(&ep->bdaddr, &con->bdaddr, sizeof(con->bdaddr)); 618878ed226SJulian Elischer 619878ed226SJulian Elischer con->state = NG_HCI_CON_W4_LP_CON_RSP; 620878ed226SJulian Elischer ng_hci_con_timeout(con); 621878ed226SJulian Elischer 622878ed226SJulian Elischer error = ng_hci_lp_con_ind(con, ep->uclass); 6230986ab12SMaksim Yevmenkin if (error != 0) { 6240986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 625878ed226SJulian Elischer ng_hci_free_con(con); 6260986ab12SMaksim Yevmenkin } 627878ed226SJulian Elischer } else 628878ed226SJulian Elischer error = ENOMEM; 629878ed226SJulian Elischer } 630878ed226SJulian Elischer 631878ed226SJulian Elischer NG_FREE_M(event); 632878ed226SJulian Elischer 633878ed226SJulian Elischer return (error); 634878ed226SJulian Elischer } /* con_req */ 635878ed226SJulian Elischer 636878ed226SJulian Elischer /* Disconnect complete event */ 637878ed226SJulian Elischer static int 638878ed226SJulian Elischer discon_compl(ng_hci_unit_p unit, struct mbuf *event) 639878ed226SJulian Elischer { 640878ed226SJulian Elischer ng_hci_discon_compl_ep *ep = NULL; 641878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 642878ed226SJulian Elischer int error = 0; 643878ed226SJulian Elischer u_int16_t h; 644878ed226SJulian Elischer 645878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 646878ed226SJulian Elischer if (event == NULL) 647878ed226SJulian Elischer return (ENOBUFS); 648878ed226SJulian Elischer 649878ed226SJulian Elischer ep = mtod(event, ng_hci_discon_compl_ep *); 650878ed226SJulian Elischer 651878ed226SJulian Elischer /* 652878ed226SJulian Elischer * XXX 653878ed226SJulian Elischer * Do we have to send notification if ep->status != 0? 654878ed226SJulian Elischer * For now we will send notification for both ACL and SCO connections 655878ed226SJulian Elischer * ONLY if ep->status == 0. 656878ed226SJulian Elischer */ 657878ed226SJulian Elischer 658878ed226SJulian Elischer if (ep->status == 0) { 659878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 660878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 661878ed226SJulian Elischer if (con != NULL) { 662878ed226SJulian Elischer error = ng_hci_lp_discon_ind(con, ep->reason); 6630986ab12SMaksim Yevmenkin 6640986ab12SMaksim Yevmenkin /* Remove all timeouts (if any) */ 6650986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 6660986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 6670986ab12SMaksim Yevmenkin 668878ed226SJulian Elischer ng_hci_free_con(con); 669878ed226SJulian Elischer } else { 670878ed226SJulian Elischer NG_HCI_ALERT( 671878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 672878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 673878ed226SJulian Elischer error = ENOENT; 674878ed226SJulian Elischer } 675878ed226SJulian Elischer } 676878ed226SJulian Elischer 677878ed226SJulian Elischer NG_FREE_M(event); 678878ed226SJulian Elischer 679878ed226SJulian Elischer return (error); 680878ed226SJulian Elischer } /* discon_compl */ 681878ed226SJulian Elischer 682f2bb1caeSJulian Elischer /* Encryption change event */ 683f2bb1caeSJulian Elischer static int 684f2bb1caeSJulian Elischer encryption_change(ng_hci_unit_p unit, struct mbuf *event) 685f2bb1caeSJulian Elischer { 686f2bb1caeSJulian Elischer ng_hci_encryption_change_ep *ep = NULL; 687f2bb1caeSJulian Elischer ng_hci_unit_con_p con = NULL; 688f2bb1caeSJulian Elischer int error = 0; 689f2bb1caeSJulian Elischer 690f2bb1caeSJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 691f2bb1caeSJulian Elischer if (event == NULL) 692f2bb1caeSJulian Elischer return (ENOBUFS); 693f2bb1caeSJulian Elischer 694f2bb1caeSJulian Elischer ep = mtod(event, ng_hci_encryption_change_ep *); 695f2bb1caeSJulian Elischer 696f2bb1caeSJulian Elischer if (ep->status == 0) { 697f2bb1caeSJulian Elischer u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 698f2bb1caeSJulian Elischer 699f2bb1caeSJulian Elischer con = ng_hci_con_by_handle(unit, h); 700f2bb1caeSJulian Elischer if (con == NULL) { 701f2bb1caeSJulian Elischer NG_HCI_ALERT( 702f2bb1caeSJulian Elischer "%s: %s - invalid connection handle=%d\n", 703f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 704f2bb1caeSJulian Elischer error = ENOENT; 705f2bb1caeSJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 706f2bb1caeSJulian Elischer NG_HCI_ALERT( 707f2bb1caeSJulian Elischer "%s: %s - invalid link type=%d\n", 708f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), 709f2bb1caeSJulian Elischer con->link_type); 710f2bb1caeSJulian Elischer error = EINVAL; 711f2bb1caeSJulian Elischer } else if (ep->encryption_enable) 712f2bb1caeSJulian Elischer /* XXX is that true? */ 713f2bb1caeSJulian Elischer con->encryption_mode = NG_HCI_ENCRYPTION_MODE_P2P; 714f2bb1caeSJulian Elischer else 715f2bb1caeSJulian Elischer con->encryption_mode = NG_HCI_ENCRYPTION_MODE_NONE; 716f2bb1caeSJulian Elischer } else 717f2bb1caeSJulian Elischer NG_HCI_ERR( 718f2bb1caeSJulian Elischer "%s: %s - failed to change encryption mode, status=%d\n", 719f2bb1caeSJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 720f2bb1caeSJulian Elischer 721f2bb1caeSJulian Elischer NG_FREE_M(event); 722f2bb1caeSJulian Elischer 723f2bb1caeSJulian Elischer return (error); 724f2bb1caeSJulian Elischer } /* encryption_change */ 725f2bb1caeSJulian Elischer 726878ed226SJulian Elischer /* Read remote feature complete event */ 727878ed226SJulian Elischer static int 728878ed226SJulian Elischer read_remote_features_compl(ng_hci_unit_p unit, struct mbuf *event) 729878ed226SJulian Elischer { 730878ed226SJulian Elischer ng_hci_read_remote_features_compl_ep *ep = NULL; 731878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 732878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 733878ed226SJulian Elischer u_int16_t h; 734878ed226SJulian Elischer int error = 0; 735878ed226SJulian Elischer 736878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 737878ed226SJulian Elischer if (event == NULL) 738878ed226SJulian Elischer return (ENOBUFS); 739878ed226SJulian Elischer 740878ed226SJulian Elischer ep = mtod(event, ng_hci_read_remote_features_compl_ep *); 741878ed226SJulian Elischer 742878ed226SJulian Elischer if (ep->status == 0) { 743878ed226SJulian Elischer /* Check if we have this connection handle */ 744878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 745878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 746878ed226SJulian Elischer if (con == NULL) { 747878ed226SJulian Elischer NG_HCI_ALERT( 748878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 749878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 750878ed226SJulian Elischer error = ENOENT; 751878ed226SJulian Elischer goto out; 752878ed226SJulian Elischer } 753878ed226SJulian Elischer 754878ed226SJulian Elischer /* Update cache entry */ 755878ed226SJulian Elischer n = ng_hci_get_neighbor(unit, &con->bdaddr); 756878ed226SJulian Elischer if (n == NULL) { 757878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 758878ed226SJulian Elischer if (n == NULL) { 759878ed226SJulian Elischer error = ENOMEM; 760878ed226SJulian Elischer goto out; 761878ed226SJulian Elischer } 762878ed226SJulian Elischer 763878ed226SJulian Elischer bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 764878ed226SJulian Elischer } else 765878ed226SJulian Elischer getmicrotime(&n->updated); 766878ed226SJulian Elischer 767878ed226SJulian Elischer bcopy(ep->features, n->features, sizeof(n->features)); 768878ed226SJulian Elischer } else 769878ed226SJulian Elischer NG_HCI_ERR( 770878ed226SJulian Elischer "%s: %s - failed to read remote unit features, status=%d\n", 771878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 772878ed226SJulian Elischer out: 773878ed226SJulian Elischer NG_FREE_M(event); 774878ed226SJulian Elischer 775878ed226SJulian Elischer return (error); 776878ed226SJulian Elischer } /* read_remote_features_compl */ 777878ed226SJulian Elischer 778878ed226SJulian Elischer /* QoS setup complete event */ 779878ed226SJulian Elischer static int 780878ed226SJulian Elischer qos_setup_compl(ng_hci_unit_p unit, struct mbuf *event) 781878ed226SJulian Elischer { 782878ed226SJulian Elischer ng_hci_qos_setup_compl_ep *ep = NULL; 783878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 784878ed226SJulian Elischer u_int16_t h; 785878ed226SJulian Elischer int error = 0; 786878ed226SJulian Elischer 787878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 788878ed226SJulian Elischer if (event == NULL) 789878ed226SJulian Elischer return (ENOBUFS); 790878ed226SJulian Elischer 791878ed226SJulian Elischer ep = mtod(event, ng_hci_qos_setup_compl_ep *); 792878ed226SJulian Elischer 793878ed226SJulian Elischer /* Check if we have this connection handle */ 794878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 795878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 796878ed226SJulian Elischer if (con == NULL) { 797878ed226SJulian Elischer NG_HCI_ALERT( 798878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 799878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 800878ed226SJulian Elischer error = ENOENT; 801878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 802878ed226SJulian Elischer NG_HCI_ALERT( 803878ed226SJulian Elischer "%s: %s - invalid link type=%d, handle=%d\n", 804878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->link_type, h); 805878ed226SJulian Elischer error = EINVAL; 806878ed226SJulian Elischer } else if (con->state != NG_HCI_CON_OPEN) { 807878ed226SJulian Elischer NG_HCI_ALERT( 808878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n", 809878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 810878ed226SJulian Elischer con->state, h); 811878ed226SJulian Elischer error = EINVAL; 812878ed226SJulian Elischer } else /* Notify upper layer */ 813878ed226SJulian Elischer error = ng_hci_lp_qos_cfm(con, ep->status); 814878ed226SJulian Elischer 815878ed226SJulian Elischer NG_FREE_M(event); 816878ed226SJulian Elischer 817878ed226SJulian Elischer return (error); 818878ed226SJulian Elischer } /* qos_setup_compl */ 819878ed226SJulian Elischer 820878ed226SJulian Elischer /* Hardware error event */ 821878ed226SJulian Elischer static int 822878ed226SJulian Elischer hardware_error(ng_hci_unit_p unit, struct mbuf *event) 823878ed226SJulian Elischer { 824878ed226SJulian Elischer NG_HCI_ALERT( 825878ed226SJulian Elischer "%s: %s - hardware error %#x\n", 826878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), *mtod(event, u_int8_t *)); 827878ed226SJulian Elischer 828878ed226SJulian Elischer NG_FREE_M(event); 829878ed226SJulian Elischer 830878ed226SJulian Elischer return (0); 831878ed226SJulian Elischer } /* hardware_error */ 832878ed226SJulian Elischer 833878ed226SJulian Elischer /* Role change event */ 834878ed226SJulian Elischer static int 835878ed226SJulian Elischer role_change(ng_hci_unit_p unit, struct mbuf *event) 836878ed226SJulian Elischer { 837878ed226SJulian Elischer ng_hci_role_change_ep *ep = NULL; 838878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 839878ed226SJulian Elischer 840878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 841878ed226SJulian Elischer if (event == NULL) 842878ed226SJulian Elischer return (ENOBUFS); 843878ed226SJulian Elischer 844878ed226SJulian Elischer ep = mtod(event, ng_hci_role_change_ep *); 845878ed226SJulian Elischer 846878ed226SJulian Elischer if (ep->status == 0) { 847878ed226SJulian Elischer /* XXX shoud we also change "role" for SCO connections? */ 848878ed226SJulian Elischer con = ng_hci_con_by_bdaddr(unit, &ep->bdaddr, NG_HCI_LINK_ACL); 849878ed226SJulian Elischer if (con != NULL) 850878ed226SJulian Elischer con->role = ep->role; 851878ed226SJulian Elischer else 852878ed226SJulian Elischer NG_HCI_ALERT( 853878ed226SJulian Elischer "%s: %s - ACL connection does not exist, bdaddr=%x:%x:%x:%x:%x:%x\n", 854878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 855878ed226SJulian Elischer ep->bdaddr.b[5], ep->bdaddr.b[4], 856878ed226SJulian Elischer ep->bdaddr.b[3], ep->bdaddr.b[2], 857878ed226SJulian Elischer ep->bdaddr.b[1], ep->bdaddr.b[0]); 858878ed226SJulian Elischer } else 859878ed226SJulian Elischer NG_HCI_ERR( 860878ed226SJulian Elischer "%s: %s - failed to change role, status=%d, bdaddr=%x:%x:%x:%x:%x:%x\n", 861878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status, 862878ed226SJulian Elischer ep->bdaddr.b[5], ep->bdaddr.b[4], ep->bdaddr.b[3], 863878ed226SJulian Elischer ep->bdaddr.b[2], ep->bdaddr.b[1], ep->bdaddr.b[0]); 864878ed226SJulian Elischer 865878ed226SJulian Elischer NG_FREE_M(event); 866878ed226SJulian Elischer 867878ed226SJulian Elischer return (0); 868878ed226SJulian Elischer } /* role_change */ 869878ed226SJulian Elischer 870878ed226SJulian Elischer /* Number of completed packets event */ 871878ed226SJulian Elischer static int 872878ed226SJulian Elischer num_compl_pkts(ng_hci_unit_p unit, struct mbuf *event) 873878ed226SJulian Elischer { 874878ed226SJulian Elischer ng_hci_num_compl_pkts_ep *ep = NULL; 875878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 876878ed226SJulian Elischer u_int16_t h, p; 877878ed226SJulian Elischer 878878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 879878ed226SJulian Elischer if (event == NULL) 880878ed226SJulian Elischer return (ENOBUFS); 881878ed226SJulian Elischer 882878ed226SJulian Elischer ep = mtod(event, ng_hci_num_compl_pkts_ep *); 883878ed226SJulian Elischer m_adj(event, sizeof(*ep)); 884878ed226SJulian Elischer 885878ed226SJulian Elischer for (; ep->num_con_handles > 0; ep->num_con_handles --) { 886878ed226SJulian Elischer /* Get connection handle */ 887878ed226SJulian Elischer m_copydata(event, 0, sizeof(h), (caddr_t) &h); 888878ed226SJulian Elischer m_adj(event, sizeof(h)); 889878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(h)); 890878ed226SJulian Elischer 891878ed226SJulian Elischer /* Get number of completed packets */ 892878ed226SJulian Elischer m_copydata(event, 0, sizeof(p), (caddr_t) &p); 893878ed226SJulian Elischer m_adj(event, sizeof(p)); 894878ed226SJulian Elischer p = le16toh(p); 895878ed226SJulian Elischer 896878ed226SJulian Elischer /* Check if we have this connection handle */ 897878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 898878ed226SJulian Elischer if (con != NULL) { 899878ed226SJulian Elischer con->pending -= p; 900878ed226SJulian Elischer if (con->pending < 0) { 901878ed226SJulian Elischer NG_HCI_WARN( 902878ed226SJulian Elischer "%s: %s - pending packet counter is out of sync! " \ 903878ed226SJulian Elischer "handle=%d, pending=%d, ncp=%d\n", __func__, NG_NODE_NAME(unit->node), 904878ed226SJulian Elischer con->con_handle, con->pending, p); 905878ed226SJulian Elischer 906878ed226SJulian Elischer con->pending = 0; 907878ed226SJulian Elischer } 908878ed226SJulian Elischer 909878ed226SJulian Elischer /* Update buffer descriptor */ 910878ed226SJulian Elischer if (con->link_type == NG_HCI_LINK_ACL) 911878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(unit->buffer, p); 912878ed226SJulian Elischer else 913878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(unit->buffer, p); 914878ed226SJulian Elischer } else 915878ed226SJulian Elischer NG_HCI_ALERT( 916878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 917878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 918878ed226SJulian Elischer } 919878ed226SJulian Elischer 920878ed226SJulian Elischer NG_FREE_M(event); 921878ed226SJulian Elischer 922878ed226SJulian Elischer /* Send more data */ 923878ed226SJulian Elischer ng_hci_send_data(unit); 924878ed226SJulian Elischer 925878ed226SJulian Elischer return (0); 926878ed226SJulian Elischer } /* num_compl_pkts */ 927878ed226SJulian Elischer 928878ed226SJulian Elischer /* Mode change event */ 929878ed226SJulian Elischer static int 930878ed226SJulian Elischer mode_change(ng_hci_unit_p unit, struct mbuf *event) 931878ed226SJulian Elischer { 932878ed226SJulian Elischer ng_hci_mode_change_ep *ep = NULL; 933878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 934878ed226SJulian Elischer int error = 0; 935878ed226SJulian Elischer 936878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 937878ed226SJulian Elischer if (event == NULL) 938878ed226SJulian Elischer return (ENOBUFS); 939878ed226SJulian Elischer 940878ed226SJulian Elischer ep = mtod(event, ng_hci_mode_change_ep *); 941878ed226SJulian Elischer 942878ed226SJulian Elischer if (ep->status == 0) { 943878ed226SJulian Elischer u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 944878ed226SJulian Elischer 945878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 946878ed226SJulian Elischer if (con == NULL) { 947878ed226SJulian Elischer NG_HCI_ALERT( 948878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 949878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 950878ed226SJulian Elischer error = ENOENT; 951878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 952878ed226SJulian Elischer NG_HCI_ALERT( 953878ed226SJulian Elischer "%s: %s - invalid link type=%d\n", 954878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 955878ed226SJulian Elischer con->link_type); 956878ed226SJulian Elischer error = EINVAL; 957878ed226SJulian Elischer } else 958878ed226SJulian Elischer con->mode = ep->unit_mode; 959878ed226SJulian Elischer } else 960878ed226SJulian Elischer NG_HCI_ERR( 961878ed226SJulian Elischer "%s: %s - failed to change mode, status=%d\n", 962878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 963878ed226SJulian Elischer 964878ed226SJulian Elischer NG_FREE_M(event); 965878ed226SJulian Elischer 966878ed226SJulian Elischer return (error); 967878ed226SJulian Elischer } /* mode_change */ 968878ed226SJulian Elischer 969878ed226SJulian Elischer /* Data buffer overflow event */ 970878ed226SJulian Elischer static int 971878ed226SJulian Elischer data_buffer_overflow(ng_hci_unit_p unit, struct mbuf *event) 972878ed226SJulian Elischer { 973878ed226SJulian Elischer NG_HCI_ALERT( 974878ed226SJulian Elischer "%s: %s - %s data buffer overflow\n", 975878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 976878ed226SJulian Elischer (*mtod(event, u_int8_t *) == NG_HCI_LINK_ACL)? "ACL" : "SCO"); 977878ed226SJulian Elischer 978878ed226SJulian Elischer NG_FREE_M(event); 979878ed226SJulian Elischer 980878ed226SJulian Elischer return (0); 981878ed226SJulian Elischer } /* data_buffer_overflow */ 982878ed226SJulian Elischer 983878ed226SJulian Elischer /* Read clock offset complete event */ 984878ed226SJulian Elischer static int 985878ed226SJulian Elischer read_clock_offset_compl(ng_hci_unit_p unit, struct mbuf *event) 986878ed226SJulian Elischer { 987878ed226SJulian Elischer ng_hci_read_clock_offset_compl_ep *ep = NULL; 988878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 989878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 990878ed226SJulian Elischer int error = 0; 991878ed226SJulian Elischer 992878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 993878ed226SJulian Elischer if (event == NULL) 994878ed226SJulian Elischer return (ENOBUFS); 995878ed226SJulian Elischer 996878ed226SJulian Elischer ep = mtod(event, ng_hci_read_clock_offset_compl_ep *); 997878ed226SJulian Elischer 998878ed226SJulian Elischer if (ep->status == 0) { 999878ed226SJulian Elischer u_int16_t h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1000878ed226SJulian Elischer 1001878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1002878ed226SJulian Elischer if (con == NULL) { 1003878ed226SJulian Elischer NG_HCI_ALERT( 1004878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1005878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1006878ed226SJulian Elischer error = ENOENT; 1007878ed226SJulian Elischer goto out; 1008878ed226SJulian Elischer } 1009878ed226SJulian Elischer 1010878ed226SJulian Elischer /* Update cache entry */ 1011878ed226SJulian Elischer n = ng_hci_get_neighbor(unit, &con->bdaddr); 1012878ed226SJulian Elischer if (n == NULL) { 1013878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1014878ed226SJulian Elischer if (n == NULL) { 1015878ed226SJulian Elischer error = ENOMEM; 1016878ed226SJulian Elischer goto out; 1017878ed226SJulian Elischer } 1018878ed226SJulian Elischer 1019878ed226SJulian Elischer bcopy(&con->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1020878ed226SJulian Elischer } else 1021878ed226SJulian Elischer getmicrotime(&n->updated); 1022878ed226SJulian Elischer 1023878ed226SJulian Elischer n->clock_offset = le16toh(ep->clock_offset); 1024878ed226SJulian Elischer } else 1025878ed226SJulian Elischer NG_HCI_ERR( 1026878ed226SJulian Elischer "%s: %s - failed to Read Remote Clock Offset, status=%d\n", 1027878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), ep->status); 1028878ed226SJulian Elischer out: 1029878ed226SJulian Elischer NG_FREE_M(event); 1030878ed226SJulian Elischer 1031878ed226SJulian Elischer return (error); 1032878ed226SJulian Elischer } /* read_clock_offset_compl */ 1033878ed226SJulian Elischer 1034878ed226SJulian Elischer /* QoS violation event */ 1035878ed226SJulian Elischer static int 1036878ed226SJulian Elischer qos_violation(ng_hci_unit_p unit, struct mbuf *event) 1037878ed226SJulian Elischer { 1038878ed226SJulian Elischer ng_hci_qos_violation_ep *ep = NULL; 1039878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 1040878ed226SJulian Elischer u_int16_t h; 1041878ed226SJulian Elischer int error = 0; 1042878ed226SJulian Elischer 1043878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1044878ed226SJulian Elischer if (event == NULL) 1045878ed226SJulian Elischer return (ENOBUFS); 1046878ed226SJulian Elischer 1047878ed226SJulian Elischer ep = mtod(event, ng_hci_qos_violation_ep *); 1048878ed226SJulian Elischer 1049878ed226SJulian Elischer /* Check if we have this connection handle */ 1050878ed226SJulian Elischer h = NG_HCI_CON_HANDLE(le16toh(ep->con_handle)); 1051878ed226SJulian Elischer con = ng_hci_con_by_handle(unit, h); 1052878ed226SJulian Elischer if (con == NULL) { 1053878ed226SJulian Elischer NG_HCI_ALERT( 1054878ed226SJulian Elischer "%s: %s - invalid connection handle=%d\n", 1055878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), h); 1056878ed226SJulian Elischer error = ENOENT; 1057878ed226SJulian Elischer } else if (con->link_type != NG_HCI_LINK_ACL) { 1058878ed226SJulian Elischer NG_HCI_ALERT( 1059878ed226SJulian Elischer "%s: %s - invalid link type=%d\n", 1060878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->link_type); 1061878ed226SJulian Elischer error = EINVAL; 1062878ed226SJulian Elischer } else if (con->state != NG_HCI_CON_OPEN) { 1063878ed226SJulian Elischer NG_HCI_ALERT( 1064878ed226SJulian Elischer "%s: %s - invalid connection state=%d, handle=%d\n", 1065878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), con->state, h); 1066878ed226SJulian Elischer error = EINVAL; 1067878ed226SJulian Elischer } else /* Notify upper layer */ 1068878ed226SJulian Elischer error = ng_hci_lp_qos_ind(con); 1069878ed226SJulian Elischer 1070878ed226SJulian Elischer NG_FREE_M(event); 1071878ed226SJulian Elischer 1072878ed226SJulian Elischer return (error); 1073878ed226SJulian Elischer } /* qos_violation */ 1074878ed226SJulian Elischer 1075878ed226SJulian Elischer /* Page scan mode change event */ 1076878ed226SJulian Elischer static int 1077878ed226SJulian Elischer page_scan_mode_change(ng_hci_unit_p unit, struct mbuf *event) 1078878ed226SJulian Elischer { 1079878ed226SJulian Elischer ng_hci_page_scan_mode_change_ep *ep = NULL; 1080878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 1081878ed226SJulian Elischer int error = 0; 1082878ed226SJulian Elischer 1083878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1084878ed226SJulian Elischer if (event == NULL) 1085878ed226SJulian Elischer return (ENOBUFS); 1086878ed226SJulian Elischer 1087878ed226SJulian Elischer ep = mtod(event, ng_hci_page_scan_mode_change_ep *); 1088878ed226SJulian Elischer 1089878ed226SJulian Elischer /* Update cache entry */ 1090878ed226SJulian Elischer n = ng_hci_get_neighbor(unit, &ep->bdaddr); 1091878ed226SJulian Elischer if (n == NULL) { 1092878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1093878ed226SJulian Elischer if (n == NULL) { 1094878ed226SJulian Elischer error = ENOMEM; 1095878ed226SJulian Elischer goto out; 1096878ed226SJulian Elischer } 1097878ed226SJulian Elischer 1098878ed226SJulian Elischer bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1099878ed226SJulian Elischer } else 1100878ed226SJulian Elischer getmicrotime(&n->updated); 1101878ed226SJulian Elischer 1102878ed226SJulian Elischer n->page_scan_mode = ep->page_scan_mode; 1103878ed226SJulian Elischer out: 1104878ed226SJulian Elischer NG_FREE_M(event); 1105878ed226SJulian Elischer 1106878ed226SJulian Elischer return (error); 1107878ed226SJulian Elischer } /* page_scan_mode_change */ 1108878ed226SJulian Elischer 1109878ed226SJulian Elischer /* Page scan repetition mode change event */ 1110878ed226SJulian Elischer static int 1111878ed226SJulian Elischer page_scan_rep_mode_change(ng_hci_unit_p unit, struct mbuf *event) 1112878ed226SJulian Elischer { 1113878ed226SJulian Elischer ng_hci_page_scan_rep_mode_change_ep *ep = NULL; 1114878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 1115878ed226SJulian Elischer int error = 0; 1116878ed226SJulian Elischer 1117878ed226SJulian Elischer NG_HCI_M_PULLUP(event, sizeof(*ep)); 1118878ed226SJulian Elischer if (event == NULL) 1119878ed226SJulian Elischer return (ENOBUFS); 1120878ed226SJulian Elischer 1121878ed226SJulian Elischer ep = mtod(event, ng_hci_page_scan_rep_mode_change_ep *); 1122878ed226SJulian Elischer 1123878ed226SJulian Elischer /* Update cache entry */ 1124878ed226SJulian Elischer n = ng_hci_get_neighbor(unit, &ep->bdaddr); 1125878ed226SJulian Elischer if (n == NULL) { 1126878ed226SJulian Elischer n = ng_hci_new_neighbor(unit); 1127878ed226SJulian Elischer if (n == NULL) { 1128878ed226SJulian Elischer error = ENOMEM; 1129878ed226SJulian Elischer goto out; 1130878ed226SJulian Elischer } 1131878ed226SJulian Elischer 1132878ed226SJulian Elischer bcopy(&ep->bdaddr, &n->bdaddr, sizeof(n->bdaddr)); 1133878ed226SJulian Elischer } else 1134878ed226SJulian Elischer getmicrotime(&n->updated); 1135878ed226SJulian Elischer 1136878ed226SJulian Elischer n->page_scan_rep_mode = ep->page_scan_rep_mode; 1137878ed226SJulian Elischer out: 1138878ed226SJulian Elischer NG_FREE_M(event); 1139878ed226SJulian Elischer 1140878ed226SJulian Elischer return (error); 1141878ed226SJulian Elischer } /* page_scan_rep_mode_change */ 1142878ed226SJulian Elischer 1143