mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-01 02:36:02 +00:00
9b1d858839
Now that phylink has a comprehensive PCS support, update the porting guide to explain the process of supporting the PCS configuration. This also removed outdated references to phylink_config fields that no longer exists. Signed-off-by: Maxime Chevallier <maxime.chevallier@bootlin.com> Signed-off-by: Paolo Abeni <pabeni@redhat.com>
406 lines
14 KiB
ReStructuredText
406 lines
14 KiB
ReStructuredText
.. SPDX-License-Identifier: GPL-2.0
|
|
|
|
=======
|
|
phylink
|
|
=======
|
|
|
|
Overview
|
|
========
|
|
|
|
phylink is a mechanism to support hot-pluggable networking modules
|
|
directly connected to a MAC without needing to re-initialise the
|
|
adapter on hot-plug events.
|
|
|
|
phylink supports conventional phylib-based setups, fixed link setups
|
|
and SFP (Small Formfactor Pluggable) modules at present.
|
|
|
|
Modes of operation
|
|
==================
|
|
|
|
phylink has several modes of operation, which depend on the firmware
|
|
settings.
|
|
|
|
1. PHY mode
|
|
|
|
In PHY mode, we use phylib to read the current link settings from
|
|
the PHY, and pass them to the MAC driver. We expect the MAC driver
|
|
to configure exactly the modes that are specified without any
|
|
negotiation being enabled on the link.
|
|
|
|
2. Fixed mode
|
|
|
|
Fixed mode is the same as PHY mode as far as the MAC driver is
|
|
concerned.
|
|
|
|
3. In-band mode
|
|
|
|
In-band mode is used with 802.3z, SGMII and similar interface modes,
|
|
and we are expecting to use and honor the in-band negotiation or
|
|
control word sent across the serdes channel.
|
|
|
|
By example, what this means is that:
|
|
|
|
.. code-block:: none
|
|
|
|
ð {
|
|
phy = <&phy>;
|
|
phy-mode = "sgmii";
|
|
};
|
|
|
|
does not use in-band SGMII signalling. The PHY is expected to follow
|
|
exactly the settings given to it in its :c:func:`mac_config` function.
|
|
The link should be forced up or down appropriately in the
|
|
:c:func:`mac_link_up` and :c:func:`mac_link_down` functions.
|
|
|
|
.. code-block:: none
|
|
|
|
ð {
|
|
managed = "in-band-status";
|
|
phy = <&phy>;
|
|
phy-mode = "sgmii";
|
|
};
|
|
|
|
uses in-band mode, where results from the PHY's negotiation are passed
|
|
to the MAC through the SGMII control word, and the MAC is expected to
|
|
acknowledge the control word. The :c:func:`mac_link_up` and
|
|
:c:func:`mac_link_down` functions must not force the MAC side link
|
|
up and down.
|
|
|
|
Rough guide to converting a network driver to sfp/phylink
|
|
=========================================================
|
|
|
|
This guide briefly describes how to convert a network driver from
|
|
phylib to the sfp/phylink support. Please send patches to improve
|
|
this documentation.
|
|
|
|
1. Optionally split the network driver's phylib update function into
|
|
two parts dealing with link-down and link-up. This can be done as
|
|
a separate preparation commit.
|
|
|
|
An older example of this preparation can be found in git commit
|
|
fc548b991fb0, although this was splitting into three parts; the
|
|
link-up part now includes configuring the MAC for the link settings.
|
|
Please see :c:func:`mac_link_up` for more information on this.
|
|
|
|
2. Replace::
|
|
|
|
select FIXED_PHY
|
|
select PHYLIB
|
|
|
|
with::
|
|
|
|
select PHYLINK
|
|
|
|
in the driver's Kconfig stanza.
|
|
|
|
3. Add::
|
|
|
|
#include <linux/phylink.h>
|
|
|
|
to the driver's list of header files.
|
|
|
|
4. Add::
|
|
|
|
struct phylink *phylink;
|
|
struct phylink_config phylink_config;
|
|
|
|
to the driver's private data structure. We shall refer to the
|
|
driver's private data pointer as ``priv`` below, and the driver's
|
|
private data structure as ``struct foo_priv``.
|
|
|
|
5. Replace the following functions:
|
|
|
|
.. flat-table::
|
|
:header-rows: 1
|
|
:widths: 1 1
|
|
:stub-columns: 0
|
|
|
|
* - Original function
|
|
- Replacement function
|
|
* - phy_start(phydev)
|
|
- phylink_start(priv->phylink)
|
|
* - phy_stop(phydev)
|
|
- phylink_stop(priv->phylink)
|
|
* - phy_mii_ioctl(phydev, ifr, cmd)
|
|
- phylink_mii_ioctl(priv->phylink, ifr, cmd)
|
|
* - phy_ethtool_get_wol(phydev, wol)
|
|
- phylink_ethtool_get_wol(priv->phylink, wol)
|
|
* - phy_ethtool_set_wol(phydev, wol)
|
|
- phylink_ethtool_set_wol(priv->phylink, wol)
|
|
* - phy_disconnect(phydev)
|
|
- phylink_disconnect_phy(priv->phylink)
|
|
|
|
Please note that some of these functions must be called under the
|
|
rtnl lock, and will warn if not. This will normally be the case,
|
|
except if these are called from the driver suspend/resume paths.
|
|
|
|
6. Add/replace ksettings get/set methods with:
|
|
|
|
.. code-block:: c
|
|
|
|
static int foo_ethtool_set_link_ksettings(struct net_device *dev,
|
|
const struct ethtool_link_ksettings *cmd)
|
|
{
|
|
struct foo_priv *priv = netdev_priv(dev);
|
|
|
|
return phylink_ethtool_ksettings_set(priv->phylink, cmd);
|
|
}
|
|
|
|
static int foo_ethtool_get_link_ksettings(struct net_device *dev,
|
|
struct ethtool_link_ksettings *cmd)
|
|
{
|
|
struct foo_priv *priv = netdev_priv(dev);
|
|
|
|
return phylink_ethtool_ksettings_get(priv->phylink, cmd);
|
|
}
|
|
|
|
7. Replace the call to::
|
|
|
|
phy_dev = of_phy_connect(dev, node, link_func, flags, phy_interface);
|
|
|
|
and associated code with a call to::
|
|
|
|
err = phylink_of_phy_connect(priv->phylink, node, flags);
|
|
|
|
For the most part, ``flags`` can be zero; these flags are passed to
|
|
the phy_attach_direct() inside this function call if a PHY is specified
|
|
in the DT node ``node``.
|
|
|
|
``node`` should be the DT node which contains the network phy property,
|
|
fixed link properties, and will also contain the sfp property.
|
|
|
|
The setup of fixed links should also be removed; these are handled
|
|
internally by phylink.
|
|
|
|
of_phy_connect() was also passed a function pointer for link updates.
|
|
This function is replaced by a different form of MAC updates
|
|
described below in (8).
|
|
|
|
Manipulation of the PHY's supported/advertised happens within phylink
|
|
based on the validate callback, see below in (8).
|
|
|
|
Note that the driver no longer needs to store the ``phy_interface``,
|
|
and also note that ``phy_interface`` becomes a dynamic property,
|
|
just like the speed, duplex etc. settings.
|
|
|
|
Finally, note that the MAC driver has no direct access to the PHY
|
|
anymore; that is because in the phylink model, the PHY can be
|
|
dynamic.
|
|
|
|
8. Add a :c:type:`struct phylink_mac_ops <phylink_mac_ops>` instance to
|
|
the driver, which is a table of function pointers, and implement
|
|
these functions. The old link update function for
|
|
:c:func:`of_phy_connect` becomes three methods: :c:func:`mac_link_up`,
|
|
:c:func:`mac_link_down`, and :c:func:`mac_config`. If step 1 was
|
|
performed, then the functionality will have been split there.
|
|
|
|
It is important that if in-band negotiation is used,
|
|
:c:func:`mac_link_up` and :c:func:`mac_link_down` do not prevent the
|
|
in-band negotiation from completing, since these functions are called
|
|
when the in-band link state changes - otherwise the link will never
|
|
come up.
|
|
|
|
The :c:func:`mac_get_caps` method is optional, and if provided should
|
|
return the phylink MAC capabilities that are supported for the passed
|
|
``interface`` mode. In general, there is no need to implement this method.
|
|
Phylink will use these capabilities in combination with permissible
|
|
capabilities for ``interface`` to determine the allowable ethtool link
|
|
modes.
|
|
|
|
The :c:func:`mac_link_state` method is used to read the link state
|
|
from the MAC, and report back the settings that the MAC is currently
|
|
using. This is particularly important for in-band negotiation
|
|
methods such as 1000base-X and SGMII.
|
|
|
|
The :c:func:`mac_link_up` method is used to inform the MAC that the
|
|
link has come up. The call includes the negotiation mode and interface
|
|
for reference only. The finalised link parameters are also supplied
|
|
(speed, duplex and flow control/pause enablement settings) which
|
|
should be used to configure the MAC when the MAC and PCS are not
|
|
tightly integrated, or when the settings are not coming from in-band
|
|
negotiation.
|
|
|
|
The :c:func:`mac_config` method is used to update the MAC with the
|
|
requested state, and must avoid unnecessarily taking the link down
|
|
when making changes to the MAC configuration. This means the
|
|
function should modify the state and only take the link down when
|
|
absolutely necessary to change the MAC configuration. An example
|
|
of how to do this can be found in :c:func:`mvneta_mac_config` in
|
|
``drivers/net/ethernet/marvell/mvneta.c``.
|
|
|
|
For further information on these methods, please see the inline
|
|
documentation in :c:type:`struct phylink_mac_ops <phylink_mac_ops>`.
|
|
|
|
9. Fill-in the :c:type:`struct phylink_config <phylink_config>` fields with
|
|
a reference to the :c:type:`struct device <device>` associated to your
|
|
:c:type:`struct net_device <net_device>`:
|
|
|
|
.. code-block:: c
|
|
|
|
priv->phylink_config.dev = &dev.dev;
|
|
priv->phylink_config.type = PHYLINK_NETDEV;
|
|
|
|
Fill-in the various speeds, pause and duplex modes your MAC can handle:
|
|
|
|
.. code-block:: c
|
|
|
|
priv->phylink_config.mac_capabilities = MAC_SYM_PAUSE | MAC_10 | MAC_100 | MAC_1000FD;
|
|
|
|
10. Some Ethernet controllers work in pair with a PCS (Physical Coding Sublayer)
|
|
block, that can handle among other things the encoding/decoding, link
|
|
establishment detection and autonegotiation. While some MACs have internal
|
|
PCS whose operation is transparent, some other require dedicated PCS
|
|
configuration for the link to become functional. In that case, phylink
|
|
provides a PCS abstraction through :c:type:`struct phylink_pcs <phylink_pcs>`.
|
|
|
|
Identify if your driver has one or more internal PCS blocks, and/or if
|
|
your controller can use an external PCS block that might be internally
|
|
connected to your controller.
|
|
|
|
If your controller doesn't have any internal PCS, you can go to step 11.
|
|
|
|
If your Ethernet controller contains one or several PCS blocks, create
|
|
one :c:type:`struct phylink_pcs <phylink_pcs>` instance per PCS block within
|
|
your driver's private data structure:
|
|
|
|
.. code-block:: c
|
|
|
|
struct phylink_pcs pcs;
|
|
|
|
Populate the relevant :c:type:`struct phylink_pcs_ops <phylink_pcs_ops>` to
|
|
configure your PCS. Create a :c:func:`pcs_get_state` function that reports
|
|
the inband link state, a :c:func:`pcs_config` function to configure your
|
|
PCS according to phylink-provided parameters, and a :c:func:`pcs_validate`
|
|
function that report to phylink all accepted configuration parameters for
|
|
your PCS:
|
|
|
|
.. code-block:: c
|
|
|
|
struct phylink_pcs_ops foo_pcs_ops = {
|
|
.pcs_validate = foo_pcs_validate,
|
|
.pcs_get_state = foo_pcs_get_state,
|
|
.pcs_config = foo_pcs_config,
|
|
};
|
|
|
|
Arrange for PCS link state interrupts to be forwarded into
|
|
phylink, via:
|
|
|
|
.. code-block:: c
|
|
|
|
phylink_pcs_change(pcs, link_is_up);
|
|
|
|
where ``link_is_up`` is true if the link is currently up or false
|
|
otherwise. If a PCS is unable to provide these interrupts, then
|
|
it should set ``pcs->pcs_poll = true;`` when creating the PCS.
|
|
|
|
11. If your controller relies on, or accepts the presence of an external PCS
|
|
controlled through its own driver, add a pointer to a phylink_pcs instance
|
|
in your driver private data structure:
|
|
|
|
.. code-block:: c
|
|
|
|
struct phylink_pcs *pcs;
|
|
|
|
The way of getting an instance of the actual PCS depends on the platform,
|
|
some PCS sit on an MDIO bus and are grabbed by passing a pointer to the
|
|
corresponding :c:type:`struct mii_bus <mii_bus>` and the PCS's address on
|
|
that bus. In this example, we assume the controller attaches to a Lynx PCS
|
|
instance:
|
|
|
|
.. code-block:: c
|
|
|
|
priv->pcs = lynx_pcs_create_mdiodev(bus, 0);
|
|
|
|
Some PCS can be recovered based on firmware information:
|
|
|
|
.. code-block:: c
|
|
|
|
priv->pcs = lynx_pcs_create_fwnode(of_fwnode_handle(node));
|
|
|
|
12. Populate the :c:func:`mac_select_pcs` callback and add it to your
|
|
:c:type:`struct phylink_mac_ops <phylink_mac_ops>` set of ops. This function
|
|
must return a pointer to the relevant :c:type:`struct phylink_pcs <phylink_pcs>`
|
|
that will be used for the requested link configuration:
|
|
|
|
.. code-block:: c
|
|
|
|
static struct phylink_pcs *foo_select_pcs(struct phylink_config *config,
|
|
phy_interface_t interface)
|
|
{
|
|
struct foo_priv *priv = container_of(config, struct foo_priv,
|
|
phylink_config);
|
|
|
|
if ( /* 'interface' needs a PCS to function */ )
|
|
return priv->pcs;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
See :c:func:`mvpp2_select_pcs` for an example of a driver that has multiple
|
|
internal PCS.
|
|
|
|
13. Fill-in all the :c:type:`phy_interface_t <phy_interface_t>` (i.e. all MAC to
|
|
PHY link modes) that your MAC can output. The following example shows a
|
|
configuration for a MAC that can handle all RGMII modes, SGMII and 1000BaseX.
|
|
You must adjust these according to what your MAC and all PCS associated
|
|
with this MAC are capable of, and not just the interface you wish to use:
|
|
|
|
.. code-block:: c
|
|
|
|
phy_interface_set_rgmii(priv->phylink_config.supported_interfaces);
|
|
__set_bit(PHY_INTERFACE_MODE_SGMII,
|
|
priv->phylink_config.supported_interfaces);
|
|
__set_bit(PHY_INTERFACE_MODE_1000BASEX,
|
|
priv->phylink_config.supported_interfaces);
|
|
|
|
14. Remove calls to of_parse_phandle() for the PHY,
|
|
of_phy_register_fixed_link() for fixed links etc. from the probe
|
|
function, and replace with:
|
|
|
|
.. code-block:: c
|
|
|
|
struct phylink *phylink;
|
|
|
|
phylink = phylink_create(&priv->phylink_config, node, phy_mode, &phylink_ops);
|
|
if (IS_ERR(phylink)) {
|
|
err = PTR_ERR(phylink);
|
|
fail probe;
|
|
}
|
|
|
|
priv->phylink = phylink;
|
|
|
|
and arrange to destroy the phylink in the probe failure path as
|
|
appropriate and the removal path too by calling:
|
|
|
|
.. code-block:: c
|
|
|
|
phylink_destroy(priv->phylink);
|
|
|
|
15. Arrange for MAC link state interrupts to be forwarded into
|
|
phylink, via:
|
|
|
|
.. code-block:: c
|
|
|
|
phylink_mac_change(priv->phylink, link_is_up);
|
|
|
|
where ``link_is_up`` is true if the link is currently up or false
|
|
otherwise.
|
|
|
|
16. Verify that the driver does not call::
|
|
|
|
netif_carrier_on()
|
|
netif_carrier_off()
|
|
|
|
as these will interfere with phylink's tracking of the link state,
|
|
and cause phylink to omit calls via the :c:func:`mac_link_up` and
|
|
:c:func:`mac_link_down` methods.
|
|
|
|
Network drivers should call phylink_stop() and phylink_start() via their
|
|
suspend/resume paths, which ensures that the appropriate
|
|
:c:type:`struct phylink_mac_ops <phylink_mac_ops>` methods are called
|
|
as necessary.
|
|
|
|
For information describing the SFP cage in DT, please see the binding
|
|
documentation in the kernel source tree
|
|
``Documentation/devicetree/bindings/net/sff,sfp.yaml``.
|