38be6befc7
setup, using the handle that they have not yet told us for the connection-to-be. Disconnecting can cause problems so just ignore zero length ACL packets on unknown connection handles. fixes a problem reported by Nick Hudson
1070 lines
25 KiB
C
1070 lines
25 KiB
C
/* $NetBSD: hci_link.c,v 1.22 2010/10/14 07:05:03 plunky Exp $ */
|
|
|
|
/*-
|
|
* Copyright (c) 2005 Iain Hibbert.
|
|
* Copyright (c) 2006 Itronix Inc.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. The name of Itronix Inc. may not be used to endorse
|
|
* or promote products derived from this software without specific
|
|
* prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY ITRONIX INC. ``AS IS'' AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
|
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
|
|
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL ITRONIX INC. BE LIABLE FOR ANY
|
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include <sys/cdefs.h>
|
|
__KERNEL_RCSID(0, "$NetBSD: hci_link.c,v 1.22 2010/10/14 07:05:03 plunky Exp $");
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/kernel.h>
|
|
#include <sys/malloc.h>
|
|
#include <sys/mbuf.h>
|
|
#include <sys/proc.h>
|
|
#include <sys/queue.h>
|
|
#include <sys/systm.h>
|
|
|
|
#include <netbt/bluetooth.h>
|
|
#include <netbt/hci.h>
|
|
#include <netbt/l2cap.h>
|
|
#include <netbt/sco.h>
|
|
|
|
/*******************************************************************************
|
|
*
|
|
* HCI ACL Connections
|
|
*/
|
|
|
|
/*
|
|
* Automatically expire unused ACL connections after this number of
|
|
* seconds (if zero, do not expire unused connections) [sysctl]
|
|
*/
|
|
int hci_acl_expiry = 10; /* seconds */
|
|
|
|
/*
|
|
* hci_acl_open(unit, bdaddr)
|
|
*
|
|
* open ACL connection to remote bdaddr. Only one ACL connection is permitted
|
|
* between any two Bluetooth devices, so we look for an existing one before
|
|
* trying to start a new one.
|
|
*/
|
|
struct hci_link *
|
|
hci_acl_open(struct hci_unit *unit, bdaddr_t *bdaddr)
|
|
{
|
|
struct hci_link *link;
|
|
struct hci_memo *memo;
|
|
hci_create_con_cp cp;
|
|
int err;
|
|
|
|
KASSERT(unit != NULL);
|
|
KASSERT(bdaddr != NULL);
|
|
|
|
link = hci_link_lookup_bdaddr(unit, bdaddr, HCI_LINK_ACL);
|
|
if (link == NULL) {
|
|
link = hci_link_alloc(unit, bdaddr, HCI_LINK_ACL);
|
|
if (link == NULL)
|
|
return NULL;
|
|
}
|
|
|
|
switch(link->hl_state) {
|
|
case HCI_LINK_CLOSED:
|
|
/*
|
|
* open connection to remote device
|
|
*/
|
|
memset(&cp, 0, sizeof(cp));
|
|
bdaddr_copy(&cp.bdaddr, bdaddr);
|
|
cp.pkt_type = htole16(unit->hci_packet_type);
|
|
|
|
memo = hci_memo_find(unit, bdaddr);
|
|
if (memo != NULL) {
|
|
cp.page_scan_rep_mode = memo->page_scan_rep_mode;
|
|
cp.page_scan_mode = memo->page_scan_mode;
|
|
cp.clock_offset = memo->clock_offset;
|
|
}
|
|
|
|
if (unit->hci_link_policy & HCI_LINK_POLICY_ENABLE_ROLE_SWITCH)
|
|
cp.accept_role_switch = 1;
|
|
|
|
err = hci_send_cmd(unit, HCI_CMD_CREATE_CON, &cp, sizeof(cp));
|
|
if (err) {
|
|
hci_link_free(link, err);
|
|
return NULL;
|
|
}
|
|
|
|
link->hl_flags |= HCI_LINK_CREATE_CON;
|
|
link->hl_state = HCI_LINK_WAIT_CONNECT;
|
|
break;
|
|
|
|
case HCI_LINK_WAIT_CONNECT:
|
|
case HCI_LINK_WAIT_AUTH:
|
|
case HCI_LINK_WAIT_ENCRYPT:
|
|
case HCI_LINK_WAIT_SECURE:
|
|
/*
|
|
* somebody else already trying to connect, we just
|
|
* sit on the bench with them..
|
|
*/
|
|
break;
|
|
|
|
case HCI_LINK_OPEN:
|
|
/*
|
|
* If already open, halt any expiry timeouts. We dont need
|
|
* to care about already invoking timeouts since refcnt >0
|
|
* will keep the link alive.
|
|
*/
|
|
callout_stop(&link->hl_expire);
|
|
break;
|
|
|
|
default:
|
|
UNKNOWN(link->hl_state);
|
|
return NULL;
|
|
}
|
|
|
|
/* open */
|
|
link->hl_refcnt++;
|
|
|
|
return link;
|
|
}
|
|
|
|
/*
|
|
* Close ACL connection. When there are no more references to this link,
|
|
* we can either close it down or schedule a delayed closedown.
|
|
*/
|
|
void
|
|
hci_acl_close(struct hci_link *link, int err)
|
|
{
|
|
|
|
KASSERT(link != NULL);
|
|
|
|
if (--link->hl_refcnt == 0) {
|
|
if (link->hl_state == HCI_LINK_CLOSED)
|
|
hci_link_free(link, err);
|
|
else if (hci_acl_expiry > 0)
|
|
callout_schedule(&link->hl_expire, hci_acl_expiry * hz);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Incoming ACL connection.
|
|
*
|
|
* Check the L2CAP listeners list and only accept when there is a
|
|
* potential listener available.
|
|
*
|
|
* There should not be a link to the same bdaddr already, we check
|
|
* anyway though its left unhandled for now.
|
|
*/
|
|
struct hci_link *
|
|
hci_acl_newconn(struct hci_unit *unit, bdaddr_t *bdaddr)
|
|
{
|
|
struct hci_link *link;
|
|
struct l2cap_channel *chan;
|
|
|
|
LIST_FOREACH(chan, &l2cap_listen_list, lc_ncid) {
|
|
if (bdaddr_same(&unit->hci_bdaddr, &chan->lc_laddr.bt_bdaddr)
|
|
|| bdaddr_any(&chan->lc_laddr.bt_bdaddr))
|
|
break;
|
|
}
|
|
|
|
if (chan == NULL) {
|
|
DPRINTF("%s: rejecting connection (no listeners)\n",
|
|
device_xname(unit->hci_dev));
|
|
|
|
return NULL;
|
|
}
|
|
|
|
link = hci_link_lookup_bdaddr(unit, bdaddr, HCI_LINK_ACL);
|
|
if (link != NULL) {
|
|
DPRINTF("%s: rejecting connection (link exists)\n",
|
|
device_xname(unit->hci_dev));
|
|
|
|
return NULL;
|
|
}
|
|
|
|
link = hci_link_alloc(unit, bdaddr, HCI_LINK_ACL);
|
|
if (link != NULL) {
|
|
link->hl_state = HCI_LINK_WAIT_CONNECT;
|
|
|
|
if (hci_acl_expiry > 0)
|
|
callout_schedule(&link->hl_expire, hci_acl_expiry * hz);
|
|
}
|
|
|
|
return link;
|
|
}
|
|
|
|
void
|
|
hci_acl_timeout(void *arg)
|
|
{
|
|
struct hci_link *link = arg;
|
|
hci_discon_cp cp;
|
|
int err;
|
|
|
|
mutex_enter(bt_lock);
|
|
callout_ack(&link->hl_expire);
|
|
|
|
if (link->hl_refcnt > 0)
|
|
goto out;
|
|
|
|
DPRINTF("link #%d expired\n", link->hl_handle);
|
|
|
|
switch (link->hl_state) {
|
|
case HCI_LINK_CLOSED:
|
|
case HCI_LINK_WAIT_CONNECT:
|
|
hci_link_free(link, ECONNRESET);
|
|
break;
|
|
|
|
case HCI_LINK_WAIT_AUTH:
|
|
case HCI_LINK_WAIT_ENCRYPT:
|
|
case HCI_LINK_WAIT_SECURE:
|
|
case HCI_LINK_OPEN:
|
|
cp.con_handle = htole16(link->hl_handle);
|
|
cp.reason = 0x13; /* "Remote User Terminated Connection" */
|
|
|
|
err = hci_send_cmd(link->hl_unit, HCI_CMD_DISCONNECT,
|
|
&cp, sizeof(cp));
|
|
|
|
if (err) {
|
|
DPRINTF("error %d sending HCI_CMD_DISCONNECT\n",
|
|
err);
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
UNKNOWN(link->hl_state);
|
|
break;
|
|
}
|
|
|
|
out:
|
|
mutex_exit(bt_lock);
|
|
}
|
|
|
|
/*
|
|
* Initiate any Link Mode change requests.
|
|
*/
|
|
int
|
|
hci_acl_setmode(struct hci_link *link)
|
|
{
|
|
int err;
|
|
|
|
KASSERT(link != NULL);
|
|
KASSERT(link->hl_unit != NULL);
|
|
|
|
if (link->hl_state != HCI_LINK_OPEN)
|
|
return EINPROGRESS;
|
|
|
|
if ((link->hl_flags & HCI_LINK_AUTH_REQ)
|
|
&& !(link->hl_flags & HCI_LINK_AUTH)) {
|
|
hci_auth_req_cp cp;
|
|
|
|
DPRINTF("requesting auth for handle #%d\n",
|
|
link->hl_handle);
|
|
|
|
link->hl_state = HCI_LINK_WAIT_AUTH;
|
|
cp.con_handle = htole16(link->hl_handle);
|
|
err = hci_send_cmd(link->hl_unit, HCI_CMD_AUTH_REQ,
|
|
&cp, sizeof(cp));
|
|
|
|
return (err == 0 ? EINPROGRESS : err);
|
|
}
|
|
|
|
if ((link->hl_flags & HCI_LINK_ENCRYPT_REQ)
|
|
&& !(link->hl_flags & HCI_LINK_ENCRYPT)) {
|
|
hci_set_con_encryption_cp cp;
|
|
|
|
/* XXX we should check features for encryption capability */
|
|
|
|
DPRINTF("requesting encryption for handle #%d\n",
|
|
link->hl_handle);
|
|
|
|
link->hl_state = HCI_LINK_WAIT_ENCRYPT;
|
|
cp.con_handle = htole16(link->hl_handle);
|
|
cp.encryption_enable = 0x01;
|
|
|
|
err = hci_send_cmd(link->hl_unit, HCI_CMD_SET_CON_ENCRYPTION,
|
|
&cp, sizeof(cp));
|
|
|
|
return (err == 0 ? EINPROGRESS : err);
|
|
}
|
|
|
|
if ((link->hl_flags & HCI_LINK_SECURE_REQ)) {
|
|
hci_change_con_link_key_cp cp;
|
|
|
|
/* always change link key for SECURE requests */
|
|
link->hl_flags &= ~HCI_LINK_SECURE;
|
|
|
|
DPRINTF("changing link key for handle #%d\n",
|
|
link->hl_handle);
|
|
|
|
link->hl_state = HCI_LINK_WAIT_SECURE;
|
|
cp.con_handle = htole16(link->hl_handle);
|
|
|
|
err = hci_send_cmd(link->hl_unit, HCI_CMD_CHANGE_CON_LINK_KEY,
|
|
&cp, sizeof(cp));
|
|
|
|
return (err == 0 ? EINPROGRESS : err);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Link Mode changed.
|
|
*
|
|
* This is called from event handlers when the mode change
|
|
* is complete. We notify upstream and restart the link.
|
|
*/
|
|
void
|
|
hci_acl_linkmode(struct hci_link *link)
|
|
{
|
|
struct l2cap_channel *chan, *next;
|
|
int err, mode = 0;
|
|
|
|
DPRINTF("handle #%d, auth %s, encrypt %s, secure %s\n",
|
|
link->hl_handle,
|
|
(link->hl_flags & HCI_LINK_AUTH ? "on" : "off"),
|
|
(link->hl_flags & HCI_LINK_ENCRYPT ? "on" : "off"),
|
|
(link->hl_flags & HCI_LINK_SECURE ? "on" : "off"));
|
|
|
|
if (link->hl_flags & HCI_LINK_AUTH)
|
|
mode |= L2CAP_LM_AUTH;
|
|
|
|
if (link->hl_flags & HCI_LINK_ENCRYPT)
|
|
mode |= L2CAP_LM_ENCRYPT;
|
|
|
|
if (link->hl_flags & HCI_LINK_SECURE)
|
|
mode |= L2CAP_LM_SECURE;
|
|
|
|
/*
|
|
* The link state will only be OPEN here if the mode change
|
|
* was successful. So, we can proceed with L2CAP connections,
|
|
* or notify already establshed channels, to allow any that
|
|
* are dissatisfied to disconnect before we restart.
|
|
*/
|
|
next = LIST_FIRST(&l2cap_active_list);
|
|
while ((chan = next) != NULL) {
|
|
next = LIST_NEXT(chan, lc_ncid);
|
|
|
|
if (chan->lc_link != link)
|
|
continue;
|
|
|
|
switch(chan->lc_state) {
|
|
case L2CAP_WAIT_SEND_CONNECT_REQ: /* we are connecting */
|
|
if ((mode & chan->lc_mode) != chan->lc_mode) {
|
|
l2cap_close(chan, ECONNABORTED);
|
|
break;
|
|
}
|
|
|
|
chan->lc_state = L2CAP_WAIT_RECV_CONNECT_RSP;
|
|
err = l2cap_send_connect_req(chan);
|
|
if (err) {
|
|
l2cap_close(chan, err);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case L2CAP_WAIT_SEND_CONNECT_RSP: /* they are connecting */
|
|
if ((mode & chan->lc_mode) != chan->lc_mode) {
|
|
l2cap_send_connect_rsp(link, chan->lc_ident,
|
|
0, chan->lc_rcid,
|
|
L2CAP_SECURITY_BLOCK);
|
|
|
|
l2cap_close(chan, ECONNABORTED);
|
|
break;
|
|
}
|
|
|
|
l2cap_send_connect_rsp(link, chan->lc_ident,
|
|
chan->lc_lcid, chan->lc_rcid,
|
|
L2CAP_SUCCESS);
|
|
|
|
chan->lc_state = L2CAP_WAIT_CONFIG;
|
|
chan->lc_flags |= (L2CAP_WAIT_CONFIG_RSP | L2CAP_WAIT_CONFIG_REQ);
|
|
err = l2cap_send_config_req(chan);
|
|
if (err) {
|
|
l2cap_close(chan, err);
|
|
break;
|
|
}
|
|
break;
|
|
|
|
case L2CAP_WAIT_RECV_CONNECT_RSP:
|
|
case L2CAP_WAIT_CONFIG:
|
|
case L2CAP_OPEN: /* already established */
|
|
(*chan->lc_proto->linkmode)(chan->lc_upper, mode);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
link->hl_state = HCI_LINK_OPEN;
|
|
hci_acl_start(link);
|
|
}
|
|
|
|
/*
|
|
* Receive ACL Data
|
|
*
|
|
* we accumulate packet fragments on the hci_link structure
|
|
* until a full L2CAP frame is ready, then send it on.
|
|
*/
|
|
void
|
|
hci_acl_recv(struct mbuf *m, struct hci_unit *unit)
|
|
{
|
|
struct hci_link *link;
|
|
hci_acldata_hdr_t hdr;
|
|
uint16_t handle, want;
|
|
int pb, got;
|
|
|
|
KASSERT(m != NULL);
|
|
KASSERT(unit != NULL);
|
|
|
|
KASSERT(m->m_pkthdr.len >= sizeof(hdr));
|
|
m_copydata(m, 0, sizeof(hdr), &hdr);
|
|
m_adj(m, sizeof(hdr));
|
|
|
|
#ifdef DIAGNOSTIC
|
|
if (hdr.type != HCI_ACL_DATA_PKT) {
|
|
aprint_error_dev(unit->hci_dev, "bad ACL packet type\n");
|
|
goto bad;
|
|
}
|
|
|
|
if (m->m_pkthdr.len != le16toh(hdr.length)) {
|
|
aprint_error_dev(unit->hci_dev,
|
|
"bad ACL packet length (%d != %d)\n",
|
|
m->m_pkthdr.len, le16toh(hdr.length));
|
|
goto bad;
|
|
}
|
|
#endif
|
|
|
|
hdr.length = le16toh(hdr.length);
|
|
hdr.con_handle = le16toh(hdr.con_handle);
|
|
handle = HCI_CON_HANDLE(hdr.con_handle);
|
|
pb = HCI_PB_FLAG(hdr.con_handle);
|
|
|
|
link = hci_link_lookup_handle(unit, handle);
|
|
if (link == NULL) {
|
|
hci_discon_cp cp;
|
|
|
|
DPRINTF("%s: dumping packet for unknown handle #%d\n",
|
|
device_xname(unit->hci_dev), handle);
|
|
|
|
/*
|
|
* There is no way to find out what this connection handle is
|
|
* for, just get rid of it. This may happen, if a USB dongle
|
|
* is plugged into a self powered hub and does not reset when
|
|
* the system is shut down.
|
|
*
|
|
* This can cause a problem with some Broadcom controllers
|
|
* which emit empty ACL packets during connection setup, so
|
|
* only disconnect where data is present.
|
|
*/
|
|
if (hdr.length > 0) {
|
|
cp.con_handle = htole16(handle);
|
|
cp.reason = 0x13;/*"Remote User Terminated Connection"*/
|
|
hci_send_cmd(unit, HCI_CMD_DISCONNECT, &cp, sizeof(cp));
|
|
}
|
|
goto bad;
|
|
}
|
|
|
|
switch (pb) {
|
|
case HCI_PACKET_START:
|
|
if (link->hl_rxp != NULL)
|
|
aprint_error_dev(unit->hci_dev,
|
|
"dropped incomplete ACL packet\n");
|
|
|
|
if (m->m_pkthdr.len < sizeof(l2cap_hdr_t)) {
|
|
aprint_error_dev(unit->hci_dev, "short ACL packet\n");
|
|
goto bad;
|
|
}
|
|
|
|
link->hl_rxp = m;
|
|
got = m->m_pkthdr.len;
|
|
break;
|
|
|
|
case HCI_PACKET_FRAGMENT:
|
|
if (link->hl_rxp == NULL) {
|
|
aprint_error_dev(unit->hci_dev,
|
|
"unexpected packet fragment\n");
|
|
|
|
goto bad;
|
|
}
|
|
|
|
got = m->m_pkthdr.len + link->hl_rxp->m_pkthdr.len;
|
|
m_cat(link->hl_rxp, m);
|
|
m = link->hl_rxp;
|
|
m->m_pkthdr.len = got;
|
|
break;
|
|
|
|
default:
|
|
aprint_error_dev(unit->hci_dev, "unknown packet type\n");
|
|
goto bad;
|
|
}
|
|
|
|
m_copydata(m, 0, sizeof(want), &want);
|
|
want = le16toh(want) + sizeof(l2cap_hdr_t) - got;
|
|
|
|
if (want > 0)
|
|
return;
|
|
|
|
link->hl_rxp = NULL;
|
|
|
|
if (want == 0) {
|
|
l2cap_recv_frame(m, link);
|
|
return;
|
|
}
|
|
|
|
bad:
|
|
m_freem(m);
|
|
}
|
|
|
|
/*
|
|
* Send ACL data on link
|
|
*
|
|
* We must fragment packets into chunks of less than unit->hci_max_acl_size and
|
|
* prepend a relevant ACL header to each fragment. We keep a PDU structure
|
|
* attached to the link, so that completed fragments can be marked off and
|
|
* more data requested from above once the PDU is sent.
|
|
*/
|
|
int
|
|
hci_acl_send(struct mbuf *m, struct hci_link *link,
|
|
struct l2cap_channel *chan)
|
|
{
|
|
struct l2cap_pdu *pdu;
|
|
struct mbuf *n = NULL;
|
|
int plen, mlen, num = 0;
|
|
|
|
KASSERT(link != NULL);
|
|
KASSERT(m != NULL);
|
|
KASSERT(m->m_flags & M_PKTHDR);
|
|
KASSERT(m->m_pkthdr.len > 0);
|
|
|
|
if (link->hl_state == HCI_LINK_CLOSED) {
|
|
m_freem(m);
|
|
return ENETDOWN;
|
|
}
|
|
|
|
pdu = pool_get(&l2cap_pdu_pool, PR_NOWAIT);
|
|
if (pdu == NULL)
|
|
goto nomem;
|
|
|
|
pdu->lp_chan = chan;
|
|
pdu->lp_pending = 0;
|
|
MBUFQ_INIT(&pdu->lp_data);
|
|
|
|
plen = m->m_pkthdr.len;
|
|
mlen = link->hl_unit->hci_max_acl_size;
|
|
|
|
DPRINTFN(5, "%s: handle #%d, plen = %d, max = %d\n",
|
|
device_xname(link->hl_unit->hci_dev), link->hl_handle, plen, mlen);
|
|
|
|
while (plen > 0) {
|
|
if (plen > mlen) {
|
|
n = m_split(m, mlen, M_DONTWAIT);
|
|
if (n == NULL)
|
|
goto nomem;
|
|
} else {
|
|
mlen = plen;
|
|
}
|
|
|
|
if (num++ == 0)
|
|
m->m_flags |= M_PROTO1; /* tag first fragment */
|
|
|
|
DPRINTFN(10, "chunk of %d (plen = %d) bytes\n", mlen, plen);
|
|
MBUFQ_ENQUEUE(&pdu->lp_data, m);
|
|
m = n;
|
|
plen -= mlen;
|
|
}
|
|
|
|
TAILQ_INSERT_TAIL(&link->hl_txq, pdu, lp_next);
|
|
link->hl_txqlen += num;
|
|
|
|
hci_acl_start(link);
|
|
|
|
return 0;
|
|
|
|
nomem:
|
|
if (m) m_freem(m);
|
|
if (pdu) {
|
|
MBUFQ_DRAIN(&pdu->lp_data);
|
|
pool_put(&l2cap_pdu_pool, pdu);
|
|
}
|
|
|
|
return ENOMEM;
|
|
}
|
|
|
|
/*
|
|
* Start sending ACL data on link.
|
|
*
|
|
* This is called when the queue may need restarting: as new data
|
|
* is queued, after link mode changes have completed, or when device
|
|
* buffers have cleared.
|
|
*
|
|
* We may use all the available packet slots. The reason that we add
|
|
* the ACL encapsulation here rather than in hci_acl_send() is that L2CAP
|
|
* signal packets may be queued before the handle is given to us..
|
|
*/
|
|
void
|
|
hci_acl_start(struct hci_link *link)
|
|
{
|
|
struct hci_unit *unit;
|
|
hci_acldata_hdr_t *hdr;
|
|
struct l2cap_pdu *pdu;
|
|
struct mbuf *m;
|
|
uint16_t handle;
|
|
|
|
KASSERT(link != NULL);
|
|
|
|
unit = link->hl_unit;
|
|
KASSERT(unit != NULL);
|
|
|
|
/* this is mainly to block ourselves (below) */
|
|
if (link->hl_state != HCI_LINK_OPEN)
|
|
return;
|
|
|
|
if (link->hl_txqlen == 0 || unit->hci_num_acl_pkts == 0)
|
|
return;
|
|
|
|
/* find first PDU with data to send */
|
|
pdu = TAILQ_FIRST(&link->hl_txq);
|
|
for (;;) {
|
|
if (pdu == NULL)
|
|
return;
|
|
|
|
if (MBUFQ_FIRST(&pdu->lp_data) != NULL)
|
|
break;
|
|
|
|
pdu = TAILQ_NEXT(pdu, lp_next);
|
|
}
|
|
|
|
while (unit->hci_num_acl_pkts > 0) {
|
|
MBUFQ_DEQUEUE(&pdu->lp_data, m);
|
|
KASSERT(m != NULL);
|
|
|
|
if (m->m_flags & M_PROTO1)
|
|
handle = HCI_MK_CON_HANDLE(link->hl_handle,
|
|
HCI_PACKET_START, 0);
|
|
else
|
|
handle = HCI_MK_CON_HANDLE(link->hl_handle,
|
|
HCI_PACKET_FRAGMENT, 0);
|
|
|
|
M_PREPEND(m, sizeof(*hdr), M_DONTWAIT);
|
|
if (m == NULL)
|
|
break;
|
|
|
|
hdr = mtod(m, hci_acldata_hdr_t *);
|
|
hdr->type = HCI_ACL_DATA_PKT;
|
|
hdr->con_handle = htole16(handle);
|
|
hdr->length = htole16(m->m_pkthdr.len - sizeof(*hdr));
|
|
|
|
link->hl_txqlen--;
|
|
pdu->lp_pending++;
|
|
|
|
hci_output_acl(unit, m);
|
|
|
|
if (MBUFQ_FIRST(&pdu->lp_data) == NULL) {
|
|
if (pdu->lp_chan) {
|
|
/*
|
|
* This should enable streaming of PDUs - when
|
|
* we have placed all the fragments on the acl
|
|
* output queue, we trigger the L2CAP layer to
|
|
* send us down one more. Use a false state so
|
|
* we dont run into ourselves coming back from
|
|
* the future..
|
|
*/
|
|
link->hl_state = HCI_LINK_BLOCK;
|
|
l2cap_start(pdu->lp_chan);
|
|
link->hl_state = HCI_LINK_OPEN;
|
|
}
|
|
|
|
pdu = TAILQ_NEXT(pdu, lp_next);
|
|
if (pdu == NULL)
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* We had our turn now, move to the back of the queue to let
|
|
* other links have a go at the output buffers..
|
|
*/
|
|
if (TAILQ_NEXT(link, hl_next)) {
|
|
TAILQ_REMOVE(&unit->hci_links, link, hl_next);
|
|
TAILQ_INSERT_TAIL(&unit->hci_links, link, hl_next);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Confirm ACL packets cleared from Controller buffers. We scan our PDU
|
|
* list to clear pending fragments and signal upstream for more data
|
|
* when a PDU is complete.
|
|
*/
|
|
void
|
|
hci_acl_complete(struct hci_link *link, int num)
|
|
{
|
|
struct l2cap_pdu *pdu;
|
|
struct l2cap_channel *chan;
|
|
|
|
DPRINTFN(5, "handle #%d (%d)\n", link->hl_handle, num);
|
|
|
|
while (num > 0) {
|
|
pdu = TAILQ_FIRST(&link->hl_txq);
|
|
if (pdu == NULL) {
|
|
aprint_error_dev(link->hl_unit->hci_dev,
|
|
"%d packets completed on handle #%x but none pending!\n",
|
|
num, link->hl_handle);
|
|
|
|
return;
|
|
}
|
|
|
|
if (num >= pdu->lp_pending) {
|
|
num -= pdu->lp_pending;
|
|
pdu->lp_pending = 0;
|
|
|
|
if (MBUFQ_FIRST(&pdu->lp_data) == NULL) {
|
|
TAILQ_REMOVE(&link->hl_txq, pdu, lp_next);
|
|
chan = pdu->lp_chan;
|
|
if (chan != NULL) {
|
|
chan->lc_pending--;
|
|
(*chan->lc_proto->complete)
|
|
(chan->lc_upper, 1);
|
|
|
|
if (chan->lc_pending == 0)
|
|
l2cap_start(chan);
|
|
}
|
|
|
|
pool_put(&l2cap_pdu_pool, pdu);
|
|
}
|
|
} else {
|
|
pdu->lp_pending -= num;
|
|
num = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/*******************************************************************************
|
|
*
|
|
* HCI SCO Connections
|
|
*/
|
|
|
|
/*
|
|
* Incoming SCO Connection. We check the list for anybody willing
|
|
* to take it.
|
|
*/
|
|
struct hci_link *
|
|
hci_sco_newconn(struct hci_unit *unit, bdaddr_t *bdaddr)
|
|
{
|
|
struct sockaddr_bt laddr, raddr;
|
|
struct sco_pcb *pcb, *new;
|
|
struct hci_link *sco, *acl;
|
|
|
|
memset(&laddr, 0, sizeof(laddr));
|
|
laddr.bt_len = sizeof(laddr);
|
|
laddr.bt_family = AF_BLUETOOTH;
|
|
bdaddr_copy(&laddr.bt_bdaddr, &unit->hci_bdaddr);
|
|
|
|
memset(&raddr, 0, sizeof(raddr));
|
|
raddr.bt_len = sizeof(raddr);
|
|
raddr.bt_family = AF_BLUETOOTH;
|
|
bdaddr_copy(&raddr.bt_bdaddr, bdaddr);
|
|
|
|
/*
|
|
* There should already be an ACL link up and running before
|
|
* the controller sends us SCO connection requests, but you
|
|
* never know..
|
|
*/
|
|
acl = hci_link_lookup_bdaddr(unit, bdaddr, HCI_LINK_ACL);
|
|
if (acl == NULL || acl->hl_state != HCI_LINK_OPEN)
|
|
return NULL;
|
|
|
|
LIST_FOREACH(pcb, &sco_pcb, sp_next) {
|
|
if ((pcb->sp_flags & SP_LISTENING) == 0)
|
|
continue;
|
|
|
|
new = (*pcb->sp_proto->newconn)(pcb->sp_upper, &laddr, &raddr);
|
|
if (new == NULL)
|
|
continue;
|
|
|
|
/*
|
|
* Ok, got new pcb so we can start a new link and fill
|
|
* in all the details.
|
|
*/
|
|
bdaddr_copy(&new->sp_laddr, &unit->hci_bdaddr);
|
|
bdaddr_copy(&new->sp_raddr, bdaddr);
|
|
|
|
sco = hci_link_alloc(unit, bdaddr, HCI_LINK_SCO);
|
|
if (sco == NULL) {
|
|
sco_detach(&new);
|
|
return NULL;
|
|
}
|
|
|
|
sco->hl_link = hci_acl_open(unit, bdaddr);
|
|
KASSERT(sco->hl_link == acl);
|
|
|
|
sco->hl_sco = new;
|
|
new->sp_link = sco;
|
|
|
|
new->sp_mtu = unit->hci_max_sco_size;
|
|
return sco;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* receive SCO packet, we only need to strip the header and send
|
|
* it to the right handler
|
|
*/
|
|
void
|
|
hci_sco_recv(struct mbuf *m, struct hci_unit *unit)
|
|
{
|
|
struct hci_link *link;
|
|
hci_scodata_hdr_t hdr;
|
|
uint16_t handle;
|
|
|
|
KASSERT(m != NULL);
|
|
KASSERT(unit != NULL);
|
|
|
|
KASSERT(m->m_pkthdr.len >= sizeof(hdr));
|
|
m_copydata(m, 0, sizeof(hdr), &hdr);
|
|
m_adj(m, sizeof(hdr));
|
|
|
|
#ifdef DIAGNOSTIC
|
|
if (hdr.type != HCI_SCO_DATA_PKT) {
|
|
aprint_error_dev(unit->hci_dev, "bad SCO packet type\n");
|
|
goto bad;
|
|
}
|
|
|
|
if (m->m_pkthdr.len != hdr.length) {
|
|
aprint_error_dev(unit->hci_dev,
|
|
"bad SCO packet length (%d != %d)\n",
|
|
m->m_pkthdr.len, hdr.length);
|
|
|
|
goto bad;
|
|
}
|
|
#endif
|
|
|
|
hdr.con_handle = le16toh(hdr.con_handle);
|
|
handle = HCI_CON_HANDLE(hdr.con_handle);
|
|
|
|
link = hci_link_lookup_handle(unit, handle);
|
|
if (link == NULL || link->hl_type == HCI_LINK_ACL) {
|
|
DPRINTF("%s: dumping packet for unknown handle #%d\n",
|
|
device_xname(unit->hci_dev), handle);
|
|
|
|
goto bad;
|
|
}
|
|
|
|
(*link->hl_sco->sp_proto->input)(link->hl_sco->sp_upper, m);
|
|
return;
|
|
|
|
bad:
|
|
m_freem(m);
|
|
}
|
|
|
|
void
|
|
hci_sco_start(struct hci_link *link)
|
|
{
|
|
}
|
|
|
|
/*
|
|
* SCO packets have completed at the controller, so we can
|
|
* signal up to free the buffer space.
|
|
*/
|
|
void
|
|
hci_sco_complete(struct hci_link *link, int num)
|
|
{
|
|
|
|
DPRINTFN(5, "handle #%d (num=%d)\n", link->hl_handle, num);
|
|
link->hl_sco->sp_pending--;
|
|
(*link->hl_sco->sp_proto->complete)(link->hl_sco->sp_upper, num);
|
|
}
|
|
|
|
/*******************************************************************************
|
|
*
|
|
* Generic HCI Connection alloc/free/lookup etc
|
|
*/
|
|
|
|
struct hci_link *
|
|
hci_link_alloc(struct hci_unit *unit, bdaddr_t *bdaddr, uint8_t type)
|
|
{
|
|
struct hci_link *link;
|
|
|
|
KASSERT(unit != NULL);
|
|
|
|
link = malloc(sizeof(struct hci_link), M_BLUETOOTH, M_NOWAIT | M_ZERO);
|
|
if (link == NULL)
|
|
return NULL;
|
|
|
|
link->hl_unit = unit;
|
|
link->hl_type = type;
|
|
link->hl_state = HCI_LINK_CLOSED;
|
|
bdaddr_copy(&link->hl_bdaddr, bdaddr);
|
|
|
|
/* init ACL portion */
|
|
callout_init(&link->hl_expire, 0);
|
|
callout_setfunc(&link->hl_expire, hci_acl_timeout, link);
|
|
|
|
TAILQ_INIT(&link->hl_txq); /* outgoing packets */
|
|
TAILQ_INIT(&link->hl_reqs); /* request queue */
|
|
|
|
link->hl_mtu = L2CAP_MTU_DEFAULT; /* L2CAP signal mtu */
|
|
link->hl_flush = L2CAP_FLUSH_TIMO_DEFAULT; /* flush timeout */
|
|
|
|
/* init SCO portion */
|
|
MBUFQ_INIT(&link->hl_data);
|
|
|
|
/* attach to unit */
|
|
TAILQ_INSERT_TAIL(&unit->hci_links, link, hl_next);
|
|
return link;
|
|
}
|
|
|
|
void
|
|
hci_link_free(struct hci_link *link, int err)
|
|
{
|
|
struct l2cap_req *req;
|
|
struct l2cap_pdu *pdu;
|
|
struct l2cap_channel *chan, *next;
|
|
|
|
KASSERT(link != NULL);
|
|
|
|
DPRINTF("#%d, type = %d, state = %d, refcnt = %d\n",
|
|
link->hl_handle, link->hl_type,
|
|
link->hl_state, link->hl_refcnt);
|
|
|
|
/* ACL reference count */
|
|
if (link->hl_refcnt > 0) {
|
|
next = LIST_FIRST(&l2cap_active_list);
|
|
while ((chan = next) != NULL) {
|
|
next = LIST_NEXT(chan, lc_ncid);
|
|
if (chan->lc_link == link)
|
|
l2cap_close(chan, err);
|
|
}
|
|
}
|
|
KASSERT(link->hl_refcnt == 0);
|
|
|
|
/* ACL L2CAP requests.. */
|
|
while ((req = TAILQ_FIRST(&link->hl_reqs)) != NULL)
|
|
l2cap_request_free(req);
|
|
|
|
KASSERT(TAILQ_EMPTY(&link->hl_reqs));
|
|
|
|
/* ACL outgoing data queue */
|
|
while ((pdu = TAILQ_FIRST(&link->hl_txq)) != NULL) {
|
|
TAILQ_REMOVE(&link->hl_txq, pdu, lp_next);
|
|
MBUFQ_DRAIN(&pdu->lp_data);
|
|
if (pdu->lp_pending)
|
|
link->hl_unit->hci_num_acl_pkts += pdu->lp_pending;
|
|
|
|
pool_put(&l2cap_pdu_pool, pdu);
|
|
}
|
|
|
|
KASSERT(TAILQ_EMPTY(&link->hl_txq));
|
|
|
|
/* ACL incoming data packet */
|
|
if (link->hl_rxp != NULL) {
|
|
m_freem(link->hl_rxp);
|
|
link->hl_rxp = NULL;
|
|
}
|
|
|
|
/* SCO master ACL link */
|
|
if (link->hl_link != NULL) {
|
|
hci_acl_close(link->hl_link, err);
|
|
link->hl_link = NULL;
|
|
}
|
|
|
|
/* SCO pcb */
|
|
if (link->hl_sco != NULL) {
|
|
struct sco_pcb *pcb;
|
|
|
|
pcb = link->hl_sco;
|
|
pcb->sp_link = NULL;
|
|
link->hl_sco = NULL;
|
|
(*pcb->sp_proto->disconnected)(pcb->sp_upper, err);
|
|
}
|
|
|
|
/* flush any SCO data */
|
|
MBUFQ_DRAIN(&link->hl_data);
|
|
|
|
/*
|
|
* Halt the callout - if its already running we cannot free the
|
|
* link structure but the timeout function will call us back in
|
|
* any case.
|
|
*/
|
|
link->hl_state = HCI_LINK_CLOSED;
|
|
callout_stop(&link->hl_expire);
|
|
if (callout_invoking(&link->hl_expire))
|
|
return;
|
|
|
|
callout_destroy(&link->hl_expire);
|
|
|
|
/*
|
|
* If we made a note of clock offset, keep it in a memo
|
|
* to facilitate reconnections to this device
|
|
*/
|
|
if (link->hl_clock != 0) {
|
|
struct hci_memo *memo;
|
|
|
|
memo = hci_memo_new(link->hl_unit, &link->hl_bdaddr);
|
|
if (memo != NULL)
|
|
memo->clock_offset = link->hl_clock;
|
|
}
|
|
|
|
TAILQ_REMOVE(&link->hl_unit->hci_links, link, hl_next);
|
|
free(link, M_BLUETOOTH);
|
|
}
|
|
|
|
/*
|
|
* Lookup HCI link by address and type. Note that for SCO links there may
|
|
* be more than one link per address, so we only return links with no
|
|
* handle (ie new links)
|
|
*/
|
|
struct hci_link *
|
|
hci_link_lookup_bdaddr(struct hci_unit *unit, bdaddr_t *bdaddr, uint8_t type)
|
|
{
|
|
struct hci_link *link;
|
|
|
|
KASSERT(unit != NULL);
|
|
KASSERT(bdaddr != NULL);
|
|
|
|
TAILQ_FOREACH(link, &unit->hci_links, hl_next) {
|
|
if (link->hl_type != type)
|
|
continue;
|
|
|
|
if (type == HCI_LINK_SCO && link->hl_handle != 0)
|
|
continue;
|
|
|
|
if (bdaddr_same(&link->hl_bdaddr, bdaddr))
|
|
break;
|
|
}
|
|
|
|
return link;
|
|
}
|
|
|
|
struct hci_link *
|
|
hci_link_lookup_handle(struct hci_unit *unit, uint16_t handle)
|
|
{
|
|
struct hci_link *link;
|
|
|
|
KASSERT(unit != NULL);
|
|
|
|
TAILQ_FOREACH(link, &unit->hci_links, hl_next) {
|
|
if (handle == link->hl_handle)
|
|
break;
|
|
}
|
|
|
|
return link;
|
|
}
|