1 /* -----------------------------------------------------------------------------
2 * Copyright (c) 2011 Ozmo Inc
3 * Released under the GNU General Public License Version 2 (GPLv2).
5 * This file provides protocol independent part of the implementation of the USB
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 * -----------------------------------------------------------------------------
14 #include <linux/init.h>
15 #include <linux/module.h>
16 #include <linux/timer.h>
17 #include <linux/sched.h>
18 #include <linux/netdevice.h>
19 #include <linux/errno.h>
20 #include <linux/input.h>
21 #include <asm/unaligned.h>
23 #include "ozprotocol.h"
32 * This is called once when the driver is loaded to initialise the USB service.
41 * This is called once when the driver is unloaded to terminate the USB service.
44 void oz_usb_term(void)
50 * This is called when the USB service is started or resumed for a PD.
53 int oz_usb_start(struct oz_pd
*pd
, int resume
)
56 struct oz_usb_ctx
*usb_ctx
;
57 struct oz_usb_ctx
*old_ctx
;
60 oz_dbg(ON
, "USB service resumed\n");
63 oz_dbg(ON
, "USB service started\n");
64 /* Create a USB context in case we need one. If we find the PD already
65 * has a USB context then we will destroy it.
67 usb_ctx
= kzalloc(sizeof(struct oz_usb_ctx
), GFP_ATOMIC
);
70 atomic_set(&usb_ctx
->ref_count
, 1);
73 /* Install the USB context if the PD doesn't already have one.
74 * If it does already have one then destroy the one we have just
77 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
78 old_ctx
= pd
->app_ctx
[OZ_APPID_USB
-1];
80 pd
->app_ctx
[OZ_APPID_USB
-1] = usb_ctx
;
81 oz_usb_get(pd
->app_ctx
[OZ_APPID_USB
-1]);
82 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
84 oz_dbg(ON
, "Already have USB context\n");
88 /* Take a reference to the PD. This will be released when
89 * the USB context is destroyed.
93 /* If we already had a USB context and had obtained a port from
94 * the USB HCD then just reset the port. If we didn't have a port
95 * then report the arrival to the USB HCD so we get one.
98 oz_hcd_pd_reset(usb_ctx
, usb_ctx
->hport
);
100 usb_ctx
->hport
= oz_hcd_pd_arrived(usb_ctx
);
101 if (usb_ctx
->hport
== NULL
) {
102 oz_dbg(ON
, "USB hub returned null port\n");
103 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
104 pd
->app_ctx
[OZ_APPID_USB
-1] = NULL
;
105 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
115 * This is called when the USB service is stopped or paused for a PD.
116 * Context: softirq or process
118 void oz_usb_stop(struct oz_pd
*pd
, int pause
)
120 struct oz_usb_ctx
*usb_ctx
;
123 oz_dbg(ON
, "USB service paused\n");
126 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
127 usb_ctx
= (struct oz_usb_ctx
*)pd
->app_ctx
[OZ_APPID_USB
-1];
128 pd
->app_ctx
[OZ_APPID_USB
-1] = NULL
;
129 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
131 struct timespec ts
, now
;
133 oz_dbg(ON
, "USB service stopping...\n");
134 usb_ctx
->stopped
= 1;
135 /* At this point the reference count on the usb context should
136 * be 2 - one from when we created it and one from the hcd
137 * which claims a reference. Since stopped = 1 no one else
138 * should get in but someone may already be in. So wait
139 * until they leave but timeout after 1 second.
141 while ((atomic_read(&usb_ctx
->ref_count
) > 2)) {
142 getnstimeofday(&now
);
143 /*Approx 1 Sec. this is not perfect calculation*/
144 if (now
.tv_sec
!= ts
.tv_sec
)
147 oz_dbg(ON
, "USB service stopped\n");
148 oz_hcd_pd_departed(usb_ctx
->hport
);
149 /* Release the reference taken in oz_usb_start.
156 * This increments the reference count of the context area for a specific PD.
157 * This ensures this context area does not disappear while still in use.
160 void oz_usb_get(void *hpd
)
162 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
164 atomic_inc(&usb_ctx
->ref_count
);
168 * This decrements the reference count of the context area for a specific PD
169 * and destroys the context area if the reference count becomes zero.
170 * Context: irq or process
172 void oz_usb_put(void *hpd
)
174 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
176 if (atomic_dec_and_test(&usb_ctx
->ref_count
)) {
177 oz_dbg(ON
, "Dealloc USB context\n");
178 oz_pd_put(usb_ctx
->pd
);
186 int oz_usb_heartbeat(struct oz_pd
*pd
)
188 struct oz_usb_ctx
*usb_ctx
;
191 spin_lock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
192 usb_ctx
= (struct oz_usb_ctx
*)pd
->app_ctx
[OZ_APPID_USB
-1];
195 spin_unlock_bh(&pd
->app_lock
[OZ_APPID_USB
-1]);
198 if (usb_ctx
->stopped
)
201 if (oz_hcd_heartbeat(usb_ctx
->hport
))
211 int oz_usb_stream_create(void *hpd
, u8 ep_num
)
213 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
214 struct oz_pd
*pd
= usb_ctx
->pd
;
216 oz_dbg(ON
, "%s: (0x%x)\n", __func__
, ep_num
);
217 if (pd
->mode
& OZ_F_ISOC_NO_ELTS
) {
218 oz_isoc_stream_create(pd
, ep_num
);
221 if (oz_elt_stream_create(&pd
->elt_buff
, ep_num
,
222 4*pd
->max_tx_size
)) {
233 int oz_usb_stream_delete(void *hpd
, u8 ep_num
)
235 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
238 struct oz_pd
*pd
= usb_ctx
->pd
;
240 oz_dbg(ON
, "%s: (0x%x)\n", __func__
, ep_num
);
241 if (pd
->mode
& OZ_F_ISOC_NO_ELTS
) {
242 oz_isoc_stream_delete(pd
, ep_num
);
244 if (oz_elt_stream_delete(&pd
->elt_buff
, ep_num
))
254 * Context: softirq or process
256 void oz_usb_request_heartbeat(void *hpd
)
258 struct oz_usb_ctx
*usb_ctx
= (struct oz_usb_ctx
*)hpd
;
260 if (usb_ctx
&& usb_ctx
->pd
)
261 oz_pd_request_heartbeat(usb_ctx
->pd
);