| /* |
| * Copyright (C) 2006 by Bryan O'Donoghue, CodeHermit |
| * bodonoghue@CodeHermit.ie |
| * |
| * References |
| * DasUBoot/drivers/usbdcore_omap1510.c, for design and implementation ideas. |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License as published by |
| * the Free Software Foundation; either version 2 of the License, or |
| * (at your option) any later version. |
| * |
| * This program is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| * GNU General Public License for more details. |
| * |
| * You should have received a copy of the GNU General Public License |
| * along with this program; if not, write to the |
| * Free Software Foundation, Inc., |
| * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. |
| * |
| */ |
| |
| /* |
| * Notes : |
| * 1. #define __SIMULATE_ERROR__ to inject a CRC error into every 2nd TX |
| * packet to force the USB re-transmit protocol. |
| * |
| * 2. #define __DEBUG_UDC__ to switch on debug tracing to serial console |
| * be careful that tracing doesn't create Hiesen-bugs with respect to |
| * response timeouts to control requests. |
| * |
| * 3. This driver should be able to support any higher level driver that |
| * that wants to do either of the two standard UDC implementations |
| * Control-Bulk-Interrupt or Bulk-IN/Bulk-Out standards. Hence |
| * gserial and cdc_acm should work with this code. |
| * |
| * 4. NAK events never actually get raised at all, the documentation |
| * is just wrong ! |
| * |
| * 5. For some reason, cbd_datlen is *always* +2 the value it should be. |
| * this means that having an RX cbd of 16 bytes is not possible, since |
| * the same size is reported for 14 bytes received as 16 bytes received |
| * until we can find out why this happens, RX cbds must be limited to 8 |
| * bytes. TODO: check errata for this behaviour. |
| * |
| * 6. Right now this code doesn't support properly powering up with the USB |
| * cable attached to the USB host my development board the Adder87x doesn't |
| * have a pull-up fitted to allow this, so it is necessary to power the |
| * board and *then* attached the USB cable to the host. However somebody |
| * with a different design in their board may be able to keep the cable |
| * constantly connected and simply enable/disable a pull-up re |
| * figure 31.1 in MPC885RM.pdf instead of having to power up the board and |
| * then attach the cable ! |
| * |
| */ |
| #include <common.h> |
| #include <config.h> |
| |
| #if defined(CONFIG_MPC885_FAMILY) && defined(CONFIG_USB_DEVICE) |
| #include <commproc.h> |
| #include "usbdcore.h" |
| #include "usbdcore_mpc8xx.h" |
| #include "usbdcore_ep0.h" |
| |
| #define ERR(fmt, args...)\ |
| serial_printf("ERROR : [%s] %s:%d: "fmt,\ |
| __FILE__,__FUNCTION__,__LINE__, ##args) |
| #ifdef __DEBUG_UDC__ |
| #define DBG(fmt,args...)\ |
| serial_printf("[%s] %s:%d: "fmt,\ |
| __FILE__,__FUNCTION__,__LINE__, ##args) |
| #else |
| #define DBG(fmt,args...) |
| #endif |
| |
| /* Static Data */ |
| #ifdef __SIMULATE_ERROR__ |
| static char err_poison_test = 0; |
| #endif |
| static struct mpc8xx_ep ep_ref[MAX_ENDPOINTS]; |
| static u32 address_base = STATE_NOT_READY; |
| static mpc8xx_udc_state_t udc_state = 0; |
| static struct usb_device_instance *udc_device = 0; |
| static volatile usb_epb_t *endpoints[MAX_ENDPOINTS]; |
| static volatile cbd_t * tx_cbd[TX_RING_SIZE]; |
| static volatile cbd_t * rx_cbd[RX_RING_SIZE]; |
| static volatile immap_t *immr = 0; |
| static volatile cpm8xx_t *cp = 0; |
| static volatile usb_pram_t *usb_paramp = 0; |
| static volatile usb_t *usbp = 0; |
| static int rx_ct = 0; |
| static int tx_ct = 0; |
| |
| /* Static Function Declarations */ |
| static void mpc8xx_udc_state_transition_up (usb_device_state_t initial, |
| usb_device_state_t final); |
| static void mpc8xx_udc_state_transition_down (usb_device_state_t initial, |
| usb_device_state_t final); |
| static void mpc8xx_udc_stall (unsigned int ep); |
| static void mpc8xx_udc_flush_tx_fifo(int epid); |
| static void mpc8xx_udc_flush_rx_fifo(void); |
| static void mpc8xx_udc_clear_rxbd (volatile cbd_t * rx_cbdp); |
| static void mpc8xx_udc_init_tx(struct usb_endpoint_instance *epi, |
| struct urb * tx_urb); |
| static void mpc8xx_udc_dump_request(struct usb_device_request *request); |
| static void mpc8xx_udc_clock_init (volatile immap_t * immr, |
| volatile cpm8xx_t * cp); |
| static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi); |
| static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp); |
| static void mpc8xx_udc_ep0_rx(volatile cbd_t * rx_cbdp); |
| static void mpc8xx_udc_cbd_init (void); |
| static void mpc8xx_udc_endpoint_init (void); |
| static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size); |
| static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment); |
| static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp); |
| static void mpc8xx_udc_set_nak (unsigned int ep); |
| static short mpc8xx_udc_handle_txerr(void); |
| static void mpc8xx_udc_advance_rx(volatile cbd_t ** rx_cbdp, int epid); |
| |
| /****************************************************************************** |
| Global Linkage |
| *****************************************************************************/ |
| |
| /* udc_init |
| * |
| * Do initial bus gluing |
| */ |
| int udc_init(void) |
| { |
| /* Init various pointers */ |
| immr = (immap_t *) CFG_IMMR; |
| cp = (cpm8xx_t *)&(immr->im_cpm); |
| usb_paramp = (usb_pram_t*)&(cp->cp_dparam[PROFF_USB]); |
| usbp = (usb_t *) &(cp->cp_scc[0]); |
| |
| memset(ep_ref, 0x00, (sizeof(struct mpc8xx_ep) * MAX_ENDPOINTS)); |
| |
| udc_device = 0; |
| udc_state = STATE_NOT_READY; |
| |
| usbp->usmod= 0x00; |
| usbp->uscom= 0; |
| |
| /* Set USB Frame #0, Respond at Address & Get a clock source */ |
| usbp->usaddr = 0x00; |
| mpc8xx_udc_clock_init (immr, cp); |
| |
| /* PA15, PA14 as perhiperal USBRXD and USBOE */ |
| immr->im_ioport.iop_padir&= ~0x0003; |
| immr->im_ioport.iop_papar|= 0x0003; |
| |
| /* PC11/PC10 as peripheral USBRXP USBRXN */ |
| immr->im_ioport.iop_pcso|= 0x0030; |
| |
| /* PC7/PC6 as perhiperal USBTXP and USBTXN */ |
| immr->im_ioport.iop_pcdir|= 0x0300; |
| immr->im_ioport.iop_pcpar|= 0x0300; |
| |
| /* Set the base address */ |
| address_base = (u32)(cp->cp_dpmem + CPM_USB_BASE); |
| |
| /* Initialise endpoints and circular buffers */ |
| mpc8xx_udc_endpoint_init(); |
| mpc8xx_udc_cbd_init(); |
| |
| /* Assign allocated Dual Port Endpoint descriptors */ |
| usb_paramp->ep0ptr = (u32)endpoints[0]; |
| usb_paramp->ep1ptr = (u32)endpoints[1]; |
| usb_paramp->ep2ptr = (u32)endpoints[2]; |
| usb_paramp->ep3ptr = (u32)endpoints[3]; |
| usb_paramp->frame_n = 0; |
| |
| DBG("ep0ptr=0x%08x ep1ptr=0x%08x ep2ptr=0x%08x ep3ptr=0x%08x\n", |
| usb_paramp->ep0ptr, usb_paramp->ep1ptr, usb_paramp->ep2ptr, |
| usb_paramp->ep3ptr); |
| |
| return 0; |
| } |
| |
| /* udc_irq |
| * |
| * Poll for whatever events may have occured |
| */ |
| void udc_irq(void) |
| { |
| int epid = 0; |
| volatile cbd_t * rx_cbdp = 0; |
| volatile cbd_t * rx_cbdp_base = 0; |
| |
| if(udc_state!=STATE_READY){ |
| return; |
| } |
| |
| if(usbp->usber&USB_E_BSY){ |
| /* This shouldn't happen. If it does then it's a bug ! */ |
| usbp->usber|=USB_E_BSY; |
| mpc8xx_udc_flush_rx_fifo(); |
| } |
| |
| |
| /* Scan all RX/Bidirectional Endpoints for RX data. */ |
| for(epid = 0; epid<MAX_ENDPOINTS; epid++){ |
| |
| if(!ep_ref[epid].prx){ |
| continue; |
| } |
| |
| rx_cbdp = rx_cbdp_base = ep_ref[epid].prx; |
| do{ |
| if(!(rx_cbdp->cbd_sc&RX_BD_E)){ |
| |
| if(rx_cbdp->cbd_sc&0x1F){ |
| /* Corrupt data discard it. |
| * Controller has NAK'd this packet. |
| */ |
| mpc8xx_udc_clear_rxbd(rx_cbdp); |
| |
| }else{ |
| if(!epid){ |
| mpc8xx_udc_ep0_rx(rx_cbdp); |
| |
| }else{ |
| /* Process data */ |
| mpc8xx_udc_set_nak(epid); |
| mpc8xx_udc_epn_rx(epid,rx_cbdp); |
| mpc8xx_udc_clear_rxbd(rx_cbdp); |
| } |
| } |
| |
| /* Advance RX CBD pointer */ |
| mpc8xx_udc_advance_rx(&rx_cbdp, epid); |
| ep_ref[epid].prx = rx_cbdp; |
| }else{ |
| /* Advance RX CBD pointer */ |
| mpc8xx_udc_advance_rx(&rx_cbdp, epid); |
| } |
| |
| }while(rx_cbdp != rx_cbdp_base); |
| } |
| |
| /* Handle TX events as appropiate, the correct place to do this is |
| * in a tx routine. Perhaps TX on epn was pre-empted by ep0 |
| */ |
| |
| if(usbp->usber&USB_E_TXB){ |
| usbp->usber|=USB_E_TXB; |
| } |
| |
| if(usbp->usber&(USB_TX_ERRMASK)){ |
| mpc8xx_udc_handle_txerr(); |
| } |
| |
| /* Switch to the default state, respond at the default address */ |
| if(usbp->usber&USB_E_RESET){ |
| usbp->usber|=USB_E_RESET; |
| usbp->usaddr = 0x00; |
| udc_device->device_state = STATE_DEFAULT; |
| } |
| |
| /*if(usbp->usber&USB_E_IDLE){ |
| We could suspend here ! |
| usbp->usber|=USB_E_IDLE; |
| DBG("idle state change\n"); |
| } |
| if(usbp->usbs){ |
| We could resume here when IDLE is deasserted ! |
| Not worth doing, so long as we are self powered though. |
| }*/ |
| |
| return; |
| } |
| |
| |
| |
| /* udc_endpoint_write |
| * |
| * Write some data to an endpoint |
| */ |
| int udc_endpoint_write(struct usb_endpoint_instance *epi) |
| { |
| int ep = 0; |
| short epid = 1, unnak = 0, ret = 0; |
| |
| if(udc_state != STATE_READY){ |
| ERR("invalid udc_state != STATE_READY!\n"); |
| return -1; |
| } |
| |
| if(!udc_device || !epi){ |
| return -1; |
| } |
| |
| if(udc_device->device_state!=STATE_CONFIGURED){ |
| return -1; |
| } |
| |
| ep = epi->endpoint_address & 0x03; |
| if(ep >= MAX_ENDPOINTS){ |
| return -1; |
| } |
| |
| /* Set NAK for all RX endpoints during TX */ |
| for(epid = 1; epid<MAX_ENDPOINTS; epid++){ |
| |
| /* Don't set NAK on DATA IN/CONTROL endpoints */ |
| if(ep_ref[epid].sc & USB_DIR_IN){ |
| continue; |
| } |
| |
| if(!(usbp->usep[epid]&( USEP_THS_NAK | USEP_RHS_NAK ))){ |
| unnak |= 1<<epid; |
| } |
| |
| mpc8xx_udc_set_nak(epid); |
| } |
| |
| mpc8xx_udc_init_tx(&udc_device->bus->endpoint_array[ep],epi->tx_urb); |
| ret = mpc8xx_udc_ep_tx(&udc_device->bus->endpoint_array[ep]); |
| |
| /* Remove temporary NAK */ |
| for(epid = 1; epid<MAX_ENDPOINTS; epid++){ |
| if(unnak&(1<<epid)){ |
| udc_unset_nak(epid); |
| } |
| } |
| |
| return ret; |
| } |
| |
| /* mpc8xx_udc_assign_urb |
| * |
| * Associate a given urb to an endpoint TX or RX transmit/receive buffers |
| */ |
| static int mpc8xx_udc_assign_urb(int ep, char direction) |
| { |
| struct usb_endpoint_instance *epi = 0; |
| |
| if(ep >= MAX_ENDPOINTS){ |
| goto err; |
| } |
| epi = &udc_device->bus->endpoint_array[ep]; |
| if(!epi){ |
| goto err; |
| } |
| |
| if(!ep_ref[ep].urb){ |
| ep_ref[ep].urb = usbd_alloc_urb(udc_device, |
| udc_device->bus->endpoint_array); |
| if(!ep_ref[ep].urb){ |
| goto err; |
| } |
| }else{ |
| ep_ref[ep].urb->actual_length = 0; |
| } |
| |
| switch(direction){ |
| case USB_DIR_IN: |
| epi->tx_urb = ep_ref[ep].urb; |
| break; |
| case USB_DIR_OUT: |
| epi->rcv_urb = ep_ref[ep].urb; |
| break; |
| default: |
| goto err; |
| } |
| return 0; |
| |
| err: |
| udc_state = STATE_ERROR; |
| return -1; |
| } |
| |
| /* udc_setup_ep |
| * |
| * Associate U-Boot software endpoints to mpc8xx endpoint parameter ram |
| * Isochronous endpoints aren't yet supported! |
| */ |
| void udc_setup_ep(struct usb_device_instance *device, unsigned int ep, |
| struct usb_endpoint_instance *epi) |
| { |
| uchar direction = 0; |
| int ep_attrib = 0; |
| |
| if(epi && (ep < MAX_ENDPOINTS)){ |
| |
| if(ep == 0){ |
| if (epi->rcv_attributes!=USB_ENDPOINT_XFER_CONTROL |
| ||epi->tx_attributes!= |
| USB_ENDPOINT_XFER_CONTROL){ |
| |
| /* ep0 must be a control endpoint*/ |
| udc_state = STATE_ERROR; |
| return; |
| |
| } |
| if(!(ep_ref[ep].sc & EP_ATTACHED)){ |
| mpc8xx_udc_cbd_attach (ep, epi->tx_packetSize, |
| epi->rcv_packetSize); |
| } |
| usbp->usep[ep] = 0x0000; |
| return; |
| } |
| |
| if ((epi->endpoint_address & USB_ENDPOINT_DIR_MASK) |
| == USB_DIR_IN) { |
| |
| direction = 1; |
| ep_attrib = epi->tx_attributes; |
| epi->rcv_packetSize = 0; |
| ep_ref[ep].sc |= USB_DIR_IN; |
| } else { |
| |
| direction = 0; |
| ep_attrib = epi->rcv_attributes; |
| epi->tx_packetSize = 0; |
| ep_ref[ep].sc &= ~USB_DIR_IN; |
| } |
| |
| if(mpc8xx_udc_assign_urb(ep, epi->endpoint_address |
| &USB_ENDPOINT_DIR_MASK)){ |
| return; |
| } |
| |
| switch(ep_attrib){ |
| case USB_ENDPOINT_XFER_CONTROL: |
| if(!(ep_ref[ep].sc & EP_ATTACHED)){ |
| mpc8xx_udc_cbd_attach (ep, |
| epi->tx_packetSize, |
| epi->rcv_packetSize); |
| } |
| usbp->usep[ep] = ep<<12; |
| epi->rcv_urb = epi->tx_urb = ep_ref[ep].urb; |
| |
| break; |
| case USB_ENDPOINT_XFER_BULK : |
| case USB_ENDPOINT_XFER_INT: |
| if(!(ep_ref[ep].sc & EP_ATTACHED)){ |
| if(direction){ |
| mpc8xx_udc_cbd_attach (ep, |
| epi->tx_packetSize, 0); |
| }else{ |
| mpc8xx_udc_cbd_attach (ep, |
| 0, epi->rcv_packetSize); |
| } |
| } |
| usbp->usep[ep]= (ep<<12)|((ep_attrib)<<8); |
| |
| break; |
| case USB_ENDPOINT_XFER_ISOC: |
| default: |
| serial_printf("Error endpoint attrib %d>3\n", |
| ep_attrib); |
| udc_state = STATE_ERROR; |
| break; |
| } |
| } |
| |
| } |
| |
| /* udc_connect |
| * |
| * Move state, switch on the USB |
| */ |
| void udc_connect(void) |
| { |
| /* Enable pull-up resistor on D+ |
| * TODO: fit a pull-up resistor to drive SE0 for > 2.5us |
| */ |
| |
| if(udc_state!=STATE_ERROR){ |
| udc_state = STATE_READY; |
| usbp->usmod|= USMOD_EN; |
| } |
| } |
| |
| /* udc_disconnect |
| * |
| * Disconnect is not used but, is included for completeness |
| */ |
| void udc_disconnect(void) |
| { |
| /* Disable pull-up resistor on D- |
| * TODO: fix a pullup resistor to control this |
| */ |
| |
| if(udc_state!=STATE_ERROR){ |
| udc_state = STATE_NOT_READY; |
| } |
| usbp->usmod&=~USMOD_EN; |
| } |
| |
| /* udc_enable |
| * |
| * Grab an EP0 URB, register interest in a subset of USB events |
| */ |
| void udc_enable(struct usb_device_instance *device) |
| { |
| if(udc_state == STATE_ERROR){ |
| return; |
| } |
| |
| udc_device = device; |
| |
| if(!ep_ref[0].urb){ |
| ep_ref[0].urb = usbd_alloc_urb(device, |
| device->bus->endpoint_array); |
| } |
| |
| /* Register interest in all events except SOF, enable transceiver */ |
| usbp->usber= 0x03FF; |
| usbp->usbmr= 0x02F7; |
| |
| return; |
| } |
| |
| /* udc_disable |
| * |
| * disable the currently hooked device |
| */ |
| void udc_disable(void) |
| { |
| int i = 0; |
| |
| if(udc_state == STATE_ERROR){ |
| DBG("Won't disable UDC. udc_state==STATE_ERROR !\n"); |
| return; |
| } |
| |
| udc_device = 0; |
| |
| for(;i<MAX_ENDPOINTS; i++){ |
| if(ep_ref[i].urb){ |
| usbd_dealloc_urb(ep_ref[i].urb); |
| ep_ref[i].urb = 0; |
| } |
| } |
| |
| usbp->usbmr= 0x00; |
| usbp->usmod= ~USMOD_EN; |
| udc_state = STATE_NOT_READY; |
| } |
| |
| /* udc_startup_events |
| * |
| * Enable the specified device |
| */ |
| void udc_startup_events(struct usb_device_instance *device) |
| { |
| udc_enable(device); |
| if(udc_state == STATE_READY){ |
| usbd_device_event_irq (device, DEVICE_CREATE, 0); |
| } |
| } |
| |
| /* udc_set_nak |
| * |
| * Allow upper layers to signal lower layers should not accept more RX data |
| * |
| */ |
| void udc_set_nak(int epid) |
| { |
| if(epid){ |
| mpc8xx_udc_set_nak(epid); |
| } |
| } |
| |
| /* udc_unset_nak |
| * |
| * Suspend sending of NAK tokens for DATA OUT tokens on a given endpoint. |
| * Switch off NAKing on this endpoint to accept more data output from host. |
| * |
| */ |
| void udc_unset_nak (int epid) |
| { |
| if(epid > MAX_ENDPOINTS){ |
| return; |
| } |
| |
| if(usbp->usep[epid]&(USEP_THS_NAK | USEP_RHS_NAK)){ |
| usbp->usep[epid]&= ~(USEP_THS_NAK | USEP_RHS_NAK); |
| __asm__ ("eieio"); |
| } |
| } |
| |
| /****************************************************************************** |
| Static Linkage |
| ******************************************************************************/ |
| |
| /* udc_state_transition_up |
| * udc_state_transition_down |
| * |
| * Helper functions to implement device state changes. The device states and |
| * the events that transition between them are: |
| * |
| * STATE_ATTACHED |
| * || /\ |
| * \/ || |
| * DEVICE_HUB_CONFIGURED DEVICE_HUB_RESET |
| * || /\ |
| * \/ || |
| * STATE_POWERED |
| * || /\ |
| * \/ || |
| * DEVICE_RESET DEVICE_POWER_INTERRUPTION |
| * || /\ |
| * \/ || |
| * STATE_DEFAULT |
| * || /\ |
| * \/ || |
| * DEVICE_ADDRESS_ASSIGNED DEVICE_RESET |
| * || /\ |
| * \/ || |
| * STATE_ADDRESSED |
| * || /\ |
| * \/ || |
| * DEVICE_CONFIGURED DEVICE_DE_CONFIGURED |
| * || /\ |
| * \/ || |
| * STATE_CONFIGURED |
| * |
| * udc_state_transition_up transitions up (in the direction from STATE_ATTACHED |
| * to STATE_CONFIGURED) from the specified initial state to the specified final |
| * state, passing through each intermediate state on the way. If the initial |
| * state is at or above (i.e. nearer to STATE_CONFIGURED) the final state, then |
| * no state transitions will take place. |
| * |
| * udc_state_transition_down transitions down (in the direction from |
| * STATE_CONFIGURED to STATE_ATTACHED) from the specified initial state to the |
| * specified final state, passing through each intermediate state on the way. |
| * If the initial state is at or below (i.e. nearer to STATE_ATTACHED) the final |
| * state, then no state transitions will take place. |
| * |
| */ |
| |
| static void mpc8xx_udc_state_transition_up (usb_device_state_t initial, |
| usb_device_state_t final) |
| { |
| if (initial < final) { |
| switch (initial) { |
| case STATE_ATTACHED: |
| usbd_device_event_irq (udc_device, |
| DEVICE_HUB_CONFIGURED, 0); |
| if (final == STATE_POWERED) |
| break; |
| case STATE_POWERED: |
| usbd_device_event_irq (udc_device, DEVICE_RESET, 0); |
| if (final == STATE_DEFAULT) |
| break; |
| case STATE_DEFAULT: |
| usbd_device_event_irq (udc_device, |
| DEVICE_ADDRESS_ASSIGNED, 0); |
| if (final == STATE_ADDRESSED) |
| break; |
| case STATE_ADDRESSED: |
| usbd_device_event_irq (udc_device, DEVICE_CONFIGURED, |
| 0); |
| case STATE_CONFIGURED: |
| break; |
| default: |
| break; |
| } |
| } |
| } |
| |
| static void mpc8xx_udc_state_transition_down (usb_device_state_t initial, |
| usb_device_state_t final) |
| { |
| if (initial > final) { |
| switch (initial) { |
| case STATE_CONFIGURED: |
| usbd_device_event_irq (udc_device, |
| DEVICE_DE_CONFIGURED, 0); |
| if (final == STATE_ADDRESSED) |
| break; |
| case STATE_ADDRESSED: |
| usbd_device_event_irq (udc_device, DEVICE_RESET, 0); |
| if (final == STATE_DEFAULT) |
| break; |
| case STATE_DEFAULT: |
| usbd_device_event_irq (udc_device, |
| DEVICE_POWER_INTERRUPTION, 0); |
| if (final == STATE_POWERED) |
| break; |
| case STATE_POWERED: |
| usbd_device_event_irq (udc_device, DEVICE_HUB_RESET, |
| 0); |
| case STATE_ATTACHED: |
| break; |
| default: |
| break; |
| } |
| } |
| } |
| |
| /* mpc8xx_udc_stall |
| * |
| * Force returning of STALL tokens on the given endpoint. Protocol or function |
| * STALL conditions are permissable here |
| */ |
| static void mpc8xx_udc_stall (unsigned int ep) |
| { |
| usbp->usep[ep] |= STALL_BITMASK; |
| } |
| |
| /* mpc8xx_udc_set_nak |
| * |
| * Force returning of NAK responses for the given endpoint as a kind of very |
| * simple flow control |
| */ |
| static void mpc8xx_udc_set_nak (unsigned int ep) |
| { |
| usbp->usep[ep] |= NAK_BITMASK; |
| __asm__ ("eieio"); |
| } |
| |
| /* mpc8xx_udc_handle_txerr |
| * |
| * Handle errors relevant to TX. Return a status code to allow calling |
| * indicative of what if anything happened |
| */ |
| static short mpc8xx_udc_handle_txerr() |
| { |
| short ep = 0, ret = 0; |
| |
| for(; ep<TX_RING_SIZE; ep++){ |
| if(usbp->usber&(0x10<<ep)){ |
| |
| /* Timeout or underrun */ |
| if(tx_cbd[ep]->cbd_sc&0x06){ |
| ret = 1; |
| mpc8xx_udc_flush_tx_fifo(ep); |
| |
| }else{ |
| if(usbp->usep[ep]&STALL_BITMASK){ |
| if(!ep){ |
| usbp->usep[ep]&= |
| ~STALL_BITMASK; |
| } |
| }/* else NAK */ |
| } |
| usbp->usber|=(0x10<<ep); |
| } |
| } |
| return ret; |
| } |
| |
| /* mpc8xx_udc_advance_rx |
| * |
| * Advance cbd rx |
| */ |
| static void mpc8xx_udc_advance_rx(volatile cbd_t ** rx_cbdp, int epid) |
| { |
| if((*rx_cbdp)->cbd_sc & RX_BD_W){ |
| *rx_cbdp = (volatile cbd_t*) |
| (endpoints[epid]->rbase + CFG_IMMR); |
| |
| }else{ |
| (*rx_cbdp)++; |
| } |
| } |
| |
| |
| /* mpc8xx_udc_flush_tx_fifo |
| * |
| * Flush a given TX fifo. Assumes one tx cbd per endpoint |
| */ |
| static void mpc8xx_udc_flush_tx_fifo(int epid) |
| { |
| volatile cbd_t * tx_cbdp = 0; |
| |
| if(epid > MAX_ENDPOINTS){ |
| return; |
| } |
| |
| /* TX stop */ |
| immr->im_cpm.cp_cpcr = ((epid<<2) | 0x1D01); |
| __asm__ ("eieio"); |
| while(immr->im_cpm.cp_cpcr & 0x01); |
| |
| usbp->uscom = 0x40 | 0; |
| |
| /* reset ring */ |
| tx_cbdp = (cbd_t*)(endpoints[epid]->tbptr + CFG_IMMR); |
| tx_cbdp->cbd_sc = (TX_BD_I | TX_BD_W); |
| |
| |
| endpoints[epid]->tptr = endpoints[epid]->tbase; |
| endpoints[epid]->tstate = 0x00; |
| endpoints[epid]->tbcnt = 0x00; |
| |
| /* TX start */ |
| immr->im_cpm.cp_cpcr = ((epid<<2) | 0x2D01); |
| __asm__ ("eieio"); |
| while(immr->im_cpm.cp_cpcr & 0x01); |
| |
| return; |
| } |
| |
| /* mpc8xx_udc_flush_rx_fifo |
| * |
| * For the sake of completeness of the namespace, it seems like |
| * a good-design-decision (tm) to include mpc8xx_udc_flush_rx_fifo(); |
| * If RX_BD_E is true => a driver bug either here or in an upper layer |
| * not polling frequently enough. If RX_BD_E is true we have told the host |
| * we have accepted data but, the CPM found it had no-where to put that data |
| * which needless to say would be a bad thing. |
| */ |
| static void mpc8xx_udc_flush_rx_fifo() |
| { |
| int i = 0; |
| for(i = 0;i<RX_RING_SIZE; i++){ |
| if(!(rx_cbd[i]->cbd_sc&RX_BD_E)){ |
| ERR("buf %p used rx data len = 0x%x sc=0x%x!\n", |
| rx_cbd[i], rx_cbd[i]->cbd_datlen, |
| rx_cbd[i]->cbd_sc); |
| |
| } |
| } |
| ERR("BUG : Input over-run\n"); |
| } |
| |
| /* mpc8xx_udc_clear_rxbd |
| * |
| * Release control of RX CBD to CP. |
| */ |
| static void mpc8xx_udc_clear_rxbd(volatile cbd_t * rx_cbdp) |
| { |
| rx_cbdp->cbd_datlen = 0x0000; |
| rx_cbdp->cbd_sc= ((rx_cbdp->cbd_sc & RX_BD_W)|(RX_BD_E | RX_BD_I)); |
| __asm__ ("eieio"); |
| } |
| |
| /* mpc8xx_udc_tx_irq |
| * |
| * Parse for tx timeout, control RX or USB reset/busy conditions |
| * Return -1 on timeout, -2 on fatal error, else return zero |
| */ |
| static int mpc8xx_udc_tx_irq(int ep) |
| { |
| int i = 0; |
| |
| if(usbp->usber&(USB_TX_ERRMASK)){ |
| if(mpc8xx_udc_handle_txerr()){ |
| /* Timeout, controlling function must retry send */ |
| return -1; |
| } |
| } |
| |
| if(usbp->usber & (USB_E_RESET|USB_E_BSY)){ |
| /* Fatal, abandon TX transaction */ |
| return -2; |
| } |
| |
| if(usbp->usber & USB_E_RXB){ |
| for(i = 0;i<RX_RING_SIZE; i++){ |
| if(!(rx_cbd[i]->cbd_sc&RX_BD_E)){ |
| if((rx_cbd[i] == ep_ref[0].prx) || ep){ |
| return -2; |
| } |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| /* mpc8xx_udc_ep_tx |
| * |
| * Transmit in a re-entrant fashion outbound USB packets. |
| * Implement retry/timeout mechanism described in USB specification |
| * Toggle DATA0/DATA1 pids as necessary |
| * Introduces non-standard tx_retry. The USB standard has no scope for slave |
| * devices to give up TX, however tx_retry stops us getting stuck in an endless |
| * TX loop. |
| */ |
| static int mpc8xx_udc_ep_tx (struct usb_endpoint_instance *epi) |
| { |
| struct urb *urb = epi->tx_urb; |
| volatile cbd_t * tx_cbdp = 0; |
| unsigned int ep = 0, pkt_len = 0, x = 0, tx_retry = 0; |
| int ret = 0; |
| |
| if(!epi || (epi->endpoint_address&0x03)>=MAX_ENDPOINTS || !urb){ |
| return -1; |
| } |
| |
| ep = epi->endpoint_address & 0x03; |
| tx_cbdp = (cbd_t*)(endpoints[ep]->tbptr + CFG_IMMR); |
| |
| if(tx_cbdp->cbd_sc&TX_BD_R || usbp->usber&USB_E_TXB){ |
| mpc8xx_udc_flush_tx_fifo(ep); |
| usbp->usber |= USB_E_TXB; |
| }; |
| |
| while(tx_retry++ < 100){ |
| ret = mpc8xx_udc_tx_irq(ep); |
| if(ret == -1){ |
| /* ignore timeout here */ |
| }else if(ret == -2){ |
| /* Abandon TX */ |
| mpc8xx_udc_flush_tx_fifo(ep); |
| return -1; |
| } |
| |
| tx_cbdp = (cbd_t*)(endpoints[ep]->tbptr + CFG_IMMR); |
| while(tx_cbdp->cbd_sc&TX_BD_R){}; |
| tx_cbdp->cbd_sc = (tx_cbdp->cbd_sc&TX_BD_W); |
| |
| pkt_len = urb->actual_length - epi->sent; |
| |
| if(pkt_len> epi->tx_packetSize || pkt_len > EP_MAX_PKT){ |
| pkt_len = MIN(epi->tx_packetSize, EP_MAX_PKT); |
| } |
| |
| for(x=0; x<pkt_len; x++){ |
| *((unsigned char*)(tx_cbdp->cbd_bufaddr+x)) = |
| urb->buffer[epi->sent + x]; |
| } |
| tx_cbdp->cbd_datlen = pkt_len; |
| tx_cbdp->cbd_sc|=(CBD_TX_BITMASK | ep_ref[ep].pid); |
| __asm__ ("eieio"); |
| |
| #ifdef __SIMULATE_ERROR__ |
| if(++err_poison_test == 2){ |
| err_poison_test = 0; |
| tx_cbdp->cbd_sc&=~TX_BD_TC; |
| } |
| #endif |
| |
| usbp->uscom = (USCOM_STR | ep); |
| |
| while(!(usbp->usber&USB_E_TXB)){ |
| ret = mpc8xx_udc_tx_irq(ep); |
| if(ret == -1){ |
| /* TX timeout */ |
| break; |
| }else if(ret == -2){ |
| if(usbp->usber & USB_E_TXB){ |
| usbp->usber|=USB_E_TXB; |
| } |
| mpc8xx_udc_flush_tx_fifo(ep); |
| return -1; |
| } |
| }; |
| |
| if(usbp->usber & USB_E_TXB){ |
| usbp->usber|=USB_E_TXB; |
| } |
| |
| /* ACK must be present <= 18bit times from TX */ |
| if(ret == -1){ |
| continue; |
| } |
| |
| /* TX ACK : USB 2.0 8.7.2, Toggle PID, Advance TX */ |
| epi->sent += pkt_len; |
| epi->last = MIN (urb->actual_length - epi->sent, |
| epi->tx_packetSize); |
| TOGGLE_TX_PID(ep_ref[ep].pid); |
| |
| if(epi->sent >= epi->tx_urb->actual_length){ |
| |
| epi->tx_urb->actual_length = 0; |
| epi->sent = 0; |
| |
| if(ep_ref[ep].sc & EP_SEND_ZLP){ |
| ep_ref[ep].sc &= ~EP_SEND_ZLP; |
| }else{ |
| return 0; |
| } |
| } |
| } |
| |
| ERR("TX fail, endpoint 0x%x tx bytes 0x%x/0x%x\n",ep, epi->sent, |
| epi->tx_urb->actual_length); |
| |
| return -1; |
| } |
| |
| /* mpc8xx_udc_dump_request |
| * |
| * Dump a control request to console |
| */ |
| static void mpc8xx_udc_dump_request(struct usb_device_request *request) |
| { |
| DBG( |
| "bmRequestType:%02x bRequest:%02x wValue:%04x " |
| "wIndex:%04x wLength:%04x ?\n", |
| request->bmRequestType, |
| request->bRequest, |
| request->wValue, |
| request->wIndex, |
| request->wLength); |
| |
| return; |
| } |
| |
| /* mpc8xx_udc_ep0_rx_setup |
| * |
| * Decode received ep0 SETUP packet. return non-zero on error |
| */ |
| static int mpc8xx_udc_ep0_rx_setup (volatile cbd_t * rx_cbdp) |
| { |
| unsigned int x = 0; |
| struct urb * purb = ep_ref[0].urb; |
| struct usb_endpoint_instance *epi = |
| &udc_device->bus->endpoint_array[0]; |
| |
| for(; x<rx_cbdp->cbd_datlen; x++){ |
| *(((unsigned char*)&ep_ref[0].urb->device_request)+x) = |
| *((unsigned char*)(rx_cbdp->cbd_bufaddr+x)); |
| } |
| |
| mpc8xx_udc_clear_rxbd(rx_cbdp); |
| |
| if (ep0_recv_setup(purb)) { |
| mpc8xx_udc_dump_request(&purb->device_request); |
| return -1; |
| } |
| |
| if ((purb->device_request.bmRequestType&USB_REQ_DIRECTION_MASK) |
| == USB_REQ_HOST2DEVICE) { |
| |
| switch (purb->device_request.bRequest){ |
| case USB_REQ_SET_ADDRESS: |
| /* Send the Status OUT ZLP */ |
| ep_ref[0].pid = TX_BD_PID_DATA1; |
| purb->actual_length = 0; |
| mpc8xx_udc_init_tx(epi,purb); |
| mpc8xx_udc_ep_tx(epi); |
| |
| /* Move to the addressed state */ |
| usbp->usaddr = udc_device->address; |
| mpc8xx_udc_state_transition_up(udc_device->device_state, |
| STATE_ADDRESSED); |
| return 0; |
| |
| case USB_REQ_SET_CONFIGURATION: |
| if(!purb->device_request.wValue){ |
| |
| /* Respond at default address */ |
| usbp->usaddr = 0x00; |
| mpc8xx_udc_state_transition_down(udc_device->device_state, |
| STATE_ADDRESSED); |
| |
| } else { |
| |
| /* TODO: Support multiple configurations */ |
| mpc8xx_udc_state_transition_up(udc_device->device_state,STATE_CONFIGURED); |
| for(x=1; x<MAX_ENDPOINTS; x++){ |
| if((udc_device->bus->endpoint_array[x].endpoint_address&USB_ENDPOINT_DIR_MASK) |
| == USB_DIR_IN){ |
| ep_ref[x].pid = TX_BD_PID_DATA0; |
| }else{ |
| ep_ref[x].pid = RX_BD_PID_DATA0; |
| } |
| /* Set configuration must unstall endpoints */ |
| usbp->usep[x]&=~STALL_BITMASK; |
| } |
| |
| } |
| break; |
| default: |
| /* CDC/Vendor specific */ |
| break; |
| } |
| |
| /* Send ZLP as ACK in Status OUT phase */ |
| ep_ref[0].pid = TX_BD_PID_DATA1; |
| purb->actual_length = 0; |
| mpc8xx_udc_init_tx(epi,purb); |
| mpc8xx_udc_ep_tx(epi); |
| |
| }else{ |
| if(purb->actual_length){ |
| ep_ref[0].pid = TX_BD_PID_DATA1; |
| mpc8xx_udc_init_tx(epi,purb); |
| |
| if(!(purb->actual_length%EP0_MAX_PACKET_SIZE)){ |
| ep_ref[0].sc |= EP_SEND_ZLP; |
| } |
| |
| if(purb->device_request.wValue== |
| USB_DESCRIPTOR_TYPE_DEVICE){ |
| if(le16_to_cpu(purb->device_request.wLength)> |
| purb->actual_length){ |
| /* Send EP0_MAX_PACKET_SIZE bytes |
| * unless correct size requested. |
| */ |
| if(purb->actual_length > |
| epi->tx_packetSize){ |
| |
| purb->actual_length = |
| epi->tx_packetSize; |
| } |
| |
| } |
| } |
| mpc8xx_udc_ep_tx(epi); |
| |
| }else{ |
| /* Corrupt SETUP packet? */ |
| ERR("Zero length data or SETUP with DATA-IN phase ?\n"); |
| return 1; |
| } |
| } |
| return 0; |
| } |
| |
| /* mpc8xx_udc_init_tx |
| * |
| * Setup some basic parameters for a TX transaction |
| */ |
| static void mpc8xx_udc_init_tx(struct usb_endpoint_instance *epi, |
| struct urb * tx_urb) |
| { |
| epi->sent = 0; |
| epi->last = 0; |
| epi->tx_urb = tx_urb; |
| } |
| |
| /* mpc8xx_udc_ep0_rx |
| * |
| * Receive ep0/control USB data. Parse and possibly send a response. |
| */ |
| static void mpc8xx_udc_ep0_rx(volatile cbd_t * rx_cbdp) |
| { |
| if(rx_cbdp->cbd_sc&RX_BD_PID_SETUP){ |
| |
| /* Unconditionally accept SETUP packets */ |
| if(mpc8xx_udc_ep0_rx_setup(rx_cbdp)){ |
| mpc8xx_udc_stall (0); |
| } |
| |
| } else { |
| |
| mpc8xx_udc_clear_rxbd(rx_cbdp); |
| |
| if((rx_cbdp->cbd_datlen-2)){ |
| /* SETUP with a DATA phase |
| * outside of SETUP packet. |
| * Reply with STALL. |
| */ |
| mpc8xx_udc_stall (0); |
| } |
| } |
| } |
| |
| /* mpc8xx_udc_epn_rx |
| * |
| * Receive some data from cbd into USB system urb data abstraction |
| * Upper layers should NAK if there is insufficient RX data space |
| */ |
| static int mpc8xx_udc_epn_rx (unsigned int epid, volatile cbd_t * rx_cbdp) |
| { |
| struct usb_endpoint_instance *epi = 0; |
| struct urb *urb = 0; |
| unsigned int x = 0; |
| |
| if(epid >= MAX_ENDPOINTS || !rx_cbdp->cbd_datlen){ |
| return 0; |
| } |
| |
| /* USB 2.0 PDF section 8.6.4 |
| * Discard data with invalid PID it is a resend. |
| */ |
| if(ep_ref[epid].pid!=(rx_cbdp->cbd_sc&0xC0)){ |
| return 1; |
| } |
| TOGGLE_RX_PID(ep_ref[epid].pid); |
| |
| epi = &udc_device->bus->endpoint_array[epid]; |
| urb = epi->rcv_urb; |
| |
| for(; x<(rx_cbdp->cbd_datlen-2); x++){ |
| *((unsigned char*)(urb->buffer + urb->actual_length +x)) = |
| *((unsigned char*)(rx_cbdp->cbd_bufaddr+x)); |
| } |
| |
| if(x){ |
| usbd_rcv_complete (epi, x, 0); |
| if(ep_ref[epid].urb->status == RECV_ERROR){ |
| DBG("RX error unset NAK\n"); |
| udc_unset_nak(epid); |
| } |
| } |
| return x; |
| } |
| |
| /* mpc8xx_udc_clock_init |
| * |
| * Obtain a clock reference for Full Speed Signaling |
| */ |
| static void mpc8xx_udc_clock_init (volatile immap_t * immr, |
| volatile cpm8xx_t * cp) |
| { |
| |
| #if defined(CFG_USB_EXTC_CLK) |
| |
| /* This has been tested with a 48MHz crystal on CLK6 */ |
| switch(CFG_USB_EXTC_CLK){ |
| case 1: |
| immr->im_ioport.iop_papar|= 0x0100; |
| immr->im_ioport.iop_padir&= ~0x0100; |
| cp->cp_sicr|= 0x24; |
| break; |
| case 2: |
| immr->im_ioport.iop_papar|= 0x0200; |
| immr->im_ioport.iop_padir&= ~0x0200; |
| cp->cp_sicr|= 0x2D; |
| break; |
| case 3: |
| immr->im_ioport.iop_papar|= 0x0400; |
| immr->im_ioport.iop_padir&= ~0x0400; |
| cp->cp_sicr|= 0x36; |
| break; |
| case 4: |
| immr->im_ioport.iop_papar|= 0x0800; |
| immr->im_ioport.iop_padir&= ~0x0800; |
| cp->cp_sicr|= 0x3F; |
| break; |
| default: |
| udc_state = STATE_ERROR; |
| break; |
| } |
| |
| #elif defined(CFG_USB_BRGCLK) |
| |
| /* This has been tested with brgclk == 50MHz */ |
| DECLARE_GLOBAL_DATA_PTR; |
| int divisor = 0; |
| |
| if(gd->cpu_clk<48000000L){ |
| ERR("brgclk is too slow for full-speed USB!\n"); |
| udc_state = STATE_ERROR; |
| return; |
| } |
| |
| /* Assume the brgclk is 'good enough', we want !(gd->cpu_clk%48Mhz) |
| * but, can /probably/ live with close-ish alternative rates. |
| */ |
| divisor = (gd->cpu_clk/48000000L)-1; |
| cp->cp_sicr &= ~0x0000003F; |
| |
| switch(CFG_USB_BRGCLK){ |
| case 1: |
| cp->cp_brgc1 |= (divisor|CPM_BRG_EN); |
| cp->cp_sicr &= ~0x2F; |
| break; |
| case 2: |
| cp->cp_brgc2 |= (divisor|CPM_BRG_EN); |
| cp->cp_sicr |= 0x00000009; |
| break; |
| case 3: |
| cp->cp_brgc3 |= (divisor|CPM_BRG_EN); |
| cp->cp_sicr |= 0x00000012; |
| break; |
| case 4: |
| cp->cp_brgc4 = (divisor|CPM_BRG_EN); |
| cp->cp_sicr |= 0x0000001B; |
| break; |
| default: |
| udc_state = STATE_ERROR; |
| break; |
| } |
| |
| #else |
| #error "CFG_USB_EXTC_CLK or CFG_USB_BRGCLK must be defined" |
| #endif |
| |
| } |
| |
| /* mpc8xx_udc_cbd_attach |
| * |
| * attach a cbd to and endpoint |
| */ |
| static void mpc8xx_udc_cbd_attach (int ep, uchar tx_size, uchar rx_size) |
| { |
| |
| if (!tx_cbd[ep] || !rx_cbd[ep] || ep >= MAX_ENDPOINTS){ |
| udc_state = STATE_ERROR; |
| return; |
| } |
| |
| if (tx_size>USB_MAX_PKT || rx_size>USB_MAX_PKT || |
| (!tx_size && !rx_size)){ |
| udc_state = STATE_ERROR; |
| return; |
| } |
| |
| /* Attach CBD to appropiate Parameter RAM Endpoint data structure */ |
| if(rx_size){ |
| endpoints[ep]->rbase = (u32)rx_cbd[rx_ct]; |
| endpoints[ep]->rbptr = (u32)rx_cbd[rx_ct]; |
| rx_ct++; |
| |
| if(!ep){ |
| |
| endpoints[ep]->rbptr = (u32)rx_cbd[rx_ct]; |
| rx_cbd[rx_ct]->cbd_sc |= RX_BD_W; |
| rx_ct++; |
| |
| }else{ |
| rx_ct += 2; |
| endpoints[ep]->rbptr = (u32)rx_cbd[rx_ct]; |
| rx_cbd[rx_ct]->cbd_sc |= RX_BD_W; |
| rx_ct++; |
| } |
| |
| /* Where we expect to RX data on this endpoint */ |
| ep_ref[ep].prx = rx_cbd[rx_ct-1]; |
| }else{ |
| |
| ep_ref[ep].prx = 0; |
| endpoints[ep]->rbase = 0; |
| endpoints[ep]->rbptr = 0; |
| } |
| |
| if(tx_size){ |
| endpoints[ep]->tbase = (u32)tx_cbd[tx_ct]; |
| endpoints[ep]->tbptr = (u32)tx_cbd[tx_ct]; |
| tx_ct++; |
| }else{ |
| endpoints[ep]->tbase = 0; |
| endpoints[ep]->tbptr = 0; |
| } |
| |
| endpoints[ep]->tstate = 0; |
| endpoints[ep]->tbcnt = 0; |
| endpoints[ep]->mrblr = EP_MAX_PKT; |
| endpoints[ep]->rfcr = 0x18; |
| endpoints[ep]->tfcr = 0x18; |
| ep_ref[ep].sc |= EP_ATTACHED; |
| |
| DBG("ep %d rbase 0x%08x rbptr 0x%08x tbase 0x%08x tbptr 0x%08x prx = %p\n", |
| ep, endpoints[ep]->rbase, endpoints[ep]->rbptr, endpoints[ep]->tbase, |
| endpoints[ep]->tbptr, ep_ref[ep].prx); |
| |
| return; |
| } |
| |
| /* mpc8xx_udc_cbd_init |
| * |
| * Allocate space for a cbd and allocate TX/RX data space |
| */ |
| static void mpc8xx_udc_cbd_init (void) |
| { |
| int i = 0; |
| |
| for(; i<TX_RING_SIZE; i++){ |
| tx_cbd[i]= (cbd_t*) |
| mpc8xx_udc_alloc(sizeof(cbd_t), sizeof(int)); |
| } |
| |
| for(i=0; i<RX_RING_SIZE; i++){ |
| rx_cbd[i]= (cbd_t*) |
| mpc8xx_udc_alloc(sizeof(cbd_t),sizeof(int)); |
| } |
| |
| for(i=0; i< TX_RING_SIZE; i++){ |
| tx_cbd[i]->cbd_bufaddr = |
| mpc8xx_udc_alloc(EP_MAX_PKT, sizeof(int)); |
| |
| tx_cbd[i]->cbd_sc = (TX_BD_I | TX_BD_W); |
| tx_cbd[i]->cbd_datlen = 0x0000; |
| } |
| |
| |
| for(i=0; i< RX_RING_SIZE; i++){ |
| rx_cbd[i]->cbd_bufaddr = |
| mpc8xx_udc_alloc(EP_MAX_PKT, sizeof(int)); |
| rx_cbd[i]->cbd_sc = (RX_BD_I | RX_BD_E); |
| rx_cbd[i]->cbd_datlen = 0x0000; |
| |
| } |
| |
| return; |
| } |
| |
| /* mpc8xx_udc_endpoint_init |
| * |
| * Attach an endpoint to some dpram |
| */ |
| static void mpc8xx_udc_endpoint_init (void) |
| { |
| int i = 0; |
| |
| for(; i<MAX_ENDPOINTS; i++){ |
| endpoints[i]= (usb_epb_t*) |
| mpc8xx_udc_alloc(sizeof(usb_epb_t) , 32); |
| } |
| } |
| |
| /* mpc8xx_udc_alloc |
| * |
| * Grab the address of some dpram |
| */ |
| static u32 mpc8xx_udc_alloc (u32 data_size, u32 alignment) |
| { |
| u32 retaddr = address_base; |
| |
| while(retaddr%alignment){ |
| retaddr++; |
| } |
| address_base+= data_size; |
| |
| return retaddr; |
| } |
| |
| #endif /* CONFIG_MPC885_FAMILY && CONFIG_USB_DEVICE) */ |