/* * 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 * @} */ #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 #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; } }