2009-07-12 15:52:28 +00:00
|
|
|
#include <usb-arch.h>
|
|
|
|
#include <gpio.h>
|
|
|
|
#include <nvic.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <debug-uart.h>
|
|
|
|
#include <usb-stm32f103.h>
|
|
|
|
|
|
|
|
/* #define DEBUG */
|
|
|
|
#ifdef DEBUG
|
|
|
|
#define PRINTF(...) printf(__VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define PRINTF(...)
|
|
|
|
#endif
|
|
|
|
typedef struct
|
|
|
|
{
|
|
|
|
vu32 EPR[8];
|
|
|
|
u32 RESERVED[8];
|
|
|
|
vu32 CNTR;
|
|
|
|
vu32 ISTR;
|
|
|
|
vu32 FNR;
|
|
|
|
vu32 DADDR;
|
|
|
|
vu32 BTABLE;
|
|
|
|
} USB_TypeDef;
|
|
|
|
|
|
|
|
#define USB_BASE (APB1PERIPH_BASE + 0x5c00)
|
|
|
|
#define USB_MEM_BASE (APB1PERIPH_BASE + 0x6000)
|
|
|
|
#define USB_MEM_SIZE (512)
|
|
|
|
#define USB ((volatile USB_TypeDef *) USB_BASE)
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
vu32 ADDR_TX;
|
|
|
|
vu32 COUNT_TX;
|
|
|
|
vu32 ADDR_RX;
|
|
|
|
vu32 COUNT_RX;
|
|
|
|
} USB_HW_Buffer;
|
|
|
|
|
|
|
|
#define USB_EP0_BUF_SIZE (2*CTRL_EP_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP1_BUF_SIZE (2*USB_EP1_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP2_BUF_SIZE (2*USB_EP2_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP3_BUF_SIZE (2*USB_EP3_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP4_BUF_SIZE (2*USB_EP4_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP5_BUF_SIZE (2*USB_EP5_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP6_BUF_SIZE (2*USB_EP6_SIZE)
|
|
|
|
|
|
|
|
#define USB_EP7_BUF_SIZE (2*USB_EP7_SIZE)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef MAX_CTRL_DATA
|
|
|
|
#define MAX_CTRL_DATA 128
|
|
|
|
#endif
|
|
|
|
/* Double buffered IN endpoint */
|
|
|
|
#define ADDR_TX_0 ADDR_TX
|
|
|
|
#define ADDR_TX_1 ADDR_RX
|
|
|
|
#define COUNT_TX_0 COUNT_TX
|
|
|
|
#define COUNT_TX_1 COUNT_RX
|
|
|
|
|
|
|
|
/* Double buffered OUT endpoint */
|
|
|
|
#define ADDR_RX_0 ADDR_TX
|
|
|
|
#define ADDR_RX_1 ADDR_RX
|
|
|
|
#define COUNT_RX_0 COUNT_TX
|
|
|
|
#define COUNT_RX_1 COUNT_RX
|
|
|
|
|
|
|
|
#define USB_EPxR_EP_TYPE_BULK 0
|
|
|
|
#define USB_EPxR_EP_TYPE_CONTROL USB_EP0R_EP_TYPE_0
|
|
|
|
#define USB_EPxR_EP_TYPE_ISO USB_EP0R_EP_TYPE_1
|
|
|
|
#define USB_EPxR_EP_TYPE_INTERRUPT (USB_EP0R_EP_TYPE_1|USB_EP0R_EP_TYPE_0)
|
|
|
|
|
|
|
|
#define USB_EPxR_EP_DBL_BUF USB_EP0R_EP_KIND
|
|
|
|
#define USB_EPxR_EP_STATUS_OUT USB_EP0R_EP_KIND
|
|
|
|
|
|
|
|
#define USB_EPxR_STAT_RX_DISABLED 0
|
|
|
|
#define USB_EPxR_STAT_RX_STALL USB_EP0R_STAT_RX_0
|
|
|
|
#define USB_EPxR_STAT_RX_NAK USB_EP0R_STAT_RX_1
|
|
|
|
#define USB_EPxR_STAT_RX_VALID (USB_EP0R_STAT_RX_1|USB_EP0R_STAT_RX_0)
|
|
|
|
|
|
|
|
#define USB_EPxR_STAT_TX_DISABLED 0
|
|
|
|
#define USB_EPxR_STAT_TX_STALL USB_EP0R_STAT_TX_0
|
|
|
|
#define USB_EPxR_STAT_TX_NAK USB_EP0R_STAT_TX_1
|
|
|
|
#define USB_EPxR_STAT_TX_VALID (USB_EP0R_STAT_TX_1|USB_EP0R_STAT_TX_0)
|
|
|
|
|
|
|
|
#define USB_EPxR_SW_BUF_TX USB_EP0R_DTOG_RX
|
|
|
|
#define USB_EPxR_SW_BUF_RX USB_EP0R_DTOG_TX
|
|
|
|
|
|
|
|
static const uint16_t ep_buffer_size[8] =
|
|
|
|
{
|
|
|
|
USB_EP0_BUF_SIZE,
|
|
|
|
USB_EP1_BUF_SIZE,
|
|
|
|
USB_EP2_BUF_SIZE,
|
|
|
|
USB_EP3_BUF_SIZE,
|
|
|
|
USB_EP4_BUF_SIZE,
|
|
|
|
USB_EP5_BUF_SIZE,
|
|
|
|
USB_EP6_BUF_SIZE,
|
|
|
|
USB_EP7_BUF_SIZE
|
|
|
|
};
|
|
|
|
|
|
|
|
#define USB_EP_BUF_SIZE(ep) ep_buffer_size[ep]
|
|
|
|
#define USB_EP_BUF_OFFSET(ep) ep_buffer_offset[ep]
|
|
|
|
#define USB_EP_BUF_ADDR(ep) (u32*)(USB_MEM_BASE + ep_buffer_offset[ep]*2);
|
|
|
|
#define USB_EP_BUF_DESC(ep) ((USB_HW_Buffer*)(USB_MEM_BASE + 16 * (ep)))
|
|
|
|
|
|
|
|
#define USB_EP0_OFFSET (8*USB_MAX_ENDPOINTS)
|
|
|
|
#define USB_EP1_OFFSET (USB_EP0_OFFSET + USB_EP0_BUF_SIZE)
|
|
|
|
#define USB_EP2_OFFSET (USB_EP1_OFFSET + USB_EP1_BUF_SIZE)
|
|
|
|
#define USB_EP3_OFFSET (USB_EP2_OFFSET + USB_EP2_BUF_SIZE)
|
|
|
|
#define USB_EP4_OFFSET (USB_EP3_OFFSET + USB_EP3_BUF_SIZE)
|
|
|
|
#define USB_EP5_OFFSET (USB_EP4_OFFSET + USB_EP4_BUF_SIZE)
|
|
|
|
#define USB_EP6_OFFSET (USB_EP5_OFFSET + USB_EP5_BUF_SIZE)
|
|
|
|
#define USB_EP7_OFFSET (USB_EP6_OFFSET + USB_EP6_BUF_SIZE)
|
|
|
|
|
|
|
|
#if (USB_EP7_OFFSET+USB_EP7_BUF_SIZE) > USB_MEM_SIZE
|
|
|
|
#error USB endpoints buffers does not fit in USB memory
|
|
|
|
#endif
|
|
|
|
static const uint16_t ep_buffer_offset[8] =
|
|
|
|
{
|
|
|
|
USB_EP0_OFFSET,
|
|
|
|
USB_EP1_OFFSET,
|
|
|
|
USB_EP2_OFFSET,
|
|
|
|
USB_EP3_OFFSET,
|
|
|
|
USB_EP4_OFFSET,
|
|
|
|
USB_EP5_OFFSET,
|
|
|
|
USB_EP6_OFFSET,
|
|
|
|
USB_EP7_OFFSET
|
|
|
|
};
|
|
|
|
|
|
|
|
#define USB_EP_BUF_CAPACITY(s) ((((s) <64)?((s)/2):(0x20 | ((s)/64)))<<10)
|
|
|
|
|
|
|
|
typedef struct _USBEndpoint USBEndpoint;
|
|
|
|
struct _USBEndpoint
|
|
|
|
{
|
|
|
|
uint16_t status;
|
|
|
|
uint8_t addr;
|
|
|
|
uint8_t flags;
|
|
|
|
USBBuffer *buffer; /* NULL if no current buffer */
|
|
|
|
struct process *event_process;
|
|
|
|
uint16_t events;
|
|
|
|
uint16_t xfer_size;
|
|
|
|
};
|
|
|
|
|
|
|
|
#define USB_EP_FLAGS_TYPE_MASK 0x03
|
|
|
|
#define USB_EP_FLAGS_TYPE_BULK 0x00
|
|
|
|
#define USB_EP_FLAGS_TYPE_CONTROL 0x01
|
|
|
|
#define USB_EP_FLAGS_TYPE_ISO 0x02
|
|
|
|
#define USB_EP_FLAGS_TYPE_INTERRUPT 0x03
|
|
|
|
|
|
|
|
#define IS_EP_TYPE(ep, type) (((ep)->flags & USB_EP_FLAGS_TYPE_MASK) == (type))
|
|
|
|
#define IS_CONTROL_EP(ep) IS_EP_TYPE(ep, USB_EP_FLAGS_TYPE_CONTROL)
|
|
|
|
#define IS_BULK_EP(ep) IS_EP_TYPE(ep, USB_EP_FLAGS_TYPE_BULK)
|
|
|
|
#define IS_INTERRUPT_EP(ep) IS_EP_TYPE(ep, USB_EP_FLAGS_TYPE_INTERRUPT)
|
|
|
|
|
|
|
|
#define USB_EP_FLAGS_ENABLED 0x04
|
|
|
|
|
|
|
|
/* A packet has been received but the data is still in hardware buffer */
|
|
|
|
#define USB_EP_FLAGS_RECV_PENDING 0x08
|
|
|
|
/* The pending packet is a SETUP packet */
|
|
|
|
#define USB_EP_FLAGS_SETUP_PENDING 0x10
|
|
|
|
|
|
|
|
/* The data in the hardware buffer is being transmitted */
|
|
|
|
#define USB_EP_FLAGS_TRANSMITTING 0x20
|
|
|
|
|
|
|
|
/* The receiver is waiting for a packet */
|
|
|
|
#define USB_EP_FLAGS_RECEIVING 0x40
|
|
|
|
|
|
|
|
/* For bulk endpoints. Both buffers are busy are in use, either by
|
|
|
|
hardware or software. */
|
|
|
|
#define USB_EP_FLAGS_DOUBLE 0x80
|
|
|
|
|
|
|
|
/* States for double buffered reception:
|
|
|
|
|
|
|
|
Packets being received 0 1 2 1 0 0
|
|
|
|
Packets pending 0 0 0 1 2 1
|
|
|
|
|
|
|
|
RECVING 0 1 1 1 0 0
|
|
|
|
RECV_PENDING 0 0 0 1 1 1
|
|
|
|
DOUBLE 0 0 1 0 1 0
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* States for double buffered transmission:
|
|
|
|
|
|
|
|
Packets being transmitted 0 1 2
|
|
|
|
|
|
|
|
TRANSMITTING 0 1 1
|
|
|
|
DOUBLE 0 0 1
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Index in endpoint array */
|
|
|
|
#define EP_INDEX(addr) ((addr) & 0x7f)
|
|
|
|
|
|
|
|
/* Get address of endpoint struct */
|
|
|
|
#define EP_STRUCT(addr) (&usb_endpoints[EP_INDEX(addr)])
|
|
|
|
|
|
|
|
/* Number of hardware endpoint */
|
|
|
|
#define EP_HW_NUM(addr) ((addr) & 0x7f)
|
|
|
|
|
|
|
|
#define USB_DISABLE_INT \
|
|
|
|
NVIC_DISABLE_INT(USB_LP_CAN_RX0_IRQChannel);\
|
|
|
|
NVIC_DISABLE_INT(USB_HP_CAN_TX_IRQChannel)
|
|
|
|
|
|
|
|
#define USB_ENABLE_INT \
|
|
|
|
NVIC_ENABLE_INT(USB_LP_CAN_RX0_IRQChannel);\
|
|
|
|
NVIC_ENABLE_INT(USB_HP_CAN_TX_IRQChannel)
|
|
|
|
|
|
|
|
static inline uint32_t
|
|
|
|
usb_save_disable_int()
|
|
|
|
{
|
|
|
|
uint32_t v = NVIC->ISER[0];
|
|
|
|
NVIC->ICER[0] = (1<<USB_HP_CAN_TX_IRQChannel | 1<<USB_LP_CAN_RX0_IRQChannel);
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void
|
|
|
|
usb_restore_int(uint32_t v)
|
|
|
|
{
|
|
|
|
NVIC->ISER[0] =
|
|
|
|
v & (1<<USB_HP_CAN_TX_IRQChannel | 1<<USB_LP_CAN_RX0_IRQChannel);
|
|
|
|
}
|
|
|
|
|
|
|
|
static USBEndpoint usb_endpoints[USB_MAX_ENDPOINTS];
|
|
|
|
struct process *event_process = 0;
|
|
|
|
volatile unsigned int events = 0;
|
|
|
|
|
|
|
|
static void
|
|
|
|
notify_process(unsigned int e)
|
|
|
|
{
|
|
|
|
events |= e;
|
|
|
|
if (event_process) {
|
|
|
|
process_poll(event_process);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
notify_ep_process(USBEndpoint *ep, unsigned int e)
|
|
|
|
{
|
|
|
|
ep->events |= e;
|
|
|
|
if (ep->event_process) {
|
|
|
|
process_poll(ep->event_process);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
usb_arch_reset(void)
|
|
|
|
{
|
|
|
|
unsigned int e;
|
|
|
|
for (e = 0; e < USB_MAX_ENDPOINTS; e++) {
|
|
|
|
if (usb_endpoints[e].flags &USB_EP_FLAGS_ENABLED) {
|
|
|
|
USBBuffer *buffer = usb_endpoints[e].buffer;
|
|
|
|
usb_endpoints[e].flags = 0;
|
|
|
|
while(buffer) {
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
usb_arch_setup_control_endpoint(0);
|
|
|
|
USB->DADDR = 0x80;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_setup(void)
|
|
|
|
{
|
|
|
|
unsigned int i;
|
|
|
|
RCC->APB1RSTR |= RCC_APB1RSTR_USBRST;
|
|
|
|
RCC->APB2ENR |= (RCC_APB2ENR_AFIOEN | RCC_APB2ENR_IOPAEN);
|
|
|
|
RCC->APB1ENR |= (RCC_APB1ENR_USBEN);
|
|
|
|
RCC->APB1RSTR &= ~RCC_APB1RSTR_USBRST;
|
|
|
|
|
|
|
|
GPIO_CONF_OUTPUT_PORT(A,11,ALT_PUSH_PULL,50);
|
|
|
|
GPIO_CONF_OUTPUT_PORT(A,12,ALT_PUSH_PULL,50);
|
|
|
|
GPIO_CONF_OUTPUT_PORT(A,10, PUSH_PULL, 2);
|
|
|
|
GPIOA->BSRR = GPIO_BSRR_BR10;
|
|
|
|
|
|
|
|
/* Turn on analog part */
|
|
|
|
USB->CNTR &= ~USB_CNTR_PDWN;
|
|
|
|
|
|
|
|
for (i = 0; i < 24; i++) asm("nop"::); /* Wait at least 1us */
|
|
|
|
/* Release reset */
|
|
|
|
USB->CNTR &= ~USB_CNTR_FRES;
|
|
|
|
/* Clear any interrupts */
|
|
|
|
USB->ISTR = ~(USB_ISTR_PMAOVR |USB_ISTR_ERR | USB_ISTR_WKUP | USB_ISTR_SUSP
|
|
|
|
| USB_ISTR_RESET);
|
|
|
|
|
|
|
|
for(i = 0; i < USB_MAX_ENDPOINTS; i++) {
|
|
|
|
usb_endpoints[i].flags = 0;
|
|
|
|
usb_endpoints[i].event_process = 0;
|
|
|
|
}
|
|
|
|
/* Put buffer table at beginning of buffer memory */
|
|
|
|
USB->BTABLE = 0;
|
|
|
|
usb_arch_reset();
|
|
|
|
GPIOA->BSRR = GPIO_BSRR_BS10;
|
|
|
|
USB->CNTR |= (USB_CNTR_CTRM | USB_CNTR_PMAOVRM | USB_CNTR_ERRM
|
|
|
|
| USB_CNTR_WKUPM| USB_CNTR_SUSPM | USB_CNTR_RESETM);
|
|
|
|
NVIC_SET_PRIORITY(USB_LP_CAN_RX0_IRQChannel, 4);
|
|
|
|
NVIC_ENABLE_INT(USB_LP_CAN_RX0_IRQChannel);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define EPR_RW (USB_EP0R_EP_TYPE|USB_EP0R_EP_KIND|USB_EP0R_EA)
|
|
|
|
#define EPR_W0 (USB_EP0R_CTR_RX|USB_EP0R_CTR_TX)
|
|
|
|
#define EPR_TOGGLE (USB_EP0R_DTOG_RX | USB_EP0R_STAT_RX \
|
|
|
|
| USB_EP0R_DTOG_TX | USB_EP0R_STAT_TX)
|
|
|
|
|
|
|
|
#define EPR_INVARIANT(epr) ((epr & (EPR_RW)) | EPR_W0)
|
|
|
|
|
|
|
|
#define EPR_TOGGLE_SET(epr, mask, set) \
|
|
|
|
((((epr) & (EPR_RW | (mask))) | EPR_W0) ^ (set))
|
|
|
|
|
|
|
|
static void
|
|
|
|
usb_arch_setup_endpoint(unsigned char addr)
|
|
|
|
{
|
|
|
|
USBEndpoint *ep = EP_STRUCT(addr);
|
|
|
|
ep->status = 0;
|
|
|
|
ep->flags = USB_EP_FLAGS_ENABLED;
|
|
|
|
ep->buffer = 0;
|
|
|
|
ep->addr = addr;
|
|
|
|
ep->events = 0;
|
|
|
|
ep->xfer_size = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_setup_control_endpoint(unsigned char addr)
|
|
|
|
{
|
|
|
|
USB_HW_Buffer *buf_desc;
|
|
|
|
unsigned int ei = EP_HW_NUM(addr);
|
|
|
|
unsigned int epr;
|
|
|
|
USBEndpoint *ep = EP_STRUCT(addr);
|
|
|
|
usb_arch_setup_endpoint(addr);
|
|
|
|
ep->flags |= USB_EP_FLAGS_TYPE_CONTROL;
|
|
|
|
|
|
|
|
buf_desc = USB_EP_BUF_DESC(ei);
|
|
|
|
buf_desc->ADDR_TX = USB_EP_BUF_OFFSET(ei);
|
|
|
|
buf_desc->COUNT_TX = USB_EP_BUF_SIZE(ei)/2;
|
|
|
|
buf_desc->ADDR_RX = USB_EP_BUF_OFFSET(ei) + USB_EP_BUF_SIZE(ei)/2;
|
|
|
|
buf_desc->COUNT_RX = USB_EP_BUF_CAPACITY(USB_EP_BUF_SIZE(ei)/2);
|
|
|
|
ep->xfer_size = USB_EP_BUF_SIZE(ei)/2;
|
|
|
|
epr = USB->EPR[EP_HW_NUM(addr)];
|
|
|
|
/* Clear interrupt flags */
|
|
|
|
epr &= ~(USB_EP0R_CTR_RX | USB_EP0R_CTR_TX);
|
|
|
|
/* NACK both directions */
|
|
|
|
epr ^= USB_EPxR_STAT_RX_NAK | USB_EPxR_STAT_TX_NAK;
|
|
|
|
/* Set control type */
|
|
|
|
epr = ((epr & ~(USB_EP0R_EP_TYPE | USB_EP0R_EP_KIND))
|
|
|
|
| USB_EPxR_EP_TYPE_CONTROL);
|
|
|
|
/* Set address */
|
|
|
|
epr = ((epr & ~USB_EP0R_EA) | addr);
|
|
|
|
USB->EPR[EP_HW_NUM(addr)] = epr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_setup_bulk_endpoint(unsigned char addr)
|
|
|
|
{
|
|
|
|
USB_HW_Buffer *buf_desc;
|
|
|
|
unsigned int ei = EP_HW_NUM(addr);
|
|
|
|
unsigned int epr;
|
|
|
|
USBEndpoint *ep = EP_STRUCT(addr);
|
|
|
|
usb_arch_setup_endpoint(addr);
|
|
|
|
ep->flags |= USB_EP_FLAGS_TYPE_BULK;
|
|
|
|
|
|
|
|
buf_desc = USB_EP_BUF_DESC(ei);
|
|
|
|
buf_desc->ADDR_TX = USB_EP_BUF_OFFSET(ei);
|
|
|
|
buf_desc->ADDR_RX = USB_EP_BUF_OFFSET(ei) + USB_EP_BUF_SIZE(ei)/2;
|
|
|
|
epr = USB->EPR[ei];
|
|
|
|
if (addr & 0x80) {
|
|
|
|
/* IN */
|
|
|
|
buf_desc->COUNT_TX_0 = 0;
|
|
|
|
buf_desc->COUNT_TX_1 = 0;
|
|
|
|
/* VALID transmission */
|
|
|
|
epr ^= USB_EPxR_STAT_TX_VALID;
|
|
|
|
} else {
|
|
|
|
/* OUT */
|
|
|
|
buf_desc->COUNT_RX_0 = USB_EP_BUF_CAPACITY(USB_EP_BUF_SIZE(ei)/2);
|
|
|
|
buf_desc->COUNT_RX_1 = USB_EP_BUF_CAPACITY(USB_EP_BUF_SIZE(ei)/2);
|
|
|
|
|
|
|
|
/* VALID reception */
|
|
|
|
epr ^= USB_EPxR_STAT_RX_VALID;
|
|
|
|
}
|
|
|
|
ep->xfer_size = USB_EP_BUF_SIZE(ei)/2;
|
|
|
|
/* Clear interrupt flags */
|
|
|
|
epr &= ~(USB_EP0R_CTR_RX | USB_EP0R_CTR_TX);
|
|
|
|
/* Set bulk type */
|
|
|
|
epr = ((epr & ~(USB_EP0R_EP_TYPE | USB_EP0R_EP_KIND))
|
|
|
|
| USB_EPxR_EP_TYPE_BULK | USB_EPxR_EP_DBL_BUF);
|
|
|
|
/* Set address */
|
|
|
|
epr = ((epr & ~USB_EP0R_EA) | addr);
|
|
|
|
USB->EPR[ei] = epr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_setup_interrupt_endpoint(unsigned char addr)
|
|
|
|
{
|
|
|
|
USB_HW_Buffer *buf_desc;
|
|
|
|
unsigned int ei = EP_HW_NUM(addr);
|
|
|
|
unsigned int epr;
|
|
|
|
USBEndpoint *ep = EP_STRUCT(addr);
|
|
|
|
usb_arch_setup_endpoint(addr);
|
|
|
|
ep->flags |= USB_EP_FLAGS_TYPE_INTERRUPT;
|
|
|
|
|
|
|
|
epr = USB->EPR[EP_HW_NUM(addr)];
|
|
|
|
|
|
|
|
buf_desc = USB_EP_BUF_DESC(ei);
|
|
|
|
if (addr & 0x80) {
|
|
|
|
/* IN */
|
|
|
|
buf_desc->ADDR_TX = USB_EP_BUF_OFFSET(ei);
|
|
|
|
buf_desc->COUNT_TX = USB_EP_BUF_SIZE(ei);
|
|
|
|
epr ^= USB_EPxR_STAT_TX_NAK;
|
|
|
|
} else {
|
|
|
|
/* OUT */
|
|
|
|
buf_desc->ADDR_RX = USB_EP_BUF_OFFSET(ei);
|
|
|
|
buf_desc->COUNT_RX = USB_EP_BUF_CAPACITY(USB_EP_BUF_SIZE(ei));
|
|
|
|
epr ^= USB_EPxR_STAT_RX_NAK;
|
|
|
|
}
|
|
|
|
ep->xfer_size = USB_EP_BUF_SIZE(ei);
|
|
|
|
/* Clear interrupt flags */
|
|
|
|
epr &= ~(USB_EP0R_CTR_RX | USB_EP0R_CTR_TX);
|
|
|
|
/* Set control type */
|
|
|
|
epr = ((epr & ~(USB_EP0R_EP_TYPE | USB_EP0R_EP_KIND))
|
|
|
|
| USB_EPxR_EP_TYPE_INTERRUPT);
|
|
|
|
/* Set address */
|
|
|
|
epr = ((epr & ~USB_EP0R_EA) | addr);
|
|
|
|
USB->EPR[EP_HW_NUM(addr)] = epr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_disable_endpoint(uint8_t addr)
|
|
|
|
{
|
|
|
|
unsigned int epr;
|
|
|
|
USBEndpoint *ep = EP_STRUCT(addr);
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_ENABLED;
|
|
|
|
|
|
|
|
epr = USB->EPR[EP_HW_NUM(addr)];
|
|
|
|
|
|
|
|
epr ^= USB_EPxR_STAT_TX_DISABLED | USB_EPxR_STAT_RX_DISABLED;
|
|
|
|
/* Clear interrupt flags */
|
|
|
|
epr &= ~(USB_EP0R_CTR_RX | USB_EP0R_CTR_TX);
|
|
|
|
USB->EPR[EP_HW_NUM(addr)] = epr;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stall_bulk_in(unsigned int hw_ep)
|
|
|
|
{
|
|
|
|
volatile uint32_t *eprp = &USB->EPR[hw_ep];
|
|
|
|
*eprp = (*eprp & (EPR_RW | USB_EP0R_STAT_TX_1)) | EPR_W0;
|
|
|
|
PRINTF("HALT IN\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stall_bulk_out(unsigned int hw_ep)
|
|
|
|
{
|
|
|
|
volatile uint32_t *eprp = &USB->EPR[hw_ep];
|
|
|
|
*eprp = (*eprp & ((EPR_RW | USB_EP0R_STAT_RX_1) & ~USB_EP0R_CTR_RX)) |EPR_W0;
|
|
|
|
PRINTF("HALT OUT\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define USB_READ_BLOCK 0x01 /* The currently submitted buffers
|
|
|
|
can't hold the received data, wait
|
|
|
|
for more buffers. No data was read
|
|
|
|
from the hardware buffer */
|
|
|
|
#define USB_READ_NOTIFY 0x02 /* Some buffers that had the
|
|
|
|
USB_BUFFER_NOTIFY flags set were
|
|
|
|
released */
|
|
|
|
#define USB_READ_FAIL 0x04 /* The received data doesn't match the
|
|
|
|
submitted buffers. The hardware
|
|
|
|
buffer is discarded. */
|
|
|
|
|
|
|
|
inline unsigned int
|
|
|
|
ep_capacity(unsigned int count)
|
|
|
|
{
|
|
|
|
return (((count & USB_COUNT0_RX_NUM_BLOCK)>>10)
|
|
|
|
* ((count & USB_COUNT0_RX_BLSIZE) ? 32 : 2));
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Skip buffers until mask and flags matches*/
|
|
|
|
static USBBuffer *
|
|
|
|
skip_buffers_until(USBBuffer *buffer, unsigned int mask, unsigned int flags,
|
|
|
|
unsigned int *resp)
|
|
|
|
{
|
|
|
|
while(buffer && !((buffer->flags & mask) == flags)) {
|
|
|
|
USBBuffer *next = buffer->next;
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED ;
|
|
|
|
buffer->flags |= USB_BUFFER_FAILED;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) *resp |= USB_READ_NOTIFY;
|
|
|
|
buffer = next;
|
|
|
|
}
|
|
|
|
return buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
read_hw_buffer(USBBuffer *buffer, unsigned int offset, unsigned int len)
|
|
|
|
{
|
|
|
|
#ifdef USB_STM32F103_ENABLE_ALT_COPY
|
|
|
|
if (buffer->flags & USB_BUFFER_ARCH_ALT_COPY) {
|
|
|
|
copy_from_hw_buffer(buffer, offset, len);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
uint8_t *data = buffer->data;
|
|
|
|
const uint32_t *hw_data = ((u32*)USB_MEM_BASE) + offset/2;
|
|
|
|
buffer->data += len;
|
|
|
|
if (offset & 1) {
|
|
|
|
*data++ = *hw_data++ >> 8;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
while(len >= 2) {
|
|
|
|
*((uint16_t*)data) = *hw_data++;
|
|
|
|
data += 2;
|
|
|
|
len -= 2;
|
|
|
|
}
|
|
|
|
if (len == 1) {
|
|
|
|
*data++ = *hw_data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#define USB_WRITE_BLOCK 0x01
|
|
|
|
#define USB_WRITE_NOTIFY 0x02
|
|
|
|
|
2010-02-05 16:23:40 +00:00
|
|
|
void
|
2009-07-12 15:52:28 +00:00
|
|
|
write_hw_buffer(USBBuffer *buffer,unsigned int offset, unsigned int len)
|
|
|
|
{
|
|
|
|
#ifdef USB_STM32F103_ENABLE_ALT_COPY
|
|
|
|
if (buffer->flags & USB_BUFFER_ARCH_ALT_COPY) {
|
|
|
|
copy_to_hw_buffer(buffer, offset, len);
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
{
|
2010-02-05 16:23:40 +00:00
|
|
|
const uint8_t *data;
|
|
|
|
uint32_t *hw_data;
|
|
|
|
if (len == 0) return;
|
|
|
|
data = buffer->data;
|
|
|
|
hw_data = ((u32*)USB_MEM_BASE) + offset/2;
|
2009-07-12 15:52:28 +00:00
|
|
|
buffer->data += len;
|
|
|
|
if (offset & 1) {
|
|
|
|
*hw_data = (*hw_data & 0xff) | (*data++ << 8);
|
|
|
|
hw_data++;
|
|
|
|
len--;
|
|
|
|
}
|
|
|
|
while(len >= 2) {
|
|
|
|
*hw_data++ = *((uint16_t*)data) ;
|
|
|
|
data += 2;
|
|
|
|
len -= 2;
|
|
|
|
}
|
|
|
|
if (len == 1) {
|
|
|
|
*hw_data = *data++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int
|
|
|
|
get_receive_capacity(USBBuffer *buffer)
|
|
|
|
{
|
|
|
|
unsigned int capacity = 0;
|
2010-02-05 16:23:40 +00:00
|
|
|
while(buffer && !(buffer->flags & (USB_BUFFER_IN| USB_BUFFER_SETUP|USB_BUFFER_HALT))) {
|
2009-07-12 15:52:28 +00:00
|
|
|
capacity += buffer->left;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
return capacity;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
handle_pending_receive(USBEndpoint *ep)
|
|
|
|
{
|
|
|
|
int short_packet;
|
|
|
|
unsigned int len;
|
|
|
|
unsigned int copy;
|
|
|
|
unsigned int res = 0;
|
|
|
|
unsigned int hw_offset;
|
|
|
|
unsigned int hw_ep = EP_HW_NUM(ep->addr);
|
|
|
|
USBBuffer *buffer = ep->buffer;
|
|
|
|
unsigned int flags = ep->flags;
|
|
|
|
USB_HW_Buffer *buf_desc = USB_EP_BUF_DESC(hw_ep);
|
|
|
|
PRINTF("handle_pending_receive:\n");
|
|
|
|
if (!(flags & USB_EP_FLAGS_ENABLED) || !buffer) return USB_READ_BLOCK;
|
|
|
|
switch(flags & USB_EP_FLAGS_TYPE_MASK) {
|
|
|
|
case USB_EP_FLAGS_TYPE_CONTROL:
|
|
|
|
len = buf_desc->COUNT_RX & USB_COUNT0_RX_COUNT0_RX;
|
|
|
|
if (flags & USB_EP_FLAGS_SETUP_PENDING) {
|
|
|
|
/* Discard buffers until we find a SETUP buffer */
|
|
|
|
buffer =
|
|
|
|
skip_buffers_until(buffer, USB_BUFFER_SETUP, USB_BUFFER_SETUP, &res);
|
|
|
|
ep->buffer = buffer;
|
|
|
|
if (!buffer || buffer->left < len) {
|
|
|
|
res |= USB_READ_BLOCK;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
/* SETUP buffer must fit in a single buffer */
|
|
|
|
if (buffer->left < len) {
|
|
|
|
buffer->flags |= USB_BUFFER_FAILED;
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED ;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res |= USB_READ_NOTIFY;
|
|
|
|
ep->buffer = buffer->next;
|
|
|
|
res |= USB_READ_FAIL;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (buffer->flags & (USB_BUFFER_SETUP|USB_BUFFER_IN)) {
|
|
|
|
buffer->flags |= USB_BUFFER_FAILED;
|
|
|
|
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED ;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res |= USB_READ_NOTIFY;
|
|
|
|
ep->buffer = buffer->next;
|
|
|
|
res |= USB_READ_FAIL;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len == 0) {
|
|
|
|
/* Status OUT */
|
|
|
|
if (buffer->left > 0) {
|
|
|
|
buffer->flags |= USB_BUFFER_FAILED;
|
|
|
|
res |= USB_READ_FAIL;
|
|
|
|
}
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED ;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res |= USB_READ_NOTIFY;
|
|
|
|
ep->buffer = buffer->next;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
if (get_receive_capacity(buffer) < len) return USB_READ_BLOCK;
|
|
|
|
}
|
|
|
|
hw_offset = buf_desc->ADDR_RX;
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_INTERRUPT:
|
|
|
|
len = buf_desc->COUNT_RX & USB_COUNT0_RX_COUNT0_RX;
|
|
|
|
if (get_receive_capacity(buffer) < len) return USB_READ_BLOCK;
|
|
|
|
hw_offset = buf_desc->ADDR_RX;
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_BULK:
|
|
|
|
if (USB->EPR[hw_ep] & USB_EPxR_SW_BUF_RX) {
|
|
|
|
len = buf_desc->COUNT_RX_1 & USB_COUNT0_RX_COUNT0_RX;
|
|
|
|
hw_offset = buf_desc->ADDR_RX_1;
|
|
|
|
} else {
|
|
|
|
len = buf_desc->COUNT_RX_0 & USB_COUNT0_RX_COUNT0_RX;
|
|
|
|
hw_offset = buf_desc->ADDR_RX_0;
|
|
|
|
}
|
|
|
|
if (get_receive_capacity(buffer) < len) return USB_READ_BLOCK;
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_ISO:
|
|
|
|
len = buf_desc->COUNT_RX & USB_COUNT0_RX_COUNT0_RX;
|
|
|
|
if (get_receive_capacity(buffer) < len) return USB_READ_BLOCK;
|
|
|
|
hw_offset = buf_desc->ADDR_RX;
|
|
|
|
}
|
|
|
|
/* printf("handle_pending_receive: %d %04x\n", len, ep->flags); */
|
|
|
|
short_packet = len < ep->xfer_size;
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (buffer->left < len) {
|
|
|
|
copy = buffer->left;
|
|
|
|
} else {
|
|
|
|
copy = len;
|
|
|
|
}
|
|
|
|
len -= copy;
|
|
|
|
buffer->left -= copy;
|
|
|
|
read_hw_buffer(buffer, hw_offset, copy);
|
|
|
|
hw_offset += copy;
|
|
|
|
|
|
|
|
if (len == 0) break;
|
|
|
|
|
|
|
|
/* Release buffer */
|
|
|
|
buffer->flags &= ~(USB_BUFFER_SUBMITTED | USB_BUFFER_SHORT_PACKET);
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res |= USB_READ_NOTIFY;
|
|
|
|
/* Use next buffer. */
|
|
|
|
buffer = buffer->next;
|
|
|
|
} while(1);
|
|
|
|
|
|
|
|
if (short_packet) {
|
|
|
|
buffer->flags |= USB_BUFFER_SHORT_PACKET;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((buffer->left == 0)
|
|
|
|
|| (buffer->flags & USB_BUFFER_PACKET_END)
|
|
|
|
|| (short_packet && (buffer->flags & USB_BUFFER_SHORT_END))) {
|
|
|
|
/* Release buffer */
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res |= USB_READ_NOTIFY;
|
|
|
|
/* Use next buffer. */
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
ep->buffer = buffer;
|
|
|
|
if (IS_BULK_EP(ep)) {
|
|
|
|
USB->EPR[hw_ep] = EPR_INVARIANT(USB->EPR[hw_ep]) | USB_EPxR_SW_BUF_RX;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PRINTF("read_endpoint %d %d\n", (int)hw_offset-buf_desc->ADDR_RX, (int)buf_desc->ADDR_RX); */
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
start_receive(USBEndpoint *ep)
|
|
|
|
{
|
|
|
|
unsigned int hw_ep = EP_HW_NUM(ep->addr);
|
|
|
|
uint32_t epr = (USB->EPR[hw_ep] | EPR_W0);
|
|
|
|
uint32_t epr_mask = EPR_RW | EPR_W0;
|
|
|
|
switch(ep->flags & USB_EP_FLAGS_TYPE_MASK) {
|
|
|
|
case USB_EP_FLAGS_TYPE_CONTROL:
|
|
|
|
case USB_EP_FLAGS_TYPE_INTERRUPT:
|
|
|
|
{
|
|
|
|
unsigned int capacity = get_receive_capacity(ep->buffer);
|
|
|
|
if (capacity <= ep->xfer_size) {
|
|
|
|
/* This is the last OUT packet of the data stage */
|
|
|
|
epr ^= USB_EPxR_STAT_TX_NAK;
|
|
|
|
} else {
|
|
|
|
epr ^= USB_EPxR_STAT_TX_STALL;
|
|
|
|
}
|
|
|
|
epr ^= USB_EPxR_STAT_RX_VALID;
|
|
|
|
epr_mask |= USB_EP0R_STAT_TX | USB_EP0R_STAT_RX;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_BULK:
|
|
|
|
case USB_EP_FLAGS_TYPE_ISO:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
ep->flags |= USB_EP_FLAGS_RECEIVING;
|
|
|
|
USB->EPR[hw_ep] = epr & epr_mask;
|
|
|
|
}
|
|
|
|
|
|
|
|
static unsigned int
|
|
|
|
get_transmit_length(USBBuffer *buffer)
|
|
|
|
{
|
|
|
|
unsigned int length = 0;
|
|
|
|
while(buffer && (buffer->flags & USB_BUFFER_IN)) {
|
|
|
|
length += buffer->left;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
return length;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
start_transmit(USBEndpoint *ep)
|
|
|
|
{
|
|
|
|
unsigned int hw_start;
|
|
|
|
unsigned int res = 0;
|
|
|
|
USBBuffer *buffer = ep->buffer;
|
|
|
|
unsigned int len;
|
|
|
|
unsigned int hw_offset;
|
|
|
|
volatile uint32_t *hw_countp;
|
|
|
|
unsigned int hw_ep = EP_HW_NUM(ep->addr);
|
|
|
|
uint32_t epr = USB->EPR[hw_ep];
|
|
|
|
unsigned int ep_flags = ep->flags;
|
|
|
|
USB_HW_Buffer *buf_desc = USB_EP_BUF_DESC(hw_ep);
|
|
|
|
len = ep->xfer_size;
|
|
|
|
if (!(ep_flags & USB_EP_FLAGS_ENABLED) || !buffer) return USB_WRITE_BLOCK;
|
|
|
|
/* PRINTF("start_transmit: %02x\n", ep->addr); */
|
|
|
|
switch(ep_flags & USB_EP_FLAGS_TYPE_MASK) {
|
|
|
|
case USB_EP_FLAGS_TYPE_CONTROL:
|
|
|
|
|
|
|
|
if (get_transmit_length(ep->buffer) <= len) {
|
|
|
|
/* This is the last IN packet of the data stage */
|
|
|
|
USB->EPR[hw_ep] = USB_EPxR_EP_STATUS_OUT
|
|
|
|
| EPR_TOGGLE_SET(epr, USB_EP0R_STAT_RX, USB_EPxR_STAT_RX_NAK);
|
|
|
|
} else {
|
|
|
|
USB->EPR[hw_ep] = USB_EPxR_EP_STATUS_OUT
|
|
|
|
| EPR_TOGGLE_SET(epr, USB_EP0R_STAT_RX, USB_EPxR_STAT_RX_STALL);
|
|
|
|
}
|
|
|
|
hw_offset = buf_desc->ADDR_TX;
|
|
|
|
hw_countp = &buf_desc->COUNT_TX;
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_BULK:
|
|
|
|
if (buffer->flags & USB_BUFFER_HALT) {
|
2010-02-05 16:23:40 +00:00
|
|
|
if (ep->status & 0x01) return USB_WRITE_BLOCK;
|
2009-07-12 15:52:28 +00:00
|
|
|
ep->status |= 0x01;
|
|
|
|
stall_bulk_in(hw_ep);
|
2010-02-05 16:23:40 +00:00
|
|
|
return USB_WRITE_BLOCK;
|
2009-07-12 15:52:28 +00:00
|
|
|
}
|
|
|
|
if (USB->EPR[hw_ep] & USB_EPxR_SW_BUF_TX) {
|
|
|
|
hw_offset = buf_desc->ADDR_TX_1;
|
|
|
|
hw_countp = &buf_desc->COUNT_TX_1;
|
|
|
|
} else {
|
|
|
|
hw_offset = buf_desc->ADDR_TX_0;
|
|
|
|
hw_countp = &buf_desc->COUNT_TX_0;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
hw_start = hw_offset;
|
|
|
|
while (buffer) {
|
|
|
|
unsigned int copy;
|
|
|
|
if (buffer->left < len) {
|
|
|
|
copy = buffer->left;
|
|
|
|
} else {
|
|
|
|
copy = len;
|
|
|
|
}
|
|
|
|
len -= copy;
|
|
|
|
buffer->left -= copy;
|
|
|
|
write_hw_buffer(buffer, hw_offset, copy);
|
|
|
|
hw_offset += copy;
|
|
|
|
if (buffer->left == 0) {
|
|
|
|
if (buffer->flags & USB_BUFFER_SHORT_END) {
|
|
|
|
if (len == 0) {
|
|
|
|
/* Send zero length packet. */
|
2010-02-05 16:23:40 +00:00
|
|
|
break; /* Leave without moving to next buffer */
|
2009-07-12 15:52:28 +00:00
|
|
|
} else {
|
|
|
|
len = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Release buffer */
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED;
|
|
|
|
if (buffer->flags & USB_BUFFER_NOTIFY) res = USB_WRITE_NOTIFY;
|
|
|
|
/* Use next buffer. */
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
if (len == 0) break;
|
|
|
|
}
|
|
|
|
ep->buffer = buffer;
|
|
|
|
if (ep->flags & USB_EP_FLAGS_TRANSMITTING) {
|
|
|
|
ep->flags |= USB_EP_FLAGS_DOUBLE;
|
|
|
|
} else {
|
|
|
|
ep->flags |= USB_EP_FLAGS_TRANSMITTING;
|
|
|
|
}
|
|
|
|
*hw_countp = hw_offset - hw_start;
|
|
|
|
/* printf("start_transmit: %02x %d %04lx\n", ep->addr, hw_offset - hw_start, USB->EPR[hw_ep]); */
|
|
|
|
switch(ep->flags & USB_EP_FLAGS_TYPE_MASK) {
|
|
|
|
case USB_EP_FLAGS_TYPE_CONTROL:
|
|
|
|
case USB_EP_FLAGS_TYPE_INTERRUPT:
|
|
|
|
USB->EPR[hw_ep] =
|
|
|
|
EPR_TOGGLE_SET(epr, USB_EP0R_STAT_TX, USB_EPxR_STAT_TX_VALID);
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_BULK:
|
|
|
|
USB->EPR[hw_ep] = EPR_INVARIANT(USB->EPR[hw_ep]) | USB_EPxR_SW_BUF_TX;
|
|
|
|
break;
|
|
|
|
case USB_EP_FLAGS_TYPE_ISO:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* printf("start_transmit: %04x\n", USB->EPR[hw_ep]); */
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
start_transfer(USBEndpoint *ep)
|
|
|
|
{
|
|
|
|
int res;
|
|
|
|
while (1) {
|
|
|
|
if (!(ep->addr & 0x80) && (IS_BULK_EP(ep) || IS_INTERRUPT_EP(ep))) {
|
|
|
|
if (ep->buffer && (ep->buffer->flags & USB_BUFFER_HALT)) {
|
|
|
|
if (ep->status & 0x01) return ;
|
|
|
|
ep->status |= 0x01;
|
|
|
|
stall_bulk_out(EP_HW_NUM(ep->addr));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!(ep->flags & USB_EP_FLAGS_RECV_PENDING)) break;
|
|
|
|
res = handle_pending_receive(ep);
|
|
|
|
if (res & USB_READ_NOTIFY) {
|
|
|
|
notify_ep_process(ep, USB_EP_EVENT_NOTIFICATION);
|
|
|
|
}
|
|
|
|
if (res & USB_READ_BLOCK) return;
|
|
|
|
if (ep->flags & USB_EP_FLAGS_DOUBLE) {
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_DOUBLE;
|
|
|
|
} else {
|
|
|
|
ep->flags &= ~(USB_EP_FLAGS_RECV_PENDING|USB_EP_FLAGS_SETUP_PENDING);
|
|
|
|
}
|
|
|
|
if (res & USB_READ_FAIL) {
|
|
|
|
/* Only fails for control endpoints */
|
|
|
|
usb_arch_control_stall(ep->addr);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ep->addr == 0x02)
|
|
|
|
PRINTF("start EPR: %04x ep->flags: %02x\n",
|
|
|
|
(unsigned int)USB->EPR[EP_HW_NUM(ep->addr)],
|
|
|
|
(unsigned int)ep->flags);
|
|
|
|
if (ep->flags & (USB_EP_FLAGS_TRANSMITTING | USB_EP_FLAGS_RECEIVING)) {
|
|
|
|
if (!IS_BULK_EP(ep) || (ep->flags & USB_EP_FLAGS_DOUBLE)) {
|
|
|
|
PRINTF("Busy\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (ep->status & 0x01) return; /* Don't start transfer if halted */
|
|
|
|
if (ep->buffer) {
|
|
|
|
if (ep->buffer->flags & USB_BUFFER_IN) {
|
|
|
|
res = start_transmit(ep);
|
|
|
|
if (res & USB_READ_NOTIFY) {
|
|
|
|
notify_ep_process(ep, USB_EP_EVENT_NOTIFICATION);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
start_receive(ep);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
transfer_complete(unsigned int hw_ep) {
|
|
|
|
uint32_t epr = USB->EPR[hw_ep];
|
|
|
|
USBEndpoint *ep = &usb_endpoints[hw_ep];
|
|
|
|
if (epr &USB_EP0R_CTR_RX) {
|
|
|
|
PRINTF("Received packet %lx %04x\n", USB_EP_BUF_DESC(hw_ep)->COUNT_RX, (int)USB->EPR[hw_ep]);
|
|
|
|
if (epr & USB_EP0R_SETUP) {
|
|
|
|
PRINTF("SETUP\n");
|
|
|
|
ep->flags |= USB_EP_FLAGS_SETUP_PENDING;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (IS_BULK_EP(ep)) {
|
|
|
|
if ((epr ^ (epr >> 8)) & USB_EP0R_DTOG_TX) {
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_DOUBLE;
|
|
|
|
} else {
|
|
|
|
ep->flags |= USB_EP_FLAGS_DOUBLE;
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_RECEIVING;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_RECEIVING;
|
|
|
|
}
|
|
|
|
ep->flags |= USB_EP_FLAGS_RECV_PENDING;
|
|
|
|
if (IS_CONTROL_EP(ep)) epr &= ~USB_EPxR_EP_STATUS_OUT;
|
|
|
|
USB->EPR[hw_ep] = EPR_INVARIANT(epr) & ~USB_EP0R_CTR_RX;
|
|
|
|
#if 0
|
|
|
|
if (ep->flags & USB_EP_FLAGS_DOUBLE) {
|
|
|
|
printf("Double\n");
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
start_transfer(ep);
|
|
|
|
}
|
|
|
|
if (epr &USB_EP0R_CTR_TX) {
|
|
|
|
PRINTF("Sent packet\n");
|
|
|
|
if (ep->flags & USB_EP_FLAGS_DOUBLE) {
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_DOUBLE;
|
|
|
|
} else {
|
|
|
|
ep->flags &= ~USB_EP_FLAGS_TRANSMITTING;
|
|
|
|
}
|
|
|
|
USB->EPR[hw_ep] = EPR_INVARIANT(USB->EPR[hw_ep]) & ~USB_EP0R_CTR_TX;
|
|
|
|
start_transfer(ep);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_set_ep_event_process(unsigned char addr, struct process *p)
|
|
|
|
{
|
|
|
|
USBEndpoint *ep = &usb_endpoints[EP_INDEX(addr)];
|
|
|
|
ep->event_process = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Select what process should be polled when a global event occurs */
|
|
|
|
void
|
|
|
|
usb_arch_set_global_event_process(struct process *p)
|
|
|
|
{
|
|
|
|
event_process = p;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int
|
|
|
|
usb_arch_get_global_events(void)
|
|
|
|
{
|
|
|
|
unsigned int e;
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
e = events;
|
|
|
|
events = 0;
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned int
|
|
|
|
usb_get_ep_events(unsigned char addr)
|
|
|
|
{
|
|
|
|
unsigned int e;
|
|
|
|
unsigned int ei = EP_HW_NUM(addr);
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
e = usb_endpoints[ei].events;
|
|
|
|
usb_endpoints[ei].events = 0;
|
|
|
|
USB_ENABLE_INT;
|
|
|
|
return e;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_submit_recv_buffer(unsigned char ep_addr, USBBuffer *buffer)
|
|
|
|
{
|
|
|
|
USBBuffer **tailp;
|
|
|
|
USBEndpoint *ep = &usb_endpoints[EP_INDEX(ep_addr)];
|
|
|
|
if (!(ep->flags & USB_EP_FLAGS_ENABLED)) return;
|
|
|
|
/* PRINTF("buffer: %p\n", ep->buffer); */
|
|
|
|
/* dbg_drain(); */
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
tailp = (USBBuffer**)&ep->buffer;
|
|
|
|
while(*tailp) {
|
|
|
|
tailp = &(*tailp)->next;
|
|
|
|
}
|
|
|
|
*tailp = buffer;
|
|
|
|
while(buffer) {
|
|
|
|
buffer->flags |= USB_BUFFER_SUBMITTED;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
start_transfer(ep);
|
|
|
|
|
|
|
|
USB_ENABLE_INT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_submit_xmit_buffer(unsigned char ep_addr, USBBuffer *buffer)
|
|
|
|
{
|
|
|
|
USBBuffer **tailp;
|
|
|
|
USBEndpoint *ep = &usb_endpoints[EP_INDEX(ep_addr)];
|
|
|
|
if (!(ep->flags & USB_EP_FLAGS_ENABLED)) return;
|
|
|
|
/* PRINTF("usb_submit_xmit_buffer %d\n", buffer->left); */
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
tailp = (USBBuffer**)&ep->buffer;
|
|
|
|
while(*tailp) {
|
|
|
|
tailp = &(*tailp)->next;
|
|
|
|
}
|
|
|
|
*tailp = buffer;
|
|
|
|
while(buffer) {
|
|
|
|
buffer->flags |= USB_BUFFER_SUBMITTED | USB_BUFFER_IN;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
start_transfer(ep);
|
|
|
|
USB_ENABLE_INT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_discard_all_buffers(unsigned char ep_addr)
|
|
|
|
{
|
|
|
|
uint32_t ints;
|
|
|
|
USBBuffer *buffer;
|
|
|
|
volatile USBEndpoint *ep = &usb_endpoints[EP_INDEX(ep_addr)];
|
|
|
|
ints = usb_save_disable_int();
|
|
|
|
buffer = ep->buffer;
|
|
|
|
ep->buffer = NULL;
|
|
|
|
#if 0
|
|
|
|
/* Set both directions to NAK */
|
|
|
|
USB->EPR[EP_HW_NUM(ep_addr)] =
|
|
|
|
EPR_TOGGLE_SET(USB->EPR[EP_HW_NUM(ep_addr)],
|
|
|
|
USB_EP0R_STAT_RX|USB_EP0R_STAT_TX,
|
|
|
|
USB_EPxR_STAT_TX_NAK| USB_EPxR_STAT_RX_NAK);
|
|
|
|
ep->flags &= ~(USB_EP_FLAGS_RECV_PENDING|USB_EP_FLAGS_SETUP_PENDING);
|
|
|
|
#endif
|
|
|
|
ep->flags &= ~(USB_EP_FLAGS_RECV_PENDING | USB_EP_FLAGS_SETUP_PENDING
|
|
|
|
| USB_EP_FLAGS_DOUBLE);
|
|
|
|
usb_restore_int(ints);
|
|
|
|
while(buffer) {
|
|
|
|
buffer->flags &= ~USB_BUFFER_SUBMITTED;
|
|
|
|
buffer = buffer->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
uint16_t
|
|
|
|
usb_arch_get_ep_status(uint8_t addr)
|
|
|
|
{
|
|
|
|
if (EP_INDEX(addr) > USB_MAX_ENDPOINTS) return 0;
|
|
|
|
return usb_endpoints[EP_INDEX(addr)].status;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_set_configuration(uint8_t usb_configuration_value)
|
|
|
|
{
|
|
|
|
/* Nothing needs to be done */
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_control_stall(unsigned char addr)
|
|
|
|
{
|
|
|
|
if (EP_INDEX(addr) > USB_MAX_ENDPOINTS) return;
|
|
|
|
uint32_t epr = USB->EPR[EP_HW_NUM(addr)];
|
|
|
|
USB->EPR[EP_HW_NUM(addr)] = EPR_TOGGLE_SET(epr,USB_EP0R_STAT_RX|USB_EP0R_STAT_TX, USB_EPxR_STAT_RX_STALL | USB_EPxR_STAT_TX_STALL);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not for control endpoints */
|
|
|
|
void
|
|
|
|
usb_arch_halt_endpoint(unsigned char ep_addr, int halt)
|
|
|
|
{
|
|
|
|
if (EP_INDEX(ep_addr) > USB_MAX_ENDPOINTS) return;
|
|
|
|
if (!usb_endpoints[EP_INDEX(ep_addr)].flags & USB_EP_FLAGS_ENABLED) return;
|
|
|
|
USB_DISABLE_INT;
|
|
|
|
if (halt) {
|
|
|
|
if (!(usb_endpoints[EP_INDEX(ep_addr)].status & 0x01)) {
|
|
|
|
usb_endpoints[EP_INDEX(ep_addr)].status |= 0x01;
|
|
|
|
if (ep_addr & 0x80) {
|
|
|
|
stall_bulk_in(EP_HW_NUM(ep_addr));
|
|
|
|
} else {
|
|
|
|
stall_bulk_out(EP_HW_NUM(ep_addr));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
USBEndpoint *ep = &usb_endpoints[EP_INDEX(ep_addr)];
|
|
|
|
if (ep->status & 0x01) {
|
|
|
|
ep->status &= ~0x01;
|
|
|
|
PRINTF("HALT clear restart EPR: %04x %p %p\n",
|
|
|
|
(unsigned int)USB->EPR[EP_HW_NUM(ep_addr)],
|
|
|
|
ep->buffer, ep->buffer->next);
|
|
|
|
/* Restore toggle state for double buffered endpoint */
|
|
|
|
if (IS_BULK_EP(ep)) {
|
|
|
|
volatile uint32_t *eprp = &USB->EPR[EP_HW_NUM(ep_addr)];
|
|
|
|
if (ep_addr & 0x80) {
|
|
|
|
ep->flags &= ~(USB_EP_FLAGS_DOUBLE |USB_EP_FLAGS_TRANSMITTING);
|
|
|
|
|
|
|
|
*eprp =(EPR_TOGGLE_SET(*eprp,(USB_EP0R_STAT_TX | USB_EP0R_DTOG_TX
|
|
|
|
| USB_EPxR_SW_BUF_TX),
|
|
|
|
USB_EPxR_STAT_TX_VALID));
|
|
|
|
} else {
|
|
|
|
ep->flags &= ~(USB_EP_FLAGS_DOUBLE | USB_EP_FLAGS_RECEIVING
|
|
|
|
| USB_EP_FLAGS_RECV_PENDING);
|
|
|
|
|
|
|
|
*eprp =(EPR_TOGGLE_SET(*eprp,(USB_EP0R_STAT_RX | USB_EP0R_DTOG_RX
|
|
|
|
| USB_EPxR_SW_BUF_RX),
|
|
|
|
USB_EPxR_STAT_RX_VALID|USB_EPxR_SW_BUF_RX));
|
|
|
|
*eprp = EPR_INVARIANT(*eprp) | USB_EPxR_SW_BUF_RX;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Release HALT buffer */
|
|
|
|
if (ep->buffer && (ep->buffer->flags & USB_BUFFER_HALT)) {
|
|
|
|
ep->buffer->flags &= ~USB_BUFFER_SUBMITTED;
|
|
|
|
if (ep->buffer->flags & USB_BUFFER_NOTIFY) {
|
|
|
|
notify_ep_process(ep,USB_EP_EVENT_NOTIFICATION);
|
|
|
|
}
|
|
|
|
ep->buffer = ep->buffer->next;
|
|
|
|
}
|
|
|
|
/* Restart transmission */
|
|
|
|
start_transfer(&usb_endpoints[EP_INDEX(ep_addr)]);
|
|
|
|
PRINTF("HALT clear restart EPR: %04x %p %p\n",
|
|
|
|
(unsigned int)USB->EPR[EP_HW_NUM(ep_addr)],
|
|
|
|
ep->buffer, ep->buffer->next);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
USB_ENABLE_INT;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_set_address(unsigned char addr)
|
|
|
|
{
|
|
|
|
USB->DADDR = 0x80 | addr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
USB_HP_CAN_TX_handler(void) __attribute__((interrupt));
|
|
|
|
|
|
|
|
void
|
|
|
|
USB_HP_CAN_TX_handler(void)
|
|
|
|
{
|
|
|
|
uint32_t status = USB->ISTR;
|
|
|
|
if (status & USB_ISTR_CTR) {
|
|
|
|
transfer_complete(status & USB_ISTR_EP_ID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
USB_LP_CAN_RX0_handler(void) __attribute__((interrupt));
|
|
|
|
void
|
|
|
|
USB_LP_CAN_RX0_handler(void)
|
|
|
|
{
|
|
|
|
uint32_t status = USB->ISTR;
|
|
|
|
if (status & USB_ISTR_CTR) {
|
|
|
|
transfer_complete(status & USB_ISTR_EP_ID);
|
|
|
|
/* PRINTF("Transfer complete ep %ld\n", status & USB_ISTR_EP_ID); */
|
|
|
|
} else if (status & USB_ISTR_PMAOVR) {
|
|
|
|
PRINTF("PMAOVR\n");
|
|
|
|
USB->ISTR &= ~USB_ISTR_PMAOVR;
|
|
|
|
} else if (status & USB_ISTR_ERR) {
|
|
|
|
PRINTF("ERR\n");
|
|
|
|
USB->ISTR &= ~USB_ISTR_ERR;
|
|
|
|
} else if (status & USB_ISTR_WKUP) {
|
|
|
|
PRINTF("WKUP\n");
|
|
|
|
USB->ISTR &= ~USB_ISTR_WKUP;
|
|
|
|
USB->CNTR &= ~USB_CNTR_FSUSP;
|
|
|
|
notify_process(USB_EVENT_RESUME);
|
|
|
|
} else if (status & USB_ISTR_SUSP) {
|
|
|
|
PRINTF("SUSP\n");
|
|
|
|
USB->ISTR &= ~USB_ISTR_SUSP;
|
|
|
|
USB->CNTR |= USB_CNTR_FSUSP;
|
|
|
|
notify_process(USB_EVENT_SUSPEND);
|
|
|
|
} else if (status & USB_ISTR_RESET) {
|
|
|
|
PRINTF("RESET\n");
|
|
|
|
USB->ISTR &= ~USB_ISTR_RESET;
|
|
|
|
usb_arch_reset();
|
|
|
|
notify_process(USB_EVENT_RESET);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
usb_arch_toggle_SW_BUF_RX()
|
|
|
|
{
|
|
|
|
USB->EPR[2] = EPR_INVARIANT(USB->EPR[2]) | USB_EPxR_SW_BUF_RX;
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
usb_arch_send_pending(uint8_t ep_addr)
|
|
|
|
{
|
|
|
|
return usb_endpoints[EP_INDEX(ep_addr)].flags & USB_EP_FLAGS_TRANSMITTING;
|
|
|
|
}
|