diff --git a/Kconfig b/Kconfig index 1afa3f364c6b4f3550299fcd5009b22c35452242..1365cbedf4880cb1fbc93eadec4160ba82b95b49 100644 --- a/Kconfig +++ b/Kconfig @@ -860,7 +860,11 @@ config FAN_HYSTERESIS_PWM_VAL PWM value used to drive fans. Range from 4 to 1000. endmenu - +config READ_SFP_DIAG_ENABLE + bool "Read SFPs Diagnostic Monitoring" + default n + help + Let HAL to read Diagnostic Monitoring from SFP's eeprom. endmenu menu "RTU HP mask" diff --git a/doc/wrs-user-manual.in b/doc/wrs-user-manual.in index 230ded00d2cd2b862f70403bad96e99a7b09782f..cca952179e7a584f120a49c910236734f04b9210 100644 --- a/doc/wrs-user-manual.in +++ b/doc/wrs-user-manual.in @@ -834,6 +834,11 @@ value is changed by the web interface, proper action is taken. below @t{CONFIG_FAN_HYSTERESIS_T_DISABLE}. These options are intended to be used during development to reduce noise generated by switch. +@item CONFIG_READ_SFP_DIAG_ENABLE + + Let HAL to read Diagnostic Monitoring from SFP's eeprom like + temperature, TX/RX power etc. + @item CONFIG_RTU_HP_MASK_ENABLE @itemx CONFIG_RTU_HP_MASK_VAL diff --git a/userspace/libwr/i2c_sfp.c b/userspace/libwr/i2c_sfp.c index 52cb30869619b0ec0750639833d46a6a124b3f74..5c633dfd73b3a0f04780b68a57889a7e5e594855 100644 --- a/userspace/libwr/i2c_sfp.c +++ b/userspace/libwr/i2c_sfp.c @@ -47,6 +47,9 @@ #define SFP_LED_WRMODE_MASK(t) ((t) ? (1 << 5) : (1 << 3)) #define SFP_TX_DISABLE_MASK(t) ((t) ? (1 << 7) : (1 << 2)) +/* Either 8 or 16 byte pages, so we use the smaller */ +#define SFP_PAGE_SIZE 8 + /* * We need these tables because the schematics are messed up * The first one is for figuring out the masks in the pca9548's @@ -318,6 +321,15 @@ void shw_sfp_print_header(struct shw_sfp_header *head) printf("\n"); } +void shw_sfp_print_dom(struct shw_sfp_dom * dom) +{ + printf("Temperature: %.3f C\n", (int8_t)dom->temp[0] + dom->temp[1]/(float)256); + printf("Voltage: %.3f V\n", (dom->vcc[0]*256 + dom->vcc[1])/(float)10000); + printf("Bias Current: %.3f uA\n", (dom->tx_bias[0]*256+dom->tx_bias[1])/(float)500000); + printf("TX power: %.3f mW\n", (dom->tx_pow[0]*256 + dom->tx_pow[1])/(float)100000); + printf("RX power: %.3f mW\n", (dom->rx_pow[0]*256 + dom->rx_pow[1])/(float)100000); +} + void shw_sfp_header_dump(struct shw_sfp_header *head) { int i; @@ -332,6 +344,20 @@ void shw_sfp_header_dump(struct shw_sfp_header *head) } +void shw_sfp_dom_dump(struct shw_sfp_dom *dom) +{ + int i; + uint8_t *dump = (uint8_t *) dom; + printf("Dom Dump:"); + for (i = 0; i < sizeof(struct shw_sfp_dom); i++) { + if (i % 8 == 0) + printf("\n"); + printf("%02X ", dump[i]); + } + printf("\n"); +} + + /* Get the SFP ID from the SFP number (0 to 17) */ inline int shw_sfp_id(int num) { @@ -369,15 +395,26 @@ int32_t shw_sfp_read(int num, uint32_t addr, int off, int len, uint8_t * buf) return i2c_transfer(bus, addr, 0, len, buf); } + int32_t shw_sfp_write(int num, uint32_t addr, int off, int len, uint8_t * buf) { int id; uint8_t byte1, byte2; + int32_t counter = 0; struct i2c_bus *bus; + int i = 0; + uint8_t page[SFP_PAGE_SIZE + 1]; + int ret; + + /* The SFP eeprom only supports 8 bit addresses */ + if (len < 1 || len > 256 || off < 0 || off > 256 || off + len < 0 + || off + len > 256) + return -1; id = shw_sfp_id(num); if (id < 0) return -1; + bus = &i2c_buses[WR_MUX_BUS]; if (id == 0 || id == 1) bus = &i2c_buses[WR_SFP0_BUS + id]; @@ -390,11 +427,44 @@ int32_t shw_sfp_write(int num, uint32_t addr, int off, int len, uint8_t * buf) i2c_transfer(bus, 0x71, 1, 0, &byte2); } - /* Send the offset we want to write to if requested */ - if (off >= 0) - i2c_transfer(bus, addr, 1, 0, (uint8_t *) & off); - /* Do the read */ - return i2c_transfer(bus, addr, len, 0, buf); + /* Write in a paged mode, 1 byte address */ + page[0] = (counter + off) & 0xff; + while (counter < len) + { + page[i + 1] = buf[counter++]; + i++; + /* When we hit a page boundary then perform a write */ + if ((off + counter) % SFP_PAGE_SIZE == 0) + { + pr_debug("Writing %d bytes to EEPROM address %02x\n", + i, page[0]); + ret = i2c_transfer(bus, addr, i + 1, 0, page); + if (ret < 0) { + pr_error("i2c_transfer error code 0x%x\n", + ret); + return -1; + } + i = 0; + page[0] = (counter + off) & 0xff; + /* Sleep 10ms for eeprom to finish writing the page */ + /* XXX this should actually be done by polling the */ + /* EEPROM and seeing if it is ready */ + usleep(10000); + } + } + /* Write the last page, if not already done */ + if (i != 0) + { + pr_debug("Writing last %d bytes to EEPROM address %02x\n", + i, page[0]); + ret = i2c_transfer(bus, addr, i + 1, 0, page); + if (ret < 0) { + pr_error("i2c_transfer error code 0x%x\n", ret); + return -1; + } + usleep(10000); + } + return counter; } uint32_t shw_sfp_module_scan(void) @@ -559,6 +629,39 @@ int shw_sfp_read_header(int num, struct shw_sfp_header *head) return 0; } +int shw_sfp_read_dom(int num, struct shw_sfp_dom *dom) +{ + int ret; + + if (shw_sfp_id(num) < 0) { + pr_error("shw_sfp_read_header: wrong SFP num %d\n", num + 1); + return -1; + } + + ret = shw_sfp_module_scan(); + if (!(ret & (1 << num))) { + pr_error("shw_sfp_read_header: SFP not present %d\n", num + 1); + return -2; + } + + ret = + shw_sfp_read(num, I2C_SFP_DOM_ADDRESS, 0x0, + sizeof(struct shw_sfp_dom), (uint8_t *) dom); + if (ret == I2C_DEV_NOT_FOUND) { + pr_error("shw_sfp_read_header: I2C_DEV_NOT_FOUND\n"); + return -ENODEV; + } + + return 0; +} + +/* Function to update SFP's Diagnostic Monitoring data from SFP's eeprom */ +int shw_sfp_update_dom(int num, struct shw_sfp_dom *dom) +{ + /* For now copy entire eeprom */ + return shw_sfp_read_dom(num, dom); +} + int shw_sfp_read_verify_header(int num, struct shw_sfp_header *head) { int ret; diff --git a/userspace/libwr/i2c_sfp.h b/userspace/libwr/i2c_sfp.h index 57f2daa52eb6c7fd49626e4173f27764f5c42d0a..e8db764252a4bf6aa49fc21df94d72db72be7e97 100644 --- a/userspace/libwr/i2c_sfp.h +++ b/userspace/libwr/i2c_sfp.h @@ -7,6 +7,8 @@ //address from AT24C01 datasheet (1k, all address lines shorted to the ground) #define I2C_SFP_ADDRESS 0x50 +// From SFF-8472, but right-shifted one bit as I2C addresses are only 7 bits. +#define I2C_SFP_DOM_ADDRESS 0x51 /* The two FPGA buses */ #define WR_FPGA_BUS0 0 diff --git a/userspace/libwr/include/libwr/hal_shmem.h b/userspace/libwr/include/libwr/hal_shmem.h index 4fa23ef9fc5819577bc4e06bb1d6de1a0e2f94b8..d96a97f94be1ab1604236fe43583cfe9ce1dec2f 100644 --- a/userspace/libwr/include/libwr/hal_shmem.h +++ b/userspace/libwr/include/libwr/hal_shmem.h @@ -12,6 +12,10 @@ #define HAL_PORT_STATE_CALIBRATION 3 #define HAL_PORT_STATE_LOCKING 4 +/* Read temperature from SFPs */ +#define READ_SFP_DIAG_ENABLE 1 +#define READ_SFP_DIAG_DISABLE 0 + #define DEFAULT_T2_PHASE_TRANS 0 #define DEFAULT_T4_PHASE_TRANS 0 @@ -46,6 +50,7 @@ typedef struct hal_port_calibration { struct shw_sfp_caldata sfp; struct shw_sfp_header sfp_header_raw; + struct shw_sfp_dom sfp_dom_raw; } hal_port_calibration_t; /* Internal port state structure */ @@ -100,6 +105,9 @@ struct hal_port_state { /* Endpoint's base address */ uint32_t ep_base; + + /* whether SFP has diagnostic Monitoring capability */ + int has_sfp_diag; }; struct hal_temp_sensors { @@ -110,13 +118,15 @@ struct hal_temp_sensors { }; /* This is the overall structure stored in shared memory */ -#define HAL_SHMEM_VERSION 9 /* Version 9 because of adding sfp_header_raw */ +#define HAL_SHMEM_VERSION 10 /* Version 10 because of adding sfp_dom_raw + has_sfp_diag and read_sfp_diag */ struct hal_shmem_header { int nports; int hal_mode; struct hal_port_state *ports; struct hal_temp_sensors temp; + int read_sfp_diag; }; static inline int state_up(int state) diff --git a/userspace/libwr/include/libwr/sfp_lib.h b/userspace/libwr/include/libwr/sfp_lib.h index 03b4a11e6ce26dfe97f1cfb046e5773498944e15..3794074b1f016d6fd3f091272ad914c1b2b02225 100644 --- a/userspace/libwr/include/libwr/sfp_lib.h +++ b/userspace/libwr/include/libwr/sfp_lib.h @@ -30,6 +30,16 @@ #define SFP_SPEED_1Gb_10 0x0A /* Unfortunatelly the above is not always true, * e.g. Cisco copper SFP (MGBT1) says simply 10 and not 13.*/ +#define SFP_DIAGNOSTIC_IMPLEMENTED (1 << 6) /* Digital diagnostic monitoring + implemented. "1" for compliance + with SFF-8472 */ +#define SFP_ADDR_CHANGE_REQ (1 << 2) /* Bit 2 indicates whether or not it is + necessary for the host to perform an + address change sequence before + accessing information at 2-wire serial + address A2h. */ + + struct shw_sfp_caldata { uint32_t flags; /* @@ -64,9 +74,9 @@ struct shw_sfp_header { uint8_t length3; /* Link length supported for 50/125 mm fiber (10m) */ uint8_t length4; /* Link length supported for 62.5/125 mm fiber (10m) */ uint8_t length5; /* Link length supported for copper (1m) */ - uint8_t reserved2; + uint8_t length6; /* Link length supported on OM3 (1m) */ uint8_t vendor_name[16]; - uint8_t reserved3; + uint8_t reserved3; /* This is now a field named transceiver */ uint8_t vendor_oui[3]; uint8_t vendor_pn[16]; uint8_t vendor_rev[4]; @@ -80,10 +90,73 @@ struct shw_sfp_header { uint8_t br_min; uint8_t vendor_serial[16]; uint8_t date_code[8]; - uint8_t reserved[3]; + uint8_t diagnostic_monitoring_type; + uint8_t enhanced_options; + uint8_t sff_8472_compliance; uint8_t cc_ext; } __attribute__ ((packed)); +struct shw_sfp_dom { +/* Treshold values, 0 - 55 */ + uint8_t temp_high_alarm[2]; + uint8_t temp_low_alarm[2]; + uint8_t temp_high_warn[2]; + uint8_t temp_low_warn[2]; + uint8_t volt_high_alarm[2]; + uint8_t volt_low_alarm[2]; + uint8_t volt_high_warn[2]; + uint8_t volt_low_warn[2]; + uint8_t bias_high_alarm[2]; + uint8_t bias_low_alarm[2]; + uint8_t bias_high_warn[2]; + uint8_t bias_low_warn[2]; + uint8_t tx_pow_high_alarm[2]; + uint8_t tx_pow_low_alarm[2]; + uint8_t tx_pow_high_warn[2]; + uint8_t tx_pow_low_warn[2]; + uint8_t rx_pow_high_alarm[2]; + uint8_t rx_pow_log_alarm[2]; + uint8_t rx_power_high_warn[2]; + uint8_t rx_power_low_warn[2]; + uint8_t unalloc0[16]; +/* Calibration data, 56-91 */ + uint8_t cal_rx_pwr4[4]; + uint8_t cal_rx_pwr3[4]; + uint8_t cal_rx_pwr2[4]; + uint8_t cal_rx_pwr1[4]; + uint8_t cal_rx_pwr0[4]; + uint8_t cal_tx_i_slope[2]; + uint8_t cal_tx_i_offset[2]; + uint8_t cal_tx_pow_slope[2]; + uint8_t cal_tx_pow_offset[2]; + uint8_t cal_T_slope[2]; + uint8_t cal_T_offset[2]; + uint8_t cal_V_slope[2]; + uint8_t cal_V_offset[2]; +/* Unallocated and checksum, 92-95 */ + uint8_t cal_unalloc[3]; + uint8_t CC_DMI; +/* Real Time Diagnostics, 96-111 */ + uint8_t temp[2]; + uint8_t vcc[2]; + uint8_t tx_bias[2]; + uint8_t tx_pow[2]; + uint8_t rx_pow[2]; + uint8_t rtd_unalloc0[4]; + uint8_t OSCB; + uint8_t rtd_unalloc1; +/* Alarms and Warnings, 112 - 117 */ + uint8_t alw[6]; +/* Extended Module Control/Status bytes 118 - 119 */ + uint8_t emcsb[2]; +/* Vendor locations 120 - 127 */ + uint8_t vendor_locations[8]; +/* User data 128 - 247 */ + uint8_t dom_user[120]; +/* Vendor specific control function locations 248 - 255 */ + uint8_t vendor_functions[8]; +} __attribute__ ((packed)); + /* Public API */ /* @@ -113,6 +186,18 @@ int shw_sfp_read_db(void); /* Read and verify the header all at once. returns -1 on failure */ int shw_sfp_read_verify_header(int num, struct shw_sfp_header *head); +/* Read the SFP diagnostics page */ +int shw_sfp_read_dom(int num, struct shw_sfp_dom *dom); + +/* Update the SFP diagnostics page */ +int shw_sfp_update_dom(int num, struct shw_sfp_dom *dom); + +/* Decode and print the SFP real time diagnostics */ +void shw_sfp_print_dom(struct shw_sfp_dom * dom); + +/* Dump the SFP diagnostics page in hex */ +void shw_sfp_dom_dump(struct shw_sfp_dom * dom); + /* return NULL if no data found */ struct shw_sfp_caldata *shw_sfp_get_cal_data(int num, struct shw_sfp_header *head); diff --git a/userspace/ppsi b/userspace/ppsi index 3ece12b3824d42910145fe0fa622d00cac1597ef..6f9508cfb22a78f00fab8db6ff2e6c84441dcb3a 160000 --- a/userspace/ppsi +++ b/userspace/ppsi @@ -1 +1 @@ -Subproject commit 3ece12b3824d42910145fe0fa622d00cac1597ef +Subproject commit 6f9508cfb22a78f00fab8db6ff2e6c84441dcb3a diff --git a/userspace/tools/wrs_dump_shmem.c b/userspace/tools/wrs_dump_shmem.c index 415a4dbf08cca65acc2bf3618777ddc5ede21a42..7d51454cc74e0a97597d03ea1d385984b6d2f90d 100644 --- a/userspace/tools/wrs_dump_shmem.c +++ b/userspace/tools/wrs_dump_shmem.c @@ -382,6 +382,7 @@ void dump_many_fields(void *addr, struct dump_info *info, int ninfo) struct dump_info hal_shmem_info [] = { DUMP_FIELD(int, nports), DUMP_FIELD(int, hal_mode), + DUMP_FIELD(int, read_sfp_diag), DUMP_FIELD(sensor_temp, temp.fpga), DUMP_FIELD(sensor_temp, temp.pll), DUMP_FIELD(sensor_temp, temp.psl), @@ -433,6 +434,7 @@ struct dump_info hal_port_info [] = { DUMP_FIELD(uint32_t, t2_phase_transition), DUMP_FIELD(uint32_t, t4_phase_transition), DUMP_FIELD(uint32_t, ep_base), + DUMP_FIELD(int, has_sfp_diag), }; int dump_hal_mem(struct wrs_shm_head *head) diff --git a/userspace/tools/wrs_sfp_dump.c b/userspace/tools/wrs_sfp_dump.c index 857f581fcc36f2431741fe48a90a3b055d0886a0..6bdf8711bfff5d351548238f98ba74972dce5bbe 100644 --- a/userspace/tools/wrs_sfp_dump.c +++ b/userspace/tools/wrs_sfp_dump.c @@ -33,11 +33,12 @@ void print_info(char *prgname) "Optional parameters:\n" " -p <num> Dump sfp header for specific port (1-18); dump sfp header info for all\n" " ports if no <-p> specified\n" - " -x Dump sfp header also in hex\n" " -a <READ|WRITE> Read/write SFP's eeprom; works only with <-I>;\n" " before READs/WRITEs disable HAL and monit!\n" " -f <file> File to READ/WRITE SFP's eeprom\n" " -H <dir> Open shmem dumps from the given directory; works only with <-L>\n" + " -d Dump sfp DOM data page\n" + " -x Dump sfp/DOM header also in hex\n" " -q Decrease verbosity\n" " -v Increase verbosity\n" " -V Print version\n" @@ -201,7 +202,8 @@ void print_version(char *prgname) __GIT_USR__); } -int hal_read(struct shw_sfp_header *sfp_header_local_copy) { +int hal_read(struct shw_sfp_header *sfp_header_local_copy, + struct shw_sfp_dom *sfp_dom) { unsigned ii; unsigned retries = 0; int port; @@ -214,7 +216,12 @@ int hal_read(struct shw_sfp_header *sfp_header_local_copy) { &hal_ports[port].calib.sfp_header_raw, sizeof(struct shw_sfp_header)); } - + if (sfp_dom) + for (port = 0; port < hal_nports_local; port++) { + memcpy(&sfp_dom[port], + &hal_ports[port].calib.sfp_dom_raw, + sizeof(struct shw_sfp_dom)); + } retries++; if (retries > 100) return -1; @@ -280,23 +287,27 @@ int main(int argc, char **argv) int c; struct shw_sfp_header sfp_hdr; struct shw_sfp_header *sfp_hdr_p; + struct shw_sfp_dom sfp_dom; + struct shw_sfp_dom *sfp_dom_p; int err; int nports; int dump_port; int i; int dump_hex_header = 0; + int dump_sfp_dom = 0; int operation = 0; char *eeprom_file = NULL; int sfp_data_source = 0; /* local copy of sfp eeprom */ struct shw_sfp_header hal_sfp_raw_header_lc[HAL_MAX_PORTS]; + struct shw_sfp_dom hal_sfp_raw_dom_lc[HAL_MAX_PORTS]; wrs_msg_init(argc, argv); nports = 18; dump_port = 1; - while ((c = getopt(argc, argv, "a:hqvp:xVf:LIH:")) != -1) { + while ((c = getopt(argc, argv, "a:hqvp:xVf:LIdH:")) != -1) { switch (c) { case 'p': dump_port = atoi(optarg); @@ -311,6 +322,9 @@ int main(int argc, char **argv) case 'x': dump_hex_header = 1; break; + case 'd': + dump_sfp_dom = 1; + break; case 'V': print_version(argv[0]); exit(0); @@ -363,7 +377,7 @@ int main(int argc, char **argv) if (sfp_data_source == READ_HAL) { hal_init_shm(); - hal_read(hal_sfp_raw_header_lc); + hal_read(hal_sfp_raw_header_lc, hal_sfp_raw_dom_lc); printf("Reading SFP eeprom from HAL\n"); } @@ -392,9 +406,13 @@ int main(int argc, char **argv) memset(&sfp_hdr, 0, sizeof(sfp_hdr)); sfp_hdr_p = &sfp_hdr; err = shw_sfp_read_header(i - 1, sfp_hdr_p); + memset(&sfp_dom, 0, sizeof(sfp_dom)); + sfp_dom_p = &sfp_dom; + shw_sfp_read_dom(i - 1, sfp_dom_p); } if (sfp_data_source == READ_HAL) { sfp_hdr_p = &hal_sfp_raw_header_lc[i - 1]; + sfp_dom_p = &hal_sfp_raw_dom_lc[i - 1]; } err = shw_sfp_header_verify(sfp_hdr_p); if (err == -2) { @@ -408,6 +426,12 @@ int main(int argc, char **argv) if (dump_hex_header) { shw_sfp_header_dump(sfp_hdr_p); } + if(dump_sfp_dom) { + shw_sfp_print_dom(sfp_dom_p); + if(dump_hex_header) { + shw_sfp_dom_dump(sfp_dom_p); + } + } } } return 0; diff --git a/userspace/wrsw_hal/hal_ports.c b/userspace/wrsw_hal/hal_ports.c index 83c3c167ae0730f99e3f8f8a62724dd6e1c35bc8..7f75ae6af3587f25a56544646180c19ca7c9a6fa 100644 --- a/userspace/wrsw_hal/hal_ports.c +++ b/userspace/wrsw_hal/hal_ports.c @@ -211,6 +211,7 @@ static int hal_port_init(int index) int hal_port_init_shmem(char *logfilename) { int index; + char *ret; pr_info("Initializing switch ports...\n"); /* default timeouts */ @@ -255,6 +256,14 @@ int hal_port_init_shmem(char *logfilename) hal_shmem->nports = hal_port_nports; hal_shmem_hdr->version = HAL_SHMEM_VERSION; hal_shmem->hal_mode = hal_get_timing_mode(); + + ret = libwr_cfg_get("READ_SFP_DIAG_ENABLE"); + if (ret && !strcmp(ret, "y")) { + pr_info("Read SFP Diagnostic Monitoring enabled\n"); + hal_shmem->read_sfp_diag = READ_SFP_DIAG_ENABLE; + } else + hal_shmem->read_sfp_diag = READ_SFP_DIAG_DISABLE; + /* Release processes waiting for HAL's to fill shm with correct data When shm is opened successfully data in shm is still not populated! Read data with wrs_shm_seqbegin and wrs_shm_seqend! @@ -478,6 +487,7 @@ static void hal_port_insert_sfp(struct hal_port_state * p) int err; memset(&shdr, 0, sizeof(struct shw_sfp_header)); + memset(&p->calib.sfp_dom_raw, 0, sizeof(struct shw_sfp_dom)); err = shw_sfp_read_verify_header(p->hw_index, &shdr); memcpy(&p->calib.sfp_header_raw, &shdr, sizeof(struct shw_sfp_header)); if (err == -2) { @@ -489,7 +499,27 @@ static void hal_port_insert_sfp(struct hal_port_state * p) p->name); return; } + if (hal_shmem->read_sfp_diag == READ_SFP_DIAG_ENABLE + && shdr.diagnostic_monitoring_type & SFP_DIAGNOSTIC_IMPLEMENTED) { + pr_info("SFP Diagnostic Monitoring implemented in SFP plugged" + " to port %d (%s)\n", p->hw_index + 1, p->name); + if (shdr.diagnostic_monitoring_type & SFP_ADDR_CHANGE_REQ) { + pr_warning("SFP in port %d (%s) requires special " + "address change before accessing Diagnostic" + " Monitoring, which is not implemented " + "right now\n", p->hw_index + 1, p->name); + } else { + /* copy coontent of SFP's Diagnostic Monitoring */ + shw_sfp_read_dom(p->hw_index, &p->calib.sfp_dom_raw); + if (err < 0) { + pr_error("Failed to read SFP Diagnostic " + "Monitoring for port %d (%s)\n", + p->hw_index + 1, p->name); + } + p->has_sfp_diag = 1; + } + } pr_info("SFP Info: Manufacturer: %.16s P/N: %.16s, S/N: %.16s\n", shdr.vendor_name, shdr.vendor_pn, shdr.vendor_serial); cdata = shw_sfp_get_cal_data(p->hw_index, &shdr); @@ -570,6 +600,8 @@ static void hal_port_remove_sfp(struct hal_port_state * p) /* clean SFP's details when removing SFP */ memset(&p->calib.sfp, 0, sizeof(p->calib.sfp)); memset(&p->calib.sfp_header_raw, 0, sizeof(struct shw_sfp_header)); + memset(&p->calib.sfp_dom_raw, 0, sizeof(struct shw_sfp_dom)); + p->has_sfp_diag = 0; } /* detects insertion/removal of SFP transceivers */ @@ -615,10 +647,20 @@ void hal_port_update_all() hal_port_poll_sfp(); for (i = 0; i < HAL_MAX_PORTS; i++) - if (ports[i].in_use) + if (ports[i].in_use) { hal_port_fsm(&ports[i]); + /* update DOM only for plugged ports with DOM + * capabilities */ + if (ports[i].state != HAL_PORT_STATE_DISABLED + && hal_shmem->read_sfp_diag == READ_SFP_DIAG_ENABLE + && (ports[i].has_sfp_diag)) { + shw_sfp_update_dom(ports[i].hw_index, + &ports[i].calib.sfp_dom_raw); + } + } /* unlock shmem */ wrs_shm_write(hal_shmem_hdr, WRS_SHM_WRITE_END); + } int hal_port_enable_tracking(const char *port_name)