| 
									
										
										
										
											2013-01-22 17:41:06 +07:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * usbd_host.c | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *  Created on: Jan 19, 2013 | 
					
						
							|  |  |  |  *      Author: hathach | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Software License Agreement (BSD License) | 
					
						
							| 
									
										
										
										
											2013-01-27 13:50:40 +07:00
										 |  |  |  * Copyright (c) 2013, hathach (tinyusb.net) | 
					
						
							| 
									
										
										
										
											2013-01-22 17:41:06 +07:00
										 |  |  |  * 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. The name of the author may not be used to endorse or promote products | 
					
						
							|  |  |  |  *    derived from this software without specific prior written permission. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``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 AUTHOR 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 tiny usb stack. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | #include "tusb_option.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-22 17:41:06 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef TUSB_CFG_HOST
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | #define _TINY_USB_SOURCE_FILE_
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INCLUDE
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | #include "common/common.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-29 11:06:30 +07:00
										 |  |  | #include "osal/osal.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 14:48:35 +07:00
										 |  |  | #include "usbh_hcd.h"
 | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // MACRO CONSTANT TYPEDEF
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INTERNAL OBJECT & FUNCTION DECLARATION
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  | static inline uint8_t get_new_address(void) ATTR_ALWAYS_INLINE; | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  | STATIC_ usbh_device_info_t usbh_device_info_pool[TUSB_CFG_HOST_DEVICE_MAX]; | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // PUBLIC API (Parameter Verification is required)
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | tusbh_device_status_t tusbh_device_status_get (tusb_handle_device_t const device_hdl) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ASSERT(device_hdl < TUSB_CFG_HOST_DEVICE_MAX, 0); | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |   return usbh_device_info_pool[device_hdl].status; | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // ENUMERATION TASK & ITS DATA
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-02-01 12:46:17 +07:00
										 |  |  | OSAL_TASK_DEF(enum_task, usbh_enumeration_task, 128, OSAL_PRIO_HIGH); | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define ENUM_QUEUE_DEPTH  5
 | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  | OSAL_QUEUE_DEF(enum_queue, ENUM_QUEUE_DEPTH, uin32_t); | 
					
						
							| 
									
										
										
										
											2013-02-01 12:46:17 +07:00
										 |  |  | osal_queue_handle_t enum_queue_hdl; | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | usbh_device_addr0_t device_addr0 TUSB_CFG_ATTR_USBRAM; | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  | uint8_t enum_data_buffer[TUSB_CFG_HOST_ENUM_BUFFER_SIZE] TUSB_CFG_ATTR_USBRAM; | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-01 12:46:17 +07:00
										 |  |  | void usbh_enumeration_task(void) | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-01 14:21:22 +07:00
										 |  |  |   tusb_error_t error; | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |   static uint8_t new_addr; | 
					
						
							| 
									
										
										
										
											2013-02-01 14:21:22 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  |   OSAL_TASK_LOOP_BEGIN | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  |   osal_queue_receive(enum_queue_hdl, (uint32_t*)(&device_addr0.enum_entry), OSAL_TIMEOUT_WAIT_FOREVER, &error); | 
					
						
							| 
									
										
										
										
											2013-02-01 14:21:22 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  |   if (device_addr0.enum_entry.hub_addr == 0) // direct connection
 | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  |     TASK_ASSERT(device_addr0.enum_entry.connect_status == hcd_port_connect_status(device_addr0.enum_entry.core_id)); // there chances the event is out-dated
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     device_addr0.speed = hcd_port_speed(device_addr0.enum_entry.core_id); | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  |     TASK_ASSERT_STATUS( hcd_addr0_open(&device_addr0) ); | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |     { // Get first 8 bytes of device descriptor to get Control Endpoint Size
 | 
					
						
							|  |  |  |       tusb_std_request_t request_device_desc = { | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  |           .bmRequestType = { .direction = TUSB_DIR_DEV_TO_HOST, .type = TUSB_REQUEST_TYPE_STANDARD, .recipient = TUSB_REQUEST_RECIPIENT_DEVICE }, | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  |           .bRequest = TUSB_REQUEST_GET_DESCRIPTOR, | 
					
						
							|  |  |  |           .wValue   = (TUSB_DESC_DEVICE << 8), | 
					
						
							|  |  |  |           .wLength  = 8 | 
					
						
							|  |  |  |       }; | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  |       hcd_pipe_control_xfer(device_addr0.pipe_hdl, &request_device_desc, enum_data_buffer); | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  |       osal_semaphore_wait(device_addr0.sem_hdl, OSAL_TIMEOUT_NORMAL, &error); // careful of local variable without static
 | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  |       TASK_ASSERT_STATUS_HANDLER(error, tusbh_device_mount_failed_cb(TUSB_ERROR_USBH_MOUNT_FAILED, NULL)); | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |     new_addr = get_new_address(); | 
					
						
							|  |  |  |     TASK_ASSERT(new_addr < TUSB_CFG_HOST_DEVICE_MAX); | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |     { // Set new address
 | 
					
						
							|  |  |  |       tusb_std_request_t request_set_address = { | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  |           .bmRequestType = { .direction = TUSB_DIR_HOST_TO_DEV, .type = TUSB_REQUEST_TYPE_STANDARD, .recipient = TUSB_REQUEST_RECIPIENT_DEVICE }, | 
					
						
							|  |  |  |           .bRequest = TUSB_REQUEST_SET_ADDRESS, | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |           .wValue   = (new_addr+1) | 
					
						
							| 
									
										
										
										
											2013-02-04 20:54:52 +07:00
										 |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       hcd_pipe_control_xfer(device_addr0.pipe_hdl, &request_set_address, NULL); | 
					
						
							|  |  |  |       osal_semaphore_wait(device_addr0.sem_hdl, OSAL_TIMEOUT_NORMAL, &error); // careful of local variable without static
 | 
					
						
							|  |  |  |       TASK_ASSERT_STATUS(error); | 
					
						
							| 
									
										
										
										
											2013-02-04 16:07:42 +07:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  |     // update data for the new device
 | 
					
						
							|  |  |  |     usbh_device_info_pool[new_addr].core_id  = device_addr0.enum_entry.core_id; | 
					
						
							|  |  |  |     usbh_device_info_pool[new_addr].hub_addr = device_addr0.enum_entry.hub_addr; | 
					
						
							|  |  |  |     usbh_device_info_pool[new_addr].hub_port = device_addr0.enum_entry.hub_port; | 
					
						
							|  |  |  |     usbh_device_info_pool[new_addr].speed    = device_addr0.speed; | 
					
						
							|  |  |  |     usbh_device_info_pool[new_addr].status   = TUSB_DEVICE_STATUS_ADDRESSED; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     hcd_addr0_close(&device_addr0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   }else // device connect via a hub
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     ASSERT_MESSAGE("%s", "Hub is not supported yet"); | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-02-02 15:36:20 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   OSAL_TASK_LOOP_END | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-04 00:03:08 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // REPORTER TASK & ITS DATA
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // CLASS-USBD API (don't require to verify parameters)
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							| 
									
										
										
										
											2013-01-29 11:06:30 +07:00
										 |  |  | tusb_error_t usbh_init(void) | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  |   uint32_t i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-06 12:03:01 +07:00
										 |  |  |   memclr_(usbh_device_info_pool, sizeof(usbh_device_info_t)*TUSB_CFG_HOST_DEVICE_MAX); | 
					
						
							| 
									
										
										
										
											2013-01-31 13:11:50 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-01 10:33:43 +07:00
										 |  |  |   for(i=0; i<TUSB_CFG_HOST_CONTROLLER_NUM; i++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     ASSERT_STATUS( hcd_init(i) ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-01 12:46:17 +07:00
										 |  |  |   ASSERT_STATUS( osal_task_create(&enum_task) ); | 
					
						
							|  |  |  |   enum_queue_hdl = osal_queue_create(&enum_queue); | 
					
						
							|  |  |  |   ASSERT_PTR(enum_queue_hdl, TUSB_ERROR_OSAL_QUEUE_FAILED); | 
					
						
							| 
									
										
										
										
											2013-01-31 13:11:50 +07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-29 11:06:30 +07:00
										 |  |  |   return TUSB_ERROR_NONE; | 
					
						
							| 
									
										
										
										
											2013-01-27 17:57:35 +07:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2013-01-22 17:41:06 +07:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-02-04 21:52:50 +07:00
										 |  |  | 
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | // INTERNAL HELPER
 | 
					
						
							|  |  |  | //--------------------------------------------------------------------+
 | 
					
						
							|  |  |  | static inline uint8_t get_new_address(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   uint8_t new_addr; | 
					
						
							|  |  |  |   for (new_addr=0; new_addr<TUSB_CFG_HOST_DEVICE_MAX; new_addr++) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     if (usbh_device_info_pool[new_addr].status == TUSB_DEVICE_STATUS_UNPLUG) | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return new_addr; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 |