2019-11-06 20:56:17 -06:00
|
|
|
#############################################################################
|
|
|
|
# Celestica
|
|
|
|
#
|
|
|
|
# Module contains an implementation of SONiC Platform Base API and
|
|
|
|
# provides the Chassis information which are available in the platform
|
|
|
|
#
|
|
|
|
#############################################################################
|
|
|
|
|
|
|
|
import sys
|
|
|
|
import re
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import json
|
|
|
|
|
|
|
|
try:
|
|
|
|
from sonic_platform_base.chassis_base import ChassisBase
|
2020-01-15 20:43:48 -06:00
|
|
|
from sonic_platform.component import Component
|
2019-11-06 20:56:17 -06:00
|
|
|
from sonic_platform.eeprom import Tlv
|
|
|
|
from sonic_platform.fan import Fan
|
2020-01-15 20:45:01 -06:00
|
|
|
from sonic_platform.sfp import Sfp
|
2019-12-04 21:00:13 -06:00
|
|
|
from sonic_platform.psu import Psu
|
2020-01-15 20:43:48 -06:00
|
|
|
from sonic_platform.thermal import Thermal
|
2020-11-25 12:28:36 -06:00
|
|
|
from .helper import APIHelper
|
2019-11-06 20:56:17 -06:00
|
|
|
except ImportError as e:
|
|
|
|
raise ImportError(str(e) + "- required module not found")
|
|
|
|
|
|
|
|
NUM_FAN_TRAY = 7
|
|
|
|
NUM_FAN = 2
|
|
|
|
NUM_PSU = 2
|
2020-01-15 20:43:48 -06:00
|
|
|
NUM_THERMAL = 10
|
2019-11-06 20:56:17 -06:00
|
|
|
NUM_SFP = 32
|
|
|
|
NUM_COMPONENT = 5
|
|
|
|
|
|
|
|
IPMI_OEM_NETFN = "0x3A"
|
|
|
|
IPMI_GET_REBOOT_CAUSE = "0x03 0x00 0x01 0x06"
|
|
|
|
|
|
|
|
|
|
|
|
class Chassis(ChassisBase):
|
|
|
|
"""Platform-specific Chassis class"""
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.config_data = {}
|
|
|
|
ChassisBase.__init__(self)
|
|
|
|
self._eeprom = Tlv()
|
|
|
|
self._api_helper = APIHelper()
|
|
|
|
|
|
|
|
for fant_index in range(0, NUM_FAN_TRAY):
|
|
|
|
for fan_index in range(0, NUM_FAN):
|
|
|
|
fan = Fan(fant_index, fan_index)
|
|
|
|
self._fan_list.append(fan)
|
2020-01-15 20:45:01 -06:00
|
|
|
|
|
|
|
for index in range(0, NUM_SFP):
|
|
|
|
sfp = Sfp(index)
|
|
|
|
self._sfp_list.append(sfp)
|
|
|
|
|
2019-12-04 21:00:13 -06:00
|
|
|
for index in range(0, NUM_PSU):
|
|
|
|
psu = Psu(index)
|
|
|
|
self._psu_list.append(psu)
|
2020-01-15 20:43:48 -06:00
|
|
|
for index in range(0, NUM_COMPONENT):
|
|
|
|
component = Component(index)
|
|
|
|
self._component_list.append(component)
|
|
|
|
for index in range(0, NUM_THERMAL):
|
|
|
|
thermal = Thermal(index)
|
|
|
|
self._thermal_list.append(thermal)
|
2019-11-06 20:56:17 -06:00
|
|
|
|
|
|
|
def get_base_mac(self):
|
|
|
|
"""
|
|
|
|
Retrieves the base MAC address for the chassis
|
|
|
|
Returns:
|
|
|
|
A string containing the MAC address in the format
|
|
|
|
'XX:XX:XX:XX:XX:XX'
|
|
|
|
"""
|
|
|
|
return self._eeprom.get_mac()
|
|
|
|
|
2020-10-18 00:00:14 -05:00
|
|
|
def get_serial(self):
|
2019-11-06 20:56:17 -06:00
|
|
|
"""
|
|
|
|
Retrieves the hardware serial number for the chassis
|
|
|
|
Returns:
|
|
|
|
A string containing the hardware serial number for this chassis.
|
|
|
|
"""
|
|
|
|
return self._eeprom.get_serial()
|
|
|
|
|
|
|
|
def get_system_eeprom_info(self):
|
|
|
|
"""
|
|
|
|
Retrieves the full content of system EEPROM information for the chassis
|
|
|
|
Returns:
|
|
|
|
A dictionary where keys are the type code defined in
|
|
|
|
OCP ONIE TlvInfo EEPROM format and values are their corresponding
|
|
|
|
values.
|
|
|
|
"""
|
|
|
|
return self._eeprom.get_eeprom()
|
|
|
|
|
|
|
|
def get_reboot_cause(self):
|
|
|
|
"""
|
|
|
|
Retrieves the cause of the previous reboot
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
A tuple (string, string) where the first element is a string
|
|
|
|
containing the cause of the previous reboot. This string must be
|
|
|
|
one of the predefined strings in this class. If the first string
|
|
|
|
is "REBOOT_CAUSE_HARDWARE_OTHER", the second string can be used
|
|
|
|
to pass a description of the reboot cause.
|
|
|
|
"""
|
|
|
|
|
|
|
|
status, raw_cause = self._api_helper.ipmi_raw(
|
|
|
|
IPMI_OEM_NETFN, IPMI_GET_REBOOT_CAUSE)
|
2020-01-15 20:43:48 -06:00
|
|
|
hx_cause = raw_cause.split()[0] if status and len(
|
|
|
|
raw_cause.split()) > 0 else 00
|
2019-11-06 20:56:17 -06:00
|
|
|
reboot_cause = {
|
|
|
|
"00": self.REBOOT_CAUSE_HARDWARE_OTHER,
|
|
|
|
"11": self.REBOOT_CAUSE_POWER_LOSS,
|
|
|
|
"22": self.REBOOT_CAUSE_NON_HARDWARE,
|
|
|
|
"33": self.REBOOT_CAUSE_HARDWARE_OTHER,
|
|
|
|
"44": self.REBOOT_CAUSE_NON_HARDWARE,
|
|
|
|
"55": self.REBOOT_CAUSE_NON_HARDWARE,
|
|
|
|
"66": self.REBOOT_CAUSE_WATCHDOG,
|
|
|
|
"77": self.REBOOT_CAUSE_NON_HARDWARE
|
|
|
|
}.get(hx_cause, self.REBOOT_CAUSE_HARDWARE_OTHER)
|
|
|
|
|
|
|
|
description = {
|
|
|
|
"00": "Unknown reason",
|
|
|
|
"11": "The last reset is Power on reset",
|
|
|
|
"22": "The last reset is soft-set CPU warm reset",
|
|
|
|
"33": "The last reset is soft-set CPU cold reset",
|
|
|
|
"44": "The last reset is CPU warm reset",
|
|
|
|
"55": "The last reset is CPU cold reset",
|
|
|
|
"66": "The last reset is watchdog reset",
|
|
|
|
"77": "The last reset is power cycle reset"
|
|
|
|
}.get(hx_cause, "Unknown reason")
|
|
|
|
|
|
|
|
return (reboot_cause, description)
|