* [PATCH v3 phy-next 0/2] phy: ti: add driver for TI DS125DF111 Dual-Channel Retimer
@ 2026-05-16 6:03 Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 1/2] dt-bindings: phy: add PHY bindings for the TI DS125DF111 Retimer PHY Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
0 siblings, 2 replies; 6+ messages in thread
From: Ioana Ciornei @ 2026-05-16 6:03 UTC (permalink / raw)
To: vkoul, neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy
Cc: devicetree, linux-kernel
This patch set adds a generic PHY driver and the corresponding DT
binding for the TI DS125DF111 Dual-Channel retimer. The datasheet on
which this driver was based on can be found at -
https://www.ti.com/lit/gpn/DS125DF111.
A separate generic PHY is registered for each of the two channels of the
retimer, so consumers can drive each channel independently. This allows
for independent control of the channels, which is especially important
since each channel can be routed to different SerDes lanes and it is not
guaranteed that the same retimer will do both directions of SerDes lane.
This was tested on a LS1088ARDB board with the Lynx10G SerDes PHY driver
yet to be submitted.
Changes in v3:
- Use reverse Christmas tree ordering
- Print a symbolic description in case of error
- Some words do not need to be capitalized
- Remove duplicated exit code path
- Return -EINVAL in case of unsupported submode received in .set_mode()
- Add a .validate() callback
- Remove comma after sentinel entry
- Add a ds125df111_rmw() helper
- Use read_poll_timeout() to wait for channel reset to complete
- Link to v2: https://lore.kernel.org/all/20260515110145.1925579-1-ioana.ciornei@nxp.com/
Changes in v2:
- Remove the label from the example
- Rename the node from 'retimer' to 'phy'
- Explicitly include all the needed headers
- Change ds125df111_xlate() so that it returns an error if args_count is
not exactly 1
- Add a MAINTAINERS entry
- Link to v1: https://lore.kernel.org/all/20260513185103.1371809-1-ioana.ciornei@nxp.com/
Ioana Ciornei (2):
dt-bindings: phy: add PHY bindings for the TI DS125DF111 Retimer PHY
phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer
.../bindings/phy/ti,ds125df111.yaml | 46 +++
MAINTAINERS | 7 +
drivers/phy/ti/Kconfig | 10 +
drivers/phy/ti/Makefile | 1 +
drivers/phy/ti/phy-ds125df111.c | 294 ++++++++++++++++++
5 files changed, 358 insertions(+)
create mode 100644 Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
create mode 100644 drivers/phy/ti/phy-ds125df111.c
--
2.25.1
^ permalink raw reply [flat|nested] 6+ messages in thread
* [PATCH v3 phy-next 1/2] dt-bindings: phy: add PHY bindings for the TI DS125DF111 Retimer PHY
2026-05-16 6:03 [PATCH v3 phy-next 0/2] phy: ti: add driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
@ 2026-05-16 6:03 ` Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
1 sibling, 0 replies; 6+ messages in thread
From: Ioana Ciornei @ 2026-05-16 6:03 UTC (permalink / raw)
To: vkoul, neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy
Cc: devicetree, linux-kernel
Add device tree binding for the TI DS125DF111 Retimer PHY.
Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
Acked-by: Conor Dooley <conor.dooley@microchip.com>
---
Changes in v3:
- none
Changes in v2:
- Remove the label from the example
- Rename the node from 'retimer' to 'phy'
---
.../bindings/phy/ti,ds125df111.yaml | 46 +++++++++++++++++++
1 file changed, 46 insertions(+)
create mode 100644 Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
diff --git a/Documentation/devicetree/bindings/phy/ti,ds125df111.yaml b/Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
new file mode 100644
index 000000000000..ca4605f1d664
--- /dev/null
+++ b/Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
@@ -0,0 +1,46 @@
+# SPDX-License-Identifier: (GPL-2.0-only OR BSD-2-Clause)
+%YAML 1.2
+---
+$id: http://devicetree.org/schemas/phy/ti,ds125df111.yaml#
+$schema: http://devicetree.org/meta-schemas/core.yaml#
+
+title: TI DS125DF111 Retimer PHY
+
+description:
+ This binding describes the TI DS125DF111 multi-protocol Retimer PHY.
+
+maintainers:
+ - Ioana Ciornei <ioana.ciornei@nxp.com>
+
+properties:
+ compatible:
+ const: ti,ds125df111
+
+ reg:
+ maxItems: 1
+
+ "#phy-cells":
+ const: 1
+ description: |
+ The phandle's argument in the PHY specifier selects one of the two
+ channels of the retimer
+
+required:
+ - compatible
+ - reg
+ - "#phy-cells"
+
+additionalProperties: false
+
+examples:
+ - |
+ i2c {
+ #address-cells = <1>;
+ #size-cells = <0>;
+
+ phy@18 {
+ compatible = "ti,ds125df111";
+ reg = <0x18>;
+ #phy-cells = <1>;
+ };
+ };
--
2.25.1
^ permalink raw reply related [flat|nested] 6+ messages in thread
* [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer
2026-05-16 6:03 [PATCH v3 phy-next 0/2] phy: ti: add driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 1/2] dt-bindings: phy: add PHY bindings for the TI DS125DF111 Retimer PHY Ioana Ciornei
@ 2026-05-16 6:03 ` Ioana Ciornei
2026-05-16 17:16 ` Vladimir Oltean
2026-05-17 16:37 ` Vinod Koul
1 sibling, 2 replies; 6+ messages in thread
From: Ioana Ciornei @ 2026-05-16 6:03 UTC (permalink / raw)
To: vkoul, neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy
Cc: devicetree, linux-kernel
Add a generic PHY driver for the TI DS125DF111 Multi-Protocol
Dual-Channel Retimer. The driver currently supports only 10G and 1G link
speeds but it can easily extended to also cover other usecases.
Since the available datasheet (https://www.ti.com/lit/gpn/DS125DF111)
does not name the registers, the name for the macros were determined by
their usage pattern.
A PHY device is created for each of the two channels present on the
retimer. This allows for independent configuration of the two channels.
This capability is especially important on retimers which have more than
2 channels that can be, depending on the board design, connected in
multiple different ways to the SerDes lanes.
Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
---
Changes in v3:
- Use reverse Christmas tree ordering
- Print a symbolic description in case of error
- Some words do not need to be capitalized
- Remove duplicated exit code path
- Return -EINVAL in case of unsupported submode received in .set_mode()
- Add a .validate() callback
- Remove comma after sentinel entry
- Add a ds125df111_rmw() helper
- Use read_poll_timeout() to wait for channel reset to complete
Changes in v2:
- Explicitly include all the needed headers
- Change ds125df111_xlate() so that it returns an error if args_count is
not exactly 1
- Add a MAINTAINERS entry
---
MAINTAINERS | 7 +
drivers/phy/ti/Kconfig | 10 ++
drivers/phy/ti/Makefile | 1 +
drivers/phy/ti/phy-ds125df111.c | 294 ++++++++++++++++++++++++++++++++
4 files changed, 312 insertions(+)
create mode 100644 drivers/phy/ti/phy-ds125df111.c
diff --git a/MAINTAINERS b/MAINTAINERS
index f877e5aaf2c7..58f410b666e7 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -26781,6 +26781,13 @@ T: git git://linuxtv.org/mhadli/v4l-dvb-davinci_devices.git
F: drivers/media/platform/ti/davinci/
F: include/media/davinci/
+TI DS125DF111 RETIMER PHY DRIVER
+M: Ioana Ciornei <ioana.ciornei@nxp.com>
+L: linux-phy@lists.infradead.org (moderated for non-subscribers)
+S: Maintained
+F: Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
+F: drivers/phy/ti/phy-ds125df111.c
+
TI ENHANCED CAPTURE (eCAP) DRIVER
M: Vignesh Raghavendra <vigneshr@ti.com>
R: Julien Panis <jpanis@baylibre.com>
diff --git a/drivers/phy/ti/Kconfig b/drivers/phy/ti/Kconfig
index b40f28019131..475e80fcd52d 100644
--- a/drivers/phy/ti/Kconfig
+++ b/drivers/phy/ti/Kconfig
@@ -111,3 +111,13 @@ config PHY_TI_GMII_SEL
help
This driver supports configuring of the TI CPSW Port mode depending on
the Ethernet PHY connected to the CPSW Port.
+
+config PHY_TI_DS125DF111
+ tristate "DS125DF111 2-Channel Retimer Driver"
+ depends on OF && I2C
+ select GENERIC_PHY
+ help
+ Enable this to add support for configuration and runtime management
+ of the TI DS125DF111 Multi-Protocol 2-Channel Retimer.
+ The retimer is modeled as a Generic PHY and supports both 10G and 1G
+ link speeds.
diff --git a/drivers/phy/ti/Makefile b/drivers/phy/ti/Makefile
index dcba2571c9bd..e68445ddd848 100644
--- a/drivers/phy/ti/Makefile
+++ b/drivers/phy/ti/Makefile
@@ -9,3 +9,4 @@ obj-$(CONFIG_TWL4030_USB) += phy-twl4030-usb.o
obj-$(CONFIG_PHY_AM654_SERDES) += phy-am654-serdes.o
obj-$(CONFIG_PHY_TI_GMII_SEL) += phy-gmii-sel.o
obj-$(CONFIG_PHY_J721E_WIZ) += phy-j721e-wiz.o
+obj-$(CONFIG_PHY_TI_DS125DF111) += phy-ds125df111.o
diff --git a/drivers/phy/ti/phy-ds125df111.c b/drivers/phy/ti/phy-ds125df111.c
new file mode 100644
index 000000000000..8788f340e3b0
--- /dev/null
+++ b/drivers/phy/ti/phy-ds125df111.c
@@ -0,0 +1,294 @@
+// SPDX-License-Identifier: GPL-2.0
+/* Copyright 2026 NXP */
+
+#include <linux/bitfield.h>
+#include <linux/bits.h>
+#include <linux/i2c.h>
+#include <linux/iopoll.h>
+#include <linux/kernel.h>
+#include <linux/module.h>
+#include <linux/phy.h>
+#include <linux/phy/phy.h>
+#include <linux/slab.h>
+
+#define DS125DF111_NUM_CH 2
+#define DS125DF111_NUM_VCO_GROUP_REG 5
+
+#define DS125DF111_CH_SELECT 0xff
+#define DS125DF111_CH_SELECT_TARGET_MASK GENMASK(3, 0)
+#define DS125DF111_CH_SELECT_EN BIT(2)
+
+#define DS125DF111_CH_CTRL 0x00
+#define DS125DF111_CH_CTRL_RESET BIT(2) /* self clearing */
+
+#define DS125DF111_CH_RST_SLEEP_US 10
+#define DS125DF111_CH_RST_TIMEOUT_US 10000
+
+#define DS125DF111_VCO_GROUP_BASE 0x60
+
+#define DS125DF111_RATIOS 0x2F
+#define DS125DF111_RATIOS_RATE_MASK GENMASK(7, 6)
+#define DS125DF111_RATIOS_SUBRATE_MASK GENMASK(5, 4)
+#define DS125DF111_RATIOS_MASK GENMASK(7, 4)
+
+struct ds125df111_ch {
+ struct phy *phy;
+ struct ds125df111_priv *priv;
+ int idx;
+};
+
+struct ds125df111_priv {
+ struct ds125df111_ch ch[DS125DF111_NUM_CH];
+ struct i2c_client *client;
+ struct mutex mutex; /* protects access to shared registers */
+};
+
+enum ds125df111_mode {
+ FREQ_1G,
+ FREQ_10G,
+};
+
+static const struct ds125df111_config {
+ u8 vco_group[DS125DF111_NUM_VCO_GROUP_REG];
+ u8 rate;
+ u8 subrate;
+} ds125df111_cfg[] = {
+ [FREQ_1G] = {
+ /* VCO group #0 = 10GHz, VCO group #1 = 10GHz */
+ .vco_group = {0x00, 0xB2, 0x00, 0xB2, 0xCC},
+ /* By using the following combination of rate and subrate we
+ * select divide ratios of 1, 2, 4, 8 on both groups
+ */
+ .rate = 0x1,
+ .subrate = 0x2,
+ },
+
+ [FREQ_10G] = {
+ /* VCO group #0 = 10.3125GHz, VCO group #1 = 10.3125GHz */
+ .vco_group = {0x90, 0xB3, 0x90, 0xB3, 0xCD},
+ /* By using the following combination of rate and subrate we
+ * select divide ratios of 1 on both groups
+ */
+ .rate = 0x1,
+ .subrate = 0x3,
+ },
+};
+
+static int ds125df111_rmw(struct ds125df111_priv *priv, u8 reg, u8 clr, u8 set)
+{
+ struct i2c_client *i2c = priv->client;
+ int err;
+ u8 val;
+
+ err = i2c_smbus_read_byte_data(i2c, reg);
+ if (err < 0)
+ return err;
+
+ val = (u8)err;
+ val &= ~clr;
+ val |= set;
+
+ err = i2c_smbus_write_byte_data(i2c, reg, val);
+ if (err < 0)
+ return err;
+
+ return 0;
+}
+
+static int ds125df111_configure(struct phy *phy,
+ const struct ds125df111_config *cfg)
+{
+ struct ds125df111_ch *ch = phy_get_drvdata(phy);
+ struct ds125df111_priv *priv = ch->priv;
+ struct i2c_client *i2c = priv->client;
+ struct device *dev = &phy->dev;
+ u8 ratios_val;
+ int err, i;
+ int val;
+
+ mutex_lock(&priv->mutex);
+
+ /* Make sure that any subsequent read/write operation will be directed
+ * only to the registers of the selected channel
+ */
+ err = ds125df111_rmw(priv, DS125DF111_CH_SELECT,
+ DS125DF111_CH_SELECT_TARGET_MASK,
+ DS125DF111_CH_SELECT_EN | ch->idx);
+ if (err < 0) {
+ dev_err(dev, "Unable to select channel: %pe\n", ERR_PTR(err));
+ goto out;
+ }
+
+ /* Reset channel registers and wait until the bit was cleared */
+ err = ds125df111_rmw(priv, DS125DF111_CH_CTRL, 0,
+ DS125DF111_CH_CTRL_RESET);
+ if (err < 0) {
+ dev_err(dev, "Error resetting channel configuration: %pe\n",
+ ERR_PTR(err));
+ goto out;
+ }
+
+ err = read_poll_timeout(i2c_smbus_read_byte_data, val,
+ val < 0 || !(val & DS125DF111_CH_CTRL_RESET),
+ DS125DF111_CH_RST_SLEEP_US,
+ DS125DF111_CH_RST_TIMEOUT_US, false, i2c,
+ DS125DF111_CH_CTRL);
+ if (err) {
+ dev_err(dev, "Timed out waiting for channel reset: %pe\n",
+ ERR_PTR(err));
+ goto out;
+ }
+
+ if (val < 0) {
+ dev_err(dev, "Error reading reset status: %pe\n", ERR_PTR(val));
+ err = val;
+ goto out;
+ }
+
+ /* Program the VCO group frequencies */
+ for (i = 0; i < DS125DF111_NUM_VCO_GROUP_REG; i++) {
+ err = i2c_smbus_write_byte_data(i2c,
+ DS125DF111_VCO_GROUP_BASE + i,
+ cfg->vco_group[i]);
+ if (err < 0) {
+ dev_err(dev, "Error programming VCO group: %pe\n",
+ ERR_PTR(err));
+ goto out;
+ }
+ }
+
+ /* Set the divide ratios for the VCO groups */
+ ratios_val = FIELD_PREP(DS125DF111_RATIOS_RATE_MASK, cfg->rate) |
+ FIELD_PREP(DS125DF111_RATIOS_SUBRATE_MASK, cfg->subrate);
+ err = ds125df111_rmw(priv, DS125DF111_RATIOS, DS125DF111_RATIOS_MASK,
+ ratios_val);
+ if (err < 0) {
+ dev_err(dev, "Error programming the divide ratios: %pe\n",
+ ERR_PTR(err));
+ goto out;
+ }
+
+out:
+ mutex_unlock(&priv->mutex);
+
+ return err;
+}
+
+static int ds125df111_set_mode(struct phy *phy, enum phy_mode mode, int submode)
+{
+ const struct ds125df111_config *cfg;
+
+ if (mode != PHY_MODE_ETHERNET)
+ return -EINVAL;
+
+ switch (submode) {
+ case PHY_INTERFACE_MODE_10GBASER:
+ cfg = &ds125df111_cfg[FREQ_10G];
+ break;
+ case PHY_INTERFACE_MODE_1000BASEX:
+ case PHY_INTERFACE_MODE_SGMII:
+ cfg = &ds125df111_cfg[FREQ_1G];
+ break;
+ default:
+ return -EINVAL;
+ }
+
+ return ds125df111_configure(phy, cfg);
+}
+
+static int ds125df111_validate(struct phy *phy, enum phy_mode mode, int submode,
+ union phy_configure_opts *opts __always_unused)
+{
+ if (mode != PHY_MODE_ETHERNET)
+ return -EINVAL;
+
+ switch (submode) {
+ case PHY_INTERFACE_MODE_10GBASER:
+ case PHY_INTERFACE_MODE_1000BASEX:
+ case PHY_INTERFACE_MODE_SGMII:
+ return 0;
+ default:
+ return -EINVAL;
+ }
+}
+
+static const struct phy_ops ds125df111_ops = {
+ .validate = ds125df111_validate,
+ .set_mode = ds125df111_set_mode,
+ .owner = THIS_MODULE,
+};
+
+static struct phy *ds125df111_xlate(struct device *dev,
+ const struct of_phandle_args *args)
+{
+ struct ds125df111_priv *priv = dev_get_drvdata(dev);
+ u32 idx;
+
+ if (args->args_count != 1)
+ return ERR_PTR(-EINVAL);
+
+ idx = args->args[0];
+ if (idx >= DS125DF111_NUM_CH) {
+ dev_err(dev, "Maximum number of channels is %d\n",
+ DS125DF111_NUM_CH);
+ return ERR_PTR(-EINVAL);
+ }
+
+ return priv->ch[idx].phy;
+}
+
+static int ds125df111_probe(struct i2c_client *client)
+{
+ struct device *dev = &client->dev;
+ struct phy_provider *provider;
+ struct ds125df111_priv *priv;
+ int i, err;
+
+ priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
+ if (!priv)
+ return -ENOMEM;
+ priv->client = client;
+ err = devm_mutex_init(dev, &priv->mutex);
+ if (err)
+ return err;
+
+ i2c_set_clientdata(client, priv);
+
+ for (i = 0; i < DS125DF111_NUM_CH; i++) {
+ struct ds125df111_ch *ch = &priv->ch[i];
+ struct phy *phy;
+
+ phy = devm_phy_create(dev, NULL, &ds125df111_ops);
+ if (IS_ERR(phy))
+ return PTR_ERR(phy);
+
+ ch->idx = i;
+ ch->priv = priv;
+ ch->phy = phy;
+
+ phy_set_drvdata(phy, ch);
+ }
+
+ provider = devm_of_phy_provider_register(dev, ds125df111_xlate);
+
+ return PTR_ERR_OR_ZERO(provider);
+}
+
+static const struct of_device_id ds125df111_dt_ids[] = {
+ { .compatible = "ti,ds125df111", },
+ {}
+};
+MODULE_DEVICE_TABLE(of, ds125df111_dt_ids);
+
+static struct i2c_driver ds125df111_driver = {
+ .driver = {
+ .name = "ds125df111",
+ .of_match_table = ds125df111_dt_ids,
+ },
+ .probe = ds125df111_probe,
+};
+module_i2c_driver(ds125df111_driver);
+
+MODULE_AUTHOR("Ioana Ciornei <ioana.ciornei@nxp.com>");
+MODULE_DESCRIPTION("TI DS125DF111 Retimer driver");
+MODULE_LICENSE("GPL");
--
2.25.1
^ permalink raw reply related [flat|nested] 6+ messages in thread
* Re: [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer
2026-05-16 6:03 ` [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
@ 2026-05-16 17:16 ` Vladimir Oltean
2026-05-17 16:37 ` Vinod Koul
1 sibling, 0 replies; 6+ messages in thread
From: Vladimir Oltean @ 2026-05-16 17:16 UTC (permalink / raw)
To: Ioana Ciornei
Cc: vkoul, neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy,
devicetree, linux-kernel
On Sat, May 16, 2026 at 09:03:09AM +0300, Ioana Ciornei wrote:
> Add a generic PHY driver for the TI DS125DF111 Multi-Protocol
> Dual-Channel Retimer. The driver currently supports only 10G and 1G link
> speeds but it can easily extended to also cover other usecases.
>
> Since the available datasheet (https://www.ti.com/lit/gpn/DS125DF111)
> does not name the registers, the name for the macros were determined by
> their usage pattern.
>
> A PHY device is created for each of the two channels present on the
> retimer. This allows for independent configuration of the two channels.
> This capability is especially important on retimers which have more than
> 2 channels that can be, depending on the board design, connected in
> multiple different ways to the SerDes lanes.
>
> Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
> ---
Reviewed-by: Vladimir Oltean <olteanv@gmail.com>
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer
2026-05-16 6:03 ` [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
2026-05-16 17:16 ` Vladimir Oltean
@ 2026-05-17 16:37 ` Vinod Koul
2026-05-18 7:22 ` Ioana Ciornei
1 sibling, 1 reply; 6+ messages in thread
From: Vinod Koul @ 2026-05-17 16:37 UTC (permalink / raw)
To: Ioana Ciornei
Cc: neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy,
devicetree, linux-kernel
On 16-05-26, 09:03, Ioana Ciornei wrote:
> Add a generic PHY driver for the TI DS125DF111 Multi-Protocol
> Dual-Channel Retimer. The driver currently supports only 10G and 1G link
> speeds but it can easily extended to also cover other usecases.
>
> Since the available datasheet (https://www.ti.com/lit/gpn/DS125DF111)
> does not name the registers, the name for the macros were determined by
> their usage pattern.
>
> A PHY device is created for each of the two channels present on the
> retimer. This allows for independent configuration of the two channels.
> This capability is especially important on retimers which have more than
> 2 channels that can be, depending on the board design, connected in
> multiple different ways to the SerDes lanes.
>
> Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
> ---
> Changes in v3:
> - Use reverse Christmas tree ordering
> - Print a symbolic description in case of error
> - Some words do not need to be capitalized
> - Remove duplicated exit code path
> - Return -EINVAL in case of unsupported submode received in .set_mode()
> - Add a .validate() callback
> - Remove comma after sentinel entry
> - Add a ds125df111_rmw() helper
> - Use read_poll_timeout() to wait for channel reset to complete
>
> Changes in v2:
> - Explicitly include all the needed headers
> - Change ds125df111_xlate() so that it returns an error if args_count is
> not exactly 1
> - Add a MAINTAINERS entry
> ---
> MAINTAINERS | 7 +
> drivers/phy/ti/Kconfig | 10 ++
> drivers/phy/ti/Makefile | 1 +
> drivers/phy/ti/phy-ds125df111.c | 294 ++++++++++++++++++++++++++++++++
> 4 files changed, 312 insertions(+)
> create mode 100644 drivers/phy/ti/phy-ds125df111.c
>
> diff --git a/MAINTAINERS b/MAINTAINERS
> index f877e5aaf2c7..58f410b666e7 100644
> --- a/MAINTAINERS
> +++ b/MAINTAINERS
> @@ -26781,6 +26781,13 @@ T: git git://linuxtv.org/mhadli/v4l-dvb-davinci_devices.git
> F: drivers/media/platform/ti/davinci/
> F: include/media/davinci/
>
> +TI DS125DF111 RETIMER PHY DRIVER
> +M: Ioana Ciornei <ioana.ciornei@nxp.com>
> +L: linux-phy@lists.infradead.org (moderated for non-subscribers)
> +S: Maintained
> +F: Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
> +F: drivers/phy/ti/phy-ds125df111.c
> +
> TI ENHANCED CAPTURE (eCAP) DRIVER
> M: Vignesh Raghavendra <vigneshr@ti.com>
> R: Julien Panis <jpanis@baylibre.com>
> diff --git a/drivers/phy/ti/Kconfig b/drivers/phy/ti/Kconfig
> index b40f28019131..475e80fcd52d 100644
> --- a/drivers/phy/ti/Kconfig
> +++ b/drivers/phy/ti/Kconfig
> @@ -111,3 +111,13 @@ config PHY_TI_GMII_SEL
> help
> This driver supports configuring of the TI CPSW Port mode depending on
> the Ethernet PHY connected to the CPSW Port.
> +
> +config PHY_TI_DS125DF111
This should be in alphabetical order, so I guess before PHY_TI_G...
> + tristate "DS125DF111 2-Channel Retimer Driver"
> + depends on OF && I2C
> + select GENERIC_PHY
> + help
> + Enable this to add support for configuration and runtime management
> + of the TI DS125DF111 Multi-Protocol 2-Channel Retimer.
> + The retimer is modeled as a Generic PHY and supports both 10G and 1G
> + link speeds.
> diff --git a/drivers/phy/ti/Makefile b/drivers/phy/ti/Makefile
> index dcba2571c9bd..e68445ddd848 100644
> --- a/drivers/phy/ti/Makefile
> +++ b/drivers/phy/ti/Makefile
> @@ -9,3 +9,4 @@ obj-$(CONFIG_TWL4030_USB) += phy-twl4030-usb.o
> obj-$(CONFIG_PHY_AM654_SERDES) += phy-am654-serdes.o
> obj-$(CONFIG_PHY_TI_GMII_SEL) += phy-gmii-sel.o
> obj-$(CONFIG_PHY_J721E_WIZ) += phy-j721e-wiz.o
> +obj-$(CONFIG_PHY_TI_DS125DF111) += phy-ds125df111.o
Here as well
> diff --git a/drivers/phy/ti/phy-ds125df111.c b/drivers/phy/ti/phy-ds125df111.c
> new file mode 100644
> index 000000000000..8788f340e3b0
> --- /dev/null
> +++ b/drivers/phy/ti/phy-ds125df111.c
> @@ -0,0 +1,294 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/* Copyright 2026 NXP */
> +
> +#include <linux/bitfield.h>
> +#include <linux/bits.h>
> +#include <linux/i2c.h>
> +#include <linux/iopoll.h>
> +#include <linux/kernel.h>
> +#include <linux/module.h>
> +#include <linux/phy.h>
> +#include <linux/phy/phy.h>
> +#include <linux/slab.h>
> +
> +#define DS125DF111_NUM_CH 2
> +#define DS125DF111_NUM_VCO_GROUP_REG 5
> +
> +#define DS125DF111_CH_SELECT 0xff
> +#define DS125DF111_CH_SELECT_TARGET_MASK GENMASK(3, 0)
> +#define DS125DF111_CH_SELECT_EN BIT(2)
> +
> +#define DS125DF111_CH_CTRL 0x00
> +#define DS125DF111_CH_CTRL_RESET BIT(2) /* self clearing */
> +
> +#define DS125DF111_CH_RST_SLEEP_US 10
> +#define DS125DF111_CH_RST_TIMEOUT_US 10000
> +
> +#define DS125DF111_VCO_GROUP_BASE 0x60
> +
> +#define DS125DF111_RATIOS 0x2F
Lower case for the hex values please
> +#define DS125DF111_RATIOS_RATE_MASK GENMASK(7, 6)
> +#define DS125DF111_RATIOS_SUBRATE_MASK GENMASK(5, 4)
> +#define DS125DF111_RATIOS_MASK GENMASK(7, 4)
> +
> +struct ds125df111_ch {
> + struct phy *phy;
> + struct ds125df111_priv *priv;
> + int idx;
> +};
> +
> +struct ds125df111_priv {
> + struct ds125df111_ch ch[DS125DF111_NUM_CH];
> + struct i2c_client *client;
> + struct mutex mutex; /* protects access to shared registers */
> +};
> +
> +enum ds125df111_mode {
> + FREQ_1G,
> + FREQ_10G,
> +};
> +
> +static const struct ds125df111_config {
> + u8 vco_group[DS125DF111_NUM_VCO_GROUP_REG];
> + u8 rate;
> + u8 subrate;
> +} ds125df111_cfg[] = {
> + [FREQ_1G] = {
> + /* VCO group #0 = 10GHz, VCO group #1 = 10GHz */
> + .vco_group = {0x00, 0xB2, 0x00, 0xB2, 0xCC},
> + /* By using the following combination of rate and subrate we
> + * select divide ratios of 1, 2, 4, 8 on both groups
> + */
> + .rate = 0x1,
> + .subrate = 0x2,
> + },
> +
> + [FREQ_10G] = {
> + /* VCO group #0 = 10.3125GHz, VCO group #1 = 10.3125GHz */
> + .vco_group = {0x90, 0xB3, 0x90, 0xB3, 0xCD},
> + /* By using the following combination of rate and subrate we
> + * select divide ratios of 1 on both groups
> + */
> + .rate = 0x1,
> + .subrate = 0x3,
> + },
> +};
> +
> +static int ds125df111_rmw(struct ds125df111_priv *priv, u8 reg, u8 clr, u8 set)
> +{
> + struct i2c_client *i2c = priv->client;
> + int err;
> + u8 val;
> +
> + err = i2c_smbus_read_byte_data(i2c, reg);
> + if (err < 0)
> + return err;
> +
> + val = (u8)err;
> + val &= ~clr;
> + val |= set;
> +
> + err = i2c_smbus_write_byte_data(i2c, reg, val);
> + if (err < 0)
> + return err;
> +
> + return 0;
> +}
> +
> +static int ds125df111_configure(struct phy *phy,
> + const struct ds125df111_config *cfg)
This should start at preceding line open braces (here and few other
places)
> +{
> + struct ds125df111_ch *ch = phy_get_drvdata(phy);
> + struct ds125df111_priv *priv = ch->priv;
> + struct i2c_client *i2c = priv->client;
> + struct device *dev = &phy->dev;
> + u8 ratios_val;
> + int err, i;
> + int val;
> +
> + mutex_lock(&priv->mutex);
> +
> + /* Make sure that any subsequent read/write operation will be directed
> + * only to the registers of the selected channel
> + */
> + err = ds125df111_rmw(priv, DS125DF111_CH_SELECT,
> + DS125DF111_CH_SELECT_TARGET_MASK,
> + DS125DF111_CH_SELECT_EN | ch->idx);
> + if (err < 0) {
> + dev_err(dev, "Unable to select channel: %pe\n", ERR_PTR(err));
> + goto out;
> + }
> +
> + /* Reset channel registers and wait until the bit was cleared */
> + err = ds125df111_rmw(priv, DS125DF111_CH_CTRL, 0,
> + DS125DF111_CH_CTRL_RESET);
> + if (err < 0) {
> + dev_err(dev, "Error resetting channel configuration: %pe\n",
> + ERR_PTR(err));
> + goto out;
> + }
> +
> + err = read_poll_timeout(i2c_smbus_read_byte_data, val,
> + val < 0 || !(val & DS125DF111_CH_CTRL_RESET),
> + DS125DF111_CH_RST_SLEEP_US,
> + DS125DF111_CH_RST_TIMEOUT_US, false, i2c,
> + DS125DF111_CH_CTRL);
> + if (err) {
> + dev_err(dev, "Timed out waiting for channel reset: %pe\n",
> + ERR_PTR(err));
> + goto out;
> + }
> +
> + if (val < 0) {
> + dev_err(dev, "Error reading reset status: %pe\n", ERR_PTR(val));
> + err = val;
> + goto out;
> + }
> +
> + /* Program the VCO group frequencies */
> + for (i = 0; i < DS125DF111_NUM_VCO_GROUP_REG; i++) {
> + err = i2c_smbus_write_byte_data(i2c,
> + DS125DF111_VCO_GROUP_BASE + i,
> + cfg->vco_group[i]);
> + if (err < 0) {
> + dev_err(dev, "Error programming VCO group: %pe\n",
> + ERR_PTR(err));
> + goto out;
> + }
> + }
> +
> + /* Set the divide ratios for the VCO groups */
> + ratios_val = FIELD_PREP(DS125DF111_RATIOS_RATE_MASK, cfg->rate) |
> + FIELD_PREP(DS125DF111_RATIOS_SUBRATE_MASK, cfg->subrate);
> + err = ds125df111_rmw(priv, DS125DF111_RATIOS, DS125DF111_RATIOS_MASK,
> + ratios_val);
> + if (err < 0) {
> + dev_err(dev, "Error programming the divide ratios: %pe\n",
> + ERR_PTR(err));
> + goto out;
> + }
> +
> +out:
> + mutex_unlock(&priv->mutex);
> +
> + return err;
> +}
> +
> +static int ds125df111_set_mode(struct phy *phy, enum phy_mode mode, int submode)
> +{
> + const struct ds125df111_config *cfg;
> +
> + if (mode != PHY_MODE_ETHERNET)
> + return -EINVAL;
> +
> + switch (submode) {
> + case PHY_INTERFACE_MODE_10GBASER:
> + cfg = &ds125df111_cfg[FREQ_10G];
> + break;
> + case PHY_INTERFACE_MODE_1000BASEX:
> + case PHY_INTERFACE_MODE_SGMII:
> + cfg = &ds125df111_cfg[FREQ_1G];
> + break;
> + default:
> + return -EINVAL;
> + }
> +
> + return ds125df111_configure(phy, cfg);
> +}
> +
> +static int ds125df111_validate(struct phy *phy, enum phy_mode mode, int submode,
> + union phy_configure_opts *opts __always_unused)
> +{
> + if (mode != PHY_MODE_ETHERNET)
> + return -EINVAL;
> +
> + switch (submode) {
> + case PHY_INTERFACE_MODE_10GBASER:
> + case PHY_INTERFACE_MODE_1000BASEX:
> + case PHY_INTERFACE_MODE_SGMII:
> + return 0;
> + default:
> + return -EINVAL;
> + }
> +}
> +
> +static const struct phy_ops ds125df111_ops = {
> + .validate = ds125df111_validate,
> + .set_mode = ds125df111_set_mode,
> + .owner = THIS_MODULE,
> +};
> +
> +static struct phy *ds125df111_xlate(struct device *dev,
> + const struct of_phandle_args *args)
> +{
> + struct ds125df111_priv *priv = dev_get_drvdata(dev);
> + u32 idx;
> +
> + if (args->args_count != 1)
> + return ERR_PTR(-EINVAL);
> +
> + idx = args->args[0];
> + if (idx >= DS125DF111_NUM_CH) {
> + dev_err(dev, "Maximum number of channels is %d\n",
> + DS125DF111_NUM_CH);
> + return ERR_PTR(-EINVAL);
> + }
> +
> + return priv->ch[idx].phy;
> +}
> +
> +static int ds125df111_probe(struct i2c_client *client)
> +{
> + struct device *dev = &client->dev;
> + struct phy_provider *provider;
> + struct ds125df111_priv *priv;
> + int i, err;
> +
> + priv = devm_kzalloc(dev, sizeof(*priv), GFP_KERNEL);
> + if (!priv)
> + return -ENOMEM;
> + priv->client = client;
> + err = devm_mutex_init(dev, &priv->mutex);
> + if (err)
> + return err;
> +
> + i2c_set_clientdata(client, priv);
> +
> + for (i = 0; i < DS125DF111_NUM_CH; i++) {
> + struct ds125df111_ch *ch = &priv->ch[i];
> + struct phy *phy;
> +
> + phy = devm_phy_create(dev, NULL, &ds125df111_ops);
> + if (IS_ERR(phy))
> + return PTR_ERR(phy);
> +
> + ch->idx = i;
> + ch->priv = priv;
> + ch->phy = phy;
> +
> + phy_set_drvdata(phy, ch);
> + }
> +
> + provider = devm_of_phy_provider_register(dev, ds125df111_xlate);
> +
> + return PTR_ERR_OR_ZERO(provider);
> +}
> +
> +static const struct of_device_id ds125df111_dt_ids[] = {
> + { .compatible = "ti,ds125df111", },
> + {}
> +};
> +MODULE_DEVICE_TABLE(of, ds125df111_dt_ids);
> +
> +static struct i2c_driver ds125df111_driver = {
> + .driver = {
> + .name = "ds125df111",
> + .of_match_table = ds125df111_dt_ids,
> + },
> + .probe = ds125df111_probe,
> +};
> +module_i2c_driver(ds125df111_driver);
> +
> +MODULE_AUTHOR("Ioana Ciornei <ioana.ciornei@nxp.com>");
> +MODULE_DESCRIPTION("TI DS125DF111 Retimer driver");
> +MODULE_LICENSE("GPL");
> --
> 2.25.1
--
~Vinod
^ permalink raw reply [flat|nested] 6+ messages in thread
* Re: [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer
2026-05-17 16:37 ` Vinod Koul
@ 2026-05-18 7:22 ` Ioana Ciornei
0 siblings, 0 replies; 6+ messages in thread
From: Ioana Ciornei @ 2026-05-18 7:22 UTC (permalink / raw)
To: Vinod Koul
Cc: neil.armstrong, robh, krzk+dt, conor+dt, johan, linux-phy,
devicetree, linux-kernel
On Sun, May 17, 2026 at 10:07:56PM +0530, Vinod Koul wrote:
> On 16-05-26, 09:03, Ioana Ciornei wrote:
> > Add a generic PHY driver for the TI DS125DF111 Multi-Protocol
> > Dual-Channel Retimer. The driver currently supports only 10G and 1G link
> > speeds but it can easily extended to also cover other usecases.
> >
> > Since the available datasheet (https://www.ti.com/lit/gpn/DS125DF111)
> > does not name the registers, the name for the macros were determined by
> > their usage pattern.
> >
> > A PHY device is created for each of the two channels present on the
> > retimer. This allows for independent configuration of the two channels.
> > This capability is especially important on retimers which have more than
> > 2 channels that can be, depending on the board design, connected in
> > multiple different ways to the SerDes lanes.
> >
> > Signed-off-by: Ioana Ciornei <ioana.ciornei@nxp.com>
> > ---
> > Changes in v3:
> > - Use reverse Christmas tree ordering
> > - Print a symbolic description in case of error
> > - Some words do not need to be capitalized
> > - Remove duplicated exit code path
> > - Return -EINVAL in case of unsupported submode received in .set_mode()
> > - Add a .validate() callback
> > - Remove comma after sentinel entry
> > - Add a ds125df111_rmw() helper
> > - Use read_poll_timeout() to wait for channel reset to complete
> >
> > Changes in v2:
> > - Explicitly include all the needed headers
> > - Change ds125df111_xlate() so that it returns an error if args_count is
> > not exactly 1
> > - Add a MAINTAINERS entry
> > ---
> > MAINTAINERS | 7 +
> > drivers/phy/ti/Kconfig | 10 ++
> > drivers/phy/ti/Makefile | 1 +
> > drivers/phy/ti/phy-ds125df111.c | 294 ++++++++++++++++++++++++++++++++
> > 4 files changed, 312 insertions(+)
> > create mode 100644 drivers/phy/ti/phy-ds125df111.c
> >
> > diff --git a/MAINTAINERS b/MAINTAINERS
> > index f877e5aaf2c7..58f410b666e7 100644
> > --- a/MAINTAINERS
> > +++ b/MAINTAINERS
> > @@ -26781,6 +26781,13 @@ T: git git://linuxtv.org/mhadli/v4l-dvb-davinci_devices.git
> > F: drivers/media/platform/ti/davinci/
> > F: include/media/davinci/
> >
> > +TI DS125DF111 RETIMER PHY DRIVER
> > +M: Ioana Ciornei <ioana.ciornei@nxp.com>
> > +L: linux-phy@lists.infradead.org (moderated for non-subscribers)
> > +S: Maintained
> > +F: Documentation/devicetree/bindings/phy/ti,ds125df111.yaml
> > +F: drivers/phy/ti/phy-ds125df111.c
> > +
> > TI ENHANCED CAPTURE (eCAP) DRIVER
> > M: Vignesh Raghavendra <vigneshr@ti.com>
> > R: Julien Panis <jpanis@baylibre.com>
> > diff --git a/drivers/phy/ti/Kconfig b/drivers/phy/ti/Kconfig
> > index b40f28019131..475e80fcd52d 100644
> > --- a/drivers/phy/ti/Kconfig
> > +++ b/drivers/phy/ti/Kconfig
> > @@ -111,3 +111,13 @@ config PHY_TI_GMII_SEL
> > help
> > This driver supports configuring of the TI CPSW Port mode depending on
> > the Ethernet PHY connected to the CPSW Port.
> > +
> > +config PHY_TI_DS125DF111
>
> This should be in alphabetical order, so I guess before PHY_TI_G...
Ok, will reorder them.
>
> > + tristate "DS125DF111 2-Channel Retimer Driver"
> > + depends on OF && I2C
> > + select GENERIC_PHY
> > + help
> > + Enable this to add support for configuration and runtime management
> > + of the TI DS125DF111 Multi-Protocol 2-Channel Retimer.
> > + The retimer is modeled as a Generic PHY and supports both 10G and 1G
> > + link speeds.
> > diff --git a/drivers/phy/ti/Makefile b/drivers/phy/ti/Makefile
> > index dcba2571c9bd..e68445ddd848 100644
> > --- a/drivers/phy/ti/Makefile
> > +++ b/drivers/phy/ti/Makefile
> > @@ -9,3 +9,4 @@ obj-$(CONFIG_TWL4030_USB) += phy-twl4030-usb.o
> > obj-$(CONFIG_PHY_AM654_SERDES) += phy-am654-serdes.o
> > obj-$(CONFIG_PHY_TI_GMII_SEL) += phy-gmii-sel.o
> > obj-$(CONFIG_PHY_J721E_WIZ) += phy-j721e-wiz.o
> > +obj-$(CONFIG_PHY_TI_DS125DF111) += phy-ds125df111.o
>
> Here as well
Sure.
>
> > diff --git a/drivers/phy/ti/phy-ds125df111.c b/drivers/phy/ti/phy-ds125df111.c
> > new file mode 100644
> > index 000000000000..8788f340e3b0
> > --- /dev/null
> > +++ b/drivers/phy/ti/phy-ds125df111.c
> > @@ -0,0 +1,294 @@
> > +// SPDX-License-Identifier: GPL-2.0
> > +/* Copyright 2026 NXP */
> > +
> > +#include <linux/bitfield.h>
> > +#include <linux/bits.h>
> > +#include <linux/i2c.h>
> > +#include <linux/iopoll.h>
> > +#include <linux/kernel.h>
> > +#include <linux/module.h>
> > +#include <linux/phy.h>
> > +#include <linux/phy/phy.h>
> > +#include <linux/slab.h>
> > +
> > +#define DS125DF111_NUM_CH 2
> > +#define DS125DF111_NUM_VCO_GROUP_REG 5
> > +
> > +#define DS125DF111_CH_SELECT 0xff
> > +#define DS125DF111_CH_SELECT_TARGET_MASK GENMASK(3, 0)
> > +#define DS125DF111_CH_SELECT_EN BIT(2)
> > +
> > +#define DS125DF111_CH_CTRL 0x00
> > +#define DS125DF111_CH_CTRL_RESET BIT(2) /* self clearing */
> > +
> > +#define DS125DF111_CH_RST_SLEEP_US 10
> > +#define DS125DF111_CH_RST_TIMEOUT_US 10000
> > +
> > +#define DS125DF111_VCO_GROUP_BASE 0x60
> > +
> > +#define DS125DF111_RATIOS 0x2F
>
> Lower case for the hex values please
>
> > +#define DS125DF111_RATIOS_RATE_MASK GENMASK(7, 6)
> > +#define DS125DF111_RATIOS_SUBRATE_MASK GENMASK(5, 4)
> > +#define DS125DF111_RATIOS_MASK GENMASK(7, 4)
> > +
> > +struct ds125df111_ch {
> > + struct phy *phy;
> > + struct ds125df111_priv *priv;
> > + int idx;
> > +};
> > +
> > +struct ds125df111_priv {
> > + struct ds125df111_ch ch[DS125DF111_NUM_CH];
> > + struct i2c_client *client;
> > + struct mutex mutex; /* protects access to shared registers */
> > +};
> > +
> > +enum ds125df111_mode {
> > + FREQ_1G,
> > + FREQ_10G,
> > +};
> > +
> > +static const struct ds125df111_config {
> > + u8 vco_group[DS125DF111_NUM_VCO_GROUP_REG];
> > + u8 rate;
> > + u8 subrate;
> > +} ds125df111_cfg[] = {
> > + [FREQ_1G] = {
> > + /* VCO group #0 = 10GHz, VCO group #1 = 10GHz */
> > + .vco_group = {0x00, 0xB2, 0x00, 0xB2, 0xCC},
> > + /* By using the following combination of rate and subrate we
> > + * select divide ratios of 1, 2, 4, 8 on both groups
> > + */
> > + .rate = 0x1,
> > + .subrate = 0x2,
> > + },
> > +
> > + [FREQ_10G] = {
> > + /* VCO group #0 = 10.3125GHz, VCO group #1 = 10.3125GHz */
> > + .vco_group = {0x90, 0xB3, 0x90, 0xB3, 0xCD},
> > + /* By using the following combination of rate and subrate we
> > + * select divide ratios of 1 on both groups
> > + */
> > + .rate = 0x1,
> > + .subrate = 0x3,
> > + },
> > +};
> > +
> > +static int ds125df111_rmw(struct ds125df111_priv *priv, u8 reg, u8 clr, u8 set)
> > +{
> > + struct i2c_client *i2c = priv->client;
> > + int err;
> > + u8 val;
> > +
> > + err = i2c_smbus_read_byte_data(i2c, reg);
> > + if (err < 0)
> > + return err;
> > +
> > + val = (u8)err;
> > + val &= ~clr;
> > + val |= set;
> > +
> > + err = i2c_smbus_write_byte_data(i2c, reg, val);
> > + if (err < 0)
> > + return err;
> > +
> > + return 0;
> > +}
> > +
> > +static int ds125df111_configure(struct phy *phy,
> > + const struct ds125df111_config *cfg)
>
> This should start at preceding line open braces (here and few other
> places)
And it does start after the open braces of the previous line.
The .patch format adds that '+' which messes up with the alignment,
mainly when tabs are used. Once you apply the patch you can see that the
arguments start where they should.
^ permalink raw reply [flat|nested] 6+ messages in thread
end of thread, other threads:[~2026-05-18 7:22 UTC | newest]
Thread overview: 6+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2026-05-16 6:03 [PATCH v3 phy-next 0/2] phy: ti: add driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 1/2] dt-bindings: phy: add PHY bindings for the TI DS125DF111 Retimer PHY Ioana Ciornei
2026-05-16 6:03 ` [PATCH v3 phy-next 2/2] phy: ti: add PHY driver for TI DS125DF111 Dual-Channel Retimer Ioana Ciornei
2026-05-16 17:16 ` Vladimir Oltean
2026-05-17 16:37 ` Vinod Koul
2026-05-18 7:22 ` Ioana Ciornei
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox