-
Notifications
You must be signed in to change notification settings - Fork 41
/
Copy pathpci_info.py
executable file
·340 lines (295 loc) · 14.5 KB
/
pci_info.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
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
#!/usr/bin/env python3
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 2 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
#
# See LICENSE for more details.
#
# Copyright: 2023 IBM
# Author: Narasimhan V <[email protected]>
# Author: Manvanthara Puttashankar <[email protected]>
# Author: Shaik Abdulla <[email protected]>
from pprint import pprint
from lib import pci
from lib import virtual
import argparse
import shutil
import os
import sys
import configparser
from lib.pci import is_sriov
from lib.logger import logger_init
from lib.helper import is_rhel8
BASE_PATH = os.path.dirname(os.path.abspath(__file__))
CONFIG_PATH = "%s/config/wrapper/pci_input.conf" % BASE_PATH
CONFIGFILE = configparser.SafeConfigParser()
CONFIGFILE.optionxform = str
CONFIGFILE.read(CONFIG_PATH)
BASE_INPUTFILE_PATH = "%s/config/inputs" % BASE_PATH
input_path = "io_input.txt"
INPUTFILE = configparser.ConfigParser()
INPUTFILE.optionxform = str
args = None
logger = logger_init(filepath=BASE_PATH).getlogger()
def create_config_inputs(orig_cfg, new_cfg, inputfile, interface, config_type):
"""
1. Creates modified configuration file name according to type of interface from original configuration file
2. Generates string with "input file option" along with input file,
3. Generates input parametes of specific device type interface.
Parameters:
orig_cfg (str): The name of the original configuration file.
new_cfg (str): The name of the new configuration file to be generated with
according to interface type.
inputfile (str): The path to the input file containing configuration data.
interface (list): The details of Interface in list format.
config_type (str): The type of configuration to generate. Ex: PCI, vNIC etc.
Returns:
test_suites :: A list, having configuration[cfg] files of different set.
input_file_string :: A string, with extension of "--input-file" option.
input_params:: A list, of different input parameters of specific interface.
"""
test_suites = []
input_file_string = ""
input_params = []
additional_params = args.add_params.split()
if len(additional_params) != 0:
additional_params = args.add_params.split(",")
# Exclude input parameters when vNIC has been created manually, as these parameters are not needed.
# exclude_inputs_params = ["manageSystem","sriov", "hmc_pwd", "hmc_username", "vios_ip","vios_username",
# "vios_pwd","slot_num", "vios_names","sriov_adapters", "sriov_ports",
# "priority", "auto_failover", "bandwidth"]
# Exclude test-cases from test bucket when vNIC created manually.
exclude_test_cases = ["NetworkVirtualization.test_add", "NetworkVirtualization.test_backingdevadd",
"NetworkVirtualization.test_backingdevremove", "NetworkVirtualization.test_remove"]
# when vNIC created manually, Read the orignal configuration file and write only required test-cases in new configuration file.
if config_type == 'vnic':
test_cases = []
with open("config/tests/host/%s.cfg" % orig_cfg, 'r') as file:
lines = file.readlines()
for testcases in lines:
if not any(exclude_test_case in testcases for exclude_test_case in exclude_test_cases):
test_cases.append(testcases)
with open("config/tests/host/%s.cfg" % new_cfg, 'w+') as output_file:
output_file.write("".join(test_cases))
else:
shutil.copy("config/tests/host/%s.cfg" % orig_cfg, "config/tests/host/%s.cfg" % new_cfg)
test_suites.append("host_%s" % new_cfg)
# adding info to input file
if not CONFIGFILE.has_section(orig_cfg) and not additional_params:
return
input_params = CONFIGFILE.items(orig_cfg)
if not input_params:
return
INPUTFILE.add_section(new_cfg)
# read the input file content and store in dict
inputfile_dict = {}
with open(inputfile, 'r') as file:
for line in file:
# Check if the line starts with '#' or '[' and skip it
if line.startswith('#') or line.startswith('['):
continue
# Split each line by '=' to separate key and value
parts = line.strip().split('=')
# Ensure there are exactly two parts (key and value)
if len(parts) == 2:
inputkey, inputvalue = parts[0].strip(), parts[1].strip()
inputfile_dict[inputkey] = inputvalue
# input params
for param in input_params:
try:
key = param[0]
if ':' not in param[1]:
value = interface[param[1]]
else:
index = param[1].split(':')[0]
index_exact = param[1].split(':')[1]
if index_exact == 'all':
# adding only first two available vNIC interfaces when
# multiple vNIC interfaces are available in system.
if config_type in ('vnic', 'veth', 'hnv'):
value = " ".join(str(item) for item in interface[index][:2])
else:
value = " ".join(interface[index])
else:
value = interface[index][int(index_exact)]
if len(interface[index]) > 1:
del interface[index][int(index_exact)]
# remove the duplicate inputfile enteries
if key in inputfile_dict:
del inputfile_dict[key]
INPUTFILE.set(new_cfg, key, "\"%s\"" % value)
except:
pass
# exclude input parameters when vNIC has been created manually.
# if config_type == 'vnic':
# for key in list(inputfile_dict.keys()):
# if any(key.startswith(exclude) for exclude in exclude_inputs_params):
# del inputfile_dict[key]
# additional params
for param in additional_params:
key = param.split('=')[0]
# handling additional params per pci
if '::' in key:
pci_root = key.split('::')[0].split('.')[0]
if pci_root != pci['pci_root']:
continue
key = key.split('::')[1]
# check if the newly added additional param is same
# as inputfile assign the values directly
if key in inputfile_dict:
inputfile_dict[key] = param.split('=')[1]
else:
# if it is completly new then directly write to new input file
value = param.split('=')[1]
INPUTFILE.set(new_cfg, key, "\"%s\"" % value)
# append the remaining input file entries to the new input file
for inputkey, inputvalue in inputfile_dict.items():
INPUTFILE.set(new_cfg, inputkey, "%s" % inputvalue)
return test_suites, input_file_string, input_params
def create_config_file(interface_details, config_type):
"""
Creates avocado test suite / config file, and input file needed for yaml files in that config files.
Parameters:
interface_details(list): The detailed differnet Interface parameters in list format.
config_type (str): The type of configuration to generate. Ex: vNIC, HNV, vETH etc.
"""
for virtual in interface_details:
cfg_name = virtual['adapter_type']
orig_cfg = "io_%s_fvt" % virtual['adapter_type']
new_cfg = "io_%s_stress_fvt" % virtual['adapter_type']
inputfile = "%s/io_%s_input.txt" % (BASE_INPUTFILE_PATH, virtual['adapter_type'])
if not os.path.exists("config/tests/host/%s.cfg" % orig_cfg):
logger.debug("ignoring hnv address as there is no cfg for %s", virtual['adapter_type'])
continue
return create_config_inputs(orig_cfg, new_cfg, inputfile, virtual, config_type=config_type)
def create_config(interface_details, config_type):
"""
Creates avocado test suite / config file, and input file needed for yaml files in that config files.
"""
if config_type == 'pci':
for pci in interface_details:
if pci['is_root_disk']:
logger.debug(
"ignoring pci address %s as it contains root disk", pci['pci_root'])
continue
# copy template cfg files and create new ones
cfg_name = "_".join(pci['pci_root'].split(':'))
if pci['adapter_type'] == 'nvmf' and is_rhel8():
orig_cfg = "io_%s_rhel8_fvt" % pci['adapter_type']
new_cfg = "io_%s_rhel8_%s_fvt" % (pci['adapter_type'], cfg_name)
inputfile = "%s/io_%s_rhel8_input.txt" % (
BASE_INPUTFILE_PATH, pci['adapter_type'])
elif pci['adapter_type'] == 'network' and is_sriov(pci[
'pci_root']):
orig_cfg = "io_nic_sriov_fvt"
new_cfg = "io_nic_sriov_%s_fvt" % cfg_name
inputfile = "%s/io_nic_sriov_input.txt" % BASE_INPUTFILE_PATH
else:
orig_cfg = "io_%s_fvt" % pci['adapter_type']
new_cfg = "io_%s_%s_fvt" % (pci['adapter_type'], cfg_name)
inputfile = "%s/io_%s_input.txt" % (
BASE_INPUTFILE_PATH, pci['adapter_type'])
if not os.path.exists("config/tests/host/%s.cfg" % orig_cfg):
logger.debug("ignoring pci address %s as there is no cfg for %s",
pci['pci_root'], pci['adapter_type'])
continue
test_suites, input_file_string, input_params = create_config_inputs(orig_cfg, new_cfg, inputfile, pci, config_type='pci')
if config_type in ('vnic', 'veth', 'hnv'):
test_suites, input_file_string, input_params = create_config_file(interface_details, config_type)
test_suites = ",".join(test_suites)
# write to input file
if input_params:
with open(input_path, 'w+') as input:
INPUTFILE.write(input)
input_file_string = "--input-file %s" % input_path
# generate avocado-setup command line
if test_suites:
cmd = "python avocado-setup.py --run-suite %s %s" % (test_suites, input_file_string)
return cmd
return ""
if __name__ == '__main__':
parser = argparse.ArgumentParser()
parser.add_argument('--pci-address', dest='pci_addr',
action='store', default='',
help='pci address, comma separated')
parser.add_argument('--vnic', dest='vnic_int', action='store', nargs='?', const='vnic_default', default=None,
help='vNIC interface name')
parser.add_argument('--veth', dest='veth_int', action='store', nargs='?', const='veth_default', default=None,
help='vETH interface name')
parser.add_argument('--hnv', dest='hnv_int', action='store', nargs='?', const='hnv_default', default=None,
help='HNV interface name')
parser.add_argument('--pci-address-blocklist', dest='pci_addr_blocklist',
action='store', default='',
help='pci address which need not be considered, comma separated')
parser.add_argument('--type', dest='pci_type',
action='store', default='All',
help='type of adapters, comma separated')
parser.add_argument('--type-blocklist', dest='type_blocklist',
action='store', default='',
help='type of adapters to blocklist, comma separated')
parser.add_argument('--show-info', dest='show_info',
action='store_true', default=False,
help='Show the pci details')
parser.add_argument('--create-config', dest='create_cfg',
action='store_true', default=False,
help='Create test config and input files')
parser.add_argument('--run-test', dest='run_test',
action='store_true', default=False,
help='Run the test suite using created test config and input files')
parser.add_argument('--additional-params', dest='add_params',
action='store', default='',
help='Additional parameters(key=value) to the input file, space separated')
args = parser.parse_args()
#if no interfaces name is provided for vNIC ,vETH or HNV, get the first aavailable interface.
if args.vnic_int == 'vnic_default':
args.vnic_int = virtual.get_vnic_interface_names()[0]
if args.veth_int == 'veth_default':
args.veth_int = virtual.get_veth_interface_names()[0]
if args.hnv_int == 'hnv_default':
args.hnv_int = virtual.get_hnv_interface_names()[0]
try:
if args.vnic_int:
vnic_details = virtual.virtual_info(args.vnic_int)
elif args.veth_int:
veth_details = virtual.virtual_info(args.veth_int)
elif args.hnv_int:
hnv_details = virtual.virtual_info(args.hnv_int)
elif args.pci_addr:
pci_details = pci.pci_info(args.pci_addr, pci_type=args.pci_type, pci_blocklist=args.pci_addr_blocklist, type_blocklist=args.type_blocklist)
else:
pci_details = pci.all_pci_info(pci_type=args.pci_type, pci_blocklist=args.pci_addr_blocklist, type_blocklist=args.type_blocklist)
except Exception as e:
if args.vnic_int:
logger.info("vNIC interface not found")
else:
logger.info("No PCI Found")
sys.exit(0)
if args.show_info:
if args.pci_addr:
pprint(pci_details)
elif args.vnic_int:
pprint(vnic_details)
elif args.veth_int:
pprint(veth_details)
elif args.hnv_int:
pprint(hnv_details)
if args.create_cfg:
if args.vnic_int:
cmd = create_config(interface_details=vnic_details, config_type='vnic')
logger.info(cmd)
elif args.veth_int:
cmd = create_config(interface_details=veth_details, config_type='veth')
logger.info(cmd)
elif args.hnv_int:
cmd = create_config(interface_details=hnv_details, config_type='hnv')
logger.info(cmd)
else:
cmd = create_config(interface_details=pci_details, config_type='pci')
logger.info(cmd)
if args.run_test:
os.system(cmd)