18c2ecf20Sopenharmony_ci/** 28c2ecf20Sopenharmony_ci * Marvell NFC-over-USB driver: USB interface related functions 38c2ecf20Sopenharmony_ci * 48c2ecf20Sopenharmony_ci * Copyright (C) 2014, Marvell International Ltd. 58c2ecf20Sopenharmony_ci * 68c2ecf20Sopenharmony_ci * This software file (the "File") is distributed by Marvell International 78c2ecf20Sopenharmony_ci * Ltd. under the terms of the GNU General Public License Version 2, June 1991 88c2ecf20Sopenharmony_ci * (the "License"). You may use, redistribute and/or modify this File in 98c2ecf20Sopenharmony_ci * accordance with the terms and conditions of the License, a copy of which 108c2ecf20Sopenharmony_ci * is available on the worldwide web at 118c2ecf20Sopenharmony_ci * http://www.gnu.org/licenses/old-licenses/gpl-2.0.txt. 128c2ecf20Sopenharmony_ci * 138c2ecf20Sopenharmony_ci * THE FILE IS DISTRIBUTED AS-IS, WITHOUT WARRANTY OF ANY KIND, AND THE 148c2ecf20Sopenharmony_ci * IMPLIED WARRANTIES OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE 158c2ecf20Sopenharmony_ci * ARE EXPRESSLY DISCLAIMED. The License provides additional details about 168c2ecf20Sopenharmony_ci * this warranty disclaimer. 178c2ecf20Sopenharmony_ci **/ 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_ci#include <linux/module.h> 208c2ecf20Sopenharmony_ci#include <linux/usb.h> 218c2ecf20Sopenharmony_ci#include <linux/nfc.h> 228c2ecf20Sopenharmony_ci#include <net/nfc/nci.h> 238c2ecf20Sopenharmony_ci#include <net/nfc/nci_core.h> 248c2ecf20Sopenharmony_ci#include "nfcmrvl.h" 258c2ecf20Sopenharmony_ci 268c2ecf20Sopenharmony_cistatic struct usb_device_id nfcmrvl_table[] = { 278c2ecf20Sopenharmony_ci { USB_DEVICE_AND_INTERFACE_INFO(0x1286, 0x2046, 288c2ecf20Sopenharmony_ci USB_CLASS_VENDOR_SPEC, 4, 1) }, 298c2ecf20Sopenharmony_ci { } /* Terminating entry */ 308c2ecf20Sopenharmony_ci}; 318c2ecf20Sopenharmony_ci 328c2ecf20Sopenharmony_ciMODULE_DEVICE_TABLE(usb, nfcmrvl_table); 338c2ecf20Sopenharmony_ci 348c2ecf20Sopenharmony_ci#define NFCMRVL_USB_BULK_RUNNING 1 358c2ecf20Sopenharmony_ci#define NFCMRVL_USB_SUSPENDING 2 368c2ecf20Sopenharmony_ci 378c2ecf20Sopenharmony_cistruct nfcmrvl_usb_drv_data { 388c2ecf20Sopenharmony_ci struct usb_device *udev; 398c2ecf20Sopenharmony_ci struct usb_interface *intf; 408c2ecf20Sopenharmony_ci unsigned long flags; 418c2ecf20Sopenharmony_ci struct work_struct waker; 428c2ecf20Sopenharmony_ci struct usb_anchor tx_anchor; 438c2ecf20Sopenharmony_ci struct usb_anchor bulk_anchor; 448c2ecf20Sopenharmony_ci struct usb_anchor deferred; 458c2ecf20Sopenharmony_ci int tx_in_flight; 468c2ecf20Sopenharmony_ci /* protects tx_in_flight */ 478c2ecf20Sopenharmony_ci spinlock_t txlock; 488c2ecf20Sopenharmony_ci struct usb_endpoint_descriptor *bulk_tx_ep; 498c2ecf20Sopenharmony_ci struct usb_endpoint_descriptor *bulk_rx_ep; 508c2ecf20Sopenharmony_ci int suspend_count; 518c2ecf20Sopenharmony_ci struct nfcmrvl_private *priv; 528c2ecf20Sopenharmony_ci}; 538c2ecf20Sopenharmony_ci 548c2ecf20Sopenharmony_cistatic int nfcmrvl_inc_tx(struct nfcmrvl_usb_drv_data *drv_data) 558c2ecf20Sopenharmony_ci{ 568c2ecf20Sopenharmony_ci unsigned long flags; 578c2ecf20Sopenharmony_ci int rv; 588c2ecf20Sopenharmony_ci 598c2ecf20Sopenharmony_ci spin_lock_irqsave(&drv_data->txlock, flags); 608c2ecf20Sopenharmony_ci rv = test_bit(NFCMRVL_USB_SUSPENDING, &drv_data->flags); 618c2ecf20Sopenharmony_ci if (!rv) 628c2ecf20Sopenharmony_ci drv_data->tx_in_flight++; 638c2ecf20Sopenharmony_ci spin_unlock_irqrestore(&drv_data->txlock, flags); 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_ci return rv; 668c2ecf20Sopenharmony_ci} 678c2ecf20Sopenharmony_ci 688c2ecf20Sopenharmony_cistatic void nfcmrvl_bulk_complete(struct urb *urb) 698c2ecf20Sopenharmony_ci{ 708c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = urb->context; 718c2ecf20Sopenharmony_ci struct sk_buff *skb; 728c2ecf20Sopenharmony_ci int err; 738c2ecf20Sopenharmony_ci 748c2ecf20Sopenharmony_ci dev_dbg(&drv_data->udev->dev, "urb %p status %d count %d\n", 758c2ecf20Sopenharmony_ci urb, urb->status, urb->actual_length); 768c2ecf20Sopenharmony_ci 778c2ecf20Sopenharmony_ci if (!test_bit(NFCMRVL_NCI_RUNNING, &drv_data->flags)) 788c2ecf20Sopenharmony_ci return; 798c2ecf20Sopenharmony_ci 808c2ecf20Sopenharmony_ci if (!urb->status) { 818c2ecf20Sopenharmony_ci skb = nci_skb_alloc(drv_data->priv->ndev, urb->actual_length, 828c2ecf20Sopenharmony_ci GFP_ATOMIC); 838c2ecf20Sopenharmony_ci if (!skb) { 848c2ecf20Sopenharmony_ci nfc_err(&drv_data->udev->dev, "failed to alloc mem\n"); 858c2ecf20Sopenharmony_ci } else { 868c2ecf20Sopenharmony_ci skb_put_data(skb, urb->transfer_buffer, 878c2ecf20Sopenharmony_ci urb->actual_length); 888c2ecf20Sopenharmony_ci if (nfcmrvl_nci_recv_frame(drv_data->priv, skb) < 0) 898c2ecf20Sopenharmony_ci nfc_err(&drv_data->udev->dev, 908c2ecf20Sopenharmony_ci "corrupted Rx packet\n"); 918c2ecf20Sopenharmony_ci } 928c2ecf20Sopenharmony_ci } 938c2ecf20Sopenharmony_ci 948c2ecf20Sopenharmony_ci if (!test_bit(NFCMRVL_USB_BULK_RUNNING, &drv_data->flags)) 958c2ecf20Sopenharmony_ci return; 968c2ecf20Sopenharmony_ci 978c2ecf20Sopenharmony_ci usb_anchor_urb(urb, &drv_data->bulk_anchor); 988c2ecf20Sopenharmony_ci usb_mark_last_busy(drv_data->udev); 998c2ecf20Sopenharmony_ci 1008c2ecf20Sopenharmony_ci err = usb_submit_urb(urb, GFP_ATOMIC); 1018c2ecf20Sopenharmony_ci if (err) { 1028c2ecf20Sopenharmony_ci /* -EPERM: urb is being killed; 1038c2ecf20Sopenharmony_ci * -ENODEV: device got disconnected 1048c2ecf20Sopenharmony_ci */ 1058c2ecf20Sopenharmony_ci if (err != -EPERM && err != -ENODEV) 1068c2ecf20Sopenharmony_ci nfc_err(&drv_data->udev->dev, 1078c2ecf20Sopenharmony_ci "urb %p failed to resubmit (%d)\n", urb, -err); 1088c2ecf20Sopenharmony_ci usb_unanchor_urb(urb); 1098c2ecf20Sopenharmony_ci } 1108c2ecf20Sopenharmony_ci} 1118c2ecf20Sopenharmony_ci 1128c2ecf20Sopenharmony_cistatic int 1138c2ecf20Sopenharmony_cinfcmrvl_submit_bulk_urb(struct nfcmrvl_usb_drv_data *drv_data, gfp_t mem_flags) 1148c2ecf20Sopenharmony_ci{ 1158c2ecf20Sopenharmony_ci struct urb *urb; 1168c2ecf20Sopenharmony_ci unsigned char *buf; 1178c2ecf20Sopenharmony_ci unsigned int pipe; 1188c2ecf20Sopenharmony_ci int err, size = NFCMRVL_NCI_MAX_EVENT_SIZE; 1198c2ecf20Sopenharmony_ci 1208c2ecf20Sopenharmony_ci if (!drv_data->bulk_rx_ep) 1218c2ecf20Sopenharmony_ci return -ENODEV; 1228c2ecf20Sopenharmony_ci 1238c2ecf20Sopenharmony_ci urb = usb_alloc_urb(0, mem_flags); 1248c2ecf20Sopenharmony_ci if (!urb) 1258c2ecf20Sopenharmony_ci return -ENOMEM; 1268c2ecf20Sopenharmony_ci 1278c2ecf20Sopenharmony_ci buf = kmalloc(size, mem_flags); 1288c2ecf20Sopenharmony_ci if (!buf) { 1298c2ecf20Sopenharmony_ci usb_free_urb(urb); 1308c2ecf20Sopenharmony_ci return -ENOMEM; 1318c2ecf20Sopenharmony_ci } 1328c2ecf20Sopenharmony_ci 1338c2ecf20Sopenharmony_ci pipe = usb_rcvbulkpipe(drv_data->udev, 1348c2ecf20Sopenharmony_ci drv_data->bulk_rx_ep->bEndpointAddress); 1358c2ecf20Sopenharmony_ci 1368c2ecf20Sopenharmony_ci usb_fill_bulk_urb(urb, drv_data->udev, pipe, buf, size, 1378c2ecf20Sopenharmony_ci nfcmrvl_bulk_complete, drv_data); 1388c2ecf20Sopenharmony_ci 1398c2ecf20Sopenharmony_ci urb->transfer_flags |= URB_FREE_BUFFER; 1408c2ecf20Sopenharmony_ci 1418c2ecf20Sopenharmony_ci usb_mark_last_busy(drv_data->udev); 1428c2ecf20Sopenharmony_ci usb_anchor_urb(urb, &drv_data->bulk_anchor); 1438c2ecf20Sopenharmony_ci 1448c2ecf20Sopenharmony_ci err = usb_submit_urb(urb, mem_flags); 1458c2ecf20Sopenharmony_ci if (err) { 1468c2ecf20Sopenharmony_ci if (err != -EPERM && err != -ENODEV) 1478c2ecf20Sopenharmony_ci nfc_err(&drv_data->udev->dev, 1488c2ecf20Sopenharmony_ci "urb %p submission failed (%d)\n", urb, -err); 1498c2ecf20Sopenharmony_ci usb_unanchor_urb(urb); 1508c2ecf20Sopenharmony_ci } 1518c2ecf20Sopenharmony_ci 1528c2ecf20Sopenharmony_ci usb_free_urb(urb); 1538c2ecf20Sopenharmony_ci 1548c2ecf20Sopenharmony_ci return err; 1558c2ecf20Sopenharmony_ci} 1568c2ecf20Sopenharmony_ci 1578c2ecf20Sopenharmony_cistatic void nfcmrvl_tx_complete(struct urb *urb) 1588c2ecf20Sopenharmony_ci{ 1598c2ecf20Sopenharmony_ci struct sk_buff *skb = urb->context; 1608c2ecf20Sopenharmony_ci struct nci_dev *ndev = (struct nci_dev *)skb->dev; 1618c2ecf20Sopenharmony_ci struct nfcmrvl_private *priv = nci_get_drvdata(ndev); 1628c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = priv->drv_data; 1638c2ecf20Sopenharmony_ci unsigned long flags; 1648c2ecf20Sopenharmony_ci 1658c2ecf20Sopenharmony_ci nfc_info(priv->dev, "urb %p status %d count %d\n", 1668c2ecf20Sopenharmony_ci urb, urb->status, urb->actual_length); 1678c2ecf20Sopenharmony_ci 1688c2ecf20Sopenharmony_ci spin_lock_irqsave(&drv_data->txlock, flags); 1698c2ecf20Sopenharmony_ci drv_data->tx_in_flight--; 1708c2ecf20Sopenharmony_ci spin_unlock_irqrestore(&drv_data->txlock, flags); 1718c2ecf20Sopenharmony_ci 1728c2ecf20Sopenharmony_ci kfree(urb->setup_packet); 1738c2ecf20Sopenharmony_ci kfree_skb(skb); 1748c2ecf20Sopenharmony_ci} 1758c2ecf20Sopenharmony_ci 1768c2ecf20Sopenharmony_cistatic int nfcmrvl_usb_nci_open(struct nfcmrvl_private *priv) 1778c2ecf20Sopenharmony_ci{ 1788c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = priv->drv_data; 1798c2ecf20Sopenharmony_ci int err; 1808c2ecf20Sopenharmony_ci 1818c2ecf20Sopenharmony_ci err = usb_autopm_get_interface(drv_data->intf); 1828c2ecf20Sopenharmony_ci if (err) 1838c2ecf20Sopenharmony_ci return err; 1848c2ecf20Sopenharmony_ci 1858c2ecf20Sopenharmony_ci drv_data->intf->needs_remote_wakeup = 1; 1868c2ecf20Sopenharmony_ci 1878c2ecf20Sopenharmony_ci err = nfcmrvl_submit_bulk_urb(drv_data, GFP_KERNEL); 1888c2ecf20Sopenharmony_ci if (err) 1898c2ecf20Sopenharmony_ci goto failed; 1908c2ecf20Sopenharmony_ci 1918c2ecf20Sopenharmony_ci set_bit(NFCMRVL_USB_BULK_RUNNING, &drv_data->flags); 1928c2ecf20Sopenharmony_ci nfcmrvl_submit_bulk_urb(drv_data, GFP_KERNEL); 1938c2ecf20Sopenharmony_ci 1948c2ecf20Sopenharmony_ci usb_autopm_put_interface(drv_data->intf); 1958c2ecf20Sopenharmony_ci return 0; 1968c2ecf20Sopenharmony_ci 1978c2ecf20Sopenharmony_cifailed: 1988c2ecf20Sopenharmony_ci usb_autopm_put_interface(drv_data->intf); 1998c2ecf20Sopenharmony_ci return err; 2008c2ecf20Sopenharmony_ci} 2018c2ecf20Sopenharmony_ci 2028c2ecf20Sopenharmony_cistatic void nfcmrvl_usb_stop_traffic(struct nfcmrvl_usb_drv_data *drv_data) 2038c2ecf20Sopenharmony_ci{ 2048c2ecf20Sopenharmony_ci usb_kill_anchored_urbs(&drv_data->bulk_anchor); 2058c2ecf20Sopenharmony_ci} 2068c2ecf20Sopenharmony_ci 2078c2ecf20Sopenharmony_cistatic int nfcmrvl_usb_nci_close(struct nfcmrvl_private *priv) 2088c2ecf20Sopenharmony_ci{ 2098c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = priv->drv_data; 2108c2ecf20Sopenharmony_ci int err; 2118c2ecf20Sopenharmony_ci 2128c2ecf20Sopenharmony_ci cancel_work_sync(&drv_data->waker); 2138c2ecf20Sopenharmony_ci 2148c2ecf20Sopenharmony_ci clear_bit(NFCMRVL_USB_BULK_RUNNING, &drv_data->flags); 2158c2ecf20Sopenharmony_ci 2168c2ecf20Sopenharmony_ci nfcmrvl_usb_stop_traffic(drv_data); 2178c2ecf20Sopenharmony_ci usb_kill_anchored_urbs(&drv_data->tx_anchor); 2188c2ecf20Sopenharmony_ci err = usb_autopm_get_interface(drv_data->intf); 2198c2ecf20Sopenharmony_ci if (err) 2208c2ecf20Sopenharmony_ci goto failed; 2218c2ecf20Sopenharmony_ci 2228c2ecf20Sopenharmony_ci drv_data->intf->needs_remote_wakeup = 0; 2238c2ecf20Sopenharmony_ci usb_autopm_put_interface(drv_data->intf); 2248c2ecf20Sopenharmony_ci 2258c2ecf20Sopenharmony_cifailed: 2268c2ecf20Sopenharmony_ci usb_scuttle_anchored_urbs(&drv_data->deferred); 2278c2ecf20Sopenharmony_ci return 0; 2288c2ecf20Sopenharmony_ci} 2298c2ecf20Sopenharmony_ci 2308c2ecf20Sopenharmony_cistatic int nfcmrvl_usb_nci_send(struct nfcmrvl_private *priv, 2318c2ecf20Sopenharmony_ci struct sk_buff *skb) 2328c2ecf20Sopenharmony_ci{ 2338c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = priv->drv_data; 2348c2ecf20Sopenharmony_ci struct urb *urb; 2358c2ecf20Sopenharmony_ci unsigned int pipe; 2368c2ecf20Sopenharmony_ci int err; 2378c2ecf20Sopenharmony_ci 2388c2ecf20Sopenharmony_ci if (!drv_data->bulk_tx_ep) 2398c2ecf20Sopenharmony_ci return -ENODEV; 2408c2ecf20Sopenharmony_ci 2418c2ecf20Sopenharmony_ci urb = usb_alloc_urb(0, GFP_ATOMIC); 2428c2ecf20Sopenharmony_ci if (!urb) 2438c2ecf20Sopenharmony_ci return -ENOMEM; 2448c2ecf20Sopenharmony_ci 2458c2ecf20Sopenharmony_ci pipe = usb_sndbulkpipe(drv_data->udev, 2468c2ecf20Sopenharmony_ci drv_data->bulk_tx_ep->bEndpointAddress); 2478c2ecf20Sopenharmony_ci 2488c2ecf20Sopenharmony_ci usb_fill_bulk_urb(urb, drv_data->udev, pipe, skb->data, skb->len, 2498c2ecf20Sopenharmony_ci nfcmrvl_tx_complete, skb); 2508c2ecf20Sopenharmony_ci 2518c2ecf20Sopenharmony_ci err = nfcmrvl_inc_tx(drv_data); 2528c2ecf20Sopenharmony_ci if (err) { 2538c2ecf20Sopenharmony_ci usb_anchor_urb(urb, &drv_data->deferred); 2548c2ecf20Sopenharmony_ci schedule_work(&drv_data->waker); 2558c2ecf20Sopenharmony_ci err = 0; 2568c2ecf20Sopenharmony_ci goto done; 2578c2ecf20Sopenharmony_ci } 2588c2ecf20Sopenharmony_ci 2598c2ecf20Sopenharmony_ci usb_anchor_urb(urb, &drv_data->tx_anchor); 2608c2ecf20Sopenharmony_ci 2618c2ecf20Sopenharmony_ci err = usb_submit_urb(urb, GFP_ATOMIC); 2628c2ecf20Sopenharmony_ci if (err) { 2638c2ecf20Sopenharmony_ci if (err != -EPERM && err != -ENODEV) 2648c2ecf20Sopenharmony_ci nfc_err(&drv_data->udev->dev, 2658c2ecf20Sopenharmony_ci "urb %p submission failed (%d)\n", urb, -err); 2668c2ecf20Sopenharmony_ci kfree(urb->setup_packet); 2678c2ecf20Sopenharmony_ci usb_unanchor_urb(urb); 2688c2ecf20Sopenharmony_ci } else { 2698c2ecf20Sopenharmony_ci usb_mark_last_busy(drv_data->udev); 2708c2ecf20Sopenharmony_ci } 2718c2ecf20Sopenharmony_ci 2728c2ecf20Sopenharmony_cidone: 2738c2ecf20Sopenharmony_ci usb_free_urb(urb); 2748c2ecf20Sopenharmony_ci return err; 2758c2ecf20Sopenharmony_ci} 2768c2ecf20Sopenharmony_ci 2778c2ecf20Sopenharmony_cistatic struct nfcmrvl_if_ops usb_ops = { 2788c2ecf20Sopenharmony_ci .nci_open = nfcmrvl_usb_nci_open, 2798c2ecf20Sopenharmony_ci .nci_close = nfcmrvl_usb_nci_close, 2808c2ecf20Sopenharmony_ci .nci_send = nfcmrvl_usb_nci_send, 2818c2ecf20Sopenharmony_ci}; 2828c2ecf20Sopenharmony_ci 2838c2ecf20Sopenharmony_cistatic void nfcmrvl_waker(struct work_struct *work) 2848c2ecf20Sopenharmony_ci{ 2858c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = 2868c2ecf20Sopenharmony_ci container_of(work, struct nfcmrvl_usb_drv_data, waker); 2878c2ecf20Sopenharmony_ci int err; 2888c2ecf20Sopenharmony_ci 2898c2ecf20Sopenharmony_ci err = usb_autopm_get_interface(drv_data->intf); 2908c2ecf20Sopenharmony_ci if (err) 2918c2ecf20Sopenharmony_ci return; 2928c2ecf20Sopenharmony_ci 2938c2ecf20Sopenharmony_ci usb_autopm_put_interface(drv_data->intf); 2948c2ecf20Sopenharmony_ci} 2958c2ecf20Sopenharmony_ci 2968c2ecf20Sopenharmony_cistatic int nfcmrvl_probe(struct usb_interface *intf, 2978c2ecf20Sopenharmony_ci const struct usb_device_id *id) 2988c2ecf20Sopenharmony_ci{ 2998c2ecf20Sopenharmony_ci struct usb_endpoint_descriptor *ep_desc; 3008c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data; 3018c2ecf20Sopenharmony_ci struct nfcmrvl_private *priv; 3028c2ecf20Sopenharmony_ci int i; 3038c2ecf20Sopenharmony_ci struct usb_device *udev = interface_to_usbdev(intf); 3048c2ecf20Sopenharmony_ci struct nfcmrvl_platform_data config; 3058c2ecf20Sopenharmony_ci 3068c2ecf20Sopenharmony_ci /* No configuration for USB */ 3078c2ecf20Sopenharmony_ci memset(&config, 0, sizeof(config)); 3088c2ecf20Sopenharmony_ci config.reset_n_io = -EINVAL; 3098c2ecf20Sopenharmony_ci 3108c2ecf20Sopenharmony_ci nfc_info(&udev->dev, "intf %p id %p\n", intf, id); 3118c2ecf20Sopenharmony_ci 3128c2ecf20Sopenharmony_ci drv_data = devm_kzalloc(&intf->dev, sizeof(*drv_data), GFP_KERNEL); 3138c2ecf20Sopenharmony_ci if (!drv_data) 3148c2ecf20Sopenharmony_ci return -ENOMEM; 3158c2ecf20Sopenharmony_ci 3168c2ecf20Sopenharmony_ci for (i = 0; i < intf->cur_altsetting->desc.bNumEndpoints; i++) { 3178c2ecf20Sopenharmony_ci ep_desc = &intf->cur_altsetting->endpoint[i].desc; 3188c2ecf20Sopenharmony_ci 3198c2ecf20Sopenharmony_ci if (!drv_data->bulk_tx_ep && 3208c2ecf20Sopenharmony_ci usb_endpoint_is_bulk_out(ep_desc)) { 3218c2ecf20Sopenharmony_ci drv_data->bulk_tx_ep = ep_desc; 3228c2ecf20Sopenharmony_ci continue; 3238c2ecf20Sopenharmony_ci } 3248c2ecf20Sopenharmony_ci 3258c2ecf20Sopenharmony_ci if (!drv_data->bulk_rx_ep && 3268c2ecf20Sopenharmony_ci usb_endpoint_is_bulk_in(ep_desc)) { 3278c2ecf20Sopenharmony_ci drv_data->bulk_rx_ep = ep_desc; 3288c2ecf20Sopenharmony_ci continue; 3298c2ecf20Sopenharmony_ci } 3308c2ecf20Sopenharmony_ci } 3318c2ecf20Sopenharmony_ci 3328c2ecf20Sopenharmony_ci if (!drv_data->bulk_tx_ep || !drv_data->bulk_rx_ep) 3338c2ecf20Sopenharmony_ci return -ENODEV; 3348c2ecf20Sopenharmony_ci 3358c2ecf20Sopenharmony_ci drv_data->udev = udev; 3368c2ecf20Sopenharmony_ci drv_data->intf = intf; 3378c2ecf20Sopenharmony_ci 3388c2ecf20Sopenharmony_ci INIT_WORK(&drv_data->waker, nfcmrvl_waker); 3398c2ecf20Sopenharmony_ci spin_lock_init(&drv_data->txlock); 3408c2ecf20Sopenharmony_ci 3418c2ecf20Sopenharmony_ci init_usb_anchor(&drv_data->tx_anchor); 3428c2ecf20Sopenharmony_ci init_usb_anchor(&drv_data->bulk_anchor); 3438c2ecf20Sopenharmony_ci init_usb_anchor(&drv_data->deferred); 3448c2ecf20Sopenharmony_ci 3458c2ecf20Sopenharmony_ci priv = nfcmrvl_nci_register_dev(NFCMRVL_PHY_USB, drv_data, &usb_ops, 3468c2ecf20Sopenharmony_ci &intf->dev, &config); 3478c2ecf20Sopenharmony_ci if (IS_ERR(priv)) 3488c2ecf20Sopenharmony_ci return PTR_ERR(priv); 3498c2ecf20Sopenharmony_ci 3508c2ecf20Sopenharmony_ci drv_data->priv = priv; 3518c2ecf20Sopenharmony_ci drv_data->priv->support_fw_dnld = false; 3528c2ecf20Sopenharmony_ci 3538c2ecf20Sopenharmony_ci usb_set_intfdata(intf, drv_data); 3548c2ecf20Sopenharmony_ci 3558c2ecf20Sopenharmony_ci return 0; 3568c2ecf20Sopenharmony_ci} 3578c2ecf20Sopenharmony_ci 3588c2ecf20Sopenharmony_cistatic void nfcmrvl_disconnect(struct usb_interface *intf) 3598c2ecf20Sopenharmony_ci{ 3608c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = usb_get_intfdata(intf); 3618c2ecf20Sopenharmony_ci 3628c2ecf20Sopenharmony_ci if (!drv_data) 3638c2ecf20Sopenharmony_ci return; 3648c2ecf20Sopenharmony_ci 3658c2ecf20Sopenharmony_ci nfc_info(&drv_data->udev->dev, "intf %p\n", intf); 3668c2ecf20Sopenharmony_ci 3678c2ecf20Sopenharmony_ci nfcmrvl_nci_unregister_dev(drv_data->priv); 3688c2ecf20Sopenharmony_ci 3698c2ecf20Sopenharmony_ci usb_set_intfdata(drv_data->intf, NULL); 3708c2ecf20Sopenharmony_ci} 3718c2ecf20Sopenharmony_ci 3728c2ecf20Sopenharmony_ci#ifdef CONFIG_PM 3738c2ecf20Sopenharmony_cistatic int nfcmrvl_suspend(struct usb_interface *intf, pm_message_t message) 3748c2ecf20Sopenharmony_ci{ 3758c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = usb_get_intfdata(intf); 3768c2ecf20Sopenharmony_ci 3778c2ecf20Sopenharmony_ci nfc_info(&drv_data->udev->dev, "intf %p\n", intf); 3788c2ecf20Sopenharmony_ci 3798c2ecf20Sopenharmony_ci if (drv_data->suspend_count++) 3808c2ecf20Sopenharmony_ci return 0; 3818c2ecf20Sopenharmony_ci 3828c2ecf20Sopenharmony_ci spin_lock_irq(&drv_data->txlock); 3838c2ecf20Sopenharmony_ci if (!(PMSG_IS_AUTO(message) && drv_data->tx_in_flight)) { 3848c2ecf20Sopenharmony_ci set_bit(NFCMRVL_USB_SUSPENDING, &drv_data->flags); 3858c2ecf20Sopenharmony_ci spin_unlock_irq(&drv_data->txlock); 3868c2ecf20Sopenharmony_ci } else { 3878c2ecf20Sopenharmony_ci spin_unlock_irq(&drv_data->txlock); 3888c2ecf20Sopenharmony_ci drv_data->suspend_count--; 3898c2ecf20Sopenharmony_ci return -EBUSY; 3908c2ecf20Sopenharmony_ci } 3918c2ecf20Sopenharmony_ci 3928c2ecf20Sopenharmony_ci nfcmrvl_usb_stop_traffic(drv_data); 3938c2ecf20Sopenharmony_ci usb_kill_anchored_urbs(&drv_data->tx_anchor); 3948c2ecf20Sopenharmony_ci 3958c2ecf20Sopenharmony_ci return 0; 3968c2ecf20Sopenharmony_ci} 3978c2ecf20Sopenharmony_ci 3988c2ecf20Sopenharmony_cistatic void nfcmrvl_play_deferred(struct nfcmrvl_usb_drv_data *drv_data) 3998c2ecf20Sopenharmony_ci{ 4008c2ecf20Sopenharmony_ci struct urb *urb; 4018c2ecf20Sopenharmony_ci int err; 4028c2ecf20Sopenharmony_ci 4038c2ecf20Sopenharmony_ci while ((urb = usb_get_from_anchor(&drv_data->deferred))) { 4048c2ecf20Sopenharmony_ci usb_anchor_urb(urb, &drv_data->tx_anchor); 4058c2ecf20Sopenharmony_ci 4068c2ecf20Sopenharmony_ci err = usb_submit_urb(urb, GFP_ATOMIC); 4078c2ecf20Sopenharmony_ci if (err) { 4088c2ecf20Sopenharmony_ci kfree(urb->setup_packet); 4098c2ecf20Sopenharmony_ci usb_unanchor_urb(urb); 4108c2ecf20Sopenharmony_ci usb_free_urb(urb); 4118c2ecf20Sopenharmony_ci break; 4128c2ecf20Sopenharmony_ci } 4138c2ecf20Sopenharmony_ci 4148c2ecf20Sopenharmony_ci drv_data->tx_in_flight++; 4158c2ecf20Sopenharmony_ci usb_free_urb(urb); 4168c2ecf20Sopenharmony_ci } 4178c2ecf20Sopenharmony_ci 4188c2ecf20Sopenharmony_ci /* Cleanup the rest deferred urbs. */ 4198c2ecf20Sopenharmony_ci while ((urb = usb_get_from_anchor(&drv_data->deferred))) { 4208c2ecf20Sopenharmony_ci kfree(urb->setup_packet); 4218c2ecf20Sopenharmony_ci usb_free_urb(urb); 4228c2ecf20Sopenharmony_ci } 4238c2ecf20Sopenharmony_ci} 4248c2ecf20Sopenharmony_ci 4258c2ecf20Sopenharmony_cistatic int nfcmrvl_resume(struct usb_interface *intf) 4268c2ecf20Sopenharmony_ci{ 4278c2ecf20Sopenharmony_ci struct nfcmrvl_usb_drv_data *drv_data = usb_get_intfdata(intf); 4288c2ecf20Sopenharmony_ci int err = 0; 4298c2ecf20Sopenharmony_ci 4308c2ecf20Sopenharmony_ci nfc_info(&drv_data->udev->dev, "intf %p\n", intf); 4318c2ecf20Sopenharmony_ci 4328c2ecf20Sopenharmony_ci if (--drv_data->suspend_count) 4338c2ecf20Sopenharmony_ci return 0; 4348c2ecf20Sopenharmony_ci 4358c2ecf20Sopenharmony_ci if (!test_bit(NFCMRVL_NCI_RUNNING, &drv_data->flags)) 4368c2ecf20Sopenharmony_ci goto done; 4378c2ecf20Sopenharmony_ci 4388c2ecf20Sopenharmony_ci if (test_bit(NFCMRVL_USB_BULK_RUNNING, &drv_data->flags)) { 4398c2ecf20Sopenharmony_ci err = nfcmrvl_submit_bulk_urb(drv_data, GFP_NOIO); 4408c2ecf20Sopenharmony_ci if (err) { 4418c2ecf20Sopenharmony_ci clear_bit(NFCMRVL_USB_BULK_RUNNING, &drv_data->flags); 4428c2ecf20Sopenharmony_ci goto failed; 4438c2ecf20Sopenharmony_ci } 4448c2ecf20Sopenharmony_ci 4458c2ecf20Sopenharmony_ci nfcmrvl_submit_bulk_urb(drv_data, GFP_NOIO); 4468c2ecf20Sopenharmony_ci } 4478c2ecf20Sopenharmony_ci 4488c2ecf20Sopenharmony_ci spin_lock_irq(&drv_data->txlock); 4498c2ecf20Sopenharmony_ci nfcmrvl_play_deferred(drv_data); 4508c2ecf20Sopenharmony_ci clear_bit(NFCMRVL_USB_SUSPENDING, &drv_data->flags); 4518c2ecf20Sopenharmony_ci spin_unlock_irq(&drv_data->txlock); 4528c2ecf20Sopenharmony_ci 4538c2ecf20Sopenharmony_ci return 0; 4548c2ecf20Sopenharmony_ci 4558c2ecf20Sopenharmony_cifailed: 4568c2ecf20Sopenharmony_ci usb_scuttle_anchored_urbs(&drv_data->deferred); 4578c2ecf20Sopenharmony_cidone: 4588c2ecf20Sopenharmony_ci spin_lock_irq(&drv_data->txlock); 4598c2ecf20Sopenharmony_ci clear_bit(NFCMRVL_USB_SUSPENDING, &drv_data->flags); 4608c2ecf20Sopenharmony_ci spin_unlock_irq(&drv_data->txlock); 4618c2ecf20Sopenharmony_ci 4628c2ecf20Sopenharmony_ci return err; 4638c2ecf20Sopenharmony_ci} 4648c2ecf20Sopenharmony_ci#endif 4658c2ecf20Sopenharmony_ci 4668c2ecf20Sopenharmony_cistatic struct usb_driver nfcmrvl_usb_driver = { 4678c2ecf20Sopenharmony_ci .name = "nfcmrvl", 4688c2ecf20Sopenharmony_ci .probe = nfcmrvl_probe, 4698c2ecf20Sopenharmony_ci .disconnect = nfcmrvl_disconnect, 4708c2ecf20Sopenharmony_ci#ifdef CONFIG_PM 4718c2ecf20Sopenharmony_ci .suspend = nfcmrvl_suspend, 4728c2ecf20Sopenharmony_ci .resume = nfcmrvl_resume, 4738c2ecf20Sopenharmony_ci .reset_resume = nfcmrvl_resume, 4748c2ecf20Sopenharmony_ci#endif 4758c2ecf20Sopenharmony_ci .id_table = nfcmrvl_table, 4768c2ecf20Sopenharmony_ci .supports_autosuspend = 1, 4778c2ecf20Sopenharmony_ci .disable_hub_initiated_lpm = 1, 4788c2ecf20Sopenharmony_ci .soft_unbind = 1, 4798c2ecf20Sopenharmony_ci}; 4808c2ecf20Sopenharmony_cimodule_usb_driver(nfcmrvl_usb_driver); 4818c2ecf20Sopenharmony_ci 4828c2ecf20Sopenharmony_ciMODULE_AUTHOR("Marvell International Ltd."); 4838c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Marvell NFC-over-USB driver"); 4848c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 485