]> Pileus Git - ~andy/linux/blob - drivers/staging/ozwpan/ozusbsvc.c
Merge remote-tracking branch 'spi/fix/core' into spi-linus
[~andy/linux] / drivers / staging / ozwpan / ozusbsvc.c
1 /* -----------------------------------------------------------------------------
2  * Copyright (c) 2011 Ozmo Inc
3  * Released under the GNU General Public License Version 2 (GPLv2).
4  *
5  * This file provides protocol independent part of the implementation of the USB
6  * service for a PD.
7  * The implementation of this service is split into two parts the first of which
8  * is protocol independent and the second contains protocol specific details.
9  * This split is to allow alternative protocols to be defined.
10  * The implementation of this service uses ozhcd.c to implement a USB HCD.
11  * -----------------------------------------------------------------------------
12  */
13
14 #include <linux/module.h>
15 #include <linux/timer.h>
16 #include <linux/sched.h>
17 #include <linux/netdevice.h>
18 #include <linux/errno.h>
19 #include <linux/input.h>
20 #include <asm/unaligned.h>
21 #include "ozdbg.h"
22 #include "ozprotocol.h"
23 #include "ozeltbuf.h"
24 #include "ozpd.h"
25 #include "ozproto.h"
26 #include "ozusbif.h"
27 #include "ozhcd.h"
28 #include "ozusbsvc.h"
29
30 /*
31  * This is called once when the driver is loaded to initialise the USB service.
32  * Context: process
33  */
34 int oz_usb_init(void)
35 {
36         return oz_hcd_init();
37 }
38
39 /*
40  * This is called once when the driver is unloaded to terminate the USB service.
41  * Context: process
42  */
43 void oz_usb_term(void)
44 {
45         oz_hcd_term();
46 }
47
48 /*
49  * This is called when the USB service is started or resumed for a PD.
50  * Context: softirq
51  */
52 int oz_usb_start(struct oz_pd *pd, int resume)
53 {
54         int rc = 0;
55         struct oz_usb_ctx *usb_ctx;
56         struct oz_usb_ctx *old_ctx;
57
58         if (resume) {
59                 oz_dbg(ON, "USB service resumed\n");
60                 return 0;
61         }
62         oz_dbg(ON, "USB service started\n");
63         /* Create a USB context in case we need one. If we find the PD already
64          * has a USB context then we will destroy it.
65          */
66         usb_ctx = kzalloc(sizeof(struct oz_usb_ctx), GFP_ATOMIC);
67         if (usb_ctx == NULL)
68                 return -ENOMEM;
69         atomic_set(&usb_ctx->ref_count, 1);
70         usb_ctx->pd = pd;
71         usb_ctx->stopped = 0;
72         /* Install the USB context if the PD doesn't already have one.
73          * If it does already have one then destroy the one we have just
74          * created.
75          */
76         spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
77         old_ctx = pd->app_ctx[OZ_APPID_USB-1];
78         if (old_ctx == NULL)
79                 pd->app_ctx[OZ_APPID_USB-1] = usb_ctx;
80         oz_usb_get(pd->app_ctx[OZ_APPID_USB-1]);
81         spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
82         if (old_ctx) {
83                 oz_dbg(ON, "Already have USB context\n");
84                 kfree(usb_ctx);
85                 usb_ctx = old_ctx;
86         } else if (usb_ctx) {
87                 /* Take a reference to the PD. This will be released when
88                  * the USB context is destroyed.
89                  */
90                 oz_pd_get(pd);
91         }
92         /* If we already had a USB context and had obtained a port from
93          * the USB HCD then just reset the port. If we didn't have a port
94          * then report the arrival to the USB HCD so we get one.
95          */
96         if (usb_ctx->hport) {
97                 oz_hcd_pd_reset(usb_ctx, usb_ctx->hport);
98         } else {
99                 usb_ctx->hport = oz_hcd_pd_arrived(usb_ctx);
100                 if (usb_ctx->hport == NULL) {
101                         oz_dbg(ON, "USB hub returned null port\n");
102                         spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
103                         pd->app_ctx[OZ_APPID_USB-1] = NULL;
104                         spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
105                         oz_usb_put(usb_ctx);
106                         rc = -1;
107                 }
108         }
109         oz_usb_put(usb_ctx);
110         return rc;
111 }
112
113 /*
114  * This is called when the USB service is stopped or paused for a PD.
115  * Context: softirq or process
116  */
117 void oz_usb_stop(struct oz_pd *pd, int pause)
118 {
119         struct oz_usb_ctx *usb_ctx;
120
121         if (pause) {
122                 oz_dbg(ON, "USB service paused\n");
123                 return;
124         }
125         spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
126         usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB-1];
127         pd->app_ctx[OZ_APPID_USB-1] = NULL;
128         spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
129         if (usb_ctx) {
130                 struct timespec ts, now;
131                 getnstimeofday(&ts);
132                 oz_dbg(ON, "USB service stopping...\n");
133                 usb_ctx->stopped = 1;
134                 /* At this point the reference count on the usb context should
135                  * be 2 - one from when we created it and one from the hcd
136                  * which claims a reference. Since stopped = 1 no one else
137                  * should get in but someone may already be in. So wait
138                  * until they leave but timeout after 1 second.
139                  */
140                 while ((atomic_read(&usb_ctx->ref_count) > 2)) {
141                         getnstimeofday(&now);
142                         /*Approx 1 Sec. this is not perfect calculation*/
143                         if (now.tv_sec != ts.tv_sec)
144                                 break;
145                 }
146                 oz_dbg(ON, "USB service stopped\n");
147                 oz_hcd_pd_departed(usb_ctx->hport);
148                 /* Release the reference taken in oz_usb_start.
149                  */
150                 oz_usb_put(usb_ctx);
151         }
152 }
153
154 /*
155  * This increments the reference count of the context area for a specific PD.
156  * This ensures this context area does not disappear while still in use.
157  * Context: softirq
158  */
159 void oz_usb_get(void *hpd)
160 {
161         struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
162
163         atomic_inc(&usb_ctx->ref_count);
164 }
165
166 /*
167  * This decrements the reference count of the context area for a specific PD
168  * and destroys the context area if the reference count becomes zero.
169  * Context: irq or process
170  */
171 void oz_usb_put(void *hpd)
172 {
173         struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
174
175         if (atomic_dec_and_test(&usb_ctx->ref_count)) {
176                 oz_dbg(ON, "Dealloc USB context\n");
177                 oz_pd_put(usb_ctx->pd);
178                 kfree(usb_ctx);
179         }
180 }
181
182 /*
183  * Context: softirq
184  */
185 int oz_usb_heartbeat(struct oz_pd *pd)
186 {
187         struct oz_usb_ctx *usb_ctx;
188         int rc = 0;
189
190         spin_lock_bh(&pd->app_lock[OZ_APPID_USB-1]);
191         usb_ctx = (struct oz_usb_ctx *)pd->app_ctx[OZ_APPID_USB-1];
192         if (usb_ctx)
193                 oz_usb_get(usb_ctx);
194         spin_unlock_bh(&pd->app_lock[OZ_APPID_USB-1]);
195         if (usb_ctx == NULL)
196                 return rc;
197         if (usb_ctx->stopped)
198                 goto done;
199         if (usb_ctx->hport)
200                 if (oz_hcd_heartbeat(usb_ctx->hport))
201                         rc = 1;
202 done:
203         oz_usb_put(usb_ctx);
204         return rc;
205 }
206
207 /*
208  * Context: softirq
209  */
210 int oz_usb_stream_create(void *hpd, u8 ep_num)
211 {
212         struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
213         struct oz_pd *pd = usb_ctx->pd;
214
215         oz_dbg(ON, "%s: (0x%x)\n", __func__, ep_num);
216         if (pd->mode & OZ_F_ISOC_NO_ELTS) {
217                 oz_isoc_stream_create(pd, ep_num);
218         } else {
219                 oz_pd_get(pd);
220                 if (oz_elt_stream_create(&pd->elt_buff, ep_num,
221                         4*pd->max_tx_size)) {
222                         oz_pd_put(pd);
223                         return -1;
224                 }
225         }
226         return 0;
227 }
228
229 /*
230  * Context: softirq
231  */
232 int oz_usb_stream_delete(void *hpd, u8 ep_num)
233 {
234         struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
235
236         if (usb_ctx) {
237                 struct oz_pd *pd = usb_ctx->pd;
238                 if (pd) {
239                         oz_dbg(ON, "%s: (0x%x)\n", __func__, ep_num);
240                         if (pd->mode & OZ_F_ISOC_NO_ELTS) {
241                                 oz_isoc_stream_delete(pd, ep_num);
242                         } else {
243                                 if (oz_elt_stream_delete(&pd->elt_buff, ep_num))
244                                         return -1;
245                                 oz_pd_put(pd);
246                         }
247                 }
248         }
249         return 0;
250 }
251
252 /*
253  * Context: softirq or process
254  */
255 void oz_usb_request_heartbeat(void *hpd)
256 {
257         struct oz_usb_ctx *usb_ctx = (struct oz_usb_ctx *)hpd;
258
259         if (usb_ctx && usb_ctx->pd)
260                 oz_pd_request_heartbeat(usb_ctx->pd);
261 }