.. _typec:
|
|
USB Type-C connector class
|
==========================
|
|
Introduction
|
------------
|
|
The typec class is meant for describing the USB Type-C ports in a system to the
|
user space in unified fashion. The class is designed to provide nothing else
|
except the user space interface implementation in hope that it can be utilized
|
on as many platforms as possible.
|
|
The platforms are expected to register every USB Type-C port they have with the
|
class. In a normal case the registration will be done by a USB Type-C or PD PHY
|
driver, but it may be a driver for firmware interface such as UCSI, driver for
|
USB PD controller or even driver for Thunderbolt3 controller. This document
|
considers the component registering the USB Type-C ports with the class as "port
|
driver".
|
|
On top of showing the capabilities, the class also offer user space control over
|
the roles and alternate modes of ports, partners and cable plugs when the port
|
driver is capable of supporting those features.
|
|
The class provides an API for the port drivers described in this document. The
|
attributes are described in Documentation/ABI/testing/sysfs-class-typec.
|
|
User space interface
|
--------------------
|
Every port will be presented as its own device under /sys/class/typec/. The
|
first port will be named "port0", the second "port1" and so on.
|
|
When connected, the partner will be presented also as its own device under
|
/sys/class/typec/. The parent of the partner device will always be the port it
|
is attached to. The partner attached to port "port0" will be named
|
"port0-partner". Full path to the device would be
|
/sys/class/typec/port0/port0-partner/.
|
|
The cable and the two plugs on it may also be optionally presented as their own
|
devices under /sys/class/typec/. The cable attached to the port "port0" port
|
will be named port0-cable and the plug on the SOP Prime end (see USB Power
|
Delivery Specification ch. 2.4) will be named "port0-plug0" and on the SOP
|
Double Prime end "port0-plug1". The parent of a cable will always be the port,
|
and the parent of the cable plugs will always be the cable.
|
|
If the port, partner or cable plug supports Alternate Modes, every supported
|
Alternate Mode SVID will have their own device describing them. Note that the
|
Alternate Mode devices will not be attached to the typec class. The parent of an
|
alternate mode will be the device that supports it, so for example an alternate
|
mode of port0-partner will be presented under /sys/class/typec/port0-partner/.
|
Every mode that is supported will have its own group under the Alternate Mode
|
device named "mode<index>", for example /sys/class/typec/port0/<alternate
|
mode>/mode1/. The requests for entering/exiting a mode can be done with "active"
|
attribute file in that group.
|
|
Driver API
|
----------
|
|
Registering the ports
|
~~~~~~~~~~~~~~~~~~~~~
|
|
The port drivers will describe every Type-C port they control with struct
|
typec_capability data structure, and register them with the following API:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_register_port typec_unregister_port
|
|
When registering the ports, the prefer_role member in struct typec_capability
|
deserves special notice. If the port that is being registered does not have
|
initial role preference, which means the port does not execute Try.SNK or
|
Try.SRC by default, the member must have value TYPEC_NO_PREFERRED_ROLE.
|
Otherwise if the port executes Try.SNK by default, the member must have value
|
TYPEC_DEVICE, and with Try.SRC the value must be TYPEC_HOST.
|
|
Registering Partners
|
~~~~~~~~~~~~~~~~~~~~
|
|
After successful connection of a partner, the port driver needs to register the
|
partner with the class. Details about the partner need to be described in struct
|
typec_partner_desc. The class copies the details of the partner during
|
registration. The class offers the following API for registering/unregistering
|
partners.
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_register_partner typec_unregister_partner
|
|
The class will provide a handle to struct typec_partner if the registration was
|
successful, or NULL.
|
|
If the partner is USB Power Delivery capable, and the port driver is able to
|
show the result of Discover Identity command, the partner descriptor structure
|
should include handle to struct usb_pd_identity instance. The class will then
|
create a sysfs directory for the identity under the partner device. The result
|
of Discover Identity command can then be reported with the following API:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_partner_set_identity
|
|
Registering Cables
|
~~~~~~~~~~~~~~~~~~
|
|
After successful connection of a cable that supports USB Power Delivery
|
Structured VDM "Discover Identity", the port driver needs to register the cable
|
and one or two plugs, depending if there is CC Double Prime controller present
|
in the cable or not. So a cable capable of SOP Prime communication, but not SOP
|
Double Prime communication, should only have one plug registered. For more
|
information about SOP communication, please read chapter about it from the
|
latest USB Power Delivery specification.
|
|
The plugs are represented as their own devices. The cable is registered first,
|
followed by registration of the cable plugs. The cable will be the parent device
|
for the plugs. Details about the cable need to be described in struct
|
typec_cable_desc and about a plug in struct typec_plug_desc. The class copies
|
the details during registration. The class offers the following API for
|
registering/unregistering cables and their plugs:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_register_cable typec_unregister_cable typec_register_plug typec_unregister_plug
|
|
The class will provide a handle to struct typec_cable and struct typec_plug if
|
the registration is successful, or NULL if it isn't.
|
|
If the cable is USB Power Delivery capable, and the port driver is able to show
|
the result of Discover Identity command, the cable descriptor structure should
|
include handle to struct usb_pd_identity instance. The class will then create a
|
sysfs directory for the identity under the cable device. The result of Discover
|
Identity command can then be reported with the following API:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_cable_set_identity
|
|
Notifications
|
~~~~~~~~~~~~~
|
|
When the partner has executed a role change, or when the default roles change
|
during connection of a partner or cable, the port driver must use the following
|
APIs to report it to the class:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_set_data_role typec_set_pwr_role typec_set_vconn_role typec_set_pwr_opmode
|
|
Alternate Modes
|
~~~~~~~~~~~~~~~
|
|
USB Type-C ports, partners and cable plugs may support Alternate Modes. Each
|
Alternate Mode will have identifier called SVID, which is either a Standard ID
|
given by USB-IF or vendor ID, and each supported SVID can have 1 - 6 modes. The
|
class provides struct typec_mode_desc for describing individual mode of a SVID,
|
and struct typec_altmode_desc which is a container for all the supported modes.
|
|
Ports that support Alternate Modes need to register each SVID they support with
|
the following API:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_port_register_altmode
|
|
If a partner or cable plug provides a list of SVIDs as response to USB Power
|
Delivery Structured VDM Discover SVIDs message, each SVID needs to be
|
registered.
|
|
API for the partners:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_partner_register_altmode
|
|
API for the Cable Plugs:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_plug_register_altmode
|
|
So ports, partners and cable plugs will register the alternate modes with their
|
own functions, but the registration will always return a handle to struct
|
typec_altmode on success, or NULL. The unregistration will happen with the same
|
function:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_unregister_altmode
|
|
If a partner or cable plug enters or exits a mode, the port driver needs to
|
notify the class with the following API:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_altmode_update_active
|
|
Multiplexer/DeMultiplexer Switches
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
USB Type-C connectors may have one or more mux/demux switches behind them. Since
|
the plugs can be inserted right-side-up or upside-down, a switch is needed to
|
route the correct data pairs from the connector to the USB controllers. If
|
Alternate or Accessory Modes are supported, another switch is needed that can
|
route the pins on the connector to some other component besides USB. USB Type-C
|
Connector Class supplies an API for registering those switches.
|
|
.. kernel-doc:: drivers/usb/typec/mux.c
|
:functions: typec_switch_register typec_switch_unregister typec_mux_register typec_mux_unregister
|
|
In most cases the same physical mux will handle both the orientation and mode.
|
However, as the port drivers will be responsible for the orientation, and the
|
alternate mode drivers for the mode, the two are always separated into their
|
own logical components: "mux" for the mode and "switch" for the orientation.
|
|
When a port is registered, USB Type-C Connector Class requests both the mux and
|
the switch for the port. The drivers can then use the following API for
|
controlling them:
|
|
.. kernel-doc:: drivers/usb/typec/class.c
|
:functions: typec_set_orientation typec_set_mode
|
|
If the connector is dual-role capable, there may also be a switch for the data
|
role. USB Type-C Connector Class does not supply separate API for them. The
|
port drivers can use USB Role Class API with those.
|
|
Illustration of the muxes behind a connector that supports an alternate mode::
|
|
------------------------
|
| Connector |
|
------------------------
|
| |
|
------------------------
|
\ Orientation /
|
--------------------
|
|
|
--------------------
|
/ Mode \
|
------------------------
|
/ \
|
------------------------ --------------------
|
| Alt Mode | / USB Role \
|
------------------------ ------------------------
|
/ \
|
------------------------ ------------------------
|
| USB Host | | USB Device |
|
------------------------ ------------------------
|