mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 07:52:44 +01:00
362 lines
12 KiB
C
362 lines
12 KiB
C
/*
|
|
* Copyright (C) 2019 Koen Zandberg
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup usbus_cdc_ecm
|
|
* @{
|
|
* @file USBUS implementation for ethernet control model
|
|
*
|
|
* @author Koen Zandberg <koen@bergzand.net>
|
|
* @}
|
|
*/
|
|
|
|
#define USB_H_USER_IS_RIOT_INTERNAL
|
|
|
|
#include "event.h"
|
|
#include "fmt.h"
|
|
#include "kernel_defines.h"
|
|
#include "luid.h"
|
|
#include "net/ethernet.h"
|
|
#include "net/eui48.h"
|
|
#include "usb/cdc.h"
|
|
#include "usb/descriptor.h"
|
|
#include "usb/usbus.h"
|
|
#include "usb/usbus/control.h"
|
|
#include "usb/usbus/cdc/ecm.h"
|
|
|
|
#include <string.h>
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
static void _event_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbus_event_usb_t event);
|
|
static int _control_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbus_control_request_state_t state,
|
|
usb_setup_t *setup);
|
|
static void _transfer_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbdev_ep_t *ep, usbus_event_transfer_t event);
|
|
static void _init(usbus_t *usbus, usbus_handler_t *handler);
|
|
static void _handle_rx_flush_ev(event_t *ev);
|
|
static void _handle_tx_xmit(event_t *ev);
|
|
|
|
static size_t _gen_full_ecm_descriptor(usbus_t *usbus, void *arg);
|
|
|
|
static const usbus_descr_gen_funcs_t _ecm_descriptor = {
|
|
.fmt_post_descriptor = _gen_full_ecm_descriptor,
|
|
.len = {
|
|
.fixed_len = sizeof(usb_desc_cdc_t) +
|
|
sizeof(usb_desc_union_t) +
|
|
sizeof(usb_desc_ecm_t),
|
|
},
|
|
.len_type = USBUS_DESCR_LEN_FIXED,
|
|
};
|
|
|
|
static size_t _gen_union_descriptor(usbus_t *usbus, usbus_cdcecm_device_t *cdcecm)
|
|
{
|
|
usb_desc_union_t uni;
|
|
|
|
/* functional union descriptor */
|
|
uni.length = sizeof(usb_desc_union_t);
|
|
uni.type = USB_TYPE_DESCRIPTOR_CDC;
|
|
uni.subtype = USB_CDC_DESCR_SUBTYPE_UNION;
|
|
uni.master_if = cdcecm->iface_ctrl.idx;
|
|
uni.slave_if = cdcecm->iface_data.idx;
|
|
usbus_control_slicer_put_bytes(usbus, (uint8_t *)&uni, sizeof(uni));
|
|
return sizeof(usb_desc_union_t);
|
|
}
|
|
|
|
static size_t _gen_ecm_descriptor(usbus_t *usbus, usbus_cdcecm_device_t *cdcecm)
|
|
{
|
|
usb_desc_ecm_t ecm;
|
|
|
|
/* functional cdc ecm descriptor */
|
|
ecm.length = sizeof(usb_desc_ecm_t);
|
|
ecm.type = USB_TYPE_DESCRIPTOR_CDC;
|
|
ecm.subtype = USB_CDC_DESCR_SUBTYPE_ETH_NET;
|
|
ecm.macaddress = cdcecm->mac_str.idx;
|
|
ecm.ethernetstatistics = 0;
|
|
ecm.maxsegmentsize = ETHERNET_FRAME_LEN;
|
|
ecm.numbermcfilters = 0x0000; /* No filtering */
|
|
ecm.numberpowerfilters = 0;
|
|
usbus_control_slicer_put_bytes(usbus, (uint8_t *)&ecm, sizeof(ecm));
|
|
return sizeof(usb_desc_ecm_t);
|
|
}
|
|
|
|
static size_t _gen_cdc_descriptor(usbus_t *usbus)
|
|
{
|
|
usb_desc_cdc_t cdc;
|
|
/* functional cdc descriptor */
|
|
cdc.length = sizeof(usb_desc_cdc_t);
|
|
cdc.bcd_cdc = USB_CDC_VERSION_BCD;
|
|
cdc.type = USB_TYPE_DESCRIPTOR_CDC;
|
|
cdc.subtype = 0x00;
|
|
usbus_control_slicer_put_bytes(usbus, (uint8_t *)&cdc, sizeof(cdc));
|
|
return sizeof(usb_desc_cdc_t);
|
|
}
|
|
|
|
static size_t _gen_full_ecm_descriptor(usbus_t *usbus, void *arg)
|
|
{
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)arg;
|
|
size_t total_size = 0;
|
|
|
|
total_size += _gen_cdc_descriptor(usbus);
|
|
total_size += _gen_union_descriptor(usbus, cdcecm);
|
|
total_size += _gen_ecm_descriptor(usbus, cdcecm);
|
|
return total_size;
|
|
}
|
|
|
|
static void _notify_link_speed(usbus_cdcecm_device_t *cdcecm)
|
|
{
|
|
DEBUG("CDC ECM: sending link speed indication\n");
|
|
usb_desc_cdcecm_speed_t *notification =
|
|
(usb_desc_cdcecm_speed_t *)cdcecm->control_in;
|
|
notification->setup.type = USB_SETUP_REQUEST_DEVICE2HOST |
|
|
USB_SETUP_REQUEST_TYPE_CLASS |
|
|
USB_SETUP_REQUEST_RECIPIENT_INTERFACE;
|
|
notification->setup.request = USB_CDC_MGNT_NOTIF_CONN_SPEED_CHANGE;
|
|
notification->setup.value = 0;
|
|
notification->setup.index = cdcecm->iface_ctrl.idx;
|
|
notification->setup.length = 8;
|
|
|
|
notification->down = CONFIG_USBUS_CDC_ECM_CONFIG_SPEED_DOWNSTREAM;
|
|
notification->up = CONFIG_USBUS_CDC_ECM_CONFIG_SPEED_UPSTREAM;
|
|
usbdev_ep_xmit(cdcecm->ep_ctrl->ep, cdcecm->control_in,
|
|
sizeof(usb_desc_cdcecm_speed_t));
|
|
cdcecm->notif = USBUS_CDCECM_NOTIF_SPEED;
|
|
}
|
|
|
|
static void _notify_link_up(usbus_cdcecm_device_t *cdcecm)
|
|
{
|
|
DEBUG("CDC ECM: sending link up indication\n");
|
|
usb_setup_t *notification = (usb_setup_t *)cdcecm->control_in;
|
|
notification->type = USB_SETUP_REQUEST_DEVICE2HOST |
|
|
USB_SETUP_REQUEST_TYPE_CLASS |
|
|
USB_SETUP_REQUEST_RECIPIENT_INTERFACE;
|
|
notification->request = USB_CDC_MGNT_NOTIF_NETWORK_CONNECTION;
|
|
notification->value = 1;
|
|
notification->index = cdcecm->iface_ctrl.idx;
|
|
notification->length = 0;
|
|
usbdev_ep_xmit(cdcecm->ep_ctrl->ep, cdcecm->control_in, sizeof(usb_setup_t));
|
|
cdcecm->notif = USBUS_CDCECM_NOTIF_LINK_UP;
|
|
}
|
|
|
|
static const usbus_handler_driver_t cdcecm_driver = {
|
|
.init = _init,
|
|
.event_handler = _event_handler,
|
|
.transfer_handler = _transfer_handler,
|
|
.control_handler = _control_handler,
|
|
};
|
|
|
|
static void _fill_ethernet(usbus_cdcecm_device_t *cdcecm)
|
|
{
|
|
uint8_t ethernet[ETHERNET_ADDR_LEN];
|
|
|
|
luid_get_eui48((eui48_t*)ethernet);
|
|
fmt_bytes_hex(cdcecm->mac_host, ethernet, sizeof(ethernet));
|
|
|
|
}
|
|
|
|
void usbus_cdcecm_init(usbus_t *usbus, usbus_cdcecm_device_t *handler)
|
|
{
|
|
assert(usbus);
|
|
assert(handler);
|
|
memset(handler, 0, sizeof(usbus_cdcecm_device_t));
|
|
mutex_init(&handler->out_lock);
|
|
_fill_ethernet(handler);
|
|
handler->usbus = usbus;
|
|
handler->handler_ctrl.driver = &cdcecm_driver;
|
|
usbus_register_event_handler(usbus, (usbus_handler_t *)handler);
|
|
}
|
|
|
|
static void _init(usbus_t *usbus, usbus_handler_t *handler)
|
|
{
|
|
DEBUG("CDC ECM: initialization\n");
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)handler;
|
|
|
|
/* Add event handlers */
|
|
cdcecm->tx_xmit.handler = _handle_tx_xmit;
|
|
cdcecm->rx_flush.handler = _handle_rx_flush_ev;
|
|
|
|
/* Set up descriptor generators */
|
|
cdcecm->ecm_descr.next = NULL;
|
|
cdcecm->ecm_descr.funcs = &_ecm_descriptor;
|
|
cdcecm->ecm_descr.arg = cdcecm;
|
|
|
|
/* Configure Interface 0 as control interface */
|
|
cdcecm->iface_ctrl.class = USB_CLASS_CDC_CONTROL;
|
|
cdcecm->iface_ctrl.subclass = USB_CDC_SUBCLASS_ENCM;
|
|
cdcecm->iface_ctrl.protocol = USB_CDC_PROTOCOL_NONE;
|
|
cdcecm->iface_ctrl.descr_gen = &cdcecm->ecm_descr;
|
|
cdcecm->iface_ctrl.handler = handler;
|
|
|
|
/* Configure second interface to handle data endpoint */
|
|
cdcecm->iface_data.class = USB_CLASS_CDC_DATA;
|
|
cdcecm->iface_data.subclass = USB_CDC_SUBCLASS_NONE;
|
|
cdcecm->iface_data.protocol = USB_CDC_PROTOCOL_NONE;
|
|
cdcecm->iface_data.descr_gen = NULL;
|
|
cdcecm->iface_data.handler = handler;
|
|
|
|
/* Add string descriptor for the host mac */
|
|
usbus_add_string_descriptor(usbus, &cdcecm->mac_str, cdcecm->mac_host);
|
|
|
|
/* Create required endpoints */
|
|
cdcecm->ep_ctrl = usbus_add_endpoint(usbus, &cdcecm->iface_ctrl,
|
|
USB_EP_TYPE_INTERRUPT,
|
|
USB_EP_DIR_IN,
|
|
USBUS_CDCECM_EP_CTRL_SIZE);
|
|
cdcecm->ep_ctrl->interval = 0x10;
|
|
|
|
cdcecm->ep_out = usbus_add_endpoint(usbus,
|
|
(usbus_interface_t *)&cdcecm->iface_data_alt,
|
|
USB_EP_TYPE_BULK,
|
|
USB_EP_DIR_OUT,
|
|
USBUS_CDCECM_EP_DATA_SIZE);
|
|
cdcecm->ep_out->interval = 0; /* Must be 0 for bulk endpoints */
|
|
cdcecm->ep_in = usbus_add_endpoint(usbus,
|
|
(usbus_interface_t *)&cdcecm->iface_data_alt,
|
|
USB_EP_TYPE_BULK,
|
|
USB_EP_DIR_IN,
|
|
USBUS_CDCECM_EP_DATA_SIZE);
|
|
cdcecm->ep_in->interval = 0; /* Must be 0 for bulk endpoints */
|
|
|
|
/* Add interfaces to the stack */
|
|
usbus_add_interface(usbus, &cdcecm->iface_ctrl);
|
|
usbus_add_interface(usbus, &cdcecm->iface_data);
|
|
|
|
usbus_add_interface_alt(&cdcecm->iface_data, &cdcecm->iface_data_alt);
|
|
|
|
usbus_enable_endpoint(cdcecm->ep_out);
|
|
usbus_enable_endpoint(cdcecm->ep_in);
|
|
usbus_enable_endpoint(cdcecm->ep_ctrl);
|
|
usbus_handler_set_flag(handler, USBUS_HANDLER_FLAG_RESET);
|
|
}
|
|
|
|
static int _control_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbus_control_request_state_t state,
|
|
usb_setup_t *setup)
|
|
{
|
|
(void)usbus;
|
|
(void)state;
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)handler;
|
|
DEBUG("CDC ECM: Request: 0x%x\n", setup->request);
|
|
switch (setup->request) {
|
|
case USB_SETUP_REQ_SET_INTERFACE:
|
|
DEBUG("CDC ECM: Changing active interface to alt %d\n",
|
|
setup->value);
|
|
cdcecm->active_iface = (uint8_t)setup->value;
|
|
if (cdcecm->active_iface == 1) {
|
|
usbdev_ep_xmit(cdcecm->ep_out->ep, cdcecm->data_out,
|
|
USBUS_CDCECM_EP_DATA_SIZE);
|
|
_notify_link_up(cdcecm);
|
|
}
|
|
break;
|
|
|
|
case USB_CDC_MGNT_REQUEST_SET_ETH_PACKET_FILTER:
|
|
/* While we do answer the request, CDC ECM filters are not really
|
|
* implemented */
|
|
DEBUG("CDC ECM: Not modifying filter to 0x%x\n", setup->value);
|
|
break;
|
|
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int _handle_in_complete(usbus_t *usbus, usbus_handler_t *handler)
|
|
{
|
|
(void)usbus;
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)handler;
|
|
mutex_unlock(&cdcecm->out_lock);
|
|
return 0;
|
|
}
|
|
|
|
static void _handle_tx_xmit(event_t *ev)
|
|
{
|
|
|
|
usbus_cdcecm_device_t *cdcecm = container_of(ev, usbus_cdcecm_device_t,
|
|
tx_xmit);
|
|
usbus_t *usbus = cdcecm->usbus;
|
|
|
|
DEBUG("CDC_ECM: Handling TX xmit from netdev\n");
|
|
if (usbus->state != USBUS_STATE_CONFIGURED || cdcecm->active_iface == 0) {
|
|
DEBUG("CDC ECM: not configured, unlocking\n");
|
|
mutex_unlock(&cdcecm->out_lock);
|
|
}
|
|
/* Data prepared by netdev_send, signal ready to usbus */
|
|
usbdev_ep_xmit(cdcecm->ep_in->ep, cdcecm->data_in, cdcecm->tx_len);
|
|
}
|
|
|
|
static void _handle_rx_flush_ev(event_t *ev)
|
|
{
|
|
usbus_cdcecm_device_t *cdcecm = container_of(ev, usbus_cdcecm_device_t,
|
|
rx_flush);
|
|
cdcecm->len = 0; /* Flush packet */
|
|
usbdev_ep_xmit(cdcecm->ep_out->ep, cdcecm->data_out, USBUS_CDCECM_EP_DATA_SIZE);
|
|
}
|
|
|
|
static void _transfer_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbdev_ep_t *ep, usbus_event_transfer_t event)
|
|
{
|
|
(void)event; /* Only receives TR_COMPLETE events */
|
|
(void)usbus;
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)handler;
|
|
if (ep == cdcecm->ep_out->ep) {
|
|
/* Retrieve incoming data */
|
|
if (cdcecm->notif == USBUS_CDCECM_NOTIF_NONE) {
|
|
_notify_link_up(cdcecm);
|
|
}
|
|
size_t len = 0;
|
|
usbdev_ep_get(ep, USBOPT_EP_AVAILABLE, &len, sizeof(size_t));
|
|
cdcecm->len += len;
|
|
if (len == USBUS_CDCECM_EP_DATA_SIZE) {
|
|
/* ready next chunk */
|
|
usbdev_ep_xmit(ep, cdcecm->data_out + cdcecm->len,
|
|
USBUS_CDCECM_EP_DATA_SIZE);
|
|
}
|
|
else {
|
|
netdev_trigger_event_isr(&cdcecm->netdev);
|
|
}
|
|
}
|
|
else if (ep == cdcecm->ep_in->ep) {
|
|
_handle_in_complete(usbus, handler);
|
|
}
|
|
else if (ep == cdcecm->ep_ctrl->ep &&
|
|
cdcecm->notif == USBUS_CDCECM_NOTIF_LINK_UP) {
|
|
_notify_link_speed(cdcecm);
|
|
}
|
|
}
|
|
|
|
static void _handle_reset(usbus_t *usbus, usbus_handler_t *handler)
|
|
{
|
|
usbus_cdcecm_device_t *cdcecm = (usbus_cdcecm_device_t *)handler;
|
|
|
|
DEBUG("CDC ECM: Reset\n");
|
|
_handle_in_complete(usbus, handler);
|
|
cdcecm->notif = USBUS_CDCECM_NOTIF_NONE;
|
|
cdcecm->len = 0; /* Flush received data */
|
|
cdcecm->active_iface = 0;
|
|
mutex_unlock(&cdcecm->out_lock);
|
|
}
|
|
|
|
static void _event_handler(usbus_t *usbus, usbus_handler_t *handler,
|
|
usbus_event_usb_t event)
|
|
{
|
|
switch (event) {
|
|
case USBUS_EVENT_USB_RESET:
|
|
_handle_reset(usbus, handler);
|
|
break;
|
|
|
|
default:
|
|
DEBUG("Unhandled event :0x%x\n", event);
|
|
break;
|
|
}
|
|
}
|