1878ed226SJulian Elischer /* 2878ed226SJulian Elischer * ng_hci_misc.c 3c398230bSWarner Losh */ 4c398230bSWarner Losh 5c398230bSWarner Losh /*- 6*fe267a55SPedro F. Giffuni * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 7*fe267a55SPedro 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_misc.c,v 1.5 2003/09/08 18:57:51 max Exp $ 33878ed226SJulian Elischer * $FreeBSD$ 34878ed226SJulian Elischer */ 35878ed226SJulian Elischer 36878ed226SJulian Elischer #include <sys/param.h> 37878ed226SJulian Elischer #include <sys/systm.h> 38878ed226SJulian Elischer #include <sys/kernel.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 ** Utility routines 55878ed226SJulian Elischer ****************************************************************************** 56878ed226SJulian Elischer ******************************************************************************/ 57878ed226SJulian Elischer 58878ed226SJulian Elischer /* 59878ed226SJulian Elischer * Give packet to RAW hook 60878ed226SJulian Elischer * Assumes input mbuf is read only. 61878ed226SJulian Elischer */ 62878ed226SJulian Elischer 63878ed226SJulian Elischer void 64878ed226SJulian Elischer ng_hci_mtap(ng_hci_unit_p unit, struct mbuf *m0) 65878ed226SJulian Elischer { 66878ed226SJulian Elischer struct mbuf *m = NULL; 67878ed226SJulian Elischer int error = 0; 68878ed226SJulian Elischer 69878ed226SJulian Elischer if (unit->raw != NULL && NG_HOOK_IS_VALID(unit->raw)) { 70eb1b1807SGleb Smirnoff m = m_dup(m0, M_NOWAIT); 71878ed226SJulian Elischer if (m != NULL) 72878ed226SJulian Elischer NG_SEND_DATA_ONLY(error, unit->raw, m); 73878ed226SJulian Elischer 74878ed226SJulian Elischer if (error != 0) 75878ed226SJulian Elischer NG_HCI_INFO( 76878ed226SJulian Elischer "%s: %s - Could not forward packet, error=%d\n", 77878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), error); 78878ed226SJulian Elischer } 79878ed226SJulian Elischer } /* ng_hci_mtap */ 80878ed226SJulian Elischer 81878ed226SJulian Elischer /* 82878ed226SJulian Elischer * Send notification to the upper layer's 83878ed226SJulian Elischer */ 84878ed226SJulian Elischer 85878ed226SJulian Elischer void 86878ed226SJulian Elischer ng_hci_node_is_up(node_p node, hook_p hook, void *arg1, int arg2) 87878ed226SJulian Elischer { 88878ed226SJulian Elischer ng_hci_unit_p unit = NULL; 89878ed226SJulian Elischer struct ng_mesg *msg = NULL; 90878ed226SJulian Elischer ng_hci_node_up_ep *ep = NULL; 91878ed226SJulian Elischer int error; 92878ed226SJulian Elischer 93878ed226SJulian Elischer if (node == NULL || NG_NODE_NOT_VALID(node) || 94878ed226SJulian Elischer hook == NULL || NG_HOOK_NOT_VALID(hook)) 95878ed226SJulian Elischer return; 96878ed226SJulian Elischer 97878ed226SJulian Elischer unit = (ng_hci_unit_p) NG_NODE_PRIVATE(node); 98878ed226SJulian Elischer if ((unit->state & NG_HCI_UNIT_READY) != NG_HCI_UNIT_READY) 99878ed226SJulian Elischer return; 100878ed226SJulian Elischer 101878ed226SJulian Elischer if (hook != unit->acl && hook != unit->sco) 102878ed226SJulian Elischer return; 103878ed226SJulian Elischer 104878ed226SJulian Elischer NG_MKMESSAGE(msg,NGM_HCI_COOKIE,NGM_HCI_NODE_UP,sizeof(*ep),M_NOWAIT); 105878ed226SJulian Elischer if (msg != NULL) { 106878ed226SJulian Elischer ep = (ng_hci_node_up_ep *)(msg->data); 107878ed226SJulian Elischer 108878ed226SJulian Elischer if (hook == unit->acl) { 109878ed226SJulian Elischer NG_HCI_BUFF_ACL_SIZE(unit->buffer, ep->pkt_size); 110878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, ep->num_pkts); 111878ed226SJulian Elischer } else { 112878ed226SJulian Elischer NG_HCI_BUFF_SCO_SIZE(unit->buffer, ep->pkt_size); 113878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, ep->num_pkts); 114878ed226SJulian Elischer } 115878ed226SJulian Elischer 116878ed226SJulian Elischer bcopy(&unit->bdaddr, &ep->bdaddr, sizeof(ep->bdaddr)); 117878ed226SJulian Elischer 1184ae439a3SMaksim Yevmenkin NG_SEND_MSG_HOOK(error, node, msg, hook, 0); 119878ed226SJulian Elischer } else 120878ed226SJulian Elischer error = ENOMEM; 121878ed226SJulian Elischer 122878ed226SJulian Elischer if (error != 0) 123878ed226SJulian Elischer NG_HCI_INFO( 124878ed226SJulian Elischer "%s: %s - failed to send NODE_UP message to hook \"%s\", error=%d\n", 125878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 126878ed226SJulian Elischer NG_HOOK_NAME(hook), error); 127878ed226SJulian Elischer } /* ng_hci_node_is_up */ 128878ed226SJulian Elischer 129878ed226SJulian Elischer /* 130878ed226SJulian Elischer * Clean unit (helper) 131878ed226SJulian Elischer */ 132878ed226SJulian Elischer 133878ed226SJulian Elischer void 134878ed226SJulian Elischer ng_hci_unit_clean(ng_hci_unit_p unit, int reason) 135878ed226SJulian Elischer { 136878ed226SJulian Elischer int size; 137878ed226SJulian Elischer 138878ed226SJulian Elischer /* Drain command queue */ 139878ed226SJulian Elischer if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) 140878ed226SJulian Elischer ng_hci_command_untimeout(unit); 141878ed226SJulian Elischer 142878ed226SJulian Elischer NG_BT_MBUFQ_DRAIN(&unit->cmdq); 143878ed226SJulian Elischer NG_HCI_BUFF_CMD_SET(unit->buffer, 1); 144878ed226SJulian Elischer 145878ed226SJulian Elischer /* Clean up connection list */ 146878ed226SJulian Elischer while (!LIST_EMPTY(&unit->con_list)) { 147878ed226SJulian Elischer ng_hci_unit_con_p con = LIST_FIRST(&unit->con_list); 148878ed226SJulian Elischer 1490986ab12SMaksim Yevmenkin /* Remove all timeouts (if any) */ 1500986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 1510986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 1520986ab12SMaksim Yevmenkin 153878ed226SJulian Elischer /* 154878ed226SJulian Elischer * Notify upper layer protocol and destroy connection 155878ed226SJulian Elischer * descriptor. Do not really care about the result. 156878ed226SJulian Elischer */ 157878ed226SJulian Elischer 158878ed226SJulian Elischer ng_hci_lp_discon_ind(con, reason); 159878ed226SJulian Elischer ng_hci_free_con(con); 160878ed226SJulian Elischer } 161878ed226SJulian Elischer 162878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, size); 163878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(unit->buffer, size); 164878ed226SJulian Elischer 165878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, size); 166878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(unit->buffer, size); 167878ed226SJulian Elischer 168878ed226SJulian Elischer /* Clean up neighbors list */ 169878ed226SJulian Elischer ng_hci_flush_neighbor_cache(unit); 170878ed226SJulian Elischer } /* ng_hci_unit_clean */ 171878ed226SJulian Elischer 172878ed226SJulian Elischer /* 173878ed226SJulian Elischer * Allocate and link new unit neighbor cache entry 174878ed226SJulian Elischer */ 175878ed226SJulian Elischer 176878ed226SJulian Elischer ng_hci_neighbor_p 177878ed226SJulian Elischer ng_hci_new_neighbor(ng_hci_unit_p unit) 178878ed226SJulian Elischer { 179878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 180878ed226SJulian Elischer 1811ede983cSDag-Erling Smørgrav n = malloc(sizeof(*n), M_NETGRAPH_HCI, 182878ed226SJulian Elischer M_NOWAIT | M_ZERO); 183878ed226SJulian Elischer if (n != NULL) { 184878ed226SJulian Elischer getmicrotime(&n->updated); 185878ed226SJulian Elischer LIST_INSERT_HEAD(&unit->neighbors, n, next); 186878ed226SJulian Elischer } 187878ed226SJulian Elischer 188878ed226SJulian Elischer return (n); 189878ed226SJulian Elischer } /* ng_hci_new_neighbor */ 190878ed226SJulian Elischer 191878ed226SJulian Elischer /* 192878ed226SJulian Elischer * Free unit neighbor cache entry 193878ed226SJulian Elischer */ 194878ed226SJulian Elischer 195878ed226SJulian Elischer void 196878ed226SJulian Elischer ng_hci_free_neighbor(ng_hci_neighbor_p n) 197878ed226SJulian Elischer { 198878ed226SJulian Elischer LIST_REMOVE(n, next); 199878ed226SJulian Elischer bzero(n, sizeof(*n)); 2001ede983cSDag-Erling Smørgrav free(n, M_NETGRAPH_HCI); 201878ed226SJulian Elischer } /* ng_hci_free_neighbor */ 202878ed226SJulian Elischer 203878ed226SJulian Elischer /* 204878ed226SJulian Elischer * Flush neighbor cache 205878ed226SJulian Elischer */ 206878ed226SJulian Elischer 207878ed226SJulian Elischer void 208878ed226SJulian Elischer ng_hci_flush_neighbor_cache(ng_hci_unit_p unit) 209878ed226SJulian Elischer { 210878ed226SJulian Elischer while (!LIST_EMPTY(&unit->neighbors)) 211878ed226SJulian Elischer ng_hci_free_neighbor(LIST_FIRST(&unit->neighbors)); 212878ed226SJulian Elischer } /* ng_hci_flush_neighbor_cache */ 213878ed226SJulian Elischer 214878ed226SJulian Elischer /* 215878ed226SJulian Elischer * Lookup unit in neighbor cache 216878ed226SJulian Elischer */ 217878ed226SJulian Elischer 218878ed226SJulian Elischer ng_hci_neighbor_p 219fbc48c2bSTakanori Watanabe ng_hci_get_neighbor(ng_hci_unit_p unit, bdaddr_p bdaddr,int link_type) 220878ed226SJulian Elischer { 221878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 222878ed226SJulian Elischer 223878ed226SJulian Elischer for (n = LIST_FIRST(&unit->neighbors); n != NULL; ) { 224878ed226SJulian Elischer ng_hci_neighbor_p nn = LIST_NEXT(n, next); 225878ed226SJulian Elischer 226878ed226SJulian Elischer if (!ng_hci_neighbor_stale(n)) { 227fbc48c2bSTakanori Watanabe if (n->addrtype == link_type && 228fbc48c2bSTakanori Watanabe bcmp(&n->bdaddr, bdaddr, sizeof(*bdaddr)) == 0) 229878ed226SJulian Elischer break; 230878ed226SJulian Elischer } else 231878ed226SJulian Elischer ng_hci_free_neighbor(n); /* remove old entry */ 232878ed226SJulian Elischer 233878ed226SJulian Elischer n = nn; 234878ed226SJulian Elischer } 235878ed226SJulian Elischer 236878ed226SJulian Elischer return (n); 237878ed226SJulian Elischer } /* ng_hci_get_neighbor */ 238878ed226SJulian Elischer 239878ed226SJulian Elischer /* 240878ed226SJulian Elischer * Check if neighbor entry is stale 241878ed226SJulian Elischer */ 242878ed226SJulian Elischer 243878ed226SJulian Elischer int 244878ed226SJulian Elischer ng_hci_neighbor_stale(ng_hci_neighbor_p n) 245878ed226SJulian Elischer { 246878ed226SJulian Elischer struct timeval now; 247878ed226SJulian Elischer 248878ed226SJulian Elischer getmicrotime(&now); 249878ed226SJulian Elischer 250878ed226SJulian Elischer return (now.tv_sec - n->updated.tv_sec > bluetooth_hci_max_neighbor_age()); 251878ed226SJulian Elischer } /* ng_hci_neighbor_stale */ 252878ed226SJulian Elischer 253878ed226SJulian Elischer /* 254878ed226SJulian Elischer * Allocate and link new connection descriptor 255878ed226SJulian Elischer */ 256878ed226SJulian Elischer 257878ed226SJulian Elischer ng_hci_unit_con_p 258878ed226SJulian Elischer ng_hci_new_con(ng_hci_unit_p unit, int link_type) 259878ed226SJulian Elischer { 260878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 261878ed226SJulian Elischer int num_pkts; 2620986ab12SMaksim Yevmenkin static int fake_con_handle = 0x0f00; 263878ed226SJulian Elischer 2641ede983cSDag-Erling Smørgrav con = malloc(sizeof(*con), M_NETGRAPH_HCI, 265878ed226SJulian Elischer M_NOWAIT | M_ZERO); 266878ed226SJulian Elischer if (con != NULL) { 267878ed226SJulian Elischer con->unit = unit; 268878ed226SJulian Elischer con->state = NG_HCI_CON_CLOSED; 2690986ab12SMaksim Yevmenkin 2700986ab12SMaksim Yevmenkin /* 2710986ab12SMaksim Yevmenkin * XXX 2720986ab12SMaksim Yevmenkin * 2730986ab12SMaksim Yevmenkin * Assign fake connection handle to the connection descriptor. 2740986ab12SMaksim Yevmenkin * Bluetooth specification marks 0x0f00 - 0x0fff connection 2750986ab12SMaksim Yevmenkin * handles as reserved. We need this fake connection handles 2760986ab12SMaksim Yevmenkin * for timeouts. Connection handle will be passed as argument 2770986ab12SMaksim Yevmenkin * to timeout so when timeout happens we can find the right 2780986ab12SMaksim Yevmenkin * connection descriptor. We can not pass pointers, because 2790986ab12SMaksim Yevmenkin * timeouts are external (to Netgraph) events and there might 2800986ab12SMaksim Yevmenkin * be a race when node/hook goes down and timeout event already 2810986ab12SMaksim Yevmenkin * went into node's queue 2820986ab12SMaksim Yevmenkin */ 2830986ab12SMaksim Yevmenkin 2840986ab12SMaksim Yevmenkin con->con_handle = fake_con_handle ++; 2850986ab12SMaksim Yevmenkin if (fake_con_handle > 0x0fff) 2860986ab12SMaksim Yevmenkin fake_con_handle = 0x0f00; 2870986ab12SMaksim Yevmenkin 288878ed226SJulian Elischer con->link_type = link_type; 289878ed226SJulian Elischer 290fbc48c2bSTakanori Watanabe if (con->link_type != NG_HCI_LINK_SCO) 291878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, num_pkts); 292878ed226SJulian Elischer else 293878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, num_pkts); 294878ed226SJulian Elischer 295878ed226SJulian Elischer NG_BT_ITEMQ_INIT(&con->conq, num_pkts); 296878ed226SJulian Elischer 297b81d7730SMaksim Yevmenkin ng_callout_init(&con->con_timo); 298878ed226SJulian Elischer 299878ed226SJulian Elischer LIST_INSERT_HEAD(&unit->con_list, con, next); 300878ed226SJulian Elischer } 301878ed226SJulian Elischer 302878ed226SJulian Elischer return (con); 303878ed226SJulian Elischer } /* ng_hci_new_con */ 304878ed226SJulian Elischer 305878ed226SJulian Elischer /* 306878ed226SJulian Elischer * Free connection descriptor 307878ed226SJulian Elischer */ 308878ed226SJulian Elischer 309878ed226SJulian Elischer void 310878ed226SJulian Elischer ng_hci_free_con(ng_hci_unit_con_p con) 311878ed226SJulian Elischer { 312878ed226SJulian Elischer LIST_REMOVE(con, next); 313878ed226SJulian Elischer 314878ed226SJulian Elischer /* 315878ed226SJulian Elischer * If we have pending packets then assume that Host Controller has 316878ed226SJulian Elischer * flushed these packets and we can free them too 317878ed226SJulian Elischer */ 318878ed226SJulian Elischer 319fbc48c2bSTakanori Watanabe if (con->link_type != NG_HCI_LINK_SCO) 320878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(con->unit->buffer, con->pending); 321878ed226SJulian Elischer else 322878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(con->unit->buffer, con->pending); 323878ed226SJulian Elischer 324878ed226SJulian Elischer NG_BT_ITEMQ_DESTROY(&con->conq); 325878ed226SJulian Elischer 326878ed226SJulian Elischer bzero(con, sizeof(*con)); 3271ede983cSDag-Erling Smørgrav free(con, M_NETGRAPH_HCI); 328878ed226SJulian Elischer } /* ng_hci_free_con */ 329878ed226SJulian Elischer 330878ed226SJulian Elischer /* 331878ed226SJulian Elischer * Lookup connection for given unit and connection handle. 332878ed226SJulian Elischer */ 333878ed226SJulian Elischer 334878ed226SJulian Elischer ng_hci_unit_con_p 335878ed226SJulian Elischer ng_hci_con_by_handle(ng_hci_unit_p unit, int con_handle) 336878ed226SJulian Elischer { 337878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 338878ed226SJulian Elischer 339878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 340878ed226SJulian Elischer if (con->con_handle == con_handle) 341878ed226SJulian Elischer break; 342878ed226SJulian Elischer 343878ed226SJulian Elischer return (con); 344878ed226SJulian Elischer } /* ng_hci_con_by_handle */ 345878ed226SJulian Elischer 346878ed226SJulian Elischer /* 347878ed226SJulian Elischer * Lookup connection for given unit, link type and remove unit address 348878ed226SJulian Elischer */ 349878ed226SJulian Elischer 350878ed226SJulian Elischer ng_hci_unit_con_p 351878ed226SJulian Elischer ng_hci_con_by_bdaddr(ng_hci_unit_p unit, bdaddr_p bdaddr, int link_type) 352878ed226SJulian Elischer { 353878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 354878ed226SJulian Elischer 355878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 356878ed226SJulian Elischer if (con->link_type == link_type && 357878ed226SJulian Elischer bcmp(&con->bdaddr, bdaddr, sizeof(bdaddr_t)) == 0) 358878ed226SJulian Elischer break; 359878ed226SJulian Elischer 360878ed226SJulian Elischer return (con); 361878ed226SJulian Elischer } /* ng_hci_con_by_bdaddr */ 362878ed226SJulian Elischer 363878ed226SJulian Elischer /* 364878ed226SJulian Elischer * Set HCI command timeout 365f9d9e1b4SGleb Smirnoff * XXX FIXME: check return code from ng_callout 366878ed226SJulian Elischer */ 367878ed226SJulian Elischer 3680986ab12SMaksim Yevmenkin int 369878ed226SJulian Elischer ng_hci_command_timeout(ng_hci_unit_p unit) 370878ed226SJulian Elischer { 3710986ab12SMaksim Yevmenkin if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) 3720986ab12SMaksim Yevmenkin panic( 3730986ab12SMaksim Yevmenkin "%s: %s - Duplicated command timeout!\n", __func__, NG_NODE_NAME(unit->node)); 3740986ab12SMaksim Yevmenkin 375878ed226SJulian Elischer unit->state |= NG_HCI_UNIT_COMMAND_PENDING; 376f9d9e1b4SGleb Smirnoff ng_callout(&unit->cmd_timo, unit->node, NULL, 3770986ab12SMaksim Yevmenkin bluetooth_hci_command_timeout(), 3780986ab12SMaksim Yevmenkin ng_hci_process_command_timeout, NULL, 0); 3790986ab12SMaksim Yevmenkin 3800986ab12SMaksim Yevmenkin return (0); 381878ed226SJulian Elischer } /* ng_hci_command_timeout */ 382878ed226SJulian Elischer 383878ed226SJulian Elischer /* 384878ed226SJulian Elischer * Unset HCI command timeout 385878ed226SJulian Elischer */ 386878ed226SJulian Elischer 3870986ab12SMaksim Yevmenkin int 388878ed226SJulian Elischer ng_hci_command_untimeout(ng_hci_unit_p unit) 389878ed226SJulian Elischer { 3900986ab12SMaksim Yevmenkin if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) 3910986ab12SMaksim Yevmenkin panic( 3920986ab12SMaksim Yevmenkin "%s: %s - No command timeout!\n", __func__, NG_NODE_NAME(unit->node)); 3930986ab12SMaksim Yevmenkin 394f9d9e1b4SGleb Smirnoff if (ng_uncallout(&unit->cmd_timo, unit->node) == 0) 3950986ab12SMaksim Yevmenkin return (ETIMEDOUT); 3960986ab12SMaksim Yevmenkin 397878ed226SJulian Elischer unit->state &= ~NG_HCI_UNIT_COMMAND_PENDING; 3980986ab12SMaksim Yevmenkin 3990986ab12SMaksim Yevmenkin return (0); 400878ed226SJulian Elischer } /* ng_hci_command_untimeout */ 401878ed226SJulian Elischer 402878ed226SJulian Elischer /* 403878ed226SJulian Elischer * Set HCI connection timeout 404f9d9e1b4SGleb Smirnoff * XXX FIXME: check return code from ng_callout 405878ed226SJulian Elischer */ 406878ed226SJulian Elischer 4070986ab12SMaksim Yevmenkin int 408878ed226SJulian Elischer ng_hci_con_timeout(ng_hci_unit_con_p con) 409878ed226SJulian Elischer { 4100986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 4110986ab12SMaksim Yevmenkin panic( 4120986ab12SMaksim Yevmenkin "%s: %s - Duplicated connection timeout!\n", 4130986ab12SMaksim Yevmenkin __func__, NG_NODE_NAME(con->unit->node)); 4140986ab12SMaksim Yevmenkin 415878ed226SJulian Elischer con->flags |= NG_HCI_CON_TIMEOUT_PENDING; 416f9d9e1b4SGleb Smirnoff ng_callout(&con->con_timo, con->unit->node, NULL, 4170986ab12SMaksim Yevmenkin bluetooth_hci_connect_timeout(), 4180986ab12SMaksim Yevmenkin ng_hci_process_con_timeout, NULL, 4190986ab12SMaksim Yevmenkin con->con_handle); 4200986ab12SMaksim Yevmenkin 4210986ab12SMaksim Yevmenkin return (0); 422878ed226SJulian Elischer } /* ng_hci_con_timeout */ 423878ed226SJulian Elischer 424878ed226SJulian Elischer /* 425878ed226SJulian Elischer * Unset HCI connection timeout 426878ed226SJulian Elischer */ 427878ed226SJulian Elischer 4280986ab12SMaksim Yevmenkin int 429878ed226SJulian Elischer ng_hci_con_untimeout(ng_hci_unit_con_p con) 430878ed226SJulian Elischer { 4310986ab12SMaksim Yevmenkin if (!(con->flags & NG_HCI_CON_TIMEOUT_PENDING)) 4320986ab12SMaksim Yevmenkin panic( 4330986ab12SMaksim Yevmenkin "%s: %s - No connection timeout!\n", __func__, NG_NODE_NAME(con->unit->node)); 4340986ab12SMaksim Yevmenkin 435f9d9e1b4SGleb Smirnoff if (ng_uncallout(&con->con_timo, con->unit->node) == 0) 4360986ab12SMaksim Yevmenkin return (ETIMEDOUT); 4370986ab12SMaksim Yevmenkin 438878ed226SJulian Elischer con->flags &= ~NG_HCI_CON_TIMEOUT_PENDING; 4390986ab12SMaksim Yevmenkin 4400986ab12SMaksim Yevmenkin return (0); 441878ed226SJulian Elischer } /* ng_hci_con_untimeout */ 442878ed226SJulian Elischer 443878ed226SJulian Elischer #if 0 444878ed226SJulian Elischer /* 445878ed226SJulian Elischer * Convert numeric error code/reason to a string 446878ed226SJulian Elischer */ 447878ed226SJulian Elischer 448878ed226SJulian Elischer char const * const 449878ed226SJulian Elischer ng_hci_str_error(u_int16_t code) 450878ed226SJulian Elischer { 451878ed226SJulian Elischer #define LAST_ERROR_CODE ((sizeof(s)/sizeof(s[0]))-1) 452878ed226SJulian Elischer static char const * const s[] = { 453878ed226SJulian Elischer /* 0x00 */ "No error", 454878ed226SJulian Elischer /* 0x01 */ "Unknown HCI command", 455878ed226SJulian Elischer /* 0x02 */ "No connection", 456878ed226SJulian Elischer /* 0x03 */ "Hardware failure", 457878ed226SJulian Elischer /* 0x04 */ "Page timeout", 458878ed226SJulian Elischer /* 0x05 */ "Authentication failure", 459878ed226SJulian Elischer /* 0x06 */ "Key missing", 460878ed226SJulian Elischer /* 0x07 */ "Memory full", 461878ed226SJulian Elischer /* 0x08 */ "Connection timeout", 462878ed226SJulian Elischer /* 0x09 */ "Max number of connections", 463878ed226SJulian Elischer /* 0x0a */ "Max number of SCO connections to a unit", 464878ed226SJulian Elischer /* 0x0b */ "ACL connection already exists", 465878ed226SJulian Elischer /* 0x0c */ "Command disallowed", 466878ed226SJulian Elischer /* 0x0d */ "Host rejected due to limited resources", 467878ed226SJulian Elischer /* 0x0e */ "Host rejected due to securiity reasons", 468878ed226SJulian Elischer /* 0x0f */ "Host rejected due to remote unit is a personal unit", 469878ed226SJulian Elischer /* 0x10 */ "Host timeout", 470878ed226SJulian Elischer /* 0x11 */ "Unsupported feature or parameter value", 471878ed226SJulian Elischer /* 0x12 */ "Invalid HCI command parameter", 472878ed226SJulian Elischer /* 0x13 */ "Other end terminated connection: User ended connection", 473878ed226SJulian Elischer /* 0x14 */ "Other end terminated connection: Low resources", 474878ed226SJulian Elischer /* 0x15 */ "Other end terminated connection: About to power off", 475878ed226SJulian Elischer /* 0x16 */ "Connection terminated by local host", 476878ed226SJulian Elischer /* 0x17 */ "Repeated attempts", 477878ed226SJulian Elischer /* 0x18 */ "Pairing not allowed", 478878ed226SJulian Elischer /* 0x19 */ "Unknown LMP PDU", 479878ed226SJulian Elischer /* 0x1a */ "Unsupported remote feature", 480878ed226SJulian Elischer /* 0x1b */ "SCO offset rejected", 481878ed226SJulian Elischer /* 0x1c */ "SCO interval rejected", 482878ed226SJulian Elischer /* 0x1d */ "SCO air mode rejected", 483878ed226SJulian Elischer /* 0x1e */ "Invalid LMP parameters", 484878ed226SJulian Elischer /* 0x1f */ "Unspecified error", 485878ed226SJulian Elischer /* 0x20 */ "Unsupported LMP parameter value", 486878ed226SJulian Elischer /* 0x21 */ "Role change not allowed", 487878ed226SJulian Elischer /* 0x22 */ "LMP response timeout", 488878ed226SJulian Elischer /* 0x23 */ "LMP error transaction collision", 489878ed226SJulian Elischer /* 0x24 */ "LMP PSU not allowed", 490878ed226SJulian Elischer /* 0x25 */ "Encryption mode not acceptable", 491878ed226SJulian Elischer /* 0x26 */ "Unit key used", 492878ed226SJulian Elischer /* 0x27 */ "QoS is not supported", 493878ed226SJulian Elischer /* 0x28 */ "Instant passed", 494878ed226SJulian Elischer /* 0x29 */ "Paring with unit key not supported", 495878ed226SJulian Elischer /* SHOULD ALWAYS BE LAST */ "Unknown error" 496878ed226SJulian Elischer }; 497878ed226SJulian Elischer 498878ed226SJulian Elischer return ((code >= LAST_ERROR_CODE)? s[LAST_ERROR_CODE] : s[code]); 499878ed226SJulian Elischer } /* ng_hci_str_error */ 500878ed226SJulian Elischer #endif 501878ed226SJulian Elischer 502