Changeset 816335c in mainline
- Timestamp:
- 2017-10-04T10:06:10Z (8 years ago)
- Branches:
- lfn, master, serial, ticket/834-toolchain-update, topic/msim-upgrade, topic/simplify-dev-export
- Children:
- c68c713c
- Parents:
- 5c5c9407
- Location:
- uspace/drv/bus/usb/xhci
- Files:
-
- 6 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/drv/bus/usb/xhci/debug.c
r5c5c9407 r816335c 288 288 ports_from = XHCI_REG_RD(ec, XHCI_EC_SP_CP_OFF); 289 289 ports_to = ports_from + XHCI_REG_RD(ec, XHCI_EC_SP_CP_COUNT) - 1; 290 usb_log_debug("\tProtocol % 4s%u.%u, ports %u-%u", name.str,290 usb_log_debug("\tProtocol %.4s%u.%u, ports %u-%u", name.str, 291 291 XHCI_REG_RD(ec, XHCI_EC_SP_MAJOR), 292 292 XHCI_REG_RD(ec, XHCI_EC_SP_MINOR), -
uspace/drv/bus/usb/xhci/hc.c
r5c5c9407 r816335c 48 48 */ 49 49 #define PSI_TO_BPS(psie, psim) (((uint64_t) psim) << (10 * psie)) 50 #define PORT_SPEED(psie, psim) { \ 50 #define PORT_SPEED(mjr, psie, psim) { \ 51 .name = "USB ", \ 52 .major = mjr, \ 53 .minor = 0, \ 51 54 .rx_bps = PSI_TO_BPS(psie, psim), \ 52 55 .tx_bps = PSI_TO_BPS(psie, psim) \ 53 56 } 54 static const xhci_port_speed_t ps_default_full = PORT_SPEED(2, 12);55 static const xhci_port_speed_t ps_default_low = PORT_SPEED( 1, 1500);56 static const xhci_port_speed_t ps_default_high = PORT_SPEED(2, 480);57 static const xhci_port_speed_t ps_default_super = PORT_SPEED(3, 5);57 static const xhci_port_speed_t ps_default_full = PORT_SPEED(2, 2, 12); 58 static const xhci_port_speed_t ps_default_low = PORT_SPEED(2, 1, 1500); 59 static const xhci_port_speed_t ps_default_high = PORT_SPEED(2, 2, 480); 60 static const xhci_port_speed_t ps_default_super = PORT_SPEED(3, 3, 5); 58 61 59 62 /** … … 62 65 static int hc_parse_ec(xhci_hc_t *hc) 63 66 { 64 unsigned psic, major; 67 unsigned psic, major, minor; 68 xhci_sp_name_t name; 69 70 xhci_port_speed_t *speeds = hc->rh.speeds; 65 71 66 72 for (xhci_extcap_t *ec = hc->xecp; ec; ec = xhci_extcap_next(ec)) { … … 74 80 psic = XHCI_REG_RD(ec, XHCI_EC_SP_PSIC); 75 81 major = XHCI_REG_RD(ec, XHCI_EC_SP_MAJOR); 82 minor = XHCI_REG_RD(ec, XHCI_EC_SP_MINOR); 83 name.packed = host2uint32_t_le(XHCI_REG_RD(ec, XHCI_EC_SP_NAME)); 84 85 if (name.packed != xhci_name_usb.packed) { 86 /** 87 * The detection of such protocol would work, 88 * but the rest of the implementation is made 89 * for the USB protocol only. 90 */ 91 usb_log_error("Unknown protocol %.4s.", name.str); 92 return ENOTSUP; 93 } 76 94 77 95 // "Implied" speed 78 96 if (psic == 0) { 79 /* 80 * According to section 7.2.2.1.2, only USB 2.0 81 * and USB 3.0 can have psic == 0. So we 82 * blindly assume the name == "USB " and minor 83 * == 0. 84 */ 85 86 unsigned ports_from = XHCI_REG_RD(ec, XHCI_EC_SP_CP_OFF); 87 unsigned ports_to = ports_from 88 + XHCI_REG_RD(ec, XHCI_EC_SP_CP_COUNT) - 1; 97 assert(minor == 0); 89 98 90 99 if (major == 2) { 91 hc->speeds[1] = ps_default_full; 92 hc->speeds[2] = ps_default_low; 93 hc->speeds[3] = ps_default_high; 94 hc->rh.usb2_port_start = ports_from; 95 hc->rh.usb2_port_end = ports_to; 100 speeds[1] = ps_default_full; 101 speeds[2] = ps_default_low; 102 speeds[3] = ps_default_high; 96 103 } else if (major == 3) { 97 hc->speeds[4] = ps_default_super; 98 hc->rh.usb3_port_start = ports_from; 99 hc->rh.usb3_port_end = ports_to; 104 speeds[4] = ps_default_super; 100 105 } else { 101 106 return EINVAL; 102 107 } 103 108 104 usb_log_debug2("Implied speed of USB %u set up.", major);109 usb_log_debug2("Implied speed of USB %u.0 set up.", major); 105 110 } else { 106 111 for (unsigned i = 0; i < psic; i++) { … … 111 116 unsigned psim = XHCI_REG_RD(psi, XHCI_PSI_PSIM); 112 117 118 speeds[psiv].major = major; 119 speeds[psiv].minor = minor; 120 str_ncpy(speeds[psiv].name, 4, name.str, 4); 121 113 122 uint64_t bps = PSI_TO_BPS(psie, psim); 114 123 115 124 if (sim == XHCI_PSI_PLT_SYMM || sim == XHCI_PSI_PLT_RX) 116 hc->speeds[psiv].rx_bps = bps;125 speeds[psiv].rx_bps = bps; 117 126 if (sim == XHCI_PSI_PLT_SYMM || sim == XHCI_PSI_PLT_TX) { 118 hc->speeds[psiv].tx_bps = bps;119 usb_log_debug2("Speed %u set up for bps %" PRIu64 " / %" PRIu64 ".", psiv, hc->speeds[psiv].rx_bps, hc->speeds[psiv].tx_bps);127 speeds[psiv].tx_bps = bps; 128 usb_log_debug2("Speed %u set up for bps %" PRIu64 " / %" PRIu64 ".", psiv, speeds[psiv].rx_bps, speeds[psiv].tx_bps); 120 129 } 121 130 } … … 202 211 goto err_scratch; 203 212 204 if ((err = xhci_rh_init(&hc->rh, hc ->op_regs)))213 if ((err = xhci_rh_init(&hc->rh, hc))) 205 214 goto err_cmd; 206 215 -
uspace/drv/bus/usb/xhci/hc.h
r5c5c9407 r816335c 49 49 } xhci_virt_device_ctx_t; 50 50 51 /**52 * xHCI lets the controller define speeds of ports it controls.53 */54 typedef struct xhci_port_speed {55 uint64_t rx_bps, tx_bps;56 } xhci_port_speed_t;57 58 51 typedef struct xhci_hc { 59 52 /* MMIO range */ … … 80 73 81 74 /* Cached capabilities */ 82 xhci_port_speed_t speeds [16];83 75 unsigned max_slots; 84 76 bool ac64; -
uspace/drv/bus/usb/xhci/hw_struct/regs.h
r5c5c9407 r816335c 241 241 * 15:14 - PIC 242 242 * 27 26 25 24 23 22 21 20 19 18 17 16 243 243 * 27:16 - WOE WDE WCE CAS CEC PLC PRC OCC WRC PEC CSC LWS 244 244 * 30 - DR 245 245 * 31 - WPR … … 546 546 } xhci_sp_name_t; 547 547 548 static const xhci_sp_name_t xhci_name_usb = { 549 .str = "USB " 550 }; 551 548 552 static inline xhci_extcap_t *xhci_extcap_next(const xhci_extcap_t *cur) 549 553 { -
uspace/drv/bus/usb/xhci/rh.c
r5c5c9407 r816335c 51 51 static usbvirt_device_ops_t ops; 52 52 53 int xhci_rh_init(xhci_rh_t *rh, xhci_ op_regs_t *op_regs)53 int xhci_rh_init(xhci_rh_t *rh, xhci_hc_t *hc) 54 54 { 55 55 assert(rh); 56 rh->op_regs = op_regs; 56 assert(hc); 57 58 rh->hc = hc; 57 59 58 60 usb_hub_descriptor_header_t *header = &rh->hub_descriptor.header; … … 172 174 { 173 175 uint8_t link_state = XHCI_REG_RD(regs, XHCI_PORT_PLS); 174 // FIXME: do we have a better way to detect if this is usb2 or usb3 device? 175 if (xhci_is_usb3_port(&hc->rh, port_id)) { 176 const xhci_port_speed_t *speed = xhci_get_port_speed(&hc->rh, port_id); 177 178 usb_log_info("Detected new %.4s%u.%u device on port %u.", speed->name, speed->major, speed->minor, port_id); 179 180 if (speed->major == 3) { 176 181 if(link_state == 0) { 177 182 /* USB3 is automatically advanced to enabled. */ 178 uint8_t port_speed = XHCI_REG_RD(regs, XHCI_PORT_PS);179 usb_log_debug("Detected new device on port %u, port speed id %u.", port_id, port_speed);180 181 183 return alloc_dev(hc, port_id, 0); 182 184 } … … 237 239 238 240 return EOK; 241 } 242 243 const xhci_port_speed_t *xhci_get_port_speed(xhci_rh_t *rh, uint8_t port) 244 { 245 xhci_port_regs_t *port_regs = &rh->hc->op_regs->portrs[port - 1]; 246 247 unsigned psiv = XHCI_REG_RD(port_regs, XHCI_PORT_PS); 248 return &rh->speeds[psiv]; 239 249 } 240 250 … … 343 353 344 354 /* The index is 1-based. */ 345 xhci_port_regs_t* regs = &hub-> op_regs->portrs[setup_packet->index - 1];355 xhci_port_regs_t* regs = &hub->hc->op_regs->portrs[setup_packet->index - 1]; 346 356 347 357 const uint32_t status = uint32_host2usb( -
uspace/drv/bus/usb/xhci/rh.h
r5c5c9407 r816335c 39 39 #include <usb/host/usb_transfer_batch.h> 40 40 #include <usbvirt/virthub_base.h> 41 #include "hw_struct/regs.h" 42 43 typedef struct xhci_hc xhci_hc_t; 41 44 42 45 enum { … … 44 47 }; 45 48 49 /** 50 * xHCI lets the controller define speeds of ports it controls. 51 */ 52 typedef struct xhci_port_speed { 53 char name [4]; 54 uint8_t major, minor; 55 uint64_t rx_bps, tx_bps; 56 } xhci_port_speed_t; 57 46 58 /* XHCI root hub instance */ 47 59 typedef struct { 48 60 /** Virtual hub instance */ 49 61 virthub_base_t base; 50 /** XHCI operational registers */ 51 xhci_op_regs_t *op_regs; 62 63 /** Host controller */ 64 xhci_hc_t *hc; 65 66 /** Port speeds reported from HC */ 67 xhci_port_speed_t speeds [16]; 68 52 69 /** USB hub descriptor describing the XHCI root hub */ 53 70 struct { … … 57 74 /** Interrupt transfer waiting for an actual interrupt to occur */ 58 75 usb_transfer_batch_t *unfinished_interrupt_transfer; 59 60 uint8_t usb2_port_start;61 uint8_t usb2_port_end;62 uint8_t usb3_port_start;63 uint8_t usb3_port_end;64 76 } xhci_rh_t; 65 77 66 int xhci_rh_init(xhci_rh_t *, xhci_ op_regs_t *);78 int xhci_rh_init(xhci_rh_t *, xhci_hc_t *); 67 79 int xhci_rh_fini(xhci_rh_t *); 80 const xhci_port_speed_t *xhci_get_port_speed(xhci_rh_t *, uint8_t); 68 81 int xhci_handle_port_status_change_event(xhci_hc_t *, xhci_trb_t *); 69 82 int xhci_get_hub_port(xhci_trb_t *); … … 84 97 } 85 98 86 static inline bool xhci_is_usb3_port(xhci_rh_t* rh, uint8_t port) { 87 return port >= rh->usb3_port_start && port <= rh->usb3_port_end; 99 static inline bool xhci_is_usb3_port(xhci_rh_t* rh, uint8_t port) 100 { 101 return xhci_get_port_speed(rh, port)->major == 3; 88 102 } 89 90 103 #endif 91 104
Note:
See TracChangeset
for help on using the changeset viewer.