refactor ehci init api

This commit is contained in:
hathach
2021-08-23 12:37:54 +07:00
parent 6a16f6ccdd
commit 4ca176c291
5 changed files with 39 additions and 47 deletions

View File

@@ -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;
@@ -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);

View File

@@ -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;

View File

@@ -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
} }

View File

@@ -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
} }

View File

@@ -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