/* * * Realtek Bluetooth USB driver * * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ #include #include #include #include #include #include #include #include #include #include #include #include #include "rtk_bt.h" #include "rtk_misc.h" #define VERSION "3.1" #ifdef BTCOEX #include "rtk_coex.h" #endif #ifdef RTKBT_SWITCH_PATCH #include #include DEFINE_SEMAPHORE(switch_sem); #endif #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 7, 1) static bool reset = 0; #endif static struct usb_driver btusb_driver; static struct usb_device_id btusb_table[] = { { .match_flags = USB_DEVICE_ID_MATCH_VENDOR | USB_DEVICE_ID_MATCH_INT_INFO, .idVendor = 0x0bda, .bInterfaceClass = 0xe0, .bInterfaceSubClass = 0x01, .bInterfaceProtocol = 0x01 }, { .match_flags = USB_DEVICE_ID_MATCH_VENDOR | USB_DEVICE_ID_MATCH_INT_INFO, .idVendor = 0x13d3, .bInterfaceClass = 0xe0, .bInterfaceSubClass = 0x01, .bInterfaceProtocol = 0x01 }, { .match_flags = USB_DEVICE_ID_MATCH_VENDOR | USB_DEVICE_ID_MATCH_INT_INFO, .idVendor = 0x0489, .bInterfaceClass = 0xe0, .bInterfaceSubClass = 0x01, .bInterfaceProtocol = 0x01 }, { .match_flags = USB_DEVICE_ID_MATCH_VENDOR | USB_DEVICE_ID_MATCH_INT_INFO, .idVendor = 0x1358, .bInterfaceClass = 0xe0, .bInterfaceSubClass = 0x01, .bInterfaceProtocol = 0x01 }, { } }; static void rtk_free(struct btusb_data *data) { #if LINUX_VERSION_CODE < KERNEL_VERSION(3, 7, 1) kfree(data); #endif return; } static struct btusb_data *rtk_alloc(struct usb_interface *intf) { struct btusb_data *data; #if LINUX_VERSION_CODE < KERNEL_VERSION(3, 7, 1) data = kzalloc(sizeof(*data), GFP_KERNEL); #else data = devm_kzalloc(&intf->dev, sizeof(*data), GFP_KERNEL); #endif return data; } MODULE_DEVICE_TABLE(usb, btusb_table); static int inc_tx(struct btusb_data *data) { unsigned long flags; int rv; spin_lock_irqsave(&data->txlock, flags); rv = test_bit(BTUSB_SUSPENDING, &data->flags); if (!rv) data->tx_in_flight++; spin_unlock_irqrestore(&data->txlock, flags); return rv; } #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 18, 0) static inline void btusb_free_frags(struct btusb_data *data) { unsigned long flags; spin_lock_irqsave(&data->rxlock, flags); kfree_skb(data->evt_skb); data->evt_skb = NULL; kfree_skb(data->acl_skb); data->acl_skb = NULL; kfree_skb(data->sco_skb); data->sco_skb = NULL; spin_unlock_irqrestore(&data->rxlock, flags); } static int btusb_recv_intr(struct btusb_data *data, void *buffer, int count) { struct sk_buff *skb; int err = 0; spin_lock(&data->rxlock); skb = data->evt_skb; while (count) { int len; if (!skb) { skb = bt_skb_alloc(HCI_MAX_EVENT_SIZE, GFP_ATOMIC); if (!skb) { err = -ENOMEM; break; } bt_cb(skb)->pkt_type = HCI_EVENT_PKT; bt_cb(skb)->expect = HCI_EVENT_HDR_SIZE; } len = min_t(uint, bt_cb(skb)->expect, count); memcpy(skb_put(skb, len), buffer, len); count -= len; buffer += len; bt_cb(skb)->expect -= len; if (skb->len == HCI_EVENT_HDR_SIZE) { /* Complete event header */ bt_cb(skb)->expect = hci_event_hdr(skb)->plen; if (skb_tailroom(skb) < bt_cb(skb)->expect) { kfree_skb(skb); skb = NULL; err = -EILSEQ; break; } } if (bt_cb(skb)->expect == 0) { /* Complete frame */ hci_recv_frame(data->hdev, skb); skb = NULL; } } data->evt_skb = skb; spin_unlock(&data->rxlock); return err; } static int btusb_recv_bulk(struct btusb_data *data, void *buffer, int count) { struct sk_buff *skb; int err = 0; spin_lock(&data->rxlock); skb = data->acl_skb; while (count) { int len; if (!skb) { skb = bt_skb_alloc(HCI_MAX_FRAME_SIZE, GFP_ATOMIC); if (!skb) { err = -ENOMEM; break; } bt_cb(skb)->pkt_type = HCI_ACLDATA_PKT; bt_cb(skb)->expect = HCI_ACL_HDR_SIZE; } len = min_t(uint, bt_cb(skb)->expect, count); memcpy(skb_put(skb, len), buffer, len); count -= len; buffer += len; bt_cb(skb)->expect -= len; if (skb->len == HCI_ACL_HDR_SIZE) { __le16 dlen = hci_acl_hdr(skb)->dlen; /* Complete ACL header */ bt_cb(skb)->expect = __le16_to_cpu(dlen); if (skb_tailroom(skb) < bt_cb(skb)->expect) { kfree_skb(skb); skb = NULL; err = -EILSEQ; break; } } if (bt_cb(skb)->expect == 0) { /* Complete frame */ hci_recv_frame(data->hdev, skb); skb = NULL; } } data->acl_skb = skb; spin_unlock(&data->rxlock); return err; } static int btusb_recv_isoc(struct btusb_data *data, void *buffer, int count) { struct sk_buff *skb; int err = 0; spin_lock(&data->rxlock); skb = data->sco_skb; while (count) { int len; if (!skb) { skb = bt_skb_alloc(HCI_MAX_SCO_SIZE, GFP_ATOMIC); if (!skb) { err = -ENOMEM; break; } bt_cb(skb)->pkt_type = HCI_SCODATA_PKT; bt_cb(skb)->expect = HCI_SCO_HDR_SIZE; } len = min_t(uint, bt_cb(skb)->expect, count); memcpy(skb_put(skb, len), buffer, len); count -= len; buffer += len; bt_cb(skb)->expect -= len; if (skb->len == HCI_SCO_HDR_SIZE) { /* Complete SCO header */ bt_cb(skb)->expect = hci_sco_hdr(skb)->dlen; if (skb_tailroom(skb) < bt_cb(skb)->expect) { kfree_skb(skb); skb = NULL; err = -EILSEQ; break; } } if (bt_cb(skb)->expect == 0) { /* Complete frame */ hci_recv_frame(data->hdev, skb); skb = NULL; } } data->sco_skb = skb; spin_unlock(&data->rxlock); return err; } #endif static void btusb_intr_complete(struct urb *urb) { struct hci_dev *hdev = urb->context; struct btusb_data *data = GET_DRV_DATA(hdev); int err; //RTKBT_DBG("%s: urb %p status %d count %d ", __func__, //urb, urb->status, urb->actual_length); if (!test_bit(HCI_RUNNING, &hdev->flags)) return; if (urb->status == 0) { hdev->stat.byte_rx += urb->actual_length; #ifdef BTCOEX rtk_btcoex_parse_event(urb->transfer_buffer, urb->actual_length); #endif #if HCI_VERSION_CODE < KERNEL_VERSION(3, 18, 0) if (hci_recv_fragment(hdev, HCI_EVENT_PKT, urb->transfer_buffer, urb->actual_length) < 0) { RTKBT_ERR("%s: Corrupted event packet", __func__); hdev->stat.err_rx++; } #else if (btusb_recv_intr(data, urb->transfer_buffer, urb->actual_length) < 0) { RTKBT_ERR("%s corrupted event packet", hdev->name); hdev->stat.err_rx++; } #endif } /* Avoid suspend failed when usb_kill_urb */ else if (urb->status == -ENOENT) { return; } if (!test_bit(BTUSB_INTR_RUNNING, &data->flags)) return; usb_mark_last_busy(data->udev); usb_anchor_urb(urb, &data->intr_anchor); err = usb_submit_urb(urb, GFP_ATOMIC); if (err < 0) { /* -EPERM: urb is being killed; * -ENODEV: device got disconnected */ if (err != -EPERM && err != -ENODEV) RTKBT_ERR("%s: Failed to re-submit urb %p, err %d", __func__, urb, err); usb_unanchor_urb(urb); } } static int btusb_submit_intr_urb(struct hci_dev *hdev, gfp_t mem_flags) { struct btusb_data *data = GET_DRV_DATA(hdev); struct urb *urb; unsigned char *buf; unsigned int pipe; int err, size; //RTKBT_DBG("%s", hdev->name); if (!data->intr_ep) return -ENODEV; urb = usb_alloc_urb(0, mem_flags); if (!urb) return -ENOMEM; size = le16_to_cpu(data->intr_ep->wMaxPacketSize); buf = kmalloc(size, mem_flags); if (!buf) { usb_free_urb(urb); return -ENOMEM; } pipe = usb_rcvintpipe(data->udev, data->intr_ep->bEndpointAddress); usb_fill_int_urb(urb, data->udev, pipe, buf, size, btusb_intr_complete, hdev, data->intr_ep->bInterval); urb->transfer_flags |= URB_FREE_BUFFER; usb_anchor_urb(urb, &data->intr_anchor); err = usb_submit_urb(urb, mem_flags); if (err < 0) { RTKBT_ERR ("btusb_submit_intr_urb %s urb %p submission failed (%d)", hdev->name, urb, -err); usb_unanchor_urb(urb); } usb_free_urb(urb); return err; } static void btusb_bulk_complete(struct urb *urb) { struct hci_dev *hdev = urb->context; struct btusb_data *data = GET_DRV_DATA(hdev); int err; //RTKBT_DBG("%s: urb %p status %d count %d", //__func__, urb, urb->status, urb->actual_length); if (!test_bit(HCI_RUNNING, &hdev->flags)) return; #ifdef BTCOEX if (urb->status == 0) rtk_btcoex_parse_l2cap_data_rx(urb->transfer_buffer, urb->actual_length); #endif if (urb->status == 0) { hdev->stat.byte_rx += urb->actual_length; #if HCI_VERSION_CODE < KERNEL_VERSION(3, 18, 0) if (hci_recv_fragment(hdev, HCI_ACLDATA_PKT, urb->transfer_buffer, urb->actual_length) < 0) { RTKBT_ERR("%s: Corrupted ACL packet", __func__); hdev->stat.err_rx++; } #else if (data->recv_bulk(data, urb->transfer_buffer, urb->actual_length) < 0) { RTKBT_ERR("%s corrupted ACL packet", hdev->name); hdev->stat.err_rx++; } #endif } /* Avoid suspend failed when usb_kill_urb */ else if (urb->status == -ENOENT) { return; } if (!test_bit(BTUSB_BULK_RUNNING, &data->flags)) return; usb_anchor_urb(urb, &data->bulk_anchor); usb_mark_last_busy(data->udev); err = usb_submit_urb(urb, GFP_ATOMIC); if (err < 0) { /* -EPERM: urb is being killed; * -ENODEV: device got disconnected */ if (err != -EPERM && err != -ENODEV) RTKBT_ERR ("btusb_bulk_complete %s urb %p failed to resubmit (%d)", hdev->name, urb, -err); usb_unanchor_urb(urb); } } static int btusb_submit_bulk_urb(struct hci_dev *hdev, gfp_t mem_flags) { struct btusb_data *data = GET_DRV_DATA(hdev); struct urb *urb; unsigned char *buf; unsigned int pipe; int err, size = HCI_MAX_FRAME_SIZE; //RTKBT_DBG("%s: hdev name %s", __func__, hdev->name); if (!data->bulk_rx_ep) return -ENODEV; urb = usb_alloc_urb(0, mem_flags); if (!urb) return -ENOMEM; buf = kmalloc(size, mem_flags); if (!buf) { usb_free_urb(urb); return -ENOMEM; } pipe = usb_rcvbulkpipe(data->udev, data->bulk_rx_ep->bEndpointAddress); usb_fill_bulk_urb(urb, data->udev, pipe, buf, size, btusb_bulk_complete, hdev); urb->transfer_flags |= URB_FREE_BUFFER; usb_mark_last_busy(data->udev); usb_anchor_urb(urb, &data->bulk_anchor); err = usb_submit_urb(urb, mem_flags); if (err < 0) { RTKBT_ERR("%s: Failed to submit urb %p, err %d", __func__, urb, err); usb_unanchor_urb(urb); } usb_free_urb(urb); return err; } static void btusb_isoc_complete(struct urb *urb) { struct hci_dev *hdev = urb->context; struct btusb_data *data = GET_DRV_DATA(hdev); int i, err; /* RTKBT_DBG("%s urb %p status %d count %d", hdev->name, urb, urb->status, urb->actual_length); */ if (!test_bit(HCI_RUNNING, &hdev->flags)) return; if (urb->status == 0) { for (i = 0; i < urb->number_of_packets; i++) { unsigned int offset = urb->iso_frame_desc[i].offset; unsigned int length = urb->iso_frame_desc[i].actual_length; if (urb->iso_frame_desc[i].status) continue; hdev->stat.byte_rx += length; #if HCI_VERSION_CODE < KERNEL_VERSION(3, 18, 0) if (hci_recv_fragment(hdev, HCI_SCODATA_PKT, urb->transfer_buffer + offset, length) < 0) { RTKBT_ERR("%s: Corrupted SCO packet", __func__); hdev->stat.err_rx++; } #else if (btusb_recv_isoc(data, urb->transfer_buffer + offset, length) < 0) { RTKBT_ERR("%s corrupted SCO packet", hdev->name); hdev->stat.err_rx++; } #endif } } /* Avoid suspend failed when usb_kill_urb */ else if (urb->status == -ENOENT) { return; } if (!test_bit(BTUSB_ISOC_RUNNING, &data->flags)) return; usb_anchor_urb(urb, &data->isoc_anchor); i = 0; retry: err = usb_submit_urb(urb, GFP_ATOMIC); if (err < 0) { /* -EPERM: urb is being killed; * -ENODEV: device got disconnected */ if (err != -EPERM && err != -ENODEV) RTKBT_ERR ("%s: Failed to re-sumbit urb %p, retry %d, err %d", __func__, urb, i, err); if (i < 10) { i++; mdelay(1); goto retry; } usb_unanchor_urb(urb); } } static inline void __fill_isoc_descriptor(struct urb *urb, int len, int mtu) { int i, offset = 0; //RTKBT_DBG("len %d mtu %d", len, mtu); for (i = 0; i < BTUSB_MAX_ISOC_FRAMES && len >= mtu; i++, offset += mtu, len -= mtu) { urb->iso_frame_desc[i].offset = offset; urb->iso_frame_desc[i].length = mtu; } if (len && i < BTUSB_MAX_ISOC_FRAMES) { urb->iso_frame_desc[i].offset = offset; urb->iso_frame_desc[i].length = len; i++; } urb->number_of_packets = i; } static int btusb_submit_isoc_urb(struct hci_dev *hdev, gfp_t mem_flags) { struct btusb_data *data = GET_DRV_DATA(hdev); struct urb *urb; unsigned char *buf; unsigned int pipe; int err, size; //RTKBT_DBG("%s", hdev->name); if (!data->isoc_rx_ep) return -ENODEV; urb = usb_alloc_urb(BTUSB_MAX_ISOC_FRAMES, mem_flags); if (!urb) return -ENOMEM; size = le16_to_cpu(data->isoc_rx_ep->wMaxPacketSize) * BTUSB_MAX_ISOC_FRAMES; buf = kmalloc(size, mem_flags); if (!buf) { usb_free_urb(urb); return -ENOMEM; } pipe = usb_rcvisocpipe(data->udev, data->isoc_rx_ep->bEndpointAddress); urb->dev = data->udev; urb->pipe = pipe; urb->context = hdev; urb->complete = btusb_isoc_complete; urb->interval = data->isoc_rx_ep->bInterval; urb->transfer_flags = URB_FREE_BUFFER | URB_ISO_ASAP; urb->transfer_buffer = buf; urb->transfer_buffer_length = size; __fill_isoc_descriptor(urb, size, le16_to_cpu(data->isoc_rx_ep->wMaxPacketSize)); usb_anchor_urb(urb, &data->isoc_anchor); err = usb_submit_urb(urb, mem_flags); if (err < 0) { RTKBT_ERR("%s %s urb %p submission failed (%d)", __func__, hdev->name, urb, err); usb_unanchor_urb(urb); } usb_free_urb(urb); return err; } static void btusb_tx_complete(struct urb *urb) { struct sk_buff *skb = urb->context; struct hci_dev *hdev = (struct hci_dev *)skb->dev; struct btusb_data *data = GET_DRV_DATA(hdev); // RTKBT_DBG("btusb_tx_complete %s urb %p status %d count %d", hdev->name, // urb, urb->status, urb->actual_length); if (!test_bit(HCI_RUNNING, &hdev->flags)) goto done; if (!urb->status) hdev->stat.byte_tx += urb->transfer_buffer_length; else hdev->stat.err_tx++; done: spin_lock(&data->txlock); data->tx_in_flight--; spin_unlock(&data->txlock); kfree(urb->setup_packet); kfree_skb(skb); } static void btusb_isoc_tx_complete(struct urb *urb) { struct sk_buff *skb = urb->context; struct hci_dev *hdev = (struct hci_dev *)skb->dev; //RTKBT_DBG("%s: urb %p status %d count %d", //__func__, urb, urb->status, urb->actual_length); if (skb && hdev) { if (!test_bit(HCI_RUNNING, &hdev->flags)) goto done; if (!urb->status) hdev->stat.byte_tx += urb->transfer_buffer_length; else hdev->stat.err_tx++; } else RTKBT_ERR("%s: skb 0x%p hdev 0x%p", __func__, skb, hdev); done: kfree(urb->setup_packet); kfree_skb(skb); } static int btusb_open(struct hci_dev *hdev) { struct btusb_data *data = GET_DRV_DATA(hdev); int err; err = usb_autopm_get_interface(data->intf); if (err < 0) return err; data->intf->needs_remote_wakeup = 1; RTKBT_DBG("%s start", __func__); /*******************************/ if (0 == atomic_read(&hdev->promisc)) { RTKBT_ERR("btusb_open hdev->promisc ==0"); err = -1; //goto failed; } err = download_patch(data->intf); if (err < 0) goto failed; /*******************************/ RTKBT_INFO("%s set HCI_RUNNING", __func__); if (test_and_set_bit(HCI_RUNNING, &hdev->flags)) goto done; if (test_and_set_bit(BTUSB_INTR_RUNNING, &data->flags)) goto done; err = btusb_submit_intr_urb(hdev, GFP_KERNEL); if (err < 0) goto failed; err = btusb_submit_bulk_urb(hdev, GFP_KERNEL); if (err < 0) { mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_kill_anchored_urbs(&data->intr_anchor); goto failed; } set_bit(BTUSB_BULK_RUNNING, &data->flags); btusb_submit_bulk_urb(hdev, GFP_KERNEL); done: usb_autopm_put_interface(data->intf); #ifdef BTCOEX rtk_btcoex_open(hdev); #endif RTKBT_DBG("%s end", __FUNCTION__); return 0; failed: clear_bit(BTUSB_INTR_RUNNING, &data->flags); clear_bit(HCI_RUNNING, &hdev->flags); usb_autopm_put_interface(data->intf); RTKBT_ERR("%s failed", __FUNCTION__); return err; } static void btusb_stop_traffic(struct btusb_data *data) { mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_kill_anchored_urbs(&data->intr_anchor); usb_kill_anchored_urbs(&data->bulk_anchor); usb_kill_anchored_urbs(&data->isoc_anchor); } static int btusb_close(struct hci_dev *hdev) { struct btusb_data *data = GET_DRV_DATA(hdev); int err; #if HCI_VERSION_CODE < KERNEL_VERSION(4, 1, 0) int i; #endif /* When in kernel 4.4.0 and greater, the HCI_RUNNING bit is * cleared in hci_dev_do_close(). */ #if HCI_VERSION_CODE < KERNEL_VERSION(4, 4, 0) if (!test_and_clear_bit(HCI_RUNNING, &hdev->flags)) return 0; #else if (test_bit(HCI_RUNNING, &hdev->flags)) { RTKBT_ERR("HCI_RUNNING is not cleared before."); return -1; } #endif RTKBT_DBG("btusb_close"); #if HCI_VERSION_CODE < KERNEL_VERSION(4, 1, 0) /*******************************/ for (i = 0; i < NUM_REASSEMBLY; i++) { if (hdev->reassembly[i]) { kfree_skb(hdev->reassembly[i]); hdev->reassembly[i] = NULL; RTKBT_DBG("%s free ressembly i=%d", __FUNCTION__, i); } } /*******************************/ #endif cancel_work_sync(&data->work); cancel_work_sync(&data->waker); clear_bit(BTUSB_ISOC_RUNNING, &data->flags); clear_bit(BTUSB_BULK_RUNNING, &data->flags); clear_bit(BTUSB_INTR_RUNNING, &data->flags); btusb_stop_traffic(data); #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 18, 0) btusb_free_frags(data); #endif err = usb_autopm_get_interface(data->intf); if (err < 0) goto failed; data->intf->needs_remote_wakeup = 0; usb_autopm_put_interface(data->intf); #ifdef BTCOEX rtk_btcoex_close(); #endif failed: mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_scuttle_anchored_urbs(&data->deferred); #ifdef RTKBT_SWITCH_PATCH down(&switch_sem); if (data->context) { struct api_context *ctx = data->context; if (ctx->flags & RTLBT_CLOSE) { ctx->flags &= ~RTLBT_CLOSE; ctx->status = 0; complete(&ctx->done); } } up(&switch_sem); #endif return 0; } static int btusb_flush(struct hci_dev *hdev) { struct btusb_data *data = GET_DRV_DATA(hdev); RTKBT_DBG("%s add delay ", __FUNCTION__); mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_kill_anchored_urbs(&data->tx_anchor); #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 18, 0) btusb_free_frags(data); #endif return 0; } const char pkt_ind[][8] = { [HCI_COMMAND_PKT] = "cmd", [HCI_ACLDATA_PKT] = "acl", [HCI_SCODATA_PKT] = "sco", }; #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 13, 0) int btusb_send_frame(struct hci_dev *hdev, struct sk_buff *skb) { #else int btusb_send_frame(struct sk_buff *skb) { struct hci_dev *hdev = (struct hci_dev *)skb->dev; #endif struct btusb_data *data = GET_DRV_DATA(hdev); struct usb_ctrlrequest *dr; struct urb *urb; unsigned int pipe; int err; //RTKBT_DBG("%s", hdev->name); if (!test_bit(HCI_RUNNING, &hdev->flags)) { /* If the parameter is wrong, the hdev isn't the correct * one. Then no HCI commands can be sent. * This issue is related to the wrong HCI_VERSION_CODE set */ RTKBT_ERR("HCI is not running"); return -EBUSY; } /* Before kernel/hci version 3.13.0, the skb->dev is set before * entering btusb_send_frame(). So there is no need to set it here. * * The skb->dev will be used in the callbacks when urb transfer * completes. See btusb_tx_complete() and btusb_isoc_tx_complete() */ #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 13, 0) skb->dev = (void *)hdev; #endif switch (bt_cb(skb)->pkt_type) { case HCI_COMMAND_PKT: print_command(skb); #ifdef BTCOEX rtk_btcoex_parse_cmd(skb->data, skb->len); #endif urb = usb_alloc_urb(0, GFP_ATOMIC); if (!urb) return -ENOMEM; dr = kmalloc(sizeof(*dr), GFP_ATOMIC); if (!dr) { usb_free_urb(urb); return -ENOMEM; } dr->bRequestType = data->cmdreq_type; 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, btusb_tx_complete, skb); hdev->stat.cmd_tx++; break; case HCI_ACLDATA_PKT: print_acl(skb, 1); #ifdef BTCOEX rtk_btcoex_parse_l2cap_data_tx(skb->data, skb->len); #endif if (!data->bulk_tx_ep) return -ENODEV; urb = usb_alloc_urb(0, GFP_ATOMIC); if (!urb) return -ENOMEM; pipe = usb_sndbulkpipe(data->udev, data->bulk_tx_ep->bEndpointAddress); usb_fill_bulk_urb(urb, data->udev, pipe, skb->data, skb->len, btusb_tx_complete, skb); hdev->stat.acl_tx++; break; case HCI_SCODATA_PKT: if (!data->isoc_tx_ep || SCO_NUM < 1) return -ENODEV; urb = usb_alloc_urb(BTUSB_MAX_ISOC_FRAMES, GFP_ATOMIC); if (!urb) return -ENOMEM; pipe = usb_sndisocpipe(data->udev, data->isoc_tx_ep->bEndpointAddress); usb_fill_int_urb(urb, data->udev, pipe, skb->data, skb->len, btusb_isoc_tx_complete, skb, data->isoc_tx_ep->bInterval); urb->transfer_flags = URB_ISO_ASAP; __fill_isoc_descriptor(urb, skb->len, le16_to_cpu(data->isoc_tx_ep-> wMaxPacketSize)); hdev->stat.sco_tx++; goto skip_waking; default: return -EILSEQ; } err = inc_tx(data); if (err) { usb_anchor_urb(urb, &data->deferred); schedule_work(&data->waker); err = 0; goto done; } skip_waking: usb_anchor_urb(urb, &data->tx_anchor); err = usb_submit_urb(urb, GFP_ATOMIC); if (err < 0) { RTKBT_ERR("%s %s urb %p submission for %s failed, err %d", __func__, hdev->name, urb, pkt_ind[bt_cb(skb)->pkt_type], err); kfree(urb->setup_packet); usb_unanchor_urb(urb); } else { usb_mark_last_busy(data->udev); } usb_free_urb(urb); done: return err; } #if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0) static void btusb_destruct(struct hci_dev *hdev) { RTKBT_DBG("btusb_destruct %s", hdev->name); hci_free_dev(hdev); } #endif static void btusb_notify(struct hci_dev *hdev, unsigned int evt) { struct btusb_data *data = GET_DRV_DATA(hdev); RTKBT_DBG("%s: %s evt %d", __func__, hdev->name, evt); if (SCO_NUM != data->sco_num) { data->sco_num = SCO_NUM; RTKBT_DBG("%s: Update sco num %d", __func__, data->sco_num); schedule_work(&data->work); } } static inline int __set_isoc_interface(struct hci_dev *hdev, int altsetting) { struct btusb_data *data = GET_DRV_DATA(hdev); struct usb_interface *intf = data->isoc; struct usb_endpoint_descriptor *ep_desc; int i, err; if (!data->isoc) return -ENODEV; RTKBT_INFO("set isoc interface: alt %d", altsetting); err = usb_set_interface(data->udev, 1, altsetting); if (err < 0) { RTKBT_ERR("%s setting interface failed (%d)", hdev->name, -err); return err; } data->isoc_altsetting = altsetting; data->isoc_tx_ep = NULL; data->isoc_rx_ep = NULL; for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) { ep_desc = &intf->cur_altsetting->endpoint[i].desc; if (!data->isoc_tx_ep && usb_endpoint_is_isoc_out(ep_desc)) { data->isoc_tx_ep = ep_desc; continue; } if (!data->isoc_rx_ep && usb_endpoint_is_isoc_in(ep_desc)) { data->isoc_rx_ep = ep_desc; continue; } } if (!data->isoc_tx_ep || !data->isoc_rx_ep) { RTKBT_ERR("%s invalid SCO descriptors", hdev->name); return -ENODEV; } return 0; } static void btusb_work(struct work_struct *work) { struct btusb_data *data = container_of(work, struct btusb_data, work); struct hci_dev *hdev = data->hdev; int err; int new_alts; RTKBT_DBG("%s: sco num %d", __func__, data->sco_num); if (data->sco_num > 0) { if (!test_bit(BTUSB_DID_ISO_RESUME, &data->flags)) { err = usb_autopm_get_interface(data->isoc ? data-> isoc : data->intf); if (err < 0) { clear_bit(BTUSB_ISOC_RUNNING, &data->flags); mdelay(URB_CANCELING_DELAY_MS); usb_kill_anchored_urbs(&data->isoc_anchor); return; } set_bit(BTUSB_DID_ISO_RESUME, &data->flags); } #if HCI_VERSION_CODE > KERNEL_VERSION(3, 7, 1) if (hdev->voice_setting & 0x0020) { static const int alts[3] = { 2, 4, 5 }; new_alts = alts[data->sco_num - 1]; } else { new_alts = data->sco_num; } if (data->isoc_altsetting != new_alts) { #else if (data->isoc_altsetting != 2) { new_alts = 2; #endif clear_bit(BTUSB_ISOC_RUNNING, &data->flags); mdelay(URB_CANCELING_DELAY_MS); usb_kill_anchored_urbs(&data->isoc_anchor); if (__set_isoc_interface(hdev, new_alts) < 0) return; } if (!test_and_set_bit(BTUSB_ISOC_RUNNING, &data->flags)) { RTKBT_INFO("submit SCO RX urb."); if (btusb_submit_isoc_urb(hdev, GFP_KERNEL) < 0) clear_bit(BTUSB_ISOC_RUNNING, &data->flags); else btusb_submit_isoc_urb(hdev, GFP_KERNEL); } } else { clear_bit(BTUSB_ISOC_RUNNING, &data->flags); mdelay(URB_CANCELING_DELAY_MS); usb_kill_anchored_urbs(&data->isoc_anchor); __set_isoc_interface(hdev, 0); if (test_and_clear_bit(BTUSB_DID_ISO_RESUME, &data->flags)) usb_autopm_put_interface(data->isoc ? data-> isoc : data->intf); } } static void btusb_waker(struct work_struct *work) { struct btusb_data *data = container_of(work, struct btusb_data, waker); int err; err = usb_autopm_get_interface(data->intf); RTKBT_DBG("%s start", __FUNCTION__); if (err < 0) return; usb_autopm_put_interface(data->intf); RTKBT_DBG("%s end", __FUNCTION__); } int rtkbt_pm_notify(struct notifier_block *notifier, ulong pm_event, void *unused) { struct btusb_data *data; struct usb_device *udev; struct usb_interface *intf; struct hci_dev *hdev; /* int err; */ #ifdef RTKBT_SWITCH_PATCH u8 *cmd; int result; static u8 hci_state = 0; struct api_context ctx; #endif data = container_of(notifier, struct btusb_data, pm_notifier); udev = data->udev; intf = data->intf; hdev = data->hdev; RTKBT_DBG("%s: pm_event %ld", __func__, pm_event); switch (pm_event) { case PM_SUSPEND_PREPARE: case PM_HIBERNATION_PREPARE: /* No need to load firmware because the download firmware * process is deprecated in resume. * We use rebind after resume instead */ /* err = usb_autopm_get_interface(data->intf); * if (err < 0) * return err; * patch_entry->fw_len = * load_firmware(dev_entry, &patch_entry->fw_cache); * usb_autopm_put_interface(data->intf); * if (patch_entry->fw_len <= 0) { * RTKBT_DBG("rtkbt_pm_notify return NOTIFY_BAD"); * return NOTIFY_BAD; * } */ RTKBT_DBG("%s: suspend prepare", __func__); if (!device_may_wakeup(&udev->dev)) { #ifdef CONFIG_NEEDS_BINDING intf->needs_binding = 1; RTKBT_DBG("Remote wakeup not support, set " "intf->needs_binding = 1"); #else RTKBT_DBG("Remote wakeup not support, no needs binding"); #endif } #ifdef RTKBT_SWITCH_PATCH if (test_bit(HCI_UP, &hdev->flags)) { unsigned long expire; init_completion(&ctx.done); hci_state = 1; down(&switch_sem); data->context = &ctx; ctx.flags = RTLBT_CLOSE; queue_work(hdev->req_workqueue, &hdev->power_off.work); up(&switch_sem); expire = msecs_to_jiffies(1000); if (!wait_for_completion_timeout(&ctx.done, expire)) RTKBT_ERR("hdev close timeout"); down(&switch_sem); data->context = NULL; up(&switch_sem); } cmd = kzalloc(16, GFP_ATOMIC); if (!cmd) { RTKBT_ERR("Can't allocate memory for cmd"); return -ENOMEM; } /* Clear patch */ cmd[0] = 0x66; cmd[1] = 0xfc; cmd[2] = 0x00; result = __rtk_send_hci_cmd(udev, cmd, 3); kfree(cmd); msleep(100); /* From FW colleague's recommendation */ result = download_lps_patch(intf); /* Tell the controller to wake up host if received special * advertising packet */ set_scan(intf); /* Send special vendor commands */ #endif break; case PM_POST_SUSPEND: case PM_POST_HIBERNATION: case PM_POST_RESTORE: /* if (patch_entry->fw_len > 0) { * kfree(patch_entry->fw_cache); * patch_entry->fw_cache = NULL; * patch_entry->fw_len = 0; * } */ #ifdef RTKBT_SWITCH_PATCH cmd = kzalloc(16, GFP_ATOMIC); if (!cmd) { RTKBT_ERR("Can't allocate memory for cmd"); return -ENOMEM; } /* Clear patch */ cmd[0] = 0x66; cmd[1] = 0xfc; cmd[2] = 0x00; result = __rtk_send_hci_cmd(udev, cmd, 3); kfree(cmd); msleep(100); /* From FW colleague's recommendation */ result = download_patch(intf); if (hci_state) { hci_state = 0; queue_work(hdev->req_workqueue, &hdev->power_on); } #endif #if BTUSB_RPM RTKBT_DBG("%s: Re-enable autosuspend", __func__); /* pm_runtime_use_autosuspend(&udev->dev); * pm_runtime_set_autosuspend_delay(&udev->dev, 2000); * pm_runtime_set_active(&udev->dev); * pm_runtime_allow(&udev->dev); * pm_runtime_mark_last_busy(&udev->dev); * pm_runtime_autosuspend(&udev->dev); * pm_runtime_put_autosuspend(&udev->dev); * usb_disable_autosuspend(udev); */ /* FIXME: usb_enable_autosuspend(udev) is useless here. * Because it is always enabled after enabled in btusb_probe() */ usb_enable_autosuspend(udev); pm_runtime_mark_last_busy(&udev->dev); #endif break; default: break; } return NOTIFY_DONE; } static int btusb_probe(struct usb_interface *intf, const struct usb_device_id *id) { struct usb_endpoint_descriptor *ep_desc; struct btusb_data *data; struct hci_dev *hdev; int i, err, flag1, flag2; struct usb_device *udev; udev = interface_to_usbdev(intf); RTKBT_DBG("btusb_probe intf->cur_altsetting->desc.bInterfaceNumber %d", intf->cur_altsetting->desc.bInterfaceNumber); /* interface numbers are hardcoded in the spec */ if (intf->cur_altsetting->desc.bInterfaceNumber != 0) return -ENODEV; /*******************************/ flag1 = device_can_wakeup(&udev->dev); flag2 = device_may_wakeup(&udev->dev); RTKBT_DBG("btusb_probe can_wakeup %x, may wakeup %x", flag1, flag2); #if BTUSB_WAKEUP_HOST device_wakeup_enable(&udev->dev); #endif //device_wakeup_enable(&udev->dev); /*device_wakeup_disable(&udev->dev); flag1=device_can_wakeup(&udev->dev); flag2=device_may_wakeup(&udev->dev); RTKBT_DBG("btusb_probe can_wakeup=%x flag2=%x",flag1,flag2); */ err = patch_add(intf); if (err < 0) return -1; /*******************************/ data = rtk_alloc(intf); if (!data) return -ENOMEM; for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) { ep_desc = &intf->cur_altsetting->endpoint[i].desc; if (!data->intr_ep && usb_endpoint_is_int_in(ep_desc)) { data->intr_ep = ep_desc; continue; } if (!data->bulk_tx_ep && usb_endpoint_is_bulk_out(ep_desc)) { data->bulk_tx_ep = ep_desc; continue; } if (!data->bulk_rx_ep && usb_endpoint_is_bulk_in(ep_desc)) { data->bulk_rx_ep = ep_desc; continue; } } if (!data->intr_ep || !data->bulk_tx_ep || !data->bulk_rx_ep) { rtk_free(data); return -ENODEV; } data->cmdreq_type = USB_TYPE_CLASS; data->udev = interface_to_usbdev(intf); data->intf = intf; spin_lock_init(&data->lock); INIT_WORK(&data->work, btusb_work); INIT_WORK(&data->waker, btusb_waker); spin_lock_init(&data->txlock); init_usb_anchor(&data->tx_anchor); init_usb_anchor(&data->intr_anchor); init_usb_anchor(&data->bulk_anchor); init_usb_anchor(&data->isoc_anchor); init_usb_anchor(&data->deferred); #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 18, 0) spin_lock_init(&data->rxlock); data->recv_bulk = btusb_recv_bulk; #endif hdev = hci_alloc_dev(); if (!hdev) { rtk_free(data); return -ENOMEM; } HDEV_BUS = HCI_USB; data->hdev = hdev; SET_HCIDEV_DEV(hdev, &intf->dev); hdev->open = btusb_open; hdev->close = btusb_close; hdev->flush = btusb_flush; hdev->send = btusb_send_frame; hdev->notify = btusb_notify; #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 4, 0) hci_set_drvdata(hdev, data); #else hdev->driver_data = data; hdev->destruct = btusb_destruct; hdev->owner = THIS_MODULE; #endif #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 7, 1) if (!reset) set_bit(HCI_QUIRK_RESET_ON_CLOSE, &hdev->quirks); RTKBT_DBG("set_bit(HCI_QUIRK_RESET_ON_CLOSE, &hdev->quirks);"); #endif /* Interface numbers are hardcoded in the specification */ data->isoc = usb_ifnum_to_if(data->udev, 1); if (data->isoc) { err = usb_driver_claim_interface(&btusb_driver, data->isoc, data); if (err < 0) { hci_free_dev(hdev); rtk_free(data); return err; } } #if HCI_VERSION_CODE >= KERNEL_VERSION(4, 1, 0) set_bit(HCI_QUIRK_SIMULTANEOUS_DISCOVERY, &hdev->quirks); #endif err = hci_register_dev(hdev); if (err < 0) { hci_free_dev(hdev); rtk_free(data); return err; } usb_set_intfdata(intf, data); /* Register PM notifier */ data->pm_notifier.notifier_call = rtkbt_pm_notify; register_pm_notifier(&data->pm_notifier); #ifdef BTCOEX rtk_btcoex_probe(hdev); #endif RTKBT_DBG("%s: done", __func__); return 0; } static void btusb_disconnect(struct usb_interface *intf) { struct btusb_data *data = usb_get_intfdata(intf); struct hci_dev *hdev; struct usb_device *udev; udev = interface_to_usbdev(intf); if (intf->cur_altsetting->desc.bInterfaceNumber != 0) return; if (!data) return; RTKBT_DBG("btusb_disconnect"); /* Un-register PM notifier */ unregister_pm_notifier(&data->pm_notifier); /*******************************/ patch_remove(intf); /*******************************/ hdev = data->hdev; #if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0) __hci_dev_hold(hdev); #endif usb_set_intfdata(data->intf, NULL); if (data->isoc) usb_set_intfdata(data->isoc, NULL); hci_unregister_dev(hdev); if (intf == data->isoc) usb_driver_release_interface(&btusb_driver, data->intf); else if (data->isoc) usb_driver_release_interface(&btusb_driver, data->isoc); #if HCI_VERSION_CODE < KERNEL_VERSION(3, 4, 0) __hci_dev_put(hdev); #endif #if HCI_VERSION_CODE >= KERNEL_VERSION(3, 18, 0) btusb_free_frags(data); #endif hci_free_dev(hdev); rtk_free(data); } #ifdef CONFIG_PM static int btusb_suspend(struct usb_interface *intf, pm_message_t message) { struct btusb_data *data = usb_get_intfdata(intf); if (intf->cur_altsetting->desc.bInterfaceNumber != 0) return 0; /*******************************/ RTKBT_DBG("btusb_suspend message.event 0x%x, data->suspend_count %d", message.event, data->suspend_count); if (!test_bit(HCI_RUNNING, &data->hdev->flags)) { RTKBT_INFO("%s: hdev is not HCI_RUNNING", __func__); /* set_scan(data->intf); */ } /*******************************/ if (data->suspend_count++) return 0; spin_lock_irq(&data->txlock); if (!((message.event & PM_EVENT_AUTO) && data->tx_in_flight)) { set_bit(BTUSB_SUSPENDING, &data->flags); spin_unlock_irq(&data->txlock); RTKBT_INFO("%s: suspending...", __func__); } else { spin_unlock_irq(&data->txlock); data->suspend_count--; return -EBUSY; } cancel_work_sync(&data->work); btusb_stop_traffic(data); mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_kill_anchored_urbs(&data->tx_anchor); return 0; } static void play_deferred(struct btusb_data *data) { struct urb *urb; int err; while ((urb = usb_get_from_anchor(&data->deferred))) { /************************************/ usb_anchor_urb(urb, &data->tx_anchor); err = usb_submit_urb(urb, GFP_ATOMIC); if (err < 0) { RTKBT_ERR("play_deferred urb %p submission failed", urb); kfree(urb->setup_packet); usb_unanchor_urb(urb); } else { usb_mark_last_busy(data->udev); } usb_free_urb(urb); /************************************/ data->tx_in_flight++; } mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_scuttle_anchored_urbs(&data->deferred); } static int btusb_resume(struct usb_interface *intf) { struct btusb_data *data = usb_get_intfdata(intf); struct hci_dev *hdev = data->hdev; int err = 0; if (intf->cur_altsetting->desc.bInterfaceNumber != 0) return 0; /*******************************/ RTKBT_DBG("%s: data->suspend_count %d", __func__, data->suspend_count); /* if intf->needs_binding is set, driver will be rebind. * The probe will be called instead of resume */ /* if (!test_bit(HCI_RUNNING, &hdev->flags)) { * RTKBT_DBG("btusb_resume-----bt is off,download patch"); * download_patch(intf); * } else * RTKBT_DBG("btusb_resume,----bt is on"); */ /*******************************/ if (--data->suspend_count) return 0; if (test_bit(BTUSB_INTR_RUNNING, &data->flags)) { err = btusb_submit_intr_urb(hdev, GFP_NOIO); if (err < 0) { clear_bit(BTUSB_INTR_RUNNING, &data->flags); goto failed; } } if (test_bit(BTUSB_BULK_RUNNING, &data->flags)) { err = btusb_submit_bulk_urb(hdev, GFP_NOIO); if (err < 0) { clear_bit(BTUSB_BULK_RUNNING, &data->flags); goto failed; } btusb_submit_bulk_urb(hdev, GFP_NOIO); } if (test_bit(BTUSB_ISOC_RUNNING, &data->flags)) { if (btusb_submit_isoc_urb(hdev, GFP_NOIO) < 0) clear_bit(BTUSB_ISOC_RUNNING, &data->flags); else btusb_submit_isoc_urb(hdev, GFP_NOIO); } spin_lock_irq(&data->txlock); play_deferred(data); clear_bit(BTUSB_SUSPENDING, &data->flags); spin_unlock_irq(&data->txlock); schedule_work(&data->work); RTKBT_DBG("%s: data->suspend_count %d, done", __func__, data->suspend_count); return 0; failed: mdelay(URB_CANCELING_DELAY_MS); // Added by Realtek usb_scuttle_anchored_urbs(&data->deferred); //done: spin_lock_irq(&data->txlock); clear_bit(BTUSB_SUSPENDING, &data->flags); spin_unlock_irq(&data->txlock); RTKBT_DBG("%s: data->suspend_count %d, fail", __func__, data->suspend_count); return err; } #endif static struct usb_driver btusb_driver = { .name = "rtk_btusb", .probe = btusb_probe, .disconnect = btusb_disconnect, #ifdef CONFIG_PM .suspend = btusb_suspend, .resume = btusb_resume, #ifdef RTKBT_SWITCH_PATCH .reset_resume = btusb_resume, #endif #endif .id_table = btusb_table, .supports_autosuspend = 1, #if LINUX_VERSION_CODE > KERNEL_VERSION(3, 7, 1) .disable_hub_initiated_lpm = 1, #endif }; static int __init btusb_init(void) { RTKBT_DBG("Realtek Bluetooth USB driver ver %s", VERSION); #ifdef BTCOEX rtk_btcoex_init(); #endif return usb_register(&btusb_driver); } static void __exit btusb_exit(void) { RTKBT_DBG("rtk_btusb: btusb_exit"); usb_deregister(&btusb_driver); #ifdef BTCOEX rtk_btcoex_exit(); #endif } module_init(btusb_init); module_exit(btusb_exit); MODULE_AUTHOR(""); MODULE_DESCRIPTION("Realtek Bluetooth USB driver ver " VERSION); MODULE_VERSION(VERSION); MODULE_LICENSE("GPL");