Skip to content

Commit

Permalink
usb/ehci: adapt hcd driver to ia32 hw
Browse files Browse the repository at this point in the history
The root hub is now declared with bDeviceProtocol 0 instead of 1
to differentiate between root hubs and real TT layer 2 hubs in the usb stack

JIRA: RTOS-937
  • Loading branch information
adamgreloch committed Jan 17, 2025
1 parent c2a89a4 commit ed8084b
Show file tree
Hide file tree
Showing 4 changed files with 203 additions and 42 deletions.
46 changes: 40 additions & 6 deletions usb/ehci/ehci-hub.c
Original file line number Diff line number Diff line change
Expand Up @@ -3,8 +3,8 @@
*
* ehci root hub implementation
*
* Copyright 2021 Phoenix Systems
* Author: Maciej Purski
* Copyright 2021, 2024 Phoenix Systems
* Author: Maciej Purski, Adam Greloch
*
* This file is part of Phoenix-RTOS.
*
Expand Down Expand Up @@ -39,7 +39,12 @@ static const struct {
.bcdUSB = 0x0200,
.bDeviceClass = USB_CLASS_HUB,
.bDeviceSubClass = 0,
#ifdef EHCI_IMX
/* imx deviation: the controller has an embedded TT */
.bDeviceProtocol = 1, /* Single TT */
#else
.bDeviceProtocol = 0, /* Root hub */
#endif
.bMaxPacketSize0 = 64,
.idVendor = 0x0,
.idProduct = 0x0,
Expand Down Expand Up @@ -88,9 +93,13 @@ static void ehci_resetPort(hcd_t *hcd, int port)
{
ehci_t *ehci = (ehci_t *)hcd->priv;
volatile int *reg = (ehci->opbase + portsc1) + (port - 1);
int tmp;

log_debug("resetting port %d", port);

*reg &= ~PORTSC_ENA;
*reg |= PORTSC_PR;
tmp = *reg;
tmp &= ~(PORTSC_ENA | PORTSC_PR);
*reg = tmp | PORTSC_PR;

#ifdef EHCI_IMX
/*
Expand All @@ -101,10 +110,27 @@ static void ehci_resetPort(hcd_t *hcd, int port)
;
usleep(20 * 1000);
#else
/* Wait for reset to complete */
usleep(50 * 1000);

/* Stop the reset sequence */
*reg = tmp;

/* Wait until reset sequence stops */
while ((*reg & PORTSC_PR) != 0)
;

usleep(20 * 1000);
*reg &= ~PORTSC_PR;
#endif

tmp = *reg;

log_debug("port %d reset done, status after reset=%x", port, tmp);

if ((tmp & PORTSC_ENA) == 0) {
log_debug("device on port %d is not a highspeed device", port);
}

ehci->portResetChange = 1 << port;

if ((*reg & PORTSC_PSPD) == PORTSC_PSPD_HS)
Expand Down Expand Up @@ -158,10 +184,16 @@ static int ehci_getPortStatus(usb_dev_t *hub, int port, usb_port_status_t *statu
if (ehci->portResetChange & (1 << port))
status->wPortChange |= USB_PORT_STAT_C_RESET;

#ifdef EHCI_IMX
if ((val & PORTSC_PSPD) >> 26 == 1)
status->wPortStatus |= USB_PORT_STAT_LOW_SPEED;
else if ((val & PORTSC_PSPD) >> 26 == 2)
status->wPortStatus |= USB_PORT_STAT_HIGH_SPEED;
#endif

/* TODO handle low/full speed devices on ia32 */

status->wPortStatus |= USB_PORT_STAT_HIGH_SPEED;

/* TODO: set indicator */

Expand Down Expand Up @@ -334,11 +366,13 @@ uint32_t ehci_getHubStatus(usb_dev_t *hub)
ehci_t *ehci = (ehci_t *)hcd->priv;

for (i = 0; i < hub->nports; i++) {
val = ehci->portsc;
val = *(ehci->opbase + portsc1 + i);
log_debug("(INT%d) port %d portsc: %x", hcd->info->irq, i + 1, val);
if (val & (PORTSC_CSC | PORTSC_PEC | PORTSC_OCC))
status |= 1 << (i + 1);
}

log_debug("(INT%d): status: %x", hcd->info->irq, status);
return status;
}

Expand Down
Loading

0 comments on commit ed8084b

Please sign in to comment.