forked from IntelPython/scikit-learn_bench
-
Notifications
You must be signed in to change notification settings - Fork 0
/
utils.py
executable file
·185 lines (160 loc) · 6.84 KB
/
utils.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
# ===============================================================================
# Copyright 2020-2021 Intel Corporation
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
# ===============================================================================
import json
import os
import pathlib
import platform
import subprocess
import sys
from typing import Any, Dict, List, Tuple, Union, cast
from datasets.load_datasets import try_load_dataset
def filter_stderr(text: str) -> str:
# delete 'Intel(R) Extension for Scikit-learn usage in sklearn' messages
fake_error_message = ('Intel(R) Extension for Scikit-learn* enabled ' +
'(https://github.com/intel/scikit-learn-intelex)')
return ''.join(text.split(fake_error_message))
def filter_stdout(text: str) -> Tuple[str, str]:
verbosity_letters = 'EWIDT'
filtered, extra = '', ''
for line in text.split('\n'):
if line == '':
continue
to_remove = False
for letter in verbosity_letters:
if line.startswith(f'[{letter}]'):
to_remove = True
break
if to_remove:
extra += line + '\n'
else:
filtered += line + '\n'
return filtered, extra
def find_the_dataset(name: str, fullpath: str) -> bool:
return os.path.isfile(fullpath) or try_load_dataset(
dataset_name=name, output_directory=pathlib.Path(fullpath).parent)
def read_output_from_command(command: str,
env: Dict[str, str] = os.environ.copy()) -> Tuple[str, str]:
if "PYTHONPATH" in env:
env["PYTHONPATH"] += ":" + os.path.dirname(os.path.abspath(__file__))
else:
env["PYTHONPATH"] = os.path.dirname(os.path.abspath(__file__))
res = subprocess.run(command.split(' '), stdout=subprocess.PIPE,
stderr=subprocess.PIPE, encoding='utf-8', env=env)
return res.stdout[:-1], res.stderr[:-1]
def parse_lscpu_lscl_info(command_output: str) -> Dict[str, str]:
res: Dict[str, str] = {}
for elem in command_output.strip().split('\n'):
splt = elem.split(':')
res[splt[0]] = splt[1]
return res
def get_hw_parameters() -> Dict[str, Union[Dict[str, Any], float]]:
if 'Linux' not in platform.platform():
return {}
hw_params: Dict[str, Union[Dict[str, str], float]] = {'CPU': {}}
# get CPU information
lscpu_info, _ = read_output_from_command('lscpu')
lscpu_info = ' '.join(lscpu_info.split())
for line in lscpu_info.split('\n'):
k, v = line.split(": ")[:2]
if k == 'CPU MHz':
continue
cast(Dict[str, str], hw_params['CPU'])[k] = v
# get RAM size
mem_info, _ = read_output_from_command('free -b')
mem_info = mem_info.split('\n')[1]
mem_info = ' '.join(mem_info.split())
hw_params['RAM size[GB]'] = int(mem_info.split(' ')[1]) / 2 ** 30
# get Intel GPU information
try:
lsgpu_info, _ = read_output_from_command(
'lscl --device-type=gpu --platform-vendor=Intel')
device_num = 0
start_idx = lsgpu_info.find('Device ')
while start_idx >= 0:
start_idx = lsgpu_info.find(':', start_idx) + 1
end_idx = lsgpu_info.find('Device ', start_idx)
hw_params[f'GPU Intel #{device_num + 1}'] = parse_lscpu_lscl_info(
lsgpu_info[start_idx: end_idx])
device_num += 1
start_idx = end_idx
except (FileNotFoundError, json.JSONDecodeError):
pass
# get Nvidia GPU information
try:
gpu_info, _ = read_output_from_command(
'nvidia-smi --query-gpu=name,memory.total,driver_version,pstate '
'--format=csv,noheader')
gpu_info_arr = gpu_info.split(', ')
if len(gpu_info_arr) == 0:
return hw_params
hw_params['GPU Nvidia'] = {
'Name': gpu_info_arr[0],
'Memory size': gpu_info_arr[1],
'Performance mode': gpu_info_arr[3]
}
except (FileNotFoundError, json.JSONDecodeError, IndexError):
pass
return hw_params
def get_sw_parameters() -> Dict[str, Dict[str, Any]]:
sw_params = {}
try:
gpu_info, _ = read_output_from_command(
'nvidia-smi --query-gpu=name,memory.total,driver_version,pstate '
'--format=csv,noheader')
info_arr = gpu_info.split(', ')
sw_params['GPU_driver'] = {'version': info_arr[2]}
# alert if GPU is already running any processes
gpu_processes, _ = read_output_from_command(
'nvidia-smi --query-compute-apps=name,pid,used_memory '
'--format=csv,noheader')
if gpu_processes != '':
print(f'There are running processes on GPU:\n{gpu_processes}',
file=sys.stderr)
except (FileNotFoundError, json.JSONDecodeError):
pass
# get python packages info from conda
try:
conda_list, _ = read_output_from_command('conda list --json')
needed_columns = ['version', 'build_string', 'channel']
conda_list_json: List[Dict[str, str]] = json.loads(conda_list)
for pkg in conda_list_json:
pkg_info = {}
for col in needed_columns:
if col in pkg:
pkg_info[col] = pkg[col]
sw_params[pkg['name']] = pkg_info
except (FileNotFoundError, json.JSONDecodeError):
pass
return sw_params
def generate_cases(params: Dict[str, Union[List[Any], Any]]) -> List[str]:
'''
Generate cases for benchmarking by iterating the parameter values
'''
commands = ['']
for param, values in params.items():
if isinstance(values, list):
prev_len = len(commands)
commands *= len(values)
dashes = '-' if len(param) == 1 else '--'
for command_num in range(prev_len):
for value_num in range(len(values)):
commands[prev_len * value_num + command_num] += ' ' + \
dashes + param + ' ' + str(values[value_num])
else:
dashes = '-' if len(param) == 1 else '--'
for command_num in range(len(commands)):
commands[command_num] += ' ' + dashes + param + ' ' + str(values)
return commands