uboot-mt623x/drivers/usbdcore_mpc8xx.c

1413 lines
33 KiB
C

/*
* 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) */