blob: da3fbba72204564e7ca6a82fc7e560a36034cb41 [file] [log] [blame]
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001/*
2 * Copyright (C) 2006 by Bryan O'Donoghue, CodeHermit
Wolfgang Denk386eda02006-06-14 18:14:56 +02003 * bodonoghue@CodeHermit.ie
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02004 *
5 * References
Jean-Christophe PLAGNIOL-VILLARD2731b9a2009-04-03 12:46:58 +02006 * DasUBoot/drivers/usb/gadget/omap1510_udc.c, for design and implementation
Marcel Ziswiler7817cb22007-12-30 03:30:46 +01007 * ideas.
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02008 *
9 * This program is free software; you can redistribute it and/or modify
10 * it under the terms of the GNU General Public License as published by
11 * the Free Software Foundation; either version 2 of the License, or
12 * (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Wolfgang Denk386eda02006-06-14 18:14:56 +020016 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020017 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the
21 * Free Software Foundation, Inc.,
22 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23 *
24 */
25
26/*
27 * Notes :
Wolfgang Denk386eda02006-06-14 18:14:56 +020028 * 1. #define __SIMULATE_ERROR__ to inject a CRC error into every 2nd TX
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020029 * packet to force the USB re-transmit protocol.
30 *
31 * 2. #define __DEBUG_UDC__ to switch on debug tracing to serial console
Wolfgang Denk386eda02006-06-14 18:14:56 +020032 * be careful that tracing doesn't create Hiesen-bugs with respect to
33 * response timeouts to control requests.
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020034 *
35 * 3. This driver should be able to support any higher level driver that
36 * that wants to do either of the two standard UDC implementations
37 * Control-Bulk-Interrupt or Bulk-IN/Bulk-Out standards. Hence
38 * gserial and cdc_acm should work with this code.
39 *
40 * 4. NAK events never actually get raised at all, the documentation
41 * is just wrong !
42 *
43 * 5. For some reason, cbd_datlen is *always* +2 the value it should be.
44 * this means that having an RX cbd of 16 bytes is not possible, since
Wolfgang Denk386eda02006-06-14 18:14:56 +020045 * the same size is reported for 14 bytes received as 16 bytes received
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020046 * until we can find out why this happens, RX cbds must be limited to 8
47 * bytes. TODO: check errata for this behaviour.
48 *
49 * 6. Right now this code doesn't support properly powering up with the USB
Wolfgang Denk386eda02006-06-14 18:14:56 +020050 * cable attached to the USB host my development board the Adder87x doesn't
51 * have a pull-up fitted to allow this, so it is necessary to power the
52 * board and *then* attached the USB cable to the host. However somebody
53 * with a different design in their board may be able to keep the cable
54 * constantly connected and simply enable/disable a pull-up re
55 * figure 31.1 in MPC885RM.pdf instead of having to power up the board and
56 * then attach the cable !
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020057 *
58 */
59#include <common.h>
60#include <config.h>
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020061#include <commproc.h>
Jean-Christophe PLAGNIOL-VILLARD2731b9a2009-04-03 12:46:58 +020062#include <usbdevice.h>
63#include <usb/mpc8xx_udc.h>
64
65#include "ep0.h"
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020066
Wolfgang Denk1218abf2007-09-15 20:48:41 +020067DECLARE_GLOBAL_DATA_PTR;
68
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020069#define ERR(fmt, args...)\
70 serial_printf("ERROR : [%s] %s:%d: "fmt,\
71 __FILE__,__FUNCTION__,__LINE__, ##args)
72#ifdef __DEBUG_UDC__
Wolfgang Denk386eda02006-06-14 18:14:56 +020073#define DBG(fmt,args...)\
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020074 serial_printf("[%s] %s:%d: "fmt,\
75 __FILE__,__FUNCTION__,__LINE__, ##args)
76#else
Wolfgang Denk386eda02006-06-14 18:14:56 +020077#define DBG(fmt,args...)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020078#endif
79
80/* Static Data */
81#ifdef __SIMULATE_ERROR__
Wolfgang Denk386eda02006-06-14 18:14:56 +020082static char err_poison_test = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020083#endif
84static struct mpc8xx_ep ep_ref[MAX_ENDPOINTS];
85static u32 address_base = STATE_NOT_READY;
86static mpc8xx_udc_state_t udc_state = 0;
87static struct usb_device_instance *udc_device = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +020088static volatile usb_epb_t *endpoints[MAX_ENDPOINTS];
89static volatile cbd_t *tx_cbd[TX_RING_SIZE];
90static volatile cbd_t *rx_cbd[RX_RING_SIZE];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +020091static volatile immap_t *immr = 0;
92static volatile cpm8xx_t *cp = 0;
93static volatile usb_pram_t *usb_paramp = 0;
94static volatile usb_t *usbp = 0;
95static int rx_ct = 0;
96static int tx_ct = 0;
97
98/* Static Function Declarations */
99static void mpc8xx_udc_state_transition_up (usb_device_state_t initial,
Wolfgang Denk386eda02006-06-14 18:14:56 +0200100 usb_device_state_t final);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200101static void mpc8xx_udc_state_transition_down (usb_device_state_t initial,
Wolfgang Denk386eda02006-06-14 18:14:56 +0200102 usb_device_state_t final);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200103static void mpc8xx_udc_stall (unsigned int ep);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200104static void mpc8xx_udc_flush_tx_fifo (int epid);
105static void mpc8xx_udc_flush_rx_fifo (void);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200106static void mpc8xx_udc_clear_rxbd (volatile cbd_t * rx_cbdp);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200107static void mpc8xx_udc_init_tx (struct usb_endpoint_instance *epi,
108 struct urb *tx_urb);
109static void mpc8xx_udc_dump_request (struct usb_device_request *request);
110static void mpc8xx_udc_clock_init (volatile immap_t * immr,
111 volatile cpm8xx_t * cp);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200112static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi);
113static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200114static void mpc8xx_udc_ep0_rx (volatile cbd_t * rx_cbdp);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200115static void mpc8xx_udc_cbd_init (void);
116static void mpc8xx_udc_endpoint_init (void);
117static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size);
118static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment);
119static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp);
120static void mpc8xx_udc_set_nak (unsigned int ep);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200121static short mpc8xx_udc_handle_txerr (void);
122static void mpc8xx_udc_advance_rx (volatile cbd_t ** rx_cbdp, int epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200123
124/******************************************************************************
Wolfgang Denk386eda02006-06-14 18:14:56 +0200125 Global Linkage
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200126 *****************************************************************************/
127
128/* udc_init
129 *
130 * Do initial bus gluing
131 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200132int udc_init (void)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200133{
134 /* Init various pointers */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200135 immr = (immap_t *) CONFIG_SYS_IMMR;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200136 cp = (cpm8xx_t *) & (immr->im_cpm);
137 usb_paramp = (usb_pram_t *) & (cp->cp_dparam[PROFF_USB]);
138 usbp = (usb_t *) & (cp->cp_scc[0]);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200139
Wolfgang Denk386eda02006-06-14 18:14:56 +0200140 memset (ep_ref, 0x00, (sizeof (struct mpc8xx_ep) * MAX_ENDPOINTS));
141
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200142 udc_device = 0;
143 udc_state = STATE_NOT_READY;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200144
145 usbp->usmod = 0x00;
146 usbp->uscom = 0;
147
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200148 /* Set USB Frame #0, Respond at Address & Get a clock source */
149 usbp->usaddr = 0x00;
150 mpc8xx_udc_clock_init (immr, cp);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200151
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200152 /* PA15, PA14 as perhiperal USBRXD and USBOE */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200153 immr->im_ioport.iop_padir &= ~0x0003;
154 immr->im_ioport.iop_papar |= 0x0003;
155
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200156 /* PC11/PC10 as peripheral USBRXP USBRXN */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200157 immr->im_ioport.iop_pcso |= 0x0030;
158
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200159 /* PC7/PC6 as perhiperal USBTXP and USBTXN */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200160 immr->im_ioport.iop_pcdir |= 0x0300;
161 immr->im_ioport.iop_pcpar |= 0x0300;
162
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200163 /* Set the base address */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200164 address_base = (u32) (cp->cp_dpmem + CPM_USB_BASE);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200165
166 /* Initialise endpoints and circular buffers */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200167 mpc8xx_udc_endpoint_init ();
168 mpc8xx_udc_cbd_init ();
169
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200170 /* Assign allocated Dual Port Endpoint descriptors */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200171 usb_paramp->ep0ptr = (u32) endpoints[0];
172 usb_paramp->ep1ptr = (u32) endpoints[1];
173 usb_paramp->ep2ptr = (u32) endpoints[2];
174 usb_paramp->ep3ptr = (u32) endpoints[3];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200175 usb_paramp->frame_n = 0;
176
Wolfgang Denk386eda02006-06-14 18:14:56 +0200177 DBG ("ep0ptr=0x%08x ep1ptr=0x%08x ep2ptr=0x%08x ep3ptr=0x%08x\n",
178 usb_paramp->ep0ptr, usb_paramp->ep1ptr, usb_paramp->ep2ptr,
179 usb_paramp->ep3ptr);
180
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200181 return 0;
182}
183
184/* udc_irq
185 *
186 * Poll for whatever events may have occured
187 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200188void udc_irq (void)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200189{
190 int epid = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200191 volatile cbd_t *rx_cbdp = 0;
192 volatile cbd_t *rx_cbdp_base = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200193
Wolfgang Denk386eda02006-06-14 18:14:56 +0200194 if (udc_state != STATE_READY) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200195 return;
196 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200197
198 if (usbp->usber & USB_E_BSY) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200199 /* This shouldn't happen. If it does then it's a bug ! */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200200 usbp->usber |= USB_E_BSY;
201 mpc8xx_udc_flush_rx_fifo ();
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200202 }
203
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200204 /* Scan all RX/Bidirectional Endpoints for RX data. */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200205 for (epid = 0; epid < MAX_ENDPOINTS; epid++) {
206 if (!ep_ref[epid].prx) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200207 continue;
208 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200209 rx_cbdp = rx_cbdp_base = ep_ref[epid].prx;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200210
211 do {
212 if (!(rx_cbdp->cbd_sc & RX_BD_E)) {
213
214 if (rx_cbdp->cbd_sc & 0x1F) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200215 /* Corrupt data discard it.
Wolfgang Denk386eda02006-06-14 18:14:56 +0200216 * Controller has NAK'd this packet.
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200217 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200218 mpc8xx_udc_clear_rxbd (rx_cbdp);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200219
Wolfgang Denk386eda02006-06-14 18:14:56 +0200220 } else {
221 if (!epid) {
222 mpc8xx_udc_ep0_rx (rx_cbdp);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200223
Wolfgang Denk386eda02006-06-14 18:14:56 +0200224 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200225 /* Process data */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200226 mpc8xx_udc_set_nak (epid);
227 mpc8xx_udc_epn_rx (epid, rx_cbdp);
228 mpc8xx_udc_clear_rxbd (rx_cbdp);
229 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200230 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200231
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200232 /* Advance RX CBD pointer */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200233 mpc8xx_udc_advance_rx (&rx_cbdp, epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200234 ep_ref[epid].prx = rx_cbdp;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200235 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200236 /* Advance RX CBD pointer */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200237 mpc8xx_udc_advance_rx (&rx_cbdp, epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200238 }
239
Wolfgang Denk386eda02006-06-14 18:14:56 +0200240 } while (rx_cbdp != rx_cbdp_base);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200241 }
242
243 /* Handle TX events as appropiate, the correct place to do this is
244 * in a tx routine. Perhaps TX on epn was pre-empted by ep0
245 */
246
Wolfgang Denk386eda02006-06-14 18:14:56 +0200247 if (usbp->usber & USB_E_TXB) {
248 usbp->usber |= USB_E_TXB;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200249 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200250
251 if (usbp->usber & (USB_TX_ERRMASK)) {
252 mpc8xx_udc_handle_txerr ();
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200253 }
254
255 /* Switch to the default state, respond at the default address */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200256 if (usbp->usber & USB_E_RESET) {
257 usbp->usber |= USB_E_RESET;
258 usbp->usaddr = 0x00;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200259 udc_device->device_state = STATE_DEFAULT;
260 }
261
Wolfgang Denk386eda02006-06-14 18:14:56 +0200262 /* if(usbp->usber&USB_E_IDLE){
263 We could suspend here !
264 usbp->usber|=USB_E_IDLE;
265 DBG("idle state change\n");
266 }
267 if(usbp->usbs){
268 We could resume here when IDLE is deasserted !
269 Not worth doing, so long as we are self powered though.
270 }
271 */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200272
273 return;
274}
275
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200276/* udc_endpoint_write
277 *
278 * Write some data to an endpoint
279 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200280int udc_endpoint_write (struct usb_endpoint_instance *epi)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200281{
282 int ep = 0;
283 short epid = 1, unnak = 0, ret = 0;
284
Wolfgang Denk386eda02006-06-14 18:14:56 +0200285 if (udc_state != STATE_READY) {
286 ERR ("invalid udc_state != STATE_READY!\n");
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200287 return -1;
288 }
289
Wolfgang Denk386eda02006-06-14 18:14:56 +0200290 if (!udc_device || !epi) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200291 return -1;
292 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200293
294 if (udc_device->device_state != STATE_CONFIGURED) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200295 return -1;
296 }
297
298 ep = epi->endpoint_address & 0x03;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200299 if (ep >= MAX_ENDPOINTS) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200300 return -1;
301 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200302
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200303 /* Set NAK for all RX endpoints during TX */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200304 for (epid = 1; epid < MAX_ENDPOINTS; epid++) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200305
306 /* Don't set NAK on DATA IN/CONTROL endpoints */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200307 if (ep_ref[epid].sc & USB_DIR_IN) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200308 continue;
309 }
310
Wolfgang Denk386eda02006-06-14 18:14:56 +0200311 if (!(usbp->usep[epid] & (USEP_THS_NAK | USEP_RHS_NAK))) {
312 unnak |= 1 << epid;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200313 }
314
Wolfgang Denk386eda02006-06-14 18:14:56 +0200315 mpc8xx_udc_set_nak (epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200316 }
317
Wolfgang Denk386eda02006-06-14 18:14:56 +0200318 mpc8xx_udc_init_tx (&udc_device->bus->endpoint_array[ep],
319 epi->tx_urb);
320 ret = mpc8xx_udc_ep_tx (&udc_device->bus->endpoint_array[ep]);
321
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200322 /* Remove temporary NAK */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200323 for (epid = 1; epid < MAX_ENDPOINTS; epid++) {
324 if (unnak & (1 << epid)) {
325 udc_unset_nak (epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200326 }
327 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200328
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200329 return ret;
330}
331
332/* mpc8xx_udc_assign_urb
333 *
334 * Associate a given urb to an endpoint TX or RX transmit/receive buffers
335 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200336static int mpc8xx_udc_assign_urb (int ep, char direction)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200337{
338 struct usb_endpoint_instance *epi = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200339
340 if (ep >= MAX_ENDPOINTS) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200341 goto err;
342 }
343 epi = &udc_device->bus->endpoint_array[ep];
Wolfgang Denk386eda02006-06-14 18:14:56 +0200344 if (!epi) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200345 goto err;
346 }
347
Wolfgang Denk386eda02006-06-14 18:14:56 +0200348 if (!ep_ref[ep].urb) {
349 ep_ref[ep].urb = usbd_alloc_urb (udc_device, udc_device->bus->endpoint_array);
350 if (!ep_ref[ep].urb) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200351 goto err;
352 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200353 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200354 ep_ref[ep].urb->actual_length = 0;
355 }
356
Wolfgang Denk386eda02006-06-14 18:14:56 +0200357 switch (direction) {
358 case USB_DIR_IN:
359 epi->tx_urb = ep_ref[ep].urb;
360 break;
361 case USB_DIR_OUT:
362 epi->rcv_urb = ep_ref[ep].urb;
363 break;
364 default:
365 goto err;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200366 }
367 return 0;
368
Wolfgang Denk386eda02006-06-14 18:14:56 +0200369 err:
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200370 udc_state = STATE_ERROR;
371 return -1;
372}
373
374/* udc_setup_ep
375 *
376 * Associate U-Boot software endpoints to mpc8xx endpoint parameter ram
377 * Isochronous endpoints aren't yet supported!
378 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200379void udc_setup_ep (struct usb_device_instance *device, unsigned int ep,
380 struct usb_endpoint_instance *epi)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200381{
382 uchar direction = 0;
383 int ep_attrib = 0;
384
Wolfgang Denk386eda02006-06-14 18:14:56 +0200385 if (epi && (ep < MAX_ENDPOINTS)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200386
Wolfgang Denk386eda02006-06-14 18:14:56 +0200387 if (ep == 0) {
388 if (epi->rcv_attributes != USB_ENDPOINT_XFER_CONTROL
389 || epi->tx_attributes !=
390 USB_ENDPOINT_XFER_CONTROL) {
391
392 /* ep0 must be a control endpoint */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200393 udc_state = STATE_ERROR;
394 return;
395
396 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200397 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
398 mpc8xx_udc_cbd_attach (ep, epi->tx_packetSize,
399 epi->rcv_packetSize);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200400 }
401 usbp->usep[ep] = 0x0000;
402 return;
403 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200404
405 if ((epi->endpoint_address & USB_ENDPOINT_DIR_MASK)
406 == USB_DIR_IN) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200407
408 direction = 1;
409 ep_attrib = epi->tx_attributes;
410 epi->rcv_packetSize = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200411 ep_ref[ep].sc |= USB_DIR_IN;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200412 } else {
Wolfgang Denk386eda02006-06-14 18:14:56 +0200413
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200414 direction = 0;
415 ep_attrib = epi->rcv_attributes;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200416 epi->tx_packetSize = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200417 ep_ref[ep].sc &= ~USB_DIR_IN;
418 }
419
Wolfgang Denk386eda02006-06-14 18:14:56 +0200420 if (mpc8xx_udc_assign_urb (ep, epi->endpoint_address
421 & USB_ENDPOINT_DIR_MASK)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200422 return;
423 }
424
Wolfgang Denk386eda02006-06-14 18:14:56 +0200425 switch (ep_attrib) {
426 case USB_ENDPOINT_XFER_CONTROL:
427 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
428 mpc8xx_udc_cbd_attach (ep,
429 epi->tx_packetSize,
430 epi->rcv_packetSize);
431 }
432 usbp->usep[ep] = ep << 12;
433 epi->rcv_urb = epi->tx_urb = ep_ref[ep].urb;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200434
Wolfgang Denk386eda02006-06-14 18:14:56 +0200435 break;
436 case USB_ENDPOINT_XFER_BULK:
437 case USB_ENDPOINT_XFER_INT:
438 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
439 if (direction) {
440 mpc8xx_udc_cbd_attach (ep,
441 epi->tx_packetSize,
442 0);
443 } else {
444 mpc8xx_udc_cbd_attach (ep,
445 0,
446 epi->rcv_packetSize);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200447 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200448 }
449 usbp->usep[ep] = (ep << 12) | ((ep_attrib) << 8);
450
451 break;
452 case USB_ENDPOINT_XFER_ISOC:
453 default:
454 serial_printf ("Error endpoint attrib %d>3\n", ep_attrib);
455 udc_state = STATE_ERROR;
456 break;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200457 }
458 }
459
460}
461
462/* udc_connect
463 *
464 * Move state, switch on the USB
465 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200466void udc_connect (void)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200467{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200468 /* Enable pull-up resistor on D+
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200469 * TODO: fit a pull-up resistor to drive SE0 for > 2.5us
470 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200471
472 if (udc_state != STATE_ERROR) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200473 udc_state = STATE_READY;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200474 usbp->usmod |= USMOD_EN;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200475 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200476}
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200477
478/* udc_disconnect
479 *
480 * Disconnect is not used but, is included for completeness
481 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200482void udc_disconnect (void)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200483{
484 /* Disable pull-up resistor on D-
485 * TODO: fix a pullup resistor to control this
486 */
487
Wolfgang Denk386eda02006-06-14 18:14:56 +0200488 if (udc_state != STATE_ERROR) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200489 udc_state = STATE_NOT_READY;
490 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200491 usbp->usmod &= ~USMOD_EN;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200492}
493
494/* udc_enable
Wolfgang Denk386eda02006-06-14 18:14:56 +0200495 *
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200496 * Grab an EP0 URB, register interest in a subset of USB events
497 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200498void udc_enable (struct usb_device_instance *device)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200499{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200500 if (udc_state == STATE_ERROR) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200501 return;
502 }
503
504 udc_device = device;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200505
506 if (!ep_ref[0].urb) {
507 ep_ref[0].urb = usbd_alloc_urb (device, device->bus->endpoint_array);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200508 }
509
510 /* Register interest in all events except SOF, enable transceiver */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200511 usbp->usber = 0x03FF;
512 usbp->usbmr = 0x02F7;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200513
514 return;
515}
516
517/* udc_disable
518 *
519 * disable the currently hooked device
520 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200521void udc_disable (void)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200522{
523 int i = 0;
524
Wolfgang Denk386eda02006-06-14 18:14:56 +0200525 if (udc_state == STATE_ERROR) {
526 DBG ("Won't disable UDC. udc_state==STATE_ERROR !\n");
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200527 return;
528 }
529
530 udc_device = 0;
531
Wolfgang Denk386eda02006-06-14 18:14:56 +0200532 for (; i < MAX_ENDPOINTS; i++) {
533 if (ep_ref[i].urb) {
534 usbd_dealloc_urb (ep_ref[i].urb);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200535 ep_ref[i].urb = 0;
536 }
537 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200538
539 usbp->usbmr = 0x00;
540 usbp->usmod = ~USMOD_EN;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200541 udc_state = STATE_NOT_READY;
542}
543
544/* udc_startup_events
545 *
546 * Enable the specified device
547 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200548void udc_startup_events (struct usb_device_instance *device)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200549{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200550 udc_enable (device);
551 if (udc_state == STATE_READY) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200552 usbd_device_event_irq (device, DEVICE_CREATE, 0);
553 }
554}
555
556/* udc_set_nak
Wolfgang Denk386eda02006-06-14 18:14:56 +0200557 *
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200558 * Allow upper layers to signal lower layers should not accept more RX data
559 *
560 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200561void udc_set_nak (int epid)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200562{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200563 if (epid) {
564 mpc8xx_udc_set_nak (epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200565 }
566}
567
Wolfgang Denk386eda02006-06-14 18:14:56 +0200568/* udc_unset_nak
569 *
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200570 * Suspend sending of NAK tokens for DATA OUT tokens on a given endpoint.
571 * Switch off NAKing on this endpoint to accept more data output from host.
572 *
573 */
574void udc_unset_nak (int epid)
575{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200576 if (epid > MAX_ENDPOINTS) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200577 return;
578 }
579
Wolfgang Denk386eda02006-06-14 18:14:56 +0200580 if (usbp->usep[epid] & (USEP_THS_NAK | USEP_RHS_NAK)) {
581 usbp->usep[epid] &= ~(USEP_THS_NAK | USEP_RHS_NAK);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200582 __asm__ ("eieio");
583 }
584}
585
586/******************************************************************************
Wolfgang Denk386eda02006-06-14 18:14:56 +0200587 Static Linkage
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200588******************************************************************************/
589
590/* udc_state_transition_up
591 * udc_state_transition_down
592 *
593 * Helper functions to implement device state changes. The device states and
594 * the events that transition between them are:
595 *
596 * STATE_ATTACHED
597 * || /\
598 * \/ ||
599 * DEVICE_HUB_CONFIGURED DEVICE_HUB_RESET
600 * || /\
601 * \/ ||
602 * STATE_POWERED
603 * || /\
604 * \/ ||
605 * DEVICE_RESET DEVICE_POWER_INTERRUPTION
606 * || /\
607 * \/ ||
608 * STATE_DEFAULT
609 * || /\
610 * \/ ||
611 * DEVICE_ADDRESS_ASSIGNED DEVICE_RESET
612 * || /\
613 * \/ ||
614 * STATE_ADDRESSED
615 * || /\
616 * \/ ||
617 * DEVICE_CONFIGURED DEVICE_DE_CONFIGURED
618 * || /\
619 * \/ ||
620 * STATE_CONFIGURED
621 *
622 * udc_state_transition_up transitions up (in the direction from STATE_ATTACHED
623 * to STATE_CONFIGURED) from the specified initial state to the specified final
624 * state, passing through each intermediate state on the way. If the initial
625 * state is at or above (i.e. nearer to STATE_CONFIGURED) the final state, then
626 * no state transitions will take place.
627 *
628 * udc_state_transition_down transitions down (in the direction from
629 * STATE_CONFIGURED to STATE_ATTACHED) from the specified initial state to the
630 * specified final state, passing through each intermediate state on the way.
631 * If the initial state is at or below (i.e. nearer to STATE_ATTACHED) the final
632 * state, then no state transitions will take place.
633 *
634 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200635
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200636static void mpc8xx_udc_state_transition_up (usb_device_state_t initial,
Wolfgang Denk386eda02006-06-14 18:14:56 +0200637 usb_device_state_t final)
638{
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200639 if (initial < final) {
640 switch (initial) {
641 case STATE_ATTACHED:
642 usbd_device_event_irq (udc_device,
643 DEVICE_HUB_CONFIGURED, 0);
644 if (final == STATE_POWERED)
645 break;
646 case STATE_POWERED:
647 usbd_device_event_irq (udc_device, DEVICE_RESET, 0);
648 if (final == STATE_DEFAULT)
649 break;
650 case STATE_DEFAULT:
651 usbd_device_event_irq (udc_device,
652 DEVICE_ADDRESS_ASSIGNED, 0);
653 if (final == STATE_ADDRESSED)
654 break;
655 case STATE_ADDRESSED:
656 usbd_device_event_irq (udc_device, DEVICE_CONFIGURED,
657 0);
658 case STATE_CONFIGURED:
659 break;
660 default:
661 break;
662 }
663 }
664}
665
666static void mpc8xx_udc_state_transition_down (usb_device_state_t initial,
Wolfgang Denk386eda02006-06-14 18:14:56 +0200667 usb_device_state_t final)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200668{
669 if (initial > final) {
670 switch (initial) {
671 case STATE_CONFIGURED:
Wolfgang Denk386eda02006-06-14 18:14:56 +0200672 usbd_device_event_irq (udc_device,
673 DEVICE_DE_CONFIGURED, 0);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200674 if (final == STATE_ADDRESSED)
675 break;
676 case STATE_ADDRESSED:
677 usbd_device_event_irq (udc_device, DEVICE_RESET, 0);
678 if (final == STATE_DEFAULT)
679 break;
680 case STATE_DEFAULT:
Wolfgang Denk386eda02006-06-14 18:14:56 +0200681 usbd_device_event_irq (udc_device,
682 DEVICE_POWER_INTERRUPTION, 0);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200683 if (final == STATE_POWERED)
684 break;
685 case STATE_POWERED:
686 usbd_device_event_irq (udc_device, DEVICE_HUB_RESET,
Wolfgang Denk386eda02006-06-14 18:14:56 +0200687 0);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200688 case STATE_ATTACHED:
689 break;
690 default:
691 break;
692 }
693 }
694}
695
696/* mpc8xx_udc_stall
697 *
698 * Force returning of STALL tokens on the given endpoint. Protocol or function
699 * STALL conditions are permissable here
700 */
701static void mpc8xx_udc_stall (unsigned int ep)
702{
703 usbp->usep[ep] |= STALL_BITMASK;
704}
705
706/* mpc8xx_udc_set_nak
707 *
708 * Force returning of NAK responses for the given endpoint as a kind of very
709 * simple flow control
Wolfgang Denk386eda02006-06-14 18:14:56 +0200710 */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200711static void mpc8xx_udc_set_nak (unsigned int ep)
712{
713 usbp->usep[ep] |= NAK_BITMASK;
714 __asm__ ("eieio");
715}
716
717/* mpc8xx_udc_handle_txerr
718 *
719 * Handle errors relevant to TX. Return a status code to allow calling
720 * indicative of what if anything happened
721 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200722static short mpc8xx_udc_handle_txerr ()
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200723{
724 short ep = 0, ret = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200725
Wolfgang Denk386eda02006-06-14 18:14:56 +0200726 for (; ep < TX_RING_SIZE; ep++) {
727 if (usbp->usber & (0x10 << ep)) {
728
729 /* Timeout or underrun */
730 if (tx_cbd[ep]->cbd_sc & 0x06) {
731 ret = 1;
732 mpc8xx_udc_flush_tx_fifo (ep);
733
734 } else {
735 if (usbp->usep[ep] & STALL_BITMASK) {
736 if (!ep) {
737 usbp->usep[ep] &= ~STALL_BITMASK;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200738 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200739 } /* else NAK */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200740 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200741 usbp->usber |= (0x10 << ep);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200742 }
743 }
744 return ret;
745}
746
747/* mpc8xx_udc_advance_rx
748 *
749 * Advance cbd rx
750 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200751static void mpc8xx_udc_advance_rx (volatile cbd_t ** rx_cbdp, int epid)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200752{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200753 if ((*rx_cbdp)->cbd_sc & RX_BD_W) {
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200754 *rx_cbdp = (volatile cbd_t *) (endpoints[epid]->rbase + CONFIG_SYS_IMMR);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200755
756 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200757 (*rx_cbdp)++;
758 }
759}
760
761
762/* mpc8xx_udc_flush_tx_fifo
763 *
764 * Flush a given TX fifo. Assumes one tx cbd per endpoint
765 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200766static void mpc8xx_udc_flush_tx_fifo (int epid)
767{
768 volatile cbd_t *tx_cbdp = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200769
Wolfgang Denk386eda02006-06-14 18:14:56 +0200770 if (epid > MAX_ENDPOINTS) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200771 return;
772 }
773
774 /* TX stop */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200775 immr->im_cpm.cp_cpcr = ((epid << 2) | 0x1D01);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200776 __asm__ ("eieio");
Wolfgang Denk386eda02006-06-14 18:14:56 +0200777 while (immr->im_cpm.cp_cpcr & 0x01);
778
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200779 usbp->uscom = 0x40 | 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200780
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200781 /* reset ring */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200782 tx_cbdp = (cbd_t *) (endpoints[epid]->tbptr + CONFIG_SYS_IMMR);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200783 tx_cbdp->cbd_sc = (TX_BD_I | TX_BD_W);
784
Wolfgang Denk386eda02006-06-14 18:14:56 +0200785
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200786 endpoints[epid]->tptr = endpoints[epid]->tbase;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200787 endpoints[epid]->tstate = 0x00;
788 endpoints[epid]->tbcnt = 0x00;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200789
790 /* TX start */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200791 immr->im_cpm.cp_cpcr = ((epid << 2) | 0x2D01);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200792 __asm__ ("eieio");
Wolfgang Denk386eda02006-06-14 18:14:56 +0200793 while (immr->im_cpm.cp_cpcr & 0x01);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200794
795 return;
796}
797
798/* mpc8xx_udc_flush_rx_fifo
799 *
800 * For the sake of completeness of the namespace, it seems like
801 * a good-design-decision (tm) to include mpc8xx_udc_flush_rx_fifo();
802 * If RX_BD_E is true => a driver bug either here or in an upper layer
803 * not polling frequently enough. If RX_BD_E is true we have told the host
804 * we have accepted data but, the CPM found it had no-where to put that data
805 * which needless to say would be a bad thing.
806 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200807static void mpc8xx_udc_flush_rx_fifo ()
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200808{
809 int i = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200810
811 for (i = 0; i < RX_RING_SIZE; i++) {
812 if (!(rx_cbd[i]->cbd_sc & RX_BD_E)) {
813 ERR ("buf %p used rx data len = 0x%x sc=0x%x!\n",
814 rx_cbd[i], rx_cbd[i]->cbd_datlen,
815 rx_cbd[i]->cbd_sc);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200816
817 }
818 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200819 ERR ("BUG : Input over-run\n");
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200820}
821
822/* mpc8xx_udc_clear_rxbd
Wolfgang Denk386eda02006-06-14 18:14:56 +0200823 *
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200824 * Release control of RX CBD to CP.
825 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200826static void mpc8xx_udc_clear_rxbd (volatile cbd_t * rx_cbdp)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200827{
828 rx_cbdp->cbd_datlen = 0x0000;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200829 rx_cbdp->cbd_sc = ((rx_cbdp->cbd_sc & RX_BD_W) | (RX_BD_E | RX_BD_I));
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200830 __asm__ ("eieio");
831}
832
833/* mpc8xx_udc_tx_irq
834 *
835 * Parse for tx timeout, control RX or USB reset/busy conditions
836 * Return -1 on timeout, -2 on fatal error, else return zero
837 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200838static int mpc8xx_udc_tx_irq (int ep)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200839{
840 int i = 0;
841
Wolfgang Denk386eda02006-06-14 18:14:56 +0200842 if (usbp->usber & (USB_TX_ERRMASK)) {
843 if (mpc8xx_udc_handle_txerr ()) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200844 /* Timeout, controlling function must retry send */
845 return -1;
846 }
847 }
848
Wolfgang Denk386eda02006-06-14 18:14:56 +0200849 if (usbp->usber & (USB_E_RESET | USB_E_BSY)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200850 /* Fatal, abandon TX transaction */
851 return -2;
852 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200853
854 if (usbp->usber & USB_E_RXB) {
855 for (i = 0; i < RX_RING_SIZE; i++) {
856 if (!(rx_cbd[i]->cbd_sc & RX_BD_E)) {
857 if ((rx_cbd[i] == ep_ref[0].prx) || ep) {
858 return -2;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200859 }
860 }
861 }
862 }
863
864 return 0;
865}
866
867/* mpc8xx_udc_ep_tx
868 *
869 * Transmit in a re-entrant fashion outbound USB packets.
870 * Implement retry/timeout mechanism described in USB specification
871 * Toggle DATA0/DATA1 pids as necessary
872 * Introduces non-standard tx_retry. The USB standard has no scope for slave
873 * devices to give up TX, however tx_retry stops us getting stuck in an endless
874 * TX loop.
875 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200876static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200877{
878 struct urb *urb = epi->tx_urb;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200879 volatile cbd_t *tx_cbdp = 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200880 unsigned int ep = 0, pkt_len = 0, x = 0, tx_retry = 0;
881 int ret = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200882
883 if (!epi || (epi->endpoint_address & 0x03) >= MAX_ENDPOINTS || !urb) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200884 return -1;
885 }
886
887 ep = epi->endpoint_address & 0x03;
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200888 tx_cbdp = (cbd_t *) (endpoints[ep]->tbptr + CONFIG_SYS_IMMR);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200889
890 if (tx_cbdp->cbd_sc & TX_BD_R || usbp->usber & USB_E_TXB) {
891 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200892 usbp->usber |= USB_E_TXB;
893 };
894
Wolfgang Denk386eda02006-06-14 18:14:56 +0200895 while (tx_retry++ < 100) {
896 ret = mpc8xx_udc_tx_irq (ep);
897 if (ret == -1) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200898 /* ignore timeout here */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200899 } else if (ret == -2) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200900 /* Abandon TX */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200901 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200902 return -1;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200903 }
904
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +0200905 tx_cbdp = (cbd_t *) (endpoints[ep]->tbptr + CONFIG_SYS_IMMR);
Wolfgang Denk386eda02006-06-14 18:14:56 +0200906 while (tx_cbdp->cbd_sc & TX_BD_R) {
907 };
908 tx_cbdp->cbd_sc = (tx_cbdp->cbd_sc & TX_BD_W);
909
910 pkt_len = urb->actual_length - epi->sent;
911
912 if (pkt_len > epi->tx_packetSize || pkt_len > EP_MAX_PKT) {
913 pkt_len = MIN (epi->tx_packetSize, EP_MAX_PKT);
914 }
915
916 for (x = 0; x < pkt_len; x++) {
917 *((unsigned char *) (tx_cbdp->cbd_bufaddr + x)) =
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200918 urb->buffer[epi->sent + x];
919 }
920 tx_cbdp->cbd_datlen = pkt_len;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200921 tx_cbdp->cbd_sc |= (CBD_TX_BITMASK | ep_ref[ep].pid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200922 __asm__ ("eieio");
923
Wolfgang Denk386eda02006-06-14 18:14:56 +0200924#ifdef __SIMULATE_ERROR__
925 if (++err_poison_test == 2) {
926 err_poison_test = 0;
927 tx_cbdp->cbd_sc &= ~TX_BD_TC;
928 }
929#endif
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200930
Wolfgang Denk386eda02006-06-14 18:14:56 +0200931 usbp->uscom = (USCOM_STR | ep);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200932
Wolfgang Denk386eda02006-06-14 18:14:56 +0200933 while (!(usbp->usber & USB_E_TXB)) {
934 ret = mpc8xx_udc_tx_irq (ep);
935 if (ret == -1) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200936 /* TX timeout */
937 break;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200938 } else if (ret == -2) {
939 if (usbp->usber & USB_E_TXB) {
940 usbp->usber |= USB_E_TXB;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200941 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200942 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200943 return -1;
944 }
945 };
946
Wolfgang Denk386eda02006-06-14 18:14:56 +0200947 if (usbp->usber & USB_E_TXB) {
948 usbp->usber |= USB_E_TXB;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200949 }
950
951 /* ACK must be present <= 18bit times from TX */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200952 if (ret == -1) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200953 continue;
954 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200955
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200956 /* TX ACK : USB 2.0 8.7.2, Toggle PID, Advance TX */
957 epi->sent += pkt_len;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200958 epi->last = MIN (urb->actual_length - epi->sent, epi->tx_packetSize);
959 TOGGLE_TX_PID (ep_ref[ep].pid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200960
Wolfgang Denk386eda02006-06-14 18:14:56 +0200961 if (epi->sent >= epi->tx_urb->actual_length) {
962
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200963 epi->tx_urb->actual_length = 0;
964 epi->sent = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200965
966 if (ep_ref[ep].sc & EP_SEND_ZLP) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200967 ep_ref[ep].sc &= ~EP_SEND_ZLP;
Wolfgang Denk386eda02006-06-14 18:14:56 +0200968 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200969 return 0;
970 }
971 }
972 }
Wolfgang Denk386eda02006-06-14 18:14:56 +0200973
974 ERR ("TX fail, endpoint 0x%x tx bytes 0x%x/0x%x\n", ep, epi->sent,
975 epi->tx_urb->actual_length);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200976
977 return -1;
978}
979
980/* mpc8xx_udc_dump_request
981 *
982 * Dump a control request to console
983 */
Wolfgang Denk386eda02006-06-14 18:14:56 +0200984static void mpc8xx_udc_dump_request (struct usb_device_request *request)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200985{
Wolfgang Denk386eda02006-06-14 18:14:56 +0200986 DBG ("bmRequestType:%02x bRequest:%02x wValue:%04x "
987 "wIndex:%04x wLength:%04x ?\n",
988 request->bmRequestType,
989 request->bRequest,
990 request->wValue, request->wIndex, request->wLength);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200991
992 return;
993}
994
Wolfgang Denk386eda02006-06-14 18:14:56 +0200995/* mpc8xx_udc_ep0_rx_setup
996 *
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +0200997 * Decode received ep0 SETUP packet. return non-zero on error
998 */
999static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp)
1000{
1001 unsigned int x = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001002 struct urb *purb = ep_ref[0].urb;
1003 struct usb_endpoint_instance *epi =
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001004 &udc_device->bus->endpoint_array[0];
1005
Wolfgang Denk386eda02006-06-14 18:14:56 +02001006 for (; x < rx_cbdp->cbd_datlen; x++) {
1007 *(((unsigned char *) &ep_ref[0].urb->device_request) + x) =
1008 *((unsigned char *) (rx_cbdp->cbd_bufaddr + x));
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001009 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001010
Wolfgang Denk386eda02006-06-14 18:14:56 +02001011 mpc8xx_udc_clear_rxbd (rx_cbdp);
1012
1013 if (ep0_recv_setup (purb)) {
1014 mpc8xx_udc_dump_request (&purb->device_request);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001015 return -1;
1016 }
1017
Wolfgang Denk386eda02006-06-14 18:14:56 +02001018 if ((purb->device_request.bmRequestType & USB_REQ_DIRECTION_MASK)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001019 == USB_REQ_HOST2DEVICE) {
1020
Wolfgang Denk386eda02006-06-14 18:14:56 +02001021 switch (purb->device_request.bRequest) {
1022 case USB_REQ_SET_ADDRESS:
1023 /* Send the Status OUT ZLP */
1024 ep_ref[0].pid = TX_BD_PID_DATA1;
1025 purb->actual_length = 0;
1026 mpc8xx_udc_init_tx (epi, purb);
1027 mpc8xx_udc_ep_tx (epi);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001028
Wolfgang Denk386eda02006-06-14 18:14:56 +02001029 /* Move to the addressed state */
1030 usbp->usaddr = udc_device->address;
1031 mpc8xx_udc_state_transition_up (udc_device->device_state,
1032 STATE_ADDRESSED);
1033 return 0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001034
Wolfgang Denk386eda02006-06-14 18:14:56 +02001035 case USB_REQ_SET_CONFIGURATION:
1036 if (!purb->device_request.wValue) {
1037 /* Respond at default address */
1038 usbp->usaddr = 0x00;
1039 mpc8xx_udc_state_transition_down (udc_device->device_state,
1040 STATE_ADDRESSED);
1041 } else {
1042 /* TODO: Support multiple configurations */
1043 mpc8xx_udc_state_transition_up (udc_device->device_state,
1044 STATE_CONFIGURED);
1045 for (x = 1; x < MAX_ENDPOINTS; x++) {
1046 if ((udc_device->bus->endpoint_array[x].endpoint_address & USB_ENDPOINT_DIR_MASK)
1047 == USB_DIR_IN) {
1048 ep_ref[x].pid = TX_BD_PID_DATA0;
1049 } else {
1050 ep_ref[x].pid = RX_BD_PID_DATA0;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001051 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001052 /* Set configuration must unstall endpoints */
1053 usbp->usep[x] &= ~STALL_BITMASK;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001054 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001055 }
1056 break;
1057 default:
1058 /* CDC/Vendor specific */
1059 break;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001060 }
1061
1062 /* Send ZLP as ACK in Status OUT phase */
1063 ep_ref[0].pid = TX_BD_PID_DATA1;
1064 purb->actual_length = 0;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001065 mpc8xx_udc_init_tx (epi, purb);
1066 mpc8xx_udc_ep_tx (epi);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001067
Wolfgang Denk386eda02006-06-14 18:14:56 +02001068 } else {
1069
1070 if (purb->actual_length) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001071 ep_ref[0].pid = TX_BD_PID_DATA1;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001072 mpc8xx_udc_init_tx (epi, purb);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001073
Wolfgang Denk386eda02006-06-14 18:14:56 +02001074 if (!(purb->actual_length % EP0_MAX_PACKET_SIZE)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001075 ep_ref[0].sc |= EP_SEND_ZLP;
1076 }
1077
Wolfgang Denk386eda02006-06-14 18:14:56 +02001078 if (purb->device_request.wValue ==
1079 USB_DESCRIPTOR_TYPE_DEVICE) {
1080 if (le16_to_cpu (purb->device_request.wLength)
1081 > purb->actual_length) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001082 /* Send EP0_MAX_PACKET_SIZE bytes
1083 * unless correct size requested.
1084 */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001085 if (purb->actual_length > epi->tx_packetSize) {
1086 purb->actual_length = epi->tx_packetSize;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001087 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001088 }
1089 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001090 mpc8xx_udc_ep_tx (epi);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001091
Wolfgang Denk386eda02006-06-14 18:14:56 +02001092 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001093 /* Corrupt SETUP packet? */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001094 ERR ("Zero length data or SETUP with DATA-IN phase ?\n");
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001095 return 1;
1096 }
1097 }
1098 return 0;
1099}
1100
1101/* mpc8xx_udc_init_tx
1102 *
1103 * Setup some basic parameters for a TX transaction
1104 */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001105static void mpc8xx_udc_init_tx (struct usb_endpoint_instance *epi,
1106 struct urb *tx_urb)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001107{
1108 epi->sent = 0;
1109 epi->last = 0;
1110 epi->tx_urb = tx_urb;
1111}
1112
1113/* mpc8xx_udc_ep0_rx
1114 *
1115 * Receive ep0/control USB data. Parse and possibly send a response.
1116 */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001117static void mpc8xx_udc_ep0_rx (volatile cbd_t * rx_cbdp)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001118{
Wolfgang Denk386eda02006-06-14 18:14:56 +02001119 if (rx_cbdp->cbd_sc & RX_BD_PID_SETUP) {
1120
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001121 /* Unconditionally accept SETUP packets */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001122 if (mpc8xx_udc_ep0_rx_setup (rx_cbdp)) {
1123 mpc8xx_udc_stall (0);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001124 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001125
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001126 } else {
Wolfgang Denk386eda02006-06-14 18:14:56 +02001127
1128 mpc8xx_udc_clear_rxbd (rx_cbdp);
1129
1130 if ((rx_cbdp->cbd_datlen - 2)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001131 /* SETUP with a DATA phase
Wolfgang Denk386eda02006-06-14 18:14:56 +02001132 * outside of SETUP packet.
1133 * Reply with STALL.
1134 */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001135 mpc8xx_udc_stall (0);
1136 }
1137 }
1138}
1139
1140/* mpc8xx_udc_epn_rx
1141 *
1142 * Receive some data from cbd into USB system urb data abstraction
Wolfgang Denk386eda02006-06-14 18:14:56 +02001143 * Upper layers should NAK if there is insufficient RX data space
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001144 */
1145static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp)
1146{
1147 struct usb_endpoint_instance *epi = 0;
1148 struct urb *urb = 0;
1149 unsigned int x = 0;
1150
Wolfgang Denk386eda02006-06-14 18:14:56 +02001151 if (epid >= MAX_ENDPOINTS || !rx_cbdp->cbd_datlen) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001152 return 0;
1153 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001154
1155 /* USB 2.0 PDF section 8.6.4
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001156 * Discard data with invalid PID it is a resend.
1157 */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001158 if (ep_ref[epid].pid != (rx_cbdp->cbd_sc & 0xC0)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001159 return 1;
1160 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001161 TOGGLE_RX_PID (ep_ref[epid].pid);
1162
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001163 epi = &udc_device->bus->endpoint_array[epid];
1164 urb = epi->rcv_urb;
1165
Wolfgang Denk386eda02006-06-14 18:14:56 +02001166 for (; x < (rx_cbdp->cbd_datlen - 2); x++) {
1167 *((unsigned char *) (urb->buffer + urb->actual_length + x)) =
1168 *((unsigned char *) (rx_cbdp->cbd_bufaddr + x));
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001169 }
1170
Wolfgang Denk386eda02006-06-14 18:14:56 +02001171 if (x) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001172 usbd_rcv_complete (epi, x, 0);
Wolfgang Denk386eda02006-06-14 18:14:56 +02001173 if (ep_ref[epid].urb->status == RECV_ERROR) {
1174 DBG ("RX error unset NAK\n");
1175 udc_unset_nak (epid);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001176 }
Wolfgang Denk386eda02006-06-14 18:14:56 +02001177 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001178 return x;
1179}
1180
1181/* mpc8xx_udc_clock_init
1182 *
Wolfgang Denk386eda02006-06-14 18:14:56 +02001183 * Obtain a clock reference for Full Speed Signaling
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001184 */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001185static void mpc8xx_udc_clock_init (volatile immap_t * immr,
1186 volatile cpm8xx_t * cp)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001187{
1188
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001189#if defined(CONFIG_SYS_USB_EXTC_CLK)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001190
1191 /* This has been tested with a 48MHz crystal on CLK6 */
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001192 switch (CONFIG_SYS_USB_EXTC_CLK) {
Wolfgang Denk386eda02006-06-14 18:14:56 +02001193 case 1:
1194 immr->im_ioport.iop_papar |= 0x0100;
1195 immr->im_ioport.iop_padir &= ~0x0100;
1196 cp->cp_sicr |= 0x24;
1197 break;
1198 case 2:
1199 immr->im_ioport.iop_papar |= 0x0200;
1200 immr->im_ioport.iop_padir &= ~0x0200;
1201 cp->cp_sicr |= 0x2D;
1202 break;
1203 case 3:
1204 immr->im_ioport.iop_papar |= 0x0400;
1205 immr->im_ioport.iop_padir &= ~0x0400;
1206 cp->cp_sicr |= 0x36;
1207 break;
1208 case 4:
1209 immr->im_ioport.iop_papar |= 0x0800;
1210 immr->im_ioport.iop_padir &= ~0x0800;
1211 cp->cp_sicr |= 0x3F;
1212 break;
1213 default:
1214 udc_state = STATE_ERROR;
1215 break;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001216 }
1217
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001218#elif defined(CONFIG_SYS_USB_BRGCLK)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001219
Wolfgang Denk386eda02006-06-14 18:14:56 +02001220 /* This has been tested with brgclk == 50MHz */
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001221 int divisor = 0;
1222
Wolfgang Denk386eda02006-06-14 18:14:56 +02001223 if (gd->cpu_clk < 48000000L) {
1224 ERR ("brgclk is too slow for full-speed USB!\n");
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001225 udc_state = STATE_ERROR;
1226 return;
1227 }
1228
Wolfgang Denk8ed44d92008-10-19 02:35:50 +02001229 /* Assume the brgclk is 'good enough', we want !(gd->cpu_clk%48MHz)
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001230 * but, can /probably/ live with close-ish alternative rates.
Wolfgang Denk386eda02006-06-14 18:14:56 +02001231 */
1232 divisor = (gd->cpu_clk / 48000000L) - 1;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001233 cp->cp_sicr &= ~0x0000003F;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001234
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001235 switch (CONFIG_SYS_USB_BRGCLK) {
Wolfgang Denk386eda02006-06-14 18:14:56 +02001236 case 1:
1237 cp->cp_brgc1 |= (divisor | CPM_BRG_EN);
1238 cp->cp_sicr &= ~0x2F;
1239 break;
1240 case 2:
1241 cp->cp_brgc2 |= (divisor | CPM_BRG_EN);
1242 cp->cp_sicr |= 0x00000009;
1243 break;
1244 case 3:
1245 cp->cp_brgc3 |= (divisor | CPM_BRG_EN);
1246 cp->cp_sicr |= 0x00000012;
1247 break;
1248 case 4:
1249 cp->cp_brgc4 = (divisor | CPM_BRG_EN);
1250 cp->cp_sicr |= 0x0000001B;
1251 break;
1252 default:
1253 udc_state = STATE_ERROR;
1254 break;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001255 }
1256
1257#else
Jean-Christophe PLAGNIOL-VILLARD6d0f6bc2008-10-16 15:01:15 +02001258#error "CONFIG_SYS_USB_EXTC_CLK or CONFIG_SYS_USB_BRGCLK must be defined"
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001259#endif
1260
1261}
1262
1263/* mpc8xx_udc_cbd_attach
1264 *
1265 * attach a cbd to and endpoint
1266 */
1267static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size)
1268{
Wolfgang Denk386eda02006-06-14 18:14:56 +02001269
1270 if (!tx_cbd[ep] || !rx_cbd[ep] || ep >= MAX_ENDPOINTS) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001271 udc_state = STATE_ERROR;
1272 return;
1273 }
1274
Wolfgang Denk386eda02006-06-14 18:14:56 +02001275 if (tx_size > USB_MAX_PKT || rx_size > USB_MAX_PKT ||
1276 (!tx_size && !rx_size)) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001277 udc_state = STATE_ERROR;
1278 return;
1279 }
1280
1281 /* Attach CBD to appropiate Parameter RAM Endpoint data structure */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001282 if (rx_size) {
1283 endpoints[ep]->rbase = (u32) rx_cbd[rx_ct];
1284 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001285 rx_ct++;
1286
Wolfgang Denk386eda02006-06-14 18:14:56 +02001287 if (!ep) {
1288
1289 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001290 rx_cbd[rx_ct]->cbd_sc |= RX_BD_W;
1291 rx_ct++;
1292
Wolfgang Denk386eda02006-06-14 18:14:56 +02001293 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001294 rx_ct += 2;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001295 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001296 rx_cbd[rx_ct]->cbd_sc |= RX_BD_W;
1297 rx_ct++;
1298 }
1299
1300 /* Where we expect to RX data on this endpoint */
Wolfgang Denk386eda02006-06-14 18:14:56 +02001301 ep_ref[ep].prx = rx_cbd[rx_ct - 1];
1302 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001303
1304 ep_ref[ep].prx = 0;
1305 endpoints[ep]->rbase = 0;
1306 endpoints[ep]->rbptr = 0;
1307 }
1308
Wolfgang Denk386eda02006-06-14 18:14:56 +02001309 if (tx_size) {
1310 endpoints[ep]->tbase = (u32) tx_cbd[tx_ct];
1311 endpoints[ep]->tbptr = (u32) tx_cbd[tx_ct];
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001312 tx_ct++;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001313 } else {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001314 endpoints[ep]->tbase = 0;
1315 endpoints[ep]->tbptr = 0;
1316 }
1317
1318 endpoints[ep]->tstate = 0;
1319 endpoints[ep]->tbcnt = 0;
1320 endpoints[ep]->mrblr = EP_MAX_PKT;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001321 endpoints[ep]->rfcr = 0x18;
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001322 endpoints[ep]->tfcr = 0x18;
1323 ep_ref[ep].sc |= EP_ATTACHED;
1324
Wolfgang Denk386eda02006-06-14 18:14:56 +02001325 DBG ("ep %d rbase 0x%08x rbptr 0x%08x tbase 0x%08x tbptr 0x%08x prx = %p\n",
1326 ep, endpoints[ep]->rbase, endpoints[ep]->rbptr,
1327 endpoints[ep]->tbase, endpoints[ep]->tbptr,
1328 ep_ref[ep].prx);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001329
1330 return;
1331}
1332
1333/* mpc8xx_udc_cbd_init
1334 *
1335 * Allocate space for a cbd and allocate TX/RX data space
1336 */
1337static void mpc8xx_udc_cbd_init (void)
1338{
1339 int i = 0;
1340
Wolfgang Denk386eda02006-06-14 18:14:56 +02001341 for (; i < TX_RING_SIZE; i++) {
1342 tx_cbd[i] = (cbd_t *)
1343 mpc8xx_udc_alloc (sizeof (cbd_t), sizeof (int));
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001344 }
1345
Wolfgang Denk386eda02006-06-14 18:14:56 +02001346 for (i = 0; i < RX_RING_SIZE; i++) {
1347 rx_cbd[i] = (cbd_t *)
1348 mpc8xx_udc_alloc (sizeof (cbd_t), sizeof (int));
1349 }
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001350
Wolfgang Denk386eda02006-06-14 18:14:56 +02001351 for (i = 0; i < TX_RING_SIZE; i++) {
1352 tx_cbd[i]->cbd_bufaddr =
1353 mpc8xx_udc_alloc (EP_MAX_PKT, sizeof (int));
1354
1355 tx_cbd[i]->cbd_sc = (TX_BD_I | TX_BD_W);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001356 tx_cbd[i]->cbd_datlen = 0x0000;
1357 }
1358
1359
Wolfgang Denk386eda02006-06-14 18:14:56 +02001360 for (i = 0; i < RX_RING_SIZE; i++) {
1361 rx_cbd[i]->cbd_bufaddr =
1362 mpc8xx_udc_alloc (EP_MAX_PKT, sizeof (int));
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001363 rx_cbd[i]->cbd_sc = (RX_BD_I | RX_BD_E);
1364 rx_cbd[i]->cbd_datlen = 0x0000;
1365
1366 }
1367
1368 return;
1369}
1370
1371/* mpc8xx_udc_endpoint_init
1372 *
1373 * Attach an endpoint to some dpram
1374 */
1375static void mpc8xx_udc_endpoint_init (void)
1376{
1377 int i = 0;
1378
Wolfgang Denk386eda02006-06-14 18:14:56 +02001379 for (; i < MAX_ENDPOINTS; i++) {
1380 endpoints[i] = (usb_epb_t *)
1381 mpc8xx_udc_alloc (sizeof (usb_epb_t), 32);
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001382 }
1383}
1384
1385/* mpc8xx_udc_alloc
1386 *
Wolfgang Denk386eda02006-06-14 18:14:56 +02001387 * Grab the address of some dpram
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001388 */
1389static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment)
1390{
1391 u32 retaddr = address_base;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001392
1393 while (retaddr % alignment) {
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001394 retaddr++;
Wolfgang Denk386eda02006-06-14 18:14:56 +02001395 }
1396 address_base += data_size;
1397
Wolfgang Denk16c8d5e2006-06-14 17:45:53 +02001398 return retaddr;
1399}