more cache, fix an similar issue with OHCI when removing an queue head

This commit is contained in:
hathach
2023-05-18 12:39:53 +07:00
parent e4f4ad5bc3
commit a0aea52a11
2 changed files with 29 additions and 23 deletions

View File

@@ -162,7 +162,7 @@ static inline void qtd_remove_1st_from_qhd (ehci_qhd_t *p_qhd);
static void qtd_init (ehci_qtd_t* qtd, void const* buffer, uint16_t total_bytes); static void qtd_init (ehci_qtd_t* qtd, void const* buffer, uint16_t total_bytes);
static inline void list_insert (ehci_link_t *current, ehci_link_t *new, uint8_t new_type); static inline void list_insert (ehci_link_t *current, ehci_link_t *new, uint8_t new_type);
static inline ehci_link_t* list_next (ehci_link_t *p_link_pointer); static inline ehci_link_t* list_next (ehci_link_t const *p_link);
TU_ATTR_WEAK void hcd_dcache_clean(void* addr, uint32_t data_size) { TU_ATTR_WEAK void hcd_dcache_clean(void* addr, uint32_t data_size) {
(void) addr; (void) addr;
@@ -237,24 +237,22 @@ tusb_speed_t hcd_port_speed_get(uint8_t rhport)
return (tusb_speed_t) ehci_data.regs->portsc_bm.nxp_port_speed; // NXP specific port speed return (tusb_speed_t) ehci_data.regs->portsc_bm.nxp_port_speed; // NXP specific port speed
} }
static void list_remove_qhd_by_addr(ehci_link_t* list_head, uint8_t dev_addr) static void list_remove_qhd_by_daddr(ehci_link_t* list_head, uint8_t dev_addr) {
{ ehci_link_t* prev = list_head;
for(ehci_link_t* prev = list_head;
!prev->terminate && (tu_align32(prev->address) != (uint32_t) list_head) && prev != NULL; while (prev && !prev->terminate) {
prev = list_next(prev) ) ehci_qhd_t* qhd = (ehci_qhd_t*) (uintptr_t) list_next(prev);
{
// TODO check type for ISO iTD and siTD // done if loop back to head
// TODO Suppress cast-align warning if ( (uintptr_t) qhd == (uintptr_t) list_head) {
#pragma GCC diagnostic push break;
#pragma GCC diagnostic ignored "-Wcast-align" }
ehci_qhd_t* qhd = (ehci_qhd_t*) list_next(prev);
#pragma GCC diagnostic pop if ( qhd->dev_addr == dev_addr ) {
if ( qhd->dev_addr == dev_addr )
{
// TODO deactivate all TD, wait for QHD to inactive before removal // TODO deactivate all TD, wait for QHD to inactive before removal
prev->address = qhd->next.address; prev->address = qhd->next.address;
// EHCI 4.8.2 link the removed qhd to async head (which always reachable by Host Controller) // EHCI 4.8.2 link the removed qhd's next to async head (which always reachable by Host Controller)
qhd->next.address = ((uint32_t) list_head) | (EHCI_QTYPE_QHD << 1); qhd->next.address = ((uint32_t) list_head) | (EHCI_QTYPE_QHD << 1);
if ( qhd->int_smask ) if ( qhd->int_smask )
@@ -267,6 +265,11 @@ static void list_remove_qhd_by_addr(ehci_link_t* list_head, uint8_t dev_addr)
// mark as removing, will completely re-usable when async advance isr occurs // mark as removing, will completely re-usable when async advance isr occurs
qhd->removing = 1; qhd->removing = 1;
} }
hcd_dcache_clean(qhd, sizeof(ehci_qhd_t));
hcd_dcache_clean(prev, sizeof(ehci_qhd_t));
}else {
prev = list_next(prev);
} }
} }
} }
@@ -275,15 +278,16 @@ static void list_remove_qhd_by_addr(ehci_link_t* list_head, uint8_t dev_addr)
void hcd_device_close(uint8_t rhport, uint8_t dev_addr) void hcd_device_close(uint8_t rhport, uint8_t dev_addr)
{ {
// skip dev0 // skip dev0
if (dev_addr == 0) return; if (dev_addr == 0) {
return;
}
// Remove from async list // Remove from async list
list_remove_qhd_by_addr( (ehci_link_t*) qhd_async_head(rhport), dev_addr ); list_remove_qhd_by_daddr((ehci_link_t *) qhd_async_head(rhport), dev_addr);
// Remove from all interval period list // Remove from all interval period list
for(uint8_t i = 0; i < TU_ARRAY_SIZE(ehci_data.period_head_arr); i++) for(uint8_t i = 0; i < TU_ARRAY_SIZE(ehci_data.period_head_arr); i++) {
{ list_remove_qhd_by_daddr((ehci_link_t *) &ehci_data.period_head_arr[i], dev_addr);
list_remove_qhd_by_addr( (ehci_link_t*) &ehci_data.period_head_arr[i], dev_addr);
} }
// Async doorbell (EHCI 4.8.2 for operational details) // Async doorbell (EHCI 4.8.2 for operational details)
@@ -990,9 +994,9 @@ static inline void list_insert(ehci_link_t *current, ehci_link_t *new, uint8_t n
current->address = ((uint32_t) new) | (new_type << 1); current->address = ((uint32_t) new) | (new_type << 1);
} }
static inline ehci_link_t* list_next(ehci_link_t *p_link_pointer) static inline ehci_link_t* list_next(ehci_link_t const *p_link)
{ {
return (ehci_link_t*) tu_align32(p_link_pointer->address); return (ehci_link_t*) tu_align32(p_link->address);
} }
#endif #endif

View File

@@ -81,6 +81,8 @@ typedef union {
}; };
}ehci_link_t; }ehci_link_t;
TU_VERIFY_STATIC( sizeof(ehci_link_t) == 4, "size is not correct" );
/// Queue Element Transfer Descriptor /// Queue Element Transfer Descriptor
/// Qtd is used to declare overlay in ehci_qhd_t -> cannot be declared with TU_ATTR_ALIGNED(32) /// Qtd is used to declare overlay in ehci_qhd_t -> cannot be declared with TU_ATTR_ALIGNED(32)
typedef struct typedef struct