Message ID | 20210111050044.22002-2-kabel@kernel.org (mailing list archive) |
---|---|
State | Superseded |
Delegated to: | Netdev Maintainers |
Headers | show |
Series | Support for RollBall 10G copper SFP modules | expand |
Context | Check | Description |
---|---|---|
netdev/cover_letter | success | Link |
netdev/fixes_present | success | Link |
netdev/patch_count | success | Link |
netdev/tree_selection | success | Clearly marked for net-next |
netdev/subject_prefix | success | Link |
netdev/cc_maintainers | warning | 2 maintainers not CCed: linux@armlinux.org.uk hkallweit1@gmail.com |
netdev/source_inline | success | Was 0 now: 0 |
netdev/verify_signedoff | success | Link |
netdev/module_param | success | Was 0 now: 0 |
netdev/build_32bit | success | Errors and warnings before: 0 this patch: 0 |
netdev/kdoc | success | Errors and warnings before: 0 this patch: 0 |
netdev/verify_fixes | success | Link |
netdev/checkpatch | success | total: 0 errors, 0 warnings, 0 checks, 380 lines checked |
netdev/build_allmodconfig_warn | success | Errors and warnings before: 0 this patch: 0 |
netdev/header_inline | success | Link |
netdev/stable | success | Stable not CCed |
On 11.01.2021 06:00, Marek Behún wrote: > Some multigig SFPs from RollBall and Hilink do not expose functional > MDIO access to the internal PHY of the SFP via I2C address 0x56 > (although there seems to be read-only clause 22 access on this address). > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > selected to 3 and the password must be filled with 0xff bytes for this > PHY communication to work. > > This extends the mdio-i2c driver to support this protocol by adding a > special parameter to mdio_i2c_alloc function via which this RollBall > protocol can be selected. > I'd think that mdio-i2c.c is for generic code. When adding a vendor-specific protocol, wouldn't it make sense to use a dedicated source file for it? > Signed-off-by: Marek Behún <kabel@kernel.org> > Cc: Andrew Lunn <andrew@lunn.ch> > Cc: Russell King <rmk+kernel@armlinux.org.uk> > --- > drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- > drivers/net/phy/sfp.c | 2 +- > include/linux/mdio/mdio-i2c.h | 8 +- > 3 files changed, 322 insertions(+), 7 deletions(-) > > diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c > index 09200a70b315..7be582c0891a 100644 > --- a/drivers/net/mdio/mdio-i2c.c > +++ b/drivers/net/mdio/mdio-i2c.c > @@ -3,6 +3,7 @@ > * MDIO I2C bridge > * > * Copyright (C) 2015-2016 Russell King > + * Copyright (C) 2021 Marek Behun > * > * Network PHYs can appear on I2C buses when they are part of SFP module. > * This driver exposes these PHYs to the networking PHY code, allowing > @@ -12,6 +13,7 @@ > #include <linux/i2c.h> > #include <linux/mdio/mdio-i2c.h> > #include <linux/phy.h> > +#include <linux/sfp.h> > > /* > * I2C bus addresses 0x50 and 0x51 are normally an EEPROM, which is > @@ -28,7 +30,7 @@ static unsigned int i2c_mii_phy_addr(int phy_id) > return phy_id + 0x40; > } > > -static int i2c_mii_read(struct mii_bus *bus, int phy_id, int reg) > +static int i2c_mii_read_default(struct mii_bus *bus, int phy_id, int reg) > { > struct i2c_adapter *i2c = bus->priv; > struct i2c_msg msgs[2]; > @@ -62,7 +64,8 @@ static int i2c_mii_read(struct mii_bus *bus, int phy_id, int reg) > return data[0] << 8 | data[1]; > } > > -static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > +static int i2c_mii_write_default(struct mii_bus *bus, int phy_id, int reg, > + u16 val) > { > struct i2c_adapter *i2c = bus->priv; > struct i2c_msg msg; > @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > return ret < 0 ? ret : 0; > } > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > + * instead via address 0x51, when SFP page is set to 0x03 and password to > + * 0xffffffff. > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > + * bus creation time, and expect it to remain set to 0x03 throughout the > + * lifetime of the module plugged into the system. If the SFP code starts > + * modifying SFP_PAGE in the future, this code will need to change. > + * > + * address size contents description > + * ------- ---- -------- ----------- > + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done > + * 0x81 1 DEV Clause 45 device > + * 0x82 2 REG Clause 45 register > + * 0x84 2 VAL Register value > + */ > +#define ROLLBALL_PHY_I2C_ADDR 0x51 > +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b > + > +#define ROLLBALL_CMD_ADDR 0x80 > +#define ROLLBALL_DATA_ADDR 0x81 > + > +#define ROLLBALL_CMD_WRITE 0x01 > +#define ROLLBALL_CMD_READ 0x02 > +#define ROLLBALL_CMD_DONE 0x04 > + > +#define SFP_PAGE_ROLLBALL_MDIO 3 > + > +static int __i2c_transfer_err(struct i2c_adapter *i2c, struct i2c_msg *msgs, > + int num) > +{ > + int ret; > + > + ret = __i2c_transfer(i2c, msgs, num); > + if (ret < 0) > + return ret; > + else if (ret != num) > + return -EIO; > + else > + return 0; > +} > + > +static int __i2c_rollball_get_page(struct i2c_adapter *i2c, int bus_addr, > + u8 *page) > +{ > + struct i2c_msg msgs[2]; > + u8 addr = SFP_PAGE; > + > + msgs[0].addr = bus_addr; > + msgs[0].flags = 0; > + msgs[0].len = 1; > + msgs[0].buf = &addr; > + > + msgs[1].addr = bus_addr; > + msgs[1].flags = I2C_M_RD; > + msgs[1].len = 1; > + msgs[1].buf = page; > + > + return __i2c_transfer_err(i2c, msgs, 2); > +} > + > +static int __i2c_rollball_set_page(struct i2c_adapter *i2c, int bus_addr, > + u8 page) > +{ > + struct i2c_msg msg; > + u8 buf[2]; > + > + buf[0] = SFP_PAGE; > + buf[1] = page; > + > + msg.addr = bus_addr; > + msg.flags = 0; > + msg.len = 2; > + msg.buf = buf; > + > + return __i2c_transfer_err(i2c, &msg, 1); > +} > + > +/* In order to not interfere with other SFP code (which possibly may manipulate > + * SFP_PAGE), for every transfer we do this: > + * 1. lock the bus > + * 2. save content of SFP_PAGE > + * 3. set SFP_PAGE to 3 > + * 4. do the transfer > + * 5. restore original SFP_PAGE > + * 6. unlock the bus > + * Note that one might think that steps 2 to 5 could be theoretically done all > + * in one call to i2c_transfer (by constructing msgs array in such a way), but > + * unfortunately tests show that this does not work :-( Changed SFP_PAGE does > + * not take into account until i2c_transfer() is done. > + */ > +static int i2c_transfer_rollball(struct i2c_adapter *i2c, > + struct i2c_msg *msgs, int num) > +{ > + u8 saved_page; > + int ret; > + > + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); > + > + /* save original page */ > + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); > + if (ret) > + goto unlock; > + > + /* change to RollBall MDIO page */ > + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); > + if (ret) > + goto unlock; > + > + /* do the transfer */ > + ret = __i2c_transfer_err(i2c, msgs, num); > + if (ret) > + goto unlock; > + > + /* restore original page */ > + ret = __i2c_rollball_set_page(i2c, msgs->addr, saved_page); > + > +unlock: > + i2c_unlock_bus(i2c, I2C_LOCK_SEGMENT); > + > + return ret; > +} > + > +static int i2c_rollball_mii_poll(struct mii_bus *bus, int bus_addr, u8 *buf, > + size_t len) > +{ > + struct i2c_adapter *i2c = bus->priv; > + struct i2c_msg msgs[2]; > + u8 cmd_addr, tmp, *res; > + int i, ret; > + > + cmd_addr = ROLLBALL_CMD_ADDR; > + > + res = buf ? buf : &tmp; > + len = buf ? len : 1; > + > + msgs[0].addr = bus_addr; > + msgs[0].flags = 0; > + msgs[0].len = 1; > + msgs[0].buf = &cmd_addr; > + > + msgs[1].addr = bus_addr; > + msgs[1].flags = I2C_M_RD; > + msgs[1].len = len; > + msgs[1].buf = res; > + > + /* By experiment it takes up to 70 ms to access a register for these > + * SFPs. Sleep 20ms between iteratios and try 10 times. > + */ > + i = 10; > + do { > + msleep(20); > + > + ret = i2c_transfer_rollball(i2c, msgs, ARRAY_SIZE(msgs)); > + if (ret) > + return ret; > + > + if (*res == ROLLBALL_CMD_DONE) > + return 0; > + } while (i-- > 0); > + > + dev_dbg(&bus->dev, "poll timed out\n"); > + > + return -ETIMEDOUT; > +} > + > +static int i2c_rollball_mii_cmd(struct mii_bus *bus, int bus_addr, u8 cmd, > + u8 *data, size_t len) > +{ > + struct i2c_adapter *i2c = bus->priv; > + struct i2c_msg msgs[2]; > + u8 cmdbuf[2]; > + > + cmdbuf[0] = ROLLBALL_CMD_ADDR; > + cmdbuf[1] = cmd; > + > + msgs[0].addr = bus_addr; > + msgs[0].flags = 0; > + msgs[0].len = len; > + msgs[0].buf = data; > + > + msgs[1].addr = bus_addr; > + msgs[1].flags = 0; > + msgs[1].len = sizeof(cmdbuf); > + msgs[1].buf = cmdbuf; > + > + return i2c_transfer_rollball(i2c, msgs, ARRAY_SIZE(msgs)); > +} > + > +static int i2c_mii_read_rollball(struct mii_bus *bus, int phy_id, int reg) > +{ > + u8 buf[4], res[6]; > + int bus_addr, ret; > + u16 val; > + > + if (!(reg & MII_ADDR_C45)) > + return -EOPNOTSUPP; > + > + bus_addr = i2c_mii_phy_addr(phy_id); > + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) > + return 0xffff; > + > + buf[0] = ROLLBALL_DATA_ADDR; > + buf[1] = (reg >> 16) & 0x1f; > + buf[2] = (reg >> 8) & 0xff; > + buf[3] = reg & 0xff; > + > + ret = i2c_rollball_mii_cmd(bus, bus_addr, ROLLBALL_CMD_READ, buf, > + sizeof(buf)); > + if (ret < 0) > + return ret; > + > + ret = i2c_rollball_mii_poll(bus, bus_addr, res, sizeof(res)); > + if (ret == -ETIMEDOUT) > + return 0xffff; > + else if (ret < 0) > + return ret; > + > + val = res[4] << 8 | res[5]; > + > + dev_dbg(&bus->dev, "read reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, > + reg & 0xffff, val); > + > + return val; > +} > + > +static int i2c_mii_write_rollball(struct mii_bus *bus, int phy_id, int reg, > + u16 val) > +{ > + int bus_addr, ret; > + u8 buf[6]; > + > + if (!(reg & MII_ADDR_C45)) > + return -EOPNOTSUPP; > + > + bus_addr = i2c_mii_phy_addr(phy_id); > + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) > + return 0; > + > + buf[0] = ROLLBALL_DATA_ADDR; > + buf[1] = (reg >> 16) & 0x1f; > + buf[2] = (reg >> 8) & 0xff; > + buf[3] = reg & 0xff; > + buf[4] = val >> 8; > + buf[5] = val & 0xff; > + > + ret = i2c_rollball_mii_cmd(bus, bus_addr, ROLLBALL_CMD_WRITE, buf, > + sizeof(buf)); > + if (ret < 0) > + return ret; > + > + ret = i2c_rollball_mii_poll(bus, bus_addr, NULL, 0); > + if (ret < 0) > + return ret; > + > + dev_dbg(&bus->dev, "write reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, > + reg & 0xffff, val); > + > + return 0; > +} > + > +static int i2c_mii_init_rollball(struct i2c_adapter *i2c) > +{ > + struct i2c_msg msg; > + u8 pw[5]; > + int ret; > + > + pw[0] = ROLLBALL_SFP_PASSWORD_ADDR; > + pw[1] = 0xff; > + pw[2] = 0xff; > + pw[3] = 0xff; > + pw[4] = 0xff; > + > + msg.addr = ROLLBALL_PHY_I2C_ADDR; > + msg.flags = 0; > + msg.len = sizeof(pw); > + msg.buf = pw; > + > + ret = i2c_transfer(i2c, &msg, 1); > + if (ret < 0) > + return ret; > + else if (ret != 1) > + return -EIO; > + else > + return 0; > +} > + > +struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c, > + enum mdio_i2c_proto protocol) > { > struct mii_bus *mii; > + int ret; > > if (!i2c_check_functionality(i2c, I2C_FUNC_I2C)) > return ERR_PTR(-EINVAL); > @@ -104,10 +395,28 @@ struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > > snprintf(mii->id, MII_BUS_ID_SIZE, "i2c:%s", dev_name(parent)); > mii->parent = parent; > - mii->read = i2c_mii_read; > - mii->write = i2c_mii_write; > mii->priv = i2c; > > + switch (protocol) { > + case MDIO_I2C_ROLLBALL: > + ret = i2c_mii_init_rollball(i2c); > + if (ret < 0) { > + dev_err(parent, > + "Cannot initialize RollBall MDIO I2C protocol: %d\n", > + ret); > + mdiobus_free(mii); > + return ERR_PTR(ret); > + } > + > + mii->read = i2c_mii_read_rollball; > + mii->write = i2c_mii_write_rollball; > + break; > + default: > + mii->read = i2c_mii_read_default; > + mii->write = i2c_mii_write_default; > + break; > + } > + > return mii; > } > EXPORT_SYMBOL_GPL(mdio_i2c_alloc); > diff --git a/drivers/net/phy/sfp.c b/drivers/net/phy/sfp.c > index 91d74c1a920a..958fd514a3b4 100644 > --- a/drivers/net/phy/sfp.c > +++ b/drivers/net/phy/sfp.c > @@ -419,7 +419,7 @@ static int sfp_i2c_configure(struct sfp *sfp, struct i2c_adapter *i2c) > sfp->read = sfp_i2c_read; > sfp->write = sfp_i2c_write; > > - i2c_mii = mdio_i2c_alloc(sfp->dev, i2c); > + i2c_mii = mdio_i2c_alloc(sfp->dev, i2c, MDIO_I2C_DEFAULT); > if (IS_ERR(i2c_mii)) > return PTR_ERR(i2c_mii); > > diff --git a/include/linux/mdio/mdio-i2c.h b/include/linux/mdio/mdio-i2c.h > index b1d27f7cd23f..53eedb0dc1d3 100644 > --- a/include/linux/mdio/mdio-i2c.h > +++ b/include/linux/mdio/mdio-i2c.h > @@ -11,6 +11,12 @@ struct device; > struct i2c_adapter; > struct mii_bus; > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c); > +enum mdio_i2c_proto { > + MDIO_I2C_DEFAULT, > + MDIO_I2C_ROLLBALL, > +}; > + > +struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c, > + enum mdio_i2c_proto protocol); > > #endif >
> I'd think that mdio-i2c.c is for generic code. When adding a > vendor-specific protocol, wouldn't it make sense to use a dedicated > source file for it? Hi Heiner There is no standardised way to access MDIO over i2c. So the existing code is vendor code, even if it is used by a few different vendors. Andrew
On 12.01.2021 15:02, Andrew Lunn wrote: >> I'd think that mdio-i2c.c is for generic code. When adding a >> vendor-specific protocol, wouldn't it make sense to use a dedicated >> source file for it? > > Hi Heiner > > There is no standardised way to access MDIO over i2c. So the existing > code is vendor code, even if it is used by a few different vendors. > > Andrew > OK, I see. Thanks, Andrew.
On Tue, 12 Jan 2021 15:02:49 +0100 Andrew Lunn <andrew@lunn.ch> wrote: > > I'd think that mdio-i2c.c is for generic code. When adding a > > vendor-specific protocol, wouldn't it make sense to use a dedicated > > source file for it? > > Hi Heiner > > There is no standardised way to access MDIO over i2c. So the existing > code is vendor code, even if it is used by a few different vendors. > > Andrew Andrew, if you could find the time, could you please review these patches? Patches 2 and 4 already have reviewed-by Russell, but 1 and 3 still need some review. Marek
On Tue, Jan 12, 2021 at 09:42:56AM +0100, Heiner Kallweit wrote: > On 11.01.2021 06:00, Marek Behún wrote: > > Some multigig SFPs from RollBall and Hilink do not expose functional > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > (although there seems to be read-only clause 22 access on this address). > > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > selected to 3 and the password must be filled with 0xff bytes for this > > PHY communication to work. > > > > This extends the mdio-i2c driver to support this protocol by adding a > > special parameter to mdio_i2c_alloc function via which this RollBall > > protocol can be selected. > > > I'd think that mdio-i2c.c is for generic code. When adding a > vendor-specific protocol, wouldn't it make sense to use a dedicated > source file for it? There is nothing in the SFP MSAs about how to access an on-board PHY on a SFP. This is something that vendors choose to do (or in some cases, not do.) mdio-i2c currently implements the access protocol for Marvell 88E1111 PHYs which have an I2C mode. It was extended to support the DM7052 module which has a Broadcom Clause 45 PHY and either a microcontroller or FPGA to convert I2C cycles to MDIO cycles - and sensibly performs clock stretching. There are modules that the clause 45 PHY is accessible via exactly the same I2C address, using exactly the same data accesses, but do not do the clock stretching thing, instead telling you that you must wait N microseconds between the bus transactions. We don't yet support these. Then there is Marek's module, where the PHY is accessible via a page in the diagnostic address - which is another entirely reasonable implementation. The solution we have here is one that I've worked with Marek for a few months now. I don't think we should grow lots of separate mdio-i2c-vendorfoo.c drivers - at least not yet. Maybe if we get lots of different access methods, it may make sense in the future.
> -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > + * instead via address 0x51, when SFP page is set to 0x03 and password to > + * 0xffffffff. > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > + * bus creation time, and expect it to remain set to 0x03 throughout the > + * lifetime of the module plugged into the system. If the SFP code starts > + * modifying SFP_PAGE in the future, this code will need to change. Just an FYI: This is likely to change. NVIDIA are working on generalizing the API ethtool -m uses. The new API should allow access to pages and banks. I've already said i will implement the API in the generic phylink SFP code. But no need to address this now. Andrew
> +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > + * instead via address 0x51, when SFP page is set to 0x03 and password to > + * 0xffffffff. > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > + * bus creation time, and expect it to remain set to 0x03 throughout the > + * lifetime of the module plugged into the system. If the SFP code starts > + * modifying SFP_PAGE in the future, this code will need to change. ... > +/* In order to not interfere with other SFP code (which possibly may manipulate > + * SFP_PAGE), for every transfer we do this: > + * 1. lock the bus > + * 2. save content of SFP_PAGE > + * 3. set SFP_PAGE to 3 > + * 4. do the transfer > + * 5. restore original SFP_PAGE > + * 6. unlock the bus These two comments seem to contradict each other? > +static int i2c_rollball_mii_poll(struct mii_bus *bus, int bus_addr, u8 *buf, > + size_t len) > +{ > + struct i2c_adapter *i2c = bus->priv; > + struct i2c_msg msgs[2]; > + u8 cmd_addr, tmp, *res; > + int i, ret; > + > + cmd_addr = ROLLBALL_CMD_ADDR; > + > + res = buf ? buf : &tmp; > + len = buf ? len : 1; > + > + msgs[0].addr = bus_addr; > + msgs[0].flags = 0; > + msgs[0].len = 1; > + msgs[0].buf = &cmd_addr; > + > + msgs[1].addr = bus_addr; > + msgs[1].flags = I2C_M_RD; > + msgs[1].len = len; > + msgs[1].buf = res; > + > + /* By experiment it takes up to 70 ms to access a register for these > + * SFPs. Sleep 20ms between iteratios and try 10 times. > + */ iterations > +static int i2c_mii_read_rollball(struct mii_bus *bus, int phy_id, int reg) > +{ > + u8 buf[4], res[6]; > + int bus_addr, ret; > + u16 val; > + > + if (!(reg & MII_ADDR_C45)) > + return -EOPNOTSUPP; > + > + bus_addr = i2c_mii_phy_addr(phy_id); > + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) > + return 0xffff; > + > + buf[0] = ROLLBALL_DATA_ADDR; > + buf[1] = (reg >> 16) & 0x1f; > + buf[2] = (reg >> 8) & 0xff; > + buf[3] = reg & 0xff; This looks odd. There are only 32 registers for C22 transactions, so it fits in one byte. You can set buf[1] and buf[2] to zero. C45 transactions allow for 65536 registers, and has the devtype field, so would need 3 bytes. Which suggests that C45 might actually be supported? At least by the protocol, if not the device. > + dev_dbg(&bus->dev, "read reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, > + reg & 0xffff, val); There is a tracepoint that allows access to this information, so you can probably remove this. Andrew
On Tue, 12 Jan 2021 21:43:29 +0100 Andrew Lunn <andrew@lunn.ch> wrote: > > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > > + * instead via address 0x51, when SFP page is set to 0x03 and password to > > + * 0xffffffff. > > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > > + * bus creation time, and expect it to remain set to 0x03 throughout the > > + * lifetime of the module plugged into the system. If the SFP code starts > > + * modifying SFP_PAGE in the future, this code will need to change. > > ... > > > +/* In order to not interfere with other SFP code (which possibly may manipulate > > + * SFP_PAGE), for every transfer we do this: > > + * 1. lock the bus > > + * 2. save content of SFP_PAGE > > + * 3. set SFP_PAGE to 3 > > + * 4. do the transfer > > + * 5. restore original SFP_PAGE > > + * 6. unlock the bus > > These two comments seem to contradict each other? I forgot about the first comment. I will rewrite it and send again. > > > +static int i2c_rollball_mii_poll(struct mii_bus *bus, int bus_addr, u8 *buf, > > + size_t len) > > +{ > > + struct i2c_adapter *i2c = bus->priv; > > + struct i2c_msg msgs[2]; > > + u8 cmd_addr, tmp, *res; > > + int i, ret; > > + > > + cmd_addr = ROLLBALL_CMD_ADDR; > > + > > + res = buf ? buf : &tmp; > > + len = buf ? len : 1; > > + > > + msgs[0].addr = bus_addr; > > + msgs[0].flags = 0; > > + msgs[0].len = 1; > > + msgs[0].buf = &cmd_addr; > > + > > + msgs[1].addr = bus_addr; > > + msgs[1].flags = I2C_M_RD; > > + msgs[1].len = len; > > + msgs[1].buf = res; > > + > > + /* By experiment it takes up to 70 ms to access a register for these > > + * SFPs. Sleep 20ms between iteratios and try 10 times. > > + */ > > iterations THX. > > > +static int i2c_mii_read_rollball(struct mii_bus *bus, int phy_id, int reg) > > +{ > > + u8 buf[4], res[6]; > > + int bus_addr, ret; > > + u16 val; > > + > > + if (!(reg & MII_ADDR_C45)) > > + return -EOPNOTSUPP; > > + > > + bus_addr = i2c_mii_phy_addr(phy_id); > > + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) > > + return 0xffff; > > + > > + buf[0] = ROLLBALL_DATA_ADDR; > > + buf[1] = (reg >> 16) & 0x1f; > > + buf[2] = (reg >> 8) & 0xff; > > + buf[3] = reg & 0xff; > > This looks odd. There are only 32 registers for C22 transactions, so > it fits in one byte. You can set buf[1] and buf[2] to zero. C22 is not supported by this protocol. A few line above there is if (!(reg & MII_ADDR_C45)) return -EOPNOTSUPP; > C45 transactions allow for 65536 registers, and has the devtype field, > so would need 3 bytes. Which suggests that C45 might actually be > supported? At least by the protocol, if not the device. > > + dev_dbg(&bus->dev, "read reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, > > + reg & 0xffff, val); > > There is a tracepoint that allows access to this information, so you > can probably remove this. > > Andrew OK then, I will remove this. Thanks, Andrew.
> +static int i2c_transfer_rollball(struct i2c_adapter *i2c, > + struct i2c_msg *msgs, int num) > +{ > + u8 saved_page; > + int ret; > + > + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); > + > + /* save original page */ > + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); > + if (ret) > + goto unlock; > + > + /* change to RollBall MDIO page */ > + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); > + if (ret) > + goto unlock; > + > + /* do the transfer */ > + ret = __i2c_transfer_err(i2c, msgs, num); > + if (ret) > + goto unlock; If get page and set page worked, and you get an error in during the actual data transfer, i wonder if you should try restoring the page before returning with the error? > + > + /* restore original page */ > + ret = __i2c_rollball_set_page(i2c, msgs->addr, saved_page); > + > +unlock: > + i2c_unlock_bus(i2c, I2C_LOCK_SEGMENT); > + > + return ret; > +}
> > > +static int i2c_mii_read_rollball(struct mii_bus *bus, int phy_id, int reg) > > > +{ > > > + u8 buf[4], res[6]; > > > + int bus_addr, ret; > > > + u16 val; > > > + > > > + if (!(reg & MII_ADDR_C45)) > > > + return -EOPNOTSUPP; > > > + > > > + bus_addr = i2c_mii_phy_addr(phy_id); > > > + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) > > > + return 0xffff; > > > + > > > + buf[0] = ROLLBALL_DATA_ADDR; > > > + buf[1] = (reg >> 16) & 0x1f; > > > + buf[2] = (reg >> 8) & 0xff; > > > + buf[3] = reg & 0xff; > > > > This looks odd. There are only 32 registers for C22 transactions, so > > it fits in one byte. You can set buf[1] and buf[2] to zero. > > C22 is not supported by this protocol. Duh! Sorry for the noise. Andrew
On Tue, 12 Jan 2021 21:54:24 +0100 Andrew Lunn <andrew@lunn.ch> wrote: > > +static int i2c_transfer_rollball(struct i2c_adapter *i2c, > > + struct i2c_msg *msgs, int num) > > +{ > > + u8 saved_page; > > + int ret; > > + > > + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); > > + > > + /* save original page */ > > + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); > > + if (ret) > > + goto unlock; > > + > > + /* change to RollBall MDIO page */ > > + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); > > + if (ret) > > + goto unlock; > > + > > + /* do the transfer */ > > + ret = __i2c_transfer_err(i2c, msgs, num); > > + if (ret) > > + goto unlock; > > If get page and set page worked, and you get an error in during the > actual data transfer, i wonder if you should try restoring the page > before returning with the error? I don't know. Can i2c trasfer fail and the next one succeed?
On Tue, Jan 12, 2021 at 09:54:24PM +0100, Andrew Lunn wrote: > > +static int i2c_transfer_rollball(struct i2c_adapter *i2c, > > + struct i2c_msg *msgs, int num) > > +{ > > + u8 saved_page; > > + int ret; > > + > > + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); > > + > > + /* save original page */ > > + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); > > + if (ret) > > + goto unlock; > > + > > + /* change to RollBall MDIO page */ > > + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); > > + if (ret) > > + goto unlock; > > + > > + /* do the transfer */ > > + ret = __i2c_transfer_err(i2c, msgs, num); > > + if (ret) > > + goto unlock; > > If get page and set page worked, and you get an error in during the > actual data transfer, i wonder if you should try restoring the page > before returning with the error? That's what we do with paged PHYs - if the access encounters an error, we still attempt to restore the page and propagate the _original_ error. We would only propagate the error from the page restore if it was the only error. See the logic and comments for phy_restore_page(). Note that phy_(save|select)_page()..phy_restore_page() deal with the locking, which is why they always have to be paired (also means that the user doesn't have to think too hard about error handling.)
On Tuesday 12 January 2021 22:01:14 Marek Behún wrote: > On Tue, 12 Jan 2021 21:54:24 +0100 > Andrew Lunn <andrew@lunn.ch> wrote: > > > > +static int i2c_transfer_rollball(struct i2c_adapter *i2c, > > > + struct i2c_msg *msgs, int num) > > > +{ > > > + u8 saved_page; > > > + int ret; > > > + > > > + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); > > > + > > > + /* save original page */ > > > + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); > > > + if (ret) > > > + goto unlock; > > > + > > > + /* change to RollBall MDIO page */ > > > + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); > > > + if (ret) > > > + goto unlock; > > > + > > > + /* do the transfer */ > > > + ret = __i2c_transfer_err(i2c, msgs, num); > > > + if (ret) > > > + goto unlock; > > > > If get page and set page worked, and you get an error in during the > > actual data transfer, i wonder if you should try restoring the page > > before returning with the error? > > I don't know. Can i2c trasfer fail and the next one succeed? I guess it can depend also on i2c driver. I'm for trying to restore previous i2c page on failure. In the worst case also restoring fails...
Hello! See my comments below. On Monday 11 January 2021 06:00:41 Marek Behún wrote: > Some multigig SFPs from RollBall and Hilink do not expose functional > MDIO access to the internal PHY of the SFP via I2C address 0x56 > (although there seems to be read-only clause 22 access on this address). Maybe it could be interesting to test if clause 22 access via i2c address 0x56 can work also in write mode after setting rollball password... > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > selected to 3 and the password must be filled with 0xff bytes for this > PHY communication to work. > > This extends the mdio-i2c driver to support this protocol by adding a > special parameter to mdio_i2c_alloc function via which this RollBall > protocol can be selected. > > Signed-off-by: Marek Behún <kabel@kernel.org> > Cc: Andrew Lunn <andrew@lunn.ch> > Cc: Russell King <rmk+kernel@armlinux.org.uk> > --- > drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- > drivers/net/phy/sfp.c | 2 +- > include/linux/mdio/mdio-i2c.h | 8 +- > 3 files changed, 322 insertions(+), 7 deletions(-) > > diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c > index 09200a70b315..7be582c0891a 100644 > --- a/drivers/net/mdio/mdio-i2c.c > +++ b/drivers/net/mdio/mdio-i2c.c ... > @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > return ret < 0 ? ret : 0; > } > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > + * instead via address 0x51, when SFP page is set to 0x03 and password to > + * 0xffffffff. > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > + * bus creation time, and expect it to remain set to 0x03 throughout the > + * lifetime of the module plugged into the system. If the SFP code starts > + * modifying SFP_PAGE in the future, this code will need to change. > + * > + * address size contents description > + * ------- ---- -------- ----------- > + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done > + * 0x81 1 DEV Clause 45 device > + * 0x82 2 REG Clause 45 register > + * 0x84 2 VAL Register value > + */ > +#define ROLLBALL_PHY_I2C_ADDR 0x51 > +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b It looks like that this password entry is standard field described in QSFP+ specifications SFF-8436 and SFF-8636. Should not it be rather named vendor-neutral (as it is not Rollball specific)? And maybe defined in include/linux/sfp.h file where also also other standard macros, like SFP_PAGE macro? > + > +#define ROLLBALL_CMD_ADDR 0x80 > +#define ROLLBALL_DATA_ADDR 0x81 > + > +#define ROLLBALL_CMD_WRITE 0x01 > +#define ROLLBALL_CMD_READ 0x02 > +#define ROLLBALL_CMD_DONE 0x04 > + > +#define SFP_PAGE_ROLLBALL_MDIO 3 ... > diff --git a/include/linux/mdio/mdio-i2c.h b/include/linux/mdio/mdio-i2c.h > index b1d27f7cd23f..53eedb0dc1d3 100644 > --- a/include/linux/mdio/mdio-i2c.h > +++ b/include/linux/mdio/mdio-i2c.h > @@ -11,6 +11,12 @@ struct device; > struct i2c_adapter; > struct mii_bus; > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c); > +enum mdio_i2c_proto { > + MDIO_I2C_DEFAULT, Russel wrote that current mdio i2c (default) access method is also vendor specific for Marvell chips. Should not we rather call it MDIO_I2C_MARVELL if both methods are vendor/chip specific? > + MDIO_I2C_ROLLBALL, > +}; > + > +struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c, > + enum mdio_i2c_proto protocol); > > #endif > -- > 2.26.2 >
On Wed, Jan 13, 2021 at 12:22:04PM +0100, Pali Rohár wrote: > Hello! See my comments below. > > On Monday 11 January 2021 06:00:41 Marek Behún wrote: > > Some multigig SFPs from RollBall and Hilink do not expose functional > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > (although there seems to be read-only clause 22 access on this address). > > Maybe it could be interesting to test if clause 22 access via i2c > address 0x56 can work also in write mode after setting rollball > password... > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > selected to 3 and the password must be filled with 0xff bytes for this > > PHY communication to work. > > > > This extends the mdio-i2c driver to support this protocol by adding a > > special parameter to mdio_i2c_alloc function via which this RollBall > > protocol can be selected. > > > > Signed-off-by: Marek Behún <kabel@kernel.org> > > Cc: Andrew Lunn <andrew@lunn.ch> > > Cc: Russell King <rmk+kernel@armlinux.org.uk> > > --- > > drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- > > drivers/net/phy/sfp.c | 2 +- > > include/linux/mdio/mdio-i2c.h | 8 +- > > 3 files changed, 322 insertions(+), 7 deletions(-) > > > > diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c > > index 09200a70b315..7be582c0891a 100644 > > --- a/drivers/net/mdio/mdio-i2c.c > > +++ b/drivers/net/mdio/mdio-i2c.c > ... > > @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > > return ret < 0 ? ret : 0; > > } > > > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > > + * instead via address 0x51, when SFP page is set to 0x03 and password to > > + * 0xffffffff. > > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > > + * bus creation time, and expect it to remain set to 0x03 throughout the > > + * lifetime of the module plugged into the system. If the SFP code starts > > + * modifying SFP_PAGE in the future, this code will need to change. > > + * > > + * address size contents description > > + * ------- ---- -------- ----------- > > + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done > > + * 0x81 1 DEV Clause 45 device > > + * 0x82 2 REG Clause 45 register > > + * 0x84 2 VAL Register value > > + */ > > +#define ROLLBALL_PHY_I2C_ADDR 0x51 > > +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b > > It looks like that this password entry is standard field described in > QSFP+ specifications SFF-8436 and SFF-8636. Should not it be rather > named vendor-neutral (as it is not Rollball specific)? And maybe defined > in include/linux/sfp.h file where also also other standard macros, like > SFP_PAGE macro? If it is generic, the functions themselves should probably move into sfp.c. Being able to swap pages is something needed for accessing the diagnostic registers under some conditions. Because we don't have support for changing the page, the HWMON support is disabled in this condition. Andrew
On Wednesday 13 January 2021 14:56:13 Andrew Lunn wrote: > On Wed, Jan 13, 2021 at 12:22:04PM +0100, Pali Rohár wrote: > > Hello! See my comments below. > > > > On Monday 11 January 2021 06:00:41 Marek Behún wrote: > > > Some multigig SFPs from RollBall and Hilink do not expose functional > > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > > (although there seems to be read-only clause 22 access on this address). > > > > Maybe it could be interesting to test if clause 22 access via i2c > > address 0x56 can work also in write mode after setting rollball > > password... > > > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > > selected to 3 and the password must be filled with 0xff bytes for this > > > PHY communication to work. > > > > > > This extends the mdio-i2c driver to support this protocol by adding a > > > special parameter to mdio_i2c_alloc function via which this RollBall > > > protocol can be selected. > > > > > > Signed-off-by: Marek Behún <kabel@kernel.org> > > > Cc: Andrew Lunn <andrew@lunn.ch> > > > Cc: Russell King <rmk+kernel@armlinux.org.uk> > > > --- > > > drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- > > > drivers/net/phy/sfp.c | 2 +- > > > include/linux/mdio/mdio-i2c.h | 8 +- > > > 3 files changed, 322 insertions(+), 7 deletions(-) > > > > > > diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c > > > index 09200a70b315..7be582c0891a 100644 > > > --- a/drivers/net/mdio/mdio-i2c.c > > > +++ b/drivers/net/mdio/mdio-i2c.c > > ... > > > @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > > > return ret < 0 ? ret : 0; > > > } > > > > > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > > > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > > > + * instead via address 0x51, when SFP page is set to 0x03 and password to > > > + * 0xffffffff. > > > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > > > + * bus creation time, and expect it to remain set to 0x03 throughout the > > > + * lifetime of the module plugged into the system. If the SFP code starts > > > + * modifying SFP_PAGE in the future, this code will need to change. > > > + * > > > + * address size contents description > > > + * ------- ---- -------- ----------- > > > + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done > > > + * 0x81 1 DEV Clause 45 device > > > + * 0x82 2 REG Clause 45 register > > > + * 0x84 2 VAL Register value > > > + */ > > > +#define ROLLBALL_PHY_I2C_ADDR 0x51 > > > +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b > > > > It looks like that this password entry is standard field described in > > QSFP+ specifications SFF-8436 and SFF-8636. Should not it be rather > > named vendor-neutral (as it is not Rollball specific)? And maybe defined > > in include/linux/sfp.h file where also also other standard macros, like > > SFP_PAGE macro? > > If it is generic, the functions themselves should probably move into > sfp.c. Being able to swap pages is something needed for accessing the > diagnostic registers under some conditions. Because we don't have > support for changing the page, the HWMON support is disabled in this > condition. Only password is described in SFF-8436 and SFF-8636, nothing more.
On Wed, Jan 13, 2021 at 02:56:13PM +0100, Andrew Lunn wrote: > On Wed, Jan 13, 2021 at 12:22:04PM +0100, Pali Rohár wrote: > > Hello! See my comments below. > > > > On Monday 11 January 2021 06:00:41 Marek Behún wrote: > > > Some multigig SFPs from RollBall and Hilink do not expose functional > > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > > (although there seems to be read-only clause 22 access on this address). > > > > Maybe it could be interesting to test if clause 22 access via i2c > > address 0x56 can work also in write mode after setting rollball > > password... > > > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > > selected to 3 and the password must be filled with 0xff bytes for this > > > PHY communication to work. > > > > > > This extends the mdio-i2c driver to support this protocol by adding a > > > special parameter to mdio_i2c_alloc function via which this RollBall > > > protocol can be selected. > > > > > > Signed-off-by: Marek Behún <kabel@kernel.org> > > > Cc: Andrew Lunn <andrew@lunn.ch> > > > Cc: Russell King <rmk+kernel@armlinux.org.uk> > > > --- > > > drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- > > > drivers/net/phy/sfp.c | 2 +- > > > include/linux/mdio/mdio-i2c.h | 8 +- > > > 3 files changed, 322 insertions(+), 7 deletions(-) > > > > > > diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c > > > index 09200a70b315..7be582c0891a 100644 > > > --- a/drivers/net/mdio/mdio-i2c.c > > > +++ b/drivers/net/mdio/mdio-i2c.c > > ... > > > @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) > > > return ret < 0 ? ret : 0; > > > } > > > > > > -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) > > > +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but > > > + * instead via address 0x51, when SFP page is set to 0x03 and password to > > > + * 0xffffffff. > > > + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at > > > + * bus creation time, and expect it to remain set to 0x03 throughout the > > > + * lifetime of the module plugged into the system. If the SFP code starts > > > + * modifying SFP_PAGE in the future, this code will need to change. > > > + * > > > + * address size contents description > > > + * ------- ---- -------- ----------- > > > + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done > > > + * 0x81 1 DEV Clause 45 device > > > + * 0x82 2 REG Clause 45 register > > > + * 0x84 2 VAL Register value > > > + */ > > > +#define ROLLBALL_PHY_I2C_ADDR 0x51 > > > +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b > > > > It looks like that this password entry is standard field described in > > QSFP+ specifications SFF-8436 and SFF-8636. Should not it be rather > > named vendor-neutral (as it is not Rollball specific)? And maybe defined > > in include/linux/sfp.h file where also also other standard macros, like > > SFP_PAGE macro? > > If it is generic, the functions themselves should probably move into > sfp.c. Being able to swap pages is something needed for accessing the > diagnostic registers under some conditions. Because we don't have > support for changing the page, the HWMON support is disabled in this > condition. QSFPs are quite different from SFPs. - the data structures in the "I2C device" are quite different. - they don't have as many controls signals either - signals such as RX_LOS, TX_DISABLE etc are handled per channel and are only exposed via the I2C bus. I've some rudimentary code for QSFP support in my CEX7 branch, which is currently entirely separate from the SFP code. I haven't done much on that for quite some time due to the historical difficulties of working with NXP, but I have over the course of the last week updated my branch to 5.10 and have patches to move SFP support forward. However, I expect further difficulties interacting with NXP and fully expect to be blocked from being able to achieve anything useful.
On Tuesday 12 January 2021 19:22:33 Russell King - ARM Linux admin wrote: > On Tue, Jan 12, 2021 at 09:42:56AM +0100, Heiner Kallweit wrote: > > On 11.01.2021 06:00, Marek Behún wrote: > > > Some multigig SFPs from RollBall and Hilink do not expose functional > > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > > (although there seems to be read-only clause 22 access on this address). > > > > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > > selected to 3 and the password must be filled with 0xff bytes for this > > > PHY communication to work. > > > > > > This extends the mdio-i2c driver to support this protocol by adding a > > > special parameter to mdio_i2c_alloc function via which this RollBall > > > protocol can be selected. > > > > > I'd think that mdio-i2c.c is for generic code. When adding a > > vendor-specific protocol, wouldn't it make sense to use a dedicated > > source file for it? > > There is nothing in the SFP MSAs about how to access an on-board PHY > on a SFP. This is something that vendors choose to do (or in some > cases, not do.) > > mdio-i2c currently implements the access protocol for Marvell 88E1111 > PHYs which have an I2C mode. It was extended to support the DM7052 > module which has a Broadcom Clause 45 PHY and either a microcontroller > or FPGA to convert I2C cycles to MDIO cycles - and sensibly performs > clock stretching. Just to note that there is also another Marvell way how to access Clause 45 registers via existing Marvell PHY Clause 22 I2C mode: https://www.ieee802.org/3/efm/public/nov02/oam/pannell_oam_1_1102.pdf Clause 45 registers are accessed via Clause 22 regs 13 and 14. > There are modules that the clause 45 PHY is accessible via exactly the > same I2C address, using exactly the same data accesses, but do not > do the clock stretching thing, instead telling you that you must wait > N microseconds between the bus transactions. We don't yet support > these. > > Then there is Marek's module, where the PHY is accessible via a page > in the diagnostic address - which is another entirely reasonable > implementation. The solution we have here is one that I've worked with > Marek for a few months now. > > I don't think we should grow lots of separate mdio-i2c-vendorfoo.c > drivers - at least not yet. Maybe if we get lots of different access > methods, it may make sense in the future. > > -- > RMK's Patch system: https://www.armlinux.org.uk/developer/patches/ > FTTP is here! 40Mbps down 10Mbps up. Decent connectivity at last!
On Mon, Jan 18, 2021 at 01:13:38PM +0100, Pali Rohár wrote: > On Tuesday 12 January 2021 19:22:33 Russell King - ARM Linux admin wrote: > > On Tue, Jan 12, 2021 at 09:42:56AM +0100, Heiner Kallweit wrote: > > > On 11.01.2021 06:00, Marek Behún wrote: > > > > Some multigig SFPs from RollBall and Hilink do not expose functional > > > > MDIO access to the internal PHY of the SFP via I2C address 0x56 > > > > (although there seems to be read-only clause 22 access on this address). > > > > > > > > Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced > > > > Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be > > > > selected to 3 and the password must be filled with 0xff bytes for this > > > > PHY communication to work. > > > > > > > > This extends the mdio-i2c driver to support this protocol by adding a > > > > special parameter to mdio_i2c_alloc function via which this RollBall > > > > protocol can be selected. > > > > > > > I'd think that mdio-i2c.c is for generic code. When adding a > > > vendor-specific protocol, wouldn't it make sense to use a dedicated > > > source file for it? > > > > There is nothing in the SFP MSAs about how to access an on-board PHY > > on a SFP. This is something that vendors choose to do (or in some > > cases, not do.) > > > > mdio-i2c currently implements the access protocol for Marvell 88E1111 > > PHYs which have an I2C mode. It was extended to support the DM7052 > > module which has a Broadcom Clause 45 PHY and either a microcontroller > > or FPGA to convert I2C cycles to MDIO cycles - and sensibly performs > > clock stretching. > > Just to note that there is also another Marvell way how to access Clause > 45 registers via existing Marvell PHY Clause 22 I2C mode: > > https://www.ieee802.org/3/efm/public/nov02/oam/pannell_oam_1_1102.pdf > > Clause 45 registers are accessed via Clause 22 regs 13 and 14. Only if the Clause 45 PHY supports the Clause 22 interface. Not all do. However, which access method really only depends on the SFP module, and how they interface the PHY to the I2C bus. As I've previously mentioned, using a wrong access method will be misinterpreted by the SFP. Even if the PHY supports both Clause 22 and Clause 45 accesses, if the I2C-to-MDIO layer in the SFP only supports Clause 45, then Clause 22 accesses will not work. As I've previously said, a Clause 45 read access is indistinguishable from a Clause 22 write access on the I2C bus.
diff --git a/drivers/net/mdio/mdio-i2c.c b/drivers/net/mdio/mdio-i2c.c index 09200a70b315..7be582c0891a 100644 --- a/drivers/net/mdio/mdio-i2c.c +++ b/drivers/net/mdio/mdio-i2c.c @@ -3,6 +3,7 @@ * MDIO I2C bridge * * Copyright (C) 2015-2016 Russell King + * Copyright (C) 2021 Marek Behun * * Network PHYs can appear on I2C buses when they are part of SFP module. * This driver exposes these PHYs to the networking PHY code, allowing @@ -12,6 +13,7 @@ #include <linux/i2c.h> #include <linux/mdio/mdio-i2c.h> #include <linux/phy.h> +#include <linux/sfp.h> /* * I2C bus addresses 0x50 and 0x51 are normally an EEPROM, which is @@ -28,7 +30,7 @@ static unsigned int i2c_mii_phy_addr(int phy_id) return phy_id + 0x40; } -static int i2c_mii_read(struct mii_bus *bus, int phy_id, int reg) +static int i2c_mii_read_default(struct mii_bus *bus, int phy_id, int reg) { struct i2c_adapter *i2c = bus->priv; struct i2c_msg msgs[2]; @@ -62,7 +64,8 @@ static int i2c_mii_read(struct mii_bus *bus, int phy_id, int reg) return data[0] << 8 | data[1]; } -static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) +static int i2c_mii_write_default(struct mii_bus *bus, int phy_id, int reg, + u16 val) { struct i2c_adapter *i2c = bus->priv; struct i2c_msg msg; @@ -91,9 +94,297 @@ static int i2c_mii_write(struct mii_bus *bus, int phy_id, int reg, u16 val) return ret < 0 ? ret : 0; } -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) +/* RollBall SFPs do not access internal PHY via I2C address 0x56, but + * instead via address 0x51, when SFP page is set to 0x03 and password to + * 0xffffffff. + * Since current SFP code does not modify SFP_PAGE, we set it to 0x03 only at + * bus creation time, and expect it to remain set to 0x03 throughout the + * lifetime of the module plugged into the system. If the SFP code starts + * modifying SFP_PAGE in the future, this code will need to change. + * + * address size contents description + * ------- ---- -------- ----------- + * 0x80 1 CMD 0x01/0x02/0x04 for write/read/done + * 0x81 1 DEV Clause 45 device + * 0x82 2 REG Clause 45 register + * 0x84 2 VAL Register value + */ +#define ROLLBALL_PHY_I2C_ADDR 0x51 +#define ROLLBALL_SFP_PASSWORD_ADDR 0x7b + +#define ROLLBALL_CMD_ADDR 0x80 +#define ROLLBALL_DATA_ADDR 0x81 + +#define ROLLBALL_CMD_WRITE 0x01 +#define ROLLBALL_CMD_READ 0x02 +#define ROLLBALL_CMD_DONE 0x04 + +#define SFP_PAGE_ROLLBALL_MDIO 3 + +static int __i2c_transfer_err(struct i2c_adapter *i2c, struct i2c_msg *msgs, + int num) +{ + int ret; + + ret = __i2c_transfer(i2c, msgs, num); + if (ret < 0) + return ret; + else if (ret != num) + return -EIO; + else + return 0; +} + +static int __i2c_rollball_get_page(struct i2c_adapter *i2c, int bus_addr, + u8 *page) +{ + struct i2c_msg msgs[2]; + u8 addr = SFP_PAGE; + + msgs[0].addr = bus_addr; + msgs[0].flags = 0; + msgs[0].len = 1; + msgs[0].buf = &addr; + + msgs[1].addr = bus_addr; + msgs[1].flags = I2C_M_RD; + msgs[1].len = 1; + msgs[1].buf = page; + + return __i2c_transfer_err(i2c, msgs, 2); +} + +static int __i2c_rollball_set_page(struct i2c_adapter *i2c, int bus_addr, + u8 page) +{ + struct i2c_msg msg; + u8 buf[2]; + + buf[0] = SFP_PAGE; + buf[1] = page; + + msg.addr = bus_addr; + msg.flags = 0; + msg.len = 2; + msg.buf = buf; + + return __i2c_transfer_err(i2c, &msg, 1); +} + +/* In order to not interfere with other SFP code (which possibly may manipulate + * SFP_PAGE), for every transfer we do this: + * 1. lock the bus + * 2. save content of SFP_PAGE + * 3. set SFP_PAGE to 3 + * 4. do the transfer + * 5. restore original SFP_PAGE + * 6. unlock the bus + * Note that one might think that steps 2 to 5 could be theoretically done all + * in one call to i2c_transfer (by constructing msgs array in such a way), but + * unfortunately tests show that this does not work :-( Changed SFP_PAGE does + * not take into account until i2c_transfer() is done. + */ +static int i2c_transfer_rollball(struct i2c_adapter *i2c, + struct i2c_msg *msgs, int num) +{ + u8 saved_page; + int ret; + + i2c_lock_bus(i2c, I2C_LOCK_SEGMENT); + + /* save original page */ + ret = __i2c_rollball_get_page(i2c, msgs->addr, &saved_page); + if (ret) + goto unlock; + + /* change to RollBall MDIO page */ + ret = __i2c_rollball_set_page(i2c, msgs->addr, SFP_PAGE_ROLLBALL_MDIO); + if (ret) + goto unlock; + + /* do the transfer */ + ret = __i2c_transfer_err(i2c, msgs, num); + if (ret) + goto unlock; + + /* restore original page */ + ret = __i2c_rollball_set_page(i2c, msgs->addr, saved_page); + +unlock: + i2c_unlock_bus(i2c, I2C_LOCK_SEGMENT); + + return ret; +} + +static int i2c_rollball_mii_poll(struct mii_bus *bus, int bus_addr, u8 *buf, + size_t len) +{ + struct i2c_adapter *i2c = bus->priv; + struct i2c_msg msgs[2]; + u8 cmd_addr, tmp, *res; + int i, ret; + + cmd_addr = ROLLBALL_CMD_ADDR; + + res = buf ? buf : &tmp; + len = buf ? len : 1; + + msgs[0].addr = bus_addr; + msgs[0].flags = 0; + msgs[0].len = 1; + msgs[0].buf = &cmd_addr; + + msgs[1].addr = bus_addr; + msgs[1].flags = I2C_M_RD; + msgs[1].len = len; + msgs[1].buf = res; + + /* By experiment it takes up to 70 ms to access a register for these + * SFPs. Sleep 20ms between iteratios and try 10 times. + */ + i = 10; + do { + msleep(20); + + ret = i2c_transfer_rollball(i2c, msgs, ARRAY_SIZE(msgs)); + if (ret) + return ret; + + if (*res == ROLLBALL_CMD_DONE) + return 0; + } while (i-- > 0); + + dev_dbg(&bus->dev, "poll timed out\n"); + + return -ETIMEDOUT; +} + +static int i2c_rollball_mii_cmd(struct mii_bus *bus, int bus_addr, u8 cmd, + u8 *data, size_t len) +{ + struct i2c_adapter *i2c = bus->priv; + struct i2c_msg msgs[2]; + u8 cmdbuf[2]; + + cmdbuf[0] = ROLLBALL_CMD_ADDR; + cmdbuf[1] = cmd; + + msgs[0].addr = bus_addr; + msgs[0].flags = 0; + msgs[0].len = len; + msgs[0].buf = data; + + msgs[1].addr = bus_addr; + msgs[1].flags = 0; + msgs[1].len = sizeof(cmdbuf); + msgs[1].buf = cmdbuf; + + return i2c_transfer_rollball(i2c, msgs, ARRAY_SIZE(msgs)); +} + +static int i2c_mii_read_rollball(struct mii_bus *bus, int phy_id, int reg) +{ + u8 buf[4], res[6]; + int bus_addr, ret; + u16 val; + + if (!(reg & MII_ADDR_C45)) + return -EOPNOTSUPP; + + bus_addr = i2c_mii_phy_addr(phy_id); + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) + return 0xffff; + + buf[0] = ROLLBALL_DATA_ADDR; + buf[1] = (reg >> 16) & 0x1f; + buf[2] = (reg >> 8) & 0xff; + buf[3] = reg & 0xff; + + ret = i2c_rollball_mii_cmd(bus, bus_addr, ROLLBALL_CMD_READ, buf, + sizeof(buf)); + if (ret < 0) + return ret; + + ret = i2c_rollball_mii_poll(bus, bus_addr, res, sizeof(res)); + if (ret == -ETIMEDOUT) + return 0xffff; + else if (ret < 0) + return ret; + + val = res[4] << 8 | res[5]; + + dev_dbg(&bus->dev, "read reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, + reg & 0xffff, val); + + return val; +} + +static int i2c_mii_write_rollball(struct mii_bus *bus, int phy_id, int reg, + u16 val) +{ + int bus_addr, ret; + u8 buf[6]; + + if (!(reg & MII_ADDR_C45)) + return -EOPNOTSUPP; + + bus_addr = i2c_mii_phy_addr(phy_id); + if (bus_addr != ROLLBALL_PHY_I2C_ADDR) + return 0; + + buf[0] = ROLLBALL_DATA_ADDR; + buf[1] = (reg >> 16) & 0x1f; + buf[2] = (reg >> 8) & 0xff; + buf[3] = reg & 0xff; + buf[4] = val >> 8; + buf[5] = val & 0xff; + + ret = i2c_rollball_mii_cmd(bus, bus_addr, ROLLBALL_CMD_WRITE, buf, + sizeof(buf)); + if (ret < 0) + return ret; + + ret = i2c_rollball_mii_poll(bus, bus_addr, NULL, 0); + if (ret < 0) + return ret; + + dev_dbg(&bus->dev, "write reg %02x:%04x = %04x\n", (reg >> 16) & 0x1f, + reg & 0xffff, val); + + return 0; +} + +static int i2c_mii_init_rollball(struct i2c_adapter *i2c) +{ + struct i2c_msg msg; + u8 pw[5]; + int ret; + + pw[0] = ROLLBALL_SFP_PASSWORD_ADDR; + pw[1] = 0xff; + pw[2] = 0xff; + pw[3] = 0xff; + pw[4] = 0xff; + + msg.addr = ROLLBALL_PHY_I2C_ADDR; + msg.flags = 0; + msg.len = sizeof(pw); + msg.buf = pw; + + ret = i2c_transfer(i2c, &msg, 1); + if (ret < 0) + return ret; + else if (ret != 1) + return -EIO; + else + return 0; +} + +struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c, + enum mdio_i2c_proto protocol) { struct mii_bus *mii; + int ret; if (!i2c_check_functionality(i2c, I2C_FUNC_I2C)) return ERR_PTR(-EINVAL); @@ -104,10 +395,28 @@ struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c) snprintf(mii->id, MII_BUS_ID_SIZE, "i2c:%s", dev_name(parent)); mii->parent = parent; - mii->read = i2c_mii_read; - mii->write = i2c_mii_write; mii->priv = i2c; + switch (protocol) { + case MDIO_I2C_ROLLBALL: + ret = i2c_mii_init_rollball(i2c); + if (ret < 0) { + dev_err(parent, + "Cannot initialize RollBall MDIO I2C protocol: %d\n", + ret); + mdiobus_free(mii); + return ERR_PTR(ret); + } + + mii->read = i2c_mii_read_rollball; + mii->write = i2c_mii_write_rollball; + break; + default: + mii->read = i2c_mii_read_default; + mii->write = i2c_mii_write_default; + break; + } + return mii; } EXPORT_SYMBOL_GPL(mdio_i2c_alloc); diff --git a/drivers/net/phy/sfp.c b/drivers/net/phy/sfp.c index 91d74c1a920a..958fd514a3b4 100644 --- a/drivers/net/phy/sfp.c +++ b/drivers/net/phy/sfp.c @@ -419,7 +419,7 @@ static int sfp_i2c_configure(struct sfp *sfp, struct i2c_adapter *i2c) sfp->read = sfp_i2c_read; sfp->write = sfp_i2c_write; - i2c_mii = mdio_i2c_alloc(sfp->dev, i2c); + i2c_mii = mdio_i2c_alloc(sfp->dev, i2c, MDIO_I2C_DEFAULT); if (IS_ERR(i2c_mii)) return PTR_ERR(i2c_mii); diff --git a/include/linux/mdio/mdio-i2c.h b/include/linux/mdio/mdio-i2c.h index b1d27f7cd23f..53eedb0dc1d3 100644 --- a/include/linux/mdio/mdio-i2c.h +++ b/include/linux/mdio/mdio-i2c.h @@ -11,6 +11,12 @@ struct device; struct i2c_adapter; struct mii_bus; -struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c); +enum mdio_i2c_proto { + MDIO_I2C_DEFAULT, + MDIO_I2C_ROLLBALL, +}; + +struct mii_bus *mdio_i2c_alloc(struct device *parent, struct i2c_adapter *i2c, + enum mdio_i2c_proto protocol); #endif
Some multigig SFPs from RollBall and Hilink do not expose functional MDIO access to the internal PHY of the SFP via I2C address 0x56 (although there seems to be read-only clause 22 access on this address). Instead these SFPs PHY can be accessed via I2C via the SFP Enhanced Digital Diagnostic Interface - I2C address 0x51. The SFP_PAGE has to be selected to 3 and the password must be filled with 0xff bytes for this PHY communication to work. This extends the mdio-i2c driver to support this protocol by adding a special parameter to mdio_i2c_alloc function via which this RollBall protocol can be selected. Signed-off-by: Marek Behún <kabel@kernel.org> Cc: Andrew Lunn <andrew@lunn.ch> Cc: Russell King <rmk+kernel@armlinux.org.uk> --- drivers/net/mdio/mdio-i2c.c | 319 +++++++++++++++++++++++++++++++++- drivers/net/phy/sfp.c | 2 +- include/linux/mdio/mdio-i2c.h | 8 +- 3 files changed, 322 insertions(+), 7 deletions(-)