2022-06-15 11:39:41 -05:00
|
|
|
#
|
|
|
|
# psuutil.py
|
|
|
|
# Platform-specific PSU status interface for SONiC
|
|
|
|
#
|
|
|
|
import logging
|
2023-01-05 18:22:09 -06:00
|
|
|
from sonic_py_common.general import getstatusoutput_noshell, getstatusoutput_noshell_pipe
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
Z9332F_MAX_PSUS = 2
|
|
|
|
FRU_PSUL = 11
|
|
|
|
FRU_PSUR = 12
|
|
|
|
|
|
|
|
|
|
|
|
try:
|
|
|
|
from sonic_psu.psu_base import PsuBase
|
|
|
|
except ImportError as e:
|
|
|
|
raise ImportError(str(e) + "- required module not found")
|
|
|
|
|
|
|
|
|
|
|
|
class PsuUtil(PsuBase):
|
|
|
|
"""Platform-specific PSUutil class"""
|
2023-01-05 18:22:09 -06:00
|
|
|
IPMI_PSU1_DATA = ["ipmitool", "raw", "0x04", "0x2d", "0x2f"]
|
|
|
|
IPMI_PSU2_DATA = ["ipmitool", "raw", "0x04", "0x2d", "0x39"]
|
|
|
|
IPMI_PSU_VOUT = ["ipmitool", "sdr", "get", ""]
|
|
|
|
IPMI_PSU_POUT = ["ipmitool", "sdr", "get", ""]
|
|
|
|
IPMI_PSU_COUT = ["ipmitool", "sdr", "get", ""]
|
|
|
|
IPMI_PSU_FAN_SPEED = ["ipmitool", "sdr", "get", ""]
|
|
|
|
IPMI_FRU = ["ipmitool", "fru"]
|
|
|
|
IPMI_FRU_DATA = ["ipmitool", "fru", "print", ""]
|
|
|
|
awk_cmd = ['awk', '{print substr($0,9,1)}']
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
PsuBase.__init__(self)
|
|
|
|
|
|
|
|
def isDockerEnv(self):
|
|
|
|
num_docker = open('/proc/self/cgroup', 'r').read().count(":/docker")
|
|
|
|
if num_docker > 0:
|
|
|
|
return True
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
|
|
|
|
def get_psu_vout(self,index):
|
|
|
|
try:
|
2023-01-05 18:22:09 -06:00
|
|
|
self.IPMI_PSU_VOUT[3] = 'PSU' + str(index) + '_VOut'
|
|
|
|
ret_status, ipmi_cmd_ret = getstatusoutput_noshell(self.IPMI_PSU_VOUT)
|
|
|
|
if ret_status == 0:
|
|
|
|
rdata = ipmi_cmd_ret.splitlines()[3].split(':')[1].split(' ')[1]
|
|
|
|
return rdata
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
except Exception:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute : %s'%(' '.join(self.IPMI_PSU_VOUT)))
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
def get_psu_cout(self,index):
|
|
|
|
try:
|
2023-01-05 18:22:09 -06:00
|
|
|
self.IPMI_PSU_COUT[3] = 'PSU' + str(index) + 'POut'
|
|
|
|
ret_status, ipmi_cmd_ret = getstatusoutput_noshell(self.IPMI_PSU_COUT)
|
|
|
|
if ret_status == 0:
|
|
|
|
rdata = ipmi_cmd_ret.splitlines()[3].split(':')[1].split(' ')[1]
|
|
|
|
return rdata
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
except Exception:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute : %s'%(' '.join(self.IPMI_PSU_COUT)))
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
def get_psu_pout(self,index):
|
|
|
|
try:
|
2023-01-05 18:22:09 -06:00
|
|
|
self.IPMI_PSU_POUT[3] = 'PSU' + str(index) + '_COut'
|
|
|
|
ret_status, ipmi_cmd_ret = getstatusoutput_noshell(self.IPMI_PSU_POUT)
|
|
|
|
if ret_status == 0:
|
|
|
|
rdata = ipmi_cmd_ret.splitlines()[3].split(':')[1].split(' ')[1]
|
|
|
|
return rdata
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
except Exception:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute : %s'%(' '.join(self.IPMI_PSU_POUT)))
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
def get_psu_fan_speed(self,index):
|
|
|
|
try:
|
2023-01-05 18:22:09 -06:00
|
|
|
self.IPMI_PSU_FAN_SPEED[3] = 'PSU' + str(index) + '_Fan'
|
|
|
|
ret_status, ipmi_cmd_ret = getstatusoutput_noshell(self.IPMI_PSU_FAN_SPEED)
|
|
|
|
if ret_status == 0:
|
|
|
|
rdata = ipmi_cmd_ret.splitlines()[3].split(':')[1].split(' ')[1]
|
|
|
|
return rdata
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
except Exception:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute : %s'%(' '.join(self.IPMI_PSU_FAN_SPEED)))
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
|
|
|
|
#Fetch FRU Data for given fruid
|
|
|
|
def get_psu_airflow(self, index):
|
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = 'FRU_PSUL'
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = 'FRU_PSUR'
|
2022-06-15 11:39:41 -05:00
|
|
|
|
2023-01-05 18:22:09 -06:00
|
|
|
ret_status, ipmi_cmd_ret = getstatusoutput_noshell(self.IPMI_FRU)
|
2022-06-15 11:39:41 -05:00
|
|
|
if ret_status:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute ipmitool: '+ self.IPMI_FRU)
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
found_fru = False
|
|
|
|
for line in ipmi_cmd_ret.splitlines():
|
|
|
|
if line.startswith('FRU Device Description') and fru_id in line.split(':')[1] :
|
2023-01-05 18:22:09 -06:00
|
|
|
found_fru = True
|
2022-06-15 11:39:41 -05:00
|
|
|
if found_fru and line.startswith(' Board Product '):
|
2023-01-05 18:22:09 -06:00
|
|
|
return ' B2F' if 'PS/IO' in line else ' F2B'
|
2022-06-15 11:39:41 -05:00
|
|
|
return ''
|
|
|
|
|
|
|
|
# Read FRU info
|
|
|
|
def get_fru_info(self,fru_id,reg_name):
|
|
|
|
output = None
|
|
|
|
Found = False
|
|
|
|
try:
|
2023-01-05 18:22:09 -06:00
|
|
|
self.IPMI_FRU_DATA[3] = str(fru_id)
|
|
|
|
status, ipmi_fru_list = getstatusoutput_noshell(self.IPMI_FRU_DATA)
|
|
|
|
if status == 0:
|
|
|
|
for item in ipmi_fru_list.split("\n"):
|
|
|
|
if reg_name == item.split(':')[0].strip(' '):
|
|
|
|
output = item.strip()
|
|
|
|
output = output.split(':')[1]
|
|
|
|
Found = True
|
|
|
|
break;
|
|
|
|
|
|
|
|
if not Found:
|
|
|
|
logging.error('\nFailed to fetch: ' + reg_name + ' sensor ')
|
|
|
|
|
|
|
|
return output
|
2022-06-15 11:39:41 -05:00
|
|
|
except Exception:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute:' + ipmi_fru_list)
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
def get_num_psus(self):
|
|
|
|
"""
|
|
|
|
Retrieves the number of PSUs available on the device
|
|
|
|
:return: An integer, the number of PSUs available on the device
|
|
|
|
"""
|
|
|
|
Z9332F_MAX_PSUS = 2
|
|
|
|
return Z9332F_MAX_PSUS
|
|
|
|
|
|
|
|
def get_psu_status(self, index):
|
|
|
|
"""
|
|
|
|
Retrieves the oprational status of power supply unit (PSU) defined
|
|
|
|
by index <index>
|
|
|
|
:param index: An integer, index of the PSU of which to query status
|
|
|
|
:return: Boolean, True if PSU is operating properly, False if PSU is\
|
|
|
|
faulty
|
|
|
|
"""
|
|
|
|
psu_status = '0'
|
2023-01-05 18:22:09 -06:00
|
|
|
ipmi_cmd = ''
|
2022-06-15 11:39:41 -05:00
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
ipmi_cmd = self.IPMI_PSU1_DATA
|
2022-06-15 11:39:41 -05:00
|
|
|
elif index == 2:
|
2023-01-05 18:22:09 -06:00
|
|
|
ipmi_cmd = self.IPMI_PSU2_DATA
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error("Invalid PSU number:" + index)
|
2022-06-15 11:39:41 -05:00
|
|
|
|
2023-01-05 18:22:09 -06:00
|
|
|
if ipmi_cmd != '':
|
|
|
|
cmd_status, psu_status = getstatusoutput_noshell_pipe(ipmi_cmd, self.awk_cmd)
|
|
|
|
if cmd_status:
|
|
|
|
logging.error('Failed to execute ipmitool')
|
2022-06-15 11:39:41 -05:00
|
|
|
return (not int(psu_status, 16) > 1)
|
|
|
|
|
|
|
|
def get_psu_presence(self, index):
|
|
|
|
"""
|
|
|
|
Retrieves the presence status of power supply unit (PSU) defined
|
|
|
|
by index <index>
|
|
|
|
:param index: An integer, index of the PSU of which to query status
|
|
|
|
:return: Boolean, True if PSU is plugged, False if not
|
|
|
|
"""
|
|
|
|
|
2023-01-05 18:22:09 -06:00
|
|
|
psu_status = '0'
|
|
|
|
ipmi_cmd = ''
|
2022-06-15 11:39:41 -05:00
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
ipmi_cmd = self.IPMI_PSU1_DATA
|
2022-06-15 11:39:41 -05:00
|
|
|
elif index == 2:
|
2023-01-05 18:22:09 -06:00
|
|
|
ipmi_cmd = self.IPMI_PSU2_DATA
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error("Invalid PSU number:" + index)
|
|
|
|
|
|
|
|
if ipmi_cmd != '':
|
|
|
|
cmd_status, psu_status = getstatusoutput_noshell_pipe(ipmi_cmd, self.awk_cmd)
|
2022-06-15 11:39:41 -05:00
|
|
|
if cmd_status:
|
2023-01-05 18:22:09 -06:00
|
|
|
logging.error('Failed to execute ipmitool')
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
return (int(psu_status, 16) & 1)
|
|
|
|
|
|
|
|
def get_output_voltage(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return 0.0
|
2022-06-15 11:39:41 -05:00
|
|
|
psuvoltage=self.get_psu_vout(index)
|
|
|
|
return float(psuvoltage.strip())
|
|
|
|
|
|
|
|
def get_output_current(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return 0.0
|
2022-06-15 11:39:41 -05:00
|
|
|
psucurrent=self.get_psu_cout(index)
|
|
|
|
return float(psucurrent.strip())
|
|
|
|
|
|
|
|
def get_output_power(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return 0.0
|
2022-06-15 11:39:41 -05:00
|
|
|
psupower=self.get_psu_pout(index)
|
|
|
|
return float(psupower.strip())
|
|
|
|
|
|
|
|
def get_fan_rpm(self, index, fan_idx):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return 0
|
2022-06-15 11:39:41 -05:00
|
|
|
fanrpm=self.get_psu_fan_speed(index)
|
|
|
|
return int(fanrpm.strip())
|
|
|
|
|
|
|
|
def get_serial(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return None
|
2022-06-15 11:39:41 -05:00
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUL
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUR
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
return self.get_fru_info(fru_id,'Board Serial')
|
|
|
|
|
|
|
|
def get_model(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return None
|
2022-06-15 11:39:41 -05:00
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUL
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUR
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
return self.get_fru_info(fru_id,'Board Part Number')
|
|
|
|
|
|
|
|
def get_mfr_id(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return None
|
2022-06-15 11:39:41 -05:00
|
|
|
if index == 1:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUL
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
fru_id = FRU_PSUR
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
return self.get_fru_info(fru_id,'Board Mfg')
|
|
|
|
|
|
|
|
def get_direction(self, index):
|
|
|
|
if index is None:
|
2023-01-05 18:22:09 -06:00
|
|
|
return None
|
2022-06-15 11:39:41 -05:00
|
|
|
|
|
|
|
direction=self.get_psu_airflow(index).strip()
|
|
|
|
if direction == 'B2F':
|
2023-01-05 18:22:09 -06:00
|
|
|
return "INTAKE"
|
2022-06-15 11:39:41 -05:00
|
|
|
elif direction == 'F2B':
|
2023-01-05 18:22:09 -06:00
|
|
|
return "EXHAUST"
|
2022-06-15 11:39:41 -05:00
|
|
|
else:
|
2023-01-05 18:22:09 -06:00
|
|
|
return None
|