|
|
121cca |
From 9a7ded387a87accd6437b3e36748d4451e8135f4 Mon Sep 17 00:00:00 2001
|
|
|
121cca |
From: Erik Ekman <erik@kryo.se>
|
|
|
121cca |
Date: Tue, 2 Nov 2021 14:56:25 +0100
|
|
|
121cca |
Subject: [PATCH 65/65] merge Github PR#77
|
|
|
121cca |
|
|
|
121cca |
---
|
|
|
121cca |
src/core/network.cc | 128 ++++++++++++++++++++++++++++++++++++++++++++
|
|
|
121cca |
1 file changed, 128 insertions(+)
|
|
|
121cca |
|
|
|
121cca |
diff --git a/src/core/network.cc b/src/core/network.cc
|
|
|
121cca |
index 01a1f51..746ac1b 100644
|
|
|
121cca |
--- a/src/core/network.cc
|
|
|
121cca |
+++ b/src/core/network.cc
|
|
|
121cca |
@@ -93,6 +93,41 @@ struct ethtool_link_settings
|
|
|
121cca |
supported, advertised, peer advertised. */
|
|
|
121cca |
};
|
|
|
121cca |
|
|
|
121cca |
+/* Recognized module eeprom standards. */
|
|
|
121cca |
+#define ETH_MODULE_SFF_8079 0x1
|
|
|
121cca |
+#define ETH_MODULE_SFF_8472 0x2
|
|
|
121cca |
+#define ETH_MODULE_SFF_8636 0x3
|
|
|
121cca |
+#define ETH_MODULE_SFF_8436 0x4
|
|
|
121cca |
+
|
|
|
121cca |
+struct ethtool_modinfo {
|
|
|
121cca |
+ u32 cmd;
|
|
|
121cca |
+ u32 type; /* SFF standard used in module */
|
|
|
121cca |
+ u32 eeprom_len; /* Length of module eeprom */
|
|
|
121cca |
+ u32 reserved[8];
|
|
|
121cca |
+};
|
|
|
121cca |
+
|
|
|
121cca |
+/* Known id types. */
|
|
|
121cca |
+#define SFF_8024_ID_SOLDERED 0x2
|
|
|
121cca |
+#define SFF_8024_ID_SFP 0x3
|
|
|
121cca |
+#define SFF_8024_EXT_ID_DEFINED_BY_2WIRE_ID 0x4
|
|
|
121cca |
+
|
|
|
121cca |
+/* Common connector types. */
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_SC 0x1
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_LC 0x7
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_OPTICAL_PIGTAIL 0xb
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_COPPER_PIGTAIL 0x21
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_RJ45 0x22
|
|
|
121cca |
+#define SFF_8024_CONNECTOR_NON_SEPARABLE 0x23
|
|
|
121cca |
+
|
|
|
121cca |
+#define MAX_EEPROM_SIZE 256
|
|
|
121cca |
+struct ethtool_eeprom {
|
|
|
121cca |
+ u32 cmd;
|
|
|
121cca |
+ u32 magic; /* Only used for eeprom writes */
|
|
|
121cca |
+ u32 offset; /* Read or write offset */
|
|
|
121cca |
+ u32 len; /* Length of read/write */
|
|
|
121cca |
+ u8 data[MAX_EEPROM_SIZE]; /* Buffer */
|
|
|
121cca |
+};
|
|
|
121cca |
+
|
|
|
121cca |
#ifndef IFNAMSIZ
|
|
|
121cca |
#define IFNAMSIZ 32
|
|
|
121cca |
#endif
|
|
|
121cca |
@@ -131,6 +166,8 @@ struct ethtool_value
|
|
|
121cca |
#define ETHTOOL_GSET 0x00000001 /* Get settings. */
|
|
|
121cca |
#define ETHTOOL_GDRVINFO 0x00000003 /* Get driver info. */
|
|
|
121cca |
#define ETHTOOL_GLINK 0x0000000a /* Get link status (ethtool_value) */
|
|
|
121cca |
+#define ETHTOOL_GMODULEINFO 0x00000042 /* Get plug-in module information */
|
|
|
121cca |
+#define ETHTOOL_GMODULEEEPROM 0x00000043 /* Get plug-in module eeprom */
|
|
|
121cca |
#define ETHTOOL_GLINKSETTINGS 0x0000004c /* Get link mode settings. */
|
|
|
121cca |
|
|
|
121cca |
/* Indicates what features are supported by the interface. */
|
|
|
121cca |
@@ -358,6 +395,96 @@ static bool isVirtual(const string & MAC)
|
|
|
121cca |
}
|
|
|
121cca |
|
|
|
121cca |
|
|
|
121cca |
+// Get data for connected transceiver module.
|
|
|
121cca |
+static void scan_module(hwNode & interface, int fd)
|
|
|
121cca |
+{
|
|
|
121cca |
+ struct ifreq ifr;
|
|
|
121cca |
+ struct ethtool_modinfo emodinfo;
|
|
|
121cca |
+ struct ethtool_eeprom eeeprom;
|
|
|
121cca |
+
|
|
|
121cca |
+ emodinfo.cmd = ETHTOOL_GMODULEINFO;
|
|
|
121cca |
+ memset(&ifr, 0, sizeof(ifr));
|
|
|
121cca |
+ strcpy(ifr.ifr_name, interface.getLogicalName().c_str());
|
|
|
121cca |
+ ifr.ifr_data = (caddr_t) &emodinfo;
|
|
|
121cca |
+ // Skip interface if module info not supported.
|
|
|
121cca |
+ if (ioctl(fd, SIOCETHTOOL, &ifr) != 0)
|
|
|
121cca |
+ return;
|
|
|
121cca |
+
|
|
|
121cca |
+ eeeprom.cmd = ETHTOOL_GMODULEEEPROM;
|
|
|
121cca |
+ eeeprom.offset = 0;
|
|
|
121cca |
+ eeeprom.len = emodinfo.eeprom_len;
|
|
|
121cca |
+ if (eeeprom.len > MAX_EEPROM_SIZE)
|
|
|
121cca |
+ eeeprom.len = MAX_EEPROM_SIZE;
|
|
|
121cca |
+ memset(&ifr, 0, sizeof(ifr));
|
|
|
121cca |
+ strcpy(ifr.ifr_name, interface.getLogicalName().c_str());
|
|
|
121cca |
+ ifr.ifr_data = (caddr_t) &eeeprom;
|
|
|
121cca |
+ if (ioctl(fd, SIOCETHTOOL, &ifr) != 0)
|
|
|
121cca |
+ return;
|
|
|
121cca |
+
|
|
|
121cca |
+ switch (emodinfo.type)
|
|
|
121cca |
+ {
|
|
|
121cca |
+ /* SFF 8472 eeprom layout starts with same data as SFF 8079. */
|
|
|
121cca |
+ case ETH_MODULE_SFF_8079:
|
|
|
121cca |
+ case ETH_MODULE_SFF_8472:
|
|
|
121cca |
+ if ((eeeprom.data[0] == SFF_8024_ID_SOLDERED || eeeprom.data[0] == SFF_8024_ID_SFP) &&
|
|
|
121cca |
+ eeeprom.data[1] == SFF_8024_EXT_ID_DEFINED_BY_2WIRE_ID)
|
|
|
121cca |
+ {
|
|
|
121cca |
+ char buffer[32];
|
|
|
121cca |
+ /* Get part number (padded with space). String is stripped inside setConfig. */
|
|
|
121cca |
+ interface.setConfig("module", string((const char*)&eeeprom.data[40], 16));
|
|
|
121cca |
+ int wavelength = eeeprom.data[60] << 8 | eeeprom.data[61];
|
|
|
121cca |
+ /* Skip wavelength for SFP+ cables, they use byte 60 for other data. */
|
|
|
121cca |
+ if ((eeeprom.data[8] & 0x0C) == 0 && wavelength > 0)
|
|
|
121cca |
+ {
|
|
|
121cca |
+ snprintf(buffer, sizeof(buffer), "%dnm", wavelength);
|
|
|
121cca |
+ interface.setConfig("wavelength", buffer);
|
|
|
121cca |
+ }
|
|
|
121cca |
+ int max_length = 0;
|
|
|
121cca |
+ int length;
|
|
|
121cca |
+ length = eeeprom.data[14] * 1000; /* SMF, km */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ length = eeeprom.data[15] * 100; /* SMF, meter */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ length = eeeprom.data[16] * 10; /* 50um (OM2), meter */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ length = eeeprom.data[17] * 10; /* 62.5um (OM1), meter */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ length = eeeprom.data[18]; /* Copper, meter */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ length = eeeprom.data[19] * 10; /* OM3, meter */
|
|
|
121cca |
+ if (length > max_length) max_length = length;
|
|
|
121cca |
+ if (max_length > 0)
|
|
|
121cca |
+ {
|
|
|
121cca |
+ snprintf(buffer, sizeof(buffer), "%dm", max_length);
|
|
|
121cca |
+ interface.setConfig("maxlength", buffer);
|
|
|
121cca |
+ }
|
|
|
121cca |
+ switch (eeeprom.data[2])
|
|
|
121cca |
+ {
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_SC:
|
|
|
121cca |
+ interface.setConfig("connector", "SC");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_LC:
|
|
|
121cca |
+ interface.setConfig("connector", "LC");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_OPTICAL_PIGTAIL:
|
|
|
121cca |
+ interface.setConfig("connector", "optical pigtail");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_COPPER_PIGTAIL:
|
|
|
121cca |
+ interface.setConfig("connector", "copper pigtail");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_RJ45:
|
|
|
121cca |
+ interface.setConfig("connector", "RJ45");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ case SFF_8024_CONNECTOR_NON_SEPARABLE:
|
|
|
121cca |
+ interface.setConfig("connector", "non separable");
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ }
|
|
|
121cca |
+ }
|
|
|
121cca |
+ break;
|
|
|
121cca |
+ }
|
|
|
121cca |
+}
|
|
|
121cca |
+
|
|
|
121cca |
+
|
|
|
121cca |
static void updateCapabilities(hwNode & interface, u32 supported, u32 supported2, u32 speed, u8 duplex, u8 port, u8 autoneg)
|
|
|
121cca |
{
|
|
|
121cca |
if(supported & SUPPORTED_TP)
|
|
|
121cca |
@@ -666,6 +793,7 @@ bool scan_network(hwNode & n)
|
|
|
121cca |
}
|
|
|
121cca |
|
|
|
121cca |
scan_modes(interface, fd);
|
|
|
121cca |
+ scan_module(interface, fd);
|
|
|
121cca |
|
|
|
121cca |
drvinfo.cmd = ETHTOOL_GDRVINFO;
|
|
|
121cca |
memset(&ifr, 0, sizeof(ifr));
|
|
|
121cca |
--
|
|
|
121cca |
2.33.1
|
|
|
121cca |
|