2024-10-15 13:03:12 +07:00
|
|
|
/*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2024 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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "tusb_option.h"
|
|
|
|
|
|
|
|
#define DWC2_COMMON_DEBUG 2
|
|
|
|
|
|
|
|
#if defined(TUP_USBIP_DWC2) && (CFG_TUH_ENABLED || CFG_TUD_ENABLED)
|
|
|
|
|
2024-10-15 17:55:24 +07:00
|
|
|
#if CFG_TUD_ENABLED
|
|
|
|
#include "device/dcd.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if CFG_TUH_ENABLED
|
|
|
|
#include "host/hcd.h"
|
|
|
|
#endif
|
|
|
|
|
2024-10-15 13:03:12 +07:00
|
|
|
#include "dwc2_common.h"
|
|
|
|
|
|
|
|
static void reset_core(dwc2_regs_t* dwc2) {
|
|
|
|
// reset core
|
|
|
|
dwc2->grstctl |= GRSTCTL_CSRST;
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
if ((dwc2->gsnpsid & DWC2_CORE_REV_MASK) < (DWC2_CORE_REV_4_20a & DWC2_CORE_REV_MASK)) {
|
|
|
|
// prior v42.0 CSRST is self-clearing
|
|
|
|
while (dwc2->grstctl & GRSTCTL_CSRST) {}
|
|
|
|
} else {
|
|
|
|
// From v4.20a CSRST bit is write only, CSRT_DONE (w1c) is introduced for checking.
|
|
|
|
// CSRST must also be explicitly cleared
|
|
|
|
while (!(dwc2->grstctl & GRSTCTL_CSRST_DONE)) {}
|
|
|
|
dwc2->grstctl = (dwc2->grstctl & ~GRSTCTL_CSRST) | GRSTCTL_CSRST_DONE;
|
|
|
|
}
|
2024-10-15 13:03:12 +07:00
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
while (!(dwc2->grstctl & GRSTCTL_AHBIDL)) {} // wait for AHB master IDLE
|
2024-10-15 13:03:12 +07:00
|
|
|
}
|
|
|
|
|
2024-10-15 17:55:24 +07:00
|
|
|
bool dwc2_core_is_highspeed(dwc2_regs_t* dwc2, const tusb_rhport_init_t* rh_init) {
|
2024-10-17 15:56:12 +07:00
|
|
|
(void)dwc2;
|
2024-10-15 13:03:12 +07:00
|
|
|
|
2024-10-15 17:55:24 +07:00
|
|
|
#if CFG_TUD_ENABLED
|
|
|
|
if (rh_init->role == TUSB_ROLE_DEVICE && !TUD_OPT_HIGH_SPEED) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#if CFG_TUH_ENABLED
|
2024-10-16 13:19:28 +07:00
|
|
|
if (rh_init->role == TUSB_ROLE_HOST && !TUH_OPT_HIGH_SPEED) {
|
2024-10-15 17:55:24 +07:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
#endif
|
2024-10-15 13:03:12 +07:00
|
|
|
|
|
|
|
return dwc2->ghwcfg2_bm.hs_phy_type != GHWCFG2_HSPHY_NOT_SUPPORTED;
|
|
|
|
}
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
static void phy_fs_init(dwc2_regs_t* dwc2, tusb_role_t role) {
|
2024-10-15 13:03:12 +07:00
|
|
|
TU_LOG(DWC2_COMMON_DEBUG, "Fullspeed PHY init\r\n");
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
uint32_t gusbcfg = dwc2->gusbcfg;
|
|
|
|
|
2024-10-15 13:03:12 +07:00
|
|
|
// Select FS PHY
|
2024-10-17 15:56:12 +07:00
|
|
|
gusbcfg |= GUSBCFG_PHYSEL;
|
|
|
|
dwc2->gusbcfg = gusbcfg;
|
2024-10-15 13:03:12 +07:00
|
|
|
|
|
|
|
// MCU specific PHY init before reset
|
|
|
|
dwc2_phy_init(dwc2, GHWCFG2_HSPHY_NOT_SUPPORTED);
|
|
|
|
|
|
|
|
// Reset core after selecting PHY
|
|
|
|
reset_core(dwc2);
|
|
|
|
|
|
|
|
// USB turnaround time is critical for certification where long cables and 5-Hubs are used.
|
|
|
|
// So if you need the AHB to run at less than 30 MHz, and if USB turnaround time is not critical,
|
|
|
|
// these bits can be programmed to a larger value. Default is 5
|
2024-10-17 15:56:12 +07:00
|
|
|
gusbcfg &= ~GUSBCFG_TRDT_Msk;
|
|
|
|
gusbcfg |= 5u << GUSBCFG_TRDT_Pos;
|
|
|
|
dwc2->gusbcfg = gusbcfg;
|
|
|
|
|
|
|
|
// FS/LS PHY Clock Select
|
|
|
|
if (role == TUSB_ROLE_HOST) {
|
|
|
|
uint32_t hcfg = dwc2->hcfg;
|
|
|
|
hcfg |= HCFG_FSLS_ONLY;
|
|
|
|
hcfg &= ~HCFG_FSLS_PHYCLK_SEL;
|
|
|
|
|
|
|
|
if (dwc2->ghwcfg2_bm.hs_phy_type == GHWCFG2_HSPHY_ULPI &&
|
|
|
|
dwc2->ghwcfg2_bm.fs_phy_type == GHWCFG2_FSPHY_DEDICATED) {
|
|
|
|
// dedicated FS PHY with 48 mhz
|
|
|
|
hcfg |= HCFG_FSLS_PHYCLK_SEL_48MHZ;
|
|
|
|
} else {
|
|
|
|
// shared HS PHY running at full speed
|
|
|
|
hcfg |= HCFG_FSLS_PHYCLK_SEL_30_60MHZ;
|
|
|
|
}
|
|
|
|
dwc2->hcfg = hcfg;
|
|
|
|
}
|
2024-10-15 13:03:12 +07:00
|
|
|
|
|
|
|
// MCU specific PHY update post reset
|
|
|
|
dwc2_phy_update(dwc2, GHWCFG2_HSPHY_NOT_SUPPORTED);
|
|
|
|
}
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
static void phy_hs_init(dwc2_regs_t* dwc2, tusb_role_t role) {
|
|
|
|
(void) role;
|
2024-10-15 13:03:12 +07:00
|
|
|
uint32_t gusbcfg = dwc2->gusbcfg;
|
|
|
|
|
|
|
|
// De-select FS PHY
|
|
|
|
gusbcfg &= ~GUSBCFG_PHYSEL;
|
|
|
|
|
|
|
|
if (dwc2->ghwcfg2_bm.hs_phy_type == GHWCFG2_HSPHY_ULPI) {
|
|
|
|
TU_LOG(DWC2_COMMON_DEBUG, "Highspeed ULPI PHY init\r\n");
|
|
|
|
|
|
|
|
// Select ULPI PHY (external)
|
|
|
|
gusbcfg |= GUSBCFG_ULPI_UTMI_SEL;
|
|
|
|
|
|
|
|
// ULPI is always 8-bit interface
|
|
|
|
gusbcfg &= ~GUSBCFG_PHYIF16;
|
|
|
|
|
|
|
|
// ULPI select single data rate
|
|
|
|
gusbcfg &= ~GUSBCFG_DDRSEL;
|
|
|
|
|
|
|
|
// default internal VBUS Indicator and Drive
|
|
|
|
gusbcfg &= ~(GUSBCFG_ULPIEVBUSD | GUSBCFG_ULPIEVBUSI);
|
|
|
|
|
|
|
|
// Disable FS/LS ULPI
|
|
|
|
gusbcfg &= ~(GUSBCFG_ULPIFSLS | GUSBCFG_ULPICSM);
|
|
|
|
} else {
|
|
|
|
TU_LOG(DWC2_COMMON_DEBUG, "Highspeed UTMI+ PHY init\r\n");
|
|
|
|
|
|
|
|
// Select UTMI+ PHY (internal)
|
|
|
|
gusbcfg &= ~GUSBCFG_ULPI_UTMI_SEL;
|
|
|
|
|
|
|
|
// Set 16-bit interface if supported
|
|
|
|
if (dwc2->ghwcfg4_bm.phy_data_width) {
|
2024-10-17 15:56:12 +07:00
|
|
|
gusbcfg |= GUSBCFG_PHYIF16; // 16 bit
|
|
|
|
} else {
|
2024-10-15 13:03:12 +07:00
|
|
|
gusbcfg &= ~GUSBCFG_PHYIF16; // 8 bit
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply config
|
|
|
|
dwc2->gusbcfg = gusbcfg;
|
|
|
|
|
|
|
|
// mcu specific phy init
|
|
|
|
dwc2_phy_init(dwc2, dwc2->ghwcfg2_bm.hs_phy_type);
|
|
|
|
|
|
|
|
// Reset core after selecting PHY
|
|
|
|
reset_core(dwc2);
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
if (role == TUSB_ROLE_HOST) {
|
|
|
|
dwc2->hcfg &= ~HCFG_FSLS_ONLY;
|
|
|
|
}
|
|
|
|
|
2024-10-15 13:03:12 +07:00
|
|
|
// Set turn-around, must after core reset otherwise it will be clear
|
|
|
|
// - 9 if using 8-bit PHY interface
|
|
|
|
// - 5 if using 16-bit PHY interface
|
|
|
|
gusbcfg &= ~GUSBCFG_TRDT_Msk;
|
|
|
|
gusbcfg |= (dwc2->ghwcfg4_bm.phy_data_width ? 5u : 9u) << GUSBCFG_TRDT_Pos;
|
|
|
|
dwc2->gusbcfg = gusbcfg;
|
|
|
|
|
|
|
|
// MCU specific PHY update post reset
|
|
|
|
dwc2_phy_update(dwc2, dwc2->ghwcfg2_bm.hs_phy_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool check_dwc2(dwc2_regs_t* dwc2) {
|
|
|
|
#if CFG_TUSB_DEBUG >= DWC2_COMMON_DEBUG
|
|
|
|
// print guid, gsnpsid, ghwcfg1, ghwcfg2, ghwcfg3, ghwcfg4
|
|
|
|
// Run 'python dwc2_info.py' and check dwc2_info.md for bit-field value and comparison with other ports
|
|
|
|
volatile uint32_t const* p = (volatile uint32_t const*) &dwc2->guid;
|
|
|
|
TU_LOG1("guid, gsnpsid, ghwcfg1, ghwcfg2, ghwcfg3, ghwcfg4\r\n");
|
|
|
|
for (size_t i = 0; i < 5; i++) {
|
|
|
|
TU_LOG1("0x%08" PRIX32 ", ", p[i]);
|
|
|
|
}
|
|
|
|
TU_LOG1("0x%08" PRIX32 "\r\n", p[5]);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// For some reason: GD32VF103 gsnpsid and all hwcfg register are always zero (skip it)
|
2024-10-17 15:56:12 +07:00
|
|
|
(void)dwc2;
|
2024-10-15 13:03:12 +07:00
|
|
|
#if !TU_CHECK_MCU(OPT_MCU_GD32VF103)
|
|
|
|
uint32_t const gsnpsid = dwc2->gsnpsid & GSNPSID_ID_MASK;
|
|
|
|
TU_ASSERT(gsnpsid == DWC2_OTG_ID || gsnpsid == DWC2_FS_IOT_ID || gsnpsid == DWC2_HS_IOT_ID);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//--------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
//--------------------------------------------------------------------
|
2024-10-15 17:55:24 +07:00
|
|
|
bool dwc2_core_init(uint8_t rhport, const tusb_rhport_init_t* rh_init) {
|
2024-10-17 15:56:12 +07:00
|
|
|
(void)rh_init;
|
2024-10-15 13:03:12 +07:00
|
|
|
dwc2_regs_t* dwc2 = DWC2_REG(rhport);
|
|
|
|
|
|
|
|
// Check Synopsys ID register, failed if controller clock/power is not enabled
|
|
|
|
TU_ASSERT(check_dwc2(dwc2));
|
|
|
|
|
2024-10-15 17:55:24 +07:00
|
|
|
if (dwc2_core_is_highspeed(dwc2, rh_init)) {
|
2024-10-17 15:56:12 +07:00
|
|
|
phy_hs_init(dwc2, rh_init->role);
|
2024-10-15 13:03:12 +07:00
|
|
|
} else {
|
2024-10-17 15:56:12 +07:00
|
|
|
phy_fs_init(dwc2, rh_init->role);
|
2024-10-15 13:03:12 +07:00
|
|
|
}
|
|
|
|
|
2024-10-15 17:55:24 +07:00
|
|
|
/* Set HS/FS Timeout Calibration to 7 (max available value).
|
|
|
|
* The number of PHY clocks that the application programs in
|
|
|
|
* this field is added to the high/full speed interpacket timeout
|
|
|
|
* duration in the core to account for any additional delays
|
|
|
|
* introduced by the PHY. This can be required, because the delay
|
|
|
|
* introduced by the PHY in generating the linestate condition
|
2024-10-16 13:19:28 +07:00
|
|
|
* can vary from one PHY to another. */
|
2024-10-15 17:55:24 +07:00
|
|
|
dwc2->gusbcfg |= (7ul << GUSBCFG_TOCAL_Pos);
|
|
|
|
|
2024-10-16 13:19:28 +07:00
|
|
|
// Enable PHY clock TODO stop/gate clock when suspended mode
|
|
|
|
dwc2->pcgcctl &= ~(PCGCCTL_STOPPCLK | PCGCCTL_GATEHCLK | PCGCCTL_PWRCLMP | PCGCCTL_RSTPDWNMODULE);
|
|
|
|
|
|
|
|
dfifo_flush_tx(dwc2, 0x10); // all tx fifo
|
|
|
|
dfifo_flush_rx(dwc2);
|
|
|
|
|
|
|
|
// Clear all pending interrupts
|
|
|
|
uint32_t int_mask;
|
|
|
|
|
|
|
|
int_mask = dwc2->gintsts;
|
|
|
|
dwc2->gintsts |= int_mask;
|
|
|
|
|
|
|
|
int_mask = dwc2->gotgint;
|
|
|
|
dwc2->gotgint |= int_mask;
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
dwc2->gintmsk = 0;
|
2024-10-16 13:19:28 +07:00
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
if (dwc2_dma_enabled(dwc2, rh_init->role)) {
|
2024-10-16 13:19:28 +07:00
|
|
|
const uint16_t epinfo_base = dma_cal_epfifo_base(rhport);
|
|
|
|
dwc2->gdfifocfg = (epinfo_base << GDFIFOCFG_EPINFOBASE_SHIFT) | epinfo_base;
|
|
|
|
|
|
|
|
// DMA seems to be only settable after a core reset, and not possible to switch on-the-fly
|
|
|
|
dwc2->gahbcfg |= GAHBCFG_DMAEN | GAHBCFG_HBSTLEN_2;
|
2024-10-17 15:56:12 +07:00
|
|
|
} else {
|
2024-10-16 13:19:28 +07:00
|
|
|
dwc2->gintmsk |= GINTMSK_RXFLVLM;
|
|
|
|
}
|
|
|
|
|
2024-10-17 15:56:12 +07:00
|
|
|
// Configure TX FIFO empty level for interrupt. Default is complete empty
|
|
|
|
dwc2->gahbcfg |= GAHBCFG_TXFELVL;
|
2024-10-16 13:19:28 +07:00
|
|
|
|
2024-10-15 13:03:12 +07:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|