xref: /openbmc/linux/drivers/net/wireless/ath/ath6kl/usb.c (revision ce932d0c5589e9766e089c22c66890dfc48fbd94)
1 /*
2  * Copyright (c) 2007-2011 Atheros Communications Inc.
3  * Copyright (c) 2011-2012 Qualcomm Atheros, Inc.
4  *
5  * Permission to use, copy, modify, and/or distribute this software for any
6  * purpose with or without fee is hereby granted, provided that the above
7  * copyright notice and this permission notice appear in all copies.
8  *
9  * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
10  * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
11  * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
12  * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
13  * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
14  * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
15  * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
16  */
17 
18 #include <linux/module.h>
19 #include <linux/usb.h>
20 
21 #include "debug.h"
22 #include "core.h"
23 
24 /* usb device object */
25 struct ath6kl_usb {
26 	struct usb_device *udev;
27 	struct usb_interface *interface;
28 	u8 *diag_cmd_buffer;
29 	u8 *diag_resp_buffer;
30 	struct ath6kl *ar;
31 };
32 
33 /* diagnostic command defnitions */
34 #define ATH6KL_USB_CONTROL_REQ_SEND_BMI_CMD        1
35 #define ATH6KL_USB_CONTROL_REQ_RECV_BMI_RESP       2
36 #define ATH6KL_USB_CONTROL_REQ_DIAG_CMD            3
37 #define ATH6KL_USB_CONTROL_REQ_DIAG_RESP           4
38 
39 #define ATH6KL_USB_CTRL_DIAG_CC_READ               0
40 #define ATH6KL_USB_CTRL_DIAG_CC_WRITE              1
41 
42 struct ath6kl_usb_ctrl_diag_cmd_write {
43 	__le32 cmd;
44 	__le32 address;
45 	__le32 value;
46 	__le32 _pad[1];
47 } __packed;
48 
49 struct ath6kl_usb_ctrl_diag_cmd_read {
50 	__le32 cmd;
51 	__le32 address;
52 } __packed;
53 
54 struct ath6kl_usb_ctrl_diag_resp_read {
55 	__le32 value;
56 } __packed;
57 
58 #define ATH6KL_USB_MAX_DIAG_CMD (sizeof(struct ath6kl_usb_ctrl_diag_cmd_write))
59 #define ATH6KL_USB_MAX_DIAG_RESP (sizeof(struct ath6kl_usb_ctrl_diag_resp_read))
60 
61 static void ath6kl_usb_destroy(struct ath6kl_usb *ar_usb)
62 {
63 	usb_set_intfdata(ar_usb->interface, NULL);
64 
65 	kfree(ar_usb->diag_cmd_buffer);
66 	kfree(ar_usb->diag_resp_buffer);
67 
68 	kfree(ar_usb);
69 }
70 
71 static struct ath6kl_usb *ath6kl_usb_create(struct usb_interface *interface)
72 {
73 	struct ath6kl_usb *ar_usb = NULL;
74 	struct usb_device *dev = interface_to_usbdev(interface);
75 	int status = 0;
76 
77 	ar_usb = kzalloc(sizeof(struct ath6kl_usb), GFP_KERNEL);
78 	if (ar_usb == NULL)
79 		goto fail_ath6kl_usb_create;
80 
81 	memset(ar_usb, 0, sizeof(struct ath6kl_usb));
82 	usb_set_intfdata(interface, ar_usb);
83 	ar_usb->udev = dev;
84 	ar_usb->interface = interface;
85 
86 	ar_usb->diag_cmd_buffer = kzalloc(ATH6KL_USB_MAX_DIAG_CMD, GFP_KERNEL);
87 	if (ar_usb->diag_cmd_buffer == NULL) {
88 		status = -ENOMEM;
89 		goto fail_ath6kl_usb_create;
90 	}
91 
92 	ar_usb->diag_resp_buffer = kzalloc(ATH6KL_USB_MAX_DIAG_RESP,
93 					   GFP_KERNEL);
94 	if (ar_usb->diag_resp_buffer == NULL) {
95 		status = -ENOMEM;
96 		goto fail_ath6kl_usb_create;
97 	}
98 
99 fail_ath6kl_usb_create:
100 	if (status != 0) {
101 		ath6kl_usb_destroy(ar_usb);
102 		ar_usb = NULL;
103 	}
104 	return ar_usb;
105 }
106 
107 static void ath6kl_usb_device_detached(struct usb_interface *interface)
108 {
109 	struct ath6kl_usb *ar_usb;
110 
111 	ar_usb = usb_get_intfdata(interface);
112 	if (ar_usb == NULL)
113 		return;
114 
115 	ath6kl_stop_txrx(ar_usb->ar);
116 
117 	ath6kl_core_cleanup(ar_usb->ar);
118 
119 	ath6kl_usb_destroy(ar_usb);
120 }
121 
122 static int ath6kl_usb_submit_ctrl_out(struct ath6kl_usb *ar_usb,
123 				   u8 req, u16 value, u16 index, void *data,
124 				   u32 size)
125 {
126 	u8 *buf = NULL;
127 	int ret;
128 
129 	if (size > 0) {
130 		buf = kmalloc(size, GFP_KERNEL);
131 		if (buf == NULL)
132 			return -ENOMEM;
133 
134 		memcpy(buf, data, size);
135 	}
136 
137 	/* note: if successful returns number of bytes transfered */
138 	ret = usb_control_msg(ar_usb->udev,
139 			      usb_sndctrlpipe(ar_usb->udev, 0),
140 			      req,
141 			      USB_DIR_OUT | USB_TYPE_VENDOR |
142 			      USB_RECIP_DEVICE, value, index, buf,
143 			      size, 1000);
144 
145 	if (ret < 0) {
146 		ath6kl_dbg(ATH6KL_DBG_USB, "%s failed,result = %d\n",
147 			   __func__, ret);
148 	}
149 
150 	kfree(buf);
151 
152 	return 0;
153 }
154 
155 static int ath6kl_usb_submit_ctrl_in(struct ath6kl_usb *ar_usb,
156 				  u8 req, u16 value, u16 index, void *data,
157 				  u32 size)
158 {
159 	u8 *buf = NULL;
160 	int ret;
161 
162 	if (size > 0) {
163 		buf = kmalloc(size, GFP_KERNEL);
164 		if (buf == NULL)
165 			return -ENOMEM;
166 	}
167 
168 	/* note: if successful returns number of bytes transfered */
169 	ret = usb_control_msg(ar_usb->udev,
170 				 usb_rcvctrlpipe(ar_usb->udev, 0),
171 				 req,
172 				 USB_DIR_IN | USB_TYPE_VENDOR |
173 				 USB_RECIP_DEVICE, value, index, buf,
174 				 size, 2 * HZ);
175 
176 	if (ret < 0) {
177 		ath6kl_dbg(ATH6KL_DBG_USB, "%s failed,result = %d\n",
178 			   __func__, ret);
179 	}
180 
181 	memcpy((u8 *) data, buf, size);
182 
183 	kfree(buf);
184 
185 	return 0;
186 }
187 
188 static int ath6kl_usb_ctrl_msg_exchange(struct ath6kl_usb *ar_usb,
189 				     u8 req_val, u8 *req_buf, u32 req_len,
190 				     u8 resp_val, u8 *resp_buf, u32 *resp_len)
191 {
192 	int ret;
193 
194 	/* send command */
195 	ret = ath6kl_usb_submit_ctrl_out(ar_usb, req_val, 0, 0,
196 					 req_buf, req_len);
197 
198 	if (ret != 0)
199 		return ret;
200 
201 	if (resp_buf == NULL) {
202 		/* no expected response */
203 		return ret;
204 	}
205 
206 	/* get response */
207 	ret = ath6kl_usb_submit_ctrl_in(ar_usb, resp_val, 0, 0,
208 					resp_buf, *resp_len);
209 
210 	return ret;
211 }
212 
213 static int ath6kl_usb_diag_read32(struct ath6kl *ar, u32 address, u32 *data)
214 {
215 	struct ath6kl_usb *ar_usb = ar->hif_priv;
216 	struct ath6kl_usb_ctrl_diag_resp_read *resp;
217 	struct ath6kl_usb_ctrl_diag_cmd_read *cmd;
218 	u32 resp_len;
219 	int ret;
220 
221 	cmd = (struct ath6kl_usb_ctrl_diag_cmd_read *) ar_usb->diag_cmd_buffer;
222 
223 	memset(cmd, 0, sizeof(*cmd));
224 	cmd->cmd = ATH6KL_USB_CTRL_DIAG_CC_READ;
225 	cmd->address = cpu_to_le32(address);
226 	resp_len = sizeof(*resp);
227 
228 	ret = ath6kl_usb_ctrl_msg_exchange(ar_usb,
229 				ATH6KL_USB_CONTROL_REQ_DIAG_CMD,
230 				(u8 *) cmd,
231 				sizeof(struct ath6kl_usb_ctrl_diag_cmd_write),
232 				ATH6KL_USB_CONTROL_REQ_DIAG_RESP,
233 				ar_usb->diag_resp_buffer, &resp_len);
234 
235 	if (ret)
236 		return ret;
237 
238 	resp = (struct ath6kl_usb_ctrl_diag_resp_read *)
239 		ar_usb->diag_resp_buffer;
240 
241 	*data = le32_to_cpu(resp->value);
242 
243 	return ret;
244 }
245 
246 static int ath6kl_usb_diag_write32(struct ath6kl *ar, u32 address, __le32 data)
247 {
248 	struct ath6kl_usb *ar_usb = ar->hif_priv;
249 	struct ath6kl_usb_ctrl_diag_cmd_write *cmd;
250 
251 	cmd = (struct ath6kl_usb_ctrl_diag_cmd_write *) ar_usb->diag_cmd_buffer;
252 
253 	memset(cmd, 0, sizeof(struct ath6kl_usb_ctrl_diag_cmd_write));
254 	cmd->cmd = cpu_to_le32(ATH6KL_USB_CTRL_DIAG_CC_WRITE);
255 	cmd->address = cpu_to_le32(address);
256 	cmd->value = data;
257 
258 	return ath6kl_usb_ctrl_msg_exchange(ar_usb,
259 					    ATH6KL_USB_CONTROL_REQ_DIAG_CMD,
260 					    (u8 *) cmd,
261 					    sizeof(*cmd),
262 					    0, NULL, NULL);
263 
264 }
265 
266 static int ath6kl_usb_bmi_read(struct ath6kl *ar, u8 *buf, u32 len)
267 {
268 	struct ath6kl_usb *ar_usb = ar->hif_priv;
269 	int ret;
270 
271 	/* get response */
272 	ret = ath6kl_usb_submit_ctrl_in(ar_usb,
273 					ATH6KL_USB_CONTROL_REQ_RECV_BMI_RESP,
274 					0, 0, buf, len);
275 	if (ret != 0) {
276 		ath6kl_err("Unable to read the bmi data from the device: %d\n",
277 			   ret);
278 		return ret;
279 	}
280 
281 	return 0;
282 }
283 
284 static int ath6kl_usb_bmi_write(struct ath6kl *ar, u8 *buf, u32 len)
285 {
286 	struct ath6kl_usb *ar_usb = ar->hif_priv;
287 	int ret;
288 
289 	/* send command */
290 	ret = ath6kl_usb_submit_ctrl_out(ar_usb,
291 					 ATH6KL_USB_CONTROL_REQ_SEND_BMI_CMD,
292 					 0, 0, buf, len);
293 	if (ret != 0) {
294 		ath6kl_err("unable to send the bmi data to the device: %d\n",
295 			   ret);
296 		return ret;
297 	}
298 
299 	return 0;
300 }
301 
302 static int ath6kl_usb_power_on(struct ath6kl *ar)
303 {
304 	return 0;
305 }
306 
307 static int ath6kl_usb_power_off(struct ath6kl *ar)
308 {
309 	return 0;
310 }
311 
312 static const struct ath6kl_hif_ops ath6kl_usb_ops = {
313 	.diag_read32 = ath6kl_usb_diag_read32,
314 	.diag_write32 = ath6kl_usb_diag_write32,
315 	.bmi_read = ath6kl_usb_bmi_read,
316 	.bmi_write = ath6kl_usb_bmi_write,
317 	.power_on = ath6kl_usb_power_on,
318 	.power_off = ath6kl_usb_power_off,
319 };
320 
321 /* ath6kl usb driver registered functions */
322 static int ath6kl_usb_probe(struct usb_interface *interface,
323 			    const struct usb_device_id *id)
324 {
325 	struct usb_device *dev = interface_to_usbdev(interface);
326 	struct ath6kl *ar;
327 	struct ath6kl_usb *ar_usb = NULL;
328 	int vendor_id, product_id;
329 	int ret = 0;
330 
331 	usb_get_dev(dev);
332 
333 	vendor_id = le16_to_cpu(dev->descriptor.idVendor);
334 	product_id = le16_to_cpu(dev->descriptor.idProduct);
335 
336 	ath6kl_dbg(ATH6KL_DBG_USB, "vendor_id = %04x\n", vendor_id);
337 	ath6kl_dbg(ATH6KL_DBG_USB, "product_id = %04x\n", product_id);
338 
339 	if (interface->cur_altsetting)
340 		ath6kl_dbg(ATH6KL_DBG_USB, "USB Interface %d\n",
341 			   interface->cur_altsetting->desc.bInterfaceNumber);
342 
343 
344 	if (dev->speed == USB_SPEED_HIGH)
345 		ath6kl_dbg(ATH6KL_DBG_USB, "USB 2.0 Host\n");
346 	else
347 		ath6kl_dbg(ATH6KL_DBG_USB, "USB 1.1 Host\n");
348 
349 	ar_usb = ath6kl_usb_create(interface);
350 
351 	if (ar_usb == NULL) {
352 		ret = -ENOMEM;
353 		goto err_usb_put;
354 	}
355 
356 	ar = ath6kl_core_create(&ar_usb->udev->dev);
357 	if (ar == NULL) {
358 		ath6kl_err("Failed to alloc ath6kl core\n");
359 		ret = -ENOMEM;
360 		goto err_usb_destroy;
361 	}
362 
363 	ar->hif_priv = ar_usb;
364 	ar->hif_type = ATH6KL_HIF_TYPE_USB;
365 	ar->hif_ops = &ath6kl_usb_ops;
366 	ar->mbox_info.block_size = 16;
367 	ar->bmi.max_data_size = 252;
368 
369 	ar_usb->ar = ar;
370 
371 	ret = ath6kl_core_init(ar);
372 	if (ret) {
373 		ath6kl_err("Failed to init ath6kl core: %d\n", ret);
374 		goto err_core_free;
375 	}
376 
377 	return ret;
378 
379 err_core_free:
380 	ath6kl_core_destroy(ar);
381 err_usb_destroy:
382 	ath6kl_usb_destroy(ar_usb);
383 err_usb_put:
384 	usb_put_dev(dev);
385 
386 	return ret;
387 }
388 
389 static void ath6kl_usb_remove(struct usb_interface *interface)
390 {
391 	usb_put_dev(interface_to_usbdev(interface));
392 	ath6kl_usb_device_detached(interface);
393 }
394 
395 /* table of devices that work with this driver */
396 static struct usb_device_id ath6kl_usb_ids[] = {
397 	{USB_DEVICE(0x0cf3, 0x9374)},
398 	{ /* Terminating entry */ },
399 };
400 
401 MODULE_DEVICE_TABLE(usb, ath6kl_usb_ids);
402 
403 static struct usb_driver ath6kl_usb_driver = {
404 	.name = "ath6kl_usb",
405 	.probe = ath6kl_usb_probe,
406 	.disconnect = ath6kl_usb_remove,
407 	.id_table = ath6kl_usb_ids,
408 };
409 
410 static int ath6kl_usb_init(void)
411 {
412 	usb_register(&ath6kl_usb_driver);
413 	return 0;
414 }
415 
416 static void ath6kl_usb_exit(void)
417 {
418 	usb_deregister(&ath6kl_usb_driver);
419 }
420 
421 module_init(ath6kl_usb_init);
422 module_exit(ath6kl_usb_exit);
423 
424 MODULE_AUTHOR("Atheros Communications, Inc.");
425 MODULE_DESCRIPTION("Driver support for Atheros AR600x USB devices");
426 MODULE_LICENSE("Dual BSD/GPL");
427 MODULE_FIRMWARE(AR6004_HW_1_0_FIRMWARE_FILE);
428 MODULE_FIRMWARE(AR6004_HW_1_0_BOARD_DATA_FILE);
429 MODULE_FIRMWARE(AR6004_HW_1_0_DEFAULT_BOARD_DATA_FILE);
430 MODULE_FIRMWARE(AR6004_HW_1_1_FIRMWARE_FILE);
431 MODULE_FIRMWARE(AR6004_HW_1_1_BOARD_DATA_FILE);
432 MODULE_FIRMWARE(AR6004_HW_1_1_DEFAULT_BOARD_DATA_FILE);
433