123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320 |
- // SPDX-License-Identifier: GPL-2.0
- /*
- * USB4 port device
- *
- * Copyright (C) 2021, Intel Corporation
- * Author: Mika Westerberg <[email protected]>
- */
- #include <linux/pm_runtime.h>
- #include <linux/component.h>
- #include <linux/property.h>
- #include "tb.h"
- static int connector_bind(struct device *dev, struct device *connector, void *data)
- {
- int ret;
- ret = sysfs_create_link(&dev->kobj, &connector->kobj, "connector");
- if (ret)
- return ret;
- ret = sysfs_create_link(&connector->kobj, &dev->kobj, dev_name(dev));
- if (ret)
- sysfs_remove_link(&dev->kobj, "connector");
- return ret;
- }
- static void connector_unbind(struct device *dev, struct device *connector, void *data)
- {
- sysfs_remove_link(&connector->kobj, dev_name(dev));
- sysfs_remove_link(&dev->kobj, "connector");
- }
- static const struct component_ops connector_ops = {
- .bind = connector_bind,
- .unbind = connector_unbind,
- };
- static ssize_t link_show(struct device *dev, struct device_attribute *attr,
- char *buf)
- {
- struct usb4_port *usb4 = tb_to_usb4_port_device(dev);
- struct tb_port *port = usb4->port;
- struct tb *tb = port->sw->tb;
- const char *link;
- if (mutex_lock_interruptible(&tb->lock))
- return -ERESTARTSYS;
- if (tb_is_upstream_port(port))
- link = port->sw->link_usb4 ? "usb4" : "tbt";
- else if (tb_port_has_remote(port))
- link = port->remote->sw->link_usb4 ? "usb4" : "tbt";
- else if (port->xdomain)
- link = port->xdomain->link_usb4 ? "usb4" : "tbt";
- else
- link = "none";
- mutex_unlock(&tb->lock);
- return sysfs_emit(buf, "%s\n", link);
- }
- static DEVICE_ATTR_RO(link);
- static struct attribute *common_attrs[] = {
- &dev_attr_link.attr,
- NULL
- };
- static const struct attribute_group common_group = {
- .attrs = common_attrs,
- };
- static int usb4_port_offline(struct usb4_port *usb4)
- {
- struct tb_port *port = usb4->port;
- int ret;
- ret = tb_acpi_power_on_retimers(port);
- if (ret)
- return ret;
- ret = usb4_port_router_offline(port);
- if (ret) {
- tb_acpi_power_off_retimers(port);
- return ret;
- }
- ret = tb_retimer_scan(port, false);
- if (ret) {
- usb4_port_router_online(port);
- tb_acpi_power_off_retimers(port);
- }
- return ret;
- }
- static void usb4_port_online(struct usb4_port *usb4)
- {
- struct tb_port *port = usb4->port;
- usb4_port_router_online(port);
- tb_acpi_power_off_retimers(port);
- }
- static ssize_t offline_show(struct device *dev,
- struct device_attribute *attr, char *buf)
- {
- struct usb4_port *usb4 = tb_to_usb4_port_device(dev);
- return sysfs_emit(buf, "%d\n", usb4->offline);
- }
- static ssize_t offline_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t count)
- {
- struct usb4_port *usb4 = tb_to_usb4_port_device(dev);
- struct tb_port *port = usb4->port;
- struct tb *tb = port->sw->tb;
- bool val;
- int ret;
- ret = kstrtobool(buf, &val);
- if (ret)
- return ret;
- pm_runtime_get_sync(&usb4->dev);
- if (mutex_lock_interruptible(&tb->lock)) {
- ret = -ERESTARTSYS;
- goto out_rpm;
- }
- if (val == usb4->offline)
- goto out_unlock;
- /* Offline mode works only for ports that are not connected */
- if (tb_port_has_remote(port)) {
- ret = -EBUSY;
- goto out_unlock;
- }
- if (val) {
- ret = usb4_port_offline(usb4);
- if (ret)
- goto out_unlock;
- } else {
- usb4_port_online(usb4);
- tb_retimer_remove_all(port);
- }
- usb4->offline = val;
- tb_port_dbg(port, "%s offline mode\n", val ? "enter" : "exit");
- out_unlock:
- mutex_unlock(&tb->lock);
- out_rpm:
- pm_runtime_mark_last_busy(&usb4->dev);
- pm_runtime_put_autosuspend(&usb4->dev);
- return ret ? ret : count;
- }
- static DEVICE_ATTR_RW(offline);
- static ssize_t rescan_store(struct device *dev,
- struct device_attribute *attr, const char *buf, size_t count)
- {
- struct usb4_port *usb4 = tb_to_usb4_port_device(dev);
- struct tb_port *port = usb4->port;
- struct tb *tb = port->sw->tb;
- bool val;
- int ret;
- ret = kstrtobool(buf, &val);
- if (ret)
- return ret;
- if (!val)
- return count;
- pm_runtime_get_sync(&usb4->dev);
- if (mutex_lock_interruptible(&tb->lock)) {
- ret = -ERESTARTSYS;
- goto out_rpm;
- }
- /* Must be in offline mode already */
- if (!usb4->offline) {
- ret = -EINVAL;
- goto out_unlock;
- }
- tb_retimer_remove_all(port);
- ret = tb_retimer_scan(port, true);
- out_unlock:
- mutex_unlock(&tb->lock);
- out_rpm:
- pm_runtime_mark_last_busy(&usb4->dev);
- pm_runtime_put_autosuspend(&usb4->dev);
- return ret ? ret : count;
- }
- static DEVICE_ATTR_WO(rescan);
- static struct attribute *service_attrs[] = {
- &dev_attr_offline.attr,
- &dev_attr_rescan.attr,
- NULL
- };
- static umode_t service_attr_is_visible(struct kobject *kobj,
- struct attribute *attr, int n)
- {
- struct device *dev = kobj_to_dev(kobj);
- struct usb4_port *usb4 = tb_to_usb4_port_device(dev);
- /*
- * Always need some platform help to cycle the modes so that
- * retimers can be accessed through the sideband.
- */
- return usb4->can_offline ? attr->mode : 0;
- }
- static const struct attribute_group service_group = {
- .attrs = service_attrs,
- .is_visible = service_attr_is_visible,
- };
- static const struct attribute_group *usb4_port_device_groups[] = {
- &common_group,
- &service_group,
- NULL
- };
- static void usb4_port_device_release(struct device *dev)
- {
- struct usb4_port *usb4 = container_of(dev, struct usb4_port, dev);
- kfree(usb4);
- }
- struct device_type usb4_port_device_type = {
- .name = "usb4_port",
- .groups = usb4_port_device_groups,
- .release = usb4_port_device_release,
- };
- /**
- * usb4_port_device_add() - Add USB4 port device
- * @port: Lane 0 adapter port to add the USB4 port
- *
- * Creates and registers a USB4 port device for @port. Returns the new
- * USB4 port device pointer or ERR_PTR() in case of error.
- */
- struct usb4_port *usb4_port_device_add(struct tb_port *port)
- {
- struct usb4_port *usb4;
- int ret;
- usb4 = kzalloc(sizeof(*usb4), GFP_KERNEL);
- if (!usb4)
- return ERR_PTR(-ENOMEM);
- usb4->port = port;
- usb4->dev.type = &usb4_port_device_type;
- usb4->dev.parent = &port->sw->dev;
- dev_set_name(&usb4->dev, "usb4_port%d", port->port);
- ret = device_register(&usb4->dev);
- if (ret) {
- put_device(&usb4->dev);
- return ERR_PTR(ret);
- }
- if (dev_fwnode(&usb4->dev)) {
- ret = component_add(&usb4->dev, &connector_ops);
- if (ret) {
- dev_err(&usb4->dev, "failed to add component\n");
- device_unregister(&usb4->dev);
- }
- }
- pm_runtime_no_callbacks(&usb4->dev);
- pm_runtime_set_active(&usb4->dev);
- pm_runtime_enable(&usb4->dev);
- pm_runtime_set_autosuspend_delay(&usb4->dev, TB_AUTOSUSPEND_DELAY);
- pm_runtime_mark_last_busy(&usb4->dev);
- pm_runtime_use_autosuspend(&usb4->dev);
- return usb4;
- }
- /**
- * usb4_port_device_remove() - Removes USB4 port device
- * @usb4: USB4 port device
- *
- * Unregisters the USB4 port device from the system. The device will be
- * released when the last reference is dropped.
- */
- void usb4_port_device_remove(struct usb4_port *usb4)
- {
- if (dev_fwnode(&usb4->dev))
- component_del(&usb4->dev, &connector_ops);
- device_unregister(&usb4->dev);
- }
- /**
- * usb4_port_device_resume() - Resumes USB4 port device
- * @usb4: USB4 port device
- *
- * Used to resume USB4 port device after sleep state.
- */
- int usb4_port_device_resume(struct usb4_port *usb4)
- {
- return usb4->offline ? usb4_port_offline(usb4) : 0;
- }
|