2019-07-02 13:05:18 -05:00
|
|
|
#############################################################################
|
|
|
|
# Celestica
|
|
|
|
#
|
|
|
|
# Component contains an implementation of SONiC Platform Base API and
|
|
|
|
# provides the components firmware management function
|
|
|
|
#
|
|
|
|
#############################################################################
|
|
|
|
|
|
|
|
import os.path
|
|
|
|
import shutil
|
|
|
|
import subprocess
|
|
|
|
|
|
|
|
try:
|
2019-09-27 14:44:16 -05:00
|
|
|
from sonic_platform_base.component_base import ComponentBase
|
2020-11-25 12:28:36 -06:00
|
|
|
from .helper import APIHelper
|
2019-07-02 13:05:18 -05:00
|
|
|
except ImportError as e:
|
|
|
|
raise ImportError(str(e) + "- required module not found")
|
|
|
|
|
|
|
|
CPLD_ADDR_MAPPING = {
|
|
|
|
"CPLD1": "0x100",
|
|
|
|
"CPLD2": "0x200",
|
|
|
|
"CPLD3": "0x280",
|
|
|
|
"CPLD4": "0x300",
|
|
|
|
"CPLD5": "0x380"
|
|
|
|
}
|
|
|
|
GETREG_PATH = "/sys/devices/platform/dx010_cpld/getreg"
|
|
|
|
BIOS_VERSION_PATH = "/sys/class/dmi/id/bios_version"
|
2023-01-18 18:27:48 -06:00
|
|
|
COMPONENT_NAME_LIST = ["CPLD1", "CPLD2", "CPLD3", "CPLD4", "CPLD5", "BIOS"]
|
2020-05-22 05:50:43 -05:00
|
|
|
COMPONENT_DES_LIST = ["Used for managing the CPU",
|
2023-01-18 18:27:48 -06:00
|
|
|
"Used for managing QSFP+ ports (1-10)", "Used for managing QSFP+ ports (11-21)",
|
|
|
|
"Used for misc status and control", "Used for managing QSFP+ ports (22-32)",
|
|
|
|
"Basic Input/Output System"]
|
2019-07-02 13:05:18 -05:00
|
|
|
|
|
|
|
|
2019-09-27 14:44:16 -05:00
|
|
|
class Component(ComponentBase):
|
2019-07-02 13:05:18 -05:00
|
|
|
"""Platform-specific Component class"""
|
|
|
|
|
|
|
|
DEVICE_TYPE = "component"
|
|
|
|
|
2019-09-27 14:44:16 -05:00
|
|
|
def __init__(self, component_index):
|
|
|
|
ComponentBase.__init__(self)
|
|
|
|
self.index = component_index
|
2020-05-22 05:50:43 -05:00
|
|
|
self._api_helper = APIHelper()
|
2019-09-27 14:44:16 -05:00
|
|
|
self.name = self.get_name()
|
2019-07-02 13:05:18 -05:00
|
|
|
|
|
|
|
def __get_bios_version(self):
|
|
|
|
# Retrieves the BIOS firmware version
|
|
|
|
try:
|
|
|
|
with open(BIOS_VERSION_PATH, 'r') as fd:
|
|
|
|
bios_version = fd.read()
|
|
|
|
return bios_version.strip()
|
|
|
|
except Exception as e:
|
|
|
|
return None
|
|
|
|
|
2019-07-08 13:26:27 -05:00
|
|
|
def get_register_value(self, register):
|
|
|
|
# Retrieves the cpld register value
|
|
|
|
cmd = "echo {1} > {0}; cat {0}".format(GETREG_PATH, register)
|
|
|
|
p = subprocess.Popen(
|
2020-11-25 12:28:36 -06:00
|
|
|
cmd, shell=True, universal_newlines=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
|
2019-07-08 13:26:27 -05:00
|
|
|
raw_data, err = p.communicate()
|
|
|
|
if err is not '':
|
|
|
|
return None
|
|
|
|
return raw_data.strip()
|
|
|
|
|
2019-07-02 13:05:18 -05:00
|
|
|
def __get_cpld_version(self):
|
|
|
|
# Retrieves the CPLD firmware version
|
|
|
|
cpld_version = dict()
|
|
|
|
for cpld_name in CPLD_ADDR_MAPPING:
|
|
|
|
try:
|
|
|
|
cpld_addr = CPLD_ADDR_MAPPING[cpld_name]
|
2019-07-08 13:26:27 -05:00
|
|
|
cpld_version_raw = self.get_register_value(cpld_addr)
|
2019-07-02 13:05:18 -05:00
|
|
|
cpld_version_str = "{}.{}".format(int(cpld_version_raw[2], 16), int(
|
|
|
|
cpld_version_raw[3], 16)) if cpld_version_raw is not None else 'None'
|
|
|
|
cpld_version[cpld_name] = cpld_version_str
|
|
|
|
except Exception as e:
|
|
|
|
cpld_version[cpld_name] = 'None'
|
|
|
|
return cpld_version
|
|
|
|
|
2019-09-27 14:44:16 -05:00
|
|
|
def get_name(self):
|
|
|
|
"""
|
|
|
|
Retrieves the name of the component
|
|
|
|
Returns:
|
|
|
|
A string containing the name of the component
|
|
|
|
"""
|
|
|
|
return COMPONENT_NAME_LIST[self.index]
|
|
|
|
|
|
|
|
def get_description(self):
|
|
|
|
"""
|
|
|
|
Retrieves the description of the component
|
|
|
|
Returns:
|
|
|
|
A string containing the description of the component
|
|
|
|
"""
|
|
|
|
return COMPONENT_DES_LIST[self.index]
|
|
|
|
|
2019-07-02 13:05:18 -05:00
|
|
|
def get_firmware_version(self):
|
|
|
|
"""
|
|
|
|
Retrieves the firmware version of module
|
|
|
|
Returns:
|
|
|
|
string: The firmware versions of the module
|
|
|
|
"""
|
|
|
|
fw_version = None
|
|
|
|
|
|
|
|
if self.name == "BIOS":
|
|
|
|
fw_version = self.__get_bios_version()
|
|
|
|
elif "CPLD" in self.name:
|
|
|
|
cpld_version = self.__get_cpld_version()
|
|
|
|
fw_version = cpld_version.get(self.name)
|
|
|
|
|
|
|
|
return fw_version
|
|
|
|
|
2021-05-28 14:56:09 -05:00
|
|
|
def get_available_firmware_version(self, image_path):
|
|
|
|
"""
|
|
|
|
Retrieves the available firmware version of the component
|
|
|
|
Note: the firmware version will be read from image
|
|
|
|
Args:
|
|
|
|
image_path: A string, path to firmware image
|
|
|
|
Returns:
|
|
|
|
A string containing the available firmware version of the component
|
|
|
|
"""
|
|
|
|
return "N/A"
|
|
|
|
|
|
|
|
def get_firmware_update_notification(self, image_path):
|
|
|
|
"""
|
|
|
|
Retrieves a notification on what should be done in order to complete
|
|
|
|
the component firmware update
|
|
|
|
Args:
|
|
|
|
image_path: A string, path to firmware image
|
|
|
|
Returns:
|
|
|
|
A string containing the component firmware update notification if required.
|
|
|
|
By default 'None' value will be used, which indicates that no actions are required
|
|
|
|
"""
|
|
|
|
return "None"
|
|
|
|
|
2019-09-27 14:44:16 -05:00
|
|
|
def install_firmware(self, image_path):
|
2019-07-02 13:05:18 -05:00
|
|
|
"""
|
|
|
|
Install firmware to module
|
|
|
|
Args:
|
|
|
|
image_path: A string, path to firmware image
|
|
|
|
Returns:
|
|
|
|
A boolean, True if install successfully, False if not
|
|
|
|
"""
|
|
|
|
if not os.path.isfile(image_path):
|
|
|
|
return False
|
|
|
|
|
|
|
|
if "CPLD" in self.name:
|
|
|
|
img_name = os.path.basename(image_path)
|
|
|
|
root, ext = os.path.splitext(img_name)
|
|
|
|
ext = ".vme" if ext == "" else ext
|
|
|
|
new_image_path = os.path.join("/tmp", (root.lower() + ext))
|
|
|
|
shutil.copy(image_path, new_image_path)
|
|
|
|
install_command = "ispvm %s" % new_image_path
|
2019-12-16 09:30:35 -06:00
|
|
|
# elif self.name == "BIOS":
|
|
|
|
# install_command = "afulnx_64 %s /p /b /n /x /r" % image_path
|
2019-07-02 13:05:18 -05:00
|
|
|
|
|
|
|
return self.__run_command(install_command)
|
2021-04-02 12:08:31 -05:00
|
|
|
|
|
|
|
|
2021-05-28 14:56:09 -05:00
|
|
|
def update_firmware(self, image_path):
|
|
|
|
"""
|
|
|
|
Updates firmware of the component
|
|
|
|
This API performs firmware update: it assumes firmware installation and loading in a single call.
|
|
|
|
In case platform component requires some extra steps (apart from calling Low Level Utility)
|
|
|
|
to load the installed firmware (e.g, reboot, power cycle, etc.) - this will be done automatically by API
|
|
|
|
Args:
|
|
|
|
image_path: A string, path to firmware image
|
|
|
|
Raises:
|
|
|
|
RuntimeError: update failed
|
|
|
|
"""
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2021-04-02 12:08:31 -05:00
|
|
|
##############################################################
|
|
|
|
###################### Device methods ########################
|
|
|
|
##############################################################
|
|
|
|
|
|
|
|
|
|
|
|
def get_presence(self):
|
|
|
|
"""
|
|
|
|
Retrieves the presence of the FAN
|
|
|
|
Returns:
|
|
|
|
bool: True if FAN is present, False if not
|
|
|
|
"""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_model(self):
|
|
|
|
"""
|
|
|
|
Retrieves the model number (or part number) of the device
|
|
|
|
Returns:
|
|
|
|
string: Model/part number of device
|
|
|
|
"""
|
|
|
|
return 'N/A'
|
|
|
|
|
|
|
|
def get_serial(self):
|
|
|
|
"""
|
|
|
|
Retrieves the serial number of the device
|
|
|
|
Returns:
|
|
|
|
string: Serial number of device
|
|
|
|
"""
|
|
|
|
return 'N/A'
|
|
|
|
|
|
|
|
def get_status(self):
|
|
|
|
"""
|
|
|
|
Retrieves the operational status of the device
|
|
|
|
Returns:
|
|
|
|
A boolean value, True if device is operating properly, False if not
|
|
|
|
"""
|
|
|
|
return True
|
|
|
|
|
|
|
|
def get_position_in_parent(self):
|
|
|
|
"""
|
|
|
|
Retrieves 1-based relative physical position in parent device.
|
|
|
|
If the agent cannot determine the parent-relative position
|
|
|
|
for some reason, or if the associated value of
|
|
|
|
entPhysicalContainedIn is'0', then the value '-1' is returned
|
|
|
|
Returns:
|
|
|
|
integer: The 1-based relative physical position in parent device
|
|
|
|
or -1 if cannot determine the position
|
|
|
|
"""
|
|
|
|
return -1
|
|
|
|
|
|
|
|
def is_replaceable(self):
|
|
|
|
"""
|
|
|
|
Indicate whether this device is replaceable.
|
|
|
|
Returns:
|
|
|
|
bool: True if it is replaceable.
|
|
|
|
"""
|
|
|
|
return False
|