blob: 1ee1c8e2b9d4aa855e9f2ac5fd082295c11e4ed7 [file] [log] [blame]
Shalabh Jainb0037c02013-01-18 12:47:40 -08001/* Copyright (c) 2011-2013, The Linux Foundation. All rights reserved.
Jack Phame32cf322011-09-26 10:20:17 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
Jack Pham76e61dd2012-06-14 18:48:53 -070013/* add additional information to our printk's */
14#define pr_fmt(fmt) "%s: " fmt "\n", __func__
15
Jack Phame32cf322011-09-26 10:20:17 -070016#include <linux/kernel.h>
17#include <linux/errno.h>
18#include <linux/init.h>
19#include <linux/slab.h>
20#include <linux/module.h>
21#include <linux/kref.h>
Jack Phamb1795542012-09-04 17:13:37 -070022#include <linux/mutex.h>
Jack Phame32cf322011-09-26 10:20:17 -070023#include <linux/platform_device.h>
Jack Phamcca03322012-06-14 19:00:31 -070024#include <linux/ratelimit.h>
Jack Phame32cf322011-09-26 10:20:17 -070025#include <linux/uaccess.h>
26#include <linux/usb.h>
Jack Pham9a08de42012-02-06 15:43:23 -080027#include <linux/debugfs.h>
Jack Phame32cf322011-09-26 10:20:17 -070028#include <mach/diag_bridge.h>
29
30#define DRIVER_DESC "USB host diag bridge driver"
Jack Phamf6ed5582011-11-02 16:08:31 -070031#define DRIVER_VERSION "1.0"
Jack Phame32cf322011-09-26 10:20:17 -070032
Shalabh Jainb0037c02013-01-18 12:47:40 -080033#define MAX_DIAG_BRIDGE_DEVS 2
Hemant Kumara3d4a6e2012-10-16 17:57:08 -070034#define AUTOSUSP_DELAY_WITH_USB 1000
35
Jack Phame32cf322011-09-26 10:20:17 -070036struct diag_bridge {
37 struct usb_device *udev;
38 struct usb_interface *ifc;
39 struct usb_anchor submitted;
40 __u8 in_epAddr;
41 __u8 out_epAddr;
Jack Pham3ae820f2011-12-14 16:21:04 -080042 int err;
Jack Phame32cf322011-09-26 10:20:17 -070043 struct kref kref;
Jack Phamb1795542012-09-04 17:13:37 -070044 struct mutex ifc_mutex;
Jack Phame32cf322011-09-26 10:20:17 -070045 struct diag_bridge_ops *ops;
46 struct platform_device *pdev;
Hemant Kumara3d4a6e2012-10-16 17:57:08 -070047 unsigned default_autosusp_delay;
Jack Pham9a08de42012-02-06 15:43:23 -080048
49 /* debugging counters */
50 unsigned long bytes_to_host;
51 unsigned long bytes_to_mdm;
52 unsigned pending_reads;
53 unsigned pending_writes;
Jack Phame32cf322011-09-26 10:20:17 -070054};
Shalabh Jainb0037c02013-01-18 12:47:40 -080055struct diag_bridge *__dev[MAX_DIAG_BRIDGE_DEVS];
Jack Phame32cf322011-09-26 10:20:17 -070056
Shalabh Jainb0037c02013-01-18 12:47:40 -080057int diag_bridge_open(int id, struct diag_bridge_ops *ops)
Jack Phame32cf322011-09-26 10:20:17 -070058{
Shalabh Jainb0037c02013-01-18 12:47:40 -080059 struct diag_bridge *dev;
Jack Phame32cf322011-09-26 10:20:17 -070060
Shalabh Jainb0037c02013-01-18 12:47:40 -080061 if (id < 0 || id >= MAX_DIAG_BRIDGE_DEVS) {
62 pr_err("Invalid device ID");
63 return -ENODEV;
64 }
65
66 dev = __dev[id];
Jack Phame32cf322011-09-26 10:20:17 -070067 if (!dev) {
Jack Pham76e61dd2012-06-14 18:48:53 -070068 pr_err("dev is null");
Jack Phame32cf322011-09-26 10:20:17 -070069 return -ENODEV;
70 }
71
Jack Phamfbd22552012-09-07 17:29:08 -070072 if (dev->ops) {
73 pr_err("bridge already opened");
74 return -EALREADY;
75 }
76
Jack Phame32cf322011-09-26 10:20:17 -070077 dev->ops = ops;
Jack Pham3ae820f2011-12-14 16:21:04 -080078 dev->err = 0;
Jack Phame32cf322011-09-26 10:20:17 -070079
Hemant Kumara3d4a6e2012-10-16 17:57:08 -070080#ifdef CONFIG_PM_RUNTIME
81 dev->default_autosusp_delay = dev->udev->dev.power.autosuspend_delay;
82#endif
83 pm_runtime_set_autosuspend_delay(&dev->udev->dev,
84 AUTOSUSP_DELAY_WITH_USB);
85
Jack Pham76e61dd2012-06-14 18:48:53 -070086 kref_get(&dev->kref);
87
Jack Phame32cf322011-09-26 10:20:17 -070088 return 0;
89}
Jack Phamf6ed5582011-11-02 16:08:31 -070090EXPORT_SYMBOL(diag_bridge_open);
Jack Phame32cf322011-09-26 10:20:17 -070091
Jack Pham76e61dd2012-06-14 18:48:53 -070092static void diag_bridge_delete(struct kref *kref)
93{
94 struct diag_bridge *dev = container_of(kref, struct diag_bridge, kref);
Shalabh Jainb0037c02013-01-18 12:47:40 -080095 int id = dev->pdev->id;
Jack Pham76e61dd2012-06-14 18:48:53 -070096
97 usb_put_dev(dev->udev);
Shalabh Jainb0037c02013-01-18 12:47:40 -080098 __dev[id] = 0;
Jack Pham76e61dd2012-06-14 18:48:53 -070099 kfree(dev);
100}
101
Shalabh Jainb0037c02013-01-18 12:47:40 -0800102void diag_bridge_close(int id)
Jack Phame32cf322011-09-26 10:20:17 -0700103{
Shalabh Jainb0037c02013-01-18 12:47:40 -0800104 struct diag_bridge *dev;
Jack Phame32cf322011-09-26 10:20:17 -0700105
Shalabh Jainb0037c02013-01-18 12:47:40 -0800106 if (id < 0 || id >= MAX_DIAG_BRIDGE_DEVS) {
107 pr_err("Invalid device ID");
108 return;
109 }
110
111 dev = __dev[id];
Jack Phamfbd22552012-09-07 17:29:08 -0700112 if (!dev) {
113 pr_err("dev is null");
114 return;
115 }
116
117 if (!dev->ops) {
118 pr_err("can't close bridge that was not open");
119 return;
120 }
121
Jack Phame8741502012-06-13 17:34:07 -0700122 dev_dbg(&dev->ifc->dev, "%s:\n", __func__);
Jack Phame32cf322011-09-26 10:20:17 -0700123
124 usb_kill_anchored_urbs(&dev->submitted);
Jack Phame32cf322011-09-26 10:20:17 -0700125 dev->ops = 0;
Hemant Kumara3d4a6e2012-10-16 17:57:08 -0700126
127 pm_runtime_set_autosuspend_delay(&dev->udev->dev,
128 dev->default_autosusp_delay);
129
Jack Pham76e61dd2012-06-14 18:48:53 -0700130 kref_put(&dev->kref, diag_bridge_delete);
Jack Phame32cf322011-09-26 10:20:17 -0700131}
Jack Phamf6ed5582011-11-02 16:08:31 -0700132EXPORT_SYMBOL(diag_bridge_close);
Jack Phame32cf322011-09-26 10:20:17 -0700133
Jack Phamf6ed5582011-11-02 16:08:31 -0700134static void diag_bridge_read_cb(struct urb *urb)
Jack Phame32cf322011-09-26 10:20:17 -0700135{
136 struct diag_bridge *dev = urb->context;
137 struct diag_bridge_ops *cbs = dev->ops;
138
Jack Phame8741502012-06-13 17:34:07 -0700139 dev_dbg(&dev->ifc->dev, "%s: status:%d actual:%d\n", __func__,
Jack Phame32cf322011-09-26 10:20:17 -0700140 urb->status, urb->actual_length);
Jack Phame32cf322011-09-26 10:20:17 -0700141
Jack Pham3ec42b62012-08-31 15:36:27 -0700142 /* save error so that subsequent read/write returns ENODEV */
143 if (urb->status == -EPROTO)
Jack Pham3ae820f2011-12-14 16:21:04 -0800144 dev->err = urb->status;
Jack Pham3ae820f2011-12-14 16:21:04 -0800145
Vamsi Krishna90125642012-05-12 16:06:13 -0700146 if (cbs && cbs->read_complete_cb)
147 cbs->read_complete_cb(cbs->ctxt,
Jack Phame32cf322011-09-26 10:20:17 -0700148 urb->transfer_buffer,
149 urb->transfer_buffer_length,
Jack Phamf6ed5582011-11-02 16:08:31 -0700150 urb->status < 0 ? urb->status : urb->actual_length);
Jack Pham9a08de42012-02-06 15:43:23 -0800151
152 dev->bytes_to_host += urb->actual_length;
153 dev->pending_reads--;
Jack Pham76e61dd2012-06-14 18:48:53 -0700154 kref_put(&dev->kref, diag_bridge_delete);
Jack Phame32cf322011-09-26 10:20:17 -0700155}
156
Shalabh Jainb0037c02013-01-18 12:47:40 -0800157int diag_bridge_read(int id, char *data, int size)
Jack Phame32cf322011-09-26 10:20:17 -0700158{
159 struct urb *urb = NULL;
160 unsigned int pipe;
Shalabh Jainb0037c02013-01-18 12:47:40 -0800161 struct diag_bridge *dev;
Jack Phame32cf322011-09-26 10:20:17 -0700162 int ret;
163
Shalabh Jainb0037c02013-01-18 12:47:40 -0800164 if (id < 0 || id >= MAX_DIAG_BRIDGE_DEVS) {
165 pr_err("Invalid device ID");
166 return -ENODEV;
167 }
168
Jack Pham76e61dd2012-06-14 18:48:53 -0700169 pr_debug("reading %d bytes", size);
170
Shalabh Jainb0037c02013-01-18 12:47:40 -0800171 dev = __dev[id];
Jack Phamb1795542012-09-04 17:13:37 -0700172 if (!dev) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700173 pr_err("device is disconnected");
174 return -ENODEV;
175 }
176
Jack Phamb1795542012-09-04 17:13:37 -0700177 mutex_lock(&dev->ifc_mutex);
178 if (!dev->ifc) {
179 ret = -ENODEV;
180 goto error;
181 }
182
Jack Pham76e61dd2012-06-14 18:48:53 -0700183 if (!dev->ops) {
184 pr_err("bridge is not open");
Jack Phamb1795542012-09-04 17:13:37 -0700185 ret = -ENODEV;
186 goto error;
Jack Pham76e61dd2012-06-14 18:48:53 -0700187 }
Jack Phame32cf322011-09-26 10:20:17 -0700188
189 if (!size) {
Jack Phame8741502012-06-13 17:34:07 -0700190 dev_err(&dev->ifc->dev, "invalid size:%d\n", size);
Jack Phamb1795542012-09-04 17:13:37 -0700191 ret = -EINVAL;
192 goto error;
Jack Phame32cf322011-09-26 10:20:17 -0700193 }
194
Jack Pham3ae820f2011-12-14 16:21:04 -0800195 /* if there was a previous unrecoverable error, just quit */
Jack Phamb1795542012-09-04 17:13:37 -0700196 if (dev->err) {
197 ret = -ENODEV;
198 goto error;
199 }
Jack Pham3ae820f2011-12-14 16:21:04 -0800200
Jack Pham76e61dd2012-06-14 18:48:53 -0700201 kref_get(&dev->kref);
202
Jack Phamb60775a2012-02-14 17:57:41 -0800203 urb = usb_alloc_urb(0, GFP_KERNEL);
Jack Phame32cf322011-09-26 10:20:17 -0700204 if (!urb) {
Jack Phame8741502012-06-13 17:34:07 -0700205 dev_err(&dev->ifc->dev, "unable to allocate urb\n");
Jack Pham76e61dd2012-06-14 18:48:53 -0700206 ret = -ENOMEM;
Jack Phamb1795542012-09-04 17:13:37 -0700207 goto put_error;
Jack Phame32cf322011-09-26 10:20:17 -0700208 }
209
Jack Phamb60775a2012-02-14 17:57:41 -0800210 ret = usb_autopm_get_interface(dev->ifc);
Jack Phamcca03322012-06-14 19:00:31 -0700211 if (ret < 0 && ret != -EAGAIN && ret != -EACCES) {
212 pr_err_ratelimited("read: autopm_get failed:%d", ret);
Jack Pham76e61dd2012-06-14 18:48:53 -0700213 goto free_error;
Jack Phamb60775a2012-02-14 17:57:41 -0800214 }
215
Jack Phame32cf322011-09-26 10:20:17 -0700216 pipe = usb_rcvbulkpipe(dev->udev, dev->in_epAddr);
217 usb_fill_bulk_urb(urb, dev->udev, pipe, data, size,
Jack Phamf6ed5582011-11-02 16:08:31 -0700218 diag_bridge_read_cb, dev);
Jack Phame32cf322011-09-26 10:20:17 -0700219 usb_anchor_urb(urb, &dev->submitted);
Jack Pham9a08de42012-02-06 15:43:23 -0800220 dev->pending_reads++;
Jack Phame32cf322011-09-26 10:20:17 -0700221
Jack Phamb60775a2012-02-14 17:57:41 -0800222 ret = usb_submit_urb(urb, GFP_KERNEL);
Jack Phame32cf322011-09-26 10:20:17 -0700223 if (ret) {
Jack Phamcca03322012-06-14 19:00:31 -0700224 pr_err_ratelimited("submitting urb failed err:%d", ret);
Jack Pham9a08de42012-02-06 15:43:23 -0800225 dev->pending_reads--;
Jack Phame32cf322011-09-26 10:20:17 -0700226 usb_unanchor_urb(urb);
Jack Phame32cf322011-09-26 10:20:17 -0700227 }
Jack Phamb60775a2012-02-14 17:57:41 -0800228 usb_autopm_put_interface(dev->ifc);
Jack Phame32cf322011-09-26 10:20:17 -0700229
Jack Pham76e61dd2012-06-14 18:48:53 -0700230free_error:
231 usb_free_urb(urb);
Jack Phamb1795542012-09-04 17:13:37 -0700232put_error:
Jack Pham76e61dd2012-06-14 18:48:53 -0700233 if (ret) /* otherwise this is done in the completion handler */
234 kref_put(&dev->kref, diag_bridge_delete);
Jack Phamb1795542012-09-04 17:13:37 -0700235error:
236 mutex_unlock(&dev->ifc_mutex);
Jack Pham76e61dd2012-06-14 18:48:53 -0700237 return ret;
Jack Phame32cf322011-09-26 10:20:17 -0700238}
Jack Phamf6ed5582011-11-02 16:08:31 -0700239EXPORT_SYMBOL(diag_bridge_read);
Jack Phame32cf322011-09-26 10:20:17 -0700240
Jack Phamf6ed5582011-11-02 16:08:31 -0700241static void diag_bridge_write_cb(struct urb *urb)
Jack Phame32cf322011-09-26 10:20:17 -0700242{
243 struct diag_bridge *dev = urb->context;
244 struct diag_bridge_ops *cbs = dev->ops;
245
Jack Phame8741502012-06-13 17:34:07 -0700246 dev_dbg(&dev->ifc->dev, "%s:\n", __func__);
Jack Phame32cf322011-09-26 10:20:17 -0700247
Jack Phamb60775a2012-02-14 17:57:41 -0800248 usb_autopm_put_interface_async(dev->ifc);
249
Jack Pham3ec42b62012-08-31 15:36:27 -0700250 /* save error so that subsequent read/write returns ENODEV */
251 if (urb->status == -EPROTO)
Jack Pham3ae820f2011-12-14 16:21:04 -0800252 dev->err = urb->status;
Jack Pham3ae820f2011-12-14 16:21:04 -0800253
Vamsi Krishna90125642012-05-12 16:06:13 -0700254 if (cbs && cbs->write_complete_cb)
255 cbs->write_complete_cb(cbs->ctxt,
Jack Phame32cf322011-09-26 10:20:17 -0700256 urb->transfer_buffer,
257 urb->transfer_buffer_length,
Jack Phamf6ed5582011-11-02 16:08:31 -0700258 urb->status < 0 ? urb->status : urb->actual_length);
Jack Pham9a08de42012-02-06 15:43:23 -0800259
260 dev->bytes_to_mdm += urb->actual_length;
261 dev->pending_writes--;
Jack Pham76e61dd2012-06-14 18:48:53 -0700262 kref_put(&dev->kref, diag_bridge_delete);
Jack Phame32cf322011-09-26 10:20:17 -0700263}
264
Shalabh Jainb0037c02013-01-18 12:47:40 -0800265int diag_bridge_write(int id, char *data, int size)
Jack Phame32cf322011-09-26 10:20:17 -0700266{
267 struct urb *urb = NULL;
268 unsigned int pipe;
Shalabh Jainb0037c02013-01-18 12:47:40 -0800269 struct diag_bridge *dev;
Jack Phame32cf322011-09-26 10:20:17 -0700270 int ret;
271
Shalabh Jainb0037c02013-01-18 12:47:40 -0800272 if (id < 0 || id >= MAX_DIAG_BRIDGE_DEVS) {
273 pr_err("Invalid device ID");
274 return -ENODEV;
275 }
276
Jack Pham76e61dd2012-06-14 18:48:53 -0700277 pr_debug("writing %d bytes", size);
278
Shalabh Jainb0037c02013-01-18 12:47:40 -0800279 dev = __dev[id];
Jack Phamb1795542012-09-04 17:13:37 -0700280 if (!dev) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700281 pr_err("device is disconnected");
282 return -ENODEV;
283 }
284
Jack Phamb1795542012-09-04 17:13:37 -0700285 mutex_lock(&dev->ifc_mutex);
286 if (!dev->ifc) {
287 ret = -ENODEV;
288 goto error;
289 }
290
Jack Pham76e61dd2012-06-14 18:48:53 -0700291 if (!dev->ops) {
292 pr_err("bridge is not open");
Jack Phamb1795542012-09-04 17:13:37 -0700293 ret = -ENODEV;
294 goto error;
Jack Pham76e61dd2012-06-14 18:48:53 -0700295 }
Jack Phame32cf322011-09-26 10:20:17 -0700296
297 if (!size) {
Jack Phame8741502012-06-13 17:34:07 -0700298 dev_err(&dev->ifc->dev, "invalid size:%d\n", size);
Jack Phamb1795542012-09-04 17:13:37 -0700299 ret = -EINVAL;
300 goto error;
Jack Phame32cf322011-09-26 10:20:17 -0700301 }
302
Jack Pham3ae820f2011-12-14 16:21:04 -0800303 /* if there was a previous unrecoverable error, just quit */
Jack Phamb1795542012-09-04 17:13:37 -0700304 if (dev->err) {
305 ret = -ENODEV;
306 goto error;
307 }
Jack Pham3ae820f2011-12-14 16:21:04 -0800308
Jack Pham76e61dd2012-06-14 18:48:53 -0700309 kref_get(&dev->kref);
310
Jack Phamb60775a2012-02-14 17:57:41 -0800311 urb = usb_alloc_urb(0, GFP_KERNEL);
Jack Phame32cf322011-09-26 10:20:17 -0700312 if (!urb) {
Jack Phame8741502012-06-13 17:34:07 -0700313 dev_err(&dev->ifc->dev, "unable to allocate urb\n");
Jack Pham76e61dd2012-06-14 18:48:53 -0700314 ret = -ENOMEM;
Jack Phamb1795542012-09-04 17:13:37 -0700315 goto put_error;
Jack Phame32cf322011-09-26 10:20:17 -0700316 }
317
Jack Phamb60775a2012-02-14 17:57:41 -0800318 ret = usb_autopm_get_interface(dev->ifc);
Jack Phamcca03322012-06-14 19:00:31 -0700319 if (ret < 0 && ret != -EAGAIN && ret != -EACCES) {
320 pr_err_ratelimited("write: autopm_get failed:%d", ret);
Jack Pham76e61dd2012-06-14 18:48:53 -0700321 goto free_error;
Jack Phamb60775a2012-02-14 17:57:41 -0800322 }
323
Jack Phame32cf322011-09-26 10:20:17 -0700324 pipe = usb_sndbulkpipe(dev->udev, dev->out_epAddr);
325 usb_fill_bulk_urb(urb, dev->udev, pipe, data, size,
Jack Phamf6ed5582011-11-02 16:08:31 -0700326 diag_bridge_write_cb, dev);
Jack Pham5a10c6f2012-06-26 11:41:28 -0700327 urb->transfer_flags |= URB_ZERO_PACKET;
Jack Phame32cf322011-09-26 10:20:17 -0700328 usb_anchor_urb(urb, &dev->submitted);
Jack Pham9a08de42012-02-06 15:43:23 -0800329 dev->pending_writes++;
Jack Phame32cf322011-09-26 10:20:17 -0700330
Jack Phamb60775a2012-02-14 17:57:41 -0800331 ret = usb_submit_urb(urb, GFP_KERNEL);
Jack Phame32cf322011-09-26 10:20:17 -0700332 if (ret) {
Jack Phamcca03322012-06-14 19:00:31 -0700333 pr_err_ratelimited("submitting urb failed err:%d", ret);
Jack Pham9a08de42012-02-06 15:43:23 -0800334 dev->pending_writes--;
Jack Phame32cf322011-09-26 10:20:17 -0700335 usb_unanchor_urb(urb);
Jack Phamb60775a2012-02-14 17:57:41 -0800336 usb_autopm_put_interface(dev->ifc);
Jack Pham76e61dd2012-06-14 18:48:53 -0700337 goto free_error;
Jack Phame32cf322011-09-26 10:20:17 -0700338 }
339
Jack Pham76e61dd2012-06-14 18:48:53 -0700340free_error:
Jack Phame32cf322011-09-26 10:20:17 -0700341 usb_free_urb(urb);
Jack Phamb1795542012-09-04 17:13:37 -0700342put_error:
Jack Pham76e61dd2012-06-14 18:48:53 -0700343 if (ret) /* otherwise this is done in the completion handler */
344 kref_put(&dev->kref, diag_bridge_delete);
Jack Phamb1795542012-09-04 17:13:37 -0700345error:
346 mutex_unlock(&dev->ifc_mutex);
Jack Pham76e61dd2012-06-14 18:48:53 -0700347 return ret;
Jack Phame32cf322011-09-26 10:20:17 -0700348}
Jack Phamf6ed5582011-11-02 16:08:31 -0700349EXPORT_SYMBOL(diag_bridge_write);
Jack Phame32cf322011-09-26 10:20:17 -0700350
Jack Pham9a08de42012-02-06 15:43:23 -0800351#if defined(CONFIG_DEBUG_FS)
352#define DEBUG_BUF_SIZE 512
353static ssize_t diag_read_stats(struct file *file, char __user *ubuf,
354 size_t count, loff_t *ppos)
355{
Jack Pham9a08de42012-02-06 15:43:23 -0800356 char *buf;
Shalabh Jainb0037c02013-01-18 12:47:40 -0800357 int i, ret = 0;
Jack Pham76e61dd2012-06-14 18:48:53 -0700358
Jack Pham9a08de42012-02-06 15:43:23 -0800359 buf = kzalloc(sizeof(char) * DEBUG_BUF_SIZE, GFP_KERNEL);
360 if (!buf)
361 return -ENOMEM;
362
Shalabh Jainb0037c02013-01-18 12:47:40 -0800363 for (i = 0; i < MAX_DIAG_BRIDGE_DEVS; i++) {
364 struct diag_bridge *dev = __dev[i];
365 if (!dev)
366 continue;
367
368 ret += scnprintf(buf, DEBUG_BUF_SIZE,
369 "epin:%d, epout:%d\n"
370 "bytes to host: %lu\n"
371 "bytes to mdm: %lu\n"
372 "pending reads: %u\n"
373 "pending writes: %u\n"
374 "last error: %d\n",
375 dev->in_epAddr, dev->out_epAddr,
376 dev->bytes_to_host, dev->bytes_to_mdm,
377 dev->pending_reads, dev->pending_writes,
378 dev->err);
379 }
Jack Pham9a08de42012-02-06 15:43:23 -0800380
381 ret = simple_read_from_buffer(ubuf, count, ppos, buf, ret);
382 kfree(buf);
383 return ret;
384}
385
386static ssize_t diag_reset_stats(struct file *file, const char __user *buf,
387 size_t count, loff_t *ppos)
388{
Shalabh Jainb0037c02013-01-18 12:47:40 -0800389 int i;
Jack Pham9a08de42012-02-06 15:43:23 -0800390
Shalabh Jainb0037c02013-01-18 12:47:40 -0800391 for (i = 0; i < MAX_DIAG_BRIDGE_DEVS; i++) {
392 struct diag_bridge *dev = __dev[i];
393 if (dev) {
394 dev->bytes_to_host = dev->bytes_to_mdm = 0;
395 dev->pending_reads = dev->pending_writes = 0;
396 }
Jack Pham76e61dd2012-06-14 18:48:53 -0700397 }
Jack Pham9a08de42012-02-06 15:43:23 -0800398
399 return count;
400}
401
402const struct file_operations diag_stats_ops = {
403 .read = diag_read_stats,
404 .write = diag_reset_stats,
405};
406
407static struct dentry *dent;
408
409static void diag_bridge_debugfs_init(void)
410{
411 struct dentry *dfile;
412
413 dent = debugfs_create_dir("diag_bridge", 0);
414 if (IS_ERR(dent))
415 return;
416
417 dfile = debugfs_create_file("status", 0444, dent, 0, &diag_stats_ops);
418 if (!dfile || IS_ERR(dfile))
419 debugfs_remove(dent);
420}
421
422static void diag_bridge_debugfs_cleanup(void)
423{
424 if (dent) {
425 debugfs_remove_recursive(dent);
426 dent = NULL;
427 }
428}
429#else
430static inline void diag_bridge_debugfs_init(void) { }
431static inline void diag_bridge_debugfs_cleanup(void) { }
432#endif
433
Jack Phame32cf322011-09-26 10:20:17 -0700434static int
Jack Phamf6ed5582011-11-02 16:08:31 -0700435diag_bridge_probe(struct usb_interface *ifc, const struct usb_device_id *id)
Jack Phame32cf322011-09-26 10:20:17 -0700436{
437 struct diag_bridge *dev;
438 struct usb_host_interface *ifc_desc;
439 struct usb_endpoint_descriptor *ep_desc;
Shalabh Jainb0037c02013-01-18 12:47:40 -0800440 int i, devid, ret = -ENOMEM;
Jack Phame32cf322011-09-26 10:20:17 -0700441 __u8 ifc_num;
442
Jack Pham76e61dd2012-06-14 18:48:53 -0700443 pr_debug("id:%lu", id->driver_info);
Jack Phame32cf322011-09-26 10:20:17 -0700444
445 ifc_num = ifc->cur_altsetting->desc.bInterfaceNumber;
446
447 /* is this interface supported ? */
Jack Phamf81139b2013-01-24 16:42:47 -0800448 if (ifc_num != (id->driver_info & 0xFF))
Jack Phame32cf322011-09-26 10:20:17 -0700449 return -ENODEV;
450
Jack Phamf81139b2013-01-24 16:42:47 -0800451 devid = (id->driver_info >> 8) & 0xFF;
Shalabh Jainb0037c02013-01-18 12:47:40 -0800452 if (devid < 0 || devid >= MAX_DIAG_BRIDGE_DEVS)
453 return -ENODEV;
454
455 /* already probed? */
456 if (__dev[devid]) {
457 pr_err("Diag device already probed");
458 return -ENODEV;
459 }
460
Jack Phame32cf322011-09-26 10:20:17 -0700461 dev = kzalloc(sizeof(*dev), GFP_KERNEL);
462 if (!dev) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700463 pr_err("unable to allocate dev");
Jack Phame32cf322011-09-26 10:20:17 -0700464 return -ENOMEM;
465 }
Shalabh Jainb0037c02013-01-18 12:47:40 -0800466 dev->pdev = platform_device_alloc("diag_bridge", devid);
Jack Phame32cf322011-09-26 10:20:17 -0700467 if (!dev->pdev) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700468 pr_err("unable to allocate platform device");
Jack Phame32cf322011-09-26 10:20:17 -0700469 kfree(dev);
470 return -ENOMEM;
471 }
Shalabh Jainb0037c02013-01-18 12:47:40 -0800472 __dev[devid] = dev;
Jack Phame32cf322011-09-26 10:20:17 -0700473
474 dev->udev = usb_get_dev(interface_to_usbdev(ifc));
475 dev->ifc = ifc;
476 kref_init(&dev->kref);
Jack Phamb1795542012-09-04 17:13:37 -0700477 mutex_init(&dev->ifc_mutex);
Jack Phame32cf322011-09-26 10:20:17 -0700478 init_usb_anchor(&dev->submitted);
479
480 ifc_desc = ifc->cur_altsetting;
481 for (i = 0; i < ifc_desc->desc.bNumEndpoints; i++) {
482 ep_desc = &ifc_desc->endpoint[i].desc;
483
484 if (!dev->in_epAddr && usb_endpoint_is_bulk_in(ep_desc))
485 dev->in_epAddr = ep_desc->bEndpointAddress;
486
487 if (!dev->out_epAddr && usb_endpoint_is_bulk_out(ep_desc))
488 dev->out_epAddr = ep_desc->bEndpointAddress;
489 }
490
491 if (!(dev->in_epAddr && dev->out_epAddr)) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700492 pr_err("could not find bulk in and bulk out endpoints");
Jack Phame32cf322011-09-26 10:20:17 -0700493 ret = -ENODEV;
494 goto error;
495 }
496
497 usb_set_intfdata(ifc, dev);
Jack Pham9a08de42012-02-06 15:43:23 -0800498 diag_bridge_debugfs_init();
Jack Phame32cf322011-09-26 10:20:17 -0700499 platform_device_add(dev->pdev);
500
Jack Phame8741502012-06-13 17:34:07 -0700501 dev_dbg(&dev->ifc->dev, "%s: complete\n", __func__);
Jack Phame32cf322011-09-26 10:20:17 -0700502
503 return 0;
504
505error:
506 if (dev)
Jack Phamf6ed5582011-11-02 16:08:31 -0700507 kref_put(&dev->kref, diag_bridge_delete);
Jack Phame32cf322011-09-26 10:20:17 -0700508
509 return ret;
510}
511
Jack Phamf6ed5582011-11-02 16:08:31 -0700512static void diag_bridge_disconnect(struct usb_interface *ifc)
Jack Phame32cf322011-09-26 10:20:17 -0700513{
514 struct diag_bridge *dev = usb_get_intfdata(ifc);
515
Jack Phame8741502012-06-13 17:34:07 -0700516 dev_dbg(&dev->ifc->dev, "%s:\n", __func__);
Jack Phame32cf322011-09-26 10:20:17 -0700517
Jack Pham1b1ba472012-06-13 16:40:15 -0700518 platform_device_unregister(dev->pdev);
Jack Phamb1795542012-09-04 17:13:37 -0700519 mutex_lock(&dev->ifc_mutex);
Jack Pham76e61dd2012-06-14 18:48:53 -0700520 dev->ifc = NULL;
Jack Phamb1795542012-09-04 17:13:37 -0700521 mutex_unlock(&dev->ifc_mutex);
Jack Pham9a08de42012-02-06 15:43:23 -0800522 diag_bridge_debugfs_cleanup();
Jack Phamf6ed5582011-11-02 16:08:31 -0700523 kref_put(&dev->kref, diag_bridge_delete);
Jack Phame32cf322011-09-26 10:20:17 -0700524 usb_set_intfdata(ifc, NULL);
525}
526
Jack Phamb60775a2012-02-14 17:57:41 -0800527static int diag_bridge_suspend(struct usb_interface *ifc, pm_message_t message)
528{
529 struct diag_bridge *dev = usb_get_intfdata(ifc);
530 struct diag_bridge_ops *cbs = dev->ops;
531 int ret = 0;
532
533 if (cbs && cbs->suspend) {
534 ret = cbs->suspend(cbs->ctxt);
535 if (ret) {
Jack Phame8741502012-06-13 17:34:07 -0700536 dev_dbg(&dev->ifc->dev,
Jack Phamb60775a2012-02-14 17:57:41 -0800537 "%s: diag veto'd suspend\n", __func__);
538 return ret;
539 }
540
541 usb_kill_anchored_urbs(&dev->submitted);
542 }
543
544 return ret;
545}
546
547static int diag_bridge_resume(struct usb_interface *ifc)
548{
549 struct diag_bridge *dev = usb_get_intfdata(ifc);
550 struct diag_bridge_ops *cbs = dev->ops;
551
552
553 if (cbs && cbs->resume)
554 cbs->resume(cbs->ctxt);
555
556 return 0;
557}
Jack Phame32cf322011-09-26 10:20:17 -0700558
559#define VALID_INTERFACE_NUM 0
Jack Phamf81139b2013-01-24 16:42:47 -0800560#define DEV_ID(n) ((n)<<8)
561
Jack Phamf6ed5582011-11-02 16:08:31 -0700562static const struct usb_device_id diag_bridge_ids[] = {
Hemant Kumar7a067f12012-01-05 15:35:15 -0800563 { USB_DEVICE(0x5c6, 0x9001),
Jack Phamf81139b2013-01-24 16:42:47 -0800564 .driver_info = VALID_INTERFACE_NUM | DEV_ID(0), },
Hemant Kumar7a067f12012-01-05 15:35:15 -0800565 { USB_DEVICE(0x5c6, 0x9034),
Jack Phamf81139b2013-01-24 16:42:47 -0800566 .driver_info = VALID_INTERFACE_NUM | DEV_ID(0), },
Hemant Kumar7f236832011-12-16 12:13:21 -0800567 { USB_DEVICE(0x5c6, 0x9048),
Jack Phamf81139b2013-01-24 16:42:47 -0800568 .driver_info = VALID_INTERFACE_NUM | DEV_ID(0), },
Hemant Kumarcfc0dad2012-02-21 14:34:39 -0800569 { USB_DEVICE(0x5c6, 0x904C),
Jack Phamf81139b2013-01-24 16:42:47 -0800570 .driver_info = VALID_INTERFACE_NUM | DEV_ID(0), },
Hemant Kumarce3c5bf2012-12-06 15:52:02 -0800571 { USB_DEVICE(0x5c6, 0x9075),
Jack Phamf81139b2013-01-24 16:42:47 -0800572 .driver_info = VALID_INTERFACE_NUM | DEV_ID(0), },
573 { USB_DEVICE(0x5c6, 0x9079),
574 .driver_info = VALID_INTERFACE_NUM | DEV_ID(1), },
Jack Phame32cf322011-09-26 10:20:17 -0700575
576 {} /* terminating entry */
577};
Jack Phamf6ed5582011-11-02 16:08:31 -0700578MODULE_DEVICE_TABLE(usb, diag_bridge_ids);
Jack Phame32cf322011-09-26 10:20:17 -0700579
Jack Phamf6ed5582011-11-02 16:08:31 -0700580static struct usb_driver diag_bridge_driver = {
581 .name = "diag_bridge",
582 .probe = diag_bridge_probe,
583 .disconnect = diag_bridge_disconnect,
Jack Phamb60775a2012-02-14 17:57:41 -0800584 .suspend = diag_bridge_suspend,
585 .resume = diag_bridge_resume,
Jack Phamf6ed5582011-11-02 16:08:31 -0700586 .id_table = diag_bridge_ids,
Jack Phamb60775a2012-02-14 17:57:41 -0800587 .supports_autosuspend = 1,
Jack Phame32cf322011-09-26 10:20:17 -0700588};
589
Jack Phamf6ed5582011-11-02 16:08:31 -0700590static int __init diag_bridge_init(void)
Jack Phame32cf322011-09-26 10:20:17 -0700591{
592 int ret;
593
Jack Phamf6ed5582011-11-02 16:08:31 -0700594 ret = usb_register(&diag_bridge_driver);
Jack Phame32cf322011-09-26 10:20:17 -0700595 if (ret) {
Jack Pham76e61dd2012-06-14 18:48:53 -0700596 pr_err("unable to register diag driver");
Jack Phame32cf322011-09-26 10:20:17 -0700597 return ret;
598 }
599
600 return 0;
601}
602
Jack Phamf6ed5582011-11-02 16:08:31 -0700603static void __exit diag_bridge_exit(void)
Jack Phame32cf322011-09-26 10:20:17 -0700604{
Jack Phamf6ed5582011-11-02 16:08:31 -0700605 usb_deregister(&diag_bridge_driver);
Jack Phame32cf322011-09-26 10:20:17 -0700606}
607
Jack Phamf6ed5582011-11-02 16:08:31 -0700608module_init(diag_bridge_init);
609module_exit(diag_bridge_exit);
Jack Phame32cf322011-09-26 10:20:17 -0700610
611MODULE_DESCRIPTION(DRIVER_DESC);
Jack Phamf6ed5582011-11-02 16:08:31 -0700612MODULE_VERSION(DRIVER_VERSION);
613MODULE_LICENSE("GPL v2");