mirror of
				https://source.denx.de/u-boot/u-boot.git
				synced 2025-10-25 22:41:21 +02:00 
			
		
		
		
	SPEAr : usbd driver support for SPEAr SoCs
SPEAr SoCs contain a synopsys usb device controller. USB Device IP can work in 2 modes - DMA mode - Slave mode The driver adds support only for slave mode operation of usb device IP. This driver is used along with standard USBTTY driver to obtain a tty interface over USB on the host Signed-off-by: Vipin <vipin.kumar@st.com>
This commit is contained in:
		
							parent
							
								
									13229557c1
								
							
						
					
					
						commit
						2f11000558
					
				
							
								
								
									
										2
									
								
								drivers/serial/usbtty.h
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										2
									
								
								drivers/serial/usbtty.h
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @ -33,6 +33,8 @@ | ||||
| #include <usb/musb_udc.h> | ||||
| #elif defined(CONFIG_PXA27X) | ||||
| #include <usb/pxa27x_udc.h> | ||||
| #elif defined(CONFIG_SPEAR3XX) || defined(CONFIG_SPEAR600) | ||||
| #include <usb/spr_udc.h> | ||||
| #endif | ||||
| 
 | ||||
| #include <version_autogenerated.h> | ||||
|  | ||||
							
								
								
									
										1
									
								
								drivers/usb/gadget/Makefile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										1
									
								
								drivers/usb/gadget/Makefile
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @ -32,6 +32,7 @@ COBJS-$(CONFIG_OMAP1510) += omap1510_udc.o | ||||
| COBJS-$(CONFIG_OMAP1610) += omap1510_udc.o | ||||
| COBJS-$(CONFIG_MPC885_FAMILY) += mpc8xx_udc.o | ||||
| COBJS-$(CONFIG_PXA27X) += pxa27x_udc.o | ||||
| COBJS-$(CONFIG_SPEARUDC) += spr_udc.o | ||||
| endif | ||||
| 
 | ||||
| COBJS	:= $(COBJS-y) | ||||
|  | ||||
							
								
								
									
										998
									
								
								drivers/usb/gadget/spr_udc.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										998
									
								
								drivers/usb/gadget/spr_udc.c
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,998 @@ | ||||
| /*
 | ||||
|  * Based on drivers/usb/gadget/omap1510_udc.c | ||||
|  * TI OMAP1510 USB bus interface driver | ||||
|  * | ||||
|  * (C) Copyright 2009 | ||||
|  * Vipin Kumar, ST Micoelectronics, vipin.kumar@st.com. | ||||
|  * | ||||
|  * See file CREDITS for list of people who contributed to this | ||||
|  * project. | ||||
|  * | ||||
|  * 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 | ||||
|  */ | ||||
| 
 | ||||
| #include <common.h> | ||||
| #include <asm/io.h> | ||||
| 
 | ||||
| #include <usbdevice.h> | ||||
| #include "ep0.h" | ||||
| #include <usb/spr_udc.h> | ||||
| #include <asm/arch/hardware.h> | ||||
| #include <asm/arch/spr_misc.h> | ||||
| 
 | ||||
| #define UDC_INIT_MDELAY		80	/* Device settle delay */ | ||||
| 
 | ||||
| /* Some kind of debugging output... */ | ||||
| #ifndef DEBUG_SPRUSBTTY | ||||
| #define UDCDBG(str) | ||||
| #define UDCDBGA(fmt, args...) | ||||
| #else | ||||
| #define UDCDBG(str) serial_printf(str "\n") | ||||
| #define UDCDBGA(fmt, args...) serial_printf(fmt "\n", ##args) | ||||
| #endif | ||||
| 
 | ||||
| static struct urb *ep0_urb; | ||||
| static struct usb_device_instance *udc_device; | ||||
| 
 | ||||
| static struct plug_regs *const plug_regs_p = | ||||
|     (struct plug_regs * const)CONFIG_SYS_PLUG_BASE; | ||||
| static struct udc_regs *const udc_regs_p = | ||||
|     (struct udc_regs * const)CONFIG_SYS_USBD_BASE; | ||||
| static struct udc_endp_regs *const outep_regs_p = | ||||
|     &((struct udc_regs * const)CONFIG_SYS_USBD_BASE)->out_regs[0]; | ||||
| static struct udc_endp_regs *const inep_regs_p = | ||||
|     &((struct udc_regs * const)CONFIG_SYS_USBD_BASE)->in_regs[0]; | ||||
| 
 | ||||
| /*
 | ||||
|  * udc_state_transition - Write the next packet to TxFIFO. | ||||
|  * @initial:	Initial state. | ||||
|  * @final:	Final state. | ||||
|  * | ||||
|  * Helper function 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 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 also 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. | ||||
|  * | ||||
|  * This function must only be called with interrupts disabled. | ||||
|  */ | ||||
| static void udc_state_transition(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; | ||||
| 		} | ||||
| 	} else 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; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /* Stall endpoint */ | ||||
| static void udc_stall_ep(u32 ep_num) | ||||
| { | ||||
| 	writel(readl(&inep_regs_p[ep_num].endp_cntl) | ENDP_CNTL_STALL, | ||||
| 	       &inep_regs_p[ep_num].endp_cntl); | ||||
| 
 | ||||
| 	writel(readl(&outep_regs_p[ep_num].endp_cntl) | ENDP_CNTL_STALL, | ||||
| 	       &outep_regs_p[ep_num].endp_cntl); | ||||
| } | ||||
| 
 | ||||
| static void *get_fifo(int ep_num, int in) | ||||
| { | ||||
| 	u32 *fifo_ptr = (u32 *)CONFIG_SYS_FIFO_BASE; | ||||
| 
 | ||||
| 	switch (ep_num) { | ||||
| 	case UDC_EP3: | ||||
| 		fifo_ptr += readl(&inep_regs_p[1].endp_bsorfn); | ||||
| 		/* break intentionally left out */ | ||||
| 
 | ||||
| 	case UDC_EP1: | ||||
| 		fifo_ptr += readl(&inep_regs_p[0].endp_bsorfn); | ||||
| 		/* break intentionally left out */ | ||||
| 
 | ||||
| 	case UDC_EP0: | ||||
| 	default: | ||||
| 		if (in) { | ||||
| 			fifo_ptr += | ||||
| 			    readl(&outep_regs_p[2].endp_maxpacksize) >> 16; | ||||
| 			/* break intentionally left out */ | ||||
| 		} else { | ||||
| 			break; | ||||
| 		} | ||||
| 
 | ||||
| 	case UDC_EP2: | ||||
| 		fifo_ptr += readl(&outep_regs_p[0].endp_maxpacksize) >> 16; | ||||
| 		/* break intentionally left out */ | ||||
| 	} | ||||
| 
 | ||||
| 	return (void *)fifo_ptr; | ||||
| } | ||||
| 
 | ||||
| static int usbgetpckfromfifo(int epNum, u8 *bufp, u32 len) | ||||
| { | ||||
| 	u8 *fifo_ptr = (u8 *)get_fifo(epNum, 0); | ||||
| 	u32 i, nw, nb; | ||||
| 	u32 *wrdp; | ||||
| 	u8 *bytp; | ||||
| 
 | ||||
| 	if (readl(&udc_regs_p->dev_stat) & DEV_STAT_RXFIFO_EMPTY) | ||||
| 		return -1; | ||||
| 
 | ||||
| 	nw = len / sizeof(u32); | ||||
| 	nb = len % sizeof(u32); | ||||
| 
 | ||||
| 	wrdp = (u32 *)bufp; | ||||
| 	for (i = 0; i < nw; i++) { | ||||
| 		writel(readl(fifo_ptr), wrdp); | ||||
| 		wrdp++; | ||||
| 	} | ||||
| 
 | ||||
| 	bytp = (u8 *)wrdp; | ||||
| 	for (i = 0; i < nb; i++) { | ||||
| 		writeb(readb(fifo_ptr), bytp); | ||||
| 		fifo_ptr++; | ||||
| 		bytp++; | ||||
| 	} | ||||
| 	readl(&outep_regs_p[epNum].write_done); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static void usbputpcktofifo(int epNum, u8 *bufp, u32 len) | ||||
| { | ||||
| 	u32 i, nw, nb; | ||||
| 	u32 *wrdp; | ||||
| 	u8 *bytp; | ||||
| 	u8 *fifo_ptr = get_fifo(epNum, 1); | ||||
| 
 | ||||
| 	nw = len / sizeof(int); | ||||
| 	nb = len % sizeof(int); | ||||
| 	wrdp = (u32 *)bufp; | ||||
| 	for (i = 0; i < nw; i++) { | ||||
| 		writel(*wrdp, fifo_ptr); | ||||
| 		wrdp++; | ||||
| 	} | ||||
| 
 | ||||
| 	bytp = (u8 *)wrdp; | ||||
| 	for (i = 0; i < nb; i++) { | ||||
| 		writeb(*bytp, fifo_ptr); | ||||
| 		fifo_ptr++; | ||||
| 		bytp++; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * spear_write_noniso_tx_fifo - Write the next packet to TxFIFO. | ||||
|  * @endpoint:		Endpoint pointer. | ||||
|  * | ||||
|  * If the endpoint has an active tx_urb, then the next packet of data from the | ||||
|  * URB is written to the tx FIFO.  The total amount of data in the urb is given | ||||
|  * by urb->actual_length.  The maximum amount of data that can be sent in any | ||||
|  * one packet is given by endpoint->tx_packetSize.  The number of data bytes | ||||
|  * from this URB that have already been transmitted is given by endpoint->sent. | ||||
|  * endpoint->last is updated by this routine with the number of data bytes | ||||
|  * transmitted in this packet. | ||||
|  * | ||||
|  */ | ||||
| static void spear_write_noniso_tx_fifo(struct usb_endpoint_instance | ||||
| 				       *endpoint) | ||||
| { | ||||
| 	struct urb *urb = endpoint->tx_urb; | ||||
| 	int align; | ||||
| 
 | ||||
| 	if (urb) { | ||||
| 		u32 last; | ||||
| 
 | ||||
| 		UDCDBGA("urb->buffer %p, buffer_length %d, actual_length %d", | ||||
| 			urb->buffer, urb->buffer_length, urb->actual_length); | ||||
| 
 | ||||
| 		last = MIN(urb->actual_length - endpoint->sent, | ||||
| 			   endpoint->tx_packetSize); | ||||
| 
 | ||||
| 		if (last) { | ||||
| 			u8 *cp = urb->buffer + endpoint->sent; | ||||
| 
 | ||||
| 			/*
 | ||||
| 			 * This ensures that USBD packet fifo is accessed | ||||
| 			 * - through word aligned pointer or | ||||
| 			 * - through non word aligned pointer but only | ||||
| 			 *   with a max length to make the next packet | ||||
| 			 *   word aligned | ||||
| 			 */ | ||||
| 
 | ||||
| 			align = ((ulong)cp % sizeof(int)); | ||||
| 			if (align) | ||||
| 				last = MIN(last, sizeof(int) - align); | ||||
| 
 | ||||
| 			UDCDBGA("endpoint->sent %d, tx_packetSize %d, last %d", | ||||
| 				endpoint->sent, endpoint->tx_packetSize, last); | ||||
| 
 | ||||
| 			usbputpcktofifo(endpoint->endpoint_address & | ||||
| 					USB_ENDPOINT_NUMBER_MASK, cp, last); | ||||
| 		} | ||||
| 		endpoint->last = last; | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Handle SETUP USB interrupt. | ||||
|  * This function implements TRM Figure 14-14. | ||||
|  */ | ||||
| static void spear_udc_setup(struct usb_endpoint_instance *endpoint) | ||||
| { | ||||
| 	u8 *datap = (u8 *)&ep0_urb->device_request; | ||||
| 	int ep_addr = endpoint->endpoint_address; | ||||
| 
 | ||||
| 	UDCDBG("-> Entering device setup"); | ||||
| 	usbgetpckfromfifo(ep_addr, datap, 8); | ||||
| 
 | ||||
| 	/* Try to process setup packet */ | ||||
| 	if (ep0_recv_setup(ep0_urb)) { | ||||
| 		/* Not a setup packet, stall next EP0 transaction */ | ||||
| 		udc_stall_ep(0); | ||||
| 		UDCDBG("can't parse setup packet, still waiting for setup"); | ||||
| 		return; | ||||
| 	} | ||||
| 
 | ||||
| 	/* Check direction */ | ||||
| 	if ((ep0_urb->device_request.bmRequestType & USB_REQ_DIRECTION_MASK) | ||||
| 	    == USB_REQ_HOST2DEVICE) { | ||||
| 		UDCDBG("control write on EP0"); | ||||
| 		if (le16_to_cpu(ep0_urb->device_request.wLength)) { | ||||
| 			/* Stall this request */ | ||||
| 			UDCDBG("Stalling unsupported EP0 control write data " | ||||
| 			       "stage."); | ||||
| 			udc_stall_ep(0); | ||||
| 		} | ||||
| 	} else { | ||||
| 
 | ||||
| 		UDCDBG("control read on EP0"); | ||||
| 		/*
 | ||||
| 		 * The ep0_recv_setup function has already placed our response | ||||
| 		 * packet data in ep0_urb->buffer and the packet length in | ||||
| 		 * ep0_urb->actual_length. | ||||
| 		 */ | ||||
| 		endpoint->tx_urb = ep0_urb; | ||||
| 		endpoint->sent = 0; | ||||
| 		/*
 | ||||
| 		 * Write packet data to the FIFO.  spear_write_noniso_tx_fifo | ||||
| 		 * will update endpoint->last with the number of bytes written | ||||
| 		 * to the FIFO. | ||||
| 		 */ | ||||
| 		spear_write_noniso_tx_fifo(endpoint); | ||||
| 
 | ||||
| 		writel(0x0, &inep_regs_p[ep_addr].write_done); | ||||
| 	} | ||||
| 
 | ||||
| 	udc_unset_nak(endpoint->endpoint_address); | ||||
| 
 | ||||
| 	UDCDBG("<- Leaving device setup"); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Handle endpoint 0 RX interrupt | ||||
|  */ | ||||
| static void spear_udc_ep0_rx(struct usb_endpoint_instance *endpoint) | ||||
| { | ||||
| 	u8 dummy[64]; | ||||
| 
 | ||||
| 	UDCDBG("RX on EP0"); | ||||
| 
 | ||||
| 	/* Check direction */ | ||||
| 	if ((ep0_urb->device_request.bmRequestType | ||||
| 	     & USB_REQ_DIRECTION_MASK) == USB_REQ_HOST2DEVICE) { | ||||
| 		/*
 | ||||
| 		 * This rx interrupt must be for a control write data | ||||
| 		 * stage packet. | ||||
| 		 * | ||||
| 		 * We don't support control write data stages. | ||||
| 		 * We should never end up here. | ||||
| 		 */ | ||||
| 
 | ||||
| 		UDCDBG("Stalling unexpected EP0 control write " | ||||
| 		       "data stage packet"); | ||||
| 		udc_stall_ep(0); | ||||
| 	} else { | ||||
| 		/*
 | ||||
| 		 * This rx interrupt must be for a control read status | ||||
| 		 * stage packet. | ||||
| 		 */ | ||||
| 		UDCDBG("ACK on EP0 control read status stage packet"); | ||||
| 		u32 len = (readl(&outep_regs_p[0].endp_status) >> 11) & 0xfff; | ||||
| 		usbgetpckfromfifo(0, dummy, len); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Handle endpoint 0 TX interrupt | ||||
|  */ | ||||
| static void spear_udc_ep0_tx(struct usb_endpoint_instance *endpoint) | ||||
| { | ||||
| 	struct usb_device_request *request = &ep0_urb->device_request; | ||||
| 	int ep_addr; | ||||
| 
 | ||||
| 	UDCDBG("TX on EP0"); | ||||
| 
 | ||||
| 	/* Check direction */ | ||||
| 	if ((request->bmRequestType & USB_REQ_DIRECTION_MASK) == | ||||
| 	    USB_REQ_HOST2DEVICE) { | ||||
| 		/*
 | ||||
| 		 * This tx interrupt must be for a control write status | ||||
| 		 * stage packet. | ||||
| 		 */ | ||||
| 		UDCDBG("ACK on EP0 control write status stage packet"); | ||||
| 	} else { | ||||
| 		/*
 | ||||
| 		 * This tx interrupt must be for a control read data | ||||
| 		 * stage packet. | ||||
| 		 */ | ||||
| 		int wLength = le16_to_cpu(request->wLength); | ||||
| 
 | ||||
| 		/*
 | ||||
| 		 * Update our count of bytes sent so far in this | ||||
| 		 * transfer. | ||||
| 		 */ | ||||
| 		endpoint->sent += endpoint->last; | ||||
| 
 | ||||
| 		/*
 | ||||
| 		 * We are finished with this transfer if we have sent | ||||
| 		 * all of the bytes in our tx urb (urb->actual_length) | ||||
| 		 * unless we need a zero-length terminating packet.  We | ||||
| 		 * need a zero-length terminating packet if we returned | ||||
| 		 * fewer bytes than were requested (wLength) by the host, | ||||
| 		 * and the number of bytes we returned is an exact | ||||
| 		 * multiple of the packet size endpoint->tx_packetSize. | ||||
| 		 */ | ||||
| 		if ((endpoint->sent == ep0_urb->actual_length) && | ||||
| 		    ((ep0_urb->actual_length == wLength) || | ||||
| 		     (endpoint->last != endpoint->tx_packetSize))) { | ||||
| 			/* Done with control read data stage. */ | ||||
| 			UDCDBG("control read data stage complete"); | ||||
| 		} else { | ||||
| 			/*
 | ||||
| 			 * We still have another packet of data to send | ||||
| 			 * in this control read data stage or else we | ||||
| 			 * need a zero-length terminating packet. | ||||
| 			 */ | ||||
| 			UDCDBG("ACK control read data stage packet"); | ||||
| 			spear_write_noniso_tx_fifo(endpoint); | ||||
| 
 | ||||
| 			ep_addr = endpoint->endpoint_address; | ||||
| 			writel(0x0, &inep_regs_p[ep_addr].write_done); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| static struct usb_endpoint_instance *spear_find_ep(int ep) | ||||
| { | ||||
| 	int i; | ||||
| 
 | ||||
| 	for (i = 0; i < udc_device->bus->max_endpoints; i++) { | ||||
| 		if ((udc_device->bus->endpoint_array[i].endpoint_address & | ||||
| 		     USB_ENDPOINT_NUMBER_MASK) == ep) | ||||
| 			return &udc_device->bus->endpoint_array[i]; | ||||
| 	} | ||||
| 	return NULL; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Handle RX transaction on non-ISO endpoint. | ||||
|  * The ep argument is a physical endpoint number for a non-ISO IN endpoint | ||||
|  * in the range 1 to 15. | ||||
|  */ | ||||
| static void spear_udc_epn_rx(int ep) | ||||
| { | ||||
| 	int nbytes = 0; | ||||
| 	struct urb *urb; | ||||
| 	struct usb_endpoint_instance *endpoint = spear_find_ep(ep); | ||||
| 
 | ||||
| 	if (endpoint) { | ||||
| 		urb = endpoint->rcv_urb; | ||||
| 
 | ||||
| 		if (urb) { | ||||
| 			u8 *cp = urb->buffer + urb->actual_length; | ||||
| 
 | ||||
| 			nbytes = (readl(&outep_regs_p[ep].endp_status) >> 11) & | ||||
| 			    0xfff; | ||||
| 			usbgetpckfromfifo(ep, cp, nbytes); | ||||
| 			usbd_rcv_complete(endpoint, nbytes, 0); | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Handle TX transaction on non-ISO endpoint. | ||||
|  * The ep argument is a physical endpoint number for a non-ISO IN endpoint | ||||
|  * in the range 16 to 30. | ||||
|  */ | ||||
| static void spear_udc_epn_tx(int ep) | ||||
| { | ||||
| 	struct usb_endpoint_instance *endpoint = spear_find_ep(ep); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * We need to transmit a terminating zero-length packet now if | ||||
| 	 * we have sent all of the data in this URB and the transfer | ||||
| 	 * size was an exact multiple of the packet size. | ||||
| 	 */ | ||||
| 	if (endpoint && endpoint->tx_urb && endpoint->tx_urb->actual_length) { | ||||
| 		if (endpoint->last == endpoint->tx_packetSize) { | ||||
| 			/* handle zero length packet here */ | ||||
| 			writel(0x0, &inep_regs_p[ep].write_done); | ||||
| 		} | ||||
| 		/* retire the data that was just sent */ | ||||
| 		usbd_tx_complete(endpoint); | ||||
| 		/*
 | ||||
| 		 * Check to see if we have more data ready to transmit | ||||
| 		 * now. | ||||
| 		 */ | ||||
| 		if (endpoint->tx_urb && endpoint->tx_urb->actual_length) { | ||||
| 			/* write data to FIFO */ | ||||
| 			spear_write_noniso_tx_fifo(endpoint); | ||||
| 			writel(0x0, &inep_regs_p[ep].write_done); | ||||
| 
 | ||||
| 		} else if (endpoint->tx_urb | ||||
| 			   && (endpoint->tx_urb->actual_length == 0)) { | ||||
| 			/* udc_set_nak(ep); */ | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Start of public functions. | ||||
|  */ | ||||
| 
 | ||||
| /* Called to start packet transmission. */ | ||||
| int udc_endpoint_write(struct usb_endpoint_instance *endpoint) | ||||
| { | ||||
| 	udc_unset_nak(endpoint->endpoint_address & USB_ENDPOINT_NUMBER_MASK); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| /* Start to initialize h/w stuff */ | ||||
| int udc_init(void) | ||||
| { | ||||
| 	int i; | ||||
| 	u32 plug_st; | ||||
| 
 | ||||
| 	udc_device = NULL; | ||||
| 
 | ||||
| 	UDCDBG("starting"); | ||||
| 
 | ||||
| 	readl(&plug_regs_p->plug_pending); | ||||
| 
 | ||||
| 	udc_disconnect(); | ||||
| 
 | ||||
| 	for (i = 0; i < UDC_INIT_MDELAY; i++) | ||||
| 		udelay(1000); | ||||
| 
 | ||||
| 	plug_st = readl(&plug_regs_p->plug_state); | ||||
| 	writel(plug_st | PLUG_STATUS_EN, &plug_regs_p->plug_state); | ||||
| 
 | ||||
| 	writel(~0x0, &udc_regs_p->endp_int); | ||||
| 	writel(~0x0, &udc_regs_p->dev_int_mask); | ||||
| 	writel(~0x0, &udc_regs_p->endp_int_mask); | ||||
| 
 | ||||
| 	writel(DEV_CONF_FS_SPEED | DEV_CONF_REMWAKEUP | DEV_CONF_SELFPOW | | ||||
| 	       /* Dev_Conf_SYNCFRAME | */ | ||||
| 	       DEV_CONF_PHYINT_16, &udc_regs_p->dev_conf); | ||||
| 
 | ||||
| 	writel(0x0, &udc_regs_p->dev_cntl); | ||||
| 
 | ||||
| 	/* Clear all interrupts pending */ | ||||
| 	writel(DEV_INT_MSK, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * udc_setup_ep - setup endpoint | ||||
|  * Associate a physical endpoint with endpoint_instance | ||||
|  */ | ||||
| void udc_setup_ep(struct usb_device_instance *device, | ||||
| 		  u32 ep, struct usb_endpoint_instance *endpoint) | ||||
| { | ||||
| 	UDCDBGA("setting up endpoint addr %x", endpoint->endpoint_address); | ||||
| 	int ep_addr; | ||||
| 	int ep_num, ep_type; | ||||
| 	int packet_size; | ||||
| 	int buffer_size; | ||||
| 	int attributes; | ||||
| 	char *tt; | ||||
| 	u32 endp_intmask; | ||||
| 
 | ||||
| 	tt = getenv("usbtty"); | ||||
| 	if (!tt) | ||||
| 		tt = "generic"; | ||||
| 
 | ||||
| 	ep_addr = endpoint->endpoint_address; | ||||
| 	ep_num = ep_addr & USB_ENDPOINT_NUMBER_MASK; | ||||
| 
 | ||||
| 	if ((ep_addr & USB_ENDPOINT_DIR_MASK) == USB_DIR_IN) { | ||||
| 		/* IN endpoint */ | ||||
| 		packet_size = endpoint->tx_packetSize; | ||||
| 		buffer_size = packet_size * 2; | ||||
| 		attributes = endpoint->tx_attributes; | ||||
| 	} else { | ||||
| 		/* OUT endpoint */ | ||||
| 		packet_size = endpoint->rcv_packetSize; | ||||
| 		buffer_size = packet_size * 2; | ||||
| 		attributes = endpoint->rcv_attributes; | ||||
| 	} | ||||
| 
 | ||||
| 	switch (attributes & USB_ENDPOINT_XFERTYPE_MASK) { | ||||
| 	case USB_ENDPOINT_XFER_CONTROL: | ||||
| 		ep_type = ENDP_EPTYPE_CNTL; | ||||
| 		break; | ||||
| 	case USB_ENDPOINT_XFER_BULK: | ||||
| 	default: | ||||
| 		ep_type = ENDP_EPTYPE_BULK; | ||||
| 		break; | ||||
| 	case USB_ENDPOINT_XFER_INT: | ||||
| 		ep_type = ENDP_EPTYPE_INT; | ||||
| 		break; | ||||
| 	case USB_ENDPOINT_XFER_ISOC: | ||||
| 		ep_type = ENDP_EPTYPE_ISO; | ||||
| 		break; | ||||
| 	} | ||||
| 
 | ||||
| 	struct udc_endp_regs *out_p = &outep_regs_p[ep_num]; | ||||
| 	struct udc_endp_regs *in_p = &inep_regs_p[ep_num]; | ||||
| 
 | ||||
| 	if (!ep_addr) { | ||||
| 		/* Setup endpoint 0 */ | ||||
| 		buffer_size = packet_size; | ||||
| 
 | ||||
| 		writel(readl(&in_p->endp_cntl) | ENDP_CNTL_CNAK, | ||||
| 		       &in_p->endp_cntl); | ||||
| 
 | ||||
| 		writel(readl(&out_p->endp_cntl) | ENDP_CNTL_CNAK, | ||||
| 		       &out_p->endp_cntl); | ||||
| 
 | ||||
| 		writel(ENDP_CNTL_CONTROL | ENDP_CNTL_FLUSH, &in_p->endp_cntl); | ||||
| 
 | ||||
| 		writel(buffer_size / sizeof(int), &in_p->endp_bsorfn); | ||||
| 
 | ||||
| 		writel(packet_size, &in_p->endp_maxpacksize); | ||||
| 
 | ||||
| 		writel(ENDP_CNTL_CONTROL | ENDP_CNTL_RRDY, &out_p->endp_cntl); | ||||
| 
 | ||||
| 		writel(packet_size | ((buffer_size / sizeof(int)) << 16), | ||||
| 		       &out_p->endp_maxpacksize); | ||||
| 
 | ||||
| 		writel((packet_size << 19) | ENDP_EPTYPE_CNTL, | ||||
| 		       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 
 | ||||
| 	} else if ((ep_addr & USB_ENDPOINT_DIR_MASK) == USB_DIR_IN) { | ||||
| 		/* Setup the IN endpoint */ | ||||
| 		writel(0x0, &in_p->endp_status); | ||||
| 		writel((ep_type << 4) | ENDP_CNTL_RRDY, &in_p->endp_cntl); | ||||
| 		writel(buffer_size / sizeof(int), &in_p->endp_bsorfn); | ||||
| 		writel(packet_size, &in_p->endp_maxpacksize); | ||||
| 
 | ||||
| 		if (!strcmp(tt, "cdc_acm")) { | ||||
| 			if (ep_type == ENDP_EPTYPE_INT) { | ||||
| 				/* Conf no. 1 Interface no. 0 */ | ||||
| 				writel((packet_size << 19) | | ||||
| 				       ENDP_EPDIR_IN | (1 << 7) | | ||||
| 				       (0 << 11) | (ep_type << 5) | ep_num, | ||||
| 				       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 			} else { | ||||
| 				/* Conf no. 1 Interface no. 1 */ | ||||
| 				writel((packet_size << 19) | | ||||
| 				       ENDP_EPDIR_IN | (1 << 7) | | ||||
| 				       (1 << 11) | (ep_type << 5) | ep_num, | ||||
| 				       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 			} | ||||
| 		} else { | ||||
| 			/* Conf no. 1 Interface no. 0 */ | ||||
| 			writel((packet_size << 19) | | ||||
| 			       ENDP_EPDIR_IN | (1 << 7) | | ||||
| 			       (0 << 11) | (ep_type << 5) | ep_num, | ||||
| 			       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 		} | ||||
| 
 | ||||
| 	} else { | ||||
| 		/* Setup the OUT endpoint */ | ||||
| 		writel(0x0, &out_p->endp_status); | ||||
| 		writel((ep_type << 4) | ENDP_CNTL_RRDY, &out_p->endp_cntl); | ||||
| 		writel(packet_size | ((buffer_size / sizeof(int)) << 16), | ||||
| 		       &out_p->endp_maxpacksize); | ||||
| 
 | ||||
| 		if (!strcmp(tt, "cdc_acm")) { | ||||
| 			writel((packet_size << 19) | | ||||
| 			       ENDP_EPDIR_OUT | (1 << 7) | | ||||
| 			       (1 << 11) | (ep_type << 5) | ep_num, | ||||
| 			       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 		} else { | ||||
| 			writel((packet_size << 19) | | ||||
| 			       ENDP_EPDIR_OUT | (1 << 7) | | ||||
| 			       (0 << 11) | (ep_type << 5) | ep_num, | ||||
| 			       &udc_regs_p->udc_endp_reg[ep_num]); | ||||
| 		} | ||||
| 
 | ||||
| 	} | ||||
| 
 | ||||
| 	endp_intmask = readl(&udc_regs_p->endp_int_mask); | ||||
| 	endp_intmask &= ~((1 << ep_num) | 0x10000 << ep_num); | ||||
| 	writel(endp_intmask, &udc_regs_p->endp_int_mask); | ||||
| } | ||||
| 
 | ||||
| /* Turn on the USB connection by enabling the pullup resistor */ | ||||
| void udc_connect(void) | ||||
| { | ||||
| 	u32 plug_st; | ||||
| 
 | ||||
| 	plug_st = readl(&plug_regs_p->plug_state); | ||||
| 	plug_st &= ~(PLUG_STATUS_PHY_RESET | PLUG_STATUS_PHY_MODE); | ||||
| 	writel(plug_st, &plug_regs_p->plug_state); | ||||
| } | ||||
| 
 | ||||
| /* Turn off the USB connection by disabling the pullup resistor */ | ||||
| void udc_disconnect(void) | ||||
| { | ||||
| 	u32 plug_st; | ||||
| 
 | ||||
| 	plug_st = readl(&plug_regs_p->plug_state); | ||||
| 	plug_st |= (PLUG_STATUS_PHY_RESET | PLUG_STATUS_PHY_MODE); | ||||
| 	writel(plug_st, &plug_regs_p->plug_state); | ||||
| } | ||||
| 
 | ||||
| /* Switch on the UDC */ | ||||
| void udc_enable(struct usb_device_instance *device) | ||||
| { | ||||
| 	UDCDBGA("enable device %p, status %d", device, device->status); | ||||
| 
 | ||||
| 	/* Save the device structure pointer */ | ||||
| 	udc_device = device; | ||||
| 
 | ||||
| 	/* Setup ep0 urb */ | ||||
| 	if (!ep0_urb) { | ||||
| 		ep0_urb = | ||||
| 		    usbd_alloc_urb(udc_device, udc_device->bus->endpoint_array); | ||||
| 	} else { | ||||
| 		serial_printf("udc_enable: ep0_urb already allocated %p\n", | ||||
| 			      ep0_urb); | ||||
| 	} | ||||
| 
 | ||||
| 	writel(DEV_INT_SOF, &udc_regs_p->dev_int_mask); | ||||
| } | ||||
| 
 | ||||
| /**
 | ||||
|  * udc_startup - allow udc code to do any additional startup | ||||
|  */ | ||||
| void udc_startup_events(struct usb_device_instance *device) | ||||
| { | ||||
| 	/* The DEVICE_INIT event puts the USB device in the state STATE_INIT. */ | ||||
| 	usbd_device_event_irq(device, DEVICE_INIT, 0); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * The DEVICE_CREATE event puts the USB device in the state | ||||
| 	 * STATE_ATTACHED. | ||||
| 	 */ | ||||
| 	usbd_device_event_irq(device, DEVICE_CREATE, 0); | ||||
| 
 | ||||
| 	/*
 | ||||
| 	 * Some USB controller driver implementations signal | ||||
| 	 * DEVICE_HUB_CONFIGURED and DEVICE_RESET events here. | ||||
| 	 * DEVICE_HUB_CONFIGURED causes a transition to the state STATE_POWERED, | ||||
| 	 * and DEVICE_RESET causes a transition to the state STATE_DEFAULT. | ||||
| 	 * The SPEAr USB client controller has the capability to detect when the | ||||
| 	 * USB cable is connected to a powered USB bus, so we will defer the | ||||
| 	 * DEVICE_HUB_CONFIGURED and DEVICE_RESET events until later. | ||||
| 	 */ | ||||
| 
 | ||||
| 	udc_enable(device); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Plug detection interrupt handling | ||||
|  */ | ||||
| void spear_udc_plug_irq(void) | ||||
| { | ||||
| 	if (readl(&plug_regs_p->plug_state) & PLUG_STATUS_ATTACHED) { | ||||
| 		/*
 | ||||
| 		 * USB cable attached | ||||
| 		 * Turn off PHY reset bit (PLUG detect). | ||||
| 		 * Switch PHY opmode to normal operation (PLUG detect). | ||||
| 		 */ | ||||
| 		udc_connect(); | ||||
| 		writel(DEV_INT_SOF, &udc_regs_p->dev_int_mask); | ||||
| 
 | ||||
| 		UDCDBG("device attached and powered"); | ||||
| 		udc_state_transition(udc_device->device_state, STATE_POWERED); | ||||
| 	} else { | ||||
| 		/*
 | ||||
| 		 * USB cable detached | ||||
| 		 * Reset the PHY and switch the mode. | ||||
| 		 */ | ||||
| 		udc_disconnect(); | ||||
| 		writel(~0x0, &udc_regs_p->dev_int_mask); | ||||
| 
 | ||||
| 		UDCDBG("device detached or unpowered"); | ||||
| 		udc_state_transition(udc_device->device_state, STATE_ATTACHED); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Device interrupt handling | ||||
|  */ | ||||
| void spear_udc_dev_irq(void) | ||||
| { | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_USBRESET) { | ||||
| 		writel(~0x0, &udc_regs_p->endp_int_mask); | ||||
| 
 | ||||
| 		udc_connect(); | ||||
| 
 | ||||
| 		writel(readl(&inep_regs_p[0].endp_cntl) | ENDP_CNTL_FLUSH, | ||||
| 		       &inep_regs_p[0].endp_cntl); | ||||
| 
 | ||||
| 		writel(DEV_INT_USBRESET, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 		UDCDBG("device reset in progess"); | ||||
| 		udc_state_transition(udc_device->device_state, STATE_DEFAULT); | ||||
| 	} | ||||
| 
 | ||||
| 	/* Device Enumeration completed */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_ENUM) { | ||||
| 		writel(DEV_INT_ENUM, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 		/* Endpoint interrupt enabled for Ctrl IN & Ctrl OUT */ | ||||
| 		writel(readl(&udc_regs_p->endp_int_mask) & ~0x10001, | ||||
| 		       &udc_regs_p->endp_int_mask); | ||||
| 
 | ||||
| 		UDCDBG("default -> addressed"); | ||||
| 		udc_state_transition(udc_device->device_state, STATE_ADDRESSED); | ||||
| 	} | ||||
| 
 | ||||
| 	/* The USB will be in SUSPEND in 3 ms */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_INACTIVE) { | ||||
| 		writel(DEV_INT_INACTIVE, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 		UDCDBG("entering inactive state"); | ||||
| 		/* usbd_device_event_irq(udc_device, DEVICE_BUS_INACTIVE, 0); */ | ||||
| 	} | ||||
| 
 | ||||
| 	/* SetConfiguration command received */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_SETCFG) { | ||||
| 		writel(DEV_INT_SETCFG, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 		UDCDBG("entering configured state"); | ||||
| 		udc_state_transition(udc_device->device_state, | ||||
| 				     STATE_CONFIGURED); | ||||
| 	} | ||||
| 
 | ||||
| 	/* SetInterface command received */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_SETINTF) | ||||
| 		writel(DEV_INT_SETINTF, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 	/* USB Suspend detected on cable */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_SUSPUSB) { | ||||
| 		writel(DEV_INT_SUSPUSB, &udc_regs_p->dev_int); | ||||
| 
 | ||||
| 		UDCDBG("entering suspended state"); | ||||
| 		usbd_device_event_irq(udc_device, DEVICE_BUS_INACTIVE, 0); | ||||
| 	} | ||||
| 
 | ||||
| 	/* USB Start-Of-Frame detected on cable */ | ||||
| 	if (readl(&udc_regs_p->dev_int) & DEV_INT_SOF) | ||||
| 		writel(DEV_INT_SOF, &udc_regs_p->dev_int); | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * Endpoint interrupt handling | ||||
|  */ | ||||
| void spear_udc_endpoint_irq(void) | ||||
| { | ||||
| 	while (readl(&udc_regs_p->endp_int) & ENDP0_INT_CTRLOUT) { | ||||
| 
 | ||||
| 		writel(ENDP0_INT_CTRLOUT, &udc_regs_p->endp_int); | ||||
| 
 | ||||
| 		if ((readl(&outep_regs_p[0].endp_status) & ENDP_STATUS_OUTMSK) | ||||
| 		    == ENDP_STATUS_OUT_SETUP) { | ||||
| 			spear_udc_setup(udc_device->bus->endpoint_array + 0); | ||||
| 			writel(ENDP_STATUS_OUT_SETUP, | ||||
| 			       &outep_regs_p[0].endp_status); | ||||
| 
 | ||||
| 		} else if ((readl(&outep_regs_p[0].endp_status) & | ||||
| 			    ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_DATA) { | ||||
| 			spear_udc_ep0_rx(udc_device->bus->endpoint_array + 0); | ||||
| 			writel(ENDP_STATUS_OUT_DATA, | ||||
| 			       &outep_regs_p[0].endp_status); | ||||
| 
 | ||||
| 		} else if ((readl(&outep_regs_p[0].endp_status) & | ||||
| 			    ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_NONE) { | ||||
| 			/* NONE received */ | ||||
| 		} | ||||
| 
 | ||||
| 		writel(0x0, &outep_regs_p[0].endp_status); | ||||
| 	} | ||||
| 
 | ||||
| 	if (readl(&udc_regs_p->endp_int) & ENDP0_INT_CTRLIN) { | ||||
| 		spear_udc_ep0_tx(udc_device->bus->endpoint_array + 0); | ||||
| 
 | ||||
| 		writel(ENDP_STATUS_IN, &inep_regs_p[0].endp_status); | ||||
| 		writel(ENDP0_INT_CTRLIN, &udc_regs_p->endp_int); | ||||
| 	} | ||||
| 
 | ||||
| 	while (readl(&udc_regs_p->endp_int) & ENDP_INT_NONISOOUT_MSK) { | ||||
| 		u32 epnum = 0; | ||||
| 		u32 ep_int = readl(&udc_regs_p->endp_int) & | ||||
| 		    ENDP_INT_NONISOOUT_MSK; | ||||
| 
 | ||||
| 		ep_int >>= 16; | ||||
| 		while (0x0 == (ep_int & 0x1)) { | ||||
| 			ep_int >>= 1; | ||||
| 			epnum++; | ||||
| 		} | ||||
| 
 | ||||
| 		writel((1 << 16) << epnum, &udc_regs_p->endp_int); | ||||
| 
 | ||||
| 		if ((readl(&outep_regs_p[epnum].endp_status) & | ||||
| 		     ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_DATA) { | ||||
| 
 | ||||
| 			spear_udc_epn_rx(epnum); | ||||
| 			writel(ENDP_STATUS_OUT_DATA, | ||||
| 			       &outep_regs_p[epnum].endp_status); | ||||
| 		} else if ((readl(&outep_regs_p[epnum].endp_status) & | ||||
| 			    ENDP_STATUS_OUTMSK) == ENDP_STATUS_OUT_NONE) { | ||||
| 			writel(0x0, &outep_regs_p[epnum].endp_status); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	if (readl(&udc_regs_p->endp_int) & ENDP_INT_NONISOIN_MSK) { | ||||
| 		u32 epnum = 0; | ||||
| 		u32 ep_int = readl(&udc_regs_p->endp_int) & | ||||
| 		    ENDP_INT_NONISOIN_MSK; | ||||
| 
 | ||||
| 		while (0x0 == (ep_int & 0x1)) { | ||||
| 			ep_int >>= 1; | ||||
| 			epnum++; | ||||
| 		} | ||||
| 
 | ||||
| 		if (readl(&inep_regs_p[epnum].endp_status) & ENDP_STATUS_IN) { | ||||
| 			writel(ENDP_STATUS_IN, | ||||
| 			       &outep_regs_p[epnum].endp_status); | ||||
| 			spear_udc_epn_tx(epnum); | ||||
| 
 | ||||
| 			writel(ENDP_STATUS_IN, | ||||
| 			       &outep_regs_p[epnum].endp_status); | ||||
| 		} | ||||
| 
 | ||||
| 		writel((1 << epnum), &udc_regs_p->endp_int); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| /*
 | ||||
|  * UDC interrupts | ||||
|  */ | ||||
| void udc_irq(void) | ||||
| { | ||||
| 	/*
 | ||||
| 	 * Loop while we have interrupts. | ||||
| 	 * If we don't do this, the input chain | ||||
| 	 * polling delay is likely to miss | ||||
| 	 * host requests. | ||||
| 	 */ | ||||
| 	while (readl(&plug_regs_p->plug_pending)) | ||||
| 		spear_udc_plug_irq(); | ||||
| 
 | ||||
| 	while (readl(&udc_regs_p->dev_int)) | ||||
| 		spear_udc_dev_irq(); | ||||
| 
 | ||||
| 	if (readl(&udc_regs_p->endp_int)) | ||||
| 		spear_udc_endpoint_irq(); | ||||
| } | ||||
| 
 | ||||
| /* Flow control */ | ||||
| void udc_set_nak(int epid) | ||||
| { | ||||
| 	writel(readl(&inep_regs_p[epid].endp_cntl) | ENDP_CNTL_SNAK, | ||||
| 	       &inep_regs_p[epid].endp_cntl); | ||||
| 
 | ||||
| 	writel(readl(&outep_regs_p[epid].endp_cntl) | ENDP_CNTL_SNAK, | ||||
| 	       &outep_regs_p[epid].endp_cntl); | ||||
| } | ||||
| 
 | ||||
| void udc_unset_nak(int epid) | ||||
| { | ||||
| 	u32 val; | ||||
| 
 | ||||
| 	val = readl(&inep_regs_p[epid].endp_cntl); | ||||
| 	val &= ~ENDP_CNTL_SNAK; | ||||
| 	val |= ENDP_CNTL_CNAK; | ||||
| 	writel(val, &inep_regs_p[epid].endp_cntl); | ||||
| 
 | ||||
| 	val = readl(&outep_regs_p[epid].endp_cntl); | ||||
| 	val &= ~ENDP_CNTL_SNAK; | ||||
| 	val |= ENDP_CNTL_CNAK; | ||||
| 	writel(val, &outep_regs_p[epid].endp_cntl); | ||||
| } | ||||
							
								
								
									
										230
									
								
								include/usb/spr_udc.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										230
									
								
								include/usb/spr_udc.h
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,230 @@ | ||||
| /*
 | ||||
|  * (C) Copyright 2009 | ||||
|  * Vipin Kumar, ST Micoelectronics, vipin.kumar@st.com. | ||||
|  * | ||||
|  * See file CREDITS for list of people who contributed to this | ||||
|  * project. | ||||
|  * | ||||
|  * 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 | ||||
|  */ | ||||
| 
 | ||||
| #ifndef __SPR_UDC_H | ||||
| #define __SPR_UDC_H | ||||
| 
 | ||||
| /*
 | ||||
|  * Defines for  USBD | ||||
|  * | ||||
|  * The udc_ahb controller has three AHB slaves: | ||||
|  * | ||||
|  * 1.  THe UDC registers | ||||
|  * 2.  The plug detect | ||||
|  * 3.  The RX/TX FIFO | ||||
|  */ | ||||
| 
 | ||||
| #define MAX_ENDPOINTS		16 | ||||
| 
 | ||||
| struct udc_endp_regs { | ||||
| 	u32 endp_cntl; | ||||
| 	u32 endp_status; | ||||
| 	u32 endp_bsorfn; | ||||
| 	u32 endp_maxpacksize; | ||||
| 	u32 reserved_1; | ||||
| 	u32 endp_desc_point; | ||||
| 	u32 reserved_2; | ||||
| 	u32 write_done; | ||||
| }; | ||||
| 
 | ||||
| /* Endpoint Control Register definitions */ | ||||
| 
 | ||||
| #define  ENDP_CNTL_STALL		0x00000001 | ||||
| #define  ENDP_CNTL_FLUSH		0x00000002 | ||||
| #define  ENDP_CNTL_SNOOP		0x00000004 | ||||
| #define  ENDP_CNTL_POLL			0x00000008 | ||||
| #define  ENDP_CNTL_CONTROL		0x00000000 | ||||
| #define  ENDP_CNTL_ISO			0x00000010 | ||||
| #define  ENDP_CNTL_BULK			0x00000020 | ||||
| #define  ENDP_CNTL_INT			0x00000030 | ||||
| #define  ENDP_CNTL_NAK			0x00000040 | ||||
| #define  ENDP_CNTL_SNAK			0x00000080 | ||||
| #define  ENDP_CNTL_CNAK			0x00000100 | ||||
| #define  ENDP_CNTL_RRDY			0x00000200 | ||||
| 
 | ||||
| /* Endpoint Satus Register definitions */ | ||||
| 
 | ||||
| #define  ENDP_STATUS_PIDMSK		0x0000000f | ||||
| #define  ENDP_STATUS_OUTMSK		0x00000030 | ||||
| #define  ENDP_STATUS_OUT_NONE		0x00000000 | ||||
| #define  ENDP_STATUS_OUT_DATA		0x00000010 | ||||
| #define  ENDP_STATUS_OUT_SETUP		0x00000020 | ||||
| #define  ENDP_STATUS_IN			0x00000040 | ||||
| #define  ENDP_STATUS_BUFFNAV		0x00000080 | ||||
| #define  ENDP_STATUS_FATERR		0x00000100 | ||||
| #define  ENDP_STATUS_HOSTBUSERR		0x00000200 | ||||
| #define  ENDP_STATUS_TDC		0x00000400 | ||||
| #define  ENDP_STATUS_RXPKTMSK		0x003ff800 | ||||
| 
 | ||||
| struct udc_regs { | ||||
| 	struct udc_endp_regs in_regs[MAX_ENDPOINTS]; | ||||
| 	struct udc_endp_regs out_regs[MAX_ENDPOINTS]; | ||||
| 	u32 dev_conf; | ||||
| 	u32 dev_cntl; | ||||
| 	u32 dev_stat; | ||||
| 	u32 dev_int; | ||||
| 	u32 dev_int_mask; | ||||
| 	u32 endp_int; | ||||
| 	u32 endp_int_mask; | ||||
| 	u32 reserved_3[0x39]; | ||||
| 	u32 reserved_4;		/* offset 0x500 */ | ||||
| 	u32 udc_endp_reg[MAX_ENDPOINTS]; | ||||
| }; | ||||
| 
 | ||||
| /* Device Configuration Register definitions */ | ||||
| 
 | ||||
| #define  DEV_CONF_HS_SPEED		0x00000000 | ||||
| #define  DEV_CONF_LS_SPEED		0x00000002 | ||||
| #define  DEV_CONF_FS_SPEED		0x00000003 | ||||
| #define  DEV_CONF_REMWAKEUP		0x00000004 | ||||
| #define  DEV_CONF_SELFPOW		0x00000008 | ||||
| #define  DEV_CONF_SYNCFRAME		0x00000010 | ||||
| #define  DEV_CONF_PHYINT_8		0x00000020 | ||||
| #define  DEV_CONF_PHYINT_16		0x00000000 | ||||
| #define  DEV_CONF_UTMI_BIDIR		0x00000040 | ||||
| #define  DEV_CONF_STATUS_STALL		0x00000080 | ||||
| 
 | ||||
| /* Device Control Register definitions */ | ||||
| 
 | ||||
| #define  DEV_CNTL_RESUME		0x00000001 | ||||
| #define  DEV_CNTL_TFFLUSH		0x00000002 | ||||
| #define  DEV_CNTL_RXDMAEN		0x00000004 | ||||
| #define  DEV_CNTL_TXDMAEN		0x00000008 | ||||
| #define  DEV_CNTL_DESCRUPD		0x00000010 | ||||
| #define  DEV_CNTL_BIGEND		0x00000020 | ||||
| #define  DEV_CNTL_BUFFILL		0x00000040 | ||||
| #define  DEV_CNTL_TSHLDEN		0x00000080 | ||||
| #define  DEV_CNTL_BURSTEN		0x00000100 | ||||
| #define  DEV_CNTL_DMAMODE		0x00000200 | ||||
| #define  DEV_CNTL_SOFTDISCONNECT	0x00000400 | ||||
| #define  DEV_CNTL_SCALEDOWN		0x00000800 | ||||
| #define  DEV_CNTL_BURSTLENU		0x00010000 | ||||
| #define  DEV_CNTL_BURSTLENMSK		0x00ff0000 | ||||
| #define  DEV_CNTL_TSHLDLENU		0x01000000 | ||||
| #define  DEV_CNTL_TSHLDLENMSK		0xff000000 | ||||
| 
 | ||||
| /* Device Status Register definitions */ | ||||
| 
 | ||||
| #define  DEV_STAT_CFG			0x0000000f | ||||
| #define  DEV_STAT_INTF			0x000000f0 | ||||
| #define  DEV_STAT_ALT			0x00000f00 | ||||
| #define  DEV_STAT_SUSP			0x00001000 | ||||
| #define  DEV_STAT_ENUM			0x00006000 | ||||
| #define  DEV_STAT_ENUM_SPEED_HS		0x00000000 | ||||
| #define  DEV_STAT_ENUM_SPEED_FS		0x00002000 | ||||
| #define  DEV_STAT_ENUM_SPEED_LS		0x00004000 | ||||
| #define  DEV_STAT_RXFIFO_EMPTY		0x00008000 | ||||
| #define  DEV_STAT_PHY_ERR		0x00010000 | ||||
| #define  DEV_STAT_TS			0xf0000000 | ||||
| 
 | ||||
| /* Device Interrupt Register definitions */ | ||||
| 
 | ||||
| #define  DEV_INT_MSK			0x0000007f | ||||
| #define  DEV_INT_SETCFG			0x00000001 | ||||
| #define  DEV_INT_SETINTF		0x00000002 | ||||
| #define  DEV_INT_INACTIVE		0x00000004 | ||||
| #define  DEV_INT_USBRESET		0x00000008 | ||||
| #define  DEV_INT_SUSPUSB		0x00000010 | ||||
| #define  DEV_INT_SOF			0x00000020 | ||||
| #define  DEV_INT_ENUM			0x00000040 | ||||
| 
 | ||||
| /* Endpoint Interrupt Register definitions */ | ||||
| 
 | ||||
| #define  ENDP0_INT_CTRLIN		0x00000001 | ||||
| #define  ENDP1_INT_BULKIN		0x00000002 | ||||
| #define  ENDP_INT_NONISOIN_MSK		0x0000AAAA | ||||
| #define  ENDP2_INT_BULKIN		0x00000004 | ||||
| #define  ENDP0_INT_CTRLOUT		0x00010000 | ||||
| #define  ENDP1_INT_BULKOUT		0x00020000 | ||||
| #define  ENDP2_INT_BULKOUT		0x00040000 | ||||
| #define  ENDP_INT_NONISOOUT_MSK		0x55540000 | ||||
| 
 | ||||
| /* Endpoint Register definitions */ | ||||
| #define  ENDP_EPDIR_OUT			0x00000000 | ||||
| #define  ENDP_EPDIR_IN			0x00000010 | ||||
| #define  ENDP_EPTYPE_CNTL		0x0 | ||||
| #define  ENDP_EPTYPE_ISO		0x1 | ||||
| #define  ENDP_EPTYPE_BULK		0x2 | ||||
| #define  ENDP_EPTYPE_INT		0x3 | ||||
| 
 | ||||
| /*
 | ||||
|  * Defines for Plug Detect | ||||
|  */ | ||||
| 
 | ||||
| struct plug_regs { | ||||
| 	u32 plug_state; | ||||
| 	u32 plug_pending; | ||||
| }; | ||||
| 
 | ||||
| /* Plug State Register definitions */ | ||||
| #define  PLUG_STATUS_EN			0x1 | ||||
| #define  PLUG_STATUS_ATTACHED		0x2 | ||||
| #define  PLUG_STATUS_PHY_RESET		0x4 | ||||
| #define  PLUG_STATUS_PHY_MODE		0x8 | ||||
| 
 | ||||
| /*
 | ||||
|  * Defines for UDC FIFO (Slave Mode) | ||||
|  */ | ||||
| struct udcfifo_regs { | ||||
| 	u32 *fifo_p; | ||||
| }; | ||||
| 
 | ||||
| /*
 | ||||
|  * USBTTY definitions | ||||
|  */ | ||||
| #define  EP0_MAX_PACKET_SIZE		64 | ||||
| #define  UDC_INT_ENDPOINT		1 | ||||
| #define  UDC_INT_PACKET_SIZE		64 | ||||
| #define  UDC_OUT_ENDPOINT		2 | ||||
| #define  UDC_BULK_PACKET_SIZE		64 | ||||
| #define  UDC_IN_ENDPOINT		3 | ||||
| #define  UDC_OUT_PACKET_SIZE		64 | ||||
| #define  UDC_IN_PACKET_SIZE		64 | ||||
| 
 | ||||
| /*
 | ||||
|  * UDC endpoint definitions | ||||
|  */ | ||||
| #define  UDC_EP0			0 | ||||
| #define  UDC_EP1			1 | ||||
| #define  UDC_EP2			2 | ||||
| #define  UDC_EP3			3 | ||||
| 
 | ||||
| /*
 | ||||
|  * Function declarations | ||||
|  */ | ||||
| 
 | ||||
| void udc_irq(void); | ||||
| 
 | ||||
| void udc_set_nak(int epid); | ||||
| void udc_unset_nak(int epid); | ||||
| int udc_endpoint_write(struct usb_endpoint_instance *endpoint); | ||||
| int udc_init(void); | ||||
| void udc_enable(struct usb_device_instance *device); | ||||
| void udc_disable(void); | ||||
| void udc_connect(void); | ||||
| void udc_disconnect(void); | ||||
| void udc_startup_events(struct usb_device_instance *device); | ||||
| void udc_setup_ep(struct usb_device_instance *device, unsigned int ep, | ||||
| 		  struct usb_endpoint_instance *endpoint); | ||||
| 
 | ||||
| #endif /* __SPR_UDC_H */ | ||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user