mirror of
https://github.com/hathach/tinyusb.git
synced 2025-03-24 07:43:38 +00:00
usbnet: OS-agnostic (Windows/Linux/macOS) network driver
This commit is contained in:
parent
13a3081d30
commit
bb3bbcc00b
examples/device/net_lwip_webserver/src
src
@ -26,13 +26,14 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
depending on the value of CFG_TUD_NET (tusb_config.h), this can be a CDC-ECM, RNDIS, or CDC-EEM USB virtual network adapter
|
depending on the value of CFG_TUD_NET (tusb_config.h), this can be a RNDIS+CDC-ECM or CDC-EEM USB virtual network adapter
|
||||||
|
|
||||||
CDC-ECM should be valid on Linux and MacOS hosts
|
OPT_NET_RNDIS_ECM : RNDIS should be valid on Linux and Windows hosts, and CDC-ECM should be valid on Linux and MacOS hosts
|
||||||
RNDIS should be valid on Linux and Windows hosts
|
OPT_NET_EEM : CDC-EEM should be valid on Linux hosts
|
||||||
CDC-EEM should be valid on Linux hosts
|
|
||||||
|
|
||||||
You *must* customize tusb_config.h to set the CFG_TUD_NET definition to the type of these network adapters to emulate.
|
OPT_NET_RNDIS_ECM should be the best choice, as it makes for a hopefully universal solution.
|
||||||
|
|
||||||
|
You *must* customize tusb_config.h to set the CFG_TUD_NET definition to the type of these network option.
|
||||||
|
|
||||||
The MCU appears to the host as IP address 192.168.7.1, and provides a DHCP server, DNS server, and web server.
|
The MCU appears to the host as IP address 192.168.7.1, and provides a DHCP server, DNS server, and web server.
|
||||||
*/
|
*/
|
||||||
|
@ -79,8 +79,7 @@
|
|||||||
#define CFG_TUD_HID 0
|
#define CFG_TUD_HID 0
|
||||||
#define CFG_TUD_MIDI 0
|
#define CFG_TUD_MIDI 0
|
||||||
#define CFG_TUD_VENDOR 0
|
#define CFG_TUD_VENDOR 0
|
||||||
//#define CFG_TUD_NET OPT_NET_ECM
|
#define CFG_TUD_NET OPT_NET_RNDIS_ECM
|
||||||
#define CFG_TUD_NET OPT_NET_RNDIS
|
|
||||||
//#define CFG_TUD_NET OPT_NET_EEM
|
//#define CFG_TUD_NET OPT_NET_EEM
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
@ -63,13 +63,17 @@ tusb_desc_device_t const desc_device =
|
|||||||
|
|
||||||
.idVendor = 0xCafe,
|
.idVendor = 0xCafe,
|
||||||
.idProduct = USB_PID,
|
.idProduct = USB_PID,
|
||||||
.bcdDevice = 0x0100,
|
.bcdDevice = 0x0101,
|
||||||
|
|
||||||
.iManufacturer = STRID_MANUFACTURER,
|
.iManufacturer = STRID_MANUFACTURER,
|
||||||
.iProduct = STRID_PRODUCT,
|
.iProduct = STRID_PRODUCT,
|
||||||
.iSerialNumber = STRID_SERIAL,
|
.iSerialNumber = STRID_SERIAL,
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
.bNumConfigurations = 0x01
|
.bNumConfigurations = 0x01
|
||||||
|
#else
|
||||||
|
.bNumConfigurations = 0x02
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
// Invoked when received GET DEVICE DESCRIPTOR
|
// Invoked when received GET DEVICE DESCRIPTOR
|
||||||
@ -85,16 +89,23 @@ uint8_t const * tud_descriptor_device_cb(void)
|
|||||||
enum
|
enum
|
||||||
{
|
{
|
||||||
ITF_NUM_CDC = 0,
|
ITF_NUM_CDC = 0,
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
ITF_NUM_CDC_DATA,
|
ITF_NUM_CDC_DATA,
|
||||||
|
#endif
|
||||||
ITF_NUM_TOTAL
|
ITF_NUM_TOTAL
|
||||||
};
|
};
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
enum
|
||||||
#define CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_CDC_ECM_DESC_LEN)
|
{
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
CONFIG_NUM_DEFAULT = 1,
|
||||||
#define CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_RNDIS_DESC_LEN)
|
CONFIG_NUM_ALTERNATE = 2,
|
||||||
#elif CFG_TUD_NET == OPT_NET_EEM
|
};
|
||||||
#define CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_CDC_EEM_DESC_LEN)
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
|
#define MAIN_CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_CDC_EEM_DESC_LEN)
|
||||||
|
#else
|
||||||
|
#define MAIN_CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_RNDIS_DESC_LEN)
|
||||||
|
#define ALT_CONFIG_TOTAL_LEN (TUD_CONFIG_DESC_LEN + TUD_CDC_ECM_DESC_LEN)
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if CFG_TUSB_MCU == OPT_MCU_LPC175X_6X || CFG_TUSB_MCU == OPT_MCU_LPC177X_8X || CFG_TUSB_MCU == OPT_MCU_LPC40XX
|
#if CFG_TUSB_MCU == OPT_MCU_LPC175X_6X || CFG_TUSB_MCU == OPT_MCU_LPC177X_8X || CFG_TUSB_MCU == OPT_MCU_LPC40XX
|
||||||
@ -105,30 +116,42 @@ enum
|
|||||||
#define EPNUM_CDC 2
|
#define EPNUM_CDC 2
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
uint8_t const desc_configuration[] =
|
static uint8_t const main_configuration[] =
|
||||||
{
|
{
|
||||||
// Config number, interface count, string index, total length, attribute, power in mA
|
// Config number, interface count, string index, total length, attribute, power in mA
|
||||||
TUD_CONFIG_DESCRIPTOR(1, ITF_NUM_TOTAL, 0, CONFIG_TOTAL_LEN, 0, 100),
|
TUD_CONFIG_DESCRIPTOR(CONFIG_NUM_DEFAULT, ITF_NUM_TOTAL, 0, MAIN_CONFIG_TOTAL_LEN, 0, 100),
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
// Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size.
|
|
||||||
TUD_CDC_ECM_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, STRID_MAC, 0x81, 64, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE, CFG_TUD_NET_MTU),
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
|
||||||
// Interface number, string index, EP notification address and size, EP data address (out, in) and size.
|
|
||||||
TUD_RNDIS_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, 0x81, 8, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE),
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_EEM
|
|
||||||
// Interface number, description string index, EP data address (out, in) and size.
|
// Interface number, description string index, EP data address (out, in) and size.
|
||||||
TUD_CDC_EEM_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE),
|
TUD_CDC_EEM_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE),
|
||||||
|
#else
|
||||||
|
// Interface number, string index, EP notification address and size, EP data address (out, in) and size.
|
||||||
|
TUD_RNDIS_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, 0x81, 8, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE),
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
|
static uint8_t const alt_configuration[] =
|
||||||
|
{
|
||||||
|
// Config number, interface count, string index, total length, attribute, power in mA
|
||||||
|
TUD_CONFIG_DESCRIPTOR(CONFIG_NUM_ALTERNATE, ITF_NUM_TOTAL, 0, ALT_CONFIG_TOTAL_LEN, 0, 100),
|
||||||
|
|
||||||
|
// Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size.
|
||||||
|
TUD_CDC_ECM_DESCRIPTOR(ITF_NUM_CDC, STRID_INTERFACE, STRID_MAC, 0x81, 64, EPNUM_CDC, 0x80 | EPNUM_CDC, CFG_TUD_NET_ENDPOINT_SIZE, CFG_TUD_NET_MTU),
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
// Invoked when received GET CONFIGURATION DESCRIPTOR
|
// Invoked when received GET CONFIGURATION DESCRIPTOR
|
||||||
// Application return pointer to descriptor
|
// Application return pointer to descriptor
|
||||||
// Descriptor contents must exist long enough for transfer to complete
|
// Descriptor contents must exist long enough for transfer to complete
|
||||||
uint8_t const * tud_descriptor_configuration_cb(uint8_t index)
|
uint8_t const * tud_descriptor_configuration_cb(uint8_t index)
|
||||||
{
|
{
|
||||||
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
(void) index; // for multiple configurations
|
(void) index; // for multiple configurations
|
||||||
return desc_configuration;
|
return main_configuration;
|
||||||
|
#else
|
||||||
|
return (0 == index) ? main_configuration : alt_configuration;
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
@ -141,8 +164,8 @@ static char const* string_desc_arr [] =
|
|||||||
[STRID_LANGID] = (const char[]) { 0x09, 0x04 }, // supported language is English (0x0409)
|
[STRID_LANGID] = (const char[]) { 0x09, 0x04 }, // supported language is English (0x0409)
|
||||||
[STRID_MANUFACTURER] = "TinyUSB", // Manufacturer
|
[STRID_MANUFACTURER] = "TinyUSB", // Manufacturer
|
||||||
[STRID_PRODUCT] = "TinyUSB Device", // Product
|
[STRID_PRODUCT] = "TinyUSB Device", // Product
|
||||||
[STRID_SERIAL] = "123456", // Serials
|
[STRID_SERIAL] = "123456", // Serial
|
||||||
[STRID_INTERFACE] = "TinyUSB Network Interface" // CDC-ECM Interface
|
[STRID_INTERFACE] = "TinyUSB Network Interface" // Interface Description
|
||||||
|
|
||||||
// STRID_MAC index is handled separately
|
// STRID_MAC index is handled separately
|
||||||
};
|
};
|
||||||
|
@ -41,39 +41,57 @@ void rndis_class_set_handler(uint8_t *data, int size); /* found in ./misc/networ
|
|||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
uint8_t itf_num;
|
uint8_t itf_num;
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
uint8_t ep_notif;
|
uint8_t ep_notif;
|
||||||
|
bool ecm_mode;
|
||||||
|
#endif
|
||||||
uint8_t ep_in;
|
uint8_t ep_in;
|
||||||
uint8_t ep_out;
|
uint8_t ep_out;
|
||||||
} netd_interface_t;
|
} netd_interface_t;
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
#define CFG_TUD_NET_PACKET_PREFIX_LEN 0
|
|
||||||
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 0
|
|
||||||
#define CFG_TUD_NET_INTERFACESUBCLASS CDC_COMM_SUBCLASS_ETHERNET_NETWORKING_CONTROL_MODEL
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
|
||||||
#define CFG_TUD_NET_PACKET_PREFIX_LEN sizeof(rndis_data_packet_t)
|
|
||||||
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 0
|
|
||||||
#define CFG_TUD_NET_INTERFACESUBCLASS TUD_RNDIS_ITF_SUBCLASS
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_EEM
|
|
||||||
#define CFG_TUD_NET_PACKET_PREFIX_LEN 2
|
#define CFG_TUD_NET_PACKET_PREFIX_LEN 2
|
||||||
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 4
|
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 4
|
||||||
#define CFG_TUD_NET_INTERFACESUBCLASS CDC_COMM_SUBCLASS_ETHERNET_EMULATION_MODEL
|
#else
|
||||||
|
#define CFG_TUD_NET_PACKET_PREFIX_LEN sizeof(rndis_data_packet_t)
|
||||||
|
#define CFG_TUD_NET_PACKET_SUFFIX_LEN 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static uint8_t received[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
|
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static uint8_t received[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
|
||||||
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static uint8_t transmitted[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
|
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static uint8_t transmitted[CFG_TUD_NET_PACKET_PREFIX_LEN + CFG_TUD_NET_MTU + CFG_TUD_NET_PACKET_PREFIX_LEN];
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
struct ecm_notify_struct
|
||||||
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static tusb_control_request_t notify =
|
{
|
||||||
{
|
tusb_control_request_t header;
|
||||||
.bmRequestType = 0x21,
|
uint32_t downlink, uplink;
|
||||||
.bRequest = 0 /* NETWORK_CONNECTION */,
|
};
|
||||||
|
|
||||||
|
static const struct ecm_notify_struct ecm_notify_nc =
|
||||||
|
{
|
||||||
|
.header = {
|
||||||
|
.bmRequestType = 0xA1,
|
||||||
|
.bRequest = 0 /* NETWORK_CONNECTION aka NetworkConnection */,
|
||||||
.wValue = 1 /* Connected */,
|
.wValue = 1 /* Connected */,
|
||||||
.wLength = 0,
|
.wLength = 0,
|
||||||
};
|
},
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
};
|
||||||
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static uint8_t rndis_buf[120];
|
|
||||||
#endif
|
static const struct ecm_notify_struct ecm_notify_csc =
|
||||||
|
{
|
||||||
|
.header = {
|
||||||
|
.bmRequestType = 0xA1,
|
||||||
|
.bRequest = 0x2A /* CONNECTION_SPEED_CHANGE aka ConnectionSpeedChange */,
|
||||||
|
.wLength = 8,
|
||||||
|
},
|
||||||
|
.downlink = 9728000,
|
||||||
|
.uplink = 9728000,
|
||||||
|
};
|
||||||
|
|
||||||
|
CFG_TUSB_MEM_SECTION CFG_TUSB_MEM_ALIGN static union
|
||||||
|
{
|
||||||
|
uint8_t rndis_buf[120];
|
||||||
|
struct ecm_notify_struct ecm_buf;
|
||||||
|
} notify;
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// INTERNAL OBJECT & FUNCTION DECLARATION
|
// INTERNAL OBJECT & FUNCTION DECLARATION
|
||||||
@ -95,7 +113,9 @@ static void do_in_xfer(uint8_t *buf, uint16_t len)
|
|||||||
|
|
||||||
void netd_report(uint8_t *buf, uint16_t len)
|
void netd_report(uint8_t *buf, uint16_t len)
|
||||||
{
|
{
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
usbd_edpt_xfer(TUD_OPT_RHPORT, _netd_itf.ep_notif, buf, len);
|
usbd_edpt_xfer(TUD_OPT_RHPORT, _netd_itf.ep_notif, buf, len);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
@ -106,6 +126,10 @@ void netd_init(void)
|
|||||||
tu_memclr(&_netd_itf, sizeof(_netd_itf));
|
tu_memclr(&_netd_itf, sizeof(_netd_itf));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void netd_init_data(void)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
void netd_reset(uint8_t rhport)
|
void netd_reset(uint8_t rhport)
|
||||||
{
|
{
|
||||||
(void) rhport;
|
(void) rhport;
|
||||||
@ -113,21 +137,26 @@ void netd_reset(uint8_t rhport)
|
|||||||
netd_init();
|
netd_init();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
bool netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length)
|
bool netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length)
|
||||||
{
|
{
|
||||||
// sanity check the descriptor
|
// sanity check the descriptor
|
||||||
TU_ASSERT (CFG_TUD_NET_INTERFACESUBCLASS == itf_desc->bInterfaceSubClass);
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
|
TU_VERIFY (CDC_COMM_SUBCLASS_ETHERNET_EMULATION_MODEL == itf_desc->bInterfaceSubClass);
|
||||||
|
#else
|
||||||
|
_netd_itf.ecm_mode = (CDC_COMM_SUBCLASS_ETHERNET_NETWORKING_CONTROL_MODEL == itf_desc->bInterfaceSubClass);
|
||||||
|
TU_VERIFY ( (TUD_RNDIS_ITF_SUBCLASS == itf_desc->bInterfaceSubClass) || _netd_itf.ecm_mode );
|
||||||
|
#endif
|
||||||
|
|
||||||
// confirm interface hasn't already been allocated
|
// confirm interface hasn't already been allocated
|
||||||
TU_ASSERT(0 == _netd_itf.ep_in);
|
TU_ASSERT(0 == _netd_itf.ep_notif);
|
||||||
|
|
||||||
//------------- first Interface -------------//
|
//------------- Management Interface -------------//
|
||||||
_netd_itf.itf_num = itf_desc->bInterfaceNumber;
|
_netd_itf.itf_num = itf_desc->bInterfaceNumber;
|
||||||
|
|
||||||
uint8_t const * p_desc = tu_desc_next( itf_desc );
|
uint8_t const * p_desc = tu_desc_next( itf_desc );
|
||||||
(*p_length) = sizeof(tusb_desc_interface_t);
|
(*p_length) = sizeof(tusb_desc_interface_t);
|
||||||
|
|
||||||
#if CFG_TUD_NET != OPT_NET_EEM
|
|
||||||
// Communication Functional Descriptors
|
// Communication Functional Descriptors
|
||||||
while ( TUSB_DESC_CS_INTERFACE == tu_desc_type(p_desc) )
|
while ( TUSB_DESC_CS_INTERFACE == tu_desc_type(p_desc) )
|
||||||
{
|
{
|
||||||
@ -143,18 +172,28 @@ bool netd_open(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t
|
|||||||
_netd_itf.ep_notif = ((tusb_desc_endpoint_t const *) p_desc)->bEndpointAddress;
|
_netd_itf.ep_notif = ((tusb_desc_endpoint_t const *) p_desc)->bEndpointAddress;
|
||||||
|
|
||||||
(*p_length) += p_desc[DESC_OFFSET_LEN];
|
(*p_length) += p_desc[DESC_OFFSET_LEN];
|
||||||
p_desc = tu_desc_next(p_desc);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//------------- second Interface -------------//
|
return true;
|
||||||
if ( (TUSB_DESC_INTERFACE == p_desc[DESC_OFFSET_TYPE]) &&
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
bool netd_open_data(uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length)
|
||||||
|
{
|
||||||
|
// confirm interface hasn't already been allocated
|
||||||
|
TU_ASSERT(0 == _netd_itf.ep_in);
|
||||||
|
|
||||||
|
uint8_t const * p_desc = tu_desc_next( itf_desc );
|
||||||
|
(*p_length) = sizeof(tusb_desc_interface_t);
|
||||||
|
|
||||||
|
//------------- Data Interface -------------//
|
||||||
|
while ( (TUSB_DESC_INTERFACE == p_desc[DESC_OFFSET_TYPE]) &&
|
||||||
(TUSB_CLASS_CDC_DATA == ((tusb_desc_interface_t const *) p_desc)->bInterfaceClass) )
|
(TUSB_CLASS_CDC_DATA == ((tusb_desc_interface_t const *) p_desc)->bInterfaceClass) )
|
||||||
{
|
{
|
||||||
// next to endpoint descriptor
|
// next to endpoint descriptor
|
||||||
p_desc = tu_desc_next(p_desc);
|
p_desc = tu_desc_next(p_desc);
|
||||||
(*p_length) += sizeof(tusb_desc_interface_t);
|
(*p_length) += sizeof(tusb_desc_interface_t);
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
|
|
||||||
if (TUSB_DESC_ENDPOINT == p_desc[DESC_OFFSET_TYPE])
|
if (TUSB_DESC_ENDPOINT == p_desc[DESC_OFFSET_TYPE])
|
||||||
{
|
{
|
||||||
@ -184,18 +223,25 @@ bool netd_control_complete(uint8_t rhport, tusb_control_request_t const * reques
|
|||||||
// Handle class request only
|
// Handle class request only
|
||||||
TU_VERIFY (request->bmRequestType_bit.type == TUSB_REQ_TYPE_CLASS);
|
TU_VERIFY (request->bmRequestType_bit.type == TUSB_REQ_TYPE_CLASS);
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
TU_VERIFY (_netd_itf.itf_num == request->wIndex);
|
TU_VERIFY (_netd_itf.itf_num == request->wIndex);
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_RNDIS
|
if ( !_netd_itf.ecm_mode && (request->bmRequestType_bit.direction == TUSB_DIR_OUT) )
|
||||||
if (request->bmRequestType_bit.direction == TUSB_DIR_OUT)
|
|
||||||
{
|
{
|
||||||
rndis_class_set_handler(rndis_buf, request->wLength);
|
rndis_class_set_handler(notify.rndis_buf, request->wLength);
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void ecm_report(bool nc)
|
||||||
|
{
|
||||||
|
notify.ecm_buf = (nc) ? ecm_notify_nc : ecm_notify_csc;
|
||||||
|
notify.ecm_buf.header.wIndex = _netd_itf.itf_num;
|
||||||
|
netd_report((uint8_t *)¬ify.ecm_buf, (nc) ? sizeof(notify.ecm_buf.header) : sizeof(notify.ecm_buf));
|
||||||
|
}
|
||||||
|
|
||||||
// Handle class control request
|
// Handle class control request
|
||||||
// return false to stall control endpoint (e.g unsupported request)
|
// return false to stall control endpoint (e.g unsupported request)
|
||||||
bool netd_control_request(uint8_t rhport, tusb_control_request_t const * request)
|
bool netd_control_request(uint8_t rhport, tusb_control_request_t const * request)
|
||||||
@ -205,28 +251,32 @@ bool netd_control_request(uint8_t rhport, tusb_control_request_t const * request
|
|||||||
|
|
||||||
TU_VERIFY (_netd_itf.itf_num == request->wIndex);
|
TU_VERIFY (_netd_itf.itf_num == request->wIndex);
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
/* the only required CDC-ECM Management Element Request is SetEthernetPacketFilter */
|
(void)rhport;
|
||||||
if (0x43 /* SET_ETHERNET_PACKET_FILTER */ == request->bRequest)
|
#else
|
||||||
|
if (_netd_itf.ecm_mode)
|
||||||
{
|
{
|
||||||
tud_control_xfer(rhport, request, NULL, 0);
|
/* the only required CDC-ECM Management Element Request is SetEthernetPacketFilter */
|
||||||
notify.wIndex = request->wIndex;
|
if (0x43 /* SET_ETHERNET_PACKET_FILTER */ == request->bRequest)
|
||||||
usbd_edpt_xfer(TUD_OPT_RHPORT, _netd_itf.ep_notif, (uint8_t *)¬ify, sizeof(notify));
|
{
|
||||||
}
|
tud_control_xfer(rhport, request, NULL, 0);
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
ecm_report(true);
|
||||||
if (request->bmRequestType_bit.direction == TUSB_DIR_IN)
|
}
|
||||||
{
|
|
||||||
rndis_generic_msg_t *rndis_msg = (rndis_generic_msg_t *)rndis_buf;
|
|
||||||
uint32_t msglen = tu_le32toh(rndis_msg->MessageLength);
|
|
||||||
TU_ASSERT(msglen <= sizeof(rndis_buf));
|
|
||||||
tud_control_xfer(rhport, request, rndis_buf, msglen);
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
tud_control_xfer(rhport, request, rndis_buf, sizeof(rndis_buf));
|
if (request->bmRequestType_bit.direction == TUSB_DIR_IN)
|
||||||
|
{
|
||||||
|
rndis_generic_msg_t *rndis_msg = (rndis_generic_msg_t *)notify.rndis_buf;
|
||||||
|
uint32_t msglen = tu_le32toh(rndis_msg->MessageLength);
|
||||||
|
TU_ASSERT(msglen <= sizeof(notify.rndis_buf));
|
||||||
|
tud_control_xfer(rhport, request, notify.rndis_buf, msglen);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
tud_control_xfer(rhport, request, notify.rndis_buf, sizeof(notify.rndis_buf));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
#else
|
|
||||||
(void)rhport;
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -244,18 +294,7 @@ static void handle_incoming_packet(uint32_t len)
|
|||||||
uint8_t *pnt = received;
|
uint8_t *pnt = received;
|
||||||
uint32_t size = 0;
|
uint32_t size = 0;
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_ECM
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
size = len;
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_RNDIS
|
|
||||||
rndis_data_packet_t *r = (rndis_data_packet_t *)pnt;
|
|
||||||
if (len >= sizeof(rndis_data_packet_t))
|
|
||||||
if ( (r->MessageType == REMOTE_NDIS_PACKET_MSG) && (r->MessageLength <= len))
|
|
||||||
if ( (r->DataOffset + offsetof(rndis_data_packet_t, DataOffset) + r->DataLength) <= len)
|
|
||||||
{
|
|
||||||
pnt = &received[r->DataOffset + offsetof(rndis_data_packet_t, DataOffset)];
|
|
||||||
size = r->DataLength;
|
|
||||||
}
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_EEM
|
|
||||||
struct cdc_eem_packet_header *hdr = (struct cdc_eem_packet_header *)pnt;
|
struct cdc_eem_packet_header *hdr = (struct cdc_eem_packet_header *)pnt;
|
||||||
|
|
||||||
(void)len;
|
(void)len;
|
||||||
@ -271,26 +310,45 @@ static void handle_incoming_packet(uint32_t len)
|
|||||||
pnt += CFG_TUD_NET_PACKET_PREFIX_LEN;
|
pnt += CFG_TUD_NET_PACKET_PREFIX_LEN;
|
||||||
size = hdr->length - 4; /* discard the unused CRC-32 */
|
size = hdr->length - 4; /* discard the unused CRC-32 */
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
if (_netd_itf.ecm_mode)
|
||||||
|
{
|
||||||
|
size = len;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
rndis_data_packet_t *r = (rndis_data_packet_t *)pnt;
|
||||||
|
if (len >= sizeof(rndis_data_packet_t))
|
||||||
|
if ( (r->MessageType == REMOTE_NDIS_PACKET_MSG) && (r->MessageLength <= len))
|
||||||
|
if ( (r->DataOffset + offsetof(rndis_data_packet_t, DataOffset) + r->DataLength) <= len)
|
||||||
|
{
|
||||||
|
pnt = &received[r->DataOffset + offsetof(rndis_data_packet_t, DataOffset)];
|
||||||
|
size = r->DataLength;
|
||||||
|
}
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
bool accepted = false;
|
||||||
|
|
||||||
if (size)
|
if (size)
|
||||||
{
|
{
|
||||||
struct pbuf *p = pbuf_alloc(PBUF_RAW, size, PBUF_POOL);
|
struct pbuf *p = pbuf_alloc(PBUF_RAW, size, PBUF_POOL);
|
||||||
bool accepted = true;
|
|
||||||
|
|
||||||
if (p)
|
if (p)
|
||||||
{
|
{
|
||||||
memcpy(p->payload, pnt, size);
|
memcpy(p->payload, pnt, size);
|
||||||
p->len = size;
|
p->len = size;
|
||||||
accepted = tud_network_recv_cb(p);
|
accepted = tud_network_recv_cb(p);
|
||||||
}
|
|
||||||
|
|
||||||
if (!p || !accepted)
|
if (!accepted) pbuf_free(p);
|
||||||
{
|
|
||||||
/* if a buffer couldn't be allocated or accepted by the callback, we must discard this packet */
|
|
||||||
tud_network_recv_renew();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!accepted)
|
||||||
|
{
|
||||||
|
/* if a buffer was never handled by user code, we must renew on the user's behalf */
|
||||||
|
tud_network_recv_renew();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool netd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
|
bool netd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes)
|
||||||
@ -320,6 +378,13 @@ bool netd_xfer_cb(uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_RNDIS_ECM
|
||||||
|
if ( _netd_itf.ecm_mode && (ep_addr == _netd_itf.ep_notif) )
|
||||||
|
{
|
||||||
|
if (sizeof(notify.ecm_buf.header) == xferred_bytes) ecm_report(false);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -337,7 +402,11 @@ void tud_network_xmit(struct pbuf *p)
|
|||||||
if (!can_xmit)
|
if (!can_xmit)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
len = CFG_TUD_NET_PACKET_PREFIX_LEN;
|
len = CFG_TUD_NET_PACKET_PREFIX_LEN;
|
||||||
|
#else
|
||||||
|
len = (_netd_itf.ecm_mode) ? 0 : CFG_TUD_NET_PACKET_PREFIX_LEN;
|
||||||
|
#endif
|
||||||
data = transmitted + len;
|
data = transmitted + len;
|
||||||
|
|
||||||
for(q = p; q != NULL; q = q->next)
|
for(q = p; q != NULL; q = q->next)
|
||||||
@ -347,14 +416,7 @@ void tud_network_xmit(struct pbuf *p)
|
|||||||
len += q->len;
|
len += q->len;
|
||||||
}
|
}
|
||||||
|
|
||||||
#if CFG_TUD_NET == OPT_NET_RNDIS
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
rndis_data_packet_t *hdr = (rndis_data_packet_t *)transmitted;
|
|
||||||
memset(hdr, 0, sizeof(rndis_data_packet_t));
|
|
||||||
hdr->MessageType = REMOTE_NDIS_PACKET_MSG;
|
|
||||||
hdr->MessageLength = len;
|
|
||||||
hdr->DataOffset = sizeof(rndis_data_packet_t) - offsetof(rndis_data_packet_t, DataOffset);
|
|
||||||
hdr->DataLength = len - sizeof(rndis_data_packet_t);
|
|
||||||
#elif CFG_TUD_NET == OPT_NET_EEM
|
|
||||||
struct cdc_eem_packet_header *hdr = (struct cdc_eem_packet_header *)transmitted;
|
struct cdc_eem_packet_header *hdr = (struct cdc_eem_packet_header *)transmitted;
|
||||||
/* append a fake CRC-32; the standard allows 0xDEADBEEF, which takes less CPU time */
|
/* append a fake CRC-32; the standard allows 0xDEADBEEF, which takes less CPU time */
|
||||||
data[0] = 0xDE; data[1] = 0xAD; data[2] = 0xBE; data[3] = 0xEF;
|
data[0] = 0xDE; data[1] = 0xAD; data[2] = 0xBE; data[3] = 0xEF;
|
||||||
@ -363,6 +425,16 @@ void tud_network_xmit(struct pbuf *p)
|
|||||||
hdr->bmType = 0; /* EEM Data Packet */
|
hdr->bmType = 0; /* EEM Data Packet */
|
||||||
hdr->length = len - sizeof(struct cdc_eem_packet_header);
|
hdr->length = len - sizeof(struct cdc_eem_packet_header);
|
||||||
hdr->bmCRC = 0; /* Ethernet Frame CRC-32 set to 0xDEADBEEF */
|
hdr->bmCRC = 0; /* Ethernet Frame CRC-32 set to 0xDEADBEEF */
|
||||||
|
#else
|
||||||
|
if (!_netd_itf.ecm_mode)
|
||||||
|
{
|
||||||
|
rndis_data_packet_t *hdr = (rndis_data_packet_t *)transmitted;
|
||||||
|
memset(hdr, 0, sizeof(rndis_data_packet_t));
|
||||||
|
hdr->MessageType = REMOTE_NDIS_PACKET_MSG;
|
||||||
|
hdr->MessageLength = len;
|
||||||
|
hdr->DataOffset = sizeof(rndis_data_packet_t) - offsetof(rndis_data_packet_t, DataOffset);
|
||||||
|
hdr->DataLength = len - sizeof(rndis_data_packet_t);
|
||||||
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
do_in_xfer(transmitted, len);
|
do_in_xfer(transmitted, len);
|
||||||
|
@ -73,8 +73,10 @@ void tud_network_xmit(struct pbuf *p);
|
|||||||
// INTERNAL USBD-CLASS DRIVER API
|
// INTERNAL USBD-CLASS DRIVER API
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
void netd_init (void);
|
void netd_init (void);
|
||||||
|
void netd_init_data (void);
|
||||||
void netd_reset (uint8_t rhport);
|
void netd_reset (uint8_t rhport);
|
||||||
bool netd_open (uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length);
|
bool netd_open (uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length);
|
||||||
|
bool netd_open_data (uint8_t rhport, tusb_desc_interface_t const * itf_desc, uint16_t *p_length);
|
||||||
bool netd_control_request (uint8_t rhport, tusb_control_request_t const * request);
|
bool netd_control_request (uint8_t rhport, tusb_control_request_t const * request);
|
||||||
bool netd_control_complete (uint8_t rhport, tusb_control_request_t const * request);
|
bool netd_control_complete (uint8_t rhport, tusb_control_request_t const * request);
|
||||||
bool netd_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes);
|
bool netd_xfer_cb (uint8_t rhport, uint8_t ep_addr, xfer_result_t result, uint32_t xferred_bytes);
|
||||||
|
@ -184,21 +184,47 @@ static usbd_class_driver_t const _usbd_driver[] =
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if CFG_TUD_NET
|
#if CFG_TUD_NET
|
||||||
|
#if CFG_TUD_NET != OPT_NET_EEM
|
||||||
|
/* RNDIS management interface */
|
||||||
{
|
{
|
||||||
.class_code =
|
.class_code = TUD_RNDIS_ITF_CLASS,
|
||||||
#if CFG_TUD_NET == OPT_NET_RNDIS
|
|
||||||
TUD_RNDIS_ITF_CLASS,
|
|
||||||
#else
|
|
||||||
TUSB_CLASS_CDC,
|
|
||||||
#endif
|
|
||||||
.init = netd_init,
|
.init = netd_init,
|
||||||
.reset = netd_reset,
|
.reset = netd_reset,
|
||||||
.open = netd_open,
|
.open = netd_open,
|
||||||
.control_request = netd_control_request,
|
.control_request = netd_control_request,
|
||||||
.control_complete = netd_control_complete,
|
.control_complete = netd_control_complete,
|
||||||
.xfer_cb = netd_xfer_cb,
|
.xfer_cb = netd_xfer_cb,
|
||||||
.sof = NULL
|
.sof = NULL,
|
||||||
},
|
},
|
||||||
|
#endif
|
||||||
|
/* CDC-ECM management interface; CDC-EEM data interface */
|
||||||
|
{
|
||||||
|
.class_code = TUSB_CLASS_CDC,
|
||||||
|
.init = netd_init,
|
||||||
|
.reset = netd_reset,
|
||||||
|
#if CFG_TUD_NET == OPT_NET_EEM
|
||||||
|
.open = netd_open_data,
|
||||||
|
#else
|
||||||
|
.open = netd_open,
|
||||||
|
#endif
|
||||||
|
.control_request = netd_control_request,
|
||||||
|
.control_complete = netd_control_complete,
|
||||||
|
.xfer_cb = netd_xfer_cb,
|
||||||
|
.sof = NULL,
|
||||||
|
},
|
||||||
|
/* RNDIS/CDC-ECM data interface */
|
||||||
|
#if CFG_TUD_NET != OPT_NET_EEM
|
||||||
|
{
|
||||||
|
.class_code = TUSB_CLASS_CDC_DATA,
|
||||||
|
.init = netd_init_data,
|
||||||
|
.reset = NULL,
|
||||||
|
.open = netd_open_data,
|
||||||
|
.control_request = NULL,
|
||||||
|
.control_complete = NULL,
|
||||||
|
.xfer_cb = netd_xfer_cb,
|
||||||
|
.sof = NULL,
|
||||||
|
},
|
||||||
|
#endif
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -342,8 +342,8 @@ TU_ATTR_WEAK bool tud_vendor_control_complete_cb(uint8_t rhport, tusb_control_re
|
|||||||
|
|
||||||
//------------- CDC-ECM -------------//
|
//------------- CDC-ECM -------------//
|
||||||
|
|
||||||
// Length of template descriptor: 62 bytes
|
// Length of template descriptor: 71 bytes
|
||||||
#define TUD_CDC_ECM_DESC_LEN (9+5+5+13+7+9+7+7)
|
#define TUD_CDC_ECM_DESC_LEN (9+5+5+13+7+9+9+7+7)
|
||||||
|
|
||||||
// CDC-ECM Descriptor Template
|
// CDC-ECM Descriptor Template
|
||||||
// Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size.
|
// Interface number, description string index, MAC address string index, EP notification address and size, EP data address (out, in), and size, max segment size.
|
||||||
@ -358,8 +358,10 @@ TU_ATTR_WEAK bool tud_vendor_control_complete_cb(uint8_t rhport, tusb_control_re
|
|||||||
13, TUSB_DESC_CS_INTERFACE, CDC_FUNC_DESC_ETHERNET_NETWORKING, _mac_stridx, 0, 0, 0, 0, U16_TO_U8S_LE(_maxsegmentsize), U16_TO_U8S_LE(0), 0,\
|
13, TUSB_DESC_CS_INTERFACE, CDC_FUNC_DESC_ETHERNET_NETWORKING, _mac_stridx, 0, 0, 0, 0, U16_TO_U8S_LE(_maxsegmentsize), U16_TO_U8S_LE(0), 0,\
|
||||||
/* Endpoint Notification */\
|
/* Endpoint Notification */\
|
||||||
7, TUSB_DESC_ENDPOINT, _ep_notif, TUSB_XFER_INTERRUPT, U16_TO_U8S_LE(_ep_notif_size), 1,\
|
7, TUSB_DESC_ENDPOINT, _ep_notif, TUSB_XFER_INTERRUPT, U16_TO_U8S_LE(_ep_notif_size), 1,\
|
||||||
/* CDC Data Interface */\
|
/* CDC Data Interface (default inactive) */\
|
||||||
9, TUSB_DESC_INTERFACE, (uint8_t)((_itfnum)+1), 0, 2, TUSB_CLASS_CDC_DATA, 0, 0, 0,\
|
9, TUSB_DESC_INTERFACE, (uint8_t)((_itfnum)+1), 0, 0, TUSB_CLASS_CDC_DATA, 0, 0, 0,\
|
||||||
|
/* CDC Data Interface (alternative active) */\
|
||||||
|
9, TUSB_DESC_INTERFACE, (uint8_t)((_itfnum)+1), 1, 2, TUSB_CLASS_CDC_DATA, 0, 0, 0,\
|
||||||
/* Endpoint In */\
|
/* Endpoint In */\
|
||||||
7, TUSB_DESC_ENDPOINT, _epin, TUSB_XFER_BULK, U16_TO_U8S_LE(_epsize), 0,\
|
7, TUSB_DESC_ENDPOINT, _epin, TUSB_XFER_BULK, U16_TO_U8S_LE(_epsize), 0,\
|
||||||
/* Endpoint Out */\
|
/* Endpoint Out */\
|
||||||
@ -372,7 +374,7 @@ TU_ATTR_WEAK bool tud_vendor_control_complete_cb(uint8_t rhport, tusb_control_re
|
|||||||
/* Windows XP */
|
/* Windows XP */
|
||||||
#define TUD_RNDIS_ITF_CLASS TUSB_CLASS_CDC
|
#define TUD_RNDIS_ITF_CLASS TUSB_CLASS_CDC
|
||||||
#define TUD_RNDIS_ITF_SUBCLASS CDC_COMM_SUBCLASS_ABSTRACT_CONTROL_MODEL
|
#define TUD_RNDIS_ITF_SUBCLASS CDC_COMM_SUBCLASS_ABSTRACT_CONTROL_MODEL
|
||||||
#define TUD_RNDIS_ITF_PROTOCOL CDC_COMM_PROTOCOL_MICROSOFT_RNDIS
|
#define TUD_RNDIS_ITF_PROTOCOL 0xFF /* CDC_COMM_PROTOCOL_MICROSOFT_RNDIS */
|
||||||
#else
|
#else
|
||||||
/* Windows 7+ */
|
/* Windows 7+ */
|
||||||
#define TUD_RNDIS_ITF_CLASS 0xE0
|
#define TUD_RNDIS_ITF_CLASS 0xE0
|
||||||
|
@ -128,9 +128,8 @@
|
|||||||
* \ref CFG_TUD_NET must be defined to one of these
|
* \ref CFG_TUD_NET must be defined to one of these
|
||||||
* @{ */
|
* @{ */
|
||||||
#define OPT_NET_NONE 0 ///< No network interface
|
#define OPT_NET_NONE 0 ///< No network interface
|
||||||
#define OPT_NET_ECM 1 ///< CDC-ECM
|
#define OPT_NET_RNDIS_ECM 1 ///< RNDIS+CDC-ECM
|
||||||
#define OPT_NET_RNDIS 2 ///< RNDIS
|
#define OPT_NET_EEM 2 ///< CDC-EEM
|
||||||
#define OPT_NET_EEM 3 ///< CDC-EEM
|
|
||||||
/** @} */
|
/** @} */
|
||||||
|
|
||||||
#ifndef CFG_TUSB_RHPORT0_MODE
|
#ifndef CFG_TUSB_RHPORT0_MODE
|
||||||
|
Loading…
x
Reference in New Issue
Block a user