Changes in uspace/drv/bus/usb/usbhub/port.c [e231d26:cae002c] in mainline
- File:
-
- 1 edited
Legend:
- Unmodified
- Added
- Removed
-
uspace/drv/bus/usb/usbhub/port.c
re231d26 rcae002c 50 50 /** Information for fibril for device discovery. */ 51 51 struct add_device_phase1 { 52 usb_hub_ info_t *hub;52 usb_hub_dev_t *hub; 53 53 usb_hub_port_t *port; 54 54 usb_speed_t speed; 55 55 }; 56 56 57 static void usb_hub_port_removed_device(usb_hub_port_t *port, 58 usb_hub_info_t *hub); 57 static int usb_hub_port_device_gone(usb_hub_port_t *port, usb_hub_dev_t *hub); 59 58 static void usb_hub_port_reset_completed(usb_hub_port_t *port, 60 59 usb_port_status_t status); 61 60 static int get_port_status(usb_hub_port_t *port, usb_port_status_t *status); 62 static int enable_port_callback( int port_no,void *arg);61 static int enable_port_callback(void *arg); 63 62 static int add_device_phase1_worker_fibril(void *arg); 64 static int create_add_device_fibril(usb_hub_port_t *port, usb_hub_ info_t *hub,63 static int create_add_device_fibril(usb_hub_port_t *port, usb_hub_dev_t *hub, 65 64 usb_speed_t speed); 66 65 66 int usb_hub_port_fini(usb_hub_port_t *port, usb_hub_dev_t *hub) 67 { 68 assert(port); 69 if (port->attached_device.fun) 70 return usb_hub_port_device_gone(port, hub); 71 return EOK; 72 } 73 /*----------------------------------------------------------------------------*/ 67 74 /** 68 75 * Clear feature on hub port. 69 76 * 70 * @param hc Host controller telephone 71 * @param address Hub address 72 * @param port_index Port 73 * @param feature Feature selector 77 * @param port Port structure. 78 * @param feature Feature selector. 74 79 * @return Operation result 75 80 */ … … 78 83 { 79 84 assert(port); 80 usb_device_request_setup_packet_t clear_request = {85 const usb_device_request_setup_packet_t clear_request = { 81 86 .request_type = USB_HUB_REQ_TYPE_CLEAR_PORT_FEATURE, 82 87 .request = USB_DEVREQ_CLEAR_FEATURE, … … 90 95 /*----------------------------------------------------------------------------*/ 91 96 /** 92 * Clear feature on hub port. 93 * 94 * @param hc Host controller telephone 95 * @param address Hub address 96 * @param port_index Port 97 * @param feature Feature selector 97 * Set feature on hub port. 98 * 99 * @param port Port structure. 100 * @param feature Feature selector. 98 101 * @return Operation result 99 102 */ … … 102 105 { 103 106 assert(port); 104 usb_device_request_setup_packet_t clear_request = {107 const usb_device_request_setup_packet_t clear_request = { 105 108 .request_type = USB_HUB_REQ_TYPE_SET_PORT_FEATURE, 106 109 .request = USB_DEVREQ_SET_FEATURE, … … 113 116 } 114 117 /*----------------------------------------------------------------------------*/ 118 /** 119 * Mark reset process as failed due to external reasons 120 * 121 * @param port Port structure 122 */ 115 123 void usb_hub_port_reset_fail(usb_hub_port_t *port) 116 124 { … … 124 132 /*----------------------------------------------------------------------------*/ 125 133 /** 126 * Process interrupts on given hubport134 * Process interrupts on given port 127 135 * 128 136 * Accepts connection, over current and port reset change. 137 * @param port port structure 129 138 * @param hub hub representation 130 * @param port port number, starting from 1 131 */ 132 void usb_hub_port_process_interrupt(usb_hub_port_t *port, usb_hub_info_t *hub) 139 */ 140 void usb_hub_port_process_interrupt(usb_hub_port_t *port, usb_hub_dev_t *hub) 133 141 { 134 142 assert(port); … … 171 179 * to that handler, it shall ACK the change too. */ 172 180 if (!(status & USB_HUB_PORT_C_STATUS_ENABLED)) { 173 usb_hub_port_ removed_device(port, hub);181 usb_hub_port_device_gone(port, hub); 174 182 } 175 183 } … … 180 188 usb_log_info("Port %zu, disabled because of errors.\n", 181 189 port->port_number); 182 usb_hub_port_ removed_device(port, hub);190 usb_hub_port_device_gone(port, hub); 183 191 const int rc = usb_hub_port_clear_feature(port, 184 192 USB_HUB_FEATURE_C_PORT_ENABLE); … … 238 246 port->port_number, status); 239 247 } 240 248 /*----------------------------------------------------------------------------*/ 241 249 /** 242 250 * routine called when a device on port has been removed … … 245 253 * Otherwise does not do anything, because DDF does not allow to remove device 246 254 * from it`s device tree. 255 * @param port port structure 247 256 * @param hub hub representation 248 * @param port port number, starting from 1 249 */ 250 static void usb_hub_port_removed_device(usb_hub_port_t *port, 251 usb_hub_info_t *hub) 257 */ 258 int usb_hub_port_device_gone(usb_hub_port_t *port, usb_hub_dev_t *hub) 252 259 { 253 260 assert(port); 254 261 assert(hub); 255 if (port->attached_device.address >= 0) { 256 fibril_mutex_lock(&port->mutex); 257 port->attached_device.address = -1; 258 port->attached_device.handle = 0; 259 fibril_mutex_unlock(&port->mutex); 260 usb_log_info("Removed device on port %zu.\n", 261 port->port_number); 262 } else { 262 if (port->attached_device.address < 0) { 263 263 usb_log_warning( 264 264 "Device on port %zu removed before being registered.\n", … … 271 271 */ 272 272 usb_hub_port_reset_fail(port); 273 } 274 } 275 273 return EOK; 274 } 275 276 fibril_mutex_lock(&port->mutex); 277 assert(port->attached_device.fun); 278 usb_log_debug("Removing device on port %zu.\n", port->port_number); 279 int ret = ddf_fun_unbind(port->attached_device.fun); 280 if (ret != EOK) { 281 usb_log_error("Failed to unbind child function on port" 282 " %zu: %s.\n", port->port_number, str_error(ret)); 283 fibril_mutex_unlock(&port->mutex); 284 return ret; 285 } 286 287 ddf_fun_destroy(port->attached_device.fun); 288 port->attached_device.fun = NULL; 289 290 ret = usb_hc_connection_open(&hub->connection); 291 if (ret == EOK) { 292 ret = usb_hc_unregister_device(&hub->connection, 293 port->attached_device.address); 294 if (ret != EOK) { 295 usb_log_warning("Failed to unregister address of the " 296 "removed device: %s.\n", str_error(ret)); 297 } 298 ret = usb_hc_connection_close(&hub->connection); 299 if (ret != EOK) { 300 usb_log_warning("Failed to close hc connection %s.\n", 301 str_error(ret)); 302 } 303 304 } else { 305 usb_log_warning("Failed to open hc connection %s.\n", 306 str_error(ret)); 307 } 308 309 port->attached_device.address = -1; 310 fibril_mutex_unlock(&port->mutex); 311 usb_log_info("Removed device on port %zu.\n", port->port_number); 312 return EOK; 313 } 314 /*----------------------------------------------------------------------------*/ 276 315 /** 277 316 * Process port reset change … … 279 318 * After this change port should be enabled, unless some problem occurred. 280 319 * This functions triggers second phase of enabling new device. 281 * @param hub 282 * @param port 283 * @param status 284 */ 285 static void usb_hub_port_reset_completed(usb_hub_port_t *port, 320 * @param port Port structure 321 * @param status Port status mask 322 */ 323 void usb_hub_port_reset_completed(usb_hub_port_t *port, 286 324 usb_port_status_t status) 287 325 { … … 313 351 /** Retrieve port status. 314 352 * 315 * @param[in] ctrl_pipe Control pipe to use. 316 * @param[in] port Port number (starting at 1). 353 * @param[in] port Port structure 317 354 * @param[out] status Where to store the port status. 318 355 * @return Error code. … … 358 395 * 359 396 * @param port_no Port number (starting at 1). 360 * @param arg Custom argument, points to @c usb_hub_ info_t.397 * @param arg Custom argument, points to @c usb_hub_dev_t. 361 398 * @return Error code. 362 399 */ 363 static int enable_port_callback( int port_no,void *arg)400 static int enable_port_callback(void *arg) 364 401 { 365 402 usb_hub_port_t *port = arg; 403 assert(port); 366 404 const int rc = 367 405 usb_hub_port_set_feature(port, USB_HUB_FEATURE_PORT_RESET); … … 380 418 fibril_mutex_unlock(&port->mutex); 381 419 382 if (port->reset_okay) { 383 return EOK; 384 } else { 385 return ESTALL; 386 } 387 } 388 420 return port->reset_okay ? EOK : ESTALL; 421 } 422 /*----------------------------------------------------------------------------*/ 389 423 /** Fibril for adding a new device. 390 424 * … … 395 429 * @return 0 Always. 396 430 */ 397 staticint add_device_phase1_worker_fibril(void *arg)431 int add_device_phase1_worker_fibril(void *arg) 398 432 { 399 433 struct add_device_phase1 *data = arg; … … 401 435 402 436 usb_address_t new_address; 403 d evman_handle_t child_handle;437 ddf_fun_t *child_fun; 404 438 405 439 const int rc = usb_hc_new_device_wrapper(data->hub->usb_device->ddf_dev, 406 &data->hub->connection, data->speed, 407 enable_port_callback, (int) data->port->port_number, 408 data->port, &new_address, &child_handle, 409 NULL, NULL, NULL); 410 411 if (rc != EOK) { 440 &data->hub->connection, data->speed, enable_port_callback, 441 data->port, &new_address, NULL, NULL, &child_fun); 442 443 if (rc == EOK) { 444 fibril_mutex_lock(&data->port->mutex); 445 data->port->attached_device.fun = child_fun; 446 data->port->attached_device.address = new_address; 447 fibril_mutex_unlock(&data->port->mutex); 448 449 usb_log_info("Detected new device on `%s' (port %zu), " 450 "address %d (handle %" PRIun ").\n", 451 data->hub->usb_device->ddf_dev->name, 452 data->port->port_number, new_address, child_fun->handle); 453 } else { 412 454 usb_log_error("Failed registering device on port %zu: %s.\n", 413 455 data->port->port_number, str_error(rc)); 414 goto leave; 415 } 416 417 fibril_mutex_lock(&data->port->mutex); 418 data->port->attached_device.handle = child_handle; 419 data->port->attached_device.address = new_address; 420 fibril_mutex_unlock(&data->port->mutex); 421 422 usb_log_info("Detected new device on `%s' (port %zu), " 423 "address %d (handle %" PRIun ").\n", 424 data->hub->usb_device->ddf_dev->name, data->port->port_number, 425 new_address, child_handle); 426 427 leave: 456 } 457 458 428 459 fibril_mutex_lock(&data->hub->pending_ops_mutex); 429 460 assert(data->hub->pending_ops_count > 0); … … 434 465 free(arg); 435 466 436 return EOK;437 } 438 467 return rc; 468 } 469 /*----------------------------------------------------------------------------*/ 439 470 /** Start device adding when connection change is detected. 440 471 * … … 446 477 * @return Error code. 447 478 */ 448 static int create_add_device_fibril(usb_hub_port_t *port, usb_hub_ info_t *hub,479 static int create_add_device_fibril(usb_hub_port_t *port, usb_hub_dev_t *hub, 449 480 usb_speed_t speed) 450 481 { … … 452 483 assert(port); 453 484 struct add_device_phase1 *data 454 = malloc(sizeof 485 = malloc(sizeof(struct add_device_phase1)); 455 486 if (data == NULL) { 456 487 return ENOMEM;
Note:
See TracChangeset
for help on using the changeset viewer.