41 #if MODE_HOST_SUPPORTED && (TUSB_CFG_MCU == MCU_LPC175X_6X)
50 #include "../usbh_hcd.h"
56 #define OHCI_REG ((ohci_registers_t *) LPC_USB_BASE)
59 OHCI_CONTROL_FUNCSTATE_RESET = 0,
60 OHCI_CONTROL_FUNCSTATE_RESUME,
61 OHCI_CONTROL_FUNCSTATE_OPERATIONAL,
62 OHCI_CONTROL_FUNCSTATE_SUSPEND
66 OHCI_CONTROL_CONTROL_BULK_RATIO = 3,
67 OHCI_CONTROL_LIST_PERIODIC_ENABLE_MASK =
BIT_(2),
68 OHCI_CONTROL_LIST_ISOCHRONOUS_ENABLE_MASK =
BIT_(3),
69 OHCI_CONTROL_LIST_CONTROL_ENABLE_MASK =
BIT_(4),
70 OHCI_CONTROL_LIST_BULK_ENABLE_MASK =
BIT_(5),
74 OHCI_FMINTERVAL_FI = 0x2EDF,
75 OHCI_FMINTERVAL_FSMPS = (6*(OHCI_FMINTERVAL_FI-210)) / 7,
79 OHCI_PERIODIC_START = 0x3E67
83 #pragma diag_suppress 66 // Suppress Keil warnings #66-D: enumeration value is out of "int" range
87 OHCI_INT_SCHEDULING_OVERUN_MASK =
BIT_(0),
88 OHCI_INT_WRITEBACK_DONEHEAD_MASK =
BIT_(1),
89 OHCI_INT_SOF_MASK =
BIT_(2),
90 OHCI_INT_RESUME_DETECTED_MASK =
BIT_(3),
91 OHCI_INT_UNRECOVERABLE_ERROR_MASK =
BIT_(4),
92 OHCI_INT_FRAME_OVERFLOW_MASK =
BIT_(5),
93 OHCI_INT_RHPORT_STATUS_CHANGE_MASK =
BIT_(6),
95 OHCI_INT_OWNERSHIP_CHANGE_MASK =
BIT_(30),
96 OHCI_INT_MASTER_ENABLE_MASK =
BIT_(31),
100 #pragma diag_default 66 // return Keil 66 to normal severity
104 OHCI_RHPORT_CURRENT_CONNECT_STATUS_MASK =
BIT_(0),
105 OHCI_RHPORT_PORT_ENABLE_STATUS_MASK =
BIT_(1),
106 OHCI_RHPORT_PORT_SUSPEND_STATUS_MASK =
BIT_(2),
107 OHCI_RHPORT_PORT_OVER_CURRENT_INDICATOR_MASK =
BIT_(3),
108 OHCI_RHPORT_PORT_RESET_STATUS_MASK =
BIT_(4),
110 OHCI_RHPORT_PORT_POWER_STATUS_MASK =
BIT_(8),
111 OHCI_RHPORT_LOW_SPEED_DEVICE_ATTACHED_MASK =
BIT_(9),
113 OHCI_RHPORT_CONNECT_STATUS_CHANGE_MASK =
BIT_(16),
114 OHCI_RHPORT_PORT_ENABLE_CHANGE_MASK =
BIT_(17),
115 OHCI_RHPORT_PORT_SUSPEND_CHANGE_MASK =
BIT_(18),
116 OHCI_RHPORT_OVER_CURRENT_CHANGE_MASK =
BIT_(19),
117 OHCI_RHPORT_PORT_RESET_CHANGE_MASK =
BIT_(20),
119 OHCI_RHPORT_ALL_CHANGE_MASK = OHCI_RHPORT_CONNECT_STATUS_CHANGE_MASK | OHCI_RHPORT_PORT_ENABLE_CHANGE_MASK |
120 OHCI_RHPORT_PORT_SUSPEND_CHANGE_MASK | OHCI_RHPORT_OVER_CURRENT_CHANGE_MASK | OHCI_RHPORT_PORT_RESET_CHANGE_MASK
124 OHCI_CCODE_NO_ERROR = 0,
126 OHCI_CCODE_BIT_STUFFING = 2,
127 OHCI_CCODE_DATA_TOGGLE_MISMATCH = 3,
128 OHCI_CCODE_STALL = 4,
129 OHCI_CCODE_DEVICE_NOT_RESPONDING = 5,
130 OHCI_CCODE_PID_CHECK_FAILURE = 6,
131 OHCI_CCODE_UNEXPECTED_PID = 7,
132 OHCI_CCODE_DATA_OVERRUN = 8,
133 OHCI_CCODE_DATA_UNDERRUN = 9,
134 OHCI_CCODE_BUFFER_OVERRUN = 12,
135 OHCI_CCODE_BUFFER_UNDERRUN = 13,
136 OHCI_CCODE_NOT_ACCESSED = 14,
140 OHCI_INT_ON_COMPLETE_YES = 0,
141 OHCI_INT_ON_COMPLETE_NO = BIN8(111)
150 [TUSB_XFER_CONTROL] = &ohci_data.control[0].ed,
151 [TUSB_XFER_BULK ] = &ohci_data.bulk_head_ed,
152 [TUSB_XFER_INTERRUPT] = &ohci_data.period_head_ed,
153 [TUSB_XFER_ISOCHRONOUS] = NULL
169 for(uint8_t i=0; i<32; i++)
171 ohci_data.hcca.interrupt_table[i] = (uint32_t) &ohci_data.period_head_ed;
174 ohci_data.control[0].ed.skip = 1;
175 ohci_data.bulk_head_ed.skip = 1;
176 ohci_data.period_head_ed.skip = 1;
179 OHCI_REG->command_status_bit.controller_reset = 1;
180 while( OHCI_REG->command_status_bit.controller_reset ) {}
183 OHCI_REG->control_head_ed = (uint32_t) &ohci_data.control[0].ed;
184 OHCI_REG->bulk_head_ed = (uint32_t) &ohci_data.bulk_head_ed;
185 OHCI_REG->hcca = (uint32_t) &ohci_data.hcca;
187 OHCI_REG->interrupt_disable = OHCI_REG->interrupt_enable;
188 OHCI_REG->interrupt_status = OHCI_REG->interrupt_status;
189 OHCI_REG->interrupt_enable = OHCI_INT_WRITEBACK_DONEHEAD_MASK | OHCI_INT_RESUME_DETECTED_MASK |
190 OHCI_INT_UNRECOVERABLE_ERROR_MASK | OHCI_INT_RHPORT_STATUS_CHANGE_MASK |
191 OHCI_INT_MASTER_ENABLE_MASK;
193 OHCI_REG->control |= OHCI_CONTROL_CONTROL_BULK_RATIO | OHCI_CONTROL_LIST_CONTROL_ENABLE_MASK |
194 OHCI_CONTROL_LIST_BULK_ENABLE_MASK | OHCI_CONTROL_LIST_PERIODIC_ENABLE_MASK;
196 OHCI_REG->frame_interval = (OHCI_FMINTERVAL_FSMPS << 16) | OHCI_FMINTERVAL_FI;
197 OHCI_REG->periodic_start = (OHCI_FMINTERVAL_FI * 9) / 10;
199 OHCI_REG->control_bit.hc_functional_state = OHCI_CONTROL_FUNCSTATE_OPERATIONAL;
200 OHCI_REG->rh_status_bit.local_power_status_change = 1;
202 return TUSB_ERROR_NONE;
208 void hcd_port_reset(uint8_t hostid)
210 OHCI_REG->rhport_status[0] = OHCI_RHPORT_PORT_RESET_STATUS_MASK;
215 return OHCI_REG->rhport_status_bit[0].current_connect_status;
220 return OHCI_REG->rhport_status_bit[0].low_speed_device_attached ? TUSB_SPEED_LOW : TUSB_SPEED_FULL;
224 void hcd_port_unplug(uint8_t hostid)
239 return (p_ed->endpoint_number == 0 ) ? TUSB_XFER_CONTROL :
240 (p_ed->is_iso ) ? TUSB_XFER_ISOCHRONOUS :
241 (p_ed->is_interrupt_xfer ) ? TUSB_XFER_INTERRUPT : TUSB_XFER_BULK;
244 static void ed_init(
ohci_ed_t *p_ed, uint8_t dev_addr, uint16_t max_packet_size, uint8_t endpoint_addr, uint8_t xfer_type, uint8_t interval)
252 p_ed->device_address = dev_addr;
253 p_ed->endpoint_number = endpoint_addr & 0x0F;
254 p_ed->direction = (xfer_type == TUSB_XFER_CONTROL) ? OHCI_PID_SETUP : ( (endpoint_addr & TUSB_DIR_DEV_TO_HOST_MASK) ? OHCI_PID_IN : OHCI_PID_OUT );
255 p_ed->speed = usbh_devices[dev_addr].speed;
256 p_ed->is_iso = (xfer_type == TUSB_XFER_ISOCHRONOUS) ? 1 : 0;
257 p_ed->max_package_size = max_packet_size;
260 p_ed->is_interrupt_xfer = (xfer_type == TUSB_XFER_INTERRUPT ? 1 : 0);
263 static void gtd_init(
ohci_gtd_t* p_td,
void* data_ptr, uint16_t total_bytes)
268 p_td->expected_bytes = total_bytes;
270 p_td->buffer_rounding = 1;
271 p_td->delay_interrupt = OHCI_INT_ON_COMPLETE_NO;
272 p_td->condition_code = OHCI_CCODE_NOT_ACCESSED;
274 p_td->current_buffer_pointer = data_ptr;
275 p_td->buffer_end = total_bytes ? (((uint8_t*) data_ptr) + total_bytes-1) : NULL;
278 tusb_error_t hcd_pipe_control_open(uint8_t dev_addr, uint8_t max_packet_size)
280 ohci_ed_t*
const p_ed = &ohci_data.control[dev_addr].ed;
282 ed_init(p_ed, dev_addr, max_packet_size, 0, TUSB_XFER_CONTROL, 0);
286 ed_list_insert( p_ed_head[TUSB_XFER_CONTROL], p_ed);
292 return TUSB_ERROR_NONE;
297 ohci_ed_t*
const p_ed = &ohci_data.control[dev_addr].ed;
299 ohci_gtd_t *p_setup = &ohci_data.control[dev_addr].gtd[0];
304 gtd_init(p_setup, (
void*) p_request, 8);
305 p_setup->index = dev_addr;
306 p_setup->pid = OHCI_PID_SETUP;
307 p_setup->data_toggle = BIN8(10);
308 p_setup->next_td = (uint32_t) p_data;
311 if (p_request->wLength > 0)
313 gtd_init(p_data, data, p_request->wLength);
314 p_data->index = dev_addr;
315 p_data->pid = p_request->bmRequestType_bit.direction ? OHCI_PID_IN : OHCI_PID_OUT;
316 p_data->data_toggle = BIN8(11);
321 p_data->next_td = (uint32_t) p_status;
324 gtd_init(p_status, NULL, 0);
325 p_status->index = dev_addr;
326 p_status->pid = p_request->bmRequestType_bit.direction ? OHCI_PID_OUT : OHCI_PID_IN;
327 p_status->data_toggle = BIN8(11);
328 p_status->delay_interrupt = OHCI_INT_ON_COMPLETE_YES;
331 p_ed->td_head.address = (uint32_t) p_setup;
333 OHCI_REG->command_status_bit.control_list_filled = 1;
335 return TUSB_ERROR_NONE;
340 ohci_ed_t*
const p_ed = &ohci_data.control[dev_addr].ed;
347 ed_list_remove( p_ed_head[ ed_get_xfer_type(p_ed)], p_ed );
350 usbh_devices[dev_addr].state = TUSB_DEVICE_STATE_UNPLUG;
353 return TUSB_ERROR_NONE;
360 static inline uint8_t ed_get_index(
ohci_ed_t const *
const p_ed)
362 return p_ed - ohci_data.device[p_ed->device_address-1].ed;
368 return &ohci_data.device[pipe_hdl.dev_addr-1].ed[pipe_hdl.index];
372 static inline ohci_ed_t * ed_find_free(uint8_t dev_addr)
374 for(uint8_t i = 0; i < HCD_MAX_ENDPOINT; i++)
376 if ( !ohci_data.device[dev_addr-1].ed[i].used )
378 return &ohci_data.device[dev_addr-1].ed[i];
387 uint32_t max_loop = HCD_MAX_ENDPOINT;
391 ASSERT_PTR(p_prev, NULL);
393 while ( align16(p_prev->next_ed) != 0 &&
394 align16(p_prev->next_ed) != (uint32_t) p_ed &&
397 p_prev = (
ohci_ed_t const *) align16(p_prev->next_ed);
401 return ( align16(p_prev->next_ed) == (uint32_t) p_ed ) ? (
ohci_ed_t*) p_prev : NULL;
406 p_ed->next_ed |= p_pre->next_ed;
407 p_pre->next_ed = (p_pre->next_ed & 0x0FUL) | ((uint32_t) p_ed);
412 ohci_ed_t *
const p_prev = ed_list_find_previous(p_head, p_ed);
414 p_prev->next_ed = (p_prev->next_ed & 0x0fUL) | align16(p_ed->next_ed);
416 p_ed->next_ed = (uint32_t) p_head;
422 pipe_handle_t const null_handle = { .dev_addr = 0, .xfer_type = 0, .index = 0 };
425 ASSERT(p_endpoint_desc->
bmAttributes.xfer != TUSB_XFER_ISOCHRONOUS, null_handle );
428 ohci_ed_t *
const p_ed = ed_find_free(dev_addr);
429 ASSERT_PTR(p_ed, null_handle);
431 ed_init( p_ed, dev_addr, p_endpoint_desc->wMaxPacketSize.
size, p_endpoint_desc->
bEndpointAddress,
433 p_ed->td_tail.class_code = class_code;
435 ed_list_insert( p_ed_head[p_endpoint_desc->
bmAttributes.xfer], p_ed );
439 .dev_addr = dev_addr,
441 .index = ed_get_index(p_ed)
445 static ohci_gtd_t * gtd_find_free(uint8_t dev_addr)
447 for(uint8_t i=0; i < HCD_MAX_XFER; i++)
449 if (!ohci_data.device[dev_addr-1].gtd[i].used)
451 return &ohci_data.device[dev_addr-1].gtd[i];
461 if ( align16(p_ed->td_head.address) == 0 )
463 p_ed->td_head.address |= (uint32_t) p_gtd;
467 ((
ohci_gtd_t*) align16(p_ed->td_head.address))->next_td = (uint32_t) p_gtd;
471 static tusb_error_t pipe_queue_xfer(
pipe_handle_t pipe_hdl, uint8_t buffer[], uint16_t total_bytes,
bool int_on_complete)
473 ohci_ed_t*
const p_ed = ed_from_pipe_handle(pipe_hdl);
477 ohci_gtd_t *
const p_gtd = gtd_find_free(pipe_hdl.dev_addr);
478 ASSERT_PTR(p_gtd, TUSB_ERROR_EHCI_NOT_ENOUGH_QTD);
480 gtd_init(p_gtd, buffer, total_bytes);
481 p_gtd->index = pipe_hdl.index;
482 if ( int_on_complete ) p_gtd->delay_interrupt = OHCI_INT_ON_COMPLETE_YES;
484 td_insert_to_ed(p_ed, p_gtd);
487 ASSERT_STATUS(TUSB_ERROR_NOT_SUPPORTED_YET);
490 return TUSB_ERROR_NONE;
495 return pipe_queue_xfer(pipe_hdl, buffer, total_bytes,
false);
500 ASSERT_STATUS( pipe_queue_xfer(pipe_hdl, buffer, total_bytes,
true) );
502 tusb_xfer_type_t xfer_type = ed_get_xfer_type( ed_from_pipe_handle(pipe_hdl) );
504 if (TUSB_XFER_BULK == xfer_type) OHCI_REG->command_status_bit.bulk_list_filled = 1;
506 return TUSB_ERROR_NONE;
514 ohci_ed_t *
const p_ed = ed_from_pipe_handle(pipe_hdl);
516 ed_list_remove( p_ed_head[ ed_get_xfer_type(p_ed)], p_ed );
518 return TUSB_ERROR_FAILED;
523 ohci_ed_t const *
const p_ed = ed_from_pipe_handle(pipe_hdl);
524 return align16(p_ed->td_head.address) != align16(p_ed->td_tail.address);
529 ohci_ed_t const *
const p_ed = ed_from_pipe_handle(pipe_hdl);
530 return p_ed->td_head.halted;
535 ohci_ed_t const *
const p_ed = ed_from_pipe_handle(pipe_hdl);
536 return p_ed->td_head.halted && p_ed->is_stalled;
541 ohci_ed_t const *
const p_ed = ed_from_pipe_handle(pipe_hdl);
542 return p_ed->endpoint_number | (p_ed->direction == OHCI_PID_IN ? TUSB_DIR_DEV_TO_HOST_MASK : 0 );
547 ohci_ed_t *
const p_ed = ed_from_pipe_handle(pipe_hdl);
549 p_ed->is_stalled = 0;
550 p_ed->td_tail.address &= 0x0Ful;
552 p_ed->td_head.toggle = 0;
553 p_ed->td_head.halted = 0;
555 if ( TUSB_XFER_BULK == ed_get_xfer_type(p_ed) ) OHCI_REG->command_status_bit.bulk_list_filled = 1;
557 return TUSB_ERROR_NONE;
568 while(td_head != NULL)
570 uint32_t next = td_head->next_td;
573 td_head->next_td = (uint32_t) td_reverse_head;
574 td_reverse_head = td_head;
579 return td_reverse_head;
583 static inline bool gtd_is_control(
ohci_gtd_t const *
const p_qtd)
585 return ((uint32_t) p_qtd) < ((uint32_t) ohci_data.device);
591 if ( gtd_is_control(p_qtd) )
593 return &ohci_data.control[p_qtd->index].ed;
596 uint8_t dev_addr_idx = (((uint32_t)p_qtd) - ((uint32_t)ohci_data.device)) /
sizeof(ohci_data.device[0]);
598 return &ohci_data.device[dev_addr_idx].ed[p_qtd->index];
603 static inline uint32_t gtd_xfer_byte_left(uint32_t buffer_end, uint32_t current_buffer)
605 return (align4k(buffer_end ^ current_buffer) ? 0x1000 : 0) +
606 offset4k(buffer_end) - offset4k(current_buffer) + 1;
609 static void done_queue_isr(uint8_t hostid)
616 while( td_head != NULL && max_loop > 0)
621 tusb_event_t
const event = (p_qtd->condition_code == OHCI_CCODE_NO_ERROR) ? TUSB_EVENT_XFER_COMPLETE :
622 (p_qtd->condition_code == OHCI_CCODE_STALL) ? TUSB_EVENT_XFER_STALLED : TUSB_EVENT_XFER_ERROR;
625 if ( (p_qtd->delay_interrupt == OHCI_INT_ON_COMPLETE_YES) || (
event != TUSB_EVENT_XFER_COMPLETE) )
627 ohci_ed_t *
const p_ed = gtd_get_ed(p_qtd);
629 uint32_t
const xferred_bytes = p_qtd->expected_bytes - gtd_xfer_byte_left((uint32_t) p_qtd->buffer_end, (uint32_t) p_qtd->current_buffer_pointer);
638 if ((event != TUSB_EVENT_XFER_COMPLETE))
640 p_ed->td_tail.address &= 0x0Ful;
641 p_ed->td_tail.address |= align16(p_ed->td_head.address);
642 if ( event == TUSB_EVENT_XFER_STALLED ) p_ed->is_stalled = 1;
647 .dev_addr = p_ed->device_address,
648 .xfer_type = ed_get_xfer_type(p_ed),
651 if ( pipe_hdl.xfer_type != TUSB_XFER_CONTROL) pipe_hdl.index = ed_get_index(p_ed);
653 usbh_xfer_isr(pipe_hdl, p_ed->td_tail.class_code, event, xferred_bytes);
661 void hcd_isr(uint8_t hostid)
663 uint32_t
const int_en = OHCI_REG->interrupt_enable;
664 uint32_t
const int_status = OHCI_REG->interrupt_status & int_en;
666 if (int_status == 0)
return;
669 if ( int_status & OHCI_INT_RHPORT_STATUS_CHANGE_MASK )
671 uint32_t
const rhport_status = OHCI_REG->rhport_status[0] & OHCI_RHPORT_ALL_CHANGE_MASK;
674 if ( rhport_status & OHCI_RHPORT_CONNECT_STATUS_CHANGE_MASK )
677 if ( OHCI_REG->rhport_status_bit[0].current_connect_status )
680 OHCI_REG->rhport_status[0] = OHCI_RHPORT_PORT_RESET_STATUS_MASK;
681 usbh_hcd_rhport_plugged_isr(0);
684 usbh_hcd_rhport_unplugged_isr(0);
688 if ( rhport_status & OHCI_RHPORT_PORT_SUSPEND_CHANGE_MASK)
693 OHCI_REG->rhport_status[0] = rhport_status;
697 if ( int_status & OHCI_INT_WRITEBACK_DONEHEAD_MASK)
699 done_queue_isr(hostid);
702 OHCI_REG->interrupt_status = int_status;
tusb_speed_t
defined base on EHCI specs value for Endpoint Speed
bool hcd_port_connect_status(uint8_t hostid) ATTR_PURE ATTR_WARN_UNUSED_RESULT
return the current connect status of roothub port
#define TUSB_CFG_HOST_DEVICE_MAX
Maximum number of device host stack can manage If hub class is not enabled, set this equal to numbe...
#define ATTR_PURE
Many functions have no effects except the return value and their return value depends only on the par...
struct tusb_descriptor_endpoint_t::@8 bmAttributes
This field describes the endpoint's attributes when it is configured using the bConfigurationValue. Bits 1..0: Transfer Type - 00 = Control - 01 = Isochronous - 10 = Bulk - 11 = Interrupt If not an isochronous endpoint, bits 5..2 are reserved and must be set to zero. If isochronous, they are defined as follows: Bits 3..2: Synchronization Type - 00 = No Synchronization - 01 = Asynchronous - 10 = Adaptive - 11 = Synchronous Bits 5..4: Usage Type - 00 = Data endpoint - 01 = Feedback endpoint - 10 = Implicit feedback Data endpoint - 11 = Reserved Refer to Chapter 5 of USB 2.0 specification for more information. All other bits are reserved and must be reset to zero. Reserved bits must be ignored by the host.
#define TUSB_CFG_ATTR_USBRAM
USB Standard Endpoint Descriptor (section 9.6.1 table 9-13)
uint8_t bInterval
Interval for polling endpoint for data transfers. Expressed in frames or microframes depending on the...
tusb_error_t
Error Code returned.
#define ATTR_ALIGNED(Bytes)
This attribute specifies a minimum alignment for the variable or structure field, measured in bytes...
#define ATTR_ALWAYS_INLINE
Generally, functions are not inlined unless optimization is specified. For functions declared inline...
#define ATTR_CONST
Many functions do not examine any values except their arguments, and have no effects except the retur...
uint16_t size
Maximum packet size this endpoint is capable of sending or receiving when this configuration is selec...
tusb_xfer_type_t
defined base on USB Specs Endpoint's bmAttributes
uint8_t bEndpointAddress
The address of the endpoint on the USB device described by this descriptor. The address is encoded as...