mirror of
https://github.com/hathach/tinyusb.git
synced 2025-01-17 05:32:55 +08:00
220 lines
7.7 KiB
C
220 lines
7.7 KiB
C
/*
|
||
* The MIT License (MIT)
|
||
*
|
||
* Copyright (c) 2019 Ha Thach (tinyusb.org)
|
||
*
|
||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
* of this software and associated documentation files (the "Software"), to deal
|
||
* in the Software without restriction, including without limitation the rights
|
||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
* copies of the Software, and to permit persons to whom the Software is
|
||
* furnished to do so, subject to the following conditions:
|
||
*
|
||
* The above copyright notice and this permission notice shall be included in
|
||
* all copies or substantial portions of the Software.
|
||
*
|
||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
* THE SOFTWARE.
|
||
*
|
||
* This file is part of the TinyUSB stack.
|
||
*/
|
||
|
||
/** \ingroup group_class
|
||
* \defgroup ClassDriver_Hub Hub (Host only)
|
||
* \details Like most PC's OS, Hub support is completely hidden from Application. In fact, application cannot determine whether
|
||
* a device is mounted directly via roothub or via a hub's port. All Hub-related procedures are performed and managed
|
||
* by tinyusb stack. Unless you are trying to develop the stack itself, there are nothing else can be used by Application.
|
||
* \note Due to my laziness, only 1-level of Hub is supported. In other way, the stack cannot mount a hub via another hub.
|
||
* @{
|
||
*/
|
||
|
||
#ifndef _TUSB_HUB_H_
|
||
#define _TUSB_HUB_H_
|
||
|
||
#include "common/tusb_common.h"
|
||
|
||
#ifdef __cplusplus
|
||
extern "C" {
|
||
#endif
|
||
|
||
//D1...D0: Logical Power Switching Mode
|
||
//00: Ganged power switching (all ports’power at
|
||
//once)
|
||
//01: Individual port power switching
|
||
//1X: Reserved. Used only on 1.0 compliant hubs
|
||
//that implement no power switching
|
||
//D2: Identifies a Compound Device
|
||
//0: Hub is not part of a compound device.
|
||
//1: Hub is part of a compound device.
|
||
//D4...D3: Over-current Protection Mode
|
||
//00: Global Over-current Protection. The hub
|
||
//reports over-current as a summation of all
|
||
//ports’current draw, without a breakdown of
|
||
//individual port over-current status.
|
||
//01: Individual Port Over-current Protection. The
|
||
//hub reports over-current on a per-port basis.
|
||
//Each port has an over-current status.
|
||
//1X: No Over-current Protection. This option is
|
||
//allowed only for bus-powered hubs that do not
|
||
//implement over-current protection.
|
||
//
|
||
//D6...D5: TT Think TIme
|
||
//00: TT requires at most 8 FS bit times of inter
|
||
//transaction gap on a full-/low-speed
|
||
//downstream bus.
|
||
//01: TT requires at most 16 FS bit times.
|
||
//10: TT requires at most 24 FS bit times.
|
||
//11: TT requires at most 32 FS bit times.
|
||
//D7: Port Indicators Supported
|
||
//0: Port Indicators are not supported on its
|
||
//downstream facing ports and the
|
||
//PORT_INDICATOR request has no effect.
|
||
//1: Port Indicators are supported on its
|
||
//downstream facing ports and the
|
||
//PORT_INDICATOR request controls the
|
||
//indicators. See Section 11.5.3.
|
||
//D15...D8: Reserved
|
||
|
||
typedef struct TU_ATTR_PACKED{
|
||
uint8_t bLength ; ///< Size of descriptor
|
||
uint8_t bDescriptorType ; ///< Other_speed_Configuration Type
|
||
uint8_t bNbrPorts;
|
||
uint16_t wHubCharacteristics;
|
||
uint8_t bPwrOn2PwrGood;
|
||
uint8_t bHubContrCurrent;
|
||
uint8_t DeviceRemovable; // bitmap each bit for a port (from bit1)
|
||
uint8_t PortPwrCtrlMask; // just for compatibility, should be 0xff
|
||
} descriptor_hub_desc_t;
|
||
|
||
TU_VERIFY_STATIC( sizeof(descriptor_hub_desc_t) == 9, "size is not correct");
|
||
|
||
enum {
|
||
HUB_REQUEST_GET_STATUS = 0 ,
|
||
HUB_REQUEST_CLEAR_FEATURE = 1 ,
|
||
|
||
HUB_REQUEST_SET_FEATURE = 3 ,
|
||
|
||
HUB_REQUEST_GET_DESCRIPTOR = 6 ,
|
||
HUB_REQUEST_SET_DESCRIPTOR = 7 ,
|
||
HUB_REQUEST_CLEAR_TT_BUFFER = 8 ,
|
||
HUB_REQUEST_RESET_TT = 9 ,
|
||
HUB_REQUEST_GET_TT_STATE = 10 ,
|
||
HUB_REQUEST_STOP_TT = 11
|
||
};
|
||
|
||
enum {
|
||
HUB_FEATURE_HUB_LOCAL_POWER_CHANGE = 0,
|
||
HUB_FEATURE_HUB_OVER_CURRENT_CHANGE
|
||
};
|
||
|
||
enum{
|
||
HUB_FEATURE_PORT_CONNECTION = 0,
|
||
HUB_FEATURE_PORT_ENABLE = 1,
|
||
HUB_FEATURE_PORT_SUSPEND = 2,
|
||
HUB_FEATURE_PORT_OVER_CURRENT = 3,
|
||
HUB_FEATURE_PORT_RESET = 4,
|
||
|
||
HUB_FEATURE_PORT_POWER = 8,
|
||
HUB_FEATURE_PORT_LOW_SPEED = 9,
|
||
|
||
HUB_FEATURE_PORT_CONNECTION_CHANGE = 16,
|
||
HUB_FEATURE_PORT_ENABLE_CHANGE = 17,
|
||
HUB_FEATURE_PORT_SUSPEND_CHANGE = 18,
|
||
HUB_FEATURE_PORT_OVER_CURRENT_CHANGE = 19,
|
||
HUB_FEATURE_PORT_RESET_CHANGE = 20,
|
||
HUB_FEATURE_PORT_TEST = 21,
|
||
HUB_FEATURE_PORT_INDICATOR = 22
|
||
};
|
||
|
||
// data in response of HUB_REQUEST_GET_STATUS, wIndex = 0 (hub)
|
||
typedef struct {
|
||
union{
|
||
struct TU_ATTR_PACKED {
|
||
uint16_t local_power_source : 1;
|
||
uint16_t over_current : 1;
|
||
uint16_t : 14;
|
||
};
|
||
|
||
uint16_t value;
|
||
} status, change;
|
||
} hub_status_response_t;
|
||
|
||
TU_VERIFY_STATIC( sizeof(hub_status_response_t) == 4, "size is not correct");
|
||
|
||
// data in response of HUB_REQUEST_GET_STATUS, wIndex = Port num
|
||
typedef struct {
|
||
union {
|
||
struct TU_ATTR_PACKED {
|
||
uint16_t connection : 1;
|
||
uint16_t port_enable : 1;
|
||
uint16_t suspend : 1;
|
||
uint16_t over_current : 1;
|
||
uint16_t reset : 1;
|
||
|
||
uint16_t : 3;
|
||
uint16_t port_power : 1;
|
||
uint16_t low_speed : 1;
|
||
uint16_t high_speed : 1;
|
||
uint16_t port_test_mode : 1;
|
||
uint16_t port_indicator_control : 1;
|
||
uint16_t TU_RESERVED : 3;
|
||
};
|
||
|
||
uint16_t value;
|
||
} status, change;
|
||
} hub_port_status_response_t;
|
||
|
||
TU_VERIFY_STATIC( sizeof(hub_port_status_response_t) == 4, "size is not correct");
|
||
|
||
// Clear feature
|
||
bool hub_port_clear_feature (uint8_t hub_addr, uint8_t hub_port, uint8_t feature,
|
||
tuh_xfer_cb_t complete_cb, uintptr_t user_data);
|
||
|
||
// Set feature
|
||
bool hub_port_set_feature (uint8_t hub_addr, uint8_t hub_port, uint8_t feature,
|
||
tuh_xfer_cb_t complete_cb, uintptr_t user_data);
|
||
|
||
// Get port status
|
||
bool hub_port_get_status (uint8_t hub_addr, uint8_t hub_port, void* resp,
|
||
tuh_xfer_cb_t complete_cb, uintptr_t user_data);
|
||
|
||
// Get status from Interrupt endpoint
|
||
bool hub_edpt_status_xfer(uint8_t dev_addr);
|
||
|
||
// Reset a port
|
||
static inline bool hub_port_reset(uint8_t hub_addr, uint8_t hub_port,
|
||
tuh_xfer_cb_t complete_cb, uintptr_t user_data)
|
||
{
|
||
return hub_port_set_feature(hub_addr, hub_port, HUB_FEATURE_PORT_RESET, complete_cb, user_data);
|
||
}
|
||
|
||
// Clear Reset Change
|
||
static inline bool hub_port_clear_reset_change(uint8_t hub_addr, uint8_t hub_port,
|
||
tuh_xfer_cb_t complete_cb, uintptr_t user_data)
|
||
{
|
||
return hub_port_clear_feature(hub_addr, hub_port, HUB_FEATURE_PORT_RESET_CHANGE, complete_cb, user_data);
|
||
}
|
||
|
||
|
||
//--------------------------------------------------------------------+
|
||
// Internal Class Driver API
|
||
//--------------------------------------------------------------------+
|
||
void hub_init (void);
|
||
bool hub_open (uint8_t rhport, uint8_t dev_addr, tusb_desc_interface_t const *itf_desc, uint16_t max_len);
|
||
bool hub_set_config (uint8_t dev_addr, uint8_t itf_num);
|
||
bool hub_xfer_cb (uint8_t dev_addr, uint8_t ep_addr, xfer_result_t event, uint32_t xferred_bytes);
|
||
void hub_close (uint8_t dev_addr);
|
||
|
||
#ifdef __cplusplus
|
||
}
|
||
#endif
|
||
|
||
#endif /* _TUSB_HUB_H_ */
|
||
|
||
/** @} */
|