/**************************************************************************/ /*! @file usbd.c @author hathach (tinyusb.org) @section LICENSE Software License Agreement (BSD License) Copyright (c) 2013, hathach (tinyusb.org) All rights reserved. Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met: 1. Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer. 2. Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution. 3. Neither the name of the copyright holders nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ''AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. This file is part of the tinyusb stack. */ /**************************************************************************/ #include "tusb_option.h" #if MODE_DEVICE_SUPPORTED #define _TINY_USB_SOURCE_FILE_ //--------------------------------------------------------------------+ // INCLUDE //--------------------------------------------------------------------+ #include "tusb.h" #include "tusb_descriptors.h" // TODO callback include #include "usbd_dcd.h" // Some MCUs cannot transfer more than 64 bytes each queue, thus require special task-alike treatment #if MCU == MCU_LPC11UXX || MCU == MCU_LPC175X_6X #define USBD_CONTROL_ONE_PACKET_EACH_XFER // for each Transfer, cannot queue more than packet size enum { USBD_COTNROL_MAX_LENGTH_EACH_XFER = 64 }; #endif //--------------------------------------------------------------------+ // MACRO CONSTANT TYPEDEF //--------------------------------------------------------------------+ usbd_device_info_t usbd_devices[CONTROLLER_DEVICE_NUMBER]; // TODO fix/compress number of class driver static usbd_class_driver_t const usbd_class_drivers[TUSB_CLASS_MAPPED_INDEX_START] = { #if DEVICE_CLASS_HID [TUSB_CLASS_HID] = { .init = hidd_init, .open = hidd_open, .control_request = hidd_control_request, .isr = hidd_isr, .bus_reset = hidd_bus_reset }, #endif #if TUSB_CFG_DEVICE_MSC [TUSB_CLASS_MSC] = { .init = mscd_init, .open = mscd_open, .control_request = mscd_control_request, .isr = mscd_isr, .bus_reset = mscd_bus_reset }, #endif #if TUSB_CFG_DEVICE_CDC [TUSB_CLASS_CDC] = { .init = cdcd_init, .open = cdcd_open, .control_request = cdcd_control_request, .isr = cdcd_isr, .bus_reset = cdcd_bus_reset }, #endif }; //--------------------------------------------------------------------+ // INTERNAL OBJECT & FUNCTION DECLARATION //--------------------------------------------------------------------+ tusb_error_t usbd_set_configure_received(uint8_t coreid, uint8_t config_number); tusb_error_t get_descriptor_subtask(uint8_t coreid, tusb_control_request_t * p_request, uint8_t const ** pp_buffer, uint16_t * p_length); //--------------------------------------------------------------------+ // APPLICATION INTERFACE //--------------------------------------------------------------------+ bool tusbd_is_configured(uint8_t coreid) { return usbd_devices[coreid].state == TUSB_DEVICE_STATE_CONFIGURED; } //--------------------------------------------------------------------+ // IMPLEMENTATION //--------------------------------------------------------------------+ //------------- OSAL Task -------------// enum { USBD_TASK_QUEUE_DEPTH = 8 }; typedef enum { USBD_EVENTID_SETUP_RECEIVED = 1 }usbd_eventid_t; typedef struct { uint8_t coreid; uint8_t event_id; uint8_t reserved[2]; }usbd_task_event_t; OSAL_TASK_DEF(usbd_task, 150, TUSB_CFG_OS_TASK_PRIO); OSAL_QUEUE_DEF(usbd_queue_def, USBD_TASK_QUEUE_DEPTH, usbd_task_event_t); OSAL_SEM_DEF(usbd_control_xfer_semaphore_def); static osal_queue_handle_t usbd_queue_hdl; static osal_semaphore_handle_t usbd_control_xfer_sem_hdl; tusb_error_t usbd_body_subtask(void) { OSAL_SUBTASK_BEGIN static uint8_t coreid; tusb_error_t error = TUSB_ERROR_NONE; usbd_task_event_t event; osal_queue_receive(usbd_queue_hdl, &event, OSAL_TIMEOUT_WAIT_FOREVER, &error); SUBTASK_ASSERT_STATUS(error); coreid = event.coreid; if ( USBD_EVENTID_SETUP_RECEIVED == event.event_id ) { // should copy to setup packet to local variable as the new one may overwrite while we processing here static tusb_control_request_t control_request; control_request = usbd_devices[coreid].control_request; //------------- Standard Control such as those in enumeration -------------// if( TUSB_REQUEST_RECIPIENT_DEVICE == control_request.bmRequestType_bit.recipient && TUSB_REQUEST_TYPE_STANDARD == control_request.bmRequestType_bit.type ) { if ( TUSB_REQUEST_GET_DESCRIPTOR == control_request.bRequest ) { static uint8_t const * p_buffer = NULL; static uint16_t length = 0; error = get_descriptor_subtask(coreid, &control_request, &p_buffer, &length); #ifdef USBD_CONTROL_ONE_PACKET_EACH_XFER while ( length > USBD_COTNROL_MAX_LENGTH_EACH_XFER && error == TUSB_ERROR_NONE ) { usbd_devices[coreid].is_waiting_control_xfer = true; dcd_pipe_control_xfer(coreid, control_request.bmRequestType_bit.direction, p_buffer, USBD_COTNROL_MAX_LENGTH_EACH_XFER); // zero length osal_semaphore_wait(usbd_control_xfer_sem_hdl, OSAL_TIMEOUT_NORMAL, &error); length -= USBD_COTNROL_MAX_LENGTH_EACH_XFER; p_buffer += USBD_COTNROL_MAX_LENGTH_EACH_XFER; usbd_devices[coreid].is_waiting_control_xfer = false; } #endif if ( TUSB_ERROR_NONE == error ) { dcd_pipe_control_xfer(coreid, control_request.bmRequestType_bit.direction, p_buffer, length); } } else if ( TUSB_REQUEST_SET_ADDRESS == control_request.bRequest ) { dcd_controller_set_address(coreid, (uint8_t) control_request.wValue); usbd_devices[coreid].state = TUSB_DEVICE_STATE_ADDRESSED; } else if ( TUSB_REQUEST_SET_CONFIGURATION == control_request.bRequest ) { usbd_set_configure_received(coreid, (uint8_t) control_request.wValue); }else { error = TUSB_ERROR_DCD_CONTROL_REQUEST_NOT_SUPPORT; } } //------------- Class/Interface Specific Request -------------// else if ( TUSB_REQUEST_RECIPIENT_INTERFACE == control_request.bmRequestType_bit.recipient) { tusb_std_class_code_t class_code = usbd_devices[coreid].interface2class[ u16_low_u8(control_request.wIndex) ]; if ( (TUSB_CLASS_AUDIO <= class_code) && (class_code <= TUSB_CLASS_AUDIO_VIDEO) && usbd_class_drivers[class_code].control_request ) { error = usbd_class_drivers[class_code].control_request(coreid, &control_request); }else { error = TUSB_ERROR_DCD_CONTROL_REQUEST_NOT_SUPPORT; } } //------------- Endpoint Request -------------// else if ( TUSB_REQUEST_RECIPIENT_ENDPOINT == control_request.bmRequestType_bit.recipient && TUSB_REQUEST_TYPE_STANDARD == control_request.bmRequestType_bit.type && TUSB_REQUEST_CLEAR_FEATURE == control_request.bRequest ) { dcd_pipe_clear_stall(coreid, u16_low_u8(control_request.wIndex) ); } else { error = TUSB_ERROR_DCD_CONTROL_REQUEST_NOT_SUPPORT; } if(TUSB_ERROR_NONE != error) { // Response with Protocol Stall if request is not supported dcd_pipe_control_stall(coreid); // ASSERT(error == TUSB_ERROR_NONE, VOID_RETURN); }else { // status phase dcd_pipe_control_xfer(coreid, 1-control_request.bmRequestType_bit.direction, NULL, 0); // zero length } } OSAL_SUBTASK_END } // To enable the TASK_ASSERT style (quick return on false condition) in a real RTOS, a task must act as a wrapper // and is used mainly to call subtasks. Within a subtask return statement can be called freely, the task with // forever loop cannot have any return at all. OSAL_TASK_FUNCTION(usbd_task) (void* p_task_para) { OSAL_TASK_LOOP_BEGIN usbd_body_subtask(); OSAL_TASK_LOOP_END } void usbd_bus_reset(uint32_t coreid) { memclr_(&usbd_devices[coreid], sizeof(usbd_device_info_t)); for (tusb_std_class_code_t class_code = TUSB_CLASS_AUDIO; class_code <= TUSB_CLASS_AUDIO_VIDEO; class_code++) { if ( usbd_class_drivers[class_code].bus_reset ) { usbd_class_drivers[class_code].bus_reset( coreid ); } } } tusb_error_t usbd_init (void) { ASSERT_STATUS ( dcd_init() ); //------------- Task init -------------// usbd_queue_hdl = osal_queue_create( OSAL_QUEUE_REF(usbd_queue_def) ); ASSERT_PTR(usbd_queue_hdl, TUSB_ERROR_OSAL_QUEUE_FAILED); usbd_control_xfer_sem_hdl = osal_semaphore_create( OSAL_SEM_REF(usbd_control_xfer_semaphore_def) ); ASSERT_PTR(usbd_queue_hdl, TUSB_ERROR_OSAL_SEMAPHORE_FAILED); ASSERT_STATUS( osal_task_create( OSAL_TASK_REF(usbd_task) )); //------------- class init -------------// for (tusb_std_class_code_t class_code = TUSB_CLASS_AUDIO; class_code <= TUSB_CLASS_AUDIO_VIDEO; class_code++) { if ( usbd_class_drivers[class_code].init ) { usbd_class_drivers[class_code].init(); } } return TUSB_ERROR_NONE; } //--------------------------------------------------------------------+ // CONTROL REQUEST //--------------------------------------------------------------------+ // TODO Host (windows) can get HID report descriptor before set configured // need to open interface before set configured tusb_error_t usbd_set_configure_received(uint8_t coreid, uint8_t config_number) { dcd_controller_set_configuration(coreid); usbd_devices[coreid].state = TUSB_DEVICE_STATE_CONFIGURED; //------------- parse configuration & open drivers -------------// uint8_t* p_desc_configure = (uint8_t*) &app_tusb_desc_configuration; uint8_t* p_desc = p_desc_configure + sizeof(tusb_descriptor_configuration_t); while( p_desc < p_desc_configure + ((tusb_descriptor_configuration_t*)p_desc_configure)->wTotalLength ) { if ( TUSB_DESC_TYPE_INTERFACE_ASSOCIATION == p_desc[DESCRIPTOR_OFFSET_TYPE]) { p_desc += p_desc[DESCRIPTOR_OFFSET_LENGTH]; // ignore IAD }else { ASSERT( TUSB_DESC_TYPE_INTERFACE == p_desc[DESCRIPTOR_OFFSET_TYPE], TUSB_ERROR_NOT_SUPPORTED_YET ); uint8_t class_index; tusb_descriptor_interface_t* p_desc_interface = (tusb_descriptor_interface_t*) p_desc; class_index = p_desc_interface->bInterfaceClass; ASSERT( class_index != 0 && usbd_class_drivers[class_index].open != NULL, TUSB_ERROR_NOT_SUPPORTED_YET ); ASSERT( 0 == usbd_devices[coreid].interface2class[p_desc_interface->bInterfaceNumber], TUSB_ERROR_FAILED); // duplicate interface number TODO alternate setting usbd_devices[coreid].interface2class[p_desc_interface->bInterfaceNumber] = class_index; uint16_t length=0; ASSERT_STATUS( usbd_class_drivers[class_index].open( coreid, p_desc_interface, &length ) ); ASSERT( length >= sizeof(tusb_descriptor_interface_t), TUSB_ERROR_FAILED ); p_desc += length; } } return TUSB_ERROR_NONE; } tusb_error_t get_descriptor_subtask(uint8_t coreid, tusb_control_request_t * p_request, uint8_t const ** pp_buffer, uint16_t * p_length) { tusb_std_descriptor_type_t const desc_type = u16_high_u8(p_request->wValue); uint8_t const desc_index = u16_low_u8( p_request->wValue ); if ( TUSB_DESC_TYPE_DEVICE == desc_type ) { (*pp_buffer) = (uint8_t const *) &app_tusb_desc_device; (*p_length) = sizeof(tusb_descriptor_device_t); } else if ( TUSB_DESC_TYPE_CONFIGURATION == desc_type ) { (*pp_buffer) = (uint8_t const *) &app_tusb_desc_configuration; (*p_length) = sizeof(app_tusb_desc_configuration); } else if ( TUSB_DESC_TYPE_STRING == desc_type ) { if ( ! (desc_index < TUSB_CFG_DEVICE_STRING_DESCRIPTOR_COUNT) ) return TUSB_ERROR_DCD_CONTROL_REQUEST_NOT_SUPPORT; (*pp_buffer) = (uint8_t const*) desc_str_table[desc_index]; (*p_length) = desc_str_table[desc_index]->bLength; }else { return TUSB_ERROR_DCD_CONTROL_REQUEST_NOT_SUPPORT; } (*p_length) = min16_of(p_request->wLength, (*p_length) ); // cannot return more than hosts requires return TUSB_ERROR_NONE; } //--------------------------------------------------------------------+ // USBD-CLASS API //--------------------------------------------------------------------+ //--------------------------------------------------------------------+ // DCD Callback API //--------------------------------------------------------------------+ void usbd_setup_received_isr(uint8_t coreid, tusb_control_request_t * p_request) { usbd_devices[coreid].control_request = (*p_request); osal_queue_send(usbd_queue_hdl, &(usbd_task_event_t){ .coreid = coreid, .event_id = USBD_EVENTID_SETUP_RECEIVED} ); } void usbd_xfer_isr(endpoint_handle_t edpt_hdl, tusb_event_t event, uint32_t xferred_bytes) { // usbd_device_info_t *p_device = &usbd_devices[edpt_hdl.coreid]; uint8_t class_index = std_class_code_to_index(edpt_hdl.class_code); if (class_index == 0 ) // Control Transfer { if (usbd_devices[edpt_hdl.coreid].is_waiting_control_xfer) { osal_semaphore_post( usbd_control_xfer_sem_hdl ); } }else if (usbd_class_drivers[class_index].isr) { usbd_class_drivers[class_index].isr(edpt_hdl, event, xferred_bytes); }else { ASSERT(false, VOID_RETURN); // something wrong, no one claims the isr's source } } //--------------------------------------------------------------------+ // HELPER //--------------------------------------------------------------------+ #endif