blob: 122793c023cfcddb6508f308f48c1bb7dfabbdd1 [file] [log] [blame]
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001/*
2 * Copyright (C) 2006 by Bryan O'Donoghue, CodeHermit
Wolfgang Denke2601822006-06-14 18:14:56 +02003 * bodonoghue@CodeHermit.ie
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02004 *
5 * References
Marcel Ziswileraea68562007-12-30 03:30:46 +01006 * DasUBoot/drivers/usb/usbdcore_omap1510.c, for design and implementation
7 * ideas.
Wolfgang Denk3f0137b2006-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 Denke2601822006-06-14 18:14:56 +020016 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
Wolfgang Denk3f0137b2006-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 Denke2601822006-06-14 18:14:56 +020028 * 1. #define __SIMULATE_ERROR__ to inject a CRC error into every 2nd TX
Wolfgang Denk3f0137b2006-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 Denke2601822006-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 Denk3f0137b2006-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 Denke2601822006-06-14 18:14:56 +020045 * the same size is reported for 14 bytes received as 16 bytes received
Wolfgang Denk3f0137b2006-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 Denke2601822006-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 Denk3f0137b2006-06-14 17:45:53 +020057 *
58 */
59#include <common.h>
60#include <config.h>
61
62#if defined(CONFIG_MPC885_FAMILY) && defined(CONFIG_USB_DEVICE)
63#include <commproc.h>
Wolfgang Denke2601822006-06-14 18:14:56 +020064#include "usbdcore.h"
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020065#include "usbdcore_mpc8xx.h"
66#include "usbdcore_ep0.h"
67
Wolfgang Denkd112a2c2007-09-15 20:48:41 +020068DECLARE_GLOBAL_DATA_PTR;
69
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020070#define ERR(fmt, args...)\
71 serial_printf("ERROR : [%s] %s:%d: "fmt,\
72 __FILE__,__FUNCTION__,__LINE__, ##args)
73#ifdef __DEBUG_UDC__
Wolfgang Denke2601822006-06-14 18:14:56 +020074#define DBG(fmt,args...)\
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020075 serial_printf("[%s] %s:%d: "fmt,\
76 __FILE__,__FUNCTION__,__LINE__, ##args)
77#else
Wolfgang Denke2601822006-06-14 18:14:56 +020078#define DBG(fmt,args...)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020079#endif
80
81/* Static Data */
82#ifdef __SIMULATE_ERROR__
Wolfgang Denke2601822006-06-14 18:14:56 +020083static char err_poison_test = 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020084#endif
85static struct mpc8xx_ep ep_ref[MAX_ENDPOINTS];
86static u32 address_base = STATE_NOT_READY;
87static mpc8xx_udc_state_t udc_state = 0;
88static struct usb_device_instance *udc_device = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +020089static volatile usb_epb_t *endpoints[MAX_ENDPOINTS];
90static volatile cbd_t *tx_cbd[TX_RING_SIZE];
91static volatile cbd_t *rx_cbd[RX_RING_SIZE];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +020092static volatile immap_t *immr = 0;
93static volatile cpm8xx_t *cp = 0;
94static volatile usb_pram_t *usb_paramp = 0;
95static volatile usb_t *usbp = 0;
96static int rx_ct = 0;
97static int tx_ct = 0;
98
99/* Static Function Declarations */
100static void mpc8xx_udc_state_transition_up (usb_device_state_t initial,
Wolfgang Denke2601822006-06-14 18:14:56 +0200101 usb_device_state_t final);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200102static void mpc8xx_udc_state_transition_down (usb_device_state_t initial,
Wolfgang Denke2601822006-06-14 18:14:56 +0200103 usb_device_state_t final);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200104static void mpc8xx_udc_stall (unsigned int ep);
Wolfgang Denke2601822006-06-14 18:14:56 +0200105static void mpc8xx_udc_flush_tx_fifo (int epid);
106static void mpc8xx_udc_flush_rx_fifo (void);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200107static void mpc8xx_udc_clear_rxbd (volatile cbd_t * rx_cbdp);
Wolfgang Denke2601822006-06-14 18:14:56 +0200108static void mpc8xx_udc_init_tx (struct usb_endpoint_instance *epi,
109 struct urb *tx_urb);
110static void mpc8xx_udc_dump_request (struct usb_device_request *request);
111static void mpc8xx_udc_clock_init (volatile immap_t * immr,
112 volatile cpm8xx_t * cp);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200113static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi);
114static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp);
Wolfgang Denke2601822006-06-14 18:14:56 +0200115static void mpc8xx_udc_ep0_rx (volatile cbd_t * rx_cbdp);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200116static void mpc8xx_udc_cbd_init (void);
117static void mpc8xx_udc_endpoint_init (void);
118static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size);
119static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment);
120static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp);
121static void mpc8xx_udc_set_nak (unsigned int ep);
Wolfgang Denke2601822006-06-14 18:14:56 +0200122static short mpc8xx_udc_handle_txerr (void);
123static void mpc8xx_udc_advance_rx (volatile cbd_t ** rx_cbdp, int epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200124
125/******************************************************************************
Wolfgang Denke2601822006-06-14 18:14:56 +0200126 Global Linkage
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200127 *****************************************************************************/
128
129/* udc_init
130 *
131 * Do initial bus gluing
132 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200133int udc_init (void)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200134{
135 /* Init various pointers */
136 immr = (immap_t *) CFG_IMMR;
Wolfgang Denke2601822006-06-14 18:14:56 +0200137 cp = (cpm8xx_t *) & (immr->im_cpm);
138 usb_paramp = (usb_pram_t *) & (cp->cp_dparam[PROFF_USB]);
139 usbp = (usb_t *) & (cp->cp_scc[0]);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200140
Wolfgang Denke2601822006-06-14 18:14:56 +0200141 memset (ep_ref, 0x00, (sizeof (struct mpc8xx_ep) * MAX_ENDPOINTS));
142
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200143 udc_device = 0;
144 udc_state = STATE_NOT_READY;
Wolfgang Denke2601822006-06-14 18:14:56 +0200145
146 usbp->usmod = 0x00;
147 usbp->uscom = 0;
148
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200149 /* Set USB Frame #0, Respond at Address & Get a clock source */
150 usbp->usaddr = 0x00;
151 mpc8xx_udc_clock_init (immr, cp);
Wolfgang Denke2601822006-06-14 18:14:56 +0200152
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200153 /* PA15, PA14 as perhiperal USBRXD and USBOE */
Wolfgang Denke2601822006-06-14 18:14:56 +0200154 immr->im_ioport.iop_padir &= ~0x0003;
155 immr->im_ioport.iop_papar |= 0x0003;
156
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200157 /* PC11/PC10 as peripheral USBRXP USBRXN */
Wolfgang Denke2601822006-06-14 18:14:56 +0200158 immr->im_ioport.iop_pcso |= 0x0030;
159
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200160 /* PC7/PC6 as perhiperal USBTXP and USBTXN */
Wolfgang Denke2601822006-06-14 18:14:56 +0200161 immr->im_ioport.iop_pcdir |= 0x0300;
162 immr->im_ioport.iop_pcpar |= 0x0300;
163
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200164 /* Set the base address */
Wolfgang Denke2601822006-06-14 18:14:56 +0200165 address_base = (u32) (cp->cp_dpmem + CPM_USB_BASE);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200166
167 /* Initialise endpoints and circular buffers */
Wolfgang Denke2601822006-06-14 18:14:56 +0200168 mpc8xx_udc_endpoint_init ();
169 mpc8xx_udc_cbd_init ();
170
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200171 /* Assign allocated Dual Port Endpoint descriptors */
Wolfgang Denke2601822006-06-14 18:14:56 +0200172 usb_paramp->ep0ptr = (u32) endpoints[0];
173 usb_paramp->ep1ptr = (u32) endpoints[1];
174 usb_paramp->ep2ptr = (u32) endpoints[2];
175 usb_paramp->ep3ptr = (u32) endpoints[3];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200176 usb_paramp->frame_n = 0;
177
Wolfgang Denke2601822006-06-14 18:14:56 +0200178 DBG ("ep0ptr=0x%08x ep1ptr=0x%08x ep2ptr=0x%08x ep3ptr=0x%08x\n",
179 usb_paramp->ep0ptr, usb_paramp->ep1ptr, usb_paramp->ep2ptr,
180 usb_paramp->ep3ptr);
181
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200182 return 0;
183}
184
185/* udc_irq
186 *
187 * Poll for whatever events may have occured
188 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200189void udc_irq (void)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200190{
191 int epid = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200192 volatile cbd_t *rx_cbdp = 0;
193 volatile cbd_t *rx_cbdp_base = 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200194
Wolfgang Denke2601822006-06-14 18:14:56 +0200195 if (udc_state != STATE_READY) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200196 return;
197 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200198
199 if (usbp->usber & USB_E_BSY) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200200 /* This shouldn't happen. If it does then it's a bug ! */
Wolfgang Denke2601822006-06-14 18:14:56 +0200201 usbp->usber |= USB_E_BSY;
202 mpc8xx_udc_flush_rx_fifo ();
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200203 }
204
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200205 /* Scan all RX/Bidirectional Endpoints for RX data. */
Wolfgang Denke2601822006-06-14 18:14:56 +0200206 for (epid = 0; epid < MAX_ENDPOINTS; epid++) {
207 if (!ep_ref[epid].prx) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200208 continue;
209 }
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200210 rx_cbdp = rx_cbdp_base = ep_ref[epid].prx;
Wolfgang Denke2601822006-06-14 18:14:56 +0200211
212 do {
213 if (!(rx_cbdp->cbd_sc & RX_BD_E)) {
214
215 if (rx_cbdp->cbd_sc & 0x1F) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200216 /* Corrupt data discard it.
Wolfgang Denke2601822006-06-14 18:14:56 +0200217 * Controller has NAK'd this packet.
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200218 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200219 mpc8xx_udc_clear_rxbd (rx_cbdp);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200220
Wolfgang Denke2601822006-06-14 18:14:56 +0200221 } else {
222 if (!epid) {
223 mpc8xx_udc_ep0_rx (rx_cbdp);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200224
Wolfgang Denke2601822006-06-14 18:14:56 +0200225 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200226 /* Process data */
Wolfgang Denke2601822006-06-14 18:14:56 +0200227 mpc8xx_udc_set_nak (epid);
228 mpc8xx_udc_epn_rx (epid, rx_cbdp);
229 mpc8xx_udc_clear_rxbd (rx_cbdp);
230 }
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200231 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200232
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200233 /* Advance RX CBD pointer */
Wolfgang Denke2601822006-06-14 18:14:56 +0200234 mpc8xx_udc_advance_rx (&rx_cbdp, epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200235 ep_ref[epid].prx = rx_cbdp;
Wolfgang Denke2601822006-06-14 18:14:56 +0200236 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200237 /* Advance RX CBD pointer */
Wolfgang Denke2601822006-06-14 18:14:56 +0200238 mpc8xx_udc_advance_rx (&rx_cbdp, epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200239 }
240
Wolfgang Denke2601822006-06-14 18:14:56 +0200241 } while (rx_cbdp != rx_cbdp_base);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200242 }
243
244 /* Handle TX events as appropiate, the correct place to do this is
245 * in a tx routine. Perhaps TX on epn was pre-empted by ep0
246 */
247
Wolfgang Denke2601822006-06-14 18:14:56 +0200248 if (usbp->usber & USB_E_TXB) {
249 usbp->usber |= USB_E_TXB;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200250 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200251
252 if (usbp->usber & (USB_TX_ERRMASK)) {
253 mpc8xx_udc_handle_txerr ();
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200254 }
255
256 /* Switch to the default state, respond at the default address */
Wolfgang Denke2601822006-06-14 18:14:56 +0200257 if (usbp->usber & USB_E_RESET) {
258 usbp->usber |= USB_E_RESET;
259 usbp->usaddr = 0x00;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200260 udc_device->device_state = STATE_DEFAULT;
261 }
262
Wolfgang Denke2601822006-06-14 18:14:56 +0200263 /* if(usbp->usber&USB_E_IDLE){
264 We could suspend here !
265 usbp->usber|=USB_E_IDLE;
266 DBG("idle state change\n");
267 }
268 if(usbp->usbs){
269 We could resume here when IDLE is deasserted !
270 Not worth doing, so long as we are self powered though.
271 }
272 */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200273
274 return;
275}
276
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200277/* udc_endpoint_write
278 *
279 * Write some data to an endpoint
280 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200281int udc_endpoint_write (struct usb_endpoint_instance *epi)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200282{
283 int ep = 0;
284 short epid = 1, unnak = 0, ret = 0;
285
Wolfgang Denke2601822006-06-14 18:14:56 +0200286 if (udc_state != STATE_READY) {
287 ERR ("invalid udc_state != STATE_READY!\n");
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200288 return -1;
289 }
290
Wolfgang Denke2601822006-06-14 18:14:56 +0200291 if (!udc_device || !epi) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200292 return -1;
293 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200294
295 if (udc_device->device_state != STATE_CONFIGURED) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200296 return -1;
297 }
298
299 ep = epi->endpoint_address & 0x03;
Wolfgang Denke2601822006-06-14 18:14:56 +0200300 if (ep >= MAX_ENDPOINTS) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200301 return -1;
302 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200303
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200304 /* Set NAK for all RX endpoints during TX */
Wolfgang Denke2601822006-06-14 18:14:56 +0200305 for (epid = 1; epid < MAX_ENDPOINTS; epid++) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200306
307 /* Don't set NAK on DATA IN/CONTROL endpoints */
Wolfgang Denke2601822006-06-14 18:14:56 +0200308 if (ep_ref[epid].sc & USB_DIR_IN) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200309 continue;
310 }
311
Wolfgang Denke2601822006-06-14 18:14:56 +0200312 if (!(usbp->usep[epid] & (USEP_THS_NAK | USEP_RHS_NAK))) {
313 unnak |= 1 << epid;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200314 }
315
Wolfgang Denke2601822006-06-14 18:14:56 +0200316 mpc8xx_udc_set_nak (epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200317 }
318
Wolfgang Denke2601822006-06-14 18:14:56 +0200319 mpc8xx_udc_init_tx (&udc_device->bus->endpoint_array[ep],
320 epi->tx_urb);
321 ret = mpc8xx_udc_ep_tx (&udc_device->bus->endpoint_array[ep]);
322
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200323 /* Remove temporary NAK */
Wolfgang Denke2601822006-06-14 18:14:56 +0200324 for (epid = 1; epid < MAX_ENDPOINTS; epid++) {
325 if (unnak & (1 << epid)) {
326 udc_unset_nak (epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200327 }
328 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200329
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200330 return ret;
331}
332
333/* mpc8xx_udc_assign_urb
334 *
335 * Associate a given urb to an endpoint TX or RX transmit/receive buffers
336 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200337static int mpc8xx_udc_assign_urb (int ep, char direction)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200338{
339 struct usb_endpoint_instance *epi = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200340
341 if (ep >= MAX_ENDPOINTS) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200342 goto err;
343 }
344 epi = &udc_device->bus->endpoint_array[ep];
Wolfgang Denke2601822006-06-14 18:14:56 +0200345 if (!epi) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200346 goto err;
347 }
348
Wolfgang Denke2601822006-06-14 18:14:56 +0200349 if (!ep_ref[ep].urb) {
350 ep_ref[ep].urb = usbd_alloc_urb (udc_device, udc_device->bus->endpoint_array);
351 if (!ep_ref[ep].urb) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200352 goto err;
353 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200354 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200355 ep_ref[ep].urb->actual_length = 0;
356 }
357
Wolfgang Denke2601822006-06-14 18:14:56 +0200358 switch (direction) {
359 case USB_DIR_IN:
360 epi->tx_urb = ep_ref[ep].urb;
361 break;
362 case USB_DIR_OUT:
363 epi->rcv_urb = ep_ref[ep].urb;
364 break;
365 default:
366 goto err;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200367 }
368 return 0;
369
Wolfgang Denke2601822006-06-14 18:14:56 +0200370 err:
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200371 udc_state = STATE_ERROR;
372 return -1;
373}
374
375/* udc_setup_ep
376 *
377 * Associate U-Boot software endpoints to mpc8xx endpoint parameter ram
378 * Isochronous endpoints aren't yet supported!
379 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200380void udc_setup_ep (struct usb_device_instance *device, unsigned int ep,
381 struct usb_endpoint_instance *epi)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200382{
383 uchar direction = 0;
384 int ep_attrib = 0;
385
Wolfgang Denke2601822006-06-14 18:14:56 +0200386 if (epi && (ep < MAX_ENDPOINTS)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200387
Wolfgang Denke2601822006-06-14 18:14:56 +0200388 if (ep == 0) {
389 if (epi->rcv_attributes != USB_ENDPOINT_XFER_CONTROL
390 || epi->tx_attributes !=
391 USB_ENDPOINT_XFER_CONTROL) {
392
393 /* ep0 must be a control endpoint */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200394 udc_state = STATE_ERROR;
395 return;
396
397 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200398 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
399 mpc8xx_udc_cbd_attach (ep, epi->tx_packetSize,
400 epi->rcv_packetSize);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200401 }
402 usbp->usep[ep] = 0x0000;
403 return;
404 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200405
406 if ((epi->endpoint_address & USB_ENDPOINT_DIR_MASK)
407 == USB_DIR_IN) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200408
409 direction = 1;
410 ep_attrib = epi->tx_attributes;
411 epi->rcv_packetSize = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200412 ep_ref[ep].sc |= USB_DIR_IN;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200413 } else {
Wolfgang Denke2601822006-06-14 18:14:56 +0200414
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200415 direction = 0;
416 ep_attrib = epi->rcv_attributes;
Wolfgang Denke2601822006-06-14 18:14:56 +0200417 epi->tx_packetSize = 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200418 ep_ref[ep].sc &= ~USB_DIR_IN;
419 }
420
Wolfgang Denke2601822006-06-14 18:14:56 +0200421 if (mpc8xx_udc_assign_urb (ep, epi->endpoint_address
422 & USB_ENDPOINT_DIR_MASK)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200423 return;
424 }
425
Wolfgang Denke2601822006-06-14 18:14:56 +0200426 switch (ep_attrib) {
427 case USB_ENDPOINT_XFER_CONTROL:
428 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
429 mpc8xx_udc_cbd_attach (ep,
430 epi->tx_packetSize,
431 epi->rcv_packetSize);
432 }
433 usbp->usep[ep] = ep << 12;
434 epi->rcv_urb = epi->tx_urb = ep_ref[ep].urb;
435
436 break;
437 case USB_ENDPOINT_XFER_BULK:
438 case USB_ENDPOINT_XFER_INT:
439 if (!(ep_ref[ep].sc & EP_ATTACHED)) {
440 if (direction) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200441 mpc8xx_udc_cbd_attach (ep,
Wolfgang Denke2601822006-06-14 18:14:56 +0200442 epi->tx_packetSize,
443 0);
444 } else {
445 mpc8xx_udc_cbd_attach (ep,
446 0,
447 epi->rcv_packetSize);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200448 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200449 }
450 usbp->usep[ep] = (ep << 12) | ((ep_attrib) << 8);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200451
Wolfgang Denke2601822006-06-14 18:14:56 +0200452 break;
453 case USB_ENDPOINT_XFER_ISOC:
454 default:
455 serial_printf ("Error endpoint attrib %d>3\n", ep_attrib);
456 udc_state = STATE_ERROR;
457 break;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200458 }
459 }
460
461}
462
463/* udc_connect
464 *
465 * Move state, switch on the USB
466 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200467void udc_connect (void)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200468{
Wolfgang Denke2601822006-06-14 18:14:56 +0200469 /* Enable pull-up resistor on D+
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200470 * TODO: fit a pull-up resistor to drive SE0 for > 2.5us
471 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200472
473 if (udc_state != STATE_ERROR) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200474 udc_state = STATE_READY;
Wolfgang Denke2601822006-06-14 18:14:56 +0200475 usbp->usmod |= USMOD_EN;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200476 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200477}
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200478
479/* udc_disconnect
480 *
481 * Disconnect is not used but, is included for completeness
482 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200483void udc_disconnect (void)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200484{
485 /* Disable pull-up resistor on D-
486 * TODO: fix a pullup resistor to control this
487 */
488
Wolfgang Denke2601822006-06-14 18:14:56 +0200489 if (udc_state != STATE_ERROR) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200490 udc_state = STATE_NOT_READY;
491 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200492 usbp->usmod &= ~USMOD_EN;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200493}
494
495/* udc_enable
Wolfgang Denke2601822006-06-14 18:14:56 +0200496 *
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200497 * Grab an EP0 URB, register interest in a subset of USB events
498 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200499void udc_enable (struct usb_device_instance *device)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200500{
Wolfgang Denke2601822006-06-14 18:14:56 +0200501 if (udc_state == STATE_ERROR) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200502 return;
503 }
504
505 udc_device = device;
Wolfgang Denke2601822006-06-14 18:14:56 +0200506
507 if (!ep_ref[0].urb) {
508 ep_ref[0].urb = usbd_alloc_urb (device, device->bus->endpoint_array);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200509 }
510
511 /* Register interest in all events except SOF, enable transceiver */
Wolfgang Denke2601822006-06-14 18:14:56 +0200512 usbp->usber = 0x03FF;
513 usbp->usbmr = 0x02F7;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200514
515 return;
516}
517
518/* udc_disable
519 *
520 * disable the currently hooked device
521 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200522void udc_disable (void)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200523{
524 int i = 0;
525
Wolfgang Denke2601822006-06-14 18:14:56 +0200526 if (udc_state == STATE_ERROR) {
527 DBG ("Won't disable UDC. udc_state==STATE_ERROR !\n");
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200528 return;
529 }
530
531 udc_device = 0;
532
Wolfgang Denke2601822006-06-14 18:14:56 +0200533 for (; i < MAX_ENDPOINTS; i++) {
534 if (ep_ref[i].urb) {
535 usbd_dealloc_urb (ep_ref[i].urb);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200536 ep_ref[i].urb = 0;
537 }
538 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200539
540 usbp->usbmr = 0x00;
541 usbp->usmod = ~USMOD_EN;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200542 udc_state = STATE_NOT_READY;
543}
544
545/* udc_startup_events
546 *
547 * Enable the specified device
548 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200549void udc_startup_events (struct usb_device_instance *device)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200550{
Wolfgang Denke2601822006-06-14 18:14:56 +0200551 udc_enable (device);
552 if (udc_state == STATE_READY) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200553 usbd_device_event_irq (device, DEVICE_CREATE, 0);
554 }
555}
556
557/* udc_set_nak
Wolfgang Denke2601822006-06-14 18:14:56 +0200558 *
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200559 * Allow upper layers to signal lower layers should not accept more RX data
560 *
561 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200562void udc_set_nak (int epid)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200563{
Wolfgang Denke2601822006-06-14 18:14:56 +0200564 if (epid) {
565 mpc8xx_udc_set_nak (epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200566 }
567}
568
Wolfgang Denke2601822006-06-14 18:14:56 +0200569/* udc_unset_nak
570 *
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200571 * Suspend sending of NAK tokens for DATA OUT tokens on a given endpoint.
572 * Switch off NAKing on this endpoint to accept more data output from host.
573 *
574 */
575void udc_unset_nak (int epid)
576{
Wolfgang Denke2601822006-06-14 18:14:56 +0200577 if (epid > MAX_ENDPOINTS) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200578 return;
579 }
580
Wolfgang Denke2601822006-06-14 18:14:56 +0200581 if (usbp->usep[epid] & (USEP_THS_NAK | USEP_RHS_NAK)) {
582 usbp->usep[epid] &= ~(USEP_THS_NAK | USEP_RHS_NAK);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200583 __asm__ ("eieio");
584 }
585}
586
587/******************************************************************************
Wolfgang Denke2601822006-06-14 18:14:56 +0200588 Static Linkage
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200589******************************************************************************/
590
591/* udc_state_transition_up
592 * udc_state_transition_down
593 *
594 * Helper functions to implement device state changes. The device states and
595 * the events that transition between them are:
596 *
597 * STATE_ATTACHED
598 * || /\
599 * \/ ||
600 * DEVICE_HUB_CONFIGURED DEVICE_HUB_RESET
601 * || /\
602 * \/ ||
603 * STATE_POWERED
604 * || /\
605 * \/ ||
606 * DEVICE_RESET DEVICE_POWER_INTERRUPTION
607 * || /\
608 * \/ ||
609 * STATE_DEFAULT
610 * || /\
611 * \/ ||
612 * DEVICE_ADDRESS_ASSIGNED DEVICE_RESET
613 * || /\
614 * \/ ||
615 * STATE_ADDRESSED
616 * || /\
617 * \/ ||
618 * DEVICE_CONFIGURED DEVICE_DE_CONFIGURED
619 * || /\
620 * \/ ||
621 * STATE_CONFIGURED
622 *
623 * udc_state_transition_up transitions up (in the direction from STATE_ATTACHED
624 * to STATE_CONFIGURED) from the specified initial state to the specified final
625 * state, passing through each intermediate state on the way. If the initial
626 * state is at or above (i.e. nearer to STATE_CONFIGURED) the final state, then
627 * no state transitions will take place.
628 *
629 * udc_state_transition_down transitions down (in the direction from
630 * STATE_CONFIGURED to STATE_ATTACHED) from the specified initial state to the
631 * specified final state, passing through each intermediate state on the way.
632 * If the initial state is at or below (i.e. nearer to STATE_ATTACHED) the final
633 * state, then no state transitions will take place.
634 *
635 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200636
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200637static void mpc8xx_udc_state_transition_up (usb_device_state_t initial,
Wolfgang Denke2601822006-06-14 18:14:56 +0200638 usb_device_state_t final)
639{
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200640 if (initial < final) {
641 switch (initial) {
642 case STATE_ATTACHED:
643 usbd_device_event_irq (udc_device,
644 DEVICE_HUB_CONFIGURED, 0);
645 if (final == STATE_POWERED)
646 break;
647 case STATE_POWERED:
648 usbd_device_event_irq (udc_device, DEVICE_RESET, 0);
649 if (final == STATE_DEFAULT)
650 break;
651 case STATE_DEFAULT:
652 usbd_device_event_irq (udc_device,
653 DEVICE_ADDRESS_ASSIGNED, 0);
654 if (final == STATE_ADDRESSED)
655 break;
656 case STATE_ADDRESSED:
657 usbd_device_event_irq (udc_device, DEVICE_CONFIGURED,
658 0);
659 case STATE_CONFIGURED:
660 break;
661 default:
662 break;
663 }
664 }
665}
666
667static void mpc8xx_udc_state_transition_down (usb_device_state_t initial,
Wolfgang Denke2601822006-06-14 18:14:56 +0200668 usb_device_state_t final)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200669{
670 if (initial > final) {
671 switch (initial) {
672 case STATE_CONFIGURED:
Wolfgang Denke2601822006-06-14 18:14:56 +0200673 usbd_device_event_irq (udc_device,
674 DEVICE_DE_CONFIGURED, 0);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200675 if (final == STATE_ADDRESSED)
676 break;
677 case STATE_ADDRESSED:
678 usbd_device_event_irq (udc_device, DEVICE_RESET, 0);
679 if (final == STATE_DEFAULT)
680 break;
681 case STATE_DEFAULT:
Wolfgang Denke2601822006-06-14 18:14:56 +0200682 usbd_device_event_irq (udc_device,
683 DEVICE_POWER_INTERRUPTION, 0);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200684 if (final == STATE_POWERED)
685 break;
686 case STATE_POWERED:
687 usbd_device_event_irq (udc_device, DEVICE_HUB_RESET,
Wolfgang Denke2601822006-06-14 18:14:56 +0200688 0);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200689 case STATE_ATTACHED:
690 break;
691 default:
692 break;
693 }
694 }
695}
696
697/* mpc8xx_udc_stall
698 *
699 * Force returning of STALL tokens on the given endpoint. Protocol or function
700 * STALL conditions are permissable here
701 */
702static void mpc8xx_udc_stall (unsigned int ep)
703{
704 usbp->usep[ep] |= STALL_BITMASK;
705}
706
707/* mpc8xx_udc_set_nak
708 *
709 * Force returning of NAK responses for the given endpoint as a kind of very
710 * simple flow control
Wolfgang Denke2601822006-06-14 18:14:56 +0200711 */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200712static void mpc8xx_udc_set_nak (unsigned int ep)
713{
714 usbp->usep[ep] |= NAK_BITMASK;
715 __asm__ ("eieio");
716}
717
718/* mpc8xx_udc_handle_txerr
719 *
720 * Handle errors relevant to TX. Return a status code to allow calling
721 * indicative of what if anything happened
722 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200723static short mpc8xx_udc_handle_txerr ()
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200724{
725 short ep = 0, ret = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200726
727 for (; ep < TX_RING_SIZE; ep++) {
728 if (usbp->usber & (0x10 << ep)) {
729
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200730 /* Timeout or underrun */
Wolfgang Denke2601822006-06-14 18:14:56 +0200731 if (tx_cbd[ep]->cbd_sc & 0x06) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200732 ret = 1;
Wolfgang Denke2601822006-06-14 18:14:56 +0200733 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200734
Wolfgang Denke2601822006-06-14 18:14:56 +0200735 } else {
736 if (usbp->usep[ep] & STALL_BITMASK) {
737 if (!ep) {
738 usbp->usep[ep] &= ~STALL_BITMASK;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200739 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200740 } /* else NAK */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200741 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200742 usbp->usber |= (0x10 << ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200743 }
744 }
745 return ret;
746}
747
748/* mpc8xx_udc_advance_rx
749 *
750 * Advance cbd rx
751 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200752static void mpc8xx_udc_advance_rx (volatile cbd_t ** rx_cbdp, int epid)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200753{
Wolfgang Denke2601822006-06-14 18:14:56 +0200754 if ((*rx_cbdp)->cbd_sc & RX_BD_W) {
755 *rx_cbdp = (volatile cbd_t *) (endpoints[epid]->rbase + CFG_IMMR);
756
757 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200758 (*rx_cbdp)++;
759 }
760}
761
762
763/* mpc8xx_udc_flush_tx_fifo
764 *
765 * Flush a given TX fifo. Assumes one tx cbd per endpoint
766 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200767static void mpc8xx_udc_flush_tx_fifo (int epid)
768{
769 volatile cbd_t *tx_cbdp = 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200770
Wolfgang Denke2601822006-06-14 18:14:56 +0200771 if (epid > MAX_ENDPOINTS) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200772 return;
773 }
774
775 /* TX stop */
Wolfgang Denke2601822006-06-14 18:14:56 +0200776 immr->im_cpm.cp_cpcr = ((epid << 2) | 0x1D01);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200777 __asm__ ("eieio");
Wolfgang Denke2601822006-06-14 18:14:56 +0200778 while (immr->im_cpm.cp_cpcr & 0x01);
779
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200780 usbp->uscom = 0x40 | 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200781
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200782 /* reset ring */
Wolfgang Denke2601822006-06-14 18:14:56 +0200783 tx_cbdp = (cbd_t *) (endpoints[epid]->tbptr + CFG_IMMR);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200784 tx_cbdp->cbd_sc = (TX_BD_I | TX_BD_W);
785
Wolfgang Denke2601822006-06-14 18:14:56 +0200786
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200787 endpoints[epid]->tptr = endpoints[epid]->tbase;
Wolfgang Denke2601822006-06-14 18:14:56 +0200788 endpoints[epid]->tstate = 0x00;
789 endpoints[epid]->tbcnt = 0x00;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200790
791 /* TX start */
Wolfgang Denke2601822006-06-14 18:14:56 +0200792 immr->im_cpm.cp_cpcr = ((epid << 2) | 0x2D01);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200793 __asm__ ("eieio");
Wolfgang Denke2601822006-06-14 18:14:56 +0200794 while (immr->im_cpm.cp_cpcr & 0x01);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200795
796 return;
797}
798
799/* mpc8xx_udc_flush_rx_fifo
800 *
801 * For the sake of completeness of the namespace, it seems like
802 * a good-design-decision (tm) to include mpc8xx_udc_flush_rx_fifo();
803 * If RX_BD_E is true => a driver bug either here or in an upper layer
804 * not polling frequently enough. If RX_BD_E is true we have told the host
805 * we have accepted data but, the CPM found it had no-where to put that data
806 * which needless to say would be a bad thing.
807 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200808static void mpc8xx_udc_flush_rx_fifo ()
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200809{
810 int i = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200811
812 for (i = 0; i < RX_RING_SIZE; i++) {
813 if (!(rx_cbd[i]->cbd_sc & RX_BD_E)) {
814 ERR ("buf %p used rx data len = 0x%x sc=0x%x!\n",
815 rx_cbd[i], rx_cbd[i]->cbd_datlen,
816 rx_cbd[i]->cbd_sc);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200817
818 }
819 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200820 ERR ("BUG : Input over-run\n");
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200821}
822
823/* mpc8xx_udc_clear_rxbd
Wolfgang Denke2601822006-06-14 18:14:56 +0200824 *
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200825 * Release control of RX CBD to CP.
826 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200827static void mpc8xx_udc_clear_rxbd (volatile cbd_t * rx_cbdp)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200828{
829 rx_cbdp->cbd_datlen = 0x0000;
Wolfgang Denke2601822006-06-14 18:14:56 +0200830 rx_cbdp->cbd_sc = ((rx_cbdp->cbd_sc & RX_BD_W) | (RX_BD_E | RX_BD_I));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200831 __asm__ ("eieio");
832}
833
834/* mpc8xx_udc_tx_irq
835 *
836 * Parse for tx timeout, control RX or USB reset/busy conditions
837 * Return -1 on timeout, -2 on fatal error, else return zero
838 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200839static int mpc8xx_udc_tx_irq (int ep)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200840{
841 int i = 0;
842
Wolfgang Denke2601822006-06-14 18:14:56 +0200843 if (usbp->usber & (USB_TX_ERRMASK)) {
844 if (mpc8xx_udc_handle_txerr ()) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200845 /* Timeout, controlling function must retry send */
846 return -1;
847 }
848 }
849
Wolfgang Denke2601822006-06-14 18:14:56 +0200850 if (usbp->usber & (USB_E_RESET | USB_E_BSY)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200851 /* Fatal, abandon TX transaction */
852 return -2;
853 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200854
855 if (usbp->usber & USB_E_RXB) {
856 for (i = 0; i < RX_RING_SIZE; i++) {
857 if (!(rx_cbd[i]->cbd_sc & RX_BD_E)) {
858 if ((rx_cbd[i] == ep_ref[0].prx) || ep) {
859 return -2;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200860 }
861 }
862 }
863 }
864
865 return 0;
866}
867
868/* mpc8xx_udc_ep_tx
869 *
870 * Transmit in a re-entrant fashion outbound USB packets.
871 * Implement retry/timeout mechanism described in USB specification
872 * Toggle DATA0/DATA1 pids as necessary
873 * Introduces non-standard tx_retry. The USB standard has no scope for slave
874 * devices to give up TX, however tx_retry stops us getting stuck in an endless
875 * TX loop.
876 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200877static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200878{
879 struct urb *urb = epi->tx_urb;
Wolfgang Denke2601822006-06-14 18:14:56 +0200880 volatile cbd_t *tx_cbdp = 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200881 unsigned int ep = 0, pkt_len = 0, x = 0, tx_retry = 0;
882 int ret = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200883
884 if (!epi || (epi->endpoint_address & 0x03) >= MAX_ENDPOINTS || !urb) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200885 return -1;
886 }
887
888 ep = epi->endpoint_address & 0x03;
Wolfgang Denke2601822006-06-14 18:14:56 +0200889 tx_cbdp = (cbd_t *) (endpoints[ep]->tbptr + CFG_IMMR);
890
891 if (tx_cbdp->cbd_sc & TX_BD_R || usbp->usber & USB_E_TXB) {
892 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200893 usbp->usber |= USB_E_TXB;
894 };
895
Wolfgang Denke2601822006-06-14 18:14:56 +0200896 while (tx_retry++ < 100) {
897 ret = mpc8xx_udc_tx_irq (ep);
898 if (ret == -1) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200899 /* ignore timeout here */
Wolfgang Denke2601822006-06-14 18:14:56 +0200900 } else if (ret == -2) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200901 /* Abandon TX */
Wolfgang Denke2601822006-06-14 18:14:56 +0200902 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200903 return -1;
Wolfgang Denke2601822006-06-14 18:14:56 +0200904 }
905
906 tx_cbdp = (cbd_t *) (endpoints[ep]->tbptr + CFG_IMMR);
907 while (tx_cbdp->cbd_sc & TX_BD_R) {
908 };
909 tx_cbdp->cbd_sc = (tx_cbdp->cbd_sc & TX_BD_W);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200910
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200911 pkt_len = urb->actual_length - epi->sent;
912
Wolfgang Denke2601822006-06-14 18:14:56 +0200913 if (pkt_len > epi->tx_packetSize || pkt_len > EP_MAX_PKT) {
914 pkt_len = MIN (epi->tx_packetSize, EP_MAX_PKT);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200915 }
916
Wolfgang Denke2601822006-06-14 18:14:56 +0200917 for (x = 0; x < pkt_len; x++) {
918 *((unsigned char *) (tx_cbdp->cbd_bufaddr + x)) =
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200919 urb->buffer[epi->sent + x];
920 }
921 tx_cbdp->cbd_datlen = pkt_len;
Wolfgang Denke2601822006-06-14 18:14:56 +0200922 tx_cbdp->cbd_sc |= (CBD_TX_BITMASK | ep_ref[ep].pid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200923 __asm__ ("eieio");
924
Wolfgang Denke2601822006-06-14 18:14:56 +0200925#ifdef __SIMULATE_ERROR__
926 if (++err_poison_test == 2) {
927 err_poison_test = 0;
928 tx_cbdp->cbd_sc &= ~TX_BD_TC;
929 }
930#endif
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200931
Wolfgang Denke2601822006-06-14 18:14:56 +0200932 usbp->uscom = (USCOM_STR | ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200933
Wolfgang Denke2601822006-06-14 18:14:56 +0200934 while (!(usbp->usber & USB_E_TXB)) {
935 ret = mpc8xx_udc_tx_irq (ep);
936 if (ret == -1) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200937 /* TX timeout */
938 break;
Wolfgang Denke2601822006-06-14 18:14:56 +0200939 } else if (ret == -2) {
940 if (usbp->usber & USB_E_TXB) {
941 usbp->usber |= USB_E_TXB;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200942 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200943 mpc8xx_udc_flush_tx_fifo (ep);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200944 return -1;
945 }
946 };
947
Wolfgang Denke2601822006-06-14 18:14:56 +0200948 if (usbp->usber & USB_E_TXB) {
949 usbp->usber |= USB_E_TXB;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200950 }
951
952 /* ACK must be present <= 18bit times from TX */
Wolfgang Denke2601822006-06-14 18:14:56 +0200953 if (ret == -1) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200954 continue;
955 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200956
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200957 /* TX ACK : USB 2.0 8.7.2, Toggle PID, Advance TX */
958 epi->sent += pkt_len;
Wolfgang Denke2601822006-06-14 18:14:56 +0200959 epi->last = MIN (urb->actual_length - epi->sent, epi->tx_packetSize);
960 TOGGLE_TX_PID (ep_ref[ep].pid);
961
962 if (epi->sent >= epi->tx_urb->actual_length) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200963
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200964 epi->tx_urb->actual_length = 0;
965 epi->sent = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +0200966
967 if (ep_ref[ep].sc & EP_SEND_ZLP) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200968 ep_ref[ep].sc &= ~EP_SEND_ZLP;
Wolfgang Denke2601822006-06-14 18:14:56 +0200969 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200970 return 0;
971 }
972 }
973 }
Wolfgang Denke2601822006-06-14 18:14:56 +0200974
975 ERR ("TX fail, endpoint 0x%x tx bytes 0x%x/0x%x\n", ep, epi->sent,
976 epi->tx_urb->actual_length);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200977
978 return -1;
979}
980
981/* mpc8xx_udc_dump_request
982 *
983 * Dump a control request to console
984 */
Wolfgang Denke2601822006-06-14 18:14:56 +0200985static void mpc8xx_udc_dump_request (struct usb_device_request *request)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200986{
Wolfgang Denke2601822006-06-14 18:14:56 +0200987 DBG ("bmRequestType:%02x bRequest:%02x wValue:%04x "
988 "wIndex:%04x wLength:%04x ?\n",
989 request->bmRequestType,
990 request->bRequest,
991 request->wValue, request->wIndex, request->wLength);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200992
993 return;
994}
995
Wolfgang Denke2601822006-06-14 18:14:56 +0200996/* mpc8xx_udc_ep0_rx_setup
997 *
Wolfgang Denk3f0137b2006-06-14 17:45:53 +0200998 * Decode received ep0 SETUP packet. return non-zero on error
999 */
1000static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp)
1001{
1002 unsigned int x = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +02001003 struct urb *purb = ep_ref[0].urb;
1004 struct usb_endpoint_instance *epi =
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001005 &udc_device->bus->endpoint_array[0];
1006
Wolfgang Denke2601822006-06-14 18:14:56 +02001007 for (; x < rx_cbdp->cbd_datlen; x++) {
1008 *(((unsigned char *) &ep_ref[0].urb->device_request) + x) =
1009 *((unsigned char *) (rx_cbdp->cbd_bufaddr + x));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001010 }
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001011
Wolfgang Denke2601822006-06-14 18:14:56 +02001012 mpc8xx_udc_clear_rxbd (rx_cbdp);
1013
1014 if (ep0_recv_setup (purb)) {
1015 mpc8xx_udc_dump_request (&purb->device_request);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001016 return -1;
1017 }
1018
Wolfgang Denke2601822006-06-14 18:14:56 +02001019 if ((purb->device_request.bmRequestType & USB_REQ_DIRECTION_MASK)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001020 == USB_REQ_HOST2DEVICE) {
1021
Wolfgang Denke2601822006-06-14 18:14:56 +02001022 switch (purb->device_request.bRequest) {
1023 case USB_REQ_SET_ADDRESS:
1024 /* Send the Status OUT ZLP */
1025 ep_ref[0].pid = TX_BD_PID_DATA1;
1026 purb->actual_length = 0;
1027 mpc8xx_udc_init_tx (epi, purb);
1028 mpc8xx_udc_ep_tx (epi);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001029
Wolfgang Denke2601822006-06-14 18:14:56 +02001030 /* Move to the addressed state */
1031 usbp->usaddr = udc_device->address;
1032 mpc8xx_udc_state_transition_up (udc_device->device_state,
1033 STATE_ADDRESSED);
1034 return 0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001035
Wolfgang Denke2601822006-06-14 18:14:56 +02001036 case USB_REQ_SET_CONFIGURATION:
1037 if (!purb->device_request.wValue) {
1038 /* Respond at default address */
1039 usbp->usaddr = 0x00;
1040 mpc8xx_udc_state_transition_down (udc_device->device_state,
1041 STATE_ADDRESSED);
1042 } else {
1043 /* TODO: Support multiple configurations */
1044 mpc8xx_udc_state_transition_up (udc_device->device_state,
1045 STATE_CONFIGURED);
1046 for (x = 1; x < MAX_ENDPOINTS; x++) {
1047 if ((udc_device->bus->endpoint_array[x].endpoint_address & USB_ENDPOINT_DIR_MASK)
1048 == USB_DIR_IN) {
1049 ep_ref[x].pid = TX_BD_PID_DATA0;
1050 } else {
1051 ep_ref[x].pid = RX_BD_PID_DATA0;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001052 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001053 /* Set configuration must unstall endpoints */
1054 usbp->usep[x] &= ~STALL_BITMASK;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001055 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001056 }
1057 break;
1058 default:
1059 /* CDC/Vendor specific */
1060 break;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001061 }
1062
1063 /* Send ZLP as ACK in Status OUT phase */
1064 ep_ref[0].pid = TX_BD_PID_DATA1;
1065 purb->actual_length = 0;
Wolfgang Denke2601822006-06-14 18:14:56 +02001066 mpc8xx_udc_init_tx (epi, purb);
1067 mpc8xx_udc_ep_tx (epi);
1068
1069 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001070
Wolfgang Denke2601822006-06-14 18:14:56 +02001071 if (purb->actual_length) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001072 ep_ref[0].pid = TX_BD_PID_DATA1;
Wolfgang Denke2601822006-06-14 18:14:56 +02001073 mpc8xx_udc_init_tx (epi, purb);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001074
Wolfgang Denke2601822006-06-14 18:14:56 +02001075 if (!(purb->actual_length % EP0_MAX_PACKET_SIZE)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001076 ep_ref[0].sc |= EP_SEND_ZLP;
1077 }
1078
Wolfgang Denke2601822006-06-14 18:14:56 +02001079 if (purb->device_request.wValue ==
1080 USB_DESCRIPTOR_TYPE_DEVICE) {
1081 if (le16_to_cpu (purb->device_request.wLength)
1082 > purb->actual_length) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001083 /* Send EP0_MAX_PACKET_SIZE bytes
1084 * unless correct size requested.
1085 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001086 if (purb->actual_length > epi->tx_packetSize) {
1087 purb->actual_length = epi->tx_packetSize;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001088 }
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001089 }
1090 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001091 mpc8xx_udc_ep_tx (epi);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001092
Wolfgang Denke2601822006-06-14 18:14:56 +02001093 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001094 /* Corrupt SETUP packet? */
Wolfgang Denke2601822006-06-14 18:14:56 +02001095 ERR ("Zero length data or SETUP with DATA-IN phase ?\n");
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001096 return 1;
1097 }
1098 }
1099 return 0;
1100}
1101
1102/* mpc8xx_udc_init_tx
1103 *
1104 * Setup some basic parameters for a TX transaction
1105 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001106static void mpc8xx_udc_init_tx (struct usb_endpoint_instance *epi,
1107 struct urb *tx_urb)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001108{
1109 epi->sent = 0;
1110 epi->last = 0;
1111 epi->tx_urb = tx_urb;
1112}
1113
1114/* mpc8xx_udc_ep0_rx
1115 *
1116 * Receive ep0/control USB data. Parse and possibly send a response.
1117 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001118static void mpc8xx_udc_ep0_rx (volatile cbd_t * rx_cbdp)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001119{
Wolfgang Denke2601822006-06-14 18:14:56 +02001120 if (rx_cbdp->cbd_sc & RX_BD_PID_SETUP) {
1121
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001122 /* Unconditionally accept SETUP packets */
Wolfgang Denke2601822006-06-14 18:14:56 +02001123 if (mpc8xx_udc_ep0_rx_setup (rx_cbdp)) {
1124 mpc8xx_udc_stall (0);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001125 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001126
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001127 } else {
Wolfgang Denke2601822006-06-14 18:14:56 +02001128
1129 mpc8xx_udc_clear_rxbd (rx_cbdp);
1130
1131 if ((rx_cbdp->cbd_datlen - 2)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001132 /* SETUP with a DATA phase
Wolfgang Denke2601822006-06-14 18:14:56 +02001133 * outside of SETUP packet.
1134 * Reply with STALL.
1135 */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001136 mpc8xx_udc_stall (0);
1137 }
1138 }
1139}
1140
1141/* mpc8xx_udc_epn_rx
1142 *
1143 * Receive some data from cbd into USB system urb data abstraction
Wolfgang Denke2601822006-06-14 18:14:56 +02001144 * Upper layers should NAK if there is insufficient RX data space
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001145 */
1146static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp)
1147{
1148 struct usb_endpoint_instance *epi = 0;
1149 struct urb *urb = 0;
1150 unsigned int x = 0;
1151
Wolfgang Denke2601822006-06-14 18:14:56 +02001152 if (epid >= MAX_ENDPOINTS || !rx_cbdp->cbd_datlen) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001153 return 0;
1154 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001155
1156 /* USB 2.0 PDF section 8.6.4
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001157 * Discard data with invalid PID it is a resend.
1158 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001159 if (ep_ref[epid].pid != (rx_cbdp->cbd_sc & 0xC0)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001160 return 1;
1161 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001162 TOGGLE_RX_PID (ep_ref[epid].pid);
1163
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001164 epi = &udc_device->bus->endpoint_array[epid];
1165 urb = epi->rcv_urb;
1166
Wolfgang Denke2601822006-06-14 18:14:56 +02001167 for (; x < (rx_cbdp->cbd_datlen - 2); x++) {
1168 *((unsigned char *) (urb->buffer + urb->actual_length + x)) =
1169 *((unsigned char *) (rx_cbdp->cbd_bufaddr + x));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001170 }
1171
Wolfgang Denke2601822006-06-14 18:14:56 +02001172 if (x) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001173 usbd_rcv_complete (epi, x, 0);
Wolfgang Denke2601822006-06-14 18:14:56 +02001174 if (ep_ref[epid].urb->status == RECV_ERROR) {
1175 DBG ("RX error unset NAK\n");
1176 udc_unset_nak (epid);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001177 }
Wolfgang Denke2601822006-06-14 18:14:56 +02001178 }
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001179 return x;
1180}
1181
1182/* mpc8xx_udc_clock_init
1183 *
Wolfgang Denke2601822006-06-14 18:14:56 +02001184 * Obtain a clock reference for Full Speed Signaling
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001185 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001186static void mpc8xx_udc_clock_init (volatile immap_t * immr,
1187 volatile cpm8xx_t * cp)
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001188{
1189
1190#if defined(CFG_USB_EXTC_CLK)
1191
1192 /* This has been tested with a 48MHz crystal on CLK6 */
Wolfgang Denke2601822006-06-14 18:14:56 +02001193 switch (CFG_USB_EXTC_CLK) {
1194 case 1:
1195 immr->im_ioport.iop_papar |= 0x0100;
1196 immr->im_ioport.iop_padir &= ~0x0100;
1197 cp->cp_sicr |= 0x24;
1198 break;
1199 case 2:
1200 immr->im_ioport.iop_papar |= 0x0200;
1201 immr->im_ioport.iop_padir &= ~0x0200;
1202 cp->cp_sicr |= 0x2D;
1203 break;
1204 case 3:
1205 immr->im_ioport.iop_papar |= 0x0400;
1206 immr->im_ioport.iop_padir &= ~0x0400;
1207 cp->cp_sicr |= 0x36;
1208 break;
1209 case 4:
1210 immr->im_ioport.iop_papar |= 0x0800;
1211 immr->im_ioport.iop_padir &= ~0x0800;
1212 cp->cp_sicr |= 0x3F;
1213 break;
1214 default:
1215 udc_state = STATE_ERROR;
1216 break;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001217 }
1218
1219#elif defined(CFG_USB_BRGCLK)
1220
Wolfgang Denke2601822006-06-14 18:14:56 +02001221 /* This has been tested with brgclk == 50MHz */
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001222 int divisor = 0;
1223
Wolfgang Denke2601822006-06-14 18:14:56 +02001224 if (gd->cpu_clk < 48000000L) {
1225 ERR ("brgclk is too slow for full-speed USB!\n");
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001226 udc_state = STATE_ERROR;
1227 return;
1228 }
1229
1230 /* Assume the brgclk is 'good enough', we want !(gd->cpu_clk%48Mhz)
1231 * but, can /probably/ live with close-ish alternative rates.
Wolfgang Denke2601822006-06-14 18:14:56 +02001232 */
1233 divisor = (gd->cpu_clk / 48000000L) - 1;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001234 cp->cp_sicr &= ~0x0000003F;
Wolfgang Denke2601822006-06-14 18:14:56 +02001235
1236 switch (CFG_USB_BRGCLK) {
1237 case 1:
1238 cp->cp_brgc1 |= (divisor | CPM_BRG_EN);
1239 cp->cp_sicr &= ~0x2F;
1240 break;
1241 case 2:
1242 cp->cp_brgc2 |= (divisor | CPM_BRG_EN);
1243 cp->cp_sicr |= 0x00000009;
1244 break;
1245 case 3:
1246 cp->cp_brgc3 |= (divisor | CPM_BRG_EN);
1247 cp->cp_sicr |= 0x00000012;
1248 break;
1249 case 4:
1250 cp->cp_brgc4 = (divisor | CPM_BRG_EN);
1251 cp->cp_sicr |= 0x0000001B;
1252 break;
1253 default:
1254 udc_state = STATE_ERROR;
1255 break;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001256 }
1257
1258#else
Wolfgang Denke2601822006-06-14 18:14:56 +02001259#error "CFG_USB_EXTC_CLK or CFG_USB_BRGCLK must be defined"
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001260#endif
1261
1262}
1263
1264/* mpc8xx_udc_cbd_attach
1265 *
1266 * attach a cbd to and endpoint
1267 */
1268static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size)
1269{
Wolfgang Denke2601822006-06-14 18:14:56 +02001270
1271 if (!tx_cbd[ep] || !rx_cbd[ep] || ep >= MAX_ENDPOINTS) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001272 udc_state = STATE_ERROR;
1273 return;
1274 }
1275
Wolfgang Denke2601822006-06-14 18:14:56 +02001276 if (tx_size > USB_MAX_PKT || rx_size > USB_MAX_PKT ||
1277 (!tx_size && !rx_size)) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001278 udc_state = STATE_ERROR;
1279 return;
1280 }
1281
1282 /* Attach CBD to appropiate Parameter RAM Endpoint data structure */
Wolfgang Denke2601822006-06-14 18:14:56 +02001283 if (rx_size) {
1284 endpoints[ep]->rbase = (u32) rx_cbd[rx_ct];
1285 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001286 rx_ct++;
1287
Wolfgang Denke2601822006-06-14 18:14:56 +02001288 if (!ep) {
1289
1290 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001291 rx_cbd[rx_ct]->cbd_sc |= RX_BD_W;
1292 rx_ct++;
1293
Wolfgang Denke2601822006-06-14 18:14:56 +02001294 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001295 rx_ct += 2;
Wolfgang Denke2601822006-06-14 18:14:56 +02001296 endpoints[ep]->rbptr = (u32) rx_cbd[rx_ct];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001297 rx_cbd[rx_ct]->cbd_sc |= RX_BD_W;
1298 rx_ct++;
1299 }
1300
1301 /* Where we expect to RX data on this endpoint */
Wolfgang Denke2601822006-06-14 18:14:56 +02001302 ep_ref[ep].prx = rx_cbd[rx_ct - 1];
1303 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001304
1305 ep_ref[ep].prx = 0;
1306 endpoints[ep]->rbase = 0;
1307 endpoints[ep]->rbptr = 0;
1308 }
1309
Wolfgang Denke2601822006-06-14 18:14:56 +02001310 if (tx_size) {
1311 endpoints[ep]->tbase = (u32) tx_cbd[tx_ct];
1312 endpoints[ep]->tbptr = (u32) tx_cbd[tx_ct];
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001313 tx_ct++;
Wolfgang Denke2601822006-06-14 18:14:56 +02001314 } else {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001315 endpoints[ep]->tbase = 0;
1316 endpoints[ep]->tbptr = 0;
1317 }
1318
1319 endpoints[ep]->tstate = 0;
1320 endpoints[ep]->tbcnt = 0;
1321 endpoints[ep]->mrblr = EP_MAX_PKT;
Wolfgang Denke2601822006-06-14 18:14:56 +02001322 endpoints[ep]->rfcr = 0x18;
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001323 endpoints[ep]->tfcr = 0x18;
1324 ep_ref[ep].sc |= EP_ATTACHED;
1325
Wolfgang Denke2601822006-06-14 18:14:56 +02001326 DBG ("ep %d rbase 0x%08x rbptr 0x%08x tbase 0x%08x tbptr 0x%08x prx = %p\n",
1327 ep, endpoints[ep]->rbase, endpoints[ep]->rbptr,
1328 endpoints[ep]->tbase, endpoints[ep]->tbptr,
1329 ep_ref[ep].prx);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001330
1331 return;
1332}
1333
1334/* mpc8xx_udc_cbd_init
1335 *
1336 * Allocate space for a cbd and allocate TX/RX data space
1337 */
1338static void mpc8xx_udc_cbd_init (void)
1339{
1340 int i = 0;
1341
Wolfgang Denke2601822006-06-14 18:14:56 +02001342 for (; i < TX_RING_SIZE; i++) {
1343 tx_cbd[i] = (cbd_t *)
1344 mpc8xx_udc_alloc (sizeof (cbd_t), sizeof (int));
1345 }
1346
1347 for (i = 0; i < RX_RING_SIZE; i++) {
1348 rx_cbd[i] = (cbd_t *)
1349 mpc8xx_udc_alloc (sizeof (cbd_t), sizeof (int));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001350 }
1351
Wolfgang Denke2601822006-06-14 18:14:56 +02001352 for (i = 0; i < TX_RING_SIZE; i++) {
1353 tx_cbd[i]->cbd_bufaddr =
1354 mpc8xx_udc_alloc (EP_MAX_PKT, sizeof (int));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001355
Wolfgang Denke2601822006-06-14 18:14:56 +02001356 tx_cbd[i]->cbd_sc = (TX_BD_I | TX_BD_W);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001357 tx_cbd[i]->cbd_datlen = 0x0000;
1358 }
1359
1360
Wolfgang Denke2601822006-06-14 18:14:56 +02001361 for (i = 0; i < RX_RING_SIZE; i++) {
1362 rx_cbd[i]->cbd_bufaddr =
1363 mpc8xx_udc_alloc (EP_MAX_PKT, sizeof (int));
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001364 rx_cbd[i]->cbd_sc = (RX_BD_I | RX_BD_E);
1365 rx_cbd[i]->cbd_datlen = 0x0000;
1366
1367 }
1368
1369 return;
1370}
1371
1372/* mpc8xx_udc_endpoint_init
1373 *
1374 * Attach an endpoint to some dpram
1375 */
1376static void mpc8xx_udc_endpoint_init (void)
1377{
1378 int i = 0;
1379
Wolfgang Denke2601822006-06-14 18:14:56 +02001380 for (; i < MAX_ENDPOINTS; i++) {
1381 endpoints[i] = (usb_epb_t *)
1382 mpc8xx_udc_alloc (sizeof (usb_epb_t), 32);
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001383 }
1384}
1385
1386/* mpc8xx_udc_alloc
1387 *
Wolfgang Denke2601822006-06-14 18:14:56 +02001388 * Grab the address of some dpram
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001389 */
1390static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment)
1391{
1392 u32 retaddr = address_base;
Wolfgang Denke2601822006-06-14 18:14:56 +02001393
1394 while (retaddr % alignment) {
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001395 retaddr++;
Wolfgang Denke2601822006-06-14 18:14:56 +02001396 }
1397 address_base += data_size;
1398
Wolfgang Denk3f0137b2006-06-14 17:45:53 +02001399 return retaddr;
1400}
1401
1402#endif /* CONFIG_MPC885_FAMILY && CONFIG_USB_DEVICE) */