forked from sonic-net/sonic-buildimage
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
[SONIC_Y_CABLE] Adding new Y cable Package for accessing Y cable QSFP…
…'s eeprom and configuring Y cable's mux through eeprom (sonic-net#128) * [SONIC_PLATFORM_COMMON] Adding new Y cable Package Signed-off-by: vaibhav-dahiya <vdahiya@microsoft.com>
- Loading branch information
Showing
3 changed files
with
335 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Empty file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,334 @@ | ||
#!/usr/bin/env python | ||
# | ||
# y_cable.py | ||
# | ||
# definitions for implementing Y cable access and configurations | ||
# API's for Y cable functionality in SONiC | ||
|
||
try: | ||
import struct | ||
|
||
import sonic_platform.platform | ||
from sonic_py_common import logger | ||
|
||
except ImportError as e: | ||
raise ImportError(str(e) + " - required module not found") | ||
|
||
# definitions of the offset with width accomodated for values | ||
# of MUX register specs of upper page 0x04 starting at 640 | ||
# info eeprom for Y Cable | ||
Y_CABLE_IDENTFIER_LOWER_PAGE = 0 | ||
Y_CABLE_IDENTFIER_UPPER_PAGE = 128 | ||
Y_CABLE_DETERMINE_CABLE_READ_SIDE = 640 | ||
Y_CABLE_CHECK_LINK_ACTIVE = 641 | ||
Y_CABLE_SWITCH_MUX_DIRECTION = 642 | ||
Y_CABLE_ACTIVE_TOR_INDICATOR = 645 | ||
Y_CABLE_MANUAL_SWITCH_COUNT = 669 | ||
|
||
# Global logger instance for helper functions and classes to log | ||
helper_logger = logger.Logger(SYSLOG_IDENTIFIER) | ||
|
||
# Global platform_chassis instance to call get_sfp required for read/write eeprom | ||
platform_chassis = None | ||
|
||
try: | ||
platform_chassis = sonic_platform.platform.Platform().get_chassis() | ||
helper_logger.log_info("chassis loaded {}".format(platform_chassis)) | ||
except Exception as e: | ||
helper_logger.log_warning("Failed to load chassis due to {}".format(repr(e))) | ||
|
||
def toggle_mux_to_torA(physical_port): | ||
""" | ||
This API specifically does a hard switch toggle of the Y cable's MUX regardless of link state to | ||
TOR A. This means if the Y cable is actively routing, the "check_active_linked_tor_side(physical_port)" | ||
API will now return Tor A. It also implies that if the link is actively routing on this port, Y cable | ||
MUX will start forwarding packets from TOR A to NIC, and drop packets from TOR B to NIC | ||
regardless of previous forwarding state. This API basically writes to upper page 4 offset 130 the | ||
value of 0x2 and expects the MUX to toggle to TOR A. Bit 0 value 0 means TOR A. | ||
Register Specification at offset 130 is documented below | ||
Byte offset bits Name Description | ||
130 7-2 Reserved Reserved | ||
1 Hard vs. soft switch 0b0 - Switch only if a valid TOR link on target; 0b1 Switch to new target regardless of link status | ||
0 Switch Target Switch Target - 0b0 - TOR#1, 0b1 - TOR#2; default is TOR #1 | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to Y end of a Y cable which can toggle the MUX | ||
Returns: | ||
a Boolean, true if the toggle succeeded and false if it did not succeed. | ||
""" | ||
|
||
buffer = bytearray([2]) | ||
curr_offset = Y_CABLE_SWITCH_MUX_DIRECTION | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).write_eeprom(curr_offset, 1, buffer) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to toggle mux to TOR A") | ||
return False | ||
|
||
return result | ||
|
||
|
||
def toggle_mux_to_torB(physical_port): | ||
""" | ||
This API specifically does a hard switch toggle of the Y cable's MUX regardless of link state to | ||
TOR B. This means if the Y cable is actively routing, the "check_active_linked_tor_side(physical_port)" | ||
API will now return Tor B. It also implies that if the link is actively routing on this port, Y cable | ||
MUX will start forwarding packets from TOR B to NIC, and drop packets from TOR A to NIC | ||
regardless of previous forwarding state. API basically writes to upper page 4 offset 130 the value | ||
of 0x3 and expects the MUX to toggle to TOR B. Bit 0 value 1 means TOR B | ||
Register Specification at offset 130 is documented below | ||
Byte offset bits Name Description | ||
130 7-2 Reserved Reserved | ||
1 Hard vs. soft switch 0b0 - Switch only if a valid TOR link on target; 0b1 Switch to new target regardless of link status | ||
0 Switch Target Switch Target - 0b0 - TOR#1, 0b1 - TOR#2; default is TOR #1 | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to Y end of a Y cable which can toggle the MUX | ||
Returns: | ||
a Boolean, true if the toggle succeeded and false if it did not succeed. | ||
""" | ||
|
||
buffer = bytearray([3]) | ||
curr_offset = Y_CABLE_SWITCH_MUX_DIRECTION | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).write_eeprom(curr_offset, 1, buffer) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to toggle mux to TOR B") | ||
return False | ||
|
||
return result | ||
|
||
|
||
def check_read_side(physical_port): | ||
""" | ||
This API specifically checks which side of the Y cable the reads are actually getting performed | ||
from, either TOR A or TOR B or NIC and returns the value. API basically reads 1 byte at upper | ||
page 4 offset 128 and checks which side of the Y cable the read is being performed from. | ||
Register Specification of upper page 0x4 at offset 128 is documented below | ||
Byte offset bits Name Description | ||
7-3 Reserved Determine which side of the cable you are reading from - specifically to differentiate TOR #1 and TOR #2: | ||
0b1 : Reading from indicated side, 0b0 not reading from that side. | ||
2 TOR #1 Side | ||
1 TOR #2 Side | ||
0 NIC Side | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to Y end of a Y cable which can which side reading the MUX from | ||
Returns: | ||
an Integer, 1 if reading the Y cable from TOR A side(TOR 1). | ||
, 2 if reading the Y cable from TOR B side(TOR 2). | ||
, 0 if reading the Y cable from NIC side. | ||
, -1 if reading the Y cable API fails. | ||
""" | ||
|
||
curr_offset = Y_CABLE_DETERMINE_CABLE_READ_SIDE | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).read_eeprom(curr_offset, 1) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to check read side") | ||
return -1 | ||
|
||
regval_read = struct.unpack(">i", result) | ||
|
||
if ((regval_read >> 2) & 0x01): | ||
helper_logger.log_info("Reading from TOR A") | ||
return 1 | ||
elif ((regval_read >> 1) & 0x01): | ||
helper_logger.log_info("Reading from TOR B") | ||
return 2 | ||
elif (regval_read & 0x01): | ||
helper_logger.log_info("Reading from NIC side") | ||
return 0 | ||
else: | ||
helper_logger.log_error( | ||
"Error: unknown status for checking which side regval = {} ".format(result)) | ||
return -1 | ||
|
||
return -1 | ||
|
||
|
||
def check_active_linked_tor_side(physical_port): | ||
""" | ||
This API specifically checks which side of the Y cable is actively linked and routing | ||
and returns either TOR A or TOR B. API basically reads 1 byte at upper page 4 offset 133 | ||
and checks which side is actively linked and routing. | ||
Register Specification of upper page 0x4 at offset 133 is documented below | ||
Byte offset bits Name Description | ||
133 7-0 TOR Active Indicator 0x00, no sides linked and routing frames, 0x01 TOR #1 linked and routing, 0x02, TOR #2 linked and routing | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to a Y cable | ||
Returns: | ||
an Integer, 1 if TOR A is actively linked and routing(TOR 1). | ||
, 2 if TOR B is actively linked and routing(TOR 2). | ||
, 0 if nothing linked and actively routing | ||
, -1 if checking which side linked for routing API fails. | ||
""" | ||
|
||
curr_offset = Y_CABLE_ACTIVE_TOR_INDICATOR | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).read_eeprom(curr_offset, 1) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to check Active Linked and routing TOR side") | ||
return -1 | ||
|
||
regval_read = struct.unpack(">i", result) | ||
|
||
if ((regval_read >> 1) & 0x01): | ||
helper_logger.log_info("TOR B active linked and actively routing") | ||
return 2 | ||
elif ((regval_read) & 0x01): | ||
helper_logger.log_info("TOR A standby linked and actively routing") | ||
return 1 | ||
elif regval_read == 0: | ||
helper_logger.log_info("Nothing linked for routing") | ||
return 0 | ||
else: | ||
helper_logger.log_error( | ||
"Error: unknown status for active TOR regval = {} ".format(result)) | ||
return -1 | ||
|
||
return -1 | ||
|
||
|
||
def check_if_link_is_active_for_NIC(physical_port): | ||
""" | ||
This API specifically checks if NIC side of the Y cable's link is active | ||
API basically reads 1 byte at upper page 4 offset 129 and checks if the link is active on NIC side | ||
Register Specification of upper page 0x4 at offset 129 is documented below | ||
Byte offset bits Name Description | ||
129 7-3 Reserved Cable link status is for each end. 0b1 : Link up, 0b0 link not up. | ||
2 TOR #1 Side | ||
1 TOR #2 Side | ||
0 NIC Side | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to a Y cable | ||
Returns: | ||
a boolean, true if the link is active | ||
, false if the link is not active | ||
""" | ||
curr_offset = Y_CABLE_CHECK_LINK_ACTIVE | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).read_eeprom(curr_offset, 1) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to check if link is Active on NIC side") | ||
return -1 | ||
|
||
regval_read = struct.unpack(">i", result) | ||
|
||
if (regval_read & 0x01): | ||
helper_logger.log_info("NIC link is up") | ||
return True | ||
else: | ||
return False | ||
|
||
|
||
def check_if_link_is_active_for_torA(physical_port): | ||
""" | ||
This API specifically checks if TOR A side of the Y cable's link is active | ||
API basically reads 1 byte at upper page 4 offset 129 and checks if the link is active on NIC side | ||
Register Specification of upper page 0x4 at offset 129 is documented below | ||
Byte offset bits Name Description | ||
129 7-3 Reserved Cable link status is for each end. 0b1 : Link up, 0b0 link not up. | ||
2 TOR #1 Side | ||
1 TOR #2 Side | ||
0 NIC Side | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to a Y cable | ||
Returns: | ||
a boolean, true if the link is active | ||
, false if the link is not active | ||
""" | ||
|
||
curr_offset = Y_CABLE_CHECK_LINK_ACTIVE | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).read_eeprom(curr_offset, 1) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to check if link is Active on TOR A side") | ||
return -1 | ||
|
||
regval_read = struct.unpack(">i", result) | ||
|
||
if ((regval_read >> 2) & 0x01): | ||
helper_logger.log_info("TOR A link is up") | ||
return True | ||
else: | ||
return False | ||
|
||
|
||
def check_if_link_is_active_for_torB(physical_port): | ||
""" | ||
This API specifically checks if TOR B side of the Y cable's link is active | ||
API basically reads 1 byte at upper page 4 offset 129 and checks if the link is active on NIC side | ||
Register Specification of upper page 0x4 at offset 129 is documented below | ||
Byte offset bits Name Description | ||
129 7-3 Reserved Cable link status is for each end. 0b1 : Link up, 0b0 link not up. | ||
2 TOR #1 Side | ||
1 TOR #2 Side | ||
0 NIC Side | ||
Args: | ||
physical_port: | ||
an Integer, the actual physical port connected to a Y cable | ||
Returns: | ||
a boolean, true if the link is active | ||
, false if the link is not active | ||
""" | ||
|
||
curr_offset = Y_CABLE_CHECK_LINK_ACTIVE | ||
|
||
if platform_chassis is not None: | ||
result = platform_chassis.get_sfp( | ||
physical_port).read_eeprom(curr_offset, 1) | ||
else: | ||
helper_logger.log_error("platform_chassis is not loaded, failed to check if link is Active on TOR B side") | ||
return -1 | ||
|
||
regval_read = struct.unpack(">i", result) | ||
|
||
if ((regval_read >> 1) & 0x01): | ||
helper_logger.log_info("TOR B link is up") | ||
return True | ||
else: | ||
return False |