1878ed226SJulian Elischer /* 2878ed226SJulian Elischer * ng_hci_misc.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_misc.c,v 1.5 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/malloc.h> 36878ed226SJulian Elischer #include <sys/mbuf.h> 37878ed226SJulian Elischer #include <sys/queue.h> 38878ed226SJulian Elischer #include <netgraph/ng_message.h> 39878ed226SJulian Elischer #include <netgraph/netgraph.h> 40b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_bluetooth.h> 41b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/include/ng_hci.h> 42b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_var.h> 43b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_cmds.h> 44b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_evnt.h> 45b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_ulpi.h> 46b84b10f9SMaksim Yevmenkin #include <netgraph/bluetooth/hci/ng_hci_misc.h> 47878ed226SJulian Elischer 48878ed226SJulian Elischer /****************************************************************************** 49878ed226SJulian Elischer ****************************************************************************** 50878ed226SJulian Elischer ** Utility routines 51878ed226SJulian Elischer ****************************************************************************** 52878ed226SJulian Elischer ******************************************************************************/ 53878ed226SJulian Elischer 54878ed226SJulian Elischer /* 55878ed226SJulian Elischer * Give packet to RAW hook 56878ed226SJulian Elischer * Assumes input mbuf is read only. 57878ed226SJulian Elischer */ 58878ed226SJulian Elischer 59878ed226SJulian Elischer void 60878ed226SJulian Elischer ng_hci_mtap(ng_hci_unit_p unit, struct mbuf *m0) 61878ed226SJulian Elischer { 62878ed226SJulian Elischer struct mbuf *m = NULL; 63878ed226SJulian Elischer int error = 0; 64878ed226SJulian Elischer 65878ed226SJulian Elischer if (unit->raw != NULL && NG_HOOK_IS_VALID(unit->raw)) { 66a163d034SWarner Losh m = m_dup(m0, M_DONTWAIT); 67878ed226SJulian Elischer if (m != NULL) 68878ed226SJulian Elischer NG_SEND_DATA_ONLY(error, unit->raw, m); 69878ed226SJulian Elischer 70878ed226SJulian Elischer if (error != 0) 71878ed226SJulian Elischer NG_HCI_INFO( 72878ed226SJulian Elischer "%s: %s - Could not forward packet, error=%d\n", 73878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), error); 74878ed226SJulian Elischer } 75878ed226SJulian Elischer } /* ng_hci_mtap */ 76878ed226SJulian Elischer 77878ed226SJulian Elischer /* 78878ed226SJulian Elischer * Send notification to the upper layer's 79878ed226SJulian Elischer */ 80878ed226SJulian Elischer 81878ed226SJulian Elischer void 82878ed226SJulian Elischer ng_hci_node_is_up(node_p node, hook_p hook, void *arg1, int arg2) 83878ed226SJulian Elischer { 84878ed226SJulian Elischer ng_hci_unit_p unit = NULL; 85878ed226SJulian Elischer struct ng_mesg *msg = NULL; 86878ed226SJulian Elischer ng_hci_node_up_ep *ep = NULL; 87878ed226SJulian Elischer int error; 88878ed226SJulian Elischer 89878ed226SJulian Elischer if (node == NULL || NG_NODE_NOT_VALID(node) || 90878ed226SJulian Elischer hook == NULL || NG_HOOK_NOT_VALID(hook)) 91878ed226SJulian Elischer return; 92878ed226SJulian Elischer 93878ed226SJulian Elischer unit = (ng_hci_unit_p) NG_NODE_PRIVATE(node); 94878ed226SJulian Elischer if ((unit->state & NG_HCI_UNIT_READY) != NG_HCI_UNIT_READY) 95878ed226SJulian Elischer return; 96878ed226SJulian Elischer 97878ed226SJulian Elischer if (hook != unit->acl && hook != unit->sco) 98878ed226SJulian Elischer return; 99878ed226SJulian Elischer 100878ed226SJulian Elischer NG_MKMESSAGE(msg,NGM_HCI_COOKIE,NGM_HCI_NODE_UP,sizeof(*ep),M_NOWAIT); 101878ed226SJulian Elischer if (msg != NULL) { 102878ed226SJulian Elischer ep = (ng_hci_node_up_ep *)(msg->data); 103878ed226SJulian Elischer 104878ed226SJulian Elischer if (hook == unit->acl) { 105878ed226SJulian Elischer NG_HCI_BUFF_ACL_SIZE(unit->buffer, ep->pkt_size); 106878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, ep->num_pkts); 107878ed226SJulian Elischer } else { 108878ed226SJulian Elischer NG_HCI_BUFF_SCO_SIZE(unit->buffer, ep->pkt_size); 109878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, ep->num_pkts); 110878ed226SJulian Elischer } 111878ed226SJulian Elischer 112878ed226SJulian Elischer bcopy(&unit->bdaddr, &ep->bdaddr, sizeof(ep->bdaddr)); 113878ed226SJulian Elischer 1144ae439a3SMaksim Yevmenkin NG_SEND_MSG_HOOK(error, node, msg, hook, 0); 115878ed226SJulian Elischer } else 116878ed226SJulian Elischer error = ENOMEM; 117878ed226SJulian Elischer 118878ed226SJulian Elischer if (error != 0) 119878ed226SJulian Elischer NG_HCI_INFO( 120878ed226SJulian Elischer "%s: %s - failed to send NODE_UP message to hook \"%s\", error=%d\n", 121878ed226SJulian Elischer __func__, NG_NODE_NAME(unit->node), 122878ed226SJulian Elischer NG_HOOK_NAME(hook), error); 123878ed226SJulian Elischer } /* ng_hci_node_is_up */ 124878ed226SJulian Elischer 125878ed226SJulian Elischer /* 126878ed226SJulian Elischer * Clean unit (helper) 127878ed226SJulian Elischer */ 128878ed226SJulian Elischer 129878ed226SJulian Elischer void 130878ed226SJulian Elischer ng_hci_unit_clean(ng_hci_unit_p unit, int reason) 131878ed226SJulian Elischer { 132878ed226SJulian Elischer int size; 133878ed226SJulian Elischer 134878ed226SJulian Elischer /* Drain command queue */ 135878ed226SJulian Elischer if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) 136878ed226SJulian Elischer ng_hci_command_untimeout(unit); 137878ed226SJulian Elischer 138878ed226SJulian Elischer NG_BT_MBUFQ_DRAIN(&unit->cmdq); 139878ed226SJulian Elischer NG_HCI_BUFF_CMD_SET(unit->buffer, 1); 140878ed226SJulian Elischer 141878ed226SJulian Elischer /* Clean up connection list */ 142878ed226SJulian Elischer while (!LIST_EMPTY(&unit->con_list)) { 143878ed226SJulian Elischer ng_hci_unit_con_p con = LIST_FIRST(&unit->con_list); 144878ed226SJulian Elischer 1450986ab12SMaksim Yevmenkin /* Remove all timeouts (if any) */ 1460986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 1470986ab12SMaksim Yevmenkin ng_hci_con_untimeout(con); 1480986ab12SMaksim Yevmenkin 149878ed226SJulian Elischer /* 150878ed226SJulian Elischer * Notify upper layer protocol and destroy connection 151878ed226SJulian Elischer * descriptor. Do not really care about the result. 152878ed226SJulian Elischer */ 153878ed226SJulian Elischer 154878ed226SJulian Elischer ng_hci_lp_discon_ind(con, reason); 155878ed226SJulian Elischer ng_hci_free_con(con); 156878ed226SJulian Elischer } 157878ed226SJulian Elischer 158878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, size); 159878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(unit->buffer, size); 160878ed226SJulian Elischer 161878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, size); 162878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(unit->buffer, size); 163878ed226SJulian Elischer 164878ed226SJulian Elischer /* Clean up neighbors list */ 165878ed226SJulian Elischer ng_hci_flush_neighbor_cache(unit); 166878ed226SJulian Elischer } /* ng_hci_unit_clean */ 167878ed226SJulian Elischer 168878ed226SJulian Elischer /* 169878ed226SJulian Elischer * Allocate and link new unit neighbor cache entry 170878ed226SJulian Elischer */ 171878ed226SJulian Elischer 172878ed226SJulian Elischer ng_hci_neighbor_p 173878ed226SJulian Elischer ng_hci_new_neighbor(ng_hci_unit_p unit) 174878ed226SJulian Elischer { 175878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 176878ed226SJulian Elischer 177878ed226SJulian Elischer MALLOC(n, ng_hci_neighbor_p, sizeof(*n), M_NETGRAPH_HCI, 178878ed226SJulian Elischer M_NOWAIT | M_ZERO); 179878ed226SJulian Elischer if (n != NULL) { 180878ed226SJulian Elischer getmicrotime(&n->updated); 181878ed226SJulian Elischer LIST_INSERT_HEAD(&unit->neighbors, n, next); 182878ed226SJulian Elischer } 183878ed226SJulian Elischer 184878ed226SJulian Elischer return (n); 185878ed226SJulian Elischer } /* ng_hci_new_neighbor */ 186878ed226SJulian Elischer 187878ed226SJulian Elischer /* 188878ed226SJulian Elischer * Free unit neighbor cache entry 189878ed226SJulian Elischer */ 190878ed226SJulian Elischer 191878ed226SJulian Elischer void 192878ed226SJulian Elischer ng_hci_free_neighbor(ng_hci_neighbor_p n) 193878ed226SJulian Elischer { 194878ed226SJulian Elischer LIST_REMOVE(n, next); 195878ed226SJulian Elischer bzero(n, sizeof(*n)); 196878ed226SJulian Elischer FREE(n, M_NETGRAPH_HCI); 197878ed226SJulian Elischer } /* ng_hci_free_neighbor */ 198878ed226SJulian Elischer 199878ed226SJulian Elischer /* 200878ed226SJulian Elischer * Flush neighbor cache 201878ed226SJulian Elischer */ 202878ed226SJulian Elischer 203878ed226SJulian Elischer void 204878ed226SJulian Elischer ng_hci_flush_neighbor_cache(ng_hci_unit_p unit) 205878ed226SJulian Elischer { 206878ed226SJulian Elischer while (!LIST_EMPTY(&unit->neighbors)) 207878ed226SJulian Elischer ng_hci_free_neighbor(LIST_FIRST(&unit->neighbors)); 208878ed226SJulian Elischer } /* ng_hci_flush_neighbor_cache */ 209878ed226SJulian Elischer 210878ed226SJulian Elischer /* 211878ed226SJulian Elischer * Lookup unit in neighbor cache 212878ed226SJulian Elischer */ 213878ed226SJulian Elischer 214878ed226SJulian Elischer ng_hci_neighbor_p 215878ed226SJulian Elischer ng_hci_get_neighbor(ng_hci_unit_p unit, bdaddr_p bdaddr) 216878ed226SJulian Elischer { 217878ed226SJulian Elischer ng_hci_neighbor_p n = NULL; 218878ed226SJulian Elischer 219878ed226SJulian Elischer for (n = LIST_FIRST(&unit->neighbors); n != NULL; ) { 220878ed226SJulian Elischer ng_hci_neighbor_p nn = LIST_NEXT(n, next); 221878ed226SJulian Elischer 222878ed226SJulian Elischer if (!ng_hci_neighbor_stale(n)) { 223878ed226SJulian Elischer if (bcmp(&n->bdaddr, bdaddr, sizeof(*bdaddr)) == 0) 224878ed226SJulian Elischer break; 225878ed226SJulian Elischer } else 226878ed226SJulian Elischer ng_hci_free_neighbor(n); /* remove old entry */ 227878ed226SJulian Elischer 228878ed226SJulian Elischer n = nn; 229878ed226SJulian Elischer } 230878ed226SJulian Elischer 231878ed226SJulian Elischer return (n); 232878ed226SJulian Elischer } /* ng_hci_get_neighbor */ 233878ed226SJulian Elischer 234878ed226SJulian Elischer /* 235878ed226SJulian Elischer * Check if neighbor entry is stale 236878ed226SJulian Elischer */ 237878ed226SJulian Elischer 238878ed226SJulian Elischer int 239878ed226SJulian Elischer ng_hci_neighbor_stale(ng_hci_neighbor_p n) 240878ed226SJulian Elischer { 241878ed226SJulian Elischer struct timeval now; 242878ed226SJulian Elischer 243878ed226SJulian Elischer getmicrotime(&now); 244878ed226SJulian Elischer 245878ed226SJulian Elischer return (now.tv_sec - n->updated.tv_sec > bluetooth_hci_max_neighbor_age()); 246878ed226SJulian Elischer } /* ng_hci_neighbor_stale */ 247878ed226SJulian Elischer 248878ed226SJulian Elischer /* 249878ed226SJulian Elischer * Allocate and link new connection descriptor 250878ed226SJulian Elischer */ 251878ed226SJulian Elischer 252878ed226SJulian Elischer ng_hci_unit_con_p 253878ed226SJulian Elischer ng_hci_new_con(ng_hci_unit_p unit, int link_type) 254878ed226SJulian Elischer { 255878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 256878ed226SJulian Elischer int num_pkts; 2570986ab12SMaksim Yevmenkin static int fake_con_handle = 0x0f00; 258878ed226SJulian Elischer 259878ed226SJulian Elischer MALLOC(con, ng_hci_unit_con_p, sizeof(*con), M_NETGRAPH_HCI, 260878ed226SJulian Elischer M_NOWAIT | M_ZERO); 261878ed226SJulian Elischer if (con != NULL) { 262878ed226SJulian Elischer con->unit = unit; 263878ed226SJulian Elischer con->state = NG_HCI_CON_CLOSED; 2640986ab12SMaksim Yevmenkin 2650986ab12SMaksim Yevmenkin /* 2660986ab12SMaksim Yevmenkin * XXX 2670986ab12SMaksim Yevmenkin * 2680986ab12SMaksim Yevmenkin * Assign fake connection handle to the connection descriptor. 2690986ab12SMaksim Yevmenkin * Bluetooth specification marks 0x0f00 - 0x0fff connection 2700986ab12SMaksim Yevmenkin * handles as reserved. We need this fake connection handles 2710986ab12SMaksim Yevmenkin * for timeouts. Connection handle will be passed as argument 2720986ab12SMaksim Yevmenkin * to timeout so when timeout happens we can find the right 2730986ab12SMaksim Yevmenkin * connection descriptor. We can not pass pointers, because 2740986ab12SMaksim Yevmenkin * timeouts are external (to Netgraph) events and there might 2750986ab12SMaksim Yevmenkin * be a race when node/hook goes down and timeout event already 2760986ab12SMaksim Yevmenkin * went into node's queue 2770986ab12SMaksim Yevmenkin */ 2780986ab12SMaksim Yevmenkin 2790986ab12SMaksim Yevmenkin con->con_handle = fake_con_handle ++; 2800986ab12SMaksim Yevmenkin if (fake_con_handle > 0x0fff) 2810986ab12SMaksim Yevmenkin fake_con_handle = 0x0f00; 2820986ab12SMaksim Yevmenkin 283878ed226SJulian Elischer con->link_type = link_type; 284878ed226SJulian Elischer 285878ed226SJulian Elischer if (con->link_type == NG_HCI_LINK_ACL) 286878ed226SJulian Elischer NG_HCI_BUFF_ACL_TOTAL(unit->buffer, num_pkts); 287878ed226SJulian Elischer else 288878ed226SJulian Elischer NG_HCI_BUFF_SCO_TOTAL(unit->buffer, num_pkts); 289878ed226SJulian Elischer 290878ed226SJulian Elischer NG_BT_ITEMQ_INIT(&con->conq, num_pkts); 291878ed226SJulian Elischer 292b81d7730SMaksim Yevmenkin ng_callout_init(&con->con_timo); 293878ed226SJulian Elischer 294878ed226SJulian Elischer LIST_INSERT_HEAD(&unit->con_list, con, next); 295878ed226SJulian Elischer } 296878ed226SJulian Elischer 297878ed226SJulian Elischer return (con); 298878ed226SJulian Elischer } /* ng_hci_new_con */ 299878ed226SJulian Elischer 300878ed226SJulian Elischer /* 301878ed226SJulian Elischer * Free connection descriptor 302878ed226SJulian Elischer */ 303878ed226SJulian Elischer 304878ed226SJulian Elischer void 305878ed226SJulian Elischer ng_hci_free_con(ng_hci_unit_con_p con) 306878ed226SJulian Elischer { 307878ed226SJulian Elischer LIST_REMOVE(con, next); 308878ed226SJulian Elischer 309878ed226SJulian Elischer /* 310878ed226SJulian Elischer * If we have pending packets then assume that Host Controller has 311878ed226SJulian Elischer * flushed these packets and we can free them too 312878ed226SJulian Elischer */ 313878ed226SJulian Elischer 314878ed226SJulian Elischer if (con->link_type == NG_HCI_LINK_ACL) 315878ed226SJulian Elischer NG_HCI_BUFF_ACL_FREE(con->unit->buffer, con->pending); 316878ed226SJulian Elischer else 317878ed226SJulian Elischer NG_HCI_BUFF_SCO_FREE(con->unit->buffer, con->pending); 318878ed226SJulian Elischer 319878ed226SJulian Elischer NG_BT_ITEMQ_DESTROY(&con->conq); 320878ed226SJulian Elischer 321878ed226SJulian Elischer bzero(con, sizeof(*con)); 322878ed226SJulian Elischer FREE(con, M_NETGRAPH_HCI); 323878ed226SJulian Elischer } /* ng_hci_free_con */ 324878ed226SJulian Elischer 325878ed226SJulian Elischer /* 326878ed226SJulian Elischer * Lookup connection for given unit and connection handle. 327878ed226SJulian Elischer */ 328878ed226SJulian Elischer 329878ed226SJulian Elischer ng_hci_unit_con_p 330878ed226SJulian Elischer ng_hci_con_by_handle(ng_hci_unit_p unit, int con_handle) 331878ed226SJulian Elischer { 332878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 333878ed226SJulian Elischer 334878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 335878ed226SJulian Elischer if (con->con_handle == con_handle) 336878ed226SJulian Elischer break; 337878ed226SJulian Elischer 338878ed226SJulian Elischer return (con); 339878ed226SJulian Elischer } /* ng_hci_con_by_handle */ 340878ed226SJulian Elischer 341878ed226SJulian Elischer /* 342878ed226SJulian Elischer * Lookup connection for given unit, link type and remove unit address 343878ed226SJulian Elischer */ 344878ed226SJulian Elischer 345878ed226SJulian Elischer ng_hci_unit_con_p 346878ed226SJulian Elischer ng_hci_con_by_bdaddr(ng_hci_unit_p unit, bdaddr_p bdaddr, int link_type) 347878ed226SJulian Elischer { 348878ed226SJulian Elischer ng_hci_unit_con_p con = NULL; 349878ed226SJulian Elischer 350878ed226SJulian Elischer LIST_FOREACH(con, &unit->con_list, next) 351878ed226SJulian Elischer if (con->link_type == link_type && 352878ed226SJulian Elischer bcmp(&con->bdaddr, bdaddr, sizeof(bdaddr_t)) == 0) 353878ed226SJulian Elischer break; 354878ed226SJulian Elischer 355878ed226SJulian Elischer return (con); 356878ed226SJulian Elischer } /* ng_hci_con_by_bdaddr */ 357878ed226SJulian Elischer 358878ed226SJulian Elischer /* 359878ed226SJulian Elischer * Set HCI command timeout 360b81d7730SMaksim Yevmenkin * XXX FIXME: check return code from ng_timeout 361878ed226SJulian Elischer */ 362878ed226SJulian Elischer 3630986ab12SMaksim Yevmenkin int 364878ed226SJulian Elischer ng_hci_command_timeout(ng_hci_unit_p unit) 365878ed226SJulian Elischer { 3660986ab12SMaksim Yevmenkin if (unit->state & NG_HCI_UNIT_COMMAND_PENDING) 3670986ab12SMaksim Yevmenkin panic( 3680986ab12SMaksim Yevmenkin "%s: %s - Duplicated command timeout!\n", __func__, NG_NODE_NAME(unit->node)); 3690986ab12SMaksim Yevmenkin 370878ed226SJulian Elischer unit->state |= NG_HCI_UNIT_COMMAND_PENDING; 371b81d7730SMaksim Yevmenkin ng_timeout(&unit->cmd_timo, unit->node, NULL, 3720986ab12SMaksim Yevmenkin bluetooth_hci_command_timeout(), 3730986ab12SMaksim Yevmenkin ng_hci_process_command_timeout, NULL, 0); 3740986ab12SMaksim Yevmenkin 3750986ab12SMaksim Yevmenkin return (0); 376878ed226SJulian Elischer } /* ng_hci_command_timeout */ 377878ed226SJulian Elischer 378878ed226SJulian Elischer /* 379878ed226SJulian Elischer * Unset HCI command timeout 380878ed226SJulian Elischer */ 381878ed226SJulian Elischer 3820986ab12SMaksim Yevmenkin int 383878ed226SJulian Elischer ng_hci_command_untimeout(ng_hci_unit_p unit) 384878ed226SJulian Elischer { 3850986ab12SMaksim Yevmenkin if (!(unit->state & NG_HCI_UNIT_COMMAND_PENDING)) 3860986ab12SMaksim Yevmenkin panic( 3870986ab12SMaksim Yevmenkin "%s: %s - No command timeout!\n", __func__, NG_NODE_NAME(unit->node)); 3880986ab12SMaksim Yevmenkin 389b81d7730SMaksim Yevmenkin if (ng_untimeout(&unit->cmd_timo, unit->node) == 0) 3900986ab12SMaksim Yevmenkin return (ETIMEDOUT); 3910986ab12SMaksim Yevmenkin 392878ed226SJulian Elischer unit->state &= ~NG_HCI_UNIT_COMMAND_PENDING; 3930986ab12SMaksim Yevmenkin 3940986ab12SMaksim Yevmenkin return (0); 395878ed226SJulian Elischer } /* ng_hci_command_untimeout */ 396878ed226SJulian Elischer 397878ed226SJulian Elischer /* 398878ed226SJulian Elischer * Set HCI connection timeout 399b81d7730SMaksim Yevmenkin * XXX FIXME: check return code from ng_timeout 400878ed226SJulian Elischer */ 401878ed226SJulian Elischer 4020986ab12SMaksim Yevmenkin int 403878ed226SJulian Elischer ng_hci_con_timeout(ng_hci_unit_con_p con) 404878ed226SJulian Elischer { 4050986ab12SMaksim Yevmenkin if (con->flags & NG_HCI_CON_TIMEOUT_PENDING) 4060986ab12SMaksim Yevmenkin panic( 4070986ab12SMaksim Yevmenkin "%s: %s - Duplicated connection timeout!\n", 4080986ab12SMaksim Yevmenkin __func__, NG_NODE_NAME(con->unit->node)); 4090986ab12SMaksim Yevmenkin 410878ed226SJulian Elischer con->flags |= NG_HCI_CON_TIMEOUT_PENDING; 411b81d7730SMaksim Yevmenkin ng_timeout(&con->con_timo, con->unit->node, NULL, 4120986ab12SMaksim Yevmenkin bluetooth_hci_connect_timeout(), 4130986ab12SMaksim Yevmenkin ng_hci_process_con_timeout, NULL, 4140986ab12SMaksim Yevmenkin con->con_handle); 4150986ab12SMaksim Yevmenkin 4160986ab12SMaksim Yevmenkin return (0); 417878ed226SJulian Elischer } /* ng_hci_con_timeout */ 418878ed226SJulian Elischer 419878ed226SJulian Elischer /* 420878ed226SJulian Elischer * Unset HCI connection timeout 421878ed226SJulian Elischer */ 422878ed226SJulian Elischer 4230986ab12SMaksim Yevmenkin int 424878ed226SJulian Elischer ng_hci_con_untimeout(ng_hci_unit_con_p con) 425878ed226SJulian Elischer { 4260986ab12SMaksim Yevmenkin if (!(con->flags & NG_HCI_CON_TIMEOUT_PENDING)) 4270986ab12SMaksim Yevmenkin panic( 4280986ab12SMaksim Yevmenkin "%s: %s - No connection timeout!\n", __func__, NG_NODE_NAME(con->unit->node)); 4290986ab12SMaksim Yevmenkin 430b81d7730SMaksim Yevmenkin if (ng_untimeout(&con->con_timo, con->unit->node) == 0) 4310986ab12SMaksim Yevmenkin return (ETIMEDOUT); 4320986ab12SMaksim Yevmenkin 433878ed226SJulian Elischer con->flags &= ~NG_HCI_CON_TIMEOUT_PENDING; 4340986ab12SMaksim Yevmenkin 4350986ab12SMaksim Yevmenkin return (0); 436878ed226SJulian Elischer } /* ng_hci_con_untimeout */ 437878ed226SJulian Elischer 438878ed226SJulian Elischer #if 0 439878ed226SJulian Elischer /* 440878ed226SJulian Elischer * Convert numeric error code/reason to a string 441878ed226SJulian Elischer */ 442878ed226SJulian Elischer 443878ed226SJulian Elischer char const * const 444878ed226SJulian Elischer ng_hci_str_error(u_int16_t code) 445878ed226SJulian Elischer { 446878ed226SJulian Elischer #define LAST_ERROR_CODE ((sizeof(s)/sizeof(s[0]))-1) 447878ed226SJulian Elischer static char const * const s[] = { 448878ed226SJulian Elischer /* 0x00 */ "No error", 449878ed226SJulian Elischer /* 0x01 */ "Unknown HCI command", 450878ed226SJulian Elischer /* 0x02 */ "No connection", 451878ed226SJulian Elischer /* 0x03 */ "Hardware failure", 452878ed226SJulian Elischer /* 0x04 */ "Page timeout", 453878ed226SJulian Elischer /* 0x05 */ "Authentication failure", 454878ed226SJulian Elischer /* 0x06 */ "Key missing", 455878ed226SJulian Elischer /* 0x07 */ "Memory full", 456878ed226SJulian Elischer /* 0x08 */ "Connection timeout", 457878ed226SJulian Elischer /* 0x09 */ "Max number of connections", 458878ed226SJulian Elischer /* 0x0a */ "Max number of SCO connections to a unit", 459878ed226SJulian Elischer /* 0x0b */ "ACL connection already exists", 460878ed226SJulian Elischer /* 0x0c */ "Command disallowed", 461878ed226SJulian Elischer /* 0x0d */ "Host rejected due to limited resources", 462878ed226SJulian Elischer /* 0x0e */ "Host rejected due to securiity reasons", 463878ed226SJulian Elischer /* 0x0f */ "Host rejected due to remote unit is a personal unit", 464878ed226SJulian Elischer /* 0x10 */ "Host timeout", 465878ed226SJulian Elischer /* 0x11 */ "Unsupported feature or parameter value", 466878ed226SJulian Elischer /* 0x12 */ "Invalid HCI command parameter", 467878ed226SJulian Elischer /* 0x13 */ "Other end terminated connection: User ended connection", 468878ed226SJulian Elischer /* 0x14 */ "Other end terminated connection: Low resources", 469878ed226SJulian Elischer /* 0x15 */ "Other end terminated connection: About to power off", 470878ed226SJulian Elischer /* 0x16 */ "Connection terminated by local host", 471878ed226SJulian Elischer /* 0x17 */ "Repeated attempts", 472878ed226SJulian Elischer /* 0x18 */ "Pairing not allowed", 473878ed226SJulian Elischer /* 0x19 */ "Unknown LMP PDU", 474878ed226SJulian Elischer /* 0x1a */ "Unsupported remote feature", 475878ed226SJulian Elischer /* 0x1b */ "SCO offset rejected", 476878ed226SJulian Elischer /* 0x1c */ "SCO interval rejected", 477878ed226SJulian Elischer /* 0x1d */ "SCO air mode rejected", 478878ed226SJulian Elischer /* 0x1e */ "Invalid LMP parameters", 479878ed226SJulian Elischer /* 0x1f */ "Unspecified error", 480878ed226SJulian Elischer /* 0x20 */ "Unsupported LMP parameter value", 481878ed226SJulian Elischer /* 0x21 */ "Role change not allowed", 482878ed226SJulian Elischer /* 0x22 */ "LMP response timeout", 483878ed226SJulian Elischer /* 0x23 */ "LMP error transaction collision", 484878ed226SJulian Elischer /* 0x24 */ "LMP PSU not allowed", 485878ed226SJulian Elischer /* 0x25 */ "Encryption mode not acceptable", 486878ed226SJulian Elischer /* 0x26 */ "Unit key used", 487878ed226SJulian Elischer /* 0x27 */ "QoS is not supported", 488878ed226SJulian Elischer /* 0x28 */ "Instant passed", 489878ed226SJulian Elischer /* 0x29 */ "Paring with unit key not supported", 490878ed226SJulian Elischer /* SHOULD ALWAYS BE LAST */ "Unknown error" 491878ed226SJulian Elischer }; 492878ed226SJulian Elischer 493878ed226SJulian Elischer return ((code >= LAST_ERROR_CODE)? s[LAST_ERROR_CODE] : s[code]); 494878ed226SJulian Elischer } /* ng_hci_str_error */ 495878ed226SJulian Elischer #endif 496878ed226SJulian Elischer 497