/* FTDI serial chip emulation for LUFA Jim Paris */ #include #include "ftdi.h" #include "queue.h" /* TX and RX queues */ volatile static uint8_t ftdi_txqueue_buf[FTDI_TX_QUEUE_LEN]; volatile static uint8_t ftdi_rxqueue_buf[FTDI_RX_QUEUE_LEN]; struct queue ftdi_txqueue = QUEUE_INITIALIZER(ftdi_txqueue_buf, sizeof(ftdi_txqueue_buf)); struct queue ftdi_rxqueue = QUEUE_INITIALIZER(ftdi_rxqueue_buf, sizeof(ftdi_rxqueue_buf)); /* Not sure if we need this, but it might make the Windows driver happy. */ static const uint8_t ftdi_eeprom[] = { 0x00, 0x40, 0x03, 0x04, 0x01, 0x60, 0x00, 0x00, 0xa0, 0x2d, 0x08, 0x00, 0x00, 0x00, 0x98, 0x0a, 0xa2, 0x20, 0xc2, 0x12, 0x23, 0x10, 0x05, 0x00, 0x0a, 0x03, 0x46, 0x00, 0x54, 0x00, 0x44, 0x00, 0x49, 0x00, 0x20, 0x03, 0x46, 0x00, 0x54, 0x00, 0x32, 0x00, 0x33, 0x00, 0x32, 0x00, 0x52, 0x00, 0x20, 0x00, 0x55, 0x00, 0x53, 0x00, 0x42, 0x00, 0x20, 0x00, 0x55, 0x00, 0x41, 0x00, 0x52, 0x00, 0x54, 0x00, 0x12, 0x03, 0x41, 0x00, 0x36, 0x00, 0x30, 0x00, 0x30, 0x00, 0x65, 0x00, 0x4f, 0x00, 0x45, 0x00, 0x37, 0x00, 0x81, 0xe1, 0x35, 0x60, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x9a, 0xa5 }; volatile static struct ftdi_status { /* modem status */ unsigned reserved_1:1; unsigned reserved_0:3; unsigned cts:1; unsigned dsr:1; unsigned ri:1; unsigned cd:1; /* ftdi status */ unsigned dr:1; unsigned oe:1; unsigned pe:1; unsigned fe:1; unsigned bi:1; unsigned thre:1; unsigned temt:1; unsigned fifoerr:1; } ftdi_status = { 0 }; volatile static struct ctrl_status { unsigned dtr:1; unsigned rts:1; } ctrl_status = { 0 }; static bool ftdi_onlcr = false; /* Translate \n to \r\n, like stty "onlcr" */ static bool ftdi_blocking_in = false; static bool ftdi_blocking_out = false; static bool ftdi_delay_init_tx = false; volatile bool got_setbaudrate; volatile bool got_first_rx; #define WAIT_FRAMES 250 volatile uint8_t frames_since_setbaudrate; volatile uint8_t latency_timer = 16; volatile uint8_t latency_count = 0; /* Fdev compatible wrappers */ static int ftdi_fdev_write(char c, FILE *stream) { if (ftdi_onlcr && c == '\n') ftdi_putchar('\r'); ftdi_putchar(c); return 0; } static int ftdi_fdev_read(FILE *stream) { return ftdi_getchar(); } static FILE _ftdi_stream = FDEV_SETUP_STREAM(ftdi_fdev_write, ftdi_fdev_read, _FDEV_SETUP_RW); FILE *ftdi_stream = &_ftdi_stream; static void reset_internal_state(void) { /* Disable endpoint interrupts */ UEIENX &= ~(1 << RXOUTE); UEIENX &= ~(1 << TXINE); /* Reset flags for ftdi_init_track_tx */ got_setbaudrate = false; got_first_rx = false; frames_since_setbaudrate = 0; /* Latency timer */ latency_timer = 16; latency_count = 0; /* Reset line state */ ftdi_status = (struct ftdi_status) { 0 }; ftdi_status.reserved_1 = 1; ftdi_status.cts = 1; ftdi_status.dsr = 1; ftdi_status.ri = 0; ftdi_status.cd = 1; ctrl_status = (struct ctrl_status) { 0 }; ctrl_status.dtr = 0; ctrl_status.rts = 0; } /* Init */ void ftdi_init(int flags) { reset_internal_state(); if (flags & FTDI_STDIO) { stdout = ftdi_stream; stdin = ftdi_stream; } if (flags & FTDI_BLOCKING_IN) ftdi_blocking_in = true; if (flags & FTDI_BLOCKING_OUT) ftdi_blocking_out = true; if (flags & FTDI_ONLCR) ftdi_onlcr = true; if (flags & FTDI_DELAY_INIT_TX) ftdi_delay_init_tx = true; USB_Device_EnableSOFEvents(); } static inline void ftdi_control_request(void) { uint16_t val; if (!Endpoint_IsSETUPReceived()) return; if ((USB_ControlRequest.bmRequestType & CONTROL_REQTYPE_TYPE) != REQTYPE_VENDOR) return; switch (USB_ControlRequest.bRequest) { case SIO_RESET_REQUEST: switch (USB_ControlRequest.wValue) { case SIO_RESET_SIO: ctrl_status.dtr = 0; ctrl_status.rts = 0; break; case SIO_RESET_PURGE_RX: while (_queue_can_get(&ftdi_rxqueue)) (void)_queue_get(&ftdi_rxqueue); break; case SIO_RESET_PURGE_TX: while (_queue_can_get(&ftdi_txqueue)) (void)_queue_get(&ftdi_txqueue); ftdi_status.temt = 1; break; } Endpoint_ClearSETUP(); Endpoint_ClearStatusStage(); return; case SIO_READ_EEPROM_REQUEST: /* Return fake EEPROM data */ val = (USB_ControlRequest.wIndex & 0xff) * 2; if (val + USB_ControlRequest.wLength > sizeof(ftdi_eeprom)) return; Endpoint_ClearSETUP(); Endpoint_Write_Control_Stream_LE((uint8_t *)&ftdi_eeprom[val], USB_ControlRequest.wLength); Endpoint_ClearStatusStage(); return; case SIO_SET_BAUDRATE_REQUEST: got_setbaudrate = true; Endpoint_ClearSETUP(); Endpoint_ClearStatusStage(); return; case SIO_WRITE_EEPROM_REQUEST: case SIO_ERASE_EEPROM_REQUEST: case SIO_SET_DATA_REQUEST: case SIO_SET_FLOW_CTRL_REQUEST: case SIO_SET_EVENT_CHAR_REQUEST: case SIO_SET_ERROR_CHAR_REQUEST: case SIO_SET_BITMODE_REQUEST: /* Ignore these */ Endpoint_ClearSETUP(); Endpoint_ClearStatusStage(); return; case SIO_SET_LATENCY_TIMER_REQUEST: val = USB_ControlRequest.wValue; if (val >= 1 && val <= 255) { latency_timer = val; latency_count = 0; } Endpoint_ClearSETUP(); Endpoint_ClearStatusStage(); return; case SIO_SET_MODEM_CTRL_REQUEST: if (USB_ControlRequest.wValue & SIO_SET_DTR_HIGH) ctrl_status.dtr = 1; else if (USB_ControlRequest.wValue & SIO_SET_DTR_LOW) ctrl_status.dtr = 0; if (USB_ControlRequest.wValue & SIO_SET_RTS_HIGH) ctrl_status.rts = 1; else if (USB_ControlRequest.wValue & SIO_SET_RTS_LOW) ctrl_status.rts = 0; Endpoint_ClearSETUP(); Endpoint_ClearStatusStage(); return; case SIO_GET_LATENCY_TIMER_REQUEST: if (USB_ControlRequest.wLength != 1) return; Endpoint_ClearSETUP(); val = latency_timer; Endpoint_Write_Control_Stream_LE((uint8_t *)&val, 1); Endpoint_ClearStatusStage(); return; case SIO_READ_PINS_REQUEST: /* Return dummy value */ if (USB_ControlRequest.wLength != 1) return; Endpoint_ClearSETUP(); val = 0; Endpoint_Write_Control_Stream_LE((uint8_t *)&val, 1); Endpoint_ClearStatusStage(); return; case SIO_POLL_MODEM_STATUS_REQUEST: /* Send status */ if (USB_ControlRequest.wLength != 2) return; Endpoint_ClearSETUP(); Endpoint_Write_Control_Stream_LE((uint8_t *)&ftdi_status, 2); Endpoint_ClearStatusStage(); return; } } /* Device descriptor */ static const USB_Descriptor_Device_t PROGMEM DeviceDescriptor = { .Header = { .Size = sizeof(USB_Descriptor_Device_t), .Type = DTYPE_Device }, .USBSpecification = VERSION_BCD(2,0,0), .Class = 0x00, .SubClass = 0x00, .Protocol = 0x00, .Endpoint0Size = FIXED_CONTROL_ENDPOINT_SIZE, .VendorID = 0x0403, /* FTDI */ .ProductID = 0x6001, /* FT232 */ .ReleaseNumber = 0x0600, /* FT232 */ .ManufacturerStrIndex = 0x01, .ProductStrIndex = 0x02, .SerialNumStrIndex = USE_INTERNAL_SERIAL, .NumberOfConfigurations = FIXED_NUM_CONFIGURATIONS }; /* Configuration descriptor */ typedef struct { USB_Descriptor_Configuration_Header_t Config; USB_Descriptor_Interface_t FTDI_Interface; USB_Descriptor_Endpoint_t FTDI_DataInEndpoint; USB_Descriptor_Endpoint_t FTDI_DataOutEndpoint; } USB_Descriptor_Config_t; static const USB_Descriptor_Config_t PROGMEM ConfigurationDescriptor = { .Config = { .Header = { .Size = sizeof(USB_Descriptor_Configuration_Header_t), .Type = DTYPE_Configuration }, .TotalConfigurationSize = sizeof(USB_Descriptor_Config_t), .TotalInterfaces = 1, .ConfigurationNumber = 1, .ConfigurationStrIndex = 0, .ConfigAttributes = USB_CONFIG_ATTR_RESERVED, .MaxPowerConsumption = USB_CONFIG_POWER_MA(100) }, .FTDI_Interface = { .Header = { .Size = sizeof(USB_Descriptor_Interface_t), .Type = DTYPE_Interface }, .InterfaceNumber = 0, .AlternateSetting = 0, .TotalEndpoints = 2, .Class = 0xff, .SubClass = 0xff, .Protocol = 0xff, .InterfaceStrIndex = 0x02, /* Reuse product string */ }, .FTDI_DataInEndpoint = { .Header = { .Size = sizeof(USB_Descriptor_Endpoint_t), .Type = DTYPE_Endpoint }, .EndpointAddress = FTDI_TX_EPADDR, .Attributes = EP_TYPE_BULK, .EndpointSize = FTDI_TXRX_EPSIZE, .PollingIntervalMS = 0x00, }, .FTDI_DataOutEndpoint = { .Header = { .Size = sizeof(USB_Descriptor_Endpoint_t), .Type = DTYPE_Endpoint }, .EndpointAddress = FTDI_RX_EPADDR, .Attributes = EP_TYPE_BULK, .EndpointSize = FTDI_TXRX_EPSIZE, .PollingIntervalMS = 0x00, }, }; /* String descriptors */ #define STRING_DESC(s) { { (sizeof(s)-2)+1+1, DTYPE_String }, s } static const USB_Descriptor_String_t PROGMEM string0 = { { USB_STRING_LEN(1), DTYPE_String }, { LANGUAGE_ID_ENG } }; static const USB_Descriptor_String_t PROGMEM string1 = STRING_DESC(L"LUFA"); static const USB_Descriptor_String_t PROGMEM string2 = STRING_DESC(L"FT232"); static const USB_Descriptor_String_t *strings[] = { &string0, &string1, &string2 }; uint16_t CALLBACK_USB_GetDescriptor(const uint16_t wValue, const uint16_t wIndex, const void** const DescriptorAddress) { const uint8_t DescriptorType = (wValue >> 8); const uint8_t DescriptorNumber = (wValue & 0xFF); const void* Address = NULL; uint16_t Size = 0; switch (DescriptorType) { case DTYPE_Device: Address = &DeviceDescriptor; Size = sizeof(USB_Descriptor_Device_t); break; case DTYPE_Configuration: Address = &ConfigurationDescriptor; Size = sizeof(USB_Descriptor_Config_t); break; case DTYPE_String: if (DescriptorNumber >= (sizeof(strings) / sizeof(strings[0]))) break; Address = strings[DescriptorNumber]; Size = pgm_read_byte( &strings[DescriptorNumber]->Header.Size); break; } *DescriptorAddress = Address; return Size; } /* LUFA events and USB interrupt handlers */ void EVENT_USB_Device_ControlRequest(void) { ftdi_control_request(); } void EVENT_USB_Device_Reset(void) { Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP); USB_INT_Enable(USB_INT_RXSTPI); reset_internal_state(); } void EVENT_USB_Device_Disconnect(void) { reset_internal_state(); } /* Configure endpoint with the best available number of banks. */ static inline bool ftdi_config_ep(uint8_t address, uint8_t type, uint16_t size) { /* Try configuring the bulk endpoint with 2 banks. */ if (Endpoint_ConfigureEndpoint(address, type, size, 2)) return true; /* On smaller parts, this will fail, so use 1 bank instead. */ return Endpoint_ConfigureEndpoint(address, type, size, 1); } static void maybe_enable_tx(void) { /* Not configured? */ if (USB_DeviceState != DEVICE_STATE_Configured) return; /* Transmitter empty, and latency timer hasn't timed out? */ if (ftdi_status.temt && latency_count < latency_timer) return; /* Delay the initial TX? */ if (ftdi_delay_init_tx) { if (!got_first_rx) { if (!got_setbaudrate) return; if (frames_since_setbaudrate < WAIT_FRAMES) return; } } /* Enable it */ Endpoint_SelectEndpoint(FTDI_TX_EPADDR); UEIENX |= (1 << TXINE); } void EVENT_USB_Device_StartOfFrame(void) { if (latency_count < latency_timer) { latency_count++; if (latency_count == latency_timer) maybe_enable_tx(); } if (!ftdi_delay_init_tx) return; if (!got_setbaudrate) return; if (frames_since_setbaudrate < WAIT_FRAMES) { frames_since_setbaudrate++; if (frames_since_setbaudrate == WAIT_FRAMES) maybe_enable_tx(); } } void EVENT_USB_Device_ConfigurationChanged(void) { uint8_t PrevSelectedEndpoint = Endpoint_GetCurrentEndpoint(); /* Configure endpoints */ ftdi_config_ep(FTDI_TX_EPADDR, EP_TYPE_BULK, FTDI_TXRX_EPSIZE); ftdi_config_ep(FTDI_RX_EPADDR, EP_TYPE_BULK, FTDI_TXRX_EPSIZE); /* Start interrupt on RX endpoint */ Endpoint_SelectEndpoint(FTDI_RX_EPADDR); UEIENX |= (1 << RXOUTE); /* If we have data to send, start interrupt on TX endpoint */ maybe_enable_tx(); /* Done */ Endpoint_SelectEndpoint(PrevSelectedEndpoint); } static const char *fmt_ch(uint8_t ch) __attribute__((unused)); static const char *fmt_ch(uint8_t ch) { static char s[10]; if (ch > 32 && ch <= 127) sprintf(s, "%c", ch); else sprintf(s, "<%d>", ch); return s; } static inline void ftdi_bulk_in(void) { int len, i; /* Data format: ftdi_status structure (2 bytes) followed by the serial data. */ len = _queue_get_len(&ftdi_txqueue); if (len > FTDI_TXRX_EPSIZE - 2) { /* There will be more data after this packet */ len = FTDI_TXRX_EPSIZE - 2; } else { /* No more data after this */ ftdi_status.temt = 1; } /* If we have data or the latency timer expired, send status bytes and the data. Otherwise, this will just be a ZLP. */ if (len > 0 || latency_count == latency_timer) { Endpoint_Write_8(((uint8_t *)&ftdi_status)[0]); Endpoint_Write_8(((uint8_t *)&ftdi_status)[1]); for (i = 0; i < len; i++) { uint8_t ch = _queue_get(&ftdi_txqueue); Endpoint_Write_8(ch); } /* Latency timer can be reset, since we're sending useful data to the host. */ latency_count = 0; } if (ftdi_status.temt) { /* If we have no more data, and we wrote a packet that was smaller than the full endpoint size, then this transfer is done and we can disable the IN interrupt. We'll reenable it when we have more data to send. */ if (len != FTDI_TXRX_EPSIZE - 2) UEIENX &= ~(1 << TXINE); } /* Send it */ Endpoint_ClearIN(); } static inline void ftdi_bulk_out(void) { int i; int len; bool overflow = false; len = Endpoint_BytesInEndpoint(); for (i = 0; i < len; i++) { /* This will discard incoming bytes if queue is full */ uint8_t ch = Endpoint_Read_8(); got_first_rx = true; /* Put into RX queue */ if (_queue_put(&ftdi_rxqueue, ch) < 0) overflow = true; } if (overflow) { /* We lost data because our RX queue was full. Disable OUT interrupt; we'll reenable after emptying some bytes from the RX queue. */ UEIENX &= ~(1 << RXOUTE); } Endpoint_ClearOUT(); } ISR(USB_COM_vect, ISR_BLOCK) { uint8_t PrevSelectedEndpoint = Endpoint_GetCurrentEndpoint(); if (UEINT & (1 << ENDPOINT_CONTROLEP)) { Endpoint_SelectEndpoint(ENDPOINT_CONTROLEP); if (Endpoint_IsSETUPReceived()) USB_Device_ProcessControlRequest(); } if (UEINT & (1 << (FTDI_TX_EPADDR & ENDPOINT_EPNUM_MASK))) { Endpoint_SelectEndpoint(FTDI_TX_EPADDR); if (Endpoint_IsINReady()) { /* Handle IN (device-to-host) */ ftdi_bulk_in(); } } if (UEINT & (1 << (FTDI_RX_EPADDR & ENDPOINT_EPNUM_MASK))) { Endpoint_SelectEndpoint(FTDI_RX_EPADDR); if (Endpoint_IsOUTReceived()) { /* Handle OUT (host-to-device) */ ftdi_bulk_out(); } } Endpoint_SelectEndpoint(PrevSelectedEndpoint); } /* Get / put routines */ void ftdi_putchar(uint8_t c) { /* Block until we can enqueue it, or drop if nonblocking */ while (queue_put(&ftdi_txqueue, c) == -1) if (ftdi_blocking_out == false) return; /* Protect against race conditions by double checking that we still have data to send, while interrupts are disabled. */ ATOMIC_BLOCK(ATOMIC_RESTORESTATE) { if (_queue_get_len(&ftdi_txqueue) > 0) { ftdi_status.temt = 0; maybe_enable_tx(); } } } int ftdi_getchar(void) { int ret; /* Block until byte is ready, or return 0 if nonblocking */ while ((ret = queue_get(&ftdi_rxqueue)) < 0) if (ftdi_blocking_in == false) return -1; /* Protect against race conditions by double checking that we still have space for new data, while interrupts are disabled. */ ATOMIC_BLOCK(ATOMIC_RESTORESTATE) { /* Reenable the OUT endpoint interrupt, since we read some data */ if (USB_DeviceState == DEVICE_STATE_Configured) { Endpoint_SelectEndpoint(FTDI_RX_EPADDR); UEIENX |= (1 << RXOUTE); } } return ret; } bool ftdi_can_get(void) { return queue_can_get(&ftdi_rxqueue); } bool ftdi_get_dtr(void) { return ctrl_status.dtr; }