Commit e24b21ec authored by Marcel Holtmann's avatar Marcel Holtmann Committed by David S. Miller

[Bluetooth] Change BPA 100/105 driver to use USB anchors

With the new support for USB anchors the driver can become more
simpler and also cleaner. This patch switches to the usage of USB
anchors for all URBs.
Signed-off-by: default avatarMarcel Holtmann <marcel@holtmann.org>
parent 6464f35f
...@@ -2,7 +2,7 @@ ...@@ -2,7 +2,7 @@
* *
* Digianswer Bluetooth USB driver * Digianswer Bluetooth USB driver
* *
* Copyright (C) 2004-2005 Marcel Holtmann <marcel@holtmann.org> * Copyright (C) 2004-2007 Marcel Holtmann <marcel@holtmann.org>
* *
* *
* This program is free software; you can redistribute it and/or modify * This program is free software; you can redistribute it and/or modify
...@@ -21,13 +21,14 @@ ...@@ -21,13 +21,14 @@
* *
*/ */
#include <linux/module.h>
#include <linux/kernel.h> #include <linux/kernel.h>
#include <linux/module.h>
#include <linux/init.h> #include <linux/init.h>
#include <linux/slab.h> #include <linux/slab.h>
#include <linux/types.h> #include <linux/types.h>
#include <linux/sched.h>
#include <linux/errno.h> #include <linux/errno.h>
#include <linux/skbuff.h>
#include <linux/usb.h> #include <linux/usb.h>
...@@ -39,7 +40,7 @@ ...@@ -39,7 +40,7 @@
#define BT_DBG(D...) #define BT_DBG(D...)
#endif #endif
#define VERSION "0.8" #define VERSION "0.9"
static int ignore = 0; static int ignore = 0;
...@@ -52,31 +53,14 @@ static struct usb_device_id bpa10x_table[] = { ...@@ -52,31 +53,14 @@ static struct usb_device_id bpa10x_table[] = {
MODULE_DEVICE_TABLE(usb, bpa10x_table); MODULE_DEVICE_TABLE(usb, bpa10x_table);
#define BPA10X_CMD_EP 0x00
#define BPA10X_EVT_EP 0x81
#define BPA10X_TX_EP 0x02
#define BPA10X_RX_EP 0x82
#define BPA10X_CMD_BUF_SIZE 252
#define BPA10X_EVT_BUF_SIZE 16
#define BPA10X_TX_BUF_SIZE 384
#define BPA10X_RX_BUF_SIZE 384
struct bpa10x_data { struct bpa10x_data {
struct hci_dev *hdev; struct hci_dev *hdev;
struct usb_device *udev; struct usb_device *udev;
rwlock_t lock; struct usb_anchor tx_anchor;
struct usb_anchor rx_anchor;
struct sk_buff_head cmd_queue;
struct urb *cmd_urb;
struct urb *evt_urb;
struct sk_buff *evt_skb;
unsigned int evt_len;
struct sk_buff_head tx_queue; struct sk_buff *rx_skb[2];
struct urb *tx_urb;
struct urb *rx_urb;
}; };
#define HCI_VENDOR_HDR_SIZE 5 #define HCI_VENDOR_HDR_SIZE 5
...@@ -87,357 +71,266 @@ struct hci_vendor_hdr { ...@@ -87,357 +71,266 @@ struct hci_vendor_hdr {
__le16 dlen; __le16 dlen;
} __attribute__ ((packed)); } __attribute__ ((packed));
static void bpa10x_recv_bulk(struct bpa10x_data *data, unsigned char *buf, int count) static int bpa10x_recv(struct hci_dev *hdev, int queue, void *buf, int count)
{ {
struct hci_acl_hdr *ah; struct bpa10x_data *data = hdev->driver_data;
struct hci_sco_hdr *sh;
struct hci_vendor_hdr *vh; BT_DBG("%s queue %d buffer %p count %d", hdev->name,
struct sk_buff *skb; queue, buf, count);
int len;
if (queue < 0 || queue > 1)
return -EILSEQ;
hdev->stat.byte_rx += count;
while (count) { while (count) {
switch (*buf++) { struct sk_buff *skb = data->rx_skb[queue];
struct { __u8 type; int expect; } *scb;
int type, len = 0;
if (!skb) {
/* Start of the frame */
type = *((__u8 *) buf);
count--; buf++;
switch (type) {
case HCI_EVENT_PKT:
if (count >= HCI_EVENT_HDR_SIZE) {
struct hci_event_hdr *h = buf;
len = HCI_EVENT_HDR_SIZE + h->plen;
} else
return -EILSEQ;
break;
case HCI_ACLDATA_PKT: case HCI_ACLDATA_PKT:
ah = (struct hci_acl_hdr *) buf; if (count >= HCI_ACL_HDR_SIZE) {
len = HCI_ACL_HDR_SIZE + __le16_to_cpu(ah->dlen); struct hci_acl_hdr *h = buf;
skb = bt_skb_alloc(len, GFP_ATOMIC); len = HCI_ACL_HDR_SIZE +
if (skb) { __le16_to_cpu(h->dlen);
memcpy(skb_put(skb, len), buf, len); } else
skb->dev = (void *) data->hdev; return -EILSEQ;
bt_cb(skb)->pkt_type = HCI_ACLDATA_PKT;
hci_recv_frame(skb);
}
break; break;
case HCI_SCODATA_PKT: case HCI_SCODATA_PKT:
sh = (struct hci_sco_hdr *) buf; if (count >= HCI_SCO_HDR_SIZE) {
len = HCI_SCO_HDR_SIZE + sh->dlen; struct hci_sco_hdr *h = buf;
skb = bt_skb_alloc(len, GFP_ATOMIC); len = HCI_SCO_HDR_SIZE + h->dlen;
if (skb) { } else
memcpy(skb_put(skb, len), buf, len); return -EILSEQ;
skb->dev = (void *) data->hdev;
bt_cb(skb)->pkt_type = HCI_SCODATA_PKT;
hci_recv_frame(skb);
}
break; break;
case HCI_VENDOR_PKT: case HCI_VENDOR_PKT:
vh = (struct hci_vendor_hdr *) buf; if (count >= HCI_VENDOR_HDR_SIZE) {
len = HCI_VENDOR_HDR_SIZE + __le16_to_cpu(vh->dlen); struct hci_vendor_hdr *h = buf;
skb = bt_skb_alloc(len, GFP_ATOMIC); len = HCI_VENDOR_HDR_SIZE +
if (skb) { __le16_to_cpu(h->dlen);
memcpy(skb_put(skb, len), buf, len); } else
skb->dev = (void *) data->hdev; return -EILSEQ;
bt_cb(skb)->pkt_type = HCI_VENDOR_PKT;
hci_recv_frame(skb);
}
break;
default:
len = count - 1;
break; break;
} }
buf += len; skb = bt_skb_alloc(len, GFP_ATOMIC);
count -= (len + 1); if (!skb) {
BT_ERR("%s no memory for packet", hdev->name);
return -ENOMEM;
} }
}
static int bpa10x_recv_event(struct bpa10x_data *data, unsigned char *buf, int size)
{
BT_DBG("data %p buf %p size %d", data, buf, size);
if (data->evt_skb) { skb->dev = (void *) hdev;
struct sk_buff *skb = data->evt_skb;
memcpy(skb_put(skb, size), buf, size); data->rx_skb[queue] = skb;
if (skb->len == data->evt_len) { scb = (void *) skb->cb;
data->evt_skb = NULL; scb->type = type;
data->evt_len = 0; scb->expect = len;
hci_recv_frame(skb);
}
} else { } else {
struct sk_buff *skb; /* Continuation */
struct hci_event_hdr *hdr;
unsigned char pkt_type;
int pkt_len = 0;
if (size < HCI_EVENT_HDR_SIZE + 1) {
BT_ERR("%s event packet block with size %d is too short",
data->hdev->name, size);
return -EILSEQ;
}
pkt_type = *buf++;
size--;
if (pkt_type != HCI_EVENT_PKT) { scb = (void *) skb->cb;
BT_ERR("%s unexpected event packet start byte 0x%02x", len = scb->expect;
data->hdev->name, pkt_type);
return -EPROTO;
} }
hdr = (struct hci_event_hdr *) buf; len = min(len, count);
pkt_len = HCI_EVENT_HDR_SIZE + hdr->plen;
skb = bt_skb_alloc(pkt_len, GFP_ATOMIC); memcpy(skb_put(skb, len), buf, len);
if (!skb) {
BT_ERR("%s no memory for new event packet", scb->expect -= len;
data->hdev->name);
return -ENOMEM;
}
skb->dev = (void *) data->hdev; if (scb->expect == 0) {
bt_cb(skb)->pkt_type = pkt_type; /* Complete frame */
memcpy(skb_put(skb, size), buf, size); data->rx_skb[queue] = NULL;
if (pkt_len == size) { bt_cb(skb)->pkt_type = scb->type;
hci_recv_frame(skb); hci_recv_frame(skb);
} else {
data->evt_skb = skb;
data->evt_len = pkt_len;
} }
count -= len; buf += len;
} }
return 0; return 0;
} }
static void bpa10x_wakeup(struct bpa10x_data *data) static void bpa10x_tx_complete(struct urb *urb)
{ {
struct urb *urb; struct sk_buff *skb = urb->context;
struct sk_buff *skb; struct hci_dev *hdev = (struct hci_dev *) skb->dev;
int err;
BT_DBG("data %p", data); BT_DBG("%s urb %p status %d count %d", hdev->name,
urb, urb->status, urb->actual_length);
urb = data->cmd_urb; if (!test_bit(HCI_RUNNING, &hdev->flags))
if (urb->status == -EINPROGRESS) goto done;
skb = NULL;
if (!urb->status)
hdev->stat.byte_tx += urb->transfer_buffer_length;
else else
skb = skb_dequeue(&data->cmd_queue); hdev->stat.err_tx++;
if (skb) { done:
struct usb_ctrlrequest *cr; kfree(urb->setup_packet);
if (skb->len > BPA10X_CMD_BUF_SIZE) {
BT_ERR("%s command packet with size %d is too big",
data->hdev->name, skb->len);
kfree_skb(skb); kfree_skb(skb);
return; }
}
cr = (struct usb_ctrlrequest *) urb->setup_packet; static void bpa10x_rx_complete(struct urb *urb)
cr->wLength = __cpu_to_le16(skb->len); {
struct hci_dev *hdev = urb->context;
struct bpa10x_data *data = hdev->driver_data;
int err;
skb_copy_from_linear_data(skb, urb->transfer_buffer, skb->len); BT_DBG("%s urb %p status %d count %d", hdev->name,
urb->transfer_buffer_length = skb->len; urb, urb->status, urb->actual_length);
err = usb_submit_urb(urb, GFP_ATOMIC); if (!test_bit(HCI_RUNNING, &hdev->flags))
if (err < 0 && err != -ENODEV) { return;
BT_ERR("%s submit failed for command urb %p with error %d",
data->hdev->name, urb, err);
skb_queue_head(&data->cmd_queue, skb);
} else
kfree_skb(skb);
}
urb = data->tx_urb; if (urb->status == 0) {
if (urb->status == -EINPROGRESS) if (bpa10x_recv(hdev, usb_pipebulk(urb->pipe),
skb = NULL; urb->transfer_buffer,
else urb->actual_length) < 0) {
skb = skb_dequeue(&data->tx_queue); BT_ERR("%s corrupted event packet", hdev->name);
hdev->stat.err_rx++;
}
}
if (skb) { usb_anchor_urb(urb, &data->rx_anchor);
skb_copy_from_linear_data(skb, urb->transfer_buffer, skb->len);
urb->transfer_buffer_length = skb->len;
err = usb_submit_urb(urb, GFP_ATOMIC); err = usb_submit_urb(urb, GFP_ATOMIC);
if (err < 0 && err != -ENODEV) { if (err < 0) {
BT_ERR("%s submit failed for command urb %p with error %d", BT_ERR("%s urb %p failed to resubmit (%d)",
data->hdev->name, urb, err); hdev->name, urb, -err);
skb_queue_head(&data->tx_queue, skb); usb_unanchor_urb(urb);
} else
kfree_skb(skb);
} }
} }
static void bpa10x_complete(struct urb *urb) static inline int bpa10x_submit_intr_urb(struct hci_dev *hdev)
{ {
struct bpa10x_data *data = urb->context; struct bpa10x_data *data = hdev->driver_data;
unsigned char *buf = urb->transfer_buffer; struct urb *urb;
int err, count = urb->actual_length; unsigned char *buf;
unsigned int pipe;
int err, size = 16;
BT_DBG("data %p urb %p buf %p count %d", data, urb, buf, count); BT_DBG("%s", hdev->name);
read_lock(&data->lock); urb = usb_alloc_urb(0, GFP_KERNEL);
if (!urb)
return -ENOMEM;
if (!test_bit(HCI_RUNNING, &data->hdev->flags)) buf = kmalloc(size, GFP_KERNEL);
goto unlock; if (!buf) {
usb_free_urb(urb);
return -ENOMEM;
}
if (urb->status < 0 || !count) pipe = usb_rcvintpipe(data->udev, 0x81);
goto resubmit;
if (usb_pipein(urb->pipe)) { usb_fill_int_urb(urb, data->udev, pipe, buf, size,
data->hdev->stat.byte_rx += count; bpa10x_rx_complete, hdev, 1);
if (usb_pipetype(urb->pipe) == PIPE_INTERRUPT) urb->transfer_flags |= URB_FREE_BUFFER;
bpa10x_recv_event(data, buf, count);
if (usb_pipetype(urb->pipe) == PIPE_BULK) usb_anchor_urb(urb, &data->rx_anchor);
bpa10x_recv_bulk(data, buf, count);
} else {
data->hdev->stat.byte_tx += count;
bpa10x_wakeup(data); err = usb_submit_urb(urb, GFP_KERNEL);
if (err < 0) {
BT_ERR("%s urb %p submission failed (%d)",
hdev->name, urb, -err);
usb_unanchor_urb(urb);
kfree(buf);
} }
resubmit: usb_free_urb(urb);
if (usb_pipein(urb->pipe)) {
err = usb_submit_urb(urb, GFP_ATOMIC);
if (err < 0 && err != -ENODEV) {
BT_ERR("%s urb %p type %d resubmit status %d",
data->hdev->name, urb, usb_pipetype(urb->pipe), err);
}
}
unlock: return err;
read_unlock(&data->lock);
} }
static inline struct urb *bpa10x_alloc_urb(struct usb_device *udev, unsigned int pipe, static inline int bpa10x_submit_bulk_urb(struct hci_dev *hdev)
size_t size, gfp_t flags, void *data)
{ {
struct bpa10x_data *data = hdev->driver_data;
struct urb *urb; struct urb *urb;
struct usb_ctrlrequest *cr;
unsigned char *buf; unsigned char *buf;
unsigned int pipe;
int err, size = 64;
BT_DBG("udev %p data %p", udev, data); BT_DBG("%s", hdev->name);
urb = usb_alloc_urb(0, flags); urb = usb_alloc_urb(0, GFP_KERNEL);
if (!urb) if (!urb)
return NULL; return -ENOMEM;
buf = kmalloc(size, flags); buf = kmalloc(size, GFP_KERNEL);
if (!buf) { if (!buf) {
usb_free_urb(urb); usb_free_urb(urb);
return NULL; return -ENOMEM;
}
switch (usb_pipetype(pipe)) {
case PIPE_CONTROL:
cr = kmalloc(sizeof(*cr), flags);
if (!cr) {
kfree(buf);
usb_free_urb(urb);
return NULL;
} }
cr->bRequestType = USB_TYPE_VENDOR; pipe = usb_rcvbulkpipe(data->udev, 0x82);
cr->bRequest = 0;
cr->wIndex = 0;
cr->wValue = 0;
cr->wLength = __cpu_to_le16(0);
usb_fill_control_urb(urb, udev, pipe, (void *) cr, buf, 0, bpa10x_complete, data); usb_fill_bulk_urb(urb, data->udev, pipe,
break; buf, size, bpa10x_rx_complete, hdev);
case PIPE_INTERRUPT: urb->transfer_flags |= URB_FREE_BUFFER;
usb_fill_int_urb(urb, udev, pipe, buf, size, bpa10x_complete, data, 1);
break;
case PIPE_BULK: usb_anchor_urb(urb, &data->rx_anchor);
usb_fill_bulk_urb(urb, udev, pipe, buf, size, bpa10x_complete, data);
break;
default: err = usb_submit_urb(urb, GFP_KERNEL);
if (err < 0) {
BT_ERR("%s urb %p submission failed (%d)",
hdev->name, urb, -err);
usb_unanchor_urb(urb);
kfree(buf); kfree(buf);
usb_free_urb(urb);
return NULL;
} }
return urb;
}
static inline void bpa10x_free_urb(struct urb *urb)
{
BT_DBG("urb %p", urb);
if (!urb)
return;
kfree(urb->setup_packet);
kfree(urb->transfer_buffer);
usb_free_urb(urb); usb_free_urb(urb);
return err;
} }
static int bpa10x_open(struct hci_dev *hdev) static int bpa10x_open(struct hci_dev *hdev)
{ {
struct bpa10x_data *data = hdev->driver_data; struct bpa10x_data *data = hdev->driver_data;
struct usb_device *udev = data->udev;
unsigned long flags;
int err; int err;
BT_DBG("hdev %p data %p", hdev, data); BT_DBG("%s", hdev->name);
if (test_and_set_bit(HCI_RUNNING, &hdev->flags)) if (test_and_set_bit(HCI_RUNNING, &hdev->flags))
return 0; return 0;
data->cmd_urb = bpa10x_alloc_urb(udev, usb_sndctrlpipe(udev, BPA10X_CMD_EP), err = bpa10x_submit_intr_urb(hdev);
BPA10X_CMD_BUF_SIZE, GFP_KERNEL, data); if (err < 0)
if (!data->cmd_urb) { goto error;
err = -ENOMEM;
goto done;
}
data->evt_urb = bpa10x_alloc_urb(udev, usb_rcvintpipe(udev, BPA10X_EVT_EP),
BPA10X_EVT_BUF_SIZE, GFP_KERNEL, data);
if (!data->evt_urb) {
bpa10x_free_urb(data->cmd_urb);
err = -ENOMEM;
goto done;
}
data->rx_urb = bpa10x_alloc_urb(udev, usb_rcvbulkpipe(udev, BPA10X_RX_EP),
BPA10X_RX_BUF_SIZE, GFP_KERNEL, data);
if (!data->rx_urb) {
bpa10x_free_urb(data->evt_urb);
bpa10x_free_urb(data->cmd_urb);
err = -ENOMEM;
goto done;
}
data->tx_urb = bpa10x_alloc_urb(udev, usb_sndbulkpipe(udev, BPA10X_TX_EP),
BPA10X_TX_BUF_SIZE, GFP_KERNEL, data);
if (!data->rx_urb) {
bpa10x_free_urb(data->rx_urb);
bpa10x_free_urb(data->evt_urb);
bpa10x_free_urb(data->cmd_urb);
err = -ENOMEM;
goto done;
}
write_lock_irqsave(&data->lock, flags); err = bpa10x_submit_bulk_urb(hdev);
if (err < 0)
goto error;
err = usb_submit_urb(data->evt_urb, GFP_ATOMIC); return 0;
if (err < 0) {
BT_ERR("%s submit failed for event urb %p with error %d",
data->hdev->name, data->evt_urb, err);
} else {
err = usb_submit_urb(data->rx_urb, GFP_ATOMIC);
if (err < 0) {
BT_ERR("%s submit failed for rx urb %p with error %d",
data->hdev->name, data->evt_urb, err);
usb_kill_urb(data->evt_urb);
}
}
write_unlock_irqrestore(&data->lock, flags); error:
usb_kill_anchored_urbs(&data->rx_anchor);
done:
if (err < 0)
clear_bit(HCI_RUNNING, &hdev->flags); clear_bit(HCI_RUNNING, &hdev->flags);
return err; return err;
...@@ -446,27 +339,13 @@ done: ...@@ -446,27 +339,13 @@ done:
static int bpa10x_close(struct hci_dev *hdev) static int bpa10x_close(struct hci_dev *hdev)
{ {
struct bpa10x_data *data = hdev->driver_data; struct bpa10x_data *data = hdev->driver_data;
unsigned long flags;
BT_DBG("hdev %p data %p", hdev, data); BT_DBG("%s", hdev->name);
if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags)) if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags))
return 0; return 0;
write_lock_irqsave(&data->lock, flags); usb_kill_anchored_urbs(&data->rx_anchor);
skb_queue_purge(&data->cmd_queue);
usb_kill_urb(data->cmd_urb);
usb_kill_urb(data->evt_urb);
usb_kill_urb(data->rx_urb);
usb_kill_urb(data->tx_urb);
write_unlock_irqrestore(&data->lock, flags);
bpa10x_free_urb(data->cmd_urb);
bpa10x_free_urb(data->evt_urb);
bpa10x_free_urb(data->rx_urb);
bpa10x_free_urb(data->tx_urb);
return 0; return 0;
} }
...@@ -475,9 +354,9 @@ static int bpa10x_flush(struct hci_dev *hdev) ...@@ -475,9 +354,9 @@ static int bpa10x_flush(struct hci_dev *hdev)
{ {
struct bpa10x_data *data = hdev->driver_data; struct bpa10x_data *data = hdev->driver_data;
BT_DBG("hdev %p data %p", hdev, data); BT_DBG("%s", hdev->name);
skb_queue_purge(&data->cmd_queue); usb_kill_anchored_urbs(&data->tx_anchor);
return 0; return 0;
} }
...@@ -485,45 +364,78 @@ static int bpa10x_flush(struct hci_dev *hdev) ...@@ -485,45 +364,78 @@ static int bpa10x_flush(struct hci_dev *hdev)
static int bpa10x_send_frame(struct sk_buff *skb) static int bpa10x_send_frame(struct sk_buff *skb)
{ {
struct hci_dev *hdev = (struct hci_dev *) skb->dev; struct hci_dev *hdev = (struct hci_dev *) skb->dev;
struct bpa10x_data *data; struct bpa10x_data *data = hdev->driver_data;
struct usb_ctrlrequest *dr;
struct urb *urb;
unsigned int pipe;
int err;
BT_DBG("hdev %p skb %p type %d len %d", hdev, skb, bt_cb(skb)->pkt_type, skb->len); BT_DBG("%s", hdev->name);
if (!hdev) {
BT_ERR("Frame for unknown HCI device");
return -ENODEV;
}
if (!test_bit(HCI_RUNNING, &hdev->flags)) if (!test_bit(HCI_RUNNING, &hdev->flags))
return -EBUSY; return -EBUSY;
data = hdev->driver_data; urb = usb_alloc_urb(0, GFP_ATOMIC);
if (!urb)
return -ENOMEM;
/* Prepend skb with frame type */ /* Prepend skb with frame type */
memcpy(skb_push(skb, 1), &bt_cb(skb)->pkt_type, 1); *skb_push(skb, 1) = bt_cb(skb)->pkt_type;
switch (bt_cb(skb)->pkt_type) { switch (bt_cb(skb)->pkt_type) {
case HCI_COMMAND_PKT: case HCI_COMMAND_PKT:
dr = kmalloc(sizeof(*dr), GFP_ATOMIC);
if (!dr) {
usb_free_urb(urb);
return -ENOMEM;
}
dr->bRequestType = USB_TYPE_VENDOR;
dr->bRequest = 0;
dr->wIndex = 0;
dr->wValue = 0;
dr->wLength = __cpu_to_le16(skb->len);
pipe = usb_sndctrlpipe(data->udev, 0x00);
usb_fill_control_urb(urb, data->udev, pipe, (void *) dr,
skb->data, skb->len, bpa10x_tx_complete, skb);
hdev->stat.cmd_tx++; hdev->stat.cmd_tx++;
skb_queue_tail(&data->cmd_queue, skb);
break; break;
case HCI_ACLDATA_PKT: case HCI_ACLDATA_PKT:
pipe = usb_sndbulkpipe(data->udev, 0x02);
usb_fill_bulk_urb(urb, data->udev, pipe,
skb->data, skb->len, bpa10x_tx_complete, skb);
hdev->stat.acl_tx++; hdev->stat.acl_tx++;
skb_queue_tail(&data->tx_queue, skb);
break; break;
case HCI_SCODATA_PKT: case HCI_SCODATA_PKT:
pipe = usb_sndbulkpipe(data->udev, 0x02);
usb_fill_bulk_urb(urb, data->udev, pipe,
skb->data, skb->len, bpa10x_tx_complete, skb);
hdev->stat.sco_tx++; hdev->stat.sco_tx++;
skb_queue_tail(&data->tx_queue, skb);
break; break;
};
read_lock(&data->lock); default:
return -EILSEQ;
}
usb_anchor_urb(urb, &data->tx_anchor);
bpa10x_wakeup(data); err = usb_submit_urb(urb, GFP_ATOMIC);
if (err < 0) {
BT_ERR("%s urb %p submission failed", hdev->name, urb);
kfree(urb->setup_packet);
usb_unanchor_urb(urb);
}
read_unlock(&data->lock); usb_free_urb(urb);
return 0; return 0;
} }
...@@ -532,16 +444,17 @@ static void bpa10x_destruct(struct hci_dev *hdev) ...@@ -532,16 +444,17 @@ static void bpa10x_destruct(struct hci_dev *hdev)
{ {
struct bpa10x_data *data = hdev->driver_data; struct bpa10x_data *data = hdev->driver_data;
BT_DBG("hdev %p data %p", hdev, data); BT_DBG("%s", hdev->name);
kfree(data->rx_skb[0]);
kfree(data->rx_skb[1]);
kfree(data); kfree(data);
} }
static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id *id) static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id *id)
{ {
struct usb_device *udev = interface_to_usbdev(intf);
struct hci_dev *hdev;
struct bpa10x_data *data; struct bpa10x_data *data;
struct hci_dev *hdev;
int err; int err;
BT_DBG("intf %p id %p", intf, id); BT_DBG("intf %p id %p", intf, id);
...@@ -549,33 +462,29 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id * ...@@ -549,33 +462,29 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id *
if (ignore) if (ignore)
return -ENODEV; return -ENODEV;
if (intf->cur_altsetting->desc.bInterfaceNumber > 0) if (intf->cur_altsetting->desc.bInterfaceNumber != 0)
return -ENODEV; return -ENODEV;
data = kzalloc(sizeof(*data), GFP_KERNEL); data = kzalloc(sizeof(*data), GFP_KERNEL);
if (!data) { if (!data)
BT_ERR("Can't allocate data structure");
return -ENOMEM; return -ENOMEM;
}
data->udev = udev;
rwlock_init(&data->lock); data->udev = interface_to_usbdev(intf);
skb_queue_head_init(&data->cmd_queue); init_usb_anchor(&data->tx_anchor);
skb_queue_head_init(&data->tx_queue); init_usb_anchor(&data->rx_anchor);
hdev = hci_alloc_dev(); hdev = hci_alloc_dev();
if (!hdev) { if (!hdev) {
BT_ERR("Can't allocate HCI device");
kfree(data); kfree(data);
return -ENOMEM; return -ENOMEM;
} }
data->hdev = hdev;
hdev->type = HCI_USB; hdev->type = HCI_USB;
hdev->driver_data = data; hdev->driver_data = data;
data->hdev = hdev;
SET_HCIDEV_DEV(hdev, &intf->dev); SET_HCIDEV_DEV(hdev, &intf->dev);
hdev->open = bpa10x_open; hdev->open = bpa10x_open;
...@@ -588,9 +497,8 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id * ...@@ -588,9 +497,8 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id *
err = hci_register_dev(hdev); err = hci_register_dev(hdev);
if (err < 0) { if (err < 0) {
BT_ERR("Can't register HCI device");
kfree(data);
hci_free_dev(hdev); hci_free_dev(hdev);
kfree(data);
return err; return err;
} }
...@@ -602,19 +510,17 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id * ...@@ -602,19 +510,17 @@ static int bpa10x_probe(struct usb_interface *intf, const struct usb_device_id *
static void bpa10x_disconnect(struct usb_interface *intf) static void bpa10x_disconnect(struct usb_interface *intf)
{ {
struct bpa10x_data *data = usb_get_intfdata(intf); struct bpa10x_data *data = usb_get_intfdata(intf);
struct hci_dev *hdev = data->hdev;
BT_DBG("intf %p", intf); BT_DBG("intf %p", intf);
if (!hdev) if (!data)
return; return;
usb_set_intfdata(intf, NULL); usb_set_intfdata(intf, NULL);
if (hci_unregister_dev(hdev) < 0) hci_unregister_dev(data->hdev);
BT_ERR("Can't unregister HCI device %s", hdev->name);
hci_free_dev(hdev); hci_free_dev(data->hdev);
} }
static struct usb_driver bpa10x_driver = { static struct usb_driver bpa10x_driver = {
...@@ -626,15 +532,9 @@ static struct usb_driver bpa10x_driver = { ...@@ -626,15 +532,9 @@ static struct usb_driver bpa10x_driver = {
static int __init bpa10x_init(void) static int __init bpa10x_init(void)
{ {
int err;
BT_INFO("Digianswer Bluetooth USB driver ver %s", VERSION); BT_INFO("Digianswer Bluetooth USB driver ver %s", VERSION);
err = usb_register(&bpa10x_driver); return usb_register(&bpa10x_driver);
if (err < 0)
BT_ERR("Failed to register USB driver");
return err;
} }
static void __exit bpa10x_exit(void) static void __exit bpa10x_exit(void)
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment