158 lines
5.3 KiB
Python
Executable File
158 lines
5.3 KiB
Python
Executable File
#!/usr/bin/python3
|
|
# -*- coding: utf-8 -*-
|
|
# Copyright (c) 2019 Huawei Technologies Co., Ltd.
|
|
# A-Tune is licensed under the Mulan PSL v1.
|
|
# You can use this software according to the terms and conditions of the Mulan PSL v1.
|
|
# You may obtain a copy of Mulan PSL v1 at:
|
|
# http://license.coscl.org.cn/MulanPSL
|
|
# THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR
|
|
# IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR
|
|
# PURPOSE.
|
|
# See the Mulan PSL v1 for more details.
|
|
# Create: 2019-10-29
|
|
|
|
"""
|
|
The sub class of the monitor, used to collect the memory topo.
|
|
"""
|
|
|
|
import sys
|
|
import logging
|
|
import subprocess
|
|
import re
|
|
import json
|
|
import prettytable
|
|
import dict2xml
|
|
|
|
if __name__ == "__main__":
|
|
sys.path.insert(0, "./../../")
|
|
from monitor.common import *
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
class MemTopo(Monitor):
|
|
"""To collect the memory topo"""
|
|
_module = "MEM"
|
|
_purpose = "TOPO"
|
|
_option = "-c memory"
|
|
|
|
def __init__(self, user=None):
|
|
Monitor.__init__(self, user)
|
|
self.__cmd = "lshw"
|
|
self.format.__func__.__doc__ = Monitor.format.__doc__ % ("json, table")
|
|
|
|
def _get(self, para=None):
|
|
with open('/dev/null', 'w') as no_print:
|
|
output = subprocess.check_output("{cmd} {opt}".format(
|
|
cmd=self.__cmd, opt=self._option).split(),
|
|
stderr=no_print)
|
|
return output.decode()
|
|
|
|
def __table_init(self, column):
|
|
self.__columns = column
|
|
data = ['Socket', 'Channel']
|
|
for i in range(self.__columns):
|
|
data.append("Slot {}".format(i))
|
|
return prettytable.PrettyTable(data)
|
|
|
|
def _table_get_locator(self, bank):
|
|
pattern = re.compile(
|
|
"DIMM.*?(\d)(\d)(\d)\s.*",
|
|
re.ASCII | re.MULTILINE)
|
|
scd = pattern.findall(bank)
|
|
if len(scd) == 0:
|
|
err = LookupError("Fail to find data")
|
|
logger.error(
|
|
"{}.{}: {}".format(
|
|
self.__class__.__name__,
|
|
sys._getframe().f_code.co_name,
|
|
str(err)))
|
|
raise err
|
|
|
|
ret = []
|
|
for i in scd[0]:
|
|
ret.append(int(i))
|
|
return ret
|
|
|
|
def _table_get_freq(self, desc):
|
|
pattern = re.compile(
|
|
".*?(\d+)\s([KMGT]?)Hz.*?",
|
|
re.ASCII | re.MULTILINE)
|
|
freq = pattern.findall(desc)
|
|
if len(freq) == 0:
|
|
err = LookupError("Fail to find data")
|
|
logger.error(
|
|
"{}.{}: {}".format(
|
|
self.__class__.__name__,
|
|
sys._getframe().f_code.co_name,
|
|
str(err)))
|
|
raise err
|
|
|
|
if freq[-1][1] == "K":
|
|
ret = int(freq[-1][0]) * 1000
|
|
elif freq[-1][1] == "M":
|
|
ret = int(freq[-1][0]) * 1000000
|
|
elif freq[-1][1] == "G":
|
|
ret = int(freq[-1][0]) * 1000000000
|
|
elif freq[-1][1] == "T":
|
|
ret = int(freq[-1][0]) * 1000000000000
|
|
else:
|
|
ret = int(freq[-1][0])
|
|
return ret
|
|
|
|
def __table_add_banks(self, table, datas, i, j):
|
|
data = [i, j]
|
|
for k in range(self.__columns):
|
|
data.append(datas[i][j][k])
|
|
table.add_row(data)
|
|
|
|
def format(self, info, fmt):
|
|
if (fmt == "json") or (fmt == "table") or (fmt == "xml"):
|
|
o_json = subprocess.check_output("{cmd} -json".format(
|
|
cmd=self.__cmd).split(), stderr=subprocess.DEVNULL)
|
|
info = o_json.decode()
|
|
all = json.loads(info)
|
|
|
|
dict_datas = get_class_type(all, "memory", "System Memory")
|
|
if (fmt == "json"):
|
|
return json.dumps(dict_datas, indent=2)
|
|
elif (fmt == "xml"):
|
|
return dict2xml.dict2xml(dict_datas, "topology")
|
|
|
|
dimms = [[["NO DIMM" for i in range(8)]
|
|
for i in range(8)] for i in range(8)]
|
|
max_socket = 0
|
|
max_slot = 0
|
|
max_channel = 0
|
|
datas = dict_datas["memorys"]
|
|
for data in datas:
|
|
for bank in data["children"]:
|
|
locator = self._table_get_locator(bank["slot"])
|
|
if locator[0] > max_socket:
|
|
max_socket = locator[0]
|
|
if locator[1] > max_channel:
|
|
max_channel = locator[1]
|
|
if locator[2] > max_slot:
|
|
max_slot = locator[2]
|
|
if "size" in bank.keys():
|
|
dimms[locator[0]][locator[1]][locator[2]] = "{} {} {} {}".format(
|
|
bank["vendor"], bank["description"], bank["size"], bank["units"])
|
|
|
|
tables = ""
|
|
for socket in range(max_socket + 1):
|
|
table = self.__table_init(max_slot + 1)
|
|
for channel in range(max_channel + 1):
|
|
self.__table_add_banks(table, dimms, socket, channel)
|
|
tables += table.get_string(hrules=prettytable.ALL) + "\n\n"
|
|
return tables
|
|
else:
|
|
return Monitor.format(self, info, fmt)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
if len(sys.argv) != 3:
|
|
print('usage: ' + sys.argv[0] + ' fmt path')
|
|
sys.exit(-1)
|
|
ct = MemTopo("UT")
|
|
print(ct.report(sys.argv[1], sys.argv[2]))
|