xref: /freebsd/sys/contrib/dev/athk/ath10k/usb.c (revision 07724ba62b4c432ea04dce9465a5ab6e2c3f5a0d)
1da8fa4e3SBjoern A. Zeeb // SPDX-License-Identifier: ISC
2da8fa4e3SBjoern A. Zeeb /*
3da8fa4e3SBjoern A. Zeeb  * Copyright (c) 2007-2011 Atheros Communications Inc.
4da8fa4e3SBjoern A. Zeeb  * Copyright (c) 2011-2012,2017 Qualcomm Atheros, Inc.
5da8fa4e3SBjoern A. Zeeb  * Copyright (c) 2016-2017 Erik Stromdahl <erik.stromdahl@gmail.com>
6da8fa4e3SBjoern A. Zeeb  */
7da8fa4e3SBjoern A. Zeeb 
8da8fa4e3SBjoern A. Zeeb #include <linux/module.h>
9da8fa4e3SBjoern A. Zeeb #include <linux/usb.h>
10da8fa4e3SBjoern A. Zeeb 
11da8fa4e3SBjoern A. Zeeb #include "debug.h"
12da8fa4e3SBjoern A. Zeeb #include "core.h"
13da8fa4e3SBjoern A. Zeeb #include "bmi.h"
14da8fa4e3SBjoern A. Zeeb #include "hif.h"
15da8fa4e3SBjoern A. Zeeb #include "htc.h"
16da8fa4e3SBjoern A. Zeeb #include "usb.h"
17da8fa4e3SBjoern A. Zeeb 
18da8fa4e3SBjoern A. Zeeb static void ath10k_usb_post_recv_transfers(struct ath10k *ar,
19da8fa4e3SBjoern A. Zeeb 					   struct ath10k_usb_pipe *recv_pipe);
20da8fa4e3SBjoern A. Zeeb 
21da8fa4e3SBjoern A. Zeeb /* inlined helper functions */
22da8fa4e3SBjoern A. Zeeb 
23da8fa4e3SBjoern A. Zeeb static inline enum ath10k_htc_ep_id
eid_from_htc_hdr(struct ath10k_htc_hdr * htc_hdr)24da8fa4e3SBjoern A. Zeeb eid_from_htc_hdr(struct ath10k_htc_hdr *htc_hdr)
25da8fa4e3SBjoern A. Zeeb {
26da8fa4e3SBjoern A. Zeeb 	return (enum ath10k_htc_ep_id)htc_hdr->eid;
27da8fa4e3SBjoern A. Zeeb }
28da8fa4e3SBjoern A. Zeeb 
is_trailer_only_msg(struct ath10k_htc_hdr * htc_hdr)29da8fa4e3SBjoern A. Zeeb static inline bool is_trailer_only_msg(struct ath10k_htc_hdr *htc_hdr)
30da8fa4e3SBjoern A. Zeeb {
31da8fa4e3SBjoern A. Zeeb 	return __le16_to_cpu(htc_hdr->len) == htc_hdr->trailer_len;
32da8fa4e3SBjoern A. Zeeb }
33da8fa4e3SBjoern A. Zeeb 
34da8fa4e3SBjoern A. Zeeb /* pipe/urb operations */
35da8fa4e3SBjoern A. Zeeb static struct ath10k_urb_context *
ath10k_usb_alloc_urb_from_pipe(struct ath10k_usb_pipe * pipe)36da8fa4e3SBjoern A. Zeeb ath10k_usb_alloc_urb_from_pipe(struct ath10k_usb_pipe *pipe)
37da8fa4e3SBjoern A. Zeeb {
38da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context = NULL;
39da8fa4e3SBjoern A. Zeeb 	unsigned long flags;
40da8fa4e3SBjoern A. Zeeb 
41da8fa4e3SBjoern A. Zeeb 	/* bail if this pipe is not initialized */
42da8fa4e3SBjoern A. Zeeb 	if (!pipe->ar_usb)
43da8fa4e3SBjoern A. Zeeb 		return NULL;
44da8fa4e3SBjoern A. Zeeb 
45da8fa4e3SBjoern A. Zeeb 	spin_lock_irqsave(&pipe->ar_usb->cs_lock, flags);
46da8fa4e3SBjoern A. Zeeb 	if (!list_empty(&pipe->urb_list_head)) {
47da8fa4e3SBjoern A. Zeeb 		urb_context = list_first_entry(&pipe->urb_list_head,
48da8fa4e3SBjoern A. Zeeb 					       struct ath10k_urb_context, link);
49da8fa4e3SBjoern A. Zeeb 		list_del(&urb_context->link);
50da8fa4e3SBjoern A. Zeeb 		pipe->urb_cnt--;
51da8fa4e3SBjoern A. Zeeb 	}
52da8fa4e3SBjoern A. Zeeb 	spin_unlock_irqrestore(&pipe->ar_usb->cs_lock, flags);
53da8fa4e3SBjoern A. Zeeb 
54da8fa4e3SBjoern A. Zeeb 	return urb_context;
55da8fa4e3SBjoern A. Zeeb }
56da8fa4e3SBjoern A. Zeeb 
ath10k_usb_free_urb_to_pipe(struct ath10k_usb_pipe * pipe,struct ath10k_urb_context * urb_context)57da8fa4e3SBjoern A. Zeeb static void ath10k_usb_free_urb_to_pipe(struct ath10k_usb_pipe *pipe,
58da8fa4e3SBjoern A. Zeeb 					struct ath10k_urb_context *urb_context)
59da8fa4e3SBjoern A. Zeeb {
60da8fa4e3SBjoern A. Zeeb 	unsigned long flags;
61da8fa4e3SBjoern A. Zeeb 
62da8fa4e3SBjoern A. Zeeb 	/* bail if this pipe is not initialized */
63da8fa4e3SBjoern A. Zeeb 	if (!pipe->ar_usb)
64da8fa4e3SBjoern A. Zeeb 		return;
65da8fa4e3SBjoern A. Zeeb 
66da8fa4e3SBjoern A. Zeeb 	spin_lock_irqsave(&pipe->ar_usb->cs_lock, flags);
67da8fa4e3SBjoern A. Zeeb 
68da8fa4e3SBjoern A. Zeeb 	pipe->urb_cnt++;
69da8fa4e3SBjoern A. Zeeb 	list_add(&urb_context->link, &pipe->urb_list_head);
70da8fa4e3SBjoern A. Zeeb 
71da8fa4e3SBjoern A. Zeeb 	spin_unlock_irqrestore(&pipe->ar_usb->cs_lock, flags);
72da8fa4e3SBjoern A. Zeeb }
73da8fa4e3SBjoern A. Zeeb 
ath10k_usb_cleanup_recv_urb(struct ath10k_urb_context * urb_context)74da8fa4e3SBjoern A. Zeeb static void ath10k_usb_cleanup_recv_urb(struct ath10k_urb_context *urb_context)
75da8fa4e3SBjoern A. Zeeb {
76da8fa4e3SBjoern A. Zeeb 	dev_kfree_skb(urb_context->skb);
77da8fa4e3SBjoern A. Zeeb 	urb_context->skb = NULL;
78da8fa4e3SBjoern A. Zeeb 
79da8fa4e3SBjoern A. Zeeb 	ath10k_usb_free_urb_to_pipe(urb_context->pipe, urb_context);
80da8fa4e3SBjoern A. Zeeb }
81da8fa4e3SBjoern A. Zeeb 
ath10k_usb_free_pipe_resources(struct ath10k * ar,struct ath10k_usb_pipe * pipe)82da8fa4e3SBjoern A. Zeeb static void ath10k_usb_free_pipe_resources(struct ath10k *ar,
83da8fa4e3SBjoern A. Zeeb 					   struct ath10k_usb_pipe *pipe)
84da8fa4e3SBjoern A. Zeeb {
85da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context;
86da8fa4e3SBjoern A. Zeeb 
87da8fa4e3SBjoern A. Zeeb 	if (!pipe->ar_usb) {
88da8fa4e3SBjoern A. Zeeb 		/* nothing allocated for this pipe */
89da8fa4e3SBjoern A. Zeeb 		return;
90da8fa4e3SBjoern A. Zeeb 	}
91da8fa4e3SBjoern A. Zeeb 
92da8fa4e3SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_USB,
93da8fa4e3SBjoern A. Zeeb 		   "usb free resources lpipe %d hpipe 0x%x urbs %d avail %d\n",
94da8fa4e3SBjoern A. Zeeb 		   pipe->logical_pipe_num, pipe->usb_pipe_handle,
95da8fa4e3SBjoern A. Zeeb 		   pipe->urb_alloc, pipe->urb_cnt);
96da8fa4e3SBjoern A. Zeeb 
97da8fa4e3SBjoern A. Zeeb 	if (pipe->urb_alloc != pipe->urb_cnt) {
98da8fa4e3SBjoern A. Zeeb 		ath10k_dbg(ar, ATH10K_DBG_USB,
99da8fa4e3SBjoern A. Zeeb 			   "usb urb leak lpipe %d hpipe 0x%x urbs %d avail %d\n",
100da8fa4e3SBjoern A. Zeeb 			   pipe->logical_pipe_num, pipe->usb_pipe_handle,
101da8fa4e3SBjoern A. Zeeb 			   pipe->urb_alloc, pipe->urb_cnt);
102da8fa4e3SBjoern A. Zeeb 	}
103da8fa4e3SBjoern A. Zeeb 
104da8fa4e3SBjoern A. Zeeb 	for (;;) {
105da8fa4e3SBjoern A. Zeeb 		urb_context = ath10k_usb_alloc_urb_from_pipe(pipe);
106da8fa4e3SBjoern A. Zeeb 
107da8fa4e3SBjoern A. Zeeb 		if (!urb_context)
108da8fa4e3SBjoern A. Zeeb 			break;
109da8fa4e3SBjoern A. Zeeb 
110da8fa4e3SBjoern A. Zeeb 		kfree(urb_context);
111da8fa4e3SBjoern A. Zeeb 	}
112da8fa4e3SBjoern A. Zeeb }
113da8fa4e3SBjoern A. Zeeb 
ath10k_usb_cleanup_pipe_resources(struct ath10k * ar)114da8fa4e3SBjoern A. Zeeb static void ath10k_usb_cleanup_pipe_resources(struct ath10k *ar)
115da8fa4e3SBjoern A. Zeeb {
116da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
117da8fa4e3SBjoern A. Zeeb 	int i;
118da8fa4e3SBjoern A. Zeeb 
119da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < ATH10K_USB_PIPE_MAX; i++)
120da8fa4e3SBjoern A. Zeeb 		ath10k_usb_free_pipe_resources(ar, &ar_usb->pipes[i]);
121da8fa4e3SBjoern A. Zeeb }
122da8fa4e3SBjoern A. Zeeb 
123da8fa4e3SBjoern A. Zeeb /* hif usb rx/tx completion functions */
124da8fa4e3SBjoern A. Zeeb 
ath10k_usb_recv_complete(struct urb * urb)125da8fa4e3SBjoern A. Zeeb static void ath10k_usb_recv_complete(struct urb *urb)
126da8fa4e3SBjoern A. Zeeb {
127da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context = urb->context;
128da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe = urb_context->pipe;
129da8fa4e3SBjoern A. Zeeb 	struct ath10k *ar = pipe->ar_usb->ar;
130da8fa4e3SBjoern A. Zeeb 	struct sk_buff *skb;
131da8fa4e3SBjoern A. Zeeb 	int status = 0;
132da8fa4e3SBjoern A. Zeeb 
133da8fa4e3SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
134da8fa4e3SBjoern A. Zeeb 		   "usb recv pipe %d stat %d len %d urb 0x%pK\n",
135da8fa4e3SBjoern A. Zeeb 		   pipe->logical_pipe_num, urb->status, urb->actual_length,
136da8fa4e3SBjoern A. Zeeb 		   urb);
137da8fa4e3SBjoern A. Zeeb 
138da8fa4e3SBjoern A. Zeeb 	if (urb->status != 0) {
139da8fa4e3SBjoern A. Zeeb 		status = -EIO;
140da8fa4e3SBjoern A. Zeeb 		switch (urb->status) {
141da8fa4e3SBjoern A. Zeeb 		case -ECONNRESET:
142da8fa4e3SBjoern A. Zeeb 		case -ENOENT:
143da8fa4e3SBjoern A. Zeeb 		case -ESHUTDOWN:
144da8fa4e3SBjoern A. Zeeb 			/* no need to spew these errors when device
145da8fa4e3SBjoern A. Zeeb 			 * removed or urb killed due to driver shutdown
146da8fa4e3SBjoern A. Zeeb 			 */
147da8fa4e3SBjoern A. Zeeb 			status = -ECANCELED;
148da8fa4e3SBjoern A. Zeeb 			break;
149da8fa4e3SBjoern A. Zeeb 		default:
150da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
151da8fa4e3SBjoern A. Zeeb 				   "usb recv pipe %d ep 0x%2.2x failed: %d\n",
152da8fa4e3SBjoern A. Zeeb 				   pipe->logical_pipe_num,
153da8fa4e3SBjoern A. Zeeb 				   pipe->ep_address, urb->status);
154da8fa4e3SBjoern A. Zeeb 			break;
155da8fa4e3SBjoern A. Zeeb 		}
156da8fa4e3SBjoern A. Zeeb 		goto cleanup_recv_urb;
157da8fa4e3SBjoern A. Zeeb 	}
158da8fa4e3SBjoern A. Zeeb 
159da8fa4e3SBjoern A. Zeeb 	if (urb->actual_length == 0)
160da8fa4e3SBjoern A. Zeeb 		goto cleanup_recv_urb;
161da8fa4e3SBjoern A. Zeeb 
162da8fa4e3SBjoern A. Zeeb 	skb = urb_context->skb;
163da8fa4e3SBjoern A. Zeeb 
164da8fa4e3SBjoern A. Zeeb 	/* we are going to pass it up */
165da8fa4e3SBjoern A. Zeeb 	urb_context->skb = NULL;
166da8fa4e3SBjoern A. Zeeb 	skb_put(skb, urb->actual_length);
167da8fa4e3SBjoern A. Zeeb 
168da8fa4e3SBjoern A. Zeeb 	/* note: queue implements a lock */
169da8fa4e3SBjoern A. Zeeb 	skb_queue_tail(&pipe->io_comp_queue, skb);
170da8fa4e3SBjoern A. Zeeb 	schedule_work(&pipe->io_complete_work);
171da8fa4e3SBjoern A. Zeeb 
172da8fa4e3SBjoern A. Zeeb cleanup_recv_urb:
173da8fa4e3SBjoern A. Zeeb 	ath10k_usb_cleanup_recv_urb(urb_context);
174da8fa4e3SBjoern A. Zeeb 
175da8fa4e3SBjoern A. Zeeb 	if (status == 0 &&
176da8fa4e3SBjoern A. Zeeb 	    pipe->urb_cnt >= pipe->urb_cnt_thresh) {
177da8fa4e3SBjoern A. Zeeb 		/* our free urbs are piling up, post more transfers */
178da8fa4e3SBjoern A. Zeeb 		ath10k_usb_post_recv_transfers(ar, pipe);
179da8fa4e3SBjoern A. Zeeb 	}
180da8fa4e3SBjoern A. Zeeb }
181da8fa4e3SBjoern A. Zeeb 
ath10k_usb_transmit_complete(struct urb * urb)182da8fa4e3SBjoern A. Zeeb static void ath10k_usb_transmit_complete(struct urb *urb)
183da8fa4e3SBjoern A. Zeeb {
184da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context = urb->context;
185da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe = urb_context->pipe;
186da8fa4e3SBjoern A. Zeeb 	struct ath10k *ar = pipe->ar_usb->ar;
187da8fa4e3SBjoern A. Zeeb 	struct sk_buff *skb;
188da8fa4e3SBjoern A. Zeeb 
189da8fa4e3SBjoern A. Zeeb 	if (urb->status != 0) {
190da8fa4e3SBjoern A. Zeeb 		ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
191da8fa4e3SBjoern A. Zeeb 			   "pipe: %d, failed:%d\n",
192da8fa4e3SBjoern A. Zeeb 			   pipe->logical_pipe_num, urb->status);
193da8fa4e3SBjoern A. Zeeb 	}
194da8fa4e3SBjoern A. Zeeb 
195da8fa4e3SBjoern A. Zeeb 	skb = urb_context->skb;
196da8fa4e3SBjoern A. Zeeb 	urb_context->skb = NULL;
197da8fa4e3SBjoern A. Zeeb 	ath10k_usb_free_urb_to_pipe(urb_context->pipe, urb_context);
198da8fa4e3SBjoern A. Zeeb 
199da8fa4e3SBjoern A. Zeeb 	/* note: queue implements a lock */
200da8fa4e3SBjoern A. Zeeb 	skb_queue_tail(&pipe->io_comp_queue, skb);
201da8fa4e3SBjoern A. Zeeb 	schedule_work(&pipe->io_complete_work);
202da8fa4e3SBjoern A. Zeeb }
203da8fa4e3SBjoern A. Zeeb 
204da8fa4e3SBjoern A. Zeeb /* pipe operations */
ath10k_usb_post_recv_transfers(struct ath10k * ar,struct ath10k_usb_pipe * recv_pipe)205da8fa4e3SBjoern A. Zeeb static void ath10k_usb_post_recv_transfers(struct ath10k *ar,
206da8fa4e3SBjoern A. Zeeb 					   struct ath10k_usb_pipe *recv_pipe)
207da8fa4e3SBjoern A. Zeeb {
208da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context;
209da8fa4e3SBjoern A. Zeeb 	struct urb *urb;
210da8fa4e3SBjoern A. Zeeb 	int usb_status;
211da8fa4e3SBjoern A. Zeeb 
212da8fa4e3SBjoern A. Zeeb 	for (;;) {
213da8fa4e3SBjoern A. Zeeb 		urb_context = ath10k_usb_alloc_urb_from_pipe(recv_pipe);
214da8fa4e3SBjoern A. Zeeb 		if (!urb_context)
215da8fa4e3SBjoern A. Zeeb 			break;
216da8fa4e3SBjoern A. Zeeb 
217da8fa4e3SBjoern A. Zeeb 		urb_context->skb = dev_alloc_skb(ATH10K_USB_RX_BUFFER_SIZE);
218da8fa4e3SBjoern A. Zeeb 		if (!urb_context->skb)
219da8fa4e3SBjoern A. Zeeb 			goto err;
220da8fa4e3SBjoern A. Zeeb 
221da8fa4e3SBjoern A. Zeeb 		urb = usb_alloc_urb(0, GFP_ATOMIC);
222da8fa4e3SBjoern A. Zeeb 		if (!urb)
223da8fa4e3SBjoern A. Zeeb 			goto err;
224da8fa4e3SBjoern A. Zeeb 
225da8fa4e3SBjoern A. Zeeb 		usb_fill_bulk_urb(urb,
226da8fa4e3SBjoern A. Zeeb 				  recv_pipe->ar_usb->udev,
227da8fa4e3SBjoern A. Zeeb 				  recv_pipe->usb_pipe_handle,
228da8fa4e3SBjoern A. Zeeb 				  urb_context->skb->data,
229da8fa4e3SBjoern A. Zeeb 				  ATH10K_USB_RX_BUFFER_SIZE,
230da8fa4e3SBjoern A. Zeeb 				  ath10k_usb_recv_complete, urb_context);
231da8fa4e3SBjoern A. Zeeb 
232da8fa4e3SBjoern A. Zeeb 		ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
233da8fa4e3SBjoern A. Zeeb 			   "usb bulk recv submit %d 0x%x ep 0x%2.2x len %d buf 0x%pK\n",
234da8fa4e3SBjoern A. Zeeb 			   recv_pipe->logical_pipe_num,
235da8fa4e3SBjoern A. Zeeb 			   recv_pipe->usb_pipe_handle, recv_pipe->ep_address,
236da8fa4e3SBjoern A. Zeeb 			   ATH10K_USB_RX_BUFFER_SIZE, urb_context->skb);
237da8fa4e3SBjoern A. Zeeb 
238da8fa4e3SBjoern A. Zeeb 		usb_anchor_urb(urb, &recv_pipe->urb_submitted);
239da8fa4e3SBjoern A. Zeeb 		usb_status = usb_submit_urb(urb, GFP_ATOMIC);
240da8fa4e3SBjoern A. Zeeb 
241da8fa4e3SBjoern A. Zeeb 		if (usb_status) {
242da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
243da8fa4e3SBjoern A. Zeeb 				   "usb bulk recv failed: %d\n",
244da8fa4e3SBjoern A. Zeeb 				   usb_status);
245da8fa4e3SBjoern A. Zeeb 			usb_unanchor_urb(urb);
246da8fa4e3SBjoern A. Zeeb 			usb_free_urb(urb);
247da8fa4e3SBjoern A. Zeeb 			goto err;
248da8fa4e3SBjoern A. Zeeb 		}
249da8fa4e3SBjoern A. Zeeb 		usb_free_urb(urb);
250da8fa4e3SBjoern A. Zeeb 	}
251da8fa4e3SBjoern A. Zeeb 
252da8fa4e3SBjoern A. Zeeb 	return;
253da8fa4e3SBjoern A. Zeeb 
254da8fa4e3SBjoern A. Zeeb err:
255da8fa4e3SBjoern A. Zeeb 	ath10k_usb_cleanup_recv_urb(urb_context);
256da8fa4e3SBjoern A. Zeeb }
257da8fa4e3SBjoern A. Zeeb 
ath10k_usb_flush_all(struct ath10k * ar)258da8fa4e3SBjoern A. Zeeb static void ath10k_usb_flush_all(struct ath10k *ar)
259da8fa4e3SBjoern A. Zeeb {
260da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
261da8fa4e3SBjoern A. Zeeb 	int i;
262da8fa4e3SBjoern A. Zeeb 
263da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < ATH10K_USB_PIPE_MAX; i++) {
264da8fa4e3SBjoern A. Zeeb 		if (ar_usb->pipes[i].ar_usb) {
265da8fa4e3SBjoern A. Zeeb 			usb_kill_anchored_urbs(&ar_usb->pipes[i].urb_submitted);
266da8fa4e3SBjoern A. Zeeb 			cancel_work_sync(&ar_usb->pipes[i].io_complete_work);
267da8fa4e3SBjoern A. Zeeb 		}
268da8fa4e3SBjoern A. Zeeb 	}
269da8fa4e3SBjoern A. Zeeb }
270da8fa4e3SBjoern A. Zeeb 
ath10k_usb_start_recv_pipes(struct ath10k * ar)271da8fa4e3SBjoern A. Zeeb static void ath10k_usb_start_recv_pipes(struct ath10k *ar)
272da8fa4e3SBjoern A. Zeeb {
273da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
274da8fa4e3SBjoern A. Zeeb 
275da8fa4e3SBjoern A. Zeeb 	ar_usb->pipes[ATH10K_USB_PIPE_RX_DATA].urb_cnt_thresh = 1;
276da8fa4e3SBjoern A. Zeeb 
277da8fa4e3SBjoern A. Zeeb 	ath10k_usb_post_recv_transfers(ar,
278da8fa4e3SBjoern A. Zeeb 				       &ar_usb->pipes[ATH10K_USB_PIPE_RX_DATA]);
279da8fa4e3SBjoern A. Zeeb }
280da8fa4e3SBjoern A. Zeeb 
ath10k_usb_tx_complete(struct ath10k * ar,struct sk_buff * skb)281da8fa4e3SBjoern A. Zeeb static void ath10k_usb_tx_complete(struct ath10k *ar, struct sk_buff *skb)
282da8fa4e3SBjoern A. Zeeb {
283da8fa4e3SBjoern A. Zeeb 	struct ath10k_htc_hdr *htc_hdr;
284da8fa4e3SBjoern A. Zeeb 	struct ath10k_htc_ep *ep;
285da8fa4e3SBjoern A. Zeeb 
286da8fa4e3SBjoern A. Zeeb 	htc_hdr = (struct ath10k_htc_hdr *)skb->data;
287da8fa4e3SBjoern A. Zeeb 	ep = &ar->htc.endpoint[htc_hdr->eid];
288da8fa4e3SBjoern A. Zeeb 	ath10k_htc_notify_tx_completion(ep, skb);
289da8fa4e3SBjoern A. Zeeb 	/* The TX complete handler now owns the skb... */
290da8fa4e3SBjoern A. Zeeb }
291da8fa4e3SBjoern A. Zeeb 
ath10k_usb_rx_complete(struct ath10k * ar,struct sk_buff * skb)292da8fa4e3SBjoern A. Zeeb static void ath10k_usb_rx_complete(struct ath10k *ar, struct sk_buff *skb)
293da8fa4e3SBjoern A. Zeeb {
294da8fa4e3SBjoern A. Zeeb 	struct ath10k_htc *htc = &ar->htc;
295da8fa4e3SBjoern A. Zeeb 	struct ath10k_htc_hdr *htc_hdr;
296da8fa4e3SBjoern A. Zeeb 	enum ath10k_htc_ep_id eid;
297da8fa4e3SBjoern A. Zeeb 	struct ath10k_htc_ep *ep;
298da8fa4e3SBjoern A. Zeeb 	u16 payload_len;
299da8fa4e3SBjoern A. Zeeb 	u8 *trailer;
300da8fa4e3SBjoern A. Zeeb 	int ret;
301da8fa4e3SBjoern A. Zeeb 
302da8fa4e3SBjoern A. Zeeb 	htc_hdr = (struct ath10k_htc_hdr *)skb->data;
303da8fa4e3SBjoern A. Zeeb 	eid = eid_from_htc_hdr(htc_hdr);
304da8fa4e3SBjoern A. Zeeb 	ep = &ar->htc.endpoint[eid];
305da8fa4e3SBjoern A. Zeeb 
306da8fa4e3SBjoern A. Zeeb 	if (ep->service_id == 0) {
307da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "ep %d is not connected\n", eid);
308da8fa4e3SBjoern A. Zeeb 		goto out_free_skb;
309da8fa4e3SBjoern A. Zeeb 	}
310da8fa4e3SBjoern A. Zeeb 
311da8fa4e3SBjoern A. Zeeb 	payload_len = le16_to_cpu(htc_hdr->len);
312da8fa4e3SBjoern A. Zeeb 	if (!payload_len) {
313da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "zero length frame received, firmware crashed?\n");
314da8fa4e3SBjoern A. Zeeb 		goto out_free_skb;
315da8fa4e3SBjoern A. Zeeb 	}
316da8fa4e3SBjoern A. Zeeb 
317da8fa4e3SBjoern A. Zeeb 	if (payload_len < htc_hdr->trailer_len) {
318da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "malformed frame received, firmware crashed?\n");
319da8fa4e3SBjoern A. Zeeb 		goto out_free_skb;
320da8fa4e3SBjoern A. Zeeb 	}
321da8fa4e3SBjoern A. Zeeb 
322da8fa4e3SBjoern A. Zeeb 	if (htc_hdr->flags & ATH10K_HTC_FLAG_TRAILER_PRESENT) {
323da8fa4e3SBjoern A. Zeeb 		trailer = skb->data + sizeof(*htc_hdr) + payload_len -
324da8fa4e3SBjoern A. Zeeb 			  htc_hdr->trailer_len;
325da8fa4e3SBjoern A. Zeeb 
326da8fa4e3SBjoern A. Zeeb 		ret = ath10k_htc_process_trailer(htc,
327da8fa4e3SBjoern A. Zeeb 						 trailer,
328da8fa4e3SBjoern A. Zeeb 						 htc_hdr->trailer_len,
329da8fa4e3SBjoern A. Zeeb 						 eid,
330da8fa4e3SBjoern A. Zeeb 						 NULL,
331da8fa4e3SBjoern A. Zeeb 						 NULL);
332da8fa4e3SBjoern A. Zeeb 		if (ret)
333da8fa4e3SBjoern A. Zeeb 			goto out_free_skb;
334da8fa4e3SBjoern A. Zeeb 
335da8fa4e3SBjoern A. Zeeb 		if (is_trailer_only_msg(htc_hdr))
336da8fa4e3SBjoern A. Zeeb 			goto out_free_skb;
337da8fa4e3SBjoern A. Zeeb 
338da8fa4e3SBjoern A. Zeeb 		/* strip off the trailer from the skb since it should not
339da8fa4e3SBjoern A. Zeeb 		 * be passed on to upper layers
340da8fa4e3SBjoern A. Zeeb 		 */
341da8fa4e3SBjoern A. Zeeb 		skb_trim(skb, skb->len - htc_hdr->trailer_len);
342da8fa4e3SBjoern A. Zeeb 	}
343da8fa4e3SBjoern A. Zeeb 
344da8fa4e3SBjoern A. Zeeb 	skb_pull(skb, sizeof(*htc_hdr));
345da8fa4e3SBjoern A. Zeeb 	ep->ep_ops.ep_rx_complete(ar, skb);
346da8fa4e3SBjoern A. Zeeb 	/* The RX complete handler now owns the skb... */
347da8fa4e3SBjoern A. Zeeb 
348*07724ba6SBjoern A. Zeeb 	if (test_bit(ATH10K_FLAG_CORE_REGISTERED, &ar->dev_flags)) {
349*07724ba6SBjoern A. Zeeb 		local_bh_disable();
350*07724ba6SBjoern A. Zeeb 		napi_schedule(&ar->napi);
351*07724ba6SBjoern A. Zeeb 		local_bh_enable();
352*07724ba6SBjoern A. Zeeb 	}
353*07724ba6SBjoern A. Zeeb 
354da8fa4e3SBjoern A. Zeeb 	return;
355da8fa4e3SBjoern A. Zeeb 
356da8fa4e3SBjoern A. Zeeb out_free_skb:
357da8fa4e3SBjoern A. Zeeb 	dev_kfree_skb(skb);
358da8fa4e3SBjoern A. Zeeb }
359da8fa4e3SBjoern A. Zeeb 
ath10k_usb_io_comp_work(struct work_struct * work)360da8fa4e3SBjoern A. Zeeb static void ath10k_usb_io_comp_work(struct work_struct *work)
361da8fa4e3SBjoern A. Zeeb {
362da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe = container_of(work,
363da8fa4e3SBjoern A. Zeeb 						    struct ath10k_usb_pipe,
364da8fa4e3SBjoern A. Zeeb 						    io_complete_work);
365da8fa4e3SBjoern A. Zeeb 	struct ath10k *ar = pipe->ar_usb->ar;
366da8fa4e3SBjoern A. Zeeb 	struct sk_buff *skb;
367da8fa4e3SBjoern A. Zeeb 
368da8fa4e3SBjoern A. Zeeb 	while ((skb = skb_dequeue(&pipe->io_comp_queue))) {
369da8fa4e3SBjoern A. Zeeb 		if (pipe->flags & ATH10K_USB_PIPE_FLAG_TX)
370da8fa4e3SBjoern A. Zeeb 			ath10k_usb_tx_complete(ar, skb);
371da8fa4e3SBjoern A. Zeeb 		else
372da8fa4e3SBjoern A. Zeeb 			ath10k_usb_rx_complete(ar, skb);
373da8fa4e3SBjoern A. Zeeb 	}
374da8fa4e3SBjoern A. Zeeb }
375da8fa4e3SBjoern A. Zeeb 
376da8fa4e3SBjoern A. Zeeb #define ATH10K_USB_MAX_DIAG_CMD (sizeof(struct ath10k_usb_ctrl_diag_cmd_write))
377da8fa4e3SBjoern A. Zeeb #define ATH10K_USB_MAX_DIAG_RESP (sizeof(struct ath10k_usb_ctrl_diag_resp_read))
378da8fa4e3SBjoern A. Zeeb 
ath10k_usb_destroy(struct ath10k * ar)379da8fa4e3SBjoern A. Zeeb static void ath10k_usb_destroy(struct ath10k *ar)
380da8fa4e3SBjoern A. Zeeb {
381da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
382da8fa4e3SBjoern A. Zeeb 
383da8fa4e3SBjoern A. Zeeb 	ath10k_usb_flush_all(ar);
384da8fa4e3SBjoern A. Zeeb 	ath10k_usb_cleanup_pipe_resources(ar);
385da8fa4e3SBjoern A. Zeeb 	usb_set_intfdata(ar_usb->interface, NULL);
386da8fa4e3SBjoern A. Zeeb 
387da8fa4e3SBjoern A. Zeeb 	kfree(ar_usb->diag_cmd_buffer);
388da8fa4e3SBjoern A. Zeeb 	kfree(ar_usb->diag_resp_buffer);
389da8fa4e3SBjoern A. Zeeb }
390da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_start(struct ath10k * ar)391da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_start(struct ath10k *ar)
392da8fa4e3SBjoern A. Zeeb {
393da8fa4e3SBjoern A. Zeeb 	int i;
394da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
395da8fa4e3SBjoern A. Zeeb 
396*07724ba6SBjoern A. Zeeb 	ath10k_core_napi_enable(ar);
397da8fa4e3SBjoern A. Zeeb 	ath10k_usb_start_recv_pipes(ar);
398da8fa4e3SBjoern A. Zeeb 
399da8fa4e3SBjoern A. Zeeb 	/* set the TX resource avail threshold for each TX pipe */
400da8fa4e3SBjoern A. Zeeb 	for (i = ATH10K_USB_PIPE_TX_CTRL;
401da8fa4e3SBjoern A. Zeeb 	     i <= ATH10K_USB_PIPE_TX_DATA_HP; i++) {
402da8fa4e3SBjoern A. Zeeb 		ar_usb->pipes[i].urb_cnt_thresh =
403da8fa4e3SBjoern A. Zeeb 		    ar_usb->pipes[i].urb_alloc / 2;
404da8fa4e3SBjoern A. Zeeb 	}
405da8fa4e3SBjoern A. Zeeb 
406da8fa4e3SBjoern A. Zeeb 	return 0;
407da8fa4e3SBjoern A. Zeeb }
408da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_tx_sg(struct ath10k * ar,u8 pipe_id,struct ath10k_hif_sg_item * items,int n_items)409da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_tx_sg(struct ath10k *ar, u8 pipe_id,
410da8fa4e3SBjoern A. Zeeb 				struct ath10k_hif_sg_item *items, int n_items)
411da8fa4e3SBjoern A. Zeeb {
412da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
413da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe = &ar_usb->pipes[pipe_id];
414da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context;
415da8fa4e3SBjoern A. Zeeb 	struct sk_buff *skb;
416da8fa4e3SBjoern A. Zeeb 	struct urb *urb;
417da8fa4e3SBjoern A. Zeeb 	int ret, i;
418da8fa4e3SBjoern A. Zeeb 
419da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < n_items; i++) {
420da8fa4e3SBjoern A. Zeeb 		urb_context = ath10k_usb_alloc_urb_from_pipe(pipe);
421da8fa4e3SBjoern A. Zeeb 		if (!urb_context) {
422da8fa4e3SBjoern A. Zeeb 			ret = -ENOMEM;
423da8fa4e3SBjoern A. Zeeb 			goto err;
424da8fa4e3SBjoern A. Zeeb 		}
425da8fa4e3SBjoern A. Zeeb 
426da8fa4e3SBjoern A. Zeeb 		skb = items[i].transfer_context;
427da8fa4e3SBjoern A. Zeeb 		urb_context->skb = skb;
428da8fa4e3SBjoern A. Zeeb 
429da8fa4e3SBjoern A. Zeeb 		urb = usb_alloc_urb(0, GFP_ATOMIC);
430da8fa4e3SBjoern A. Zeeb 		if (!urb) {
431da8fa4e3SBjoern A. Zeeb 			ret = -ENOMEM;
432da8fa4e3SBjoern A. Zeeb 			goto err_free_urb_to_pipe;
433da8fa4e3SBjoern A. Zeeb 		}
434da8fa4e3SBjoern A. Zeeb 
435da8fa4e3SBjoern A. Zeeb 		usb_fill_bulk_urb(urb,
436da8fa4e3SBjoern A. Zeeb 				  ar_usb->udev,
437da8fa4e3SBjoern A. Zeeb 				  pipe->usb_pipe_handle,
438da8fa4e3SBjoern A. Zeeb 				  skb->data,
439da8fa4e3SBjoern A. Zeeb 				  skb->len,
440da8fa4e3SBjoern A. Zeeb 				  ath10k_usb_transmit_complete, urb_context);
441da8fa4e3SBjoern A. Zeeb 
442da8fa4e3SBjoern A. Zeeb 		if (!(skb->len % pipe->max_packet_size)) {
443da8fa4e3SBjoern A. Zeeb 			/* hit a max packet boundary on this pipe */
444da8fa4e3SBjoern A. Zeeb 			urb->transfer_flags |= URB_ZERO_PACKET;
445da8fa4e3SBjoern A. Zeeb 		}
446da8fa4e3SBjoern A. Zeeb 
447da8fa4e3SBjoern A. Zeeb 		usb_anchor_urb(urb, &pipe->urb_submitted);
448da8fa4e3SBjoern A. Zeeb 		ret = usb_submit_urb(urb, GFP_ATOMIC);
449da8fa4e3SBjoern A. Zeeb 		if (ret) {
450da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB_BULK,
451da8fa4e3SBjoern A. Zeeb 				   "usb bulk transmit failed: %d\n", ret);
452da8fa4e3SBjoern A. Zeeb 			usb_unanchor_urb(urb);
453da8fa4e3SBjoern A. Zeeb 			usb_free_urb(urb);
454da8fa4e3SBjoern A. Zeeb 			ret = -EINVAL;
455da8fa4e3SBjoern A. Zeeb 			goto err_free_urb_to_pipe;
456da8fa4e3SBjoern A. Zeeb 		}
457da8fa4e3SBjoern A. Zeeb 
458da8fa4e3SBjoern A. Zeeb 		usb_free_urb(urb);
459da8fa4e3SBjoern A. Zeeb 	}
460da8fa4e3SBjoern A. Zeeb 
461da8fa4e3SBjoern A. Zeeb 	return 0;
462da8fa4e3SBjoern A. Zeeb 
463da8fa4e3SBjoern A. Zeeb err_free_urb_to_pipe:
464da8fa4e3SBjoern A. Zeeb 	ath10k_usb_free_urb_to_pipe(urb_context->pipe, urb_context);
465da8fa4e3SBjoern A. Zeeb err:
466da8fa4e3SBjoern A. Zeeb 	return ret;
467da8fa4e3SBjoern A. Zeeb }
468da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_stop(struct ath10k * ar)469da8fa4e3SBjoern A. Zeeb static void ath10k_usb_hif_stop(struct ath10k *ar)
470da8fa4e3SBjoern A. Zeeb {
471da8fa4e3SBjoern A. Zeeb 	ath10k_usb_flush_all(ar);
472*07724ba6SBjoern A. Zeeb 	ath10k_core_napi_sync_disable(ar);
473da8fa4e3SBjoern A. Zeeb }
474da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_get_free_queue_number(struct ath10k * ar,u8 pipe_id)475da8fa4e3SBjoern A. Zeeb static u16 ath10k_usb_hif_get_free_queue_number(struct ath10k *ar, u8 pipe_id)
476da8fa4e3SBjoern A. Zeeb {
477da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
478da8fa4e3SBjoern A. Zeeb 
479da8fa4e3SBjoern A. Zeeb 	return ar_usb->pipes[pipe_id].urb_cnt;
480da8fa4e3SBjoern A. Zeeb }
481da8fa4e3SBjoern A. Zeeb 
ath10k_usb_submit_ctrl_out(struct ath10k * ar,u8 req,u16 value,u16 index,void * data,u32 size)482da8fa4e3SBjoern A. Zeeb static int ath10k_usb_submit_ctrl_out(struct ath10k *ar,
483da8fa4e3SBjoern A. Zeeb 				      u8 req, u16 value, u16 index, void *data,
484da8fa4e3SBjoern A. Zeeb 				      u32 size)
485da8fa4e3SBjoern A. Zeeb {
486da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
487da8fa4e3SBjoern A. Zeeb 	u8 *buf = NULL;
488da8fa4e3SBjoern A. Zeeb 	int ret;
489da8fa4e3SBjoern A. Zeeb 
490da8fa4e3SBjoern A. Zeeb 	if (size > 0) {
491da8fa4e3SBjoern A. Zeeb 		buf = kmemdup(data, size, GFP_KERNEL);
492da8fa4e3SBjoern A. Zeeb 		if (!buf)
493da8fa4e3SBjoern A. Zeeb 			return -ENOMEM;
494da8fa4e3SBjoern A. Zeeb 	}
495da8fa4e3SBjoern A. Zeeb 
496da8fa4e3SBjoern A. Zeeb 	/* note: if successful returns number of bytes transferred */
497da8fa4e3SBjoern A. Zeeb 	ret = usb_control_msg(ar_usb->udev,
498da8fa4e3SBjoern A. Zeeb 			      usb_sndctrlpipe(ar_usb->udev, 0),
499da8fa4e3SBjoern A. Zeeb 			      req,
500da8fa4e3SBjoern A. Zeeb 			      USB_DIR_OUT | USB_TYPE_VENDOR |
501da8fa4e3SBjoern A. Zeeb 			      USB_RECIP_DEVICE, value, index, buf,
502da8fa4e3SBjoern A. Zeeb 			      size, 1000);
503da8fa4e3SBjoern A. Zeeb 
504da8fa4e3SBjoern A. Zeeb 	if (ret < 0) {
505da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "Failed to submit usb control message: %d\n",
506da8fa4e3SBjoern A. Zeeb 			    ret);
507da8fa4e3SBjoern A. Zeeb 		kfree(buf);
508da8fa4e3SBjoern A. Zeeb 		return ret;
509da8fa4e3SBjoern A. Zeeb 	}
510da8fa4e3SBjoern A. Zeeb 
511da8fa4e3SBjoern A. Zeeb 	kfree(buf);
512da8fa4e3SBjoern A. Zeeb 
513da8fa4e3SBjoern A. Zeeb 	return 0;
514da8fa4e3SBjoern A. Zeeb }
515da8fa4e3SBjoern A. Zeeb 
ath10k_usb_submit_ctrl_in(struct ath10k * ar,u8 req,u16 value,u16 index,void * data,u32 size)516da8fa4e3SBjoern A. Zeeb static int ath10k_usb_submit_ctrl_in(struct ath10k *ar,
517da8fa4e3SBjoern A. Zeeb 				     u8 req, u16 value, u16 index, void *data,
518da8fa4e3SBjoern A. Zeeb 				     u32 size)
519da8fa4e3SBjoern A. Zeeb {
520da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
521da8fa4e3SBjoern A. Zeeb 	u8 *buf = NULL;
522da8fa4e3SBjoern A. Zeeb 	int ret;
523da8fa4e3SBjoern A. Zeeb 
524da8fa4e3SBjoern A. Zeeb 	if (size > 0) {
525da8fa4e3SBjoern A. Zeeb 		buf = kmalloc(size, GFP_KERNEL);
526da8fa4e3SBjoern A. Zeeb 		if (!buf)
527da8fa4e3SBjoern A. Zeeb 			return -ENOMEM;
528da8fa4e3SBjoern A. Zeeb 	}
529da8fa4e3SBjoern A. Zeeb 
530da8fa4e3SBjoern A. Zeeb 	/* note: if successful returns number of bytes transferred */
531da8fa4e3SBjoern A. Zeeb 	ret = usb_control_msg(ar_usb->udev,
532da8fa4e3SBjoern A. Zeeb 			      usb_rcvctrlpipe(ar_usb->udev, 0),
533da8fa4e3SBjoern A. Zeeb 			      req,
534da8fa4e3SBjoern A. Zeeb 			      USB_DIR_IN | USB_TYPE_VENDOR |
535da8fa4e3SBjoern A. Zeeb 			      USB_RECIP_DEVICE, value, index, buf,
536da8fa4e3SBjoern A. Zeeb 			      size, 2000);
537da8fa4e3SBjoern A. Zeeb 
538da8fa4e3SBjoern A. Zeeb 	if (ret < 0) {
539da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "Failed to read usb control message: %d\n",
540da8fa4e3SBjoern A. Zeeb 			    ret);
541da8fa4e3SBjoern A. Zeeb 		kfree(buf);
542da8fa4e3SBjoern A. Zeeb 		return ret;
543da8fa4e3SBjoern A. Zeeb 	}
544da8fa4e3SBjoern A. Zeeb 
545da8fa4e3SBjoern A. Zeeb 	memcpy((u8 *)data, buf, size);
546da8fa4e3SBjoern A. Zeeb 
547da8fa4e3SBjoern A. Zeeb 	kfree(buf);
548da8fa4e3SBjoern A. Zeeb 
549da8fa4e3SBjoern A. Zeeb 	return 0;
550da8fa4e3SBjoern A. Zeeb }
551da8fa4e3SBjoern A. Zeeb 
ath10k_usb_ctrl_msg_exchange(struct ath10k * ar,u8 req_val,u8 * req_buf,u32 req_len,u8 resp_val,u8 * resp_buf,u32 * resp_len)552da8fa4e3SBjoern A. Zeeb static int ath10k_usb_ctrl_msg_exchange(struct ath10k *ar,
553da8fa4e3SBjoern A. Zeeb 					u8 req_val, u8 *req_buf, u32 req_len,
554da8fa4e3SBjoern A. Zeeb 					u8 resp_val, u8 *resp_buf,
555da8fa4e3SBjoern A. Zeeb 					u32 *resp_len)
556da8fa4e3SBjoern A. Zeeb {
557da8fa4e3SBjoern A. Zeeb 	int ret;
558da8fa4e3SBjoern A. Zeeb 
559da8fa4e3SBjoern A. Zeeb 	/* send command */
560da8fa4e3SBjoern A. Zeeb 	ret = ath10k_usb_submit_ctrl_out(ar, req_val, 0, 0,
561da8fa4e3SBjoern A. Zeeb 					 req_buf, req_len);
562da8fa4e3SBjoern A. Zeeb 	if (ret)
563da8fa4e3SBjoern A. Zeeb 		goto err;
564da8fa4e3SBjoern A. Zeeb 
565da8fa4e3SBjoern A. Zeeb 	/* get response */
566da8fa4e3SBjoern A. Zeeb 	if (resp_buf) {
567da8fa4e3SBjoern A. Zeeb 		ret = ath10k_usb_submit_ctrl_in(ar, resp_val, 0, 0,
568da8fa4e3SBjoern A. Zeeb 						resp_buf, *resp_len);
569da8fa4e3SBjoern A. Zeeb 		if (ret)
570da8fa4e3SBjoern A. Zeeb 			goto err;
571da8fa4e3SBjoern A. Zeeb 	}
572da8fa4e3SBjoern A. Zeeb 
573da8fa4e3SBjoern A. Zeeb 	return 0;
574da8fa4e3SBjoern A. Zeeb err:
575da8fa4e3SBjoern A. Zeeb 	return ret;
576da8fa4e3SBjoern A. Zeeb }
577da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_diag_read(struct ath10k * ar,u32 address,void * buf,size_t buf_len)578da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_diag_read(struct ath10k *ar, u32 address, void *buf,
579da8fa4e3SBjoern A. Zeeb 				    size_t buf_len)
580da8fa4e3SBjoern A. Zeeb {
581da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
582da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_ctrl_diag_cmd_read *cmd;
583da8fa4e3SBjoern A. Zeeb 	u32 resp_len;
584da8fa4e3SBjoern A. Zeeb 	int ret;
585da8fa4e3SBjoern A. Zeeb 
586da8fa4e3SBjoern A. Zeeb 	if (buf_len < sizeof(struct ath10k_usb_ctrl_diag_resp_read))
587da8fa4e3SBjoern A. Zeeb 		return -EINVAL;
588da8fa4e3SBjoern A. Zeeb 
589da8fa4e3SBjoern A. Zeeb 	cmd = (struct ath10k_usb_ctrl_diag_cmd_read *)ar_usb->diag_cmd_buffer;
590da8fa4e3SBjoern A. Zeeb 	memset(cmd, 0, sizeof(*cmd));
591da8fa4e3SBjoern A. Zeeb 	cmd->cmd = ATH10K_USB_CTRL_DIAG_CC_READ;
592da8fa4e3SBjoern A. Zeeb 	cmd->address = cpu_to_le32(address);
593da8fa4e3SBjoern A. Zeeb 	resp_len = sizeof(struct ath10k_usb_ctrl_diag_resp_read);
594da8fa4e3SBjoern A. Zeeb 
595da8fa4e3SBjoern A. Zeeb 	ret = ath10k_usb_ctrl_msg_exchange(ar,
596da8fa4e3SBjoern A. Zeeb 					   ATH10K_USB_CONTROL_REQ_DIAG_CMD,
597da8fa4e3SBjoern A. Zeeb 					   (u8 *)cmd,
598da8fa4e3SBjoern A. Zeeb 					   sizeof(*cmd),
599da8fa4e3SBjoern A. Zeeb 					   ATH10K_USB_CONTROL_REQ_DIAG_RESP,
600da8fa4e3SBjoern A. Zeeb 					   ar_usb->diag_resp_buffer, &resp_len);
601da8fa4e3SBjoern A. Zeeb 	if (ret)
602da8fa4e3SBjoern A. Zeeb 		return ret;
603da8fa4e3SBjoern A. Zeeb 
604da8fa4e3SBjoern A. Zeeb 	if (resp_len != sizeof(struct ath10k_usb_ctrl_diag_resp_read))
605da8fa4e3SBjoern A. Zeeb 		return -EMSGSIZE;
606da8fa4e3SBjoern A. Zeeb 
607da8fa4e3SBjoern A. Zeeb 	memcpy(buf, ar_usb->diag_resp_buffer,
608da8fa4e3SBjoern A. Zeeb 	       sizeof(struct ath10k_usb_ctrl_diag_resp_read));
609da8fa4e3SBjoern A. Zeeb 
610da8fa4e3SBjoern A. Zeeb 	return 0;
611da8fa4e3SBjoern A. Zeeb }
612da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_diag_write(struct ath10k * ar,u32 address,const void * data,int nbytes)613da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_diag_write(struct ath10k *ar, u32 address,
614da8fa4e3SBjoern A. Zeeb 				     const void *data, int nbytes)
615da8fa4e3SBjoern A. Zeeb {
616da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
617da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_ctrl_diag_cmd_write *cmd;
618da8fa4e3SBjoern A. Zeeb 	int ret;
619da8fa4e3SBjoern A. Zeeb 
620da8fa4e3SBjoern A. Zeeb 	if (nbytes != sizeof(cmd->value))
621da8fa4e3SBjoern A. Zeeb 		return -EINVAL;
622da8fa4e3SBjoern A. Zeeb 
623da8fa4e3SBjoern A. Zeeb 	cmd = (struct ath10k_usb_ctrl_diag_cmd_write *)ar_usb->diag_cmd_buffer;
624da8fa4e3SBjoern A. Zeeb 	memset(cmd, 0, sizeof(*cmd));
625da8fa4e3SBjoern A. Zeeb 	cmd->cmd = cpu_to_le32(ATH10K_USB_CTRL_DIAG_CC_WRITE);
626da8fa4e3SBjoern A. Zeeb 	cmd->address = cpu_to_le32(address);
627da8fa4e3SBjoern A. Zeeb 	memcpy(&cmd->value, data, nbytes);
628da8fa4e3SBjoern A. Zeeb 
629da8fa4e3SBjoern A. Zeeb 	ret = ath10k_usb_ctrl_msg_exchange(ar,
630da8fa4e3SBjoern A. Zeeb 					   ATH10K_USB_CONTROL_REQ_DIAG_CMD,
631da8fa4e3SBjoern A. Zeeb 					   (u8 *)cmd,
632da8fa4e3SBjoern A. Zeeb 					   sizeof(*cmd),
633da8fa4e3SBjoern A. Zeeb 					   0, NULL, NULL);
634da8fa4e3SBjoern A. Zeeb 	if (ret)
635da8fa4e3SBjoern A. Zeeb 		return ret;
636da8fa4e3SBjoern A. Zeeb 
637da8fa4e3SBjoern A. Zeeb 	return 0;
638da8fa4e3SBjoern A. Zeeb }
639da8fa4e3SBjoern A. Zeeb 
ath10k_usb_bmi_exchange_msg(struct ath10k * ar,void * req,u32 req_len,void * resp,u32 * resp_len)640da8fa4e3SBjoern A. Zeeb static int ath10k_usb_bmi_exchange_msg(struct ath10k *ar,
641da8fa4e3SBjoern A. Zeeb 				       void *req, u32 req_len,
642da8fa4e3SBjoern A. Zeeb 				       void *resp, u32 *resp_len)
643da8fa4e3SBjoern A. Zeeb {
644da8fa4e3SBjoern A. Zeeb 	int ret;
645da8fa4e3SBjoern A. Zeeb 
646da8fa4e3SBjoern A. Zeeb 	if (req) {
647da8fa4e3SBjoern A. Zeeb 		ret = ath10k_usb_submit_ctrl_out(ar,
648da8fa4e3SBjoern A. Zeeb 						 ATH10K_USB_CONTROL_REQ_SEND_BMI_CMD,
649da8fa4e3SBjoern A. Zeeb 						 0, 0, req, req_len);
650da8fa4e3SBjoern A. Zeeb 		if (ret) {
651da8fa4e3SBjoern A. Zeeb 			ath10k_warn(ar,
652da8fa4e3SBjoern A. Zeeb 				    "unable to send the bmi data to the device: %d\n",
653da8fa4e3SBjoern A. Zeeb 				    ret);
654da8fa4e3SBjoern A. Zeeb 			return ret;
655da8fa4e3SBjoern A. Zeeb 		}
656da8fa4e3SBjoern A. Zeeb 	}
657da8fa4e3SBjoern A. Zeeb 
658da8fa4e3SBjoern A. Zeeb 	if (resp) {
659da8fa4e3SBjoern A. Zeeb 		ret = ath10k_usb_submit_ctrl_in(ar,
660da8fa4e3SBjoern A. Zeeb 						ATH10K_USB_CONTROL_REQ_RECV_BMI_RESP,
661da8fa4e3SBjoern A. Zeeb 						0, 0, resp, *resp_len);
662da8fa4e3SBjoern A. Zeeb 		if (ret) {
663da8fa4e3SBjoern A. Zeeb 			ath10k_warn(ar,
664da8fa4e3SBjoern A. Zeeb 				    "Unable to read the bmi data from the device: %d\n",
665da8fa4e3SBjoern A. Zeeb 				    ret);
666da8fa4e3SBjoern A. Zeeb 			return ret;
667da8fa4e3SBjoern A. Zeeb 		}
668da8fa4e3SBjoern A. Zeeb 	}
669da8fa4e3SBjoern A. Zeeb 
670da8fa4e3SBjoern A. Zeeb 	return 0;
671da8fa4e3SBjoern A. Zeeb }
672da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_get_default_pipe(struct ath10k * ar,u8 * ul_pipe,u8 * dl_pipe)673da8fa4e3SBjoern A. Zeeb static void ath10k_usb_hif_get_default_pipe(struct ath10k *ar,
674da8fa4e3SBjoern A. Zeeb 					    u8 *ul_pipe, u8 *dl_pipe)
675da8fa4e3SBjoern A. Zeeb {
676da8fa4e3SBjoern A. Zeeb 	*ul_pipe = ATH10K_USB_PIPE_TX_CTRL;
677da8fa4e3SBjoern A. Zeeb 	*dl_pipe = ATH10K_USB_PIPE_RX_CTRL;
678da8fa4e3SBjoern A. Zeeb }
679da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_map_service_to_pipe(struct ath10k * ar,u16 svc_id,u8 * ul_pipe,u8 * dl_pipe)680da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_map_service_to_pipe(struct ath10k *ar, u16 svc_id,
681da8fa4e3SBjoern A. Zeeb 					      u8 *ul_pipe, u8 *dl_pipe)
682da8fa4e3SBjoern A. Zeeb {
683da8fa4e3SBjoern A. Zeeb 	switch (svc_id) {
684da8fa4e3SBjoern A. Zeeb 	case ATH10K_HTC_SVC_ID_RSVD_CTRL:
685da8fa4e3SBjoern A. Zeeb 	case ATH10K_HTC_SVC_ID_WMI_CONTROL:
686da8fa4e3SBjoern A. Zeeb 		*ul_pipe = ATH10K_USB_PIPE_TX_CTRL;
687da8fa4e3SBjoern A. Zeeb 		/* due to large control packets, shift to data pipe */
688da8fa4e3SBjoern A. Zeeb 		*dl_pipe = ATH10K_USB_PIPE_RX_DATA;
689da8fa4e3SBjoern A. Zeeb 		break;
690da8fa4e3SBjoern A. Zeeb 	case ATH10K_HTC_SVC_ID_HTT_DATA_MSG:
691da8fa4e3SBjoern A. Zeeb 		*ul_pipe = ATH10K_USB_PIPE_TX_DATA_LP;
692da8fa4e3SBjoern A. Zeeb 		/* Disable rxdata2 directly, it will be enabled
693da8fa4e3SBjoern A. Zeeb 		 * if FW enable rxdata2
694da8fa4e3SBjoern A. Zeeb 		 */
695da8fa4e3SBjoern A. Zeeb 		*dl_pipe = ATH10K_USB_PIPE_RX_DATA;
696da8fa4e3SBjoern A. Zeeb 		break;
697da8fa4e3SBjoern A. Zeeb 	default:
698da8fa4e3SBjoern A. Zeeb 		return -EPERM;
699da8fa4e3SBjoern A. Zeeb 	}
700da8fa4e3SBjoern A. Zeeb 
701da8fa4e3SBjoern A. Zeeb 	return 0;
702da8fa4e3SBjoern A. Zeeb }
703da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_power_up(struct ath10k * ar,enum ath10k_firmware_mode fw_mode)704da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_power_up(struct ath10k *ar,
705da8fa4e3SBjoern A. Zeeb 				   enum ath10k_firmware_mode fw_mode)
706da8fa4e3SBjoern A. Zeeb {
707da8fa4e3SBjoern A. Zeeb 	return 0;
708da8fa4e3SBjoern A. Zeeb }
709da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_power_down(struct ath10k * ar)710da8fa4e3SBjoern A. Zeeb static void ath10k_usb_hif_power_down(struct ath10k *ar)
711da8fa4e3SBjoern A. Zeeb {
712da8fa4e3SBjoern A. Zeeb 	ath10k_usb_flush_all(ar);
713da8fa4e3SBjoern A. Zeeb }
714da8fa4e3SBjoern A. Zeeb 
715da8fa4e3SBjoern A. Zeeb #ifdef CONFIG_PM
716da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_suspend(struct ath10k * ar)717da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_suspend(struct ath10k *ar)
718da8fa4e3SBjoern A. Zeeb {
719da8fa4e3SBjoern A. Zeeb 	return -EOPNOTSUPP;
720da8fa4e3SBjoern A. Zeeb }
721da8fa4e3SBjoern A. Zeeb 
ath10k_usb_hif_resume(struct ath10k * ar)722da8fa4e3SBjoern A. Zeeb static int ath10k_usb_hif_resume(struct ath10k *ar)
723da8fa4e3SBjoern A. Zeeb {
724da8fa4e3SBjoern A. Zeeb 	return -EOPNOTSUPP;
725da8fa4e3SBjoern A. Zeeb }
726da8fa4e3SBjoern A. Zeeb #endif
727da8fa4e3SBjoern A. Zeeb 
728da8fa4e3SBjoern A. Zeeb static const struct ath10k_hif_ops ath10k_usb_hif_ops = {
729da8fa4e3SBjoern A. Zeeb 	.tx_sg			= ath10k_usb_hif_tx_sg,
730da8fa4e3SBjoern A. Zeeb 	.diag_read		= ath10k_usb_hif_diag_read,
731da8fa4e3SBjoern A. Zeeb 	.diag_write		= ath10k_usb_hif_diag_write,
732da8fa4e3SBjoern A. Zeeb 	.exchange_bmi_msg	= ath10k_usb_bmi_exchange_msg,
733da8fa4e3SBjoern A. Zeeb 	.start			= ath10k_usb_hif_start,
734da8fa4e3SBjoern A. Zeeb 	.stop			= ath10k_usb_hif_stop,
735da8fa4e3SBjoern A. Zeeb 	.map_service_to_pipe	= ath10k_usb_hif_map_service_to_pipe,
736da8fa4e3SBjoern A. Zeeb 	.get_default_pipe	= ath10k_usb_hif_get_default_pipe,
737da8fa4e3SBjoern A. Zeeb 	.get_free_queue_number	= ath10k_usb_hif_get_free_queue_number,
738da8fa4e3SBjoern A. Zeeb 	.power_up		= ath10k_usb_hif_power_up,
739da8fa4e3SBjoern A. Zeeb 	.power_down		= ath10k_usb_hif_power_down,
740da8fa4e3SBjoern A. Zeeb #ifdef CONFIG_PM
741da8fa4e3SBjoern A. Zeeb 	.suspend		= ath10k_usb_hif_suspend,
742da8fa4e3SBjoern A. Zeeb 	.resume			= ath10k_usb_hif_resume,
743da8fa4e3SBjoern A. Zeeb #endif
744da8fa4e3SBjoern A. Zeeb };
745da8fa4e3SBjoern A. Zeeb 
ath10k_usb_get_logical_pipe_num(u8 ep_address,int * urb_count)746da8fa4e3SBjoern A. Zeeb static u8 ath10k_usb_get_logical_pipe_num(u8 ep_address, int *urb_count)
747da8fa4e3SBjoern A. Zeeb {
748da8fa4e3SBjoern A. Zeeb 	u8 pipe_num = ATH10K_USB_PIPE_INVALID;
749da8fa4e3SBjoern A. Zeeb 
750da8fa4e3SBjoern A. Zeeb 	switch (ep_address) {
751da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_CTRL_IN:
752da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_RX_CTRL;
753da8fa4e3SBjoern A. Zeeb 		*urb_count = RX_URB_COUNT;
754da8fa4e3SBjoern A. Zeeb 		break;
755da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_DATA_IN:
756da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_RX_DATA;
757da8fa4e3SBjoern A. Zeeb 		*urb_count = RX_URB_COUNT;
758da8fa4e3SBjoern A. Zeeb 		break;
759da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_INT_IN:
760da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_RX_INT;
761da8fa4e3SBjoern A. Zeeb 		*urb_count = RX_URB_COUNT;
762da8fa4e3SBjoern A. Zeeb 		break;
763da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_DATA2_IN:
764da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_RX_DATA2;
765da8fa4e3SBjoern A. Zeeb 		*urb_count = RX_URB_COUNT;
766da8fa4e3SBjoern A. Zeeb 		break;
767da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_CTRL_OUT:
768da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_TX_CTRL;
769da8fa4e3SBjoern A. Zeeb 		*urb_count = TX_URB_COUNT;
770da8fa4e3SBjoern A. Zeeb 		break;
771da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_DATA_LP_OUT:
772da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_TX_DATA_LP;
773da8fa4e3SBjoern A. Zeeb 		*urb_count = TX_URB_COUNT;
774da8fa4e3SBjoern A. Zeeb 		break;
775da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_DATA_MP_OUT:
776da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_TX_DATA_MP;
777da8fa4e3SBjoern A. Zeeb 		*urb_count = TX_URB_COUNT;
778da8fa4e3SBjoern A. Zeeb 		break;
779da8fa4e3SBjoern A. Zeeb 	case ATH10K_USB_EP_ADDR_APP_DATA_HP_OUT:
780da8fa4e3SBjoern A. Zeeb 		pipe_num = ATH10K_USB_PIPE_TX_DATA_HP;
781da8fa4e3SBjoern A. Zeeb 		*urb_count = TX_URB_COUNT;
782da8fa4e3SBjoern A. Zeeb 		break;
783da8fa4e3SBjoern A. Zeeb 	default:
784da8fa4e3SBjoern A. Zeeb 		/* note: there may be endpoints not currently used */
785da8fa4e3SBjoern A. Zeeb 		break;
786da8fa4e3SBjoern A. Zeeb 	}
787da8fa4e3SBjoern A. Zeeb 
788da8fa4e3SBjoern A. Zeeb 	return pipe_num;
789da8fa4e3SBjoern A. Zeeb }
790da8fa4e3SBjoern A. Zeeb 
ath10k_usb_alloc_pipe_resources(struct ath10k * ar,struct ath10k_usb_pipe * pipe,int urb_cnt)791da8fa4e3SBjoern A. Zeeb static int ath10k_usb_alloc_pipe_resources(struct ath10k *ar,
792da8fa4e3SBjoern A. Zeeb 					   struct ath10k_usb_pipe *pipe,
793da8fa4e3SBjoern A. Zeeb 					   int urb_cnt)
794da8fa4e3SBjoern A. Zeeb {
795da8fa4e3SBjoern A. Zeeb 	struct ath10k_urb_context *urb_context;
796da8fa4e3SBjoern A. Zeeb 	int i;
797da8fa4e3SBjoern A. Zeeb 
798da8fa4e3SBjoern A. Zeeb 	INIT_LIST_HEAD(&pipe->urb_list_head);
799da8fa4e3SBjoern A. Zeeb 	init_usb_anchor(&pipe->urb_submitted);
800da8fa4e3SBjoern A. Zeeb 
801da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < urb_cnt; i++) {
802da8fa4e3SBjoern A. Zeeb 		urb_context = kzalloc(sizeof(*urb_context), GFP_KERNEL);
803da8fa4e3SBjoern A. Zeeb 		if (!urb_context)
804da8fa4e3SBjoern A. Zeeb 			return -ENOMEM;
805da8fa4e3SBjoern A. Zeeb 
806da8fa4e3SBjoern A. Zeeb 		urb_context->pipe = pipe;
807da8fa4e3SBjoern A. Zeeb 
808da8fa4e3SBjoern A. Zeeb 		/* we are only allocate the urb contexts here, the actual URB
809da8fa4e3SBjoern A. Zeeb 		 * is allocated from the kernel as needed to do a transaction
810da8fa4e3SBjoern A. Zeeb 		 */
811da8fa4e3SBjoern A. Zeeb 		pipe->urb_alloc++;
812da8fa4e3SBjoern A. Zeeb 		ath10k_usb_free_urb_to_pipe(pipe, urb_context);
813da8fa4e3SBjoern A. Zeeb 	}
814da8fa4e3SBjoern A. Zeeb 
815da8fa4e3SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_USB,
816da8fa4e3SBjoern A. Zeeb 		   "usb alloc resources lpipe %d hpipe 0x%x urbs %d\n",
817da8fa4e3SBjoern A. Zeeb 		   pipe->logical_pipe_num, pipe->usb_pipe_handle,
818da8fa4e3SBjoern A. Zeeb 		   pipe->urb_alloc);
819da8fa4e3SBjoern A. Zeeb 
820da8fa4e3SBjoern A. Zeeb 	return 0;
821da8fa4e3SBjoern A. Zeeb }
822da8fa4e3SBjoern A. Zeeb 
ath10k_usb_setup_pipe_resources(struct ath10k * ar,struct usb_interface * interface)823da8fa4e3SBjoern A. Zeeb static int ath10k_usb_setup_pipe_resources(struct ath10k *ar,
824da8fa4e3SBjoern A. Zeeb 					   struct usb_interface *interface)
825da8fa4e3SBjoern A. Zeeb {
826da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
827da8fa4e3SBjoern A. Zeeb 	struct usb_host_interface *iface_desc = interface->cur_altsetting;
828da8fa4e3SBjoern A. Zeeb 	struct usb_endpoint_descriptor *endpoint;
829da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe;
830da8fa4e3SBjoern A. Zeeb 	int ret, i, urbcount;
831da8fa4e3SBjoern A. Zeeb 	u8 pipe_num;
832da8fa4e3SBjoern A. Zeeb 
833da8fa4e3SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_USB, "usb setting up pipes using interface\n");
834da8fa4e3SBjoern A. Zeeb 
835da8fa4e3SBjoern A. Zeeb 	/* walk descriptors and setup pipes */
836da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < iface_desc->desc.bNumEndpoints; ++i) {
837da8fa4e3SBjoern A. Zeeb 		endpoint = &iface_desc->endpoint[i].desc;
838da8fa4e3SBjoern A. Zeeb 
839da8fa4e3SBjoern A. Zeeb 		if (ATH10K_USB_IS_BULK_EP(endpoint->bmAttributes)) {
840da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB,
841da8fa4e3SBjoern A. Zeeb 				   "usb %s bulk ep 0x%2.2x maxpktsz %d\n",
842da8fa4e3SBjoern A. Zeeb 				   ATH10K_USB_IS_DIR_IN
843da8fa4e3SBjoern A. Zeeb 				   (endpoint->bEndpointAddress) ?
844da8fa4e3SBjoern A. Zeeb 				   "rx" : "tx", endpoint->bEndpointAddress,
845da8fa4e3SBjoern A. Zeeb 				   le16_to_cpu(endpoint->wMaxPacketSize));
846da8fa4e3SBjoern A. Zeeb 		} else if (ATH10K_USB_IS_INT_EP(endpoint->bmAttributes)) {
847da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB,
848da8fa4e3SBjoern A. Zeeb 				   "usb %s int ep 0x%2.2x maxpktsz %d interval %d\n",
849da8fa4e3SBjoern A. Zeeb 				   ATH10K_USB_IS_DIR_IN
850da8fa4e3SBjoern A. Zeeb 				   (endpoint->bEndpointAddress) ?
851da8fa4e3SBjoern A. Zeeb 				   "rx" : "tx", endpoint->bEndpointAddress,
852da8fa4e3SBjoern A. Zeeb 				   le16_to_cpu(endpoint->wMaxPacketSize),
853da8fa4e3SBjoern A. Zeeb 				   endpoint->bInterval);
854da8fa4e3SBjoern A. Zeeb 		} else if (ATH10K_USB_IS_ISOC_EP(endpoint->bmAttributes)) {
855da8fa4e3SBjoern A. Zeeb 			/* TODO for ISO */
856da8fa4e3SBjoern A. Zeeb 			ath10k_dbg(ar, ATH10K_DBG_USB,
857da8fa4e3SBjoern A. Zeeb 				   "usb %s isoc ep 0x%2.2x maxpktsz %d interval %d\n",
858da8fa4e3SBjoern A. Zeeb 				   ATH10K_USB_IS_DIR_IN
859da8fa4e3SBjoern A. Zeeb 				   (endpoint->bEndpointAddress) ?
860da8fa4e3SBjoern A. Zeeb 				   "rx" : "tx", endpoint->bEndpointAddress,
861da8fa4e3SBjoern A. Zeeb 				   le16_to_cpu(endpoint->wMaxPacketSize),
862da8fa4e3SBjoern A. Zeeb 				   endpoint->bInterval);
863da8fa4e3SBjoern A. Zeeb 		}
864da8fa4e3SBjoern A. Zeeb 
865da8fa4e3SBjoern A. Zeeb 		/* Ignore broken descriptors. */
866da8fa4e3SBjoern A. Zeeb 		if (usb_endpoint_maxp(endpoint) == 0)
867da8fa4e3SBjoern A. Zeeb 			continue;
868da8fa4e3SBjoern A. Zeeb 
869da8fa4e3SBjoern A. Zeeb 		urbcount = 0;
870da8fa4e3SBjoern A. Zeeb 
871da8fa4e3SBjoern A. Zeeb 		pipe_num =
872da8fa4e3SBjoern A. Zeeb 		    ath10k_usb_get_logical_pipe_num(endpoint->bEndpointAddress,
873da8fa4e3SBjoern A. Zeeb 						    &urbcount);
874da8fa4e3SBjoern A. Zeeb 		if (pipe_num == ATH10K_USB_PIPE_INVALID)
875da8fa4e3SBjoern A. Zeeb 			continue;
876da8fa4e3SBjoern A. Zeeb 
877da8fa4e3SBjoern A. Zeeb 		pipe = &ar_usb->pipes[pipe_num];
878da8fa4e3SBjoern A. Zeeb 		if (pipe->ar_usb)
879da8fa4e3SBjoern A. Zeeb 			/* hmmm..pipe was already setup */
880da8fa4e3SBjoern A. Zeeb 			continue;
881da8fa4e3SBjoern A. Zeeb 
882da8fa4e3SBjoern A. Zeeb 		pipe->ar_usb = ar_usb;
883da8fa4e3SBjoern A. Zeeb 		pipe->logical_pipe_num = pipe_num;
884da8fa4e3SBjoern A. Zeeb 		pipe->ep_address = endpoint->bEndpointAddress;
885da8fa4e3SBjoern A. Zeeb 		pipe->max_packet_size = le16_to_cpu(endpoint->wMaxPacketSize);
886da8fa4e3SBjoern A. Zeeb 
887da8fa4e3SBjoern A. Zeeb 		if (ATH10K_USB_IS_BULK_EP(endpoint->bmAttributes)) {
888da8fa4e3SBjoern A. Zeeb 			if (ATH10K_USB_IS_DIR_IN(pipe->ep_address)) {
889da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
890da8fa4e3SBjoern A. Zeeb 				    usb_rcvbulkpipe(ar_usb->udev,
891da8fa4e3SBjoern A. Zeeb 						    pipe->ep_address);
892da8fa4e3SBjoern A. Zeeb 			} else {
893da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
894da8fa4e3SBjoern A. Zeeb 				    usb_sndbulkpipe(ar_usb->udev,
895da8fa4e3SBjoern A. Zeeb 						    pipe->ep_address);
896da8fa4e3SBjoern A. Zeeb 			}
897da8fa4e3SBjoern A. Zeeb 		} else if (ATH10K_USB_IS_INT_EP(endpoint->bmAttributes)) {
898da8fa4e3SBjoern A. Zeeb 			if (ATH10K_USB_IS_DIR_IN(pipe->ep_address)) {
899da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
900da8fa4e3SBjoern A. Zeeb 				    usb_rcvintpipe(ar_usb->udev,
901da8fa4e3SBjoern A. Zeeb 						   pipe->ep_address);
902da8fa4e3SBjoern A. Zeeb 			} else {
903da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
904da8fa4e3SBjoern A. Zeeb 				    usb_sndintpipe(ar_usb->udev,
905da8fa4e3SBjoern A. Zeeb 						   pipe->ep_address);
906da8fa4e3SBjoern A. Zeeb 			}
907da8fa4e3SBjoern A. Zeeb 		} else if (ATH10K_USB_IS_ISOC_EP(endpoint->bmAttributes)) {
908da8fa4e3SBjoern A. Zeeb 			/* TODO for ISO */
909da8fa4e3SBjoern A. Zeeb 			if (ATH10K_USB_IS_DIR_IN(pipe->ep_address)) {
910da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
911da8fa4e3SBjoern A. Zeeb 				    usb_rcvisocpipe(ar_usb->udev,
912da8fa4e3SBjoern A. Zeeb 						    pipe->ep_address);
913da8fa4e3SBjoern A. Zeeb 			} else {
914da8fa4e3SBjoern A. Zeeb 				pipe->usb_pipe_handle =
915da8fa4e3SBjoern A. Zeeb 				    usb_sndisocpipe(ar_usb->udev,
916da8fa4e3SBjoern A. Zeeb 						    pipe->ep_address);
917da8fa4e3SBjoern A. Zeeb 			}
918da8fa4e3SBjoern A. Zeeb 		}
919da8fa4e3SBjoern A. Zeeb 
920da8fa4e3SBjoern A. Zeeb 		pipe->ep_desc = endpoint;
921da8fa4e3SBjoern A. Zeeb 
922da8fa4e3SBjoern A. Zeeb 		if (!ATH10K_USB_IS_DIR_IN(pipe->ep_address))
923da8fa4e3SBjoern A. Zeeb 			pipe->flags |= ATH10K_USB_PIPE_FLAG_TX;
924da8fa4e3SBjoern A. Zeeb 
925da8fa4e3SBjoern A. Zeeb 		ret = ath10k_usb_alloc_pipe_resources(ar, pipe, urbcount);
926da8fa4e3SBjoern A. Zeeb 		if (ret)
927da8fa4e3SBjoern A. Zeeb 			return ret;
928da8fa4e3SBjoern A. Zeeb 	}
929da8fa4e3SBjoern A. Zeeb 
930da8fa4e3SBjoern A. Zeeb 	return 0;
931da8fa4e3SBjoern A. Zeeb }
932da8fa4e3SBjoern A. Zeeb 
ath10k_usb_create(struct ath10k * ar,struct usb_interface * interface)933da8fa4e3SBjoern A. Zeeb static int ath10k_usb_create(struct ath10k *ar,
934da8fa4e3SBjoern A. Zeeb 			     struct usb_interface *interface)
935da8fa4e3SBjoern A. Zeeb {
936da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = ath10k_usb_priv(ar);
937da8fa4e3SBjoern A. Zeeb 	struct usb_device *dev = interface_to_usbdev(interface);
938da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb_pipe *pipe;
939da8fa4e3SBjoern A. Zeeb 	int ret, i;
940da8fa4e3SBjoern A. Zeeb 
941da8fa4e3SBjoern A. Zeeb 	usb_set_intfdata(interface, ar_usb);
942da8fa4e3SBjoern A. Zeeb 	spin_lock_init(&ar_usb->cs_lock);
943da8fa4e3SBjoern A. Zeeb 	ar_usb->udev = dev;
944da8fa4e3SBjoern A. Zeeb 	ar_usb->interface = interface;
945da8fa4e3SBjoern A. Zeeb 
946da8fa4e3SBjoern A. Zeeb 	for (i = 0; i < ATH10K_USB_PIPE_MAX; i++) {
947da8fa4e3SBjoern A. Zeeb 		pipe = &ar_usb->pipes[i];
948da8fa4e3SBjoern A. Zeeb 		INIT_WORK(&pipe->io_complete_work,
949da8fa4e3SBjoern A. Zeeb 			  ath10k_usb_io_comp_work);
950da8fa4e3SBjoern A. Zeeb 		skb_queue_head_init(&pipe->io_comp_queue);
951da8fa4e3SBjoern A. Zeeb 	}
952da8fa4e3SBjoern A. Zeeb 
953da8fa4e3SBjoern A. Zeeb 	ar_usb->diag_cmd_buffer = kzalloc(ATH10K_USB_MAX_DIAG_CMD, GFP_KERNEL);
954da8fa4e3SBjoern A. Zeeb 	if (!ar_usb->diag_cmd_buffer) {
955da8fa4e3SBjoern A. Zeeb 		ret = -ENOMEM;
956da8fa4e3SBjoern A. Zeeb 		goto err;
957da8fa4e3SBjoern A. Zeeb 	}
958da8fa4e3SBjoern A. Zeeb 
959da8fa4e3SBjoern A. Zeeb 	ar_usb->diag_resp_buffer = kzalloc(ATH10K_USB_MAX_DIAG_RESP,
960da8fa4e3SBjoern A. Zeeb 					   GFP_KERNEL);
961da8fa4e3SBjoern A. Zeeb 	if (!ar_usb->diag_resp_buffer) {
962da8fa4e3SBjoern A. Zeeb 		ret = -ENOMEM;
963da8fa4e3SBjoern A. Zeeb 		goto err;
964da8fa4e3SBjoern A. Zeeb 	}
965da8fa4e3SBjoern A. Zeeb 
966da8fa4e3SBjoern A. Zeeb 	ret = ath10k_usb_setup_pipe_resources(ar, interface);
967da8fa4e3SBjoern A. Zeeb 	if (ret)
968da8fa4e3SBjoern A. Zeeb 		goto err;
969da8fa4e3SBjoern A. Zeeb 
970da8fa4e3SBjoern A. Zeeb 	return 0;
971da8fa4e3SBjoern A. Zeeb 
972da8fa4e3SBjoern A. Zeeb err:
973da8fa4e3SBjoern A. Zeeb 	ath10k_usb_destroy(ar);
974da8fa4e3SBjoern A. Zeeb 	return ret;
975da8fa4e3SBjoern A. Zeeb }
976da8fa4e3SBjoern A. Zeeb 
ath10k_usb_napi_poll(struct napi_struct * ctx,int budget)977*07724ba6SBjoern A. Zeeb static int ath10k_usb_napi_poll(struct napi_struct *ctx, int budget)
978*07724ba6SBjoern A. Zeeb {
979*07724ba6SBjoern A. Zeeb 	struct ath10k *ar = container_of(ctx, struct ath10k, napi);
980*07724ba6SBjoern A. Zeeb 	int done;
981*07724ba6SBjoern A. Zeeb 
982*07724ba6SBjoern A. Zeeb 	done = ath10k_htt_rx_hl_indication(ar, budget);
983*07724ba6SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_USB, "napi poll: done: %d, budget:%d\n", done, budget);
984*07724ba6SBjoern A. Zeeb 
985*07724ba6SBjoern A. Zeeb 	if (done < budget)
986*07724ba6SBjoern A. Zeeb 		napi_complete_done(ctx, done);
987*07724ba6SBjoern A. Zeeb 
988*07724ba6SBjoern A. Zeeb 	return done;
989*07724ba6SBjoern A. Zeeb }
990*07724ba6SBjoern A. Zeeb 
991da8fa4e3SBjoern A. Zeeb /* ath10k usb driver registered functions */
ath10k_usb_probe(struct usb_interface * interface,const struct usb_device_id * id)992da8fa4e3SBjoern A. Zeeb static int ath10k_usb_probe(struct usb_interface *interface,
993da8fa4e3SBjoern A. Zeeb 			    const struct usb_device_id *id)
994da8fa4e3SBjoern A. Zeeb {
995da8fa4e3SBjoern A. Zeeb 	struct ath10k *ar;
996da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb;
997da8fa4e3SBjoern A. Zeeb 	struct usb_device *dev = interface_to_usbdev(interface);
998da8fa4e3SBjoern A. Zeeb 	int ret, vendor_id, product_id;
999da8fa4e3SBjoern A. Zeeb 	enum ath10k_hw_rev hw_rev;
1000da8fa4e3SBjoern A. Zeeb 	struct ath10k_bus_params bus_params = {};
1001da8fa4e3SBjoern A. Zeeb 
1002da8fa4e3SBjoern A. Zeeb 	/* Assumption: All USB based chipsets (so far) are QCA9377 based.
1003da8fa4e3SBjoern A. Zeeb 	 * If there will be newer chipsets that does not use the hw reg
1004da8fa4e3SBjoern A. Zeeb 	 * setup as defined in qca6174_regs and qca6174_values, this
1005da8fa4e3SBjoern A. Zeeb 	 * assumption is no longer valid and hw_rev must be setup differently
1006da8fa4e3SBjoern A. Zeeb 	 * depending on chipset.
1007da8fa4e3SBjoern A. Zeeb 	 */
1008da8fa4e3SBjoern A. Zeeb 	hw_rev = ATH10K_HW_QCA9377;
1009da8fa4e3SBjoern A. Zeeb 
1010da8fa4e3SBjoern A. Zeeb 	ar = ath10k_core_create(sizeof(*ar_usb), &dev->dev, ATH10K_BUS_USB,
1011da8fa4e3SBjoern A. Zeeb 				hw_rev, &ath10k_usb_hif_ops);
1012da8fa4e3SBjoern A. Zeeb 	if (!ar) {
1013da8fa4e3SBjoern A. Zeeb 		dev_err(&dev->dev, "failed to allocate core\n");
1014da8fa4e3SBjoern A. Zeeb 		return -ENOMEM;
1015da8fa4e3SBjoern A. Zeeb 	}
1016da8fa4e3SBjoern A. Zeeb 
1017*07724ba6SBjoern A. Zeeb 	netif_napi_add(&ar->napi_dev, &ar->napi, ath10k_usb_napi_poll);
1018*07724ba6SBjoern A. Zeeb 
1019da8fa4e3SBjoern A. Zeeb 	usb_get_dev(dev);
1020da8fa4e3SBjoern A. Zeeb 	vendor_id = le16_to_cpu(dev->descriptor.idVendor);
1021da8fa4e3SBjoern A. Zeeb 	product_id = le16_to_cpu(dev->descriptor.idProduct);
1022da8fa4e3SBjoern A. Zeeb 
1023da8fa4e3SBjoern A. Zeeb 	ath10k_dbg(ar, ATH10K_DBG_BOOT,
1024da8fa4e3SBjoern A. Zeeb 		   "usb new func vendor 0x%04x product 0x%04x\n",
1025da8fa4e3SBjoern A. Zeeb 		   vendor_id, product_id);
1026da8fa4e3SBjoern A. Zeeb 
1027da8fa4e3SBjoern A. Zeeb 	ar_usb = ath10k_usb_priv(ar);
1028da8fa4e3SBjoern A. Zeeb 	ret = ath10k_usb_create(ar, interface);
1029da8fa4e3SBjoern A. Zeeb 	if (ret)
1030da8fa4e3SBjoern A. Zeeb 		goto err;
1031da8fa4e3SBjoern A. Zeeb 	ar_usb->ar = ar;
1032da8fa4e3SBjoern A. Zeeb 
1033da8fa4e3SBjoern A. Zeeb 	ar->dev_id = product_id;
1034da8fa4e3SBjoern A. Zeeb 	ar->id.vendor = vendor_id;
1035da8fa4e3SBjoern A. Zeeb 	ar->id.device = product_id;
1036da8fa4e3SBjoern A. Zeeb 
1037da8fa4e3SBjoern A. Zeeb 	bus_params.dev_type = ATH10K_DEV_TYPE_HL;
1038da8fa4e3SBjoern A. Zeeb 	/* TODO: don't know yet how to get chip_id with USB */
1039da8fa4e3SBjoern A. Zeeb 	bus_params.chip_id = 0;
1040*07724ba6SBjoern A. Zeeb 	bus_params.hl_msdu_ids = true;
1041da8fa4e3SBjoern A. Zeeb 	ret = ath10k_core_register(ar, &bus_params);
1042da8fa4e3SBjoern A. Zeeb 	if (ret) {
1043da8fa4e3SBjoern A. Zeeb 		ath10k_warn(ar, "failed to register driver core: %d\n", ret);
1044da8fa4e3SBjoern A. Zeeb 		goto err_usb_destroy;
1045da8fa4e3SBjoern A. Zeeb 	}
1046da8fa4e3SBjoern A. Zeeb 
1047da8fa4e3SBjoern A. Zeeb 	/* TODO: remove this once USB support is fully implemented */
1048da8fa4e3SBjoern A. Zeeb 	ath10k_warn(ar, "Warning: ath10k USB support is incomplete, don't expect anything to work!\n");
1049da8fa4e3SBjoern A. Zeeb 
1050da8fa4e3SBjoern A. Zeeb 	return 0;
1051da8fa4e3SBjoern A. Zeeb 
1052da8fa4e3SBjoern A. Zeeb err_usb_destroy:
1053da8fa4e3SBjoern A. Zeeb 	ath10k_usb_destroy(ar);
1054da8fa4e3SBjoern A. Zeeb 
1055da8fa4e3SBjoern A. Zeeb err:
1056da8fa4e3SBjoern A. Zeeb 	ath10k_core_destroy(ar);
1057da8fa4e3SBjoern A. Zeeb 
1058da8fa4e3SBjoern A. Zeeb 	usb_put_dev(dev);
1059da8fa4e3SBjoern A. Zeeb 
1060da8fa4e3SBjoern A. Zeeb 	return ret;
1061da8fa4e3SBjoern A. Zeeb }
1062da8fa4e3SBjoern A. Zeeb 
ath10k_usb_remove(struct usb_interface * interface)1063da8fa4e3SBjoern A. Zeeb static void ath10k_usb_remove(struct usb_interface *interface)
1064da8fa4e3SBjoern A. Zeeb {
1065da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb;
1066da8fa4e3SBjoern A. Zeeb 
1067da8fa4e3SBjoern A. Zeeb 	ar_usb = usb_get_intfdata(interface);
1068da8fa4e3SBjoern A. Zeeb 	if (!ar_usb)
1069da8fa4e3SBjoern A. Zeeb 		return;
1070da8fa4e3SBjoern A. Zeeb 
1071da8fa4e3SBjoern A. Zeeb 	ath10k_core_unregister(ar_usb->ar);
1072*07724ba6SBjoern A. Zeeb 	netif_napi_del(&ar_usb->ar->napi);
1073da8fa4e3SBjoern A. Zeeb 	ath10k_usb_destroy(ar_usb->ar);
1074da8fa4e3SBjoern A. Zeeb 	usb_put_dev(interface_to_usbdev(interface));
1075da8fa4e3SBjoern A. Zeeb 	ath10k_core_destroy(ar_usb->ar);
1076da8fa4e3SBjoern A. Zeeb }
1077da8fa4e3SBjoern A. Zeeb 
1078da8fa4e3SBjoern A. Zeeb #ifdef CONFIG_PM
1079da8fa4e3SBjoern A. Zeeb 
ath10k_usb_pm_suspend(struct usb_interface * interface,pm_message_t message)1080da8fa4e3SBjoern A. Zeeb static int ath10k_usb_pm_suspend(struct usb_interface *interface,
1081da8fa4e3SBjoern A. Zeeb 				 pm_message_t message)
1082da8fa4e3SBjoern A. Zeeb {
1083da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = usb_get_intfdata(interface);
1084da8fa4e3SBjoern A. Zeeb 
1085da8fa4e3SBjoern A. Zeeb 	ath10k_usb_flush_all(ar_usb->ar);
1086da8fa4e3SBjoern A. Zeeb 	return 0;
1087da8fa4e3SBjoern A. Zeeb }
1088da8fa4e3SBjoern A. Zeeb 
ath10k_usb_pm_resume(struct usb_interface * interface)1089da8fa4e3SBjoern A. Zeeb static int ath10k_usb_pm_resume(struct usb_interface *interface)
1090da8fa4e3SBjoern A. Zeeb {
1091da8fa4e3SBjoern A. Zeeb 	struct ath10k_usb *ar_usb = usb_get_intfdata(interface);
1092da8fa4e3SBjoern A. Zeeb 	struct ath10k *ar = ar_usb->ar;
1093da8fa4e3SBjoern A. Zeeb 
1094da8fa4e3SBjoern A. Zeeb 	ath10k_usb_post_recv_transfers(ar,
1095da8fa4e3SBjoern A. Zeeb 				       &ar_usb->pipes[ATH10K_USB_PIPE_RX_DATA]);
1096da8fa4e3SBjoern A. Zeeb 
1097da8fa4e3SBjoern A. Zeeb 	return 0;
1098da8fa4e3SBjoern A. Zeeb }
1099da8fa4e3SBjoern A. Zeeb 
1100da8fa4e3SBjoern A. Zeeb #else
1101da8fa4e3SBjoern A. Zeeb 
1102da8fa4e3SBjoern A. Zeeb #define ath10k_usb_pm_suspend NULL
1103da8fa4e3SBjoern A. Zeeb #define ath10k_usb_pm_resume NULL
1104da8fa4e3SBjoern A. Zeeb 
1105da8fa4e3SBjoern A. Zeeb #endif
1106da8fa4e3SBjoern A. Zeeb 
1107da8fa4e3SBjoern A. Zeeb /* table of devices that work with this driver */
1108da8fa4e3SBjoern A. Zeeb static struct usb_device_id ath10k_usb_ids[] = {
1109da8fa4e3SBjoern A. Zeeb 	{USB_DEVICE(0x13b1, 0x0042)}, /* Linksys WUSB6100M */
1110da8fa4e3SBjoern A. Zeeb 	{ /* Terminating entry */ },
1111da8fa4e3SBjoern A. Zeeb };
1112da8fa4e3SBjoern A. Zeeb 
1113da8fa4e3SBjoern A. Zeeb MODULE_DEVICE_TABLE(usb, ath10k_usb_ids);
1114da8fa4e3SBjoern A. Zeeb 
1115da8fa4e3SBjoern A. Zeeb static struct usb_driver ath10k_usb_driver = {
1116da8fa4e3SBjoern A. Zeeb 	.name = "ath10k_usb",
1117da8fa4e3SBjoern A. Zeeb 	.probe = ath10k_usb_probe,
1118da8fa4e3SBjoern A. Zeeb 	.suspend = ath10k_usb_pm_suspend,
1119da8fa4e3SBjoern A. Zeeb 	.resume = ath10k_usb_pm_resume,
1120da8fa4e3SBjoern A. Zeeb 	.disconnect = ath10k_usb_remove,
1121da8fa4e3SBjoern A. Zeeb 	.id_table = ath10k_usb_ids,
1122da8fa4e3SBjoern A. Zeeb 	.supports_autosuspend = true,
1123da8fa4e3SBjoern A. Zeeb 	.disable_hub_initiated_lpm = 1,
1124da8fa4e3SBjoern A. Zeeb };
1125da8fa4e3SBjoern A. Zeeb 
1126da8fa4e3SBjoern A. Zeeb module_usb_driver(ath10k_usb_driver);
1127da8fa4e3SBjoern A. Zeeb 
1128da8fa4e3SBjoern A. Zeeb MODULE_AUTHOR("Atheros Communications, Inc.");
1129da8fa4e3SBjoern A. Zeeb MODULE_DESCRIPTION("Driver support for Qualcomm Atheros 802.11ac WLAN USB devices");
1130da8fa4e3SBjoern A. Zeeb MODULE_LICENSE("Dual BSD/GPL");
1131