sonic-buildimage/device/ragile/x86_64-ragile_ra-b6510-48v8c-r0/plugins/ssd_util.py

312 lines
12 KiB
Python
Raw Normal View History

#
# ssd_util.py
#
# Generic implementation of the SSD health API
# SSD models supported:
# - InnoDisk
# - StorFly
# - Virtium
try:
import re
import os
import subprocess
from sonic_platform_base.sonic_ssd.ssd_base import SsdBase
except ImportError as e:
raise ImportError (str(e) + "- required module not found")
SMARTCTL = "smartctl {} -a"
INNODISK = "iSmart -d {}"
VIRTIUM = "SmartCmd -m {}"
DISK_LIST_CMD = "fdisk -l -o Device"
DISK_FREE_CMD = "df -h"
MOUNT_CMD = "mount"
NOT_AVAILABLE = "N/A"
PE_CYCLE = 3000
FAIL_PERCENT = 95
# Set Vendor Specific IDs
INNODISK_HEALTH_ID = 169
INNODISK_TEMPERATURE_ID = 194
class SsdUtil(SsdBase):
"""
Generic implementation of the SSD health API
"""
model = NOT_AVAILABLE
serial = NOT_AVAILABLE
firmware = NOT_AVAILABLE
temperature = NOT_AVAILABLE
health = NOT_AVAILABLE
remaining_life = NOT_AVAILABLE
sata_rate = NOT_AVAILABLE
ssd_info = NOT_AVAILABLE
vendor_ssd_info = NOT_AVAILABLE
def __init__(self, diskdev):
self.vendor_ssd_utility = {
"Generic" : { "utility" : SMARTCTL, "parser" : self.parse_generic_ssd_info },
"InnoDisk" : { "utility" : INNODISK, "parser" : self.parse_innodisk_info },
"M.2" : { "utility" : INNODISK, "parser" : self.parse_innodisk_info },
"StorFly" : { "utility" : VIRTIUM, "parser" : self.parse_virtium_info },
"Virtium" : { "utility" : VIRTIUM, "parser" : self.parse_virtium_info }
}
"""
The dict model_attr keys relate the vendors
LITEON : "ER2-GD","AF2MA31DTDLT"
Intel : "SSDSCKKB"
SMI : "SM619GXC"
samsung: "MZNLH"
ADATA : "IM2S3134N"
"""
self.model_attr = {
"ER2-GD" : { "temperature" : "\n190\s+(.+?)\n", "remainingLife" : "\n202\s+(.+?)\n" },
"AF2MA31DTDLT" : { "temperature" : "\n194\s+(.+?)\n", "remainingLife" : "\n202\s+(.+?)\n" },
"SSDSCK" : { "temperature" : "\n194\s+(.+?)\n", "remainingLife" : "\n233\s+(.+?)\n" },
"SM619GXC" : { "temperature" : "\n194\s+(.+?)\n", "remainingLife" : "\n169\s+(.+?)\n" },
"MZNLH" : { "temperature" : "\n190\s+(.+?)\n", "remainingLife" : "\n245\s+(.+?)\n" },
"IM2S3134N" : { "temperature" : "\n194\s+(.+?)\n", "remainingLife" : "\n231\s+(.+?)\n" }
}
self.key_list = list(self.model_attr.keys())
self.attr_info_rule = "[\s\S]*SMART Attributes Data Structure revision number: 1|SMART Error Log Version[\s\S]*"
self.dev = diskdev
# Generic part
self.fetch_generic_ssd_info(diskdev)
self.parse_generic_ssd_info()
self.fetch_vendor_ssd_info(diskdev, "Generic")
# Known vendor part
if self.model:
model_short = self.model.split()[0]
if model_short in self.vendor_ssd_utility:
self.fetch_vendor_ssd_info(diskdev, model_short)
self.parse_vendor_ssd_info(model_short)
else:
# No handler registered for this disk model
pass
else:
# Failed to get disk model
self.model = "Unknown"
def _execute_shell(self, cmd):
process = subprocess.Popen(cmd.split(), universal_newlines=True, stdout=subprocess.PIPE)
output, error = process.communicate()
exit_code = process.returncode
if exit_code:
return None
return output
def _parse_re(self, pattern, buffer):
res_list = re.findall(pattern, str(buffer))
return res_list[0] if res_list else NOT_AVAILABLE
def fetch_generic_ssd_info(self, diskdev):
self.ssd_info = self._execute_shell(self.vendor_ssd_utility["Generic"]["utility"].format(diskdev))
# Health and temperature values may be overwritten with vendor specific data
def parse_generic_ssd_info(self):
if "nvme" in self.dev:
self.model = self._parse_re('Model Number:\s*(.+?)\n', self.ssd_info)
health_raw = self._parse_re('Percentage Used\s*(.+?)\n', self.ssd_info)
if health_raw == NOT_AVAILABLE:
self.health = NOT_AVAILABLE
else:
health_raw = health_raw.split()[-1]
self.health = 100 - float(health_raw.strip('%'))
temp_raw = self._parse_re('Temperature\s*(.+?)\n', self.ssd_info)
if temp_raw == NOT_AVAILABLE:
self.temperature = NOT_AVAILABLE
else:
temp_raw = temp_raw.split()[-2]
self.temperature = float(temp_raw)
else:
self.model = self._parse_re('Device Model:\s*(.+?)\n', self.ssd_info)
model_key = ""
for key in self.key_list:
if re.search(key, self.model):
model_key = key
break
if model_key != "":
self.remaining_life = self._parse_re(self.model_attr[model_key]["remainingLife"], re.sub(self.attr_info_rule,"",self.ssd_info)).split()[2]
self.temperature = self._parse_re(self.model_attr[model_key]["temperature"], re.sub(self.attr_info_rule,"",self.ssd_info)).split()[8]
self.health = self.remaining_life
# Get the LITEON ssd health value by (PE CYCLE - AVG ERASE CYCLE )/(PE CYCLE)
if model_key in ["ER2-GD", "AF2MA31DTDLT"]:
avg_erase = int(self._parse_re('\n173\s+(.+?)\n' ,re.sub(self.attr_info_rule,"",self.ssd_info)).split()[-1])
self.health = int(round((PE_CYCLE - avg_erase)/PE_CYCLE*100,0))
if self.remaining_life != NOT_AVAILABLE and int(self.remaining_life) < FAIL_PERCENT:
self.remaining_life = "Fail"
self.sata_rate = self._parse_re('SATA Version is:.*current: (.+?)\)\n', self.ssd_info)
self.serial = self._parse_re('Serial Number:\s*(.+?)\n', self.ssd_info)
self.firmware = self._parse_re('Firmware Version:\s*(.+?)\n', self.ssd_info)
def parse_innodisk_info(self):
if self.vendor_ssd_info:
self.health = self._parse_re('Health:\s*(.+?)%', self.vendor_ssd_info)
self.temperature = self._parse_re('Temperature\s*\[\s*(.+?)\]', self.vendor_ssd_info)
else:
if self.health == NOT_AVAILABLE:
health_raw = self.parse_id_number(INNODISK_HEALTH_ID)
self.health = health_raw.split()[-1]
if self.temperature == NOT_AVAILABLE:
temp_raw = self.parse_id_number(INNODISK_TEMPERATURE_ID)
self.temperature = temp_raw.split()[-6]
def parse_virtium_info(self):
if self.vendor_ssd_info:
self.temperature = self._parse_re('Temperature_Celsius\s*\d*\s*(\d+?)\s+', self.vendor_ssd_info)
nand_endurance = self._parse_re('NAND_Endurance\s*\d*\s*(\d+?)\s+', self.vendor_ssd_info)
avg_erase_count = self._parse_re('Average_Erase_Count\s*\d*\s*(\d+?)\s+', self.vendor_ssd_info)
try:
self.health = 100 - (float(avg_erase_count) * 100 / float(nand_endurance))
except (ValueError, ZeroDivisionError):
# Invalid avg_erase_count or nand_endurance.
pass
def fetch_vendor_ssd_info(self, diskdev, model):
self.vendor_ssd_info = self._execute_shell(self.vendor_ssd_utility[model]["utility"].format(diskdev))
def parse_vendor_ssd_info(self, model):
self.vendor_ssd_utility[model]["parser"]()
def check_readonly2(self, partition, filesystem):
# parse mount cmd output info
mount_info = self._execute_shell(MOUNT_CMD)
for line in mount_info.split('\n'):
column_list = line.split()
if line == '':
continue
if column_list[0] == partition and column_list[2] == filesystem:
if column_list[5].split(',')[0][1:] == "ro":
return partition
else:
return NOT_AVAILABLE
return NOT_AVAILABLE
def check_readonly(self, partition, filesystem):
ret = os.access(filesystem, os.W_OK)
if ret == False:
return partition
else:
return NOT_AVAILABLE
def get_health(self):
"""
Retrieves current disk health in percentages
Returns:
A float number of current ssd health
e.g. 83.5
"""
return float(self.health)
def get_temperature(self):
"""
Retrieves current disk temperature in Celsius
Returns:
A float number of current temperature in Celsius
e.g. 40.1
"""
return float(self.temperature)
def get_model(self):
"""
Retrieves model for the given disk device
Returns:
A string holding disk model as provided by the manufacturer
"""
return self.model
def get_firmware(self):
"""
Retrieves firmware version for the given disk device
Returns:
A string holding disk firmware version as provided by the manufacturer
"""
return self.firmware
def get_serial(self):
"""
Retrieves serial number for the given disk device
Returns:
A string holding disk serial number as provided by the manufacturer
"""
return self.serial
def get_sata_rate(self):
"""
Retrieves SATA rate for the given disk device
Returns:
A string holding current SATA rate as provided by the manufacturer
"""
return self.sata_rate
def get_remaining_life(self):
"""
Retrieves remaining life for the given disk device
Returns:
A string holding disk remaining life as provided by the manufacturer
"""
return self.remaining_life
def get_vendor_output(self):
"""
Retrieves vendor specific data for the given disk device
Returns:
A string holding some vendor specific disk information
"""
return self.vendor_ssd_info
def parse_id_number(self, id):
return self._parse_re('{}\s*(.+?)\n'.format(id), self.ssd_info)
def get_readonly_partition(self):
"""
Check the partition mount filesystem is readonly status,then output the result.
Returns:
The readonly partition list
"""
ro_partition_list = []
partition_list = []
# parse fdisk cmd output info
disk_info = self._execute_shell(DISK_LIST_CMD)
begin_flag = False
for line in disk_info.split('\n'):
if line == "Device":
begin_flag = True
continue
if begin_flag:
if line != "":
partition_list.append(line)
else:
break
# parse df cmd output info
disk_free = self._execute_shell(DISK_FREE_CMD)
disk_dict = {}
line_num = 0
for line in disk_free.split('\n'):
line_num = line_num + 1
if line_num == 1 or line == "":
continue
column_list = line.split()
disk_dict[column_list[0]] = column_list[5]
# get partition which is readonly
for partition in partition_list:
if partition in disk_dict:
ret = self.check_readonly(partition, disk_dict[partition])
if (ret != NOT_AVAILABLE):
ro_partition_list.append(ret)
return ro_partition_list