refactor ehci init api
This commit is contained in:
@@ -37,13 +37,25 @@
|
|||||||
|
|
||||||
#include "host/hcd.h"
|
#include "host/hcd.h"
|
||||||
#include "host/usbh_hcd.h"
|
#include "host/usbh_hcd.h"
|
||||||
#include "hcd_ehci.h"
|
#include "ehci_api.h"
|
||||||
#include "ehci.h"
|
#include "ehci.h"
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// MACRO CONSTANT TYPEDEF
|
// MACRO CONSTANT TYPEDEF
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
|
|
||||||
|
// Debug level of EHCI
|
||||||
|
#define EHCI_DBG 2
|
||||||
|
|
||||||
|
// Framelist size as small as possible
|
||||||
|
// - Standard EHCI : 256 elements
|
||||||
|
// - NXP Transdimension: 8 elements
|
||||||
|
#define EHCI_CFG_FRAMELIST_SIZE_BITS 7
|
||||||
|
#define EHCI_FRAMELIST_SIZE (1024 >> EHCI_CFG_FRAMELIST_SIZE_BITS)
|
||||||
|
|
||||||
|
TU_VERIFY_STATIC(EHCI_CFG_FRAMELIST_SIZE_BITS <= 7, "incorrect value");
|
||||||
|
|
||||||
|
|
||||||
typedef struct
|
typedef struct
|
||||||
{
|
{
|
||||||
ehci_link_t period_framelist[EHCI_FRAMELIST_SIZE];
|
ehci_link_t period_framelist[EHCI_FRAMELIST_SIZE];
|
||||||
@@ -65,9 +77,7 @@ typedef struct
|
|||||||
|
|
||||||
volatile uint32_t uframe_number;
|
volatile uint32_t uframe_number;
|
||||||
}ehci_data_t;
|
}ehci_data_t;
|
||||||
//--------------------------------------------------------------------+
|
|
||||||
// INTERNAL OBJECT & FUNCTION DECLARATION
|
|
||||||
//--------------------------------------------------------------------+
|
|
||||||
// Periodic frame list must be 4K alignment
|
// Periodic frame list must be 4K alignment
|
||||||
CFG_TUSB_MEM_SECTION TU_ATTR_ALIGNED(4096) static ehci_data_t ehci_data;
|
CFG_TUSB_MEM_SECTION TU_ATTR_ALIGNED(4096) static ehci_data_t ehci_data;
|
||||||
|
|
||||||
@@ -109,7 +119,7 @@ static inline bool qhd_has_xact_error (ehci_qhd_t * p_qhd)
|
|||||||
//p_qhd->qtd_overlay.non_hs_period_missed_uframe || p_qhd->qtd_overlay.pingstate_err TODO split transaction error
|
//p_qhd->qtd_overlay.non_hs_period_missed_uframe || p_qhd->qtd_overlay.pingstate_err TODO split transaction error
|
||||||
}
|
}
|
||||||
|
|
||||||
static void qhd_init (ehci_qhd_t *p_qhd, uint8_t dev_addr, tusb_desc_endpoint_t const * ep_desc);
|
static void qhd_init(ehci_qhd_t *p_qhd, uint8_t dev_addr, tusb_desc_endpoint_t const * ep_desc);
|
||||||
|
|
||||||
static inline ehci_qtd_t* qtd_find_free (void);
|
static inline ehci_qtd_t* qtd_find_free (void);
|
||||||
static inline ehci_qtd_t* qtd_next (ehci_qtd_t const * p_qtd);
|
static inline ehci_qtd_t* qtd_next (ehci_qtd_t const * p_qtd);
|
||||||
@@ -218,12 +228,13 @@ void hcd_device_close(uint8_t rhport, uint8_t dev_addr)
|
|||||||
ehci_data.regs->command_bm.async_adv_doorbell = 1;
|
ehci_data.regs->command_bm.async_adv_doorbell = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// EHCI controller init
|
bool ehci_init(uint8_t rhport, uint32_t capability_reg, uint32_t operatial_reg)
|
||||||
bool hcd_ehci_init(uint8_t rhport)
|
|
||||||
{
|
{
|
||||||
|
(void) capability_reg; // not used yet
|
||||||
|
|
||||||
tu_memclr(&ehci_data, sizeof(ehci_data_t));
|
tu_memclr(&ehci_data, sizeof(ehci_data_t));
|
||||||
|
|
||||||
ehci_data.regs = (ehci_registers_t* ) hcd_ehci_register_addr(rhport);
|
ehci_data.regs = (ehci_registers_t* ) operatial_reg;
|
||||||
|
|
||||||
ehci_registers_t* regs = ehci_data.regs;
|
ehci_registers_t* regs = ehci_data.regs;
|
||||||
|
|
||||||
@@ -470,16 +481,16 @@ static void async_advance_isr(uint8_t rhport)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void port_connect_status_change_isr(uint8_t hostid)
|
static void port_connect_status_change_isr(uint8_t rhport)
|
||||||
{
|
{
|
||||||
// NOTE There is an sequence plug->unplug->…..-> plug if device is powering with pre-plugged device
|
// NOTE There is an sequence plug->unplug->…..-> plug if device is powering with pre-plugged device
|
||||||
if (ehci_data.regs->portsc_bm.current_connect_status)
|
if (ehci_data.regs->portsc_bm.current_connect_status)
|
||||||
{
|
{
|
||||||
hcd_port_reset(hostid);
|
hcd_port_reset(rhport);
|
||||||
hcd_event_device_attach(hostid, true);
|
hcd_event_device_attach(rhport, true);
|
||||||
}else // device unplugged
|
}else // device unplugged
|
||||||
{
|
{
|
||||||
hcd_event_device_remove(hostid, true);
|
hcd_event_device_remove(rhport, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -656,6 +667,8 @@ void hcd_int_handler(uint8_t rhport)
|
|||||||
{
|
{
|
||||||
uint32_t port_status = regs->portsc & EHCI_PORTSC_MASK_ALL;
|
uint32_t port_status = regs->portsc & EHCI_PORTSC_MASK_ALL;
|
||||||
|
|
||||||
|
TU_LOG_HEX(EHCI_DBG, regs->portsc);
|
||||||
|
|
||||||
if (regs->portsc_bm.connect_status_change)
|
if (regs->portsc_bm.connect_status_change)
|
||||||
{
|
{
|
||||||
port_connect_status_change_isr(rhport);
|
port_connect_status_change_isr(rhport);
|
||||||
|
@@ -46,8 +46,6 @@
|
|||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// EHCI CONFIGURATION & CONSTANTS
|
// EHCI CONFIGURATION & CONSTANTS
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
#define EHCI_CFG_FRAMELIST_SIZE_BITS 7 /// Framelist Size (NXP specific) (0:1024) - (1:512) - (2:256) - (3:128) - (4:64) - (5:32) - (6:16) - (7:8)
|
|
||||||
#define EHCI_FRAMELIST_SIZE (1024 >> EHCI_CFG_FRAMELIST_SIZE_BITS)
|
|
||||||
|
|
||||||
// TODO merge OHCI with EHCI
|
// TODO merge OHCI with EHCI
|
||||||
enum {
|
enum {
|
||||||
@@ -55,9 +53,6 @@ enum {
|
|||||||
EHCI_MAX_SITD = 16
|
EHCI_MAX_SITD = 16
|
||||||
};
|
};
|
||||||
|
|
||||||
//------------- Validation -------------//
|
|
||||||
TU_VERIFY_STATIC(EHCI_CFG_FRAMELIST_SIZE_BITS <= 7, "incorrect value");
|
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// EHCI Data Structure
|
// EHCI Data Structure
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
@@ -411,7 +406,7 @@ typedef volatile struct
|
|||||||
uint32_t wake_on_over_current_enable : 1; ///< Enables over-current conditions as wake-up events
|
uint32_t wake_on_over_current_enable : 1; ///< Enables over-current conditions as wake-up events
|
||||||
uint32_t nxp_phy_clock_disable : 1; ///< NXP customized: the PHY can be put into Low Power Suspend – Clock Disable when the downstream device has been put into suspend mode or when no downstream device is connected. Low power suspend is completely under the control of software. 0: enable PHY clock, 1: disable PHY clock
|
uint32_t nxp_phy_clock_disable : 1; ///< NXP customized: the PHY can be put into Low Power Suspend – Clock Disable when the downstream device has been put into suspend mode or when no downstream device is connected. Low power suspend is completely under the control of software. 0: enable PHY clock, 1: disable PHY clock
|
||||||
uint32_t nxp_port_force_fullspeed : 1; ///< NXP customized: Writing this bit to a 1 will force the port to only connect at Full Speed. It disables the chirp sequence that allowsthe port to identify itself as High Speed. This is useful for testing FS configurations with a HS host, hub or device.
|
uint32_t nxp_port_force_fullspeed : 1; ///< NXP customized: Writing this bit to a 1 will force the port to only connect at Full Speed. It disables the chirp sequence that allowsthe port to identify itself as High Speed. This is useful for testing FS configurations with a HS host, hub or device.
|
||||||
uint32_t : 1;
|
uint32_t TU_RESERVED : 1;
|
||||||
uint32_t nxp_port_speed : 2; ///< NXP customized: This register field indicates the speed atwhich the port is operating. For HS mode operation in the host controllerand HS/FS operation in the device controller the port routing steers data to the Protocol engine. For FS and LS mode operation in the host controller, the port routing steers data to the Protocol Engine w/ Embedded Transaction Translator. 0x0: Fullspeed, 0x1: Lowspeed, 0x2: Highspeed
|
uint32_t nxp_port_speed : 2; ///< NXP customized: This register field indicates the speed atwhich the port is operating. For HS mode operation in the host controllerand HS/FS operation in the device controller the port routing steers data to the Protocol engine. For FS and LS mode operation in the host controller, the port routing steers data to the Protocol Engine w/ Embedded Transaction Translator. 0x0: Fullspeed, 0x1: Lowspeed, 0x2: Highspeed
|
||||||
uint32_t TU_RESERVED : 4;
|
uint32_t TU_RESERVED : 4;
|
||||||
}portsc_bm;
|
}portsc_bm;
|
||||||
|
@@ -24,27 +24,19 @@
|
|||||||
* This file is part of the TinyUSB stack.
|
* This file is part of the TinyUSB stack.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _TUSB_HCD_EHCI_H_
|
#ifndef _TUSB_EHCI_API_H_
|
||||||
#define _TUSB_HCD_EHCI_H_
|
#define _TUSB_EHCI_API_H_
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
|
||||||
// API Implemented by HCD
|
|
||||||
//--------------------------------------------------------------------+
|
|
||||||
|
|
||||||
// Get operational address i.e EHCI Command register
|
|
||||||
uint32_t hcd_ehci_register_addr(uint8_t rhport);
|
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// API Implemented by EHCI
|
// API Implemented by EHCI
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
|
|
||||||
// Initialize EHCI driver
|
// Initialize EHCI driver
|
||||||
extern bool hcd_ehci_init (uint8_t rhport);
|
bool ehci_init(uint8_t rhport, uint32_t capability_reg, uint32_t operatial_reg);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
@@ -127,7 +127,7 @@ typedef struct
|
|||||||
__I uint32_t ENDPTSTAT; ///< Endpoint Status
|
__I uint32_t ENDPTSTAT; ///< Endpoint Status
|
||||||
__IO uint32_t ENDPTCOMPLETE; ///< Endpoint Complete
|
__IO uint32_t ENDPTCOMPLETE; ///< Endpoint Complete
|
||||||
__IO uint32_t ENDPTCTRL[8]; ///< Endpoint Control 0 - 7
|
__IO uint32_t ENDPTCTRL[8]; ///< Endpoint Control 0 - 7
|
||||||
} dcd_registers_t;
|
} dcd_registers_t, hcd_registers_t;
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
|
@@ -43,7 +43,7 @@
|
|||||||
|
|
||||||
#include "common/tusb_common.h"
|
#include "common/tusb_common.h"
|
||||||
#include "common_transdimension.h"
|
#include "common_transdimension.h"
|
||||||
#include "portable/ehci/hcd_ehci.h"
|
#include "portable/ehci/ehci_api.h"
|
||||||
|
|
||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
// MACRO CONSTANT TYPEDEF
|
// MACRO CONSTANT TYPEDEF
|
||||||
@@ -82,26 +82,26 @@ typedef struct
|
|||||||
|
|
||||||
bool hcd_init(uint8_t rhport)
|
bool hcd_init(uint8_t rhport)
|
||||||
{
|
{
|
||||||
dcd_registers_t* dcd_reg = (dcd_registers_t*) _hcd_controller[rhport].regs_base;
|
hcd_registers_t* hcd_reg = (hcd_registers_t*) _hcd_controller[rhport].regs_base;
|
||||||
|
|
||||||
// Reset controller
|
// Reset controller
|
||||||
dcd_reg->USBCMD |= USBCMD_RESET;
|
hcd_reg->USBCMD |= USBCMD_RESET;
|
||||||
while( dcd_reg->USBCMD & USBCMD_RESET ) {}
|
while( hcd_reg->USBCMD & USBCMD_RESET ) {}
|
||||||
|
|
||||||
// Set mode to device, must be set immediately after reset
|
// Set mode to device, must be set immediately after reset
|
||||||
#if CFG_TUSB_MCU == OPT_MCU_LPC18XX || CFG_TUSB_MCU == OPT_MCU_LPC43XX
|
#if CFG_TUSB_MCU == OPT_MCU_LPC18XX || CFG_TUSB_MCU == OPT_MCU_LPC43XX
|
||||||
// LPC18XX/43XX need to set VBUS Power Select to HIGH
|
// LPC18XX/43XX need to set VBUS Power Select to HIGH
|
||||||
// RHPORT1 is fullspeed only (need external PHY for Highspeed)
|
// RHPORT1 is fullspeed only (need external PHY for Highspeed)
|
||||||
dcd_reg->USBMODE = USBMODE_CM_HOST | USBMODE_VBUS_POWER_SELECT;
|
hcd_reg->USBMODE = USBMODE_CM_HOST | USBMODE_VBUS_POWER_SELECT;
|
||||||
if (rhport == 1) dcd_reg->PORTSC1 |= PORTSC1_FORCE_FULL_SPEED;
|
if (rhport == 1) hcd_reg->PORTSC1 |= PORTSC1_FORCE_FULL_SPEED;
|
||||||
#else
|
#else
|
||||||
dcd_reg->USBMODE = USBMODE_CM_HOST;
|
hcd_reg->USBMODE = USBMODE_CM_HOST;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// FIXME force full speed, still have issue with Highspeed enumeration
|
// FIXME force full speed, still have issue with Highspeed enumeration
|
||||||
dcd_reg->PORTSC1 |= PORTSC1_FORCE_FULL_SPEED;
|
hcd_reg->PORTSC1 |= PORTSC1_FORCE_FULL_SPEED;
|
||||||
|
|
||||||
return hcd_ehci_init(rhport);
|
return ehci_init(rhport, (uint32_t) &hcd_reg->CAPLENGTH, (uint32_t) &hcd_reg->USBCMD);
|
||||||
}
|
}
|
||||||
|
|
||||||
void hcd_int_enable(uint8_t rhport)
|
void hcd_int_enable(uint8_t rhport)
|
||||||
@@ -114,12 +114,4 @@ void hcd_int_disable(uint8_t rhport)
|
|||||||
NVIC_DisableIRQ(_hcd_controller[rhport].irqnum);
|
NVIC_DisableIRQ(_hcd_controller[rhport].irqnum);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t hcd_ehci_register_addr(uint8_t rhport)
|
|
||||||
{
|
|
||||||
dcd_registers_t* hcd_reg = (dcd_registers_t*) _hcd_controller[rhport].regs_base;
|
|
||||||
|
|
||||||
// EHCI USBCMD has same address within dcd_register_t
|
|
||||||
return (uint32_t) &hcd_reg->USBCMD;
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Reference in New Issue
Block a user