Commit 5cab3035 authored by David S. Miller's avatar David S. Miller

Merge branch 'Add-support-for-VSOL-V2801F-CarlitoxxPro-CPGOS03-GPON-mo

dule'

Russell King says:

====================
Add support for VSOL V2801F/CarlitoxxPro CPGOS03 GPON module

This patch set adds support for the V2801F / CarlitoxxPro module. This
requires two changes:

1) the module only supports single byte reads to the ID EEPROM,
   while we need to still permit sequential reads to the diagnostics
   EEPROM for atomicity reasons.

2) we need to relax the encoding check when we have no reported
   capabilities to allow 1000base-X based on the module bitrate.

Thanks to Pali Rohár for responsive testing over the last two days.

(Resending, dropping the utf-8 characters in Pali's name so the patches
 get through vger. Added Andrew's r-b tags.)
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 6b21c0bb 7a77233e
...@@ -334,14 +334,13 @@ void sfp_parse_support(struct sfp_bus *bus, const struct sfp_eeprom_id *id, ...@@ -334,14 +334,13 @@ void sfp_parse_support(struct sfp_bus *bus, const struct sfp_eeprom_id *id,
} }
/* If we haven't discovered any modes that this module supports, try /* If we haven't discovered any modes that this module supports, try
* the encoding and bitrate to determine supported modes. Some BiDi * the bitrate to determine supported modes. Some BiDi modules (eg,
* modules (eg, 1310nm/1550nm) are not 1000BASE-BX compliant due to * 1310nm/1550nm) are not 1000BASE-BX compliant due to the differing
* the differing wavelengths, so do not set any transceiver bits. * wavelengths, so do not set any transceiver bits.
*/ */
if (bitmap_empty(modes, __ETHTOOL_LINK_MODE_MASK_NBITS)) { if (bitmap_empty(modes, __ETHTOOL_LINK_MODE_MASK_NBITS)) {
/* If the encoding and bit rate allows 1000baseX */ /* If the bit rate allows 1000baseX */
if (id->base.encoding == SFF8024_ENCODING_8B10B && br_nom && if (br_nom && br_min <= 1300 && br_max >= 1200)
br_min <= 1300 && br_max >= 1200)
phylink_set(modes, 1000baseX_Full); phylink_set(modes, 1000baseX_Full);
} }
......
...@@ -219,6 +219,7 @@ struct sfp { ...@@ -219,6 +219,7 @@ struct sfp {
struct sfp_bus *sfp_bus; struct sfp_bus *sfp_bus;
struct phy_device *mod_phy; struct phy_device *mod_phy;
const struct sff_data *type; const struct sff_data *type;
size_t i2c_block_size;
u32 max_power_mW; u32 max_power_mW;
unsigned int (*get_state)(struct sfp *); unsigned int (*get_state)(struct sfp *);
...@@ -335,10 +336,19 @@ static int sfp_i2c_read(struct sfp *sfp, bool a2, u8 dev_addr, void *buf, ...@@ -335,10 +336,19 @@ static int sfp_i2c_read(struct sfp *sfp, bool a2, u8 dev_addr, void *buf,
size_t len) size_t len)
{ {
struct i2c_msg msgs[2]; struct i2c_msg msgs[2];
u8 bus_addr = a2 ? 0x51 : 0x50; size_t block_size;
size_t this_len; size_t this_len;
u8 bus_addr;
int ret; int ret;
if (a2) {
block_size = 16;
bus_addr = 0x51;
} else {
block_size = sfp->i2c_block_size;
bus_addr = 0x50;
}
msgs[0].addr = bus_addr; msgs[0].addr = bus_addr;
msgs[0].flags = 0; msgs[0].flags = 0;
msgs[0].len = 1; msgs[0].len = 1;
...@@ -350,8 +360,8 @@ static int sfp_i2c_read(struct sfp *sfp, bool a2, u8 dev_addr, void *buf, ...@@ -350,8 +360,8 @@ static int sfp_i2c_read(struct sfp *sfp, bool a2, u8 dev_addr, void *buf,
while (len) { while (len) {
this_len = len; this_len = len;
if (this_len > 16) if (this_len > block_size)
this_len = 16; this_len = block_size;
msgs[1].len = this_len; msgs[1].len = this_len;
...@@ -1632,6 +1642,28 @@ static int sfp_sm_mod_hpower(struct sfp *sfp, bool enable) ...@@ -1632,6 +1642,28 @@ static int sfp_sm_mod_hpower(struct sfp *sfp, bool enable)
return 0; return 0;
} }
/* Some modules (Nokia 3FE46541AA) lock up if byte 0x51 is read as a
* single read. Switch back to reading 16 byte blocks unless we have
* a CarlitoxxPro module (rebranded VSOL V2801F). Even more annoyingly,
* some VSOL V2801F have the vendor name changed to OEM.
*/
static int sfp_quirk_i2c_block_size(const struct sfp_eeprom_base *base)
{
if (!memcmp(base->vendor_name, "VSOL ", 16))
return 1;
if (!memcmp(base->vendor_name, "OEM ", 16) &&
!memcmp(base->vendor_pn, "V2801F ", 16))
return 1;
/* Some modules can't cope with long reads */
return 16;
}
static void sfp_quirks_base(struct sfp *sfp, const struct sfp_eeprom_base *base)
{
sfp->i2c_block_size = sfp_quirk_i2c_block_size(base);
}
static int sfp_cotsworks_fixup_check(struct sfp *sfp, struct sfp_eeprom_id *id) static int sfp_cotsworks_fixup_check(struct sfp *sfp, struct sfp_eeprom_id *id)
{ {
u8 check; u8 check;
...@@ -1673,14 +1705,20 @@ static int sfp_sm_mod_probe(struct sfp *sfp, bool report) ...@@ -1673,14 +1705,20 @@ static int sfp_sm_mod_probe(struct sfp *sfp, bool report)
u8 check; u8 check;
int ret; int ret;
ret = sfp_read(sfp, false, 0, &id, sizeof(id)); /* Some modules (CarlitoxxPro CPGOS03-0490) do not support multibyte
* reads from the EEPROM, so start by reading the base identifying
* information one byte at a time.
*/
sfp->i2c_block_size = 1;
ret = sfp_read(sfp, false, 0, &id.base, sizeof(id.base));
if (ret < 0) { if (ret < 0) {
if (report) if (report)
dev_err(sfp->dev, "failed to read EEPROM: %d\n", ret); dev_err(sfp->dev, "failed to read EEPROM: %d\n", ret);
return -EAGAIN; return -EAGAIN;
} }
if (ret != sizeof(id)) { if (ret != sizeof(id.base)) {
dev_err(sfp->dev, "EEPROM short read: %d\n", ret); dev_err(sfp->dev, "EEPROM short read: %d\n", ret);
return -EAGAIN; return -EAGAIN;
} }
...@@ -1719,6 +1757,21 @@ static int sfp_sm_mod_probe(struct sfp *sfp, bool report) ...@@ -1719,6 +1757,21 @@ static int sfp_sm_mod_probe(struct sfp *sfp, bool report)
} }
} }
/* Apply any early module-specific quirks */
sfp_quirks_base(sfp, &id.base);
ret = sfp_read(sfp, false, SFP_CC_BASE + 1, &id.ext, sizeof(id.ext));
if (ret < 0) {
if (report)
dev_err(sfp->dev, "failed to read EEPROM: %d\n", ret);
return -EAGAIN;
}
if (ret != sizeof(id.ext)) {
dev_err(sfp->dev, "EEPROM short read: %d\n", ret);
return -EAGAIN;
}
check = sfp_check(&id.ext, sizeof(id.ext) - 1); check = sfp_check(&id.ext, sizeof(id.ext) - 1);
if (check != id.ext.cc_ext) { if (check != id.ext.cc_ext) {
if (cotsworks) { if (cotsworks) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment