Lines Matching +full:usb +full:- +full:hub

1 // SPDX-License-Identifier: GPL-2.0+
3 * Driver for Microchip USB251xB USB 2.0 Hi-Speed Hub Controller
9 * a not-accepted patch by Fabien Lahoudere, see:
110 #define DRIVER_DESC "Microchip USB 2.0 Hi-Speed Hub Controller"
234 if (dev->type == &i2c_adapter_type) { in usb251xb_check_dev_children()
242 static int usb251x_check_gpio_chip(struct usb251xb *hub) in usb251x_check_gpio_chip() argument
244 struct gpio_chip *gc = gpiod_to_chip(hub->gpio_reset); in usb251x_check_gpio_chip()
245 struct i2c_adapter *adap = hub->i2c->adapter; in usb251x_check_gpio_chip()
248 if (!hub->gpio_reset) in usb251x_check_gpio_chip()
252 return -EINVAL; in usb251x_check_gpio_chip()
254 ret = usb251xb_check_dev_children(&adap->dev, gc->parent); in usb251x_check_gpio_chip()
256 dev_err(hub->dev, "Reset GPIO chip is at the same i2c-bus\n"); in usb251x_check_gpio_chip()
257 return -EINVAL; in usb251x_check_gpio_chip()
263 static int usb251x_check_gpio_chip(struct usb251xb *hub) in usb251x_check_gpio_chip() argument
269 static void usb251xb_reset(struct usb251xb *hub) in usb251xb_reset() argument
271 if (!hub->gpio_reset) in usb251xb_reset()
274 i2c_lock_bus(hub->i2c->adapter, I2C_LOCK_SEGMENT); in usb251xb_reset()
276 gpiod_set_value_cansleep(hub->gpio_reset, 1); in usb251xb_reset()
278 gpiod_set_value_cansleep(hub->gpio_reset, 0); in usb251xb_reset()
280 /* wait for hub recovery/stabilization */ in usb251xb_reset()
283 i2c_unlock_bus(hub->i2c->adapter, I2C_LOCK_SEGMENT); in usb251xb_reset()
286 static int usb251xb_connect(struct usb251xb *hub) in usb251xb_connect() argument
288 struct device *dev = hub->dev; in usb251xb_connect()
294 if (hub->skip_config) { in usb251xb_connect()
295 dev_info(dev, "Skip hub configuration, only attach.\n"); in usb251xb_connect()
299 usb251xb_reset(hub); in usb251xb_connect()
301 err = i2c_smbus_write_i2c_block_data(hub->i2c, in usb251xb_connect()
304 dev_err(dev, "attaching hub failed: %d\n", err); in usb251xb_connect()
310 i2c_wb[USB251XB_ADDR_VENDOR_ID_MSB] = (hub->vendor_id >> 8) & 0xFF; in usb251xb_connect()
311 i2c_wb[USB251XB_ADDR_VENDOR_ID_LSB] = hub->vendor_id & 0xFF; in usb251xb_connect()
312 i2c_wb[USB251XB_ADDR_PRODUCT_ID_MSB] = (hub->product_id >> 8) & 0xFF; in usb251xb_connect()
313 i2c_wb[USB251XB_ADDR_PRODUCT_ID_LSB] = hub->product_id & 0xFF; in usb251xb_connect()
314 i2c_wb[USB251XB_ADDR_DEVICE_ID_MSB] = (hub->device_id >> 8) & 0xFF; in usb251xb_connect()
315 i2c_wb[USB251XB_ADDR_DEVICE_ID_LSB] = hub->device_id & 0xFF; in usb251xb_connect()
316 i2c_wb[USB251XB_ADDR_CONFIG_DATA_1] = hub->conf_data1; in usb251xb_connect()
317 i2c_wb[USB251XB_ADDR_CONFIG_DATA_2] = hub->conf_data2; in usb251xb_connect()
318 i2c_wb[USB251XB_ADDR_CONFIG_DATA_3] = hub->conf_data3; in usb251xb_connect()
319 i2c_wb[USB251XB_ADDR_NON_REMOVABLE_DEVICES] = hub->non_rem_dev; in usb251xb_connect()
320 i2c_wb[USB251XB_ADDR_PORT_DISABLE_SELF] = hub->port_disable_sp; in usb251xb_connect()
321 i2c_wb[USB251XB_ADDR_PORT_DISABLE_BUS] = hub->port_disable_bp; in usb251xb_connect()
322 i2c_wb[USB251XB_ADDR_MAX_POWER_SELF] = hub->max_power_sp; in usb251xb_connect()
323 i2c_wb[USB251XB_ADDR_MAX_POWER_BUS] = hub->max_power_bp; in usb251xb_connect()
324 i2c_wb[USB251XB_ADDR_MAX_CURRENT_SELF] = hub->max_current_sp; in usb251xb_connect()
325 i2c_wb[USB251XB_ADDR_MAX_CURRENT_BUS] = hub->max_current_bp; in usb251xb_connect()
326 i2c_wb[USB251XB_ADDR_POWER_ON_TIME] = hub->power_on_time; in usb251xb_connect()
327 i2c_wb[USB251XB_ADDR_LANGUAGE_ID_HIGH] = (hub->lang_id >> 8) & 0xFF; in usb251xb_connect()
328 i2c_wb[USB251XB_ADDR_LANGUAGE_ID_LOW] = hub->lang_id & 0xFF; in usb251xb_connect()
329 i2c_wb[USB251XB_ADDR_MANUFACTURER_STRING_LEN] = hub->manufacturer_len; in usb251xb_connect()
330 i2c_wb[USB251XB_ADDR_PRODUCT_STRING_LEN] = hub->product_len; in usb251xb_connect()
331 i2c_wb[USB251XB_ADDR_SERIAL_STRING_LEN] = hub->serial_len; in usb251xb_connect()
332 memcpy(&i2c_wb[USB251XB_ADDR_MANUFACTURER_STRING], hub->manufacturer, in usb251xb_connect()
334 memcpy(&i2c_wb[USB251XB_ADDR_SERIAL_STRING], hub->serial, in usb251xb_connect()
336 memcpy(&i2c_wb[USB251XB_ADDR_PRODUCT_STRING], hub->product, in usb251xb_connect()
338 i2c_wb[USB251XB_ADDR_BATTERY_CHARGING_ENABLE] = hub->bat_charge_en; in usb251xb_connect()
339 i2c_wb[USB251XB_ADDR_BOOST_UP] = hub->boost_up; in usb251xb_connect()
340 i2c_wb[USB251XB_ADDR_BOOST_57] = hub->boost_57; in usb251xb_connect()
341 i2c_wb[USB251XB_ADDR_BOOST_14] = hub->boost_14; in usb251xb_connect()
342 i2c_wb[USB251XB_ADDR_PORT_SWAP] = hub->port_swap; in usb251xb_connect()
343 i2c_wb[USB251XB_ADDR_PORT_MAP_12] = hub->port_map12; in usb251xb_connect()
344 i2c_wb[USB251XB_ADDR_PORT_MAP_34] = hub->port_map34; in usb251xb_connect()
345 i2c_wb[USB251XB_ADDR_PORT_MAP_56] = hub->port_map56; in usb251xb_connect()
346 i2c_wb[USB251XB_ADDR_PORT_MAP_7] = hub->port_map7; in usb251xb_connect()
349 usb251xb_reset(hub); in usb251xb_connect()
356 /* The first data byte transferred tells the hub how many data in usb251xb_connect()
365 err = i2c_smbus_write_i2c_block_data(hub->i2c, offset, in usb251xb_connect()
372 dev_info(dev, "Hub configuration was successful.\n"); in usb251xb_connect()
381 static void usb251xb_get_ports_field(struct usb251xb *hub, in usb251xb_get_ports_field() argument
385 struct device *dev = hub->dev; in usb251xb_get_ports_field()
390 of_property_for_each_u32(dev->of_node, prop_name, prop, p, port) { in usb251xb_get_ports_field()
398 static int usb251xb_get_ofdata(struct usb251xb *hub, in usb251xb_get_ofdata() argument
401 struct device *dev = hub->dev; in usb251xb_get_ofdata()
402 struct device_node *np = dev->of_node; in usb251xb_get_ofdata()
410 return -ENODEV; in usb251xb_get_ofdata()
413 if (of_get_property(np, "skip-config", NULL)) in usb251xb_get_ofdata()
414 hub->skip_config = 1; in usb251xb_get_ofdata()
416 hub->skip_config = 0; in usb251xb_get_ofdata()
418 hub->gpio_reset = devm_gpiod_get_optional(dev, "reset", GPIOD_OUT_HIGH); in usb251xb_get_ofdata()
419 if (IS_ERR(hub->gpio_reset)) in usb251xb_get_ofdata()
420 return dev_err_probe(dev, PTR_ERR(hub->gpio_reset), in usb251xb_get_ofdata()
423 if (of_property_read_u16_array(np, "vendor-id", &hub->vendor_id, 1)) in usb251xb_get_ofdata()
424 hub->vendor_id = USB251XB_DEF_VENDOR_ID; in usb251xb_get_ofdata()
426 if (of_property_read_u16_array(np, "product-id", in usb251xb_get_ofdata()
427 &hub->product_id, 1)) in usb251xb_get_ofdata()
428 hub->product_id = data->product_id; in usb251xb_get_ofdata()
430 if (of_property_read_u16_array(np, "device-id", &hub->device_id, 1)) in usb251xb_get_ofdata()
431 hub->device_id = USB251XB_DEF_DEVICE_ID; in usb251xb_get_ofdata()
433 hub->conf_data1 = USB251XB_DEF_CONFIG_DATA_1; in usb251xb_get_ofdata()
434 if (of_get_property(np, "self-powered", NULL)) { in usb251xb_get_ofdata()
435 hub->conf_data1 |= BIT(7); in usb251xb_get_ofdata()
437 /* Configure Over-Current sens when self-powered */ in usb251xb_get_ofdata()
438 hub->conf_data1 &= ~BIT(2); in usb251xb_get_ofdata()
439 if (of_get_property(np, "ganged-sensing", NULL)) in usb251xb_get_ofdata()
440 hub->conf_data1 &= ~BIT(1); in usb251xb_get_ofdata()
441 else if (of_get_property(np, "individual-sensing", NULL)) in usb251xb_get_ofdata()
442 hub->conf_data1 |= BIT(1); in usb251xb_get_ofdata()
443 } else if (of_get_property(np, "bus-powered", NULL)) { in usb251xb_get_ofdata()
444 hub->conf_data1 &= ~BIT(7); in usb251xb_get_ofdata()
446 /* Disable Over-Current sense when bus-powered */ in usb251xb_get_ofdata()
447 hub->conf_data1 |= BIT(2); in usb251xb_get_ofdata()
450 if (of_get_property(np, "disable-hi-speed", NULL)) in usb251xb_get_ofdata()
451 hub->conf_data1 |= BIT(5); in usb251xb_get_ofdata()
453 if (of_get_property(np, "multi-tt", NULL)) in usb251xb_get_ofdata()
454 hub->conf_data1 |= BIT(4); in usb251xb_get_ofdata()
455 else if (of_get_property(np, "single-tt", NULL)) in usb251xb_get_ofdata()
456 hub->conf_data1 &= ~BIT(4); in usb251xb_get_ofdata()
458 if (of_get_property(np, "disable-eop", NULL)) in usb251xb_get_ofdata()
459 hub->conf_data1 |= BIT(3); in usb251xb_get_ofdata()
461 if (of_get_property(np, "individual-port-switching", NULL)) in usb251xb_get_ofdata()
462 hub->conf_data1 |= BIT(0); in usb251xb_get_ofdata()
463 else if (of_get_property(np, "ganged-port-switching", NULL)) in usb251xb_get_ofdata()
464 hub->conf_data1 &= ~BIT(0); in usb251xb_get_ofdata()
466 hub->conf_data2 = USB251XB_DEF_CONFIG_DATA_2; in usb251xb_get_ofdata()
467 if (of_get_property(np, "dynamic-power-switching", NULL)) in usb251xb_get_ofdata()
468 hub->conf_data2 |= BIT(7); in usb251xb_get_ofdata()
470 if (!of_property_read_u32(np, "oc-delay-us", &property_u32)) { in usb251xb_get_ofdata()
473 hub->conf_data2 &= ~BIT(5); in usb251xb_get_ofdata()
474 hub->conf_data2 &= ~BIT(4); in usb251xb_get_ofdata()
477 hub->conf_data2 &= ~BIT(5); in usb251xb_get_ofdata()
478 hub->conf_data2 |= BIT(4); in usb251xb_get_ofdata()
481 hub->conf_data2 |= BIT(5); in usb251xb_get_ofdata()
482 hub->conf_data2 |= BIT(4); in usb251xb_get_ofdata()
485 hub->conf_data2 |= BIT(5); in usb251xb_get_ofdata()
486 hub->conf_data2 &= ~BIT(4); in usb251xb_get_ofdata()
490 if (of_get_property(np, "compound-device", NULL)) in usb251xb_get_ofdata()
491 hub->conf_data2 |= BIT(3); in usb251xb_get_ofdata()
493 hub->conf_data3 = USB251XB_DEF_CONFIG_DATA_3; in usb251xb_get_ofdata()
494 if (of_get_property(np, "port-mapping-mode", NULL)) in usb251xb_get_ofdata()
495 hub->conf_data3 |= BIT(3); in usb251xb_get_ofdata()
497 if (data->led_support && of_get_property(np, "led-usb-mode", NULL)) in usb251xb_get_ofdata()
498 hub->conf_data3 &= ~BIT(1); in usb251xb_get_ofdata()
500 if (of_get_property(np, "string-support", NULL)) in usb251xb_get_ofdata()
501 hub->conf_data3 |= BIT(0); in usb251xb_get_ofdata()
503 hub->non_rem_dev = USB251XB_DEF_NON_REMOVABLE_DEVICES; in usb251xb_get_ofdata()
504 usb251xb_get_ports_field(hub, "non-removable-ports", data->port_cnt, in usb251xb_get_ofdata()
505 true, &hub->non_rem_dev); in usb251xb_get_ofdata()
507 hub->port_disable_sp = USB251XB_DEF_PORT_DISABLE_SELF; in usb251xb_get_ofdata()
508 usb251xb_get_ports_field(hub, "sp-disabled-ports", data->port_cnt, in usb251xb_get_ofdata()
509 true, &hub->port_disable_sp); in usb251xb_get_ofdata()
511 hub->port_disable_bp = USB251XB_DEF_PORT_DISABLE_BUS; in usb251xb_get_ofdata()
512 usb251xb_get_ports_field(hub, "bp-disabled-ports", data->port_cnt, in usb251xb_get_ofdata()
513 true, &hub->port_disable_bp); in usb251xb_get_ofdata()
515 hub->max_power_sp = USB251XB_DEF_MAX_POWER_SELF; in usb251xb_get_ofdata()
516 if (!of_property_read_u32(np, "sp-max-total-current-microamp", in usb251xb_get_ofdata()
518 hub->max_power_sp = min_t(u8, property_u32 / 2000, 50); in usb251xb_get_ofdata()
520 hub->max_power_bp = USB251XB_DEF_MAX_POWER_BUS; in usb251xb_get_ofdata()
521 if (!of_property_read_u32(np, "bp-max-total-current-microamp", in usb251xb_get_ofdata()
523 hub->max_power_bp = min_t(u8, property_u32 / 2000, 255); in usb251xb_get_ofdata()
525 hub->max_current_sp = USB251XB_DEF_MAX_CURRENT_SELF; in usb251xb_get_ofdata()
526 if (!of_property_read_u32(np, "sp-max-removable-current-microamp", in usb251xb_get_ofdata()
528 hub->max_current_sp = min_t(u8, property_u32 / 2000, 50); in usb251xb_get_ofdata()
530 hub->max_current_bp = USB251XB_DEF_MAX_CURRENT_BUS; in usb251xb_get_ofdata()
531 if (!of_property_read_u32(np, "bp-max-removable-current-microamp", in usb251xb_get_ofdata()
533 hub->max_current_bp = min_t(u8, property_u32 / 2000, 255); in usb251xb_get_ofdata()
535 hub->power_on_time = USB251XB_DEF_POWER_ON_TIME; in usb251xb_get_ofdata()
536 if (!of_property_read_u32(np, "power-on-time-ms", &property_u32)) in usb251xb_get_ofdata()
537 hub->power_on_time = min_t(u8, property_u32 / 2, 255); in usb251xb_get_ofdata()
539 if (of_property_read_u16_array(np, "language-id", &hub->lang_id, 1)) in usb251xb_get_ofdata()
540 hub->lang_id = USB251XB_DEF_LANGUAGE_ID; in usb251xb_get_ofdata()
542 if (of_property_read_u8(np, "boost-up", &hub->boost_up)) in usb251xb_get_ofdata()
543 hub->boost_up = USB251XB_DEF_BOOST_UP; in usb251xb_get_ofdata()
548 hub->manufacturer_len = strlen(str) & 0xFF; in usb251xb_get_ofdata()
549 memset(hub->manufacturer, 0, USB251XB_STRING_BUFSIZE); in usb251xb_get_ofdata()
552 (wchar_t *)hub->manufacturer, in usb251xb_get_ofdata()
556 strscpy(str, cproperty_char ? : data->product_str, sizeof(str)); in usb251xb_get_ofdata()
557 hub->product_len = strlen(str) & 0xFF; in usb251xb_get_ofdata()
558 memset(hub->product, 0, USB251XB_STRING_BUFSIZE); in usb251xb_get_ofdata()
561 (wchar_t *)hub->product, in usb251xb_get_ofdata()
567 hub->serial_len = strlen(str) & 0xFF; in usb251xb_get_ofdata()
568 memset(hub->serial, 0, USB251XB_STRING_BUFSIZE); in usb251xb_get_ofdata()
571 (wchar_t *)hub->serial, in usb251xb_get_ofdata()
576 * register controls the USB DP/DM signal swapping for each port. in usb251xb_get_ofdata()
578 hub->port_swap = USB251XB_DEF_PORT_SWAP; in usb251xb_get_ofdata()
579 usb251xb_get_ports_field(hub, "swap-dx-lanes", data->port_cnt, in usb251xb_get_ofdata()
580 false, &hub->port_swap); in usb251xb_get_ofdata()
585 hub->bat_charge_en = USB251XB_DEF_BATTERY_CHARGING_ENABLE; in usb251xb_get_ofdata()
586 hub->boost_57 = USB251XB_DEF_BOOST_57; in usb251xb_get_ofdata()
587 hub->boost_14 = USB251XB_DEF_BOOST_14; in usb251xb_get_ofdata()
588 hub->port_map12 = USB251XB_DEF_PORT_MAP_12; in usb251xb_get_ofdata()
589 hub->port_map34 = USB251XB_DEF_PORT_MAP_34; in usb251xb_get_ofdata()
590 hub->port_map56 = USB251XB_DEF_PORT_MAP_56; in usb251xb_get_ofdata()
591 hub->port_map7 = USB251XB_DEF_PORT_MAP_7; in usb251xb_get_ofdata()
630 static int usb251xb_get_ofdata(struct usb251xb *hub, in usb251xb_get_ofdata() argument
639 struct usb251xb *hub = data; in usb251xb_regulator_disable_action() local
641 regulator_disable(hub->vdd); in usb251xb_regulator_disable_action()
644 static int usb251xb_probe(struct usb251xb *hub) in usb251xb_probe() argument
646 struct device *dev = hub->dev; in usb251xb_probe()
647 struct device_node *np = dev->of_node; in usb251xb_probe()
652 err = usb251xb_get_ofdata(hub, usb_data); in usb251xb_probe()
660 * usb251x SMBus-slave SCL lane is muxed with CFG_SEL0 pin. So if anyone in usb251xb_probe()
661 * tries to work with the bus at the moment the hub reset is released, in usb251xb_probe()
663 * one of the config modes makes the hub loading a default registers in usb251xb_probe()
664 * value without SMBus-slave interface activation. If the hub in usb251xb_probe()
665 * accidentally gets this mode, this will cause the driver SMBus- in usb251xb_probe()
666 * functions failure. Normally we could just lock the SMBus-segment the in usb251xb_probe()
667 * hub i2c-interface resides for the device-specific reset timing. But in usb251xb_probe()
668 * the GPIO controller, which is used to handle the hub reset, might be in usb251xb_probe()
669 * placed at the same i2c-bus segment. In this case an error should be in usb251xb_probe()
671 * reset state (it may affect the hub configuration) and we can't lock in usb251xb_probe()
672 * the i2c-bus segment (it will cause a deadlock). in usb251xb_probe()
674 err = usb251x_check_gpio_chip(hub); in usb251xb_probe()
678 hub->vdd = devm_regulator_get(dev, "vdd"); in usb251xb_probe()
679 if (IS_ERR(hub->vdd)) in usb251xb_probe()
680 return PTR_ERR(hub->vdd); in usb251xb_probe()
682 err = regulator_enable(hub->vdd); in usb251xb_probe()
687 usb251xb_regulator_disable_action, hub); in usb251xb_probe()
691 err = usb251xb_connect(hub); in usb251xb_probe()
693 dev_err(dev, "Failed to connect hub (%d)\n", err); in usb251xb_probe()
697 dev_info(dev, "Hub probed successfully\n"); in usb251xb_probe()
705 struct usb251xb *hub; in usb251xb_i2c_probe() local
707 hub = devm_kzalloc(&i2c->dev, sizeof(struct usb251xb), GFP_KERNEL); in usb251xb_i2c_probe()
708 if (!hub) in usb251xb_i2c_probe()
709 return -ENOMEM; in usb251xb_i2c_probe()
711 i2c_set_clientdata(i2c, hub); in usb251xb_i2c_probe()
712 hub->dev = &i2c->dev; in usb251xb_i2c_probe()
713 hub->i2c = i2c; in usb251xb_i2c_probe()
715 return usb251xb_probe(hub); in usb251xb_i2c_probe()
721 struct usb251xb *hub = i2c_get_clientdata(client); in usb251xb_suspend() local
723 return regulator_disable(hub->vdd); in usb251xb_suspend()
729 struct usb251xb *hub = i2c_get_clientdata(client); in usb251xb_resume() local
732 err = regulator_enable(hub->vdd); in usb251xb_resume()
736 return usb251xb_connect(hub); in usb251xb_resume()
768 MODULE_DESCRIPTION("USB251x/xBi USB 2.0 Hub Controller Driver");