more ohci clean up
This commit is contained in:
@@ -232,7 +232,7 @@ void hcd_device_remove(uint8_t rhport, uint8_t dev_addr)
|
|||||||
//--------------------------------------------------------------------+
|
//--------------------------------------------------------------------+
|
||||||
static inline tusb_xfer_type_t ed_get_xfer_type(ohci_ed_t const * const p_ed)
|
static inline tusb_xfer_type_t ed_get_xfer_type(ohci_ed_t const * const p_ed)
|
||||||
{
|
{
|
||||||
return (p_ed->endpoint_number == 0 ) ? TUSB_XFER_CONTROL :
|
return (p_ed->ep_number == 0 ) ? TUSB_XFER_CONTROL :
|
||||||
(p_ed->is_iso ) ? TUSB_XFER_ISOCHRONOUS :
|
(p_ed->is_iso ) ? TUSB_XFER_ISOCHRONOUS :
|
||||||
(p_ed->is_interrupt_xfer ) ? TUSB_XFER_INTERRUPT : TUSB_XFER_BULK;
|
(p_ed->is_interrupt_xfer ) ? TUSB_XFER_INTERRUPT : TUSB_XFER_BULK;
|
||||||
}
|
}
|
||||||
@@ -247,12 +247,12 @@ static void ed_init(ohci_ed_t *p_ed, uint8_t dev_addr, uint16_t max_packet_size,
|
|||||||
tu_memclr(p_ed, sizeof(ohci_ed_t));
|
tu_memclr(p_ed, sizeof(ohci_ed_t));
|
||||||
}
|
}
|
||||||
|
|
||||||
p_ed->device_address = dev_addr;
|
p_ed->dev_addr = dev_addr;
|
||||||
p_ed->endpoint_number = endpoint_addr & 0x0F;
|
p_ed->ep_number = endpoint_addr & 0x0F;
|
||||||
p_ed->direction = (xfer_type == TUSB_XFER_CONTROL) ? OHCI_PID_SETUP : ( (endpoint_addr & TUSB_DIR_IN_MASK) ? OHCI_PID_IN : OHCI_PID_OUT );
|
p_ed->pid = (xfer_type == TUSB_XFER_CONTROL) ? OHCI_PID_SETUP : ( (endpoint_addr & TUSB_DIR_IN_MASK) ? OHCI_PID_IN : OHCI_PID_OUT );
|
||||||
p_ed->speed = _usbh_devices[dev_addr].speed;
|
p_ed->speed = _usbh_devices[dev_addr].speed;
|
||||||
p_ed->is_iso = (xfer_type == TUSB_XFER_ISOCHRONOUS) ? 1 : 0;
|
p_ed->is_iso = (xfer_type == TUSB_XFER_ISOCHRONOUS) ? 1 : 0;
|
||||||
p_ed->max_package_size = max_packet_size;
|
p_ed->max_packet_size = max_packet_size;
|
||||||
|
|
||||||
p_ed->used = 1;
|
p_ed->used = 1;
|
||||||
p_ed->is_interrupt_xfer = (xfer_type == TUSB_XFER_INTERRUPT ? 1 : 0);
|
p_ed->is_interrupt_xfer = (xfer_type == TUSB_XFER_INTERRUPT ? 1 : 0);
|
||||||
@@ -360,8 +360,8 @@ static inline ohci_ed_t * ed_from_addr(uint8_t dev_addr, uint8_t ep_addr)
|
|||||||
|
|
||||||
for(uint32_t i=0; i<HCD_MAX_ENDPOINT; i++)
|
for(uint32_t i=0; i<HCD_MAX_ENDPOINT; i++)
|
||||||
{
|
{
|
||||||
if ( (ed_pool[i].device_address == dev_addr) &&
|
if ( (ed_pool[i].dev_addr == dev_addr) &&
|
||||||
ep_addr == edpt_addr(ed_pool[i].endpoint_number, ed_pool[i].direction == OHCI_PID_IN) )
|
ep_addr == edpt_addr(ed_pool[i].ep_number, ed_pool[i].pid == OHCI_PID_IN) )
|
||||||
{
|
{
|
||||||
return &ed_pool[i];
|
return &ed_pool[i];
|
||||||
}
|
}
|
||||||
@@ -390,30 +390,30 @@ static ohci_ed_t * ed_list_find_previous(ohci_ed_t const * p_head, ohci_ed_t con
|
|||||||
|
|
||||||
TU_ASSERT(p_prev, NULL);
|
TU_ASSERT(p_prev, NULL);
|
||||||
|
|
||||||
while ( tu_align16(p_prev->next_ed) != 0 && /* not reach null */
|
while ( tu_align16(p_prev->next) != 0 && /* not reach null */
|
||||||
tu_align16(p_prev->next_ed) != (uint32_t) p_ed && /* not found yet */
|
tu_align16(p_prev->next) != (uint32_t) p_ed && /* not found yet */
|
||||||
max_loop > 0)
|
max_loop > 0)
|
||||||
{
|
{
|
||||||
p_prev = (ohci_ed_t const *) tu_align16(p_prev->next_ed);
|
p_prev = (ohci_ed_t const *) tu_align16(p_prev->next);
|
||||||
max_loop--;
|
max_loop--;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ( tu_align16(p_prev->next_ed) == (uint32_t) p_ed ) ? (ohci_ed_t*) p_prev : NULL;
|
return ( tu_align16(p_prev->next) == (uint32_t) p_ed ) ? (ohci_ed_t*) p_prev : NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ed_list_insert(ohci_ed_t * p_pre, ohci_ed_t * p_ed)
|
static void ed_list_insert(ohci_ed_t * p_pre, ohci_ed_t * p_ed)
|
||||||
{
|
{
|
||||||
p_ed->next_ed |= p_pre->next_ed; // to reserve 4 lsb bits
|
p_ed->next |= p_pre->next; // to reserve 4 lsb bits
|
||||||
p_pre->next_ed = (p_pre->next_ed & 0x0FUL) | ((uint32_t) p_ed);
|
p_pre->next = (p_pre->next & 0x0FUL) | ((uint32_t) p_ed);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ed_list_remove(ohci_ed_t * p_head, ohci_ed_t * p_ed)
|
static void ed_list_remove(ohci_ed_t * p_head, ohci_ed_t * p_ed)
|
||||||
{
|
{
|
||||||
ohci_ed_t * const p_prev = ed_list_find_previous(p_head, p_ed);
|
ohci_ed_t * const p_prev = ed_list_find_previous(p_head, p_ed);
|
||||||
|
|
||||||
p_prev->next_ed = (p_prev->next_ed & 0x0fUL) | tu_align16(p_ed->next_ed);
|
p_prev->next = (p_prev->next & 0x0fUL) | tu_align16(p_ed->next);
|
||||||
// point the removed ED's next pointer to list head to make sure HC can always safely move away from this ED
|
// point the removed ED's next pointer to list head to make sure HC can always safely move away from this ED
|
||||||
p_ed->next_ed = (uint32_t) p_head;
|
p_ed->next = (uint32_t) p_head;
|
||||||
p_ed->used = 0; // free ED
|
p_ed->used = 0; // free ED
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -470,7 +470,7 @@ static void td_insert_to_ed(ohci_ed_t* p_ed, ohci_gtd_t * p_gtd)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{ // TODO currently only support queue up to 2 TD each endpoint at a time
|
{ // TODO currently only support queue up to 2 TD each endpoint at a time
|
||||||
((ohci_gtd_t*) tu_align16(p_ed->td_head.address))->next_td = (uint32_t) p_gtd;
|
((ohci_gtd_t*) tu_align16(p_ed->td_head.address))->next = (uint32_t) p_gtd;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -527,7 +527,7 @@ bool hcd_pipe_close(uint8_t rhport, uint8_t dev_addr, uint8_t ep_addr)
|
|||||||
bool hcd_edpt_busy(uint8_t dev_addr, uint8_t ep_addr)
|
bool hcd_edpt_busy(uint8_t dev_addr, uint8_t ep_addr)
|
||||||
{
|
{
|
||||||
ohci_ed_t const * const p_ed = ed_from_addr(dev_addr, ep_addr);
|
ohci_ed_t const * const p_ed = ed_from_addr(dev_addr, ep_addr);
|
||||||
return tu_align16(p_ed->td_head.address) != tu_align16(p_ed->td_tail.address);
|
return tu_align16(p_ed->td_head.address) != tu_align16(p_ed->td_tail);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool hcd_edpt_stalled(uint8_t dev_addr, uint8_t ep_addr)
|
bool hcd_edpt_stalled(uint8_t dev_addr, uint8_t ep_addr)
|
||||||
@@ -540,11 +540,11 @@ bool hcd_edpt_clear_stall(uint8_t dev_addr, uint8_t ep_addr)
|
|||||||
{
|
{
|
||||||
ohci_ed_t * const p_ed = ed_from_addr(dev_addr, ep_addr);
|
ohci_ed_t * const p_ed = ed_from_addr(dev_addr, ep_addr);
|
||||||
|
|
||||||
p_ed->is_stalled = 0;
|
p_ed->is_stalled = 0;
|
||||||
p_ed->td_tail.address &= 0x0Ful; // set tail pointer back to NULL
|
p_ed->td_tail &= 0x0Ful; // set tail pointer back to NULL
|
||||||
|
|
||||||
p_ed->td_head.toggle = 0; // reset data toggle
|
p_ed->td_head.toggle = 0; // reset data toggle
|
||||||
p_ed->td_head.halted = 0;
|
p_ed->td_head.halted = 0;
|
||||||
|
|
||||||
if ( TUSB_XFER_BULK == ed_get_xfer_type(p_ed) ) OHCI_REG->command_status_bit.bulk_list_filled = 1;
|
if ( TUSB_XFER_BULK == ed_get_xfer_type(p_ed) ) OHCI_REG->command_status_bit.bulk_list_filled = 1;
|
||||||
|
|
||||||
@@ -561,10 +561,10 @@ static ohci_td_item_t* list_reverse(ohci_td_item_t* td_head)
|
|||||||
|
|
||||||
while(td_head != NULL)
|
while(td_head != NULL)
|
||||||
{
|
{
|
||||||
uint32_t next = td_head->next_td;
|
uint32_t next = td_head->next;
|
||||||
|
|
||||||
// make current's item become reverse's first item
|
// make current's item become reverse's first item
|
||||||
td_head->next_td = (uint32_t) td_reverse_head;
|
td_head->next = (uint32_t) td_reverse_head;
|
||||||
td_reverse_head = td_head;
|
td_reverse_head = td_head;
|
||||||
|
|
||||||
td_head = (ohci_td_item_t*) next; // advance to next item
|
td_head = (ohci_td_item_t*) next; // advance to next item
|
||||||
@@ -626,17 +626,17 @@ static void done_queue_isr(uint8_t hostid)
|
|||||||
// the TailP must be set back to NULL for processing remaining TDs
|
// the TailP must be set back to NULL for processing remaining TDs
|
||||||
if ((event != XFER_RESULT_SUCCESS))
|
if ((event != XFER_RESULT_SUCCESS))
|
||||||
{
|
{
|
||||||
p_ed->td_tail.address &= 0x0Ful;
|
p_ed->td_tail &= 0x0Ful;
|
||||||
p_ed->td_tail.address |= tu_align16(p_ed->td_head.address); // mark halted EP as empty queue
|
p_ed->td_tail |= tu_align16(p_ed->td_head.address); // mark halted EP as empty queue
|
||||||
if ( event == XFER_RESULT_STALLED ) p_ed->is_stalled = 1;
|
if ( event == XFER_RESULT_STALLED ) p_ed->is_stalled = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
hcd_event_xfer_complete(p_ed->device_address,
|
hcd_event_xfer_complete(p_ed->dev_addr,
|
||||||
edpt_addr(p_ed->endpoint_number, p_ed->direction == OHCI_PID_IN),
|
edpt_addr(p_ed->ep_number, p_ed->pid == OHCI_PID_IN),
|
||||||
event, xferred_bytes);
|
event, xferred_bytes);
|
||||||
}
|
}
|
||||||
|
|
||||||
td_head = (ohci_td_item_t*) td_head->next_td;
|
td_head = (ohci_td_item_t*) td_head->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -83,7 +83,7 @@ TU_VERIFY_STATIC( sizeof(ohci_hcca_t) == 256, "size is not correct" );
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint32_t reserved[2];
|
uint32_t reserved[2];
|
||||||
volatile uint32_t next_td;
|
volatile uint32_t next;
|
||||||
uint32_t reserved2;
|
uint32_t reserved2;
|
||||||
}ohci_td_item_t;
|
}ohci_td_item_t;
|
||||||
|
|
||||||
@@ -105,8 +105,8 @@ typedef struct ATTR_ALIGNED(16)
|
|||||||
// Word 1
|
// Word 1
|
||||||
volatile uint8_t* current_buffer_pointer;
|
volatile uint8_t* current_buffer_pointer;
|
||||||
|
|
||||||
// Word 2
|
// Word 2 : next TD
|
||||||
volatile uint32_t next_td;
|
volatile uint32_t next;
|
||||||
|
|
||||||
// Word 3
|
// Word 3
|
||||||
uint8_t* buffer_end;
|
uint8_t* buffer_end;
|
||||||
@@ -114,28 +114,26 @@ typedef struct ATTR_ALIGNED(16)
|
|||||||
|
|
||||||
TU_VERIFY_STATIC( sizeof(ohci_gtd_t) == 16, "size is not correct" );
|
TU_VERIFY_STATIC( sizeof(ohci_gtd_t) == 16, "size is not correct" );
|
||||||
|
|
||||||
typedef struct ATTR_ALIGNED(16) {
|
typedef struct ATTR_ALIGNED(16)
|
||||||
//------------- Word 0 -------------//
|
{
|
||||||
uint32_t device_address : 7;
|
// Word 0
|
||||||
uint32_t endpoint_number : 4;
|
uint32_t dev_addr : 7;
|
||||||
uint32_t direction : 2;
|
uint32_t ep_number : 4;
|
||||||
uint32_t speed : 1;
|
uint32_t pid : 2; // 00b from TD, 01b Out, 10b In
|
||||||
uint32_t skip : 1;
|
uint32_t speed : 1;
|
||||||
uint32_t is_iso : 1;
|
uint32_t skip : 1;
|
||||||
uint32_t max_package_size : 11;
|
uint32_t is_iso : 1;
|
||||||
|
uint32_t max_packet_size : 11;
|
||||||
uint32_t used : 1; // HCD
|
// HCD: make use of 5 reserved bits
|
||||||
|
uint32_t used : 1;
|
||||||
uint32_t is_interrupt_xfer : 1;
|
uint32_t is_interrupt_xfer : 1;
|
||||||
uint32_t is_stalled : 1;
|
uint32_t is_stalled : 1;
|
||||||
uint32_t : 2;
|
uint32_t : 2;
|
||||||
|
|
||||||
|
// Word 1
|
||||||
|
uint32_t td_tail;
|
||||||
|
|
||||||
//------------- Word 1 -------------//
|
// Word 2
|
||||||
union {
|
|
||||||
uint32_t address;
|
|
||||||
}td_tail;
|
|
||||||
|
|
||||||
//------------- Word 2 -------------//
|
|
||||||
volatile union {
|
volatile union {
|
||||||
uint32_t address;
|
uint32_t address;
|
||||||
struct {
|
struct {
|
||||||
@@ -145,13 +143,14 @@ typedef struct ATTR_ALIGNED(16) {
|
|||||||
};
|
};
|
||||||
}td_head;
|
}td_head;
|
||||||
|
|
||||||
//------------- Word 3 -------------//
|
// Word 3: next ED (4 lsb bits are free to use )
|
||||||
uint32_t next_ed; // 4 lsb bits are free to use
|
uint32_t next;
|
||||||
} ohci_ed_t;
|
} ohci_ed_t;
|
||||||
|
|
||||||
TU_VERIFY_STATIC( sizeof(ohci_ed_t) == 16, "size is not correct" );
|
TU_VERIFY_STATIC( sizeof(ohci_ed_t) == 16, "size is not correct" );
|
||||||
|
|
||||||
typedef struct ATTR_ALIGNED(32) {
|
typedef struct ATTR_ALIGNED(32)
|
||||||
|
{
|
||||||
/*---------- Word 1 ----------*/
|
/*---------- Word 1 ----------*/
|
||||||
uint32_t starting_frame : 16;
|
uint32_t starting_frame : 16;
|
||||||
uint32_t : 5; // can be used
|
uint32_t : 5; // can be used
|
||||||
@@ -159,13 +158,12 @@ typedef struct ATTR_ALIGNED(32) {
|
|||||||
uint32_t frame_count : 3;
|
uint32_t frame_count : 3;
|
||||||
uint32_t : 1; // can be used
|
uint32_t : 1; // can be used
|
||||||
volatile uint32_t condition_code : 4;
|
volatile uint32_t condition_code : 4;
|
||||||
/*---------- End Word 1 ----------*/
|
|
||||||
|
|
||||||
/*---------- Word 2 ----------*/
|
/*---------- Word 2 ----------*/
|
||||||
uint32_t buffer_page0; // 12 lsb bits can be used
|
uint32_t buffer_page0; // 12 lsb bits can be used
|
||||||
|
|
||||||
/*---------- Word 3 ----------*/
|
/*---------- Word 3 ----------*/
|
||||||
volatile uint32_t next_td;
|
volatile uint32_t next;
|
||||||
|
|
||||||
/*---------- Word 4 ----------*/
|
/*---------- Word 4 ----------*/
|
||||||
uint32_t buffer_end;
|
uint32_t buffer_end;
|
||||||
|
Reference in New Issue
Block a user