2021-10-14 09:04:00 +01:00
|
|
|
#include <stdint.h>
|
|
|
|
#include <esp_log.h>
|
|
|
|
#include <freertos/FreeRTOS.h>
|
|
|
|
#include <freertos/task.h>
|
2021-10-17 15:06:53 +01:00
|
|
|
#include <freertos/stream_buffer.h>
|
2021-10-14 09:04:00 +01:00
|
|
|
#include <tinyusb.h>
|
|
|
|
#include <tusb_cdc_acm.h>
|
|
|
|
#include <sdkconfig.h>
|
|
|
|
|
|
|
|
static const char* TAG = "usb-cdc";
|
2021-10-17 15:06:53 +01:00
|
|
|
static uint8_t buffer_rx[CONFIG_USB_CDC_RX_BUFSIZE + 1];
|
2021-10-14 09:04:00 +01:00
|
|
|
|
2021-10-17 15:06:53 +01:00
|
|
|
// #define buffer_tx_size 10000
|
|
|
|
// static uint8_t buffer_tx[buffer_tx_size + 1];
|
|
|
|
// static size_t buffer_tx_counter = 0;
|
2021-10-14 09:04:00 +01:00
|
|
|
|
2021-10-17 15:06:53 +01:00
|
|
|
typedef struct {
|
|
|
|
volatile bool connected;
|
|
|
|
volatile bool dtr;
|
|
|
|
volatile bool rts;
|
|
|
|
|
|
|
|
StreamBufferHandle_t rx_stream;
|
|
|
|
volatile bool rx_stream_full;
|
|
|
|
} FuriHalVcp;
|
|
|
|
|
|
|
|
static FuriHalVcp furi_hal_vcp;
|
|
|
|
|
|
|
|
size_t usb_cdc_rx(uint8_t* buffer, size_t size) {
|
|
|
|
size_t received = xStreamBufferReceive(furi_hal_vcp.rx_stream, buffer, size, portMAX_DELAY);
|
|
|
|
|
|
|
|
if(furi_hal_vcp.rx_stream_full &&
|
|
|
|
xStreamBufferSpacesAvailable(furi_hal_vcp.rx_stream) >= CONFIG_USB_CDC_RX_BUFSIZE) {
|
|
|
|
furi_hal_vcp.rx_stream_full = false;
|
|
|
|
ESP_LOGW(TAG, "Stream is not full now");
|
|
|
|
}
|
|
|
|
|
|
|
|
return received;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t usb_cdc_rx_with_timeout(uint8_t* buffer, size_t size, uint32_t timeout) {
|
|
|
|
size_t received = xStreamBufferReceive(furi_hal_vcp.rx_stream, buffer, size, timeout);
|
|
|
|
|
|
|
|
if(furi_hal_vcp.rx_stream_full &&
|
|
|
|
xStreamBufferSpacesAvailable(furi_hal_vcp.rx_stream) >= CONFIG_USB_CDC_RX_BUFSIZE) {
|
|
|
|
furi_hal_vcp.rx_stream_full = false;
|
|
|
|
ESP_LOGW(TAG, "Stream is not full now");
|
2021-10-14 09:04:00 +01:00
|
|
|
}
|
|
|
|
|
2021-10-17 15:06:53 +01:00
|
|
|
return received;
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cdc_tx(uint8_t* buffer, size_t size, bool flush) {
|
|
|
|
/*if(furi_hal_vcp.connected) {
|
|
|
|
//printf("%c", buffer[0]);
|
|
|
|
if((buffer_tx_counter + size) < buffer_tx_size) {
|
|
|
|
memcpy(&buffer_tx[buffer_tx_counter], buffer, size);
|
|
|
|
buffer_tx_counter += size;
|
|
|
|
}
|
|
|
|
|
|
|
|
while(size > 0) {
|
|
|
|
size_t tx_size = tinyusb_cdcacm_write_queue(TINYUSB_USBDEV_0, buffer, size);
|
|
|
|
size -= tx_size;
|
|
|
|
buffer += tx_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(flush) {
|
|
|
|
buffer_tx[buffer_tx_counter] = '\0';
|
|
|
|
ESP_LOGI(TAG, "tx> %s", buffer_tx);
|
|
|
|
buffer_tx_counter = 0;
|
|
|
|
|
|
|
|
// TODO: timeout size
|
|
|
|
ESP_ERROR_CHECK_WITHOUT_ABORT(tinyusb_cdcacm_write_flush(TINYUSB_USBDEV_0, 10));
|
|
|
|
}
|
|
|
|
}*/
|
|
|
|
while(size > 0) {
|
|
|
|
size_t tx_size = tinyusb_cdcacm_write_queue(TINYUSB_USBDEV_0, buffer, size);
|
|
|
|
size -= tx_size;
|
|
|
|
buffer += tx_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(flush) {
|
|
|
|
// TODO: timeout size
|
|
|
|
ESP_ERROR_CHECK_WITHOUT_ABORT(tinyusb_cdcacm_write_flush(TINYUSB_USBDEV_0, 100));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cdc_rx_callback(int itf, cdcacm_event_t* event) {
|
|
|
|
uint16_t max_len = xStreamBufferSpacesAvailable(furi_hal_vcp.rx_stream);
|
|
|
|
|
|
|
|
if(max_len > 0) {
|
|
|
|
if(max_len > CONFIG_USB_CDC_RX_BUFSIZE) max_len = CONFIG_USB_CDC_RX_BUFSIZE;
|
|
|
|
size_t rx_size = 0;
|
|
|
|
esp_err_t err = tinyusb_cdcacm_read(itf, buffer_rx, max_len, &rx_size);
|
|
|
|
|
|
|
|
if(err == ESP_OK) {
|
|
|
|
if(rx_size > 0) {
|
|
|
|
BaseType_t xHigherPriorityTaskWoken = pdFALSE;
|
|
|
|
size_t ret = xStreamBufferSendFromISR(
|
|
|
|
furi_hal_vcp.rx_stream, buffer_rx, rx_size, &xHigherPriorityTaskWoken);
|
|
|
|
ESP_ERROR_CHECK(ret != rx_size);
|
|
|
|
// buffer_rx[rx_size] = '\0';
|
|
|
|
// ESP_LOGI(TAG, "rx< %s", buffer_rx);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ESP_LOGE(TAG, "Read error");
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
furi_hal_vcp.rx_stream_full = true;
|
|
|
|
ESP_LOGE(TAG, "Stream is full");
|
|
|
|
};
|
2021-10-14 09:04:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cdc_line_state_changed_callback(int itf, cdcacm_event_t* event) {
|
2021-10-17 15:06:53 +01:00
|
|
|
furi_hal_vcp.dtr = event->line_state_changed_data.dtr;
|
|
|
|
furi_hal_vcp.rts = event->line_state_changed_data.rts;
|
2021-10-14 09:04:00 +01:00
|
|
|
|
2021-10-17 15:06:53 +01:00
|
|
|
ESP_LOGI(TAG, "Line state changed! dtr:%d, rst:%d", furi_hal_vcp.dtr, furi_hal_vcp.rts);
|
2021-10-14 09:04:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void usb_cdc_line_coding_changed_callback(int itf, cdcacm_event_t* event) {
|
|
|
|
uint32_t bit_rate = event->line_coding_changed_data.p_line_coding->bit_rate;
|
|
|
|
uint8_t stop_bits = event->line_coding_changed_data.p_line_coding->stop_bits;
|
|
|
|
uint8_t parity = event->line_coding_changed_data.p_line_coding->parity;
|
|
|
|
uint8_t data_bits = event->line_coding_changed_data.p_line_coding->data_bits;
|
|
|
|
|
|
|
|
ESP_LOGI(
|
|
|
|
TAG,
|
|
|
|
"Line coding changed! bit_rate:%d, stop_bits:%d, parity:%d, data_bits:%d",
|
|
|
|
bit_rate,
|
|
|
|
stop_bits,
|
|
|
|
parity,
|
|
|
|
data_bits);
|
|
|
|
}
|
|
|
|
|
2021-10-17 15:06:53 +01:00
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
// Device callbacks
|
|
|
|
//--------------------------------------------------------------------+
|
|
|
|
void tud_mount_cb(void) {
|
|
|
|
ESP_LOGI(TAG, "Mount");
|
|
|
|
furi_hal_vcp.connected = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void tud_umount_cb(void) {
|
|
|
|
ESP_LOGI(TAG, "Unmount");
|
|
|
|
furi_hal_vcp.connected = false;
|
|
|
|
}
|
|
|
|
|
2021-10-14 09:04:00 +01:00
|
|
|
void usb_cdc_init(void) {
|
2021-10-17 15:06:53 +01:00
|
|
|
furi_hal_vcp.connected = false;
|
|
|
|
furi_hal_vcp.rx_stream = xStreamBufferCreate((CONFIG_USB_CDC_RX_BUFSIZE * 16), 1);
|
|
|
|
furi_hal_vcp.rx_stream_full = false;
|
|
|
|
|
2021-10-14 09:04:00 +01:00
|
|
|
ESP_LOGI(TAG, "USB initialization");
|
|
|
|
tinyusb_config_t tusb_cfg = {};
|
|
|
|
ESP_ERROR_CHECK(tinyusb_driver_install(&tusb_cfg));
|
|
|
|
|
|
|
|
tinyusb_config_cdcacm_t amc_cfg = {
|
|
|
|
.usb_dev = TINYUSB_USBDEV_0,
|
|
|
|
.cdc_port = TINYUSB_CDC_ACM_0,
|
|
|
|
.rx_unread_buf_sz = 64,
|
|
|
|
.callback_rx = &usb_cdc_rx_callback,
|
|
|
|
.callback_rx_wanted_char = NULL,
|
|
|
|
.callback_line_state_changed = &usb_cdc_line_state_changed_callback,
|
|
|
|
.callback_line_coding_changed = &usb_cdc_line_coding_changed_callback};
|
|
|
|
|
|
|
|
ESP_ERROR_CHECK(tusb_cdc_acm_init(&amc_cfg));
|
|
|
|
|
|
|
|
ESP_LOGI(TAG, "USB initialization DONE");
|
|
|
|
}
|