From c9f37223ce1cd22cf0b901beefe1a53871e18f8c Mon Sep 17 00:00:00 2001
From: iceman1001 <iceman@iuse.se>
Date: Wed, 11 Oct 2017 10:15:41 +0200
Subject: [PATCH] FIX:  at91sam7s256 has some limits on endpoints maxpacket
 sizes.  Took me sometime to find it.

---
 common/usb_cdc.c | 127 +++++++++++++++++++++++++++--------------------
 1 file changed, 73 insertions(+), 54 deletions(-)

diff --git a/common/usb_cdc.c b/common/usb_cdc.c
index 6e780ab75..a96d82183 100644
--- a/common/usb_cdc.c
+++ b/common/usb_cdc.c
@@ -34,14 +34,24 @@
 
 #include "usb_cdc.h"
 
+/*
+AT91SAM7S256  USB Device Port
+• Embedded 328-byte dual-port RAM for endpoints
+• Four endpoints 
+– Endpoint 0: 8 bytes
+– Endpoint 1 and 2: 64 bytes ping-pong
+– Endpoint 3: 64 bytes
+– Ping-pong Mode (two memory banks) for bulk endpoints
+*/
+
 #define AT91C_EP_CONTROL      0
-#define AT91C_EP_CONTROL_SIZE 0x40
+#define AT91C_EP_CONTROL_SIZE 0x08
 
 #define AT91C_EP_IN          2
-#define AT91C_EP_IN_SIZE  0x200
+#define AT91C_EP_IN_SIZE  0x40
 
 #define AT91C_EP_OUT         1
-#define AT91C_EP_OUT_SIZE 0x200
+#define AT91C_EP_OUT_SIZE 0x40
 
 // Section: USB Descriptors
 #define USB_DESCRIPTOR_DEVICE           0x01    // bDescriptorType for a Device Descriptor.
@@ -83,8 +93,8 @@
 #define MS_VENDOR_CODE					0x1C
 #define MS_EXTENDED_COMPAT_ID           0x04
 #define MS_EXTENDED_PROPERTIES          0x05
-#define MS_WCID_GET_DESCRIPTOR			((MS_VENDOR_CODE << 8) | 0xC0)
-#define MS_WCID_GET_FEATURE_DESCRIPTOR	((MS_VENDOR_CODE << 8 )| 0xC1)
+#define MS_WCID_GET_DESCRIPTOR			0xC0
+#define MS_WCID_GET_FEATURE_DESCRIPTOR	0xC1
 
 static const char devDescriptor[] = {
 	/* Device descriptor */
@@ -94,7 +104,7 @@ static const char devDescriptor[] = {
 	0x02,      // Device Class:    CDC class code
 	0x02,      // Device Subclass: CDC class sub code ACM [ice 0x02 = win10 virtual comport :) ]
 	0x00,      // Device Protocol: CDC Device protocol
-	0x40,      // MaxPacket Size0  (ice 0x40, rflder 0x40, pm3, 0x08
+	AT91C_EP_CONTROL_SIZE,      // MaxPacketSize0
     0xc4,0x9a, // Vendor ID  [0x9ac4 = J. Westhues]
     0x8f,0x4b, // Product ID [0x4b8f = Proxmark-3 RFID Instrument]
 	0x01,0x00, // Device release number (0001)
@@ -113,7 +123,7 @@ static const char cfgDescriptor[] = {
 	0x02,   // Number of Interfaces
 	0x01,   // Index value of this Configuration
 	0x00,   // Configuration string index
-	_DEFAULT | _SELF,   // Attributes 0xA0
+	_DEFAULT,   // Attributes 0xA0
 	0xFA,   // Max Power consumption
 
 	/* Communication Class Interface Descriptor Requirement */
@@ -137,7 +147,7 @@ static const char cfgDescriptor[] = {
 	0x04, // Function Length
 	0x24, // Descriptor Type: CS_INTERFACE
 	0x02, // Descriptor Subtype: ACM Func Desc
-	0x0F, // Capabilities  (rfidler 0x04,  pm3 0x02,  zero should also work  )
+	0x02, // Capabilities  (rfidler 0x04,  pm3 0x02,  zero should also work  )
 
 	/* Union Functional Descriptor */
 	0x05, // Function Length
@@ -150,7 +160,7 @@ static const char cfgDescriptor[] = {
 	0x05, // Function Length
 	0x24, // Descriptor Type: CS_INTERFACE
 	0x01, // Descriptor Subtype: Call Management Func Desc 
-	0x03, // Capabilities: D1 + D0   (ice 0x03, pm3 0x00, rfidler 0x00)
+	0x00, // Capabilities: D1 + D0   (ice 0x03, pm3 0x00, rfidler 0x00)
 	0x01, // Data Interface: Data Class Interface 1
 
 	/* Endpoint descriptor */
@@ -158,8 +168,8 @@ static const char cfgDescriptor[] = {
 	USB_DESCRIPTOR_ENDPOINT, // Descriptor Type
 	_EP03_IN,   // EndpointAddress, Endpoint 03-IN
 	_INTERRUPT, // Attributes
-	0x00, 0x02, // MaxPacket Size  (ice 0x200, pm3 0x08)
-	0xff,		// Interval polling (rfidler 0x02,  pm3 0xff)
+	0x40, 0x00, // MaxPacket Size  (ice 0x200, pm3 0x08)
+	0x02,		// Interval polling (rfidler 0x02,  pm3 0xff)
 
 	/* Data Class Interface Descriptor Requirement */
 	0x09, // Length
@@ -177,7 +187,7 @@ static const char cfgDescriptor[] = {
 	USB_DESCRIPTOR_ENDPOINT, // Descriptor Type
 	_EP01_OUT, 	// Endpoint Address, Endpoint 01-OUT
 	_BULK,   	// Attributes      BULK
-	0x00, 0x02, // MaxPacket Size	
+	0x40, 0x00, // MaxPacket Size	
 	0x00,   	// Interval
 
 	/* Endpoint descriptor */
@@ -185,7 +195,7 @@ static const char cfgDescriptor[] = {
 	USB_DESCRIPTOR_ENDPOINT, // DescriptorType
 	_EP02_IN,   // Endpoint Address, Endpoint 02-IN
 	_BULK,   	// Attributes      BULK
-	0x00, 0x02, // MaxPacket Size	
+	0x40, 0x00, // MaxPacket Size	
 	0x00    	// Interval
 };
 
@@ -417,8 +427,8 @@ AT91S_CDC_LINE_CODING line = {
 	8};     // 8 Data bits
 
 AT91PS_UDP pUdp = AT91C_BASE_UDP;
-byte_t btConfiguration = 0;
-byte_t btConnection    = 0;
+uint16_t btConfiguration = 0;
+uint16_t btConnection    = 0;
 byte_t btReceiveBank   = AT91C_UDP_RX_DATA_BK0;
 
 //*----------------------------------------------------------------------------
@@ -430,7 +440,7 @@ void usb_disable() {
 	AT91C_BASE_PIOA->PIO_ODR = GPIO_USB_PU;
 
 	// Clear all lingering interrupts
-	if(pUdp->UDP_ISR & AT91C_UDP_ENDBUSRES) {
+	if (pUdp->UDP_ISR & AT91C_UDP_ENDBUSRES) {
 		pUdp->UDP_ICR = AT91C_UDP_ENDBUSRES;
 	}
 }
@@ -528,6 +538,8 @@ bool usb_check() {
 	}
 	*/
 
+	WDT_HIT();
+	
 	// interrupt status register
 	AT91_REG isr = pUdp->UDP_ISR;
 
@@ -539,9 +551,8 @@ bool usb_check() {
 		pUdp->UDP_RSTEP  = 0;
 		// Enable the function
 		pUdp->UDP_FADDR = AT91C_UDP_FEN;
-		// Configure endpoint 0
+		// Configure endpoint 0  (enable control endpoint)
 		UDP_SET_EP_FLAGS(AT91C_EP_CONTROL, (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_CTRL));
-		//pUdp->UDP_CSR[AT91C_EP_CONTROL] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_CTRL);
 		// clear it
 		pUdp->UDP_ICR |= AT91C_UDP_ENDBUSRES;
 	}
@@ -588,7 +599,7 @@ uint32_t usb_read(byte_t* data, size_t len) {
 		if ( pUdp->UDP_CSR[AT91C_EP_OUT] & bank ) {
 			packetSize = MIN(pUdp->UDP_CSR[AT91C_EP_OUT] >> 16, len);
 			len -= packetSize;
-			while(packetSize--)
+			while (packetSize--)
 				data[nbBytesRcv++] = pUdp->UDP_FDR[AT91C_EP_OUT];
 
 			UDP_CLEAR_EP_FLAGS(AT91C_EP_OUT, bank)			
@@ -655,10 +666,9 @@ uint32_t usb_write(const byte_t* data, const size_t len) {
 //*----------------------------------------------------------------------------
 void AT91F_USB_SendData(AT91PS_UDP pUdp, const char *pData, uint32_t length) {
 	uint32_t cpt = 0;
-	AT91_REG csr;
 
 	do {
-		cpt = MIN(length, 0x40);
+		cpt = MIN(length, 8);
 		length -= cpt;
 
 		while (cpt--)
@@ -672,16 +682,13 @@ void AT91F_USB_SendData(AT91PS_UDP pUdp, const char *pData, uint32_t length) {
 		UDP_SET_EP_FLAGS(AT91C_EP_CONTROL, AT91C_UDP_TXPKTRDY)
 
 		do {
-			csr = pUdp->UDP_CSR[AT91C_EP_CONTROL];
-
 			// Data IN stage has been stopped by a status OUT
-			if (csr & AT91C_UDP_RX_DATA_BK0) {
+			if ( pUdp->UDP_CSR[AT91C_EP_CONTROL] & AT91C_UDP_RX_DATA_BK0) {
 				
 				UDP_CLEAR_EP_FLAGS(AT91C_EP_CONTROL, AT91C_UDP_RX_DATA_BK0)
-
 				return;
 			}
-		} while ( !(csr & AT91C_UDP_TXCOMP) );
+		} while ( !( pUdp->UDP_CSR[AT91C_EP_CONTROL] & AT91C_UDP_TXCOMP) );
 
 	} while (length);
 
@@ -706,6 +713,7 @@ void AT91F_USB_SendZlp(AT91PS_UDP pUdp) {
 //*----------------------------------------------------------------------------
 void AT91F_USB_SendStall(AT91PS_UDP pUdp) {
 	UDP_SET_EP_FLAGS(AT91C_EP_CONTROL, AT91C_UDP_FORCESTALL);
+	while ( !(pUdp->UDP_CSR[AT91C_EP_CONTROL] & AT91C_UDP_ISOERROR) );
 	UDP_CLEAR_EP_FLAGS(AT91C_EP_CONTROL, (AT91C_UDP_FORCESTALL | AT91C_UDP_ISOERROR));	
 }
 
@@ -734,26 +742,26 @@ void AT91F_CDC_Enumerate() {
 	}
 	UDP_CLEAR_EP_FLAGS(AT91C_EP_CONTROL, AT91C_UDP_RXSETUP)
 
-	// if ( bRequest == MS_VENDOR_CODE) {
-		// if ( bmRequestType == MS_WCID_GET_DESCRIPTOR ) { // C0
+	if ( bRequest == MS_VENDOR_CODE) {
+		if ( bmRequestType == MS_WCID_GET_DESCRIPTOR ) { // C0
 		
-			// if ( wIndex == MS_EXTENDED_COMPAT_ID ) {  // 4
-				// AT91F_USB_SendData(pUdp, CompatIDFeatureDescriptor, MIN(sizeof(CompatIDFeatureDescriptor), wLength));
-			// } else {
-				// AT91F_USB_SendStall(pUdp);
-			// }
-			// return;
-		// }
-		// if ( bmRequestType == MS_WCID_GET_FEATURE_DESCRIPTOR ) {  //C1
-			// if ( wIndex == MS_EXTENDED_PROPERTIES ) { // 5
-				// AT91F_USB_SendData(pUdp, (char *)&OSPropertyDescriptor, MIN(sizeof(OSPropertyDescriptor), wLength));
-				// AT91F_USB_SendData(pUdp, OSprop, MIN(sizeof(OSprop), wLength));
-			// } else {
-				// AT91F_USB_SendStall(pUdp);
-			// }
-			// return;
-		// }
-	// }
+			if ( wIndex == MS_EXTENDED_COMPAT_ID ) {  // 4
+				AT91F_USB_SendData(pUdp, CompatIDFeatureDescriptor, MIN(sizeof(CompatIDFeatureDescriptor), wLength));
+			} else {
+				AT91F_USB_SendStall(pUdp);
+			}
+			return;
+		}
+		if ( bmRequestType == MS_WCID_GET_FEATURE_DESCRIPTOR ) {  //C1
+			if ( wIndex == MS_EXTENDED_PROPERTIES ) { // 5
+				AT91F_USB_SendData(pUdp, (char *)&OSPropertyDescriptor, MIN(sizeof(OSPropertyDescriptor), wLength));
+				AT91F_USB_SendData(pUdp, OSprop, MIN(sizeof(OSprop), wLength));
+			} else {
+				AT91F_USB_SendStall(pUdp);
+			}
+			return;
+		}
+	}
 	// Handle supported standard device request Cf Table 9-3 in USB specification Rev 1.1
 	switch ((bRequest << 8) | bmRequestType) {
 	case STD_GET_DESCRIPTOR:
@@ -782,9 +790,13 @@ void AT91F_CDC_Enumerate() {
 		btConfiguration = wValue;
 		AT91F_USB_SendZlp(pUdp);
 		pUdp->UDP_GLBSTATE  = (wValue) ? AT91C_UDP_CONFG : AT91C_UDP_FADDEN;
-		pUdp->UDP_CSR[1] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT) : 0;
-		pUdp->UDP_CSR[2] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN)  : 0;
-		pUdp->UDP_CSR[3] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_INT_IN)   : 0;
+		
+		UDP_SET_EP_FLAGS(1, (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT) : 0 );
+		UDP_SET_EP_FLAGS(2, (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN)  : 0 ); 
+		UDP_SET_EP_FLAGS(3, (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_INT_IN)   : 0 );
+//		pUdp->UDP_CSR[1] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT) : 0;
+//		pUdp->UDP_CSR[2] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN)  : 0;
+//		pUdp->UDP_CSR[3] = (wValue) ? (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_INT_IN)   : 0;
 		break;
 	case STD_GET_CONFIGURATION:
 		AT91F_USB_SendData(pUdp, (char *) &(btConfiguration), sizeof(btConfiguration));
@@ -835,12 +847,18 @@ void AT91F_CDC_Enumerate() {
 	case STD_CLEAR_FEATURE_ENDPOINT:
 		wIndex &= 0x0F;
 		if ((wValue == 0) && wIndex && (wIndex <= 3)) {
-			if (wIndex == 1)
-				pUdp->UDP_CSR[1] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT);
-			else if (wIndex == 2)
-				pUdp->UDP_CSR[2] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN);
-			else if (wIndex == 3)
-				pUdp->UDP_CSR[3] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_ISO_IN);
+			if (wIndex == 1) {
+				//pUdp->UDP_CSR[1] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT);
+				UDP_SET_EP_FLAGS(1, (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_OUT) );
+			}
+			else if (wIndex == 2) {
+				//pUdp->UDP_CSR[2] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN);
+				UDP_SET_EP_FLAGS(2, (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_BULK_IN) );
+			}
+			else if (wIndex == 3) {
+				//pUdp->UDP_CSR[3] = (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_ISO_IN);
+				UDP_SET_EP_FLAGS(3, (AT91C_UDP_EPEDS | AT91C_UDP_EPTYPE_ISO_IN) );
+			}
 			AT91F_USB_SendZlp(pUdp);
 		} else {
 			AT91F_USB_SendStall(pUdp);
@@ -849,6 +867,7 @@ void AT91F_CDC_Enumerate() {
 
 	// handle CDC class requests
 	case SET_LINE_CODING:
+		// ignor SET_LINE_CODING...
 		while ( !(pUdp->UDP_CSR[AT91C_EP_CONTROL] & AT91C_UDP_RX_DATA_BK0) );
 		UDP_CLEAR_EP_FLAGS(AT91C_EP_CONTROL, AT91C_UDP_RX_DATA_BK0);
 		AT91F_USB_SendZlp(pUdp);