#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include "bsp/board.h"
#include "tusb.h"
#include "usb_descriptors.h"
enum {
BLINK_NOT_MOUNTED = 250,
BLINK_MOUNTED = 1000,
BLINK_SUSPENDED = 2500,
BLINK_ALWAYS_ON = UINT32_MAX,
BLINK_ALWAYS_OFF = 0
};
static uint32_t blink_interval_ms = BLINK_NOT_MOUNTED;
#define URL "example.tinyusb.org/webusb-serial/"
const tusb_desc_webusb_url_t desc_url =
{
.bLength = 3 + sizeof(URL) - 1,
.bDescriptorType = 3, .bScheme = 1, .url = URL
};
static bool web_serial_connected = false;
void led_blinking_task(void);
void cdc_task(void);
void webserial_task(void);
int main(void)
{
board_init();
tusb_init();
while (1)
{
tud_task(); cdc_task();
webserial_task();
led_blinking_task();
}
return 0;
}
void echo_all(uint8_t buf[], uint32_t count)
{
if ( web_serial_connected )
{
tud_vendor_write(buf, count);
}
if ( tud_cdc_connected() )
{
for(uint32_t i=0; i<count; i++)
{
tud_cdc_write_char(buf[i]);
if ( buf[i] == '\r' ) tud_cdc_write_char('\n');
}
tud_cdc_write_flush();
}
}
void tud_mount_cb(void)
{
blink_interval_ms = BLINK_MOUNTED;
}
void tud_umount_cb(void)
{
blink_interval_ms = BLINK_NOT_MOUNTED;
}
void tud_suspend_cb(bool remote_wakeup_en)
{
(void) remote_wakeup_en;
blink_interval_ms = BLINK_SUSPENDED;
}
void tud_resume_cb(void)
{
blink_interval_ms = BLINK_MOUNTED;
}
bool tud_vendor_control_xfer_cb(uint8_t rhport, uint8_t stage, tusb_control_request_t const * request)
{
if (stage != CONTROL_STAGE_SETUP) return true;
switch (request->bmRequestType_bit.type)
{
case TUSB_REQ_TYPE_VENDOR:
switch (request->bRequest)
{
case VENDOR_REQUEST_WEBUSB:
return tud_control_xfer(rhport, request, (void*)(uintptr_t) &desc_url, desc_url.bLength);
case VENDOR_REQUEST_MICROSOFT:
if ( request->wIndex == 7 )
{
uint16_t total_len;
memcpy(&total_len, desc_ms_os_20+8, 2);
return tud_control_xfer(rhport, request, (void*)(uintptr_t) desc_ms_os_20, total_len);
}else
{
return false;
}
default: break;
}
break;
case TUSB_REQ_TYPE_CLASS:
if (request->bRequest == 0x22)
{
web_serial_connected = (request->wValue != 0);
if ( web_serial_connected )
{
board_led_write(true);
blink_interval_ms = BLINK_ALWAYS_ON;
tud_vendor_write_str("\r\nTinyUSB WebUSB device example\r\n");
}else
{
blink_interval_ms = BLINK_MOUNTED;
}
return tud_control_status(rhport, request);
}
break;
default: break;
}
return false;
}
void webserial_task(void)
{
if ( web_serial_connected )
{
if ( tud_vendor_available() )
{
uint8_t buf[64];
uint32_t count = tud_vendor_read(buf, sizeof(buf));
echo_all(buf, count);
}
}
}
void cdc_task(void)
{
if ( tud_cdc_connected() )
{
if ( tud_cdc_available() )
{
uint8_t buf[64];
uint32_t count = tud_cdc_read(buf, sizeof(buf));
echo_all(buf, count);
}
}
}
void tud_cdc_line_state_cb(uint8_t itf, bool dtr, bool rts)
{
(void) itf;
if ( dtr && rts )
{
tud_cdc_write_str("\r\nTinyUSB WebUSB device example\r\n");
}
}
void tud_cdc_rx_cb(uint8_t itf)
{
(void) itf;
}
void led_blinking_task(void)
{
static uint32_t start_ms = 0;
static bool led_state = false;
if ( board_millis() - start_ms < blink_interval_ms) return; start_ms += blink_interval_ms;
board_led_write(led_state);
led_state = 1 - led_state; }