mirror of
https://github.com/arduino/Arduino.git
synced 2025-01-17 06:52:18 +01:00
Merge branch 'master' into upload-16u2
Conflicts: hardware/arduino/sam/cores/arduino/USB/CDC.cpp hardware/arduino/sam/cores/arduino/USB/USBCore.cpp
This commit is contained in:
commit
d21f734357
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
#ifdef CDC_ENABLED
|
#ifdef CDC_ENABLED
|
||||||
|
|
||||||
#define CDC_SERIAL_BUFFER_SIZE 64
|
#define CDC_SERIAL_BUFFER_SIZE 512
|
||||||
|
|
||||||
/* For information purpose only since RTS is not always handled by the terminal application */
|
/* For information purpose only since RTS is not always handled by the terminal application */
|
||||||
#define CDC_LINESTATE_DTR 0x01 // Data Terminal Ready
|
#define CDC_LINESTATE_DTR 0x01 // Data Terminal Ready
|
||||||
@ -46,7 +46,13 @@ typedef struct
|
|||||||
uint8_t lineState;
|
uint8_t lineState;
|
||||||
} LineInfo;
|
} LineInfo;
|
||||||
|
|
||||||
static volatile LineInfo _usbLineInfo = { 57600, 0x00, 0x00, 0x00, 0x00 };
|
static volatile LineInfo _usbLineInfo = {
|
||||||
|
57600, // dWDTERate
|
||||||
|
0x00, // bCharFormat
|
||||||
|
0x00, // bParityType
|
||||||
|
0x08, // bDataBits
|
||||||
|
0x00 // lineState
|
||||||
|
};
|
||||||
|
|
||||||
_Pragma("pack(1)")
|
_Pragma("pack(1)")
|
||||||
static const CDCDescriptor _cdcInterface =
|
static const CDCDescriptor _cdcInterface =
|
||||||
@ -59,12 +65,29 @@ static const CDCDescriptor _cdcInterface =
|
|||||||
D_CDCCS(CDC_CALL_MANAGEMENT,1,1), // Device handles call management (not)
|
D_CDCCS(CDC_CALL_MANAGEMENT,1,1), // Device handles call management (not)
|
||||||
D_CDCCS4(CDC_ABSTRACT_CONTROL_MANAGEMENT,6), // SET_LINE_CODING, GET_LINE_CODING, SET_CONTROL_LINE_STATE supported
|
D_CDCCS4(CDC_ABSTRACT_CONTROL_MANAGEMENT,6), // SET_LINE_CODING, GET_LINE_CODING, SET_CONTROL_LINE_STATE supported
|
||||||
D_CDCCS(CDC_UNION,CDC_ACM_INTERFACE,CDC_DATA_INTERFACE), // Communication interface is master, data interface is slave 0
|
D_CDCCS(CDC_UNION,CDC_ACM_INTERFACE,CDC_DATA_INTERFACE), // Communication interface is master, data interface is slave 0
|
||||||
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_ACM),USB_ENDPOINT_TYPE_INTERRUPT,0x10,0x40),
|
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_ACM),USB_ENDPOINT_TYPE_INTERRUPT,0x10, 0x10),
|
||||||
|
|
||||||
// CDC data interface
|
// CDC data interface
|
||||||
D_INTERFACE(CDC_DATA_INTERFACE,2,CDC_DATA_INTERFACE_CLASS,0,0),
|
D_INTERFACE(CDC_DATA_INTERFACE,2,CDC_DATA_INTERFACE_CLASS,0,0),
|
||||||
D_ENDPOINT(USB_ENDPOINT_OUT(CDC_ENDPOINT_OUT),USB_ENDPOINT_TYPE_BULK,0x40,0),
|
D_ENDPOINT(USB_ENDPOINT_OUT(CDC_ENDPOINT_OUT),USB_ENDPOINT_TYPE_BULK,512,0),
|
||||||
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_IN ),USB_ENDPOINT_TYPE_BULK,0x40,0)
|
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_IN ),USB_ENDPOINT_TYPE_BULK,512,0)
|
||||||
|
};
|
||||||
|
static const CDCDescriptor _cdcOtherInterface =
|
||||||
|
{
|
||||||
|
D_IAD(0,2,CDC_COMMUNICATION_INTERFACE_CLASS,CDC_ABSTRACT_CONTROL_MODEL,1),
|
||||||
|
|
||||||
|
// CDC communication interface
|
||||||
|
D_INTERFACE(CDC_ACM_INTERFACE,1,CDC_COMMUNICATION_INTERFACE_CLASS,CDC_ABSTRACT_CONTROL_MODEL,0),
|
||||||
|
D_CDCCS(CDC_HEADER,0x10,0x01), // Header (1.10 bcd)
|
||||||
|
D_CDCCS(CDC_CALL_MANAGEMENT,1,1), // Device handles call management (not)
|
||||||
|
D_CDCCS4(CDC_ABSTRACT_CONTROL_MANAGEMENT,6), // SET_LINE_CODING, GET_LINE_CODING, SET_CONTROL_LINE_STATE supported
|
||||||
|
D_CDCCS(CDC_UNION,CDC_ACM_INTERFACE,CDC_DATA_INTERFACE), // Communication interface is master, data interface is slave 0
|
||||||
|
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_ACM),USB_ENDPOINT_TYPE_INTERRUPT,0x10, 0x10),
|
||||||
|
|
||||||
|
// CDC data interface
|
||||||
|
D_INTERFACE(CDC_DATA_INTERFACE,2,CDC_DATA_INTERFACE_CLASS,0,0),
|
||||||
|
D_ENDPOINT(USB_ENDPOINT_OUT(CDC_ENDPOINT_OUT),USB_ENDPOINT_TYPE_BULK,64,0),
|
||||||
|
D_ENDPOINT(USB_ENDPOINT_IN (CDC_ENDPOINT_IN ),USB_ENDPOINT_TYPE_BULK,64,0)
|
||||||
};
|
};
|
||||||
_Pragma("pack()")
|
_Pragma("pack()")
|
||||||
|
|
||||||
@ -74,6 +97,12 @@ int WEAK CDC_GetInterface(uint8_t* interfaceNum)
|
|||||||
return USBD_SendControl(0,&_cdcInterface,sizeof(_cdcInterface));
|
return USBD_SendControl(0,&_cdcInterface,sizeof(_cdcInterface));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int WEAK CDC_GetOtherInterface(uint8_t* interfaceNum)
|
||||||
|
{
|
||||||
|
interfaceNum[0] += 2; // uses 2
|
||||||
|
return USBD_SendControl(0,&_cdcOtherInterface,sizeof(_cdcOtherInterface));
|
||||||
|
}
|
||||||
|
|
||||||
bool WEAK CDC_Setup(Setup& setup)
|
bool WEAK CDC_Setup(Setup& setup)
|
||||||
{
|
{
|
||||||
uint8_t r = setup.bRequest;
|
uint8_t r = setup.bRequest;
|
||||||
@ -182,7 +211,7 @@ void Serial_::flush(void)
|
|||||||
USBD_Flush(CDC_TX);
|
USBD_Flush(CDC_TX);
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t Serial_::write(uint8_t c)
|
size_t Serial_::write(const uint8_t *buffer, size_t size)
|
||||||
{
|
{
|
||||||
/* only try to send bytes if the high-level CDC connection itself
|
/* only try to send bytes if the high-level CDC connection itself
|
||||||
is open (not just the pipe) - the OS should set lineState when the port
|
is open (not just the pipe) - the OS should set lineState when the port
|
||||||
@ -195,10 +224,10 @@ size_t Serial_::write(uint8_t c)
|
|||||||
// or locks up, or host virtual serial port hangs)
|
// or locks up, or host virtual serial port hangs)
|
||||||
if (_usbLineInfo.lineState > 0)
|
if (_usbLineInfo.lineState > 0)
|
||||||
{
|
{
|
||||||
int r = USBD_Send(CDC_TX,&c,1);
|
int r = USBD_Send(CDC_TX, buffer, size);
|
||||||
|
|
||||||
if (r > 0)
|
if (r > 0)
|
||||||
{
|
{
|
||||||
return r;
|
return r;
|
||||||
} else
|
} else
|
||||||
{
|
{
|
||||||
@ -210,6 +239,10 @@ size_t Serial_::write(uint8_t c)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t Serial_::write(uint8_t c) {
|
||||||
|
return write(&c, 1);
|
||||||
|
}
|
||||||
|
|
||||||
// This operator is a convenient way for a sketch to check whether the
|
// This operator is a convenient way for a sketch to check whether the
|
||||||
// port has actually been configured and opened by the host (as opposed
|
// port has actually been configured and opened by the host (as opposed
|
||||||
// to just being connected to the host). It can be used, for example, in
|
// to just being connected to the host). It can be used, for example, in
|
||||||
|
@ -57,7 +57,8 @@ public:
|
|||||||
virtual int read(void);
|
virtual int read(void);
|
||||||
virtual void flush(void);
|
virtual void flush(void);
|
||||||
virtual size_t write(uint8_t);
|
virtual size_t write(uint8_t);
|
||||||
using Print::write; // pull in write(str) and write(buf, size) from Print
|
virtual size_t write(const uint8_t *buffer, size_t size);
|
||||||
|
using Print::write; // pull in write(str) from Print
|
||||||
operator bool();
|
operator bool();
|
||||||
};
|
};
|
||||||
extern Serial_ Serial;
|
extern Serial_ Serial;
|
||||||
@ -190,6 +191,7 @@ bool MSC_Data(uint8_t rx,uint8_t tx);
|
|||||||
// CSC 'Driver'
|
// CSC 'Driver'
|
||||||
|
|
||||||
int CDC_GetInterface(uint8_t* interfaceNum);
|
int CDC_GetInterface(uint8_t* interfaceNum);
|
||||||
|
int CDC_GetOtherInterface(uint8_t* interfaceNum);
|
||||||
int CDC_GetDescriptor(int i);
|
int CDC_GetDescriptor(int i);
|
||||||
bool CDC_Setup(Setup& setup);
|
bool CDC_Setup(Setup& setup);
|
||||||
|
|
||||||
|
@ -27,9 +27,9 @@ static const uint32_t EndPoints[] =
|
|||||||
EP_TYPE_CONTROL,
|
EP_TYPE_CONTROL,
|
||||||
|
|
||||||
#ifdef CDC_ENABLED
|
#ifdef CDC_ENABLED
|
||||||
EP_TYPE_BULK_IN, // CDC_ENDPOINT_IN
|
|
||||||
EP_TYPE_BULK_OUT, // CDC_ENDPOINT_OUT
|
|
||||||
EP_TYPE_INTERRUPT_IN, // CDC_ENDPOINT_ACM
|
EP_TYPE_INTERRUPT_IN, // CDC_ENDPOINT_ACM
|
||||||
|
EP_TYPE_BULK_OUT, // CDC_ENDPOINT_OUT
|
||||||
|
EP_TYPE_BULK_IN, // CDC_ENDPOINT_IN
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef HID_ENABLED
|
#ifdef HID_ENABLED
|
||||||
@ -41,7 +41,8 @@ static const uint32_t EndPoints[] =
|
|||||||
#define TX_RX_LED_PULSE_MS 100
|
#define TX_RX_LED_PULSE_MS 100
|
||||||
volatile uint8_t TxLEDPulse; /**< Milliseconds remaining for data Tx LED pulse */
|
volatile uint8_t TxLEDPulse; /**< Milliseconds remaining for data Tx LED pulse */
|
||||||
volatile uint8_t RxLEDPulse; /**< Milliseconds remaining for data Rx LED pulse */
|
volatile uint8_t RxLEDPulse; /**< Milliseconds remaining for data Rx LED pulse */
|
||||||
|
static char isRemoteWakeUpEnabled = 0;
|
||||||
|
static char isEndpointHalt = 0;
|
||||||
//==================================================================
|
//==================================================================
|
||||||
//==================================================================
|
//==================================================================
|
||||||
|
|
||||||
@ -87,11 +88,16 @@ const DeviceDescriptor USB_DeviceDescriptor =
|
|||||||
const DeviceDescriptor USB_DeviceDescriptorA =
|
const DeviceDescriptor USB_DeviceDescriptorA =
|
||||||
D_DEVICE(DEVICE_CLASS,0x00,0x00,64,USB_VID,USB_PID,0x100,IMANUFACTURER,IPRODUCT,0,1);
|
D_DEVICE(DEVICE_CLASS,0x00,0x00,64,USB_VID,USB_PID,0x100,IMANUFACTURER,IPRODUCT,0,1);
|
||||||
|
|
||||||
|
const DeviceDescriptor USB_DeviceQualifier =
|
||||||
|
D_QUALIFIER(0x00,0x00,0x00,64,1);
|
||||||
|
|
||||||
|
|
||||||
//==================================================================
|
//==================================================================
|
||||||
//==================================================================
|
//==================================================================
|
||||||
|
|
||||||
volatile uint32_t _usbConfiguration = 0;
|
volatile uint32_t _usbConfiguration = 0;
|
||||||
volatile uint32_t _usbInitialized = 0;
|
volatile uint32_t _usbInitialized = 0;
|
||||||
|
uint32_t _usbSetInterface = 0;
|
||||||
uint32_t _cdcComposite = 0;
|
uint32_t _cdcComposite = 0;
|
||||||
|
|
||||||
//==================================================================
|
//==================================================================
|
||||||
@ -149,44 +155,42 @@ uint32_t USBD_Recv(uint32_t ep)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Space in send EP
|
// Space in send EP
|
||||||
uint32_t USBD_SendSpace(uint32_t ep)
|
//uint32_t USBD_SendSpace(uint32_t ep)
|
||||||
{
|
//{
|
||||||
LockEP lock(ep);
|
//LockEP lock(ep);
|
||||||
if (!UDD_ReadWriteAllowed(ep & 0xF))
|
//// if (!UDD_ReadWriteAllowed(ep & 0xF))
|
||||||
return 0;
|
////{
|
||||||
return 64 - UDD_FifoByteCount(ep & 0xF);
|
////printf("pb "); // UOTGHS->UOTGHS_DEVEPTISR[%d]=0x%X\n\r", ep, UOTGHS->UOTGHS_DEVEPTISR[ep]);
|
||||||
}
|
////return 0;
|
||||||
|
////}
|
||||||
|
|
||||||
|
//if(ep==0) return 64 - UDD_FifoByteCount(ep & 0xF); // EP0_SIZE jcb
|
||||||
|
//else return 512 - UDD_FifoByteCount(ep & 0xF); // EPX_SIZE jcb
|
||||||
|
//}
|
||||||
|
|
||||||
// Blocking Send of data to an endpoint
|
// Blocking Send of data to an endpoint
|
||||||
uint32_t USBD_Send(uint32_t ep, const void* d, uint32_t len)
|
uint32_t USBD_Send(uint32_t ep, const void* d, uint32_t len)
|
||||||
{
|
{
|
||||||
if (!_usbConfiguration)
|
uint32_t n;
|
||||||
return -1;
|
|
||||||
|
|
||||||
int r = len;
|
int r = len;
|
||||||
const uint8_t* data = (const uint8_t*)d;
|
const uint8_t* data = (const uint8_t*)d;
|
||||||
uint8_t timeout = 250; // 250ms timeout on send? TODO
|
|
||||||
|
if (!_usbConfiguration)
|
||||||
|
{
|
||||||
|
printf("pb conf\n\r");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
while (len)
|
while (len)
|
||||||
{
|
{
|
||||||
uint8_t n = USBD_SendSpace(ep);
|
if(ep==0) n= EP0_SIZE;
|
||||||
if (n == 0)
|
else n = EPX_SIZE;
|
||||||
{
|
|
||||||
if (!(--timeout))
|
|
||||||
return -1;
|
|
||||||
delay(1);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (n > len)
|
if (n > len)
|
||||||
n = len;
|
n = len;
|
||||||
len -= n;
|
len -= n;
|
||||||
|
|
||||||
UDD_Send(ep & 0xF, data, n);
|
UDD_Send(ep & 0xF, data, n);
|
||||||
|
}
|
||||||
if (!UDD_ReadWriteAllowed(ep & 0xF) || ((len == 0) && (ep & TRANSFER_RELEASE))) // Release full buffer
|
|
||||||
UDD_ReleaseTX(ep & 0xF);
|
|
||||||
}
|
|
||||||
//TXLED1; // light the TX LED
|
//TXLED1; // light the TX LED
|
||||||
//TxLEDPulse = TX_RX_LED_PULSE_MS;
|
//TxLEDPulse = TX_RX_LED_PULSE_MS;
|
||||||
return r;
|
return r;
|
||||||
@ -281,6 +285,24 @@ int USBD_SendInterfaces(void)
|
|||||||
return interfaces;
|
return interfaces;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int USBD_SendOtherInterfaces(void)
|
||||||
|
{
|
||||||
|
int total = 0;
|
||||||
|
uint8_t interfaces = 0;
|
||||||
|
|
||||||
|
#ifdef CDC_ENABLED
|
||||||
|
total = CDC_GetOtherInterface(&interfaces);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef HID_ENABLED
|
||||||
|
total += HID_GetInterface(&interfaces);
|
||||||
|
#endif
|
||||||
|
|
||||||
|
total = total; // Get rid of compiler warning
|
||||||
|
TRACE_CORE(printf("=> USBD_SendInterfaces, total=%d interfaces=%d\r\n", total, interfaces);)
|
||||||
|
return interfaces;
|
||||||
|
}
|
||||||
|
|
||||||
// Construct a dynamic configuration descriptor
|
// Construct a dynamic configuration descriptor
|
||||||
// This really needs dynamic endpoint allocation etc
|
// This really needs dynamic endpoint allocation etc
|
||||||
// TODO
|
// TODO
|
||||||
@ -307,6 +329,29 @@ _Pragma("pack()")
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool USBD_SendOtherConfiguration(int maxlen)
|
||||||
|
{
|
||||||
|
// Count and measure interfaces
|
||||||
|
USBD_InitControl(0);
|
||||||
|
//TRACE_CORE(printf("=> USBD_SendConfiguration _cmark1=%d\r\n", _cmark);)
|
||||||
|
int interfaces = USBD_SendOtherInterfaces();
|
||||||
|
//TRACE_CORE(printf("=> USBD_SendConfiguration _cmark2=%d\r\n", _cmark);)
|
||||||
|
//TRACE_CORE(printf("=> USBD_SendConfiguration sizeof=%d\r\n", sizeof(ConfigDescriptor));)
|
||||||
|
|
||||||
|
_Pragma("pack(1)")
|
||||||
|
ConfigDescriptor config = D_OTHERCONFIG(_cmark + sizeof(ConfigDescriptor),interfaces);
|
||||||
|
_Pragma("pack()")
|
||||||
|
//TRACE_CORE(printf("=> USBD_SendConfiguration clen=%d\r\n", config.clen);)
|
||||||
|
|
||||||
|
//TRACE_CORE(printf("=> USBD_SendConfiguration maxlen=%d\r\n", maxlen);)
|
||||||
|
|
||||||
|
// Now send them
|
||||||
|
USBD_InitControl(maxlen);
|
||||||
|
USBD_SendControl(0,&config,sizeof(ConfigDescriptor));
|
||||||
|
USBD_SendOtherInterfaces();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static bool USBD_SendDescriptor(Setup& setup)
|
static bool USBD_SendDescriptor(Setup& setup)
|
||||||
{
|
{
|
||||||
uint8_t t = setup.wValueH;
|
uint8_t t = setup.wValueH;
|
||||||
@ -340,15 +385,39 @@ static bool USBD_SendDescriptor(Setup& setup)
|
|||||||
else if (USB_STRING_DESCRIPTOR_TYPE == t)
|
else if (USB_STRING_DESCRIPTOR_TYPE == t)
|
||||||
{
|
{
|
||||||
TRACE_CORE(puts("=> USBD_SendDescriptor : USB_STRING_DESCRIPTOR_TYPE\r\n");)
|
TRACE_CORE(puts("=> USBD_SendDescriptor : USB_STRING_DESCRIPTOR_TYPE\r\n");)
|
||||||
if (setup.wValueL == 0)
|
if (setup.wValueL == 0) {
|
||||||
desc_addr = (const uint8_t*)&STRING_LANGUAGE;
|
desc_addr = (const uint8_t*)&STRING_LANGUAGE;
|
||||||
else if (setup.wValueL == IPRODUCT)
|
}
|
||||||
|
else if (setup.wValueL == IPRODUCT) {
|
||||||
desc_addr = (const uint8_t*)&STRING_IPRODUCT;
|
desc_addr = (const uint8_t*)&STRING_IPRODUCT;
|
||||||
else if (setup.wValueL == IMANUFACTURER)
|
}
|
||||||
|
else if (setup.wValueL == IMANUFACTURER) {
|
||||||
desc_addr = (const uint8_t*)&STRING_IMANUFACTURER;
|
desc_addr = (const uint8_t*)&STRING_IMANUFACTURER;
|
||||||
else
|
}
|
||||||
|
else {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
if( *desc_addr > setup.wLength ) {
|
||||||
|
desc_length = setup.wLength;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
else if (USB_DEVICE_QUALIFIER == t)
|
||||||
|
{
|
||||||
|
// Device qualifier descriptor requested
|
||||||
|
desc_addr = (const uint8_t*)&USB_DeviceQualifier;
|
||||||
|
if( *desc_addr > setup.wLength ) {
|
||||||
|
desc_length = setup.wLength;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (USB_OTHER_SPEED_CONFIGURATION == t)
|
||||||
|
{
|
||||||
|
// Other configuration descriptor requested
|
||||||
|
return USBD_SendOtherConfiguration(setup.wLength);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
//printf("Device ERROR");
|
||||||
|
}
|
||||||
|
|
||||||
if (desc_addr == 0)
|
if (desc_addr == 0)
|
||||||
{
|
{
|
||||||
@ -366,12 +435,15 @@ static bool USBD_SendDescriptor(Setup& setup)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//unsigned int iii=0;
|
||||||
// Endpoint 0 interrupt
|
// Endpoint 0 interrupt
|
||||||
static void USB_ISR(void)
|
static void USB_ISR(void)
|
||||||
{
|
{
|
||||||
Reset.tick();
|
Reset.tick();
|
||||||
|
|
||||||
// End of Reset
|
// printf("ISR=0x%X\n\r", UOTGHS->UOTGHS_DEVISR); // jcb
|
||||||
|
// if( iii++ > 1500 ) while(1); // jcb
|
||||||
|
// End of bus reset
|
||||||
if (Is_udd_reset())
|
if (Is_udd_reset())
|
||||||
{
|
{
|
||||||
TRACE_CORE(printf(">>> End of Reset\r\n");)
|
TRACE_CORE(printf(">>> End of Reset\r\n");)
|
||||||
@ -404,14 +476,13 @@ static void USB_ISR(void)
|
|||||||
if (Is_udd_sof())
|
if (Is_udd_sof())
|
||||||
{
|
{
|
||||||
udd_ack_sof();
|
udd_ack_sof();
|
||||||
USBD_Flush(CDC_TX);
|
// USBD_Flush(CDC_TX); // jcb
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// EP 0 Interrupt
|
// EP 0 Interrupt
|
||||||
if (Is_udd_endpoint_interrupt(0))
|
if (Is_udd_endpoint_interrupt(0) )
|
||||||
{
|
{
|
||||||
|
|
||||||
if (!UDD_ReceivedSetupInt())
|
if (!UDD_ReceivedSetupInt())
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
@ -440,15 +511,70 @@ static void USB_ISR(void)
|
|||||||
uint8_t r = setup.bRequest;
|
uint8_t r = setup.bRequest;
|
||||||
if (GET_STATUS == r)
|
if (GET_STATUS == r)
|
||||||
{
|
{
|
||||||
TRACE_CORE(puts(">>> EP0 Int: GET_STATUS\r\n");)
|
if( setup.bmRequestType == 0 ) // device
|
||||||
UDD_Send8(EP0, 0); // TODO
|
{
|
||||||
UDD_Send8(EP0, 0);
|
// Send the device status
|
||||||
|
TRACE_CORE(puts(">>> EP0 Int: GET_STATUS\r\n");)
|
||||||
|
// Check current configuration for power mode (if device is configured)
|
||||||
|
// TODO
|
||||||
|
// Check if remote wake-up is enabled
|
||||||
|
// TODO
|
||||||
|
UDD_Send8(EP0, 0); // TODO
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
|
// if( setup.bmRequestType == 2 ) // Endpoint:
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Send the endpoint status
|
||||||
|
// Check if the endpoint if currently halted
|
||||||
|
if( isEndpointHalt == 1 )
|
||||||
|
UDD_Send8(EP0, 1); // TODO
|
||||||
|
else
|
||||||
|
UDD_Send8(EP0, 0); // TODO
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (CLEAR_FEATURE == r)
|
else if (CLEAR_FEATURE == r)
|
||||||
{
|
{
|
||||||
}
|
// Check which is the selected feature
|
||||||
|
if( setup.wValueL == 1) // DEVICEREMOTEWAKEUP
|
||||||
|
{
|
||||||
|
// Enable remote wake-up and send a ZLP
|
||||||
|
if( isRemoteWakeUpEnabled == 1 )
|
||||||
|
UDD_Send8(EP0, 1);
|
||||||
|
else
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
|
else // if( setup.wValueL == 0) // ENDPOINTHALT
|
||||||
|
{
|
||||||
|
isEndpointHalt = 0; // TODO
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
else if (SET_FEATURE == r)
|
else if (SET_FEATURE == r)
|
||||||
{
|
{
|
||||||
|
// Check which is the selected feature
|
||||||
|
if( setup.wValueL == 1) // DEVICEREMOTEWAKEUP
|
||||||
|
{
|
||||||
|
// Enable remote wake-up and send a ZLP
|
||||||
|
isRemoteWakeUpEnabled = 1;
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
|
if( setup.wValueL == 0) // ENDPOINTHALT
|
||||||
|
{
|
||||||
|
// Halt endpoint
|
||||||
|
isEndpointHalt = 1;
|
||||||
|
//USBD_Halt(USBGenericRequest_GetEndpointNumber(pRequest));
|
||||||
|
UDD_Send8(EP0, 0);
|
||||||
|
}
|
||||||
|
if( setup.wValueL == 2) // TEST_MODE
|
||||||
|
{
|
||||||
|
// 7.1.20 Test Mode Support, 9.4.9 SetFeature
|
||||||
|
// TODO
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (SET_ADDRESS == r)
|
else if (SET_ADDRESS == r)
|
||||||
{
|
{
|
||||||
@ -469,7 +595,7 @@ static void USB_ISR(void)
|
|||||||
else if (GET_CONFIGURATION == r)
|
else if (GET_CONFIGURATION == r)
|
||||||
{
|
{
|
||||||
TRACE_CORE(puts(">>> EP0 Int: GET_CONFIGURATION\r\n");)
|
TRACE_CORE(puts(">>> EP0 Int: GET_CONFIGURATION\r\n");)
|
||||||
UDD_Send8(EP0, 1);
|
UDD_Send8(EP0, _usbConfiguration);
|
||||||
}
|
}
|
||||||
else if (SET_CONFIGURATION == r)
|
else if (SET_CONFIGURATION == r)
|
||||||
{
|
{
|
||||||
@ -495,9 +621,11 @@ static void USB_ISR(void)
|
|||||||
else if (GET_INTERFACE == r)
|
else if (GET_INTERFACE == r)
|
||||||
{
|
{
|
||||||
TRACE_CORE(puts(">>> EP0 Int: GET_INTERFACE\r\n");)
|
TRACE_CORE(puts(">>> EP0 Int: GET_INTERFACE\r\n");)
|
||||||
|
UDD_Send8(EP0, _usbSetInterface);
|
||||||
}
|
}
|
||||||
else if (SET_INTERFACE == r)
|
else if (SET_INTERFACE == r)
|
||||||
{
|
{
|
||||||
|
_usbSetInterface = setup.wValueL;
|
||||||
TRACE_CORE(puts(">>> EP0 Int: SET_INTERFACE\r\n");)
|
TRACE_CORE(puts(">>> EP0 Int: SET_INTERFACE\r\n");)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -77,6 +77,8 @@
|
|||||||
#define USB_STRING_DESCRIPTOR_TYPE 3
|
#define USB_STRING_DESCRIPTOR_TYPE 3
|
||||||
#define USB_INTERFACE_DESCRIPTOR_TYPE 4
|
#define USB_INTERFACE_DESCRIPTOR_TYPE 4
|
||||||
#define USB_ENDPOINT_DESCRIPTOR_TYPE 5
|
#define USB_ENDPOINT_DESCRIPTOR_TYPE 5
|
||||||
|
#define USB_DEVICE_QUALIFIER 6
|
||||||
|
#define USB_OTHER_SPEED_CONFIGURATION 7
|
||||||
|
|
||||||
#define USB_DEVICE_CLASS_COMMUNICATIONS 0x02
|
#define USB_DEVICE_CLASS_COMMUNICATIONS 0x02
|
||||||
#define USB_DEVICE_CLASS_HUMAN_INTERFACE 0x03
|
#define USB_DEVICE_CLASS_HUMAN_INTERFACE 0x03
|
||||||
@ -285,12 +287,18 @@ _Pragma("pack()")
|
|||||||
#define D_CONFIG(_totalLength,_interfaces) \
|
#define D_CONFIG(_totalLength,_interfaces) \
|
||||||
{ 9, 2, _totalLength,_interfaces, 1, 0, USB_CONFIG_SELF_POWERED, USB_CONFIG_POWER_MA(500) }
|
{ 9, 2, _totalLength,_interfaces, 1, 0, USB_CONFIG_SELF_POWERED, USB_CONFIG_POWER_MA(500) }
|
||||||
|
|
||||||
|
#define D_OTHERCONFIG(_totalLength,_interfaces) \
|
||||||
|
{ 9, 7, _totalLength,_interfaces, 1, 0, USB_CONFIG_SELF_POWERED, USB_CONFIG_POWER_MA(500) }
|
||||||
|
|
||||||
#define D_INTERFACE(_n,_numEndpoints,_class,_subClass,_protocol) \
|
#define D_INTERFACE(_n,_numEndpoints,_class,_subClass,_protocol) \
|
||||||
{ 9, 4, _n, 0, _numEndpoints, _class,_subClass, _protocol, 0 }
|
{ 9, 4, _n, 0, _numEndpoints, _class,_subClass, _protocol, 0 }
|
||||||
|
|
||||||
#define D_ENDPOINT(_addr,_attr,_packetSize, _interval) \
|
#define D_ENDPOINT(_addr,_attr,_packetSize, _interval) \
|
||||||
{ 7, 5, _addr,_attr,_packetSize, _interval }
|
{ 7, 5, _addr,_attr,_packetSize, _interval }
|
||||||
|
|
||||||
|
#define D_QUALIFIER(_class,_subClass,_proto,_packetSize0,_configs) \
|
||||||
|
{ 10, 6, 0x200, _class,_subClass,_proto,_packetSize0,_configs }
|
||||||
|
|
||||||
#define D_IAD(_firstInterface, _count, _class, _subClass, _protocol) \
|
#define D_IAD(_firstInterface, _count, _class, _subClass, _protocol) \
|
||||||
{ 8, 11, _firstInterface, _count, _class, _subClass, _protocol, 0 }
|
{ 8, 11, _firstInterface, _count, _class, _subClass, _protocol, 0 }
|
||||||
|
|
||||||
|
@ -34,7 +34,7 @@
|
|||||||
#define MAX_ENDPOINTS 10
|
#define MAX_ENDPOINTS 10
|
||||||
#define EP0 0
|
#define EP0 0
|
||||||
#define EP0_SIZE 64
|
#define EP0_SIZE 64
|
||||||
#define EPX_SIZE 1024
|
#define EPX_SIZE 512
|
||||||
|
|
||||||
#define EP_SINGLE_64 (0x32UL) // EP0
|
#define EP_SINGLE_64 (0x32UL) // EP0
|
||||||
#define EP_DOUBLE_64 (0x36UL) // Other endpoints
|
#define EP_DOUBLE_64 (0x36UL) // Other endpoints
|
||||||
@ -116,7 +116,7 @@
|
|||||||
//! Get maximal number of banks of endpoints
|
//! Get maximal number of banks of endpoints
|
||||||
#define udd_get_endpoint_bank_max_nbr(ep) ((ep == 0) ? 1 : (( ep <= 2) ? 3 : 2))
|
#define udd_get_endpoint_bank_max_nbr(ep) ((ep == 0) ? 1 : (( ep <= 2) ? 3 : 2))
|
||||||
//! Get maximal size of endpoint (3X, 1024/64)
|
//! Get maximal size of endpoint (3X, 1024/64)
|
||||||
#define udd_get_endpoint_size_max(ep) (((ep) == 0) ? 64 : 1024)
|
#define udd_get_endpoint_size_max(ep) (((ep) == 0) ? 64 : 512) // for bulk
|
||||||
//! Get DMA support of endpoints
|
//! Get DMA support of endpoints
|
||||||
#define Is_udd_endpoint_dma_supported(ep) ((((ep) >= 1) && ((ep) <= 6)) ? true : false)
|
#define Is_udd_endpoint_dma_supported(ep) ((((ep) >= 1) && ((ep) <= 6)) ? true : false)
|
||||||
//! Get High Band Width support of endpoints
|
//! Get High Band Width support of endpoints
|
||||||
|
@ -85,7 +85,7 @@ uint32_t UDD_Init(void)
|
|||||||
|
|
||||||
// Enable High Speed
|
// Enable High Speed
|
||||||
udd_low_speed_disable();
|
udd_low_speed_disable();
|
||||||
udd_high_speed_disable();
|
udd_high_speed_enable();
|
||||||
|
|
||||||
//otg_ack_vbus_transition();
|
//otg_ack_vbus_transition();
|
||||||
// Force Vbus interrupt in case of Vbus always with a high level
|
// Force Vbus interrupt in case of Vbus always with a high level
|
||||||
@ -115,7 +115,7 @@ void UDD_Attach(void)
|
|||||||
|
|
||||||
// Enable USB line events
|
// Enable USB line events
|
||||||
udd_enable_reset_interrupt();
|
udd_enable_reset_interrupt();
|
||||||
udd_enable_sof_interrupt();
|
// udd_enable_sof_interrupt();
|
||||||
|
|
||||||
cpu_irq_restore(flags);
|
cpu_irq_restore(flags);
|
||||||
}
|
}
|
||||||
@ -139,6 +139,7 @@ void UDD_InitEP( uint32_t ul_ep_nb, uint32_t ul_ep_cfg )
|
|||||||
|
|
||||||
if (!Is_udd_endpoint_configured(ul_ep_nb)) {
|
if (!Is_udd_endpoint_configured(ul_ep_nb)) {
|
||||||
TRACE_UOTGHS_DEVICE(printf("=> UDD_InitEP : ERROR FAILED TO INIT EP %lu\r\n", ul_ep_nb);)
|
TRACE_UOTGHS_DEVICE(printf("=> UDD_InitEP : ERROR FAILED TO INIT EP %lu\r\n", ul_ep_nb);)
|
||||||
|
while(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -156,6 +157,7 @@ void UDD_InitEndpoints(const uint32_t* eps_table, const uint32_t ul_eps_table_si
|
|||||||
|
|
||||||
if (!Is_udd_endpoint_configured(ul_ep_nb)) {
|
if (!Is_udd_endpoint_configured(ul_ep_nb)) {
|
||||||
TRACE_UOTGHS_DEVICE(printf("=> UDD_InitEP : ERROR FAILED TO INIT EP %lu\r\n", ul_ep_nb);)
|
TRACE_UOTGHS_DEVICE(printf("=> UDD_InitEP : ERROR FAILED TO INIT EP %lu\r\n", ul_ep_nb);)
|
||||||
|
while(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -215,6 +217,8 @@ uint32_t UDD_Send(uint32_t ep, const void* data, uint32_t len)
|
|||||||
|
|
||||||
TRACE_UOTGHS_DEVICE(printf("=> UDD_Send (1): ep=%lu ul_send_fifo_ptr=%lu len=%lu\r\n", ep, ul_send_fifo_ptr[ep], len);)
|
TRACE_UOTGHS_DEVICE(printf("=> UDD_Send (1): ep=%lu ul_send_fifo_ptr=%lu len=%lu\r\n", ep, ul_send_fifo_ptr[ep], len);)
|
||||||
|
|
||||||
|
while( UOTGHS_DEVEPTISR_TXINI != (UOTGHS->UOTGHS_DEVEPTISR[ep] & UOTGHS_DEVEPTISR_TXINI )) {}
|
||||||
|
|
||||||
if (ep == EP0)
|
if (ep == EP0)
|
||||||
{
|
{
|
||||||
if (ul_send_fifo_ptr[ep] + len > EP0_SIZE)
|
if (ul_send_fifo_ptr[ep] + len > EP0_SIZE)
|
||||||
@ -222,34 +226,26 @@ uint32_t UDD_Send(uint32_t ep, const void* data, uint32_t len)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (ul_send_fifo_ptr[ep] + len > EPX_SIZE)
|
ul_send_fifo_ptr[ep] = 0;
|
||||||
len = EPX_SIZE - ul_send_fifo_ptr[ep];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
for (i = 0, ptr_dest += ul_send_fifo_ptr[ep]; i < len; ++i)
|
for (i = 0, ptr_dest += ul_send_fifo_ptr[ep]; i < len; ++i)
|
||||||
*ptr_dest++ = *ptr_src++;
|
*ptr_dest++ = *ptr_src++;
|
||||||
|
|
||||||
ul_send_fifo_ptr[ep] += i;
|
ul_send_fifo_ptr[ep] += i;
|
||||||
|
|
||||||
|
|
||||||
if (ep == EP0)
|
if (ep == EP0)
|
||||||
{
|
{
|
||||||
TRACE_UOTGHS_DEVICE(printf("=> UDD_Send (2): ep=%lu ptr_dest=%lu maxlen=%d\r\n", ep, ul_send_fifo_ptr[ep], EP0_SIZE);)
|
TRACE_UOTGHS_DEVICE(printf("=> UDD_Send (2): ep=%lu ptr_dest=%lu maxlen=%d\r\n", ep, ul_send_fifo_ptr[ep], EP0_SIZE);)
|
||||||
if (ul_send_fifo_ptr[ep] == EP0_SIZE)
|
if (ul_send_fifo_ptr[ep] == EP0_SIZE)
|
||||||
{
|
{
|
||||||
UDD_ClearIN(); // Fifo is full, release this packet
|
UDD_ClearIN(); // Fifo is full, release this packet // UOTGHS->UOTGHS_DEVEPTICR[EP0] = UOTGHS_DEVEPTICR_TXINIC;
|
||||||
UDD_WaitIN(); // Wait for new FIFO buffer to be ready
|
}
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (ul_send_fifo_ptr[ep] == EPX_SIZE)
|
UOTGHS->UOTGHS_DEVEPTICR[ep] = UOTGHS_DEVEPTICR_TXINIC;
|
||||||
{
|
UOTGHS->UOTGHS_DEVEPTIDR[ep] = UOTGHS_DEVEPTIDR_FIFOCONC;
|
||||||
UDD_ClearIN(); // Fifo is full, release this packet
|
|
||||||
UDD_WaitIN(); // Wait for new FIFO buffer to be ready
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return len;
|
return len;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -301,7 +297,8 @@ uint32_t UDD_FifoByteCount(uint32_t ep)
|
|||||||
void UDD_ReleaseRX(uint32_t ep)
|
void UDD_ReleaseRX(uint32_t ep)
|
||||||
{
|
{
|
||||||
TRACE_UOTGHS_DEVICE(puts("=> UDD_ReleaseRX\r\n");)
|
TRACE_UOTGHS_DEVICE(puts("=> UDD_ReleaseRX\r\n");)
|
||||||
UOTGHS->UOTGHS_DEVEPTICR[ep] = (UOTGHS_DEVEPTICR_NAKOUTIC | UOTGHS_DEVEPTICR_RXOUTIC);
|
// UOTGHS->UOTGHS_DEVEPTICR[ep] = (UOTGHS_DEVEPTICR_NAKOUTIC | UOTGHS_DEVEPTICR_RXOUTIC);
|
||||||
|
UOTGHS->UOTGHS_DEVEPTICR[ep] = UOTGHS_DEVEPTICR_RXOUTIC;
|
||||||
UOTGHS->UOTGHS_DEVEPTIDR[ep] = UOTGHS_DEVEPTIDR_FIFOCONC;
|
UOTGHS->UOTGHS_DEVEPTIDR[ep] = UOTGHS_DEVEPTIDR_FIFOCONC;
|
||||||
ul_recv_fifo_ptr[ep] = 0;
|
ul_recv_fifo_ptr[ep] = 0;
|
||||||
}
|
}
|
||||||
@ -309,7 +306,8 @@ void UDD_ReleaseRX(uint32_t ep)
|
|||||||
void UDD_ReleaseTX(uint32_t ep)
|
void UDD_ReleaseTX(uint32_t ep)
|
||||||
{
|
{
|
||||||
TRACE_UOTGHS_DEVICE(printf("=> UDD_ReleaseTX ep=%lu\r\n", ep);)
|
TRACE_UOTGHS_DEVICE(printf("=> UDD_ReleaseTX ep=%lu\r\n", ep);)
|
||||||
UOTGHS->UOTGHS_DEVEPTICR[ep] = (UOTGHS_DEVEPTICR_NAKINIC | UOTGHS_DEVEPTICR_RXOUTIC | UOTGHS_DEVEPTICR_TXINIC);
|
// UOTGHS->UOTGHS_DEVEPTICR[ep] = (UOTGHS_DEVEPTICR_NAKINIC | UOTGHS_DEVEPTICR_RXOUTIC | UOTGHS_DEVEPTICR_TXINIC);
|
||||||
|
UOTGHS->UOTGHS_DEVEPTICR[ep] = UOTGHS_DEVEPTICR_TXINIC;
|
||||||
UOTGHS->UOTGHS_DEVEPTIDR[ep] = UOTGHS_DEVEPTIDR_FIFOCONC;
|
UOTGHS->UOTGHS_DEVEPTIDR[ep] = UOTGHS_DEVEPTIDR_FIFOCONC;
|
||||||
ul_send_fifo_ptr[ep] = 0;
|
ul_send_fifo_ptr[ep] = 0;
|
||||||
}
|
}
|
||||||
|
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user