VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/tests/storage/tdStorageBenchmark1.py@ 65145

Last change on this file since 65145 was 64847, checked in by vboxsync, 8 years ago

ValdiationKit/storaage: Increase the timeout for the benchmarks to 2 hourse because running the complete benchmark with QCOW exceeds the timeout. The 4K block size hurts performance too much

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 47.2 KB
Line 
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3# $Id: tdStorageBenchmark1.py 64847 2016-12-13 12:42:29Z vboxsync $
4
5"""
6VirtualBox Validation Kit - Storage benchmark.
7"""
8
9__copyright__ = \
10"""
11Copyright (C) 2012-2016 Oracle Corporation
12
13This file is part of VirtualBox Open Source Edition (OSE), as
14available from http://www.virtualbox.org. This file is free software;
15you can redistribute it and/or modify it under the terms of the GNU
16General Public License (GPL) as published by the Free Software
17Foundation, in version 2 as it comes in the "COPYING" file of the
18VirtualBox OSE distribution. VirtualBox OSE is distributed in the
19hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
20
21The contents of this file may alternatively be used under the terms
22of the Common Development and Distribution License Version 1.0
23(CDDL) only, as it comes in the "COPYING.CDDL" file of the
24VirtualBox OSE distribution, in which case the provisions of the
25CDDL are applicable instead of those of the GPL.
26
27You may elect to license modified versions of this file under the
28terms and conditions of either the GPL or the CDDL or both.
29"""
30__version__ = "$Revision: 64847 $"
31
32
33# Standard Python imports.
34import os;
35import socket;
36import sys;
37import StringIO;
38
39# Only the main script needs to modify the path.
40try: __file__
41except: __file__ = sys.argv[0];
42g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))));
43sys.path.append(g_ksValidationKitDir);
44
45# Validation Kit imports.
46from common import constants;
47from common import utils;
48from testdriver import reporter;
49from testdriver import base;
50from testdriver import vbox;
51from testdriver import vboxcon;
52
53import remoteexecutor;
54import storagecfg;
55
56def _ControllerTypeToName(eControllerType):
57 """ Translate a controller type to a name. """
58 if eControllerType == vboxcon.StorageControllerType_PIIX3 or eControllerType == vboxcon.StorageControllerType_PIIX4:
59 sType = "IDE Controller";
60 elif eControllerType == vboxcon.StorageControllerType_IntelAhci:
61 sType = "SATA Controller";
62 elif eControllerType == vboxcon.StorageControllerType_LsiLogicSas:
63 sType = "SAS Controller";
64 elif eControllerType == vboxcon.StorageControllerType_LsiLogic or eControllerType == vboxcon.StorageControllerType_BusLogic:
65 sType = "SCSI Controller";
66 elif eControllerType == vboxcon.StorageControllerType_NVMe:
67 sType = "NVMe Controller";
68 else:
69 sType = "Storage Controller";
70 return sType;
71
72class FioTest(object):
73 """
74 Flexible I/O tester testcase.
75 """
76
77 kdHostIoEngine = {
78 'solaris': ('solarisaio', False),
79 'linux': ('libaio', True)
80 };
81
82 def __init__(self, oExecutor, dCfg = None):
83 self.oExecutor = oExecutor;
84 self.sCfgFileId = None;
85 self.dCfg = dCfg;
86
87 def prepare(self, cMsTimeout = 30000):
88 """ Prepares the testcase """
89
90 sTargetOs = self.dCfg.get('TargetOs', 'linux');
91 sIoEngine, fDirectIo = self.kdHostIoEngine.get(sTargetOs);
92 if sIoEngine is None:
93 return False;
94
95 cfgBuf = StringIO.StringIO();
96 cfgBuf.write('[global]\n');
97 cfgBuf.write('bs=' + self.dCfg.get('RecordSize', '4k') + '\n');
98 cfgBuf.write('ioengine=' + sIoEngine + '\n');
99 cfgBuf.write('iodepth=' + self.dCfg.get('QueueDepth', '32') + '\n');
100 cfgBuf.write('size=' + self.dCfg.get('TestsetSize', '2g') + '\n');
101 if fDirectIo:
102 cfgBuf.write('direct=1\n');
103 else:
104 cfgBuf.write('direct=0\n');
105 cfgBuf.write('directory=' + self.dCfg.get('FilePath', '/mnt') + '\n');
106
107 cfgBuf.write('[seq-write]\n');
108 cfgBuf.write('rw=write\n');
109 cfgBuf.write('stonewall\n');
110
111 cfgBuf.write('[rand-write]\n');
112 cfgBuf.write('rw=randwrite\n');
113 cfgBuf.write('stonewall\n');
114
115 cfgBuf.write('[seq-read]\n');
116 cfgBuf.write('rw=read\n');
117 cfgBuf.write('stonewall\n');
118
119 cfgBuf.write('[rand-read]\n');
120 cfgBuf.write('rw=randread\n');
121 cfgBuf.write('stonewall\n');
122
123 self.sCfgFileId = self.oExecutor.copyString(cfgBuf.getvalue(), 'aio-test', cMsTimeout);
124 return self.sCfgFileId is not None;
125
126 def run(self, cMsTimeout = 30000):
127 """ Runs the testcase """
128 _ = cMsTimeout
129 fRc, sOutput = self.oExecutor.execBinary('fio', (self.sCfgFileId,), cMsTimeout = cMsTimeout);
130 # @todo: Parse output.
131 _ = sOutput;
132 return fRc;
133
134 def cleanup(self):
135 """ Cleans up any leftovers from the testcase. """
136
137 def reportResult(self):
138 """
139 Reports the test results to the test manager.
140 """
141 return True;
142
143class IozoneTest(object):
144 """
145 I/O zone testcase.
146 """
147 def __init__(self, oExecutor, dCfg = None):
148 self.oExecutor = oExecutor;
149 self.sResult = None;
150 self.lstTests = [ ('initial writers', 'FirstWrite'),
151 ('rewriters', 'Rewrite'),
152 ('re-readers', 'ReRead'),
153 ('stride readers', 'StrideRead'),
154 ('reverse readers', 'ReverseRead'),
155 ('random readers', 'RandomRead'),
156 ('mixed workload', 'MixedWorkload'),
157 ('random writers', 'RandomWrite'),
158 ('pwrite writers', 'PWrite'),
159 ('pread readers', 'PRead'),
160 ('fwriters', 'FWrite'),
161 ('freaders', 'FRead'),
162 ('readers', 'FirstRead')];
163 self.sRecordSize = dCfg.get('RecordSize', '4k');
164 self.sTestsetSize = dCfg.get('TestsetSize', '2g');
165 self.sQueueDepth = dCfg.get('QueueDepth', '32');
166 self.sFilePath = dCfg.get('FilePath', '/mnt/iozone');
167 self.fDirectIo = True;
168
169 sTargetOs = dCfg.get('TargetOs');
170 if sTargetOs == 'solaris':
171 self.fDirectIo = False;
172
173 def prepare(self, cMsTimeout = 30000):
174 """ Prepares the testcase """
175 _ = cMsTimeout;
176 return True; # Nothing to do.
177
178 def run(self, cMsTimeout = 30000):
179 """ Runs the testcase """
180 tupArgs = ('-r', self.sRecordSize, '-s', self.sTestsetSize, \
181 '-t', '1', '-T', '-F', self.sFilePath + '/iozone.tmp');
182 if self.fDirectIo:
183 tupArgs += ('-I',);
184 fRc, sOutput = self.oExecutor.execBinary('iozone', tupArgs, cMsTimeout = cMsTimeout);
185 if fRc:
186 self.sResult = sOutput;
187
188 _ = cMsTimeout;
189 return fRc;
190
191 def cleanup(self):
192 """ Cleans up any leftovers from the testcase. """
193 return True;
194
195 def reportResult(self):
196 """
197 Reports the test results to the test manager.
198 """
199
200 fRc = True;
201 if self.sResult is not None:
202 try:
203 asLines = self.sResult.splitlines();
204 for sLine in asLines:
205 sLine = sLine.strip();
206 if sLine.startswith('Children') is True:
207 # Extract the value
208 idxValue = sLine.rfind('=');
209 if idxValue is -1:
210 raise Exception('IozoneTest: Invalid state');
211
212 idxValue += 1;
213 while sLine[idxValue] == ' ':
214 idxValue += 1;
215
216 # Get the reported value, cut off after the decimal point
217 # it is not supported by the testmanager yet and is not really
218 # relevant anyway.
219 idxValueEnd = idxValue;
220 while sLine[idxValueEnd].isdigit():
221 idxValueEnd += 1;
222
223 for sNeedle, sTestVal in self.lstTests:
224 if sLine.rfind(sNeedle) is not -1:
225 reporter.testValue(sTestVal, sLine[idxValue:idxValueEnd],
226 constants.valueunit.g_asNames[constants.valueunit.KILOBYTES_PER_SEC]);
227 break;
228 except:
229 fRc = False;
230 else:
231 fRc = False;
232
233 return fRc;
234
235
236class StorTestCfgMgr(object):
237 """
238 Manages the different testcases.
239 """
240
241 def __init__(self, aasTestLvls, aasTestsBlacklist, fnIsCfgSupported = None):
242 self.aasTestsBlacklist = aasTestsBlacklist;
243 self.at3TestLvls = [];
244 self.iTestLvl = 0;
245 self.fnIsCfgSupported = fnIsCfgSupported;
246 for asTestLvl in aasTestLvls:
247 if isinstance(asTestLvl, tuple):
248 asTestLvl, fnTestFmt = asTestLvl;
249 self.at3TestLvls.append((0, fnTestFmt, asTestLvl));
250 else:
251 self.at3TestLvls.append((0, None, asTestLvl));
252
253 self.at3TestLvls.reverse();
254
255 # Get the first non blacklisted test.
256 asTestCfg = self.getCurrentTestCfg();
257 while len(asTestCfg) > 0 and self.isTestCfgBlacklisted(asTestCfg):
258 asTestCfg = self.advanceTestCfg();
259
260 iLvl = 0;
261 for sCfg in asTestCfg:
262 reporter.testStart('%s' % (self.getTestIdString(sCfg, iLvl)));
263 iLvl += 1;
264
265 def __del__(self):
266 # Make sure the tests are marked as done.
267 while self.iTestLvl < len(self.at3TestLvls):
268 reporter.testDone();
269 self.iTestLvl += 1;
270
271 def getTestIdString(self, oCfg, iLvl):
272 """
273 Returns a potentially formatted string for the test name.
274 """
275
276 # The order of the test levels is reversed so get the level starting
277 # from the end.
278 _, fnTestFmt, _ = self.at3TestLvls[len(self.at3TestLvls) - 1 - iLvl];
279 if fnTestFmt is not None:
280 return fnTestFmt(oCfg);
281 else:
282 return oCfg;
283
284 def isTestCfgBlacklisted(self, asTestCfg):
285 """
286 Returns whether the given test config is black listed.
287 """
288 fBlacklisted = False;
289
290 for asTestBlacklist in self.aasTestsBlacklist:
291 iLvl = 0;
292 fBlacklisted = True;
293 while iLvl < len(asTestBlacklist) and iLvl < len(asTestCfg):
294 if asTestBlacklist[iLvl] != asTestCfg[iLvl] and asTestBlacklist[iLvl] != '*':
295 fBlacklisted = False;
296 break;
297
298 iLvl += 1;
299
300 if not fBlacklisted and self.fnIsCfgSupported is not None:
301 fBlacklisted = not self.fnIsCfgSupported(asTestCfg);
302
303 return fBlacklisted;
304
305 def advanceTestCfg(self):
306 """
307 Advances to the next test config and returns it as an
308 array of strings or an empty config if there is no test left anymore.
309 """
310 iTestCfg, fnTestFmt, asTestCfg = self.at3TestLvls[self.iTestLvl];
311 iTestCfg += 1;
312 self.at3TestLvls[self.iTestLvl] = (iTestCfg, fnTestFmt, asTestCfg);
313 while iTestCfg == len(asTestCfg) and self.iTestLvl < len(self.at3TestLvls):
314 self.at3TestLvls[self.iTestLvl] = (0, fnTestFmt, asTestCfg);
315 self.iTestLvl += 1;
316 if self.iTestLvl < len(self.at3TestLvls):
317 iTestCfg, fnTestFmt, asTestCfg = self.at3TestLvls[self.iTestLvl];
318 iTestCfg += 1;
319 self.at3TestLvls[self.iTestLvl] = (iTestCfg, fnTestFmt, asTestCfg);
320 if iTestCfg < len(asTestCfg):
321 self.iTestLvl = 0;
322 break;
323 else:
324 break; # We reached the end of our tests.
325
326 return self.getCurrentTestCfg();
327
328 def getCurrentTestCfg(self):
329 """
330 Returns the current not black listed test config as an array of strings.
331 """
332 asTestCfg = [];
333
334 if self.iTestLvl < len(self.at3TestLvls):
335 for t3TestLvl in self.at3TestLvls:
336 iTestCfg, _, asTestLvl = t3TestLvl;
337 asTestCfg.append(asTestLvl[iTestCfg]);
338
339 asTestCfg.reverse()
340
341 return asTestCfg;
342
343 def getNextTestCfg(self, fSkippedLast = False):
344 """
345 Returns the next not blacklisted test config or an empty list if
346 there is no test left.
347 """
348 asTestCfgCur = self.getCurrentTestCfg();
349
350 asTestCfg = self.advanceTestCfg();
351 while len(asTestCfg) > 0 and self.isTestCfgBlacklisted(asTestCfg):
352 asTestCfg = self.advanceTestCfg();
353
354 # Compare the current and next config and close the approriate test
355 # categories.
356 reporter.testDone(fSkippedLast);
357 if len(asTestCfg) > 0:
358 idxSame = 0;
359 while asTestCfgCur[idxSame] == asTestCfg[idxSame]:
360 idxSame += 1;
361
362 for i in range(idxSame, len(asTestCfg) - 1):
363 reporter.testDone();
364
365 for i in range(idxSame, len(asTestCfg)):
366 reporter.testStart('%s' % (self.getTestIdString(asTestCfg[i], i)));
367
368 else:
369 # No more tests, mark all tests as done
370 for i in range(0, len(asTestCfgCur) - 1):
371 reporter.testDone();
372
373 return asTestCfg;
374
375class tdStorageBenchmark(vbox.TestDriver): # pylint: disable=R0902
376 """
377 Storage benchmark.
378 """
379
380 # Global storage configs for the testbox
381 kdStorageCfgs = {
382 'testboxstor1.de.oracle.com': r'c[3-9]t\dd0\Z',
383 'adaris': [ '/dev/sda' ]
384 };
385
386 # Available test sets.
387 kdTestSets = {
388 # Mostly for developing and debugging the testcase.
389 'Fast': {
390 'RecordSize': '64k',
391 'TestsetSize': '100m',
392 'QueueDepth': '32',
393 'DiskSizeGb': 2
394 },
395 # For quick functionality tests where benchmark results are not required.
396 'Functionality': {
397 'RecordSize': '64k',
398 'TestsetSize': '2g',
399 'QueueDepth': '32',
400 'DiskSizeGb': 10
401 },
402 # For benchmarking the I/O stack.
403 'Benchmark': {
404 'RecordSize': '64k',
405 'TestsetSize': '20g',
406 'QueueDepth': '32',
407 'DiskSizeGb': 100
408 },
409 # For stress testing which takes a lot of time.
410 'Stress': {
411 'RecordSize': '64k',
412 'TestsetSize': '2t',
413 'QueueDepth': '32',
414 'DiskSizeGb': 10000
415 },
416 };
417
418 # Dictionary mapping the virtualization mode mnemonics to a little less cryptic
419 # strings used in test descriptions.
420 kdVirtModeDescs = {
421 'raw' : 'Raw-mode',
422 'hwvirt' : 'HwVirt',
423 'hwvirt-np' : 'NestedPaging'
424 };
425
426 kdHostIoCacheDescs = {
427 'default' : 'HostCacheDef',
428 'hostiocache' : 'HostCacheOn',
429 'no-hostiocache' : 'HostCacheOff'
430 };
431
432 # Array indexes for the test configs.
433 kiVmName = 0;
434 kiStorageCtrl = 1;
435 kiHostIoCache = 2;
436 kiDiskFmt = 3;
437 kiDiskVar = 4;
438 kiCpuCount = 5;
439 kiVirtMode = 6;
440 kiIoTest = 7;
441 kiTestSet = 8;
442
443 def __init__(self):
444 vbox.TestDriver.__init__(self);
445 self.asRsrcs = None;
446 self.asTestVMsDef = ['tst-storage', 'tst-storage32'];
447 self.asTestVMs = self.asTestVMsDef;
448 self.asSkipVMs = [];
449 self.asVirtModesDef = ['hwvirt', 'hwvirt-np', 'raw',]
450 self.asVirtModes = self.asVirtModesDef;
451 self.acCpusDef = [1, 2];
452 self.acCpus = self.acCpusDef;
453 self.asStorageCtrlsDef = ['AHCI', 'IDE', 'LsiLogicSAS', 'LsiLogic', 'BusLogic', 'NVMe'];
454 self.asStorageCtrls = self.asStorageCtrlsDef;
455 self.asHostIoCacheDef = ['default', 'hostiocache', 'no-hostiocache'];
456 self.asHostIoCache = self.asHostIoCacheDef;
457 self.asDiskFormatsDef = ['VDI', 'VMDK', 'VHD', 'QED', 'Parallels', 'QCOW', 'iSCSI'];
458 self.asDiskFormats = self.asDiskFormatsDef;
459 self.asDiskVariantsDef = ['Dynamic', 'Fixed', 'DynamicSplit2G', 'FixedSplit2G', 'Network'];
460 self.asDiskVariants = self.asDiskVariantsDef;
461 self.asTestsDef = ['iozone', 'fio'];
462 self.asTests = self.asTestsDef;
463 self.asTestSetsDef = ['Fast', 'Functionality', 'Benchmark', 'Stress'];
464 self.asTestSets = self.asTestSetsDef;
465 self.asIscsiTargetsDef = [ ]; # @todo: Configure one target for basic iSCSI testing
466 self.asIscsiTargets = self.asIscsiTargetsDef;
467 self.cDiffLvlsDef = 0;
468 self.cDiffLvls = self.cDiffLvlsDef;
469 self.fTestHost = False;
470 self.fUseScratch = False;
471 self.fRecreateStorCfg = True;
472 self.fReportBenchmarkResults = True;
473 self.oStorCfg = None;
474
475 #
476 # Overridden methods.
477 #
478 def showUsage(self):
479 rc = vbox.TestDriver.showUsage(self);
480 reporter.log('');
481 reporter.log('tdStorageBenchmark1 Options:');
482 reporter.log(' --virt-modes <m1[:m2[:]]');
483 reporter.log(' Default: %s' % (':'.join(self.asVirtModesDef)));
484 reporter.log(' --cpu-counts <c1[:c2[:]]');
485 reporter.log(' Default: %s' % (':'.join(str(c) for c in self.acCpusDef)));
486 reporter.log(' --storage-ctrls <type1[:type2[:...]]>');
487 reporter.log(' Default: %s' % (':'.join(self.asStorageCtrlsDef)));
488 reporter.log(' --host-io-cache <setting1[:setting2[:...]]>');
489 reporter.log(' Default: %s' % (':'.join(self.asHostIoCacheDef)));
490 reporter.log(' --disk-formats <type1[:type2[:...]]>');
491 reporter.log(' Default: %s' % (':'.join(self.asDiskFormatsDef)));
492 reporter.log(' --disk-variants <variant1[:variant2[:...]]>');
493 reporter.log(' Default: %s' % (':'.join(self.asDiskVariantsDef)));
494 reporter.log(' --iscsi-targets <target1[:target2[:...]]>');
495 reporter.log(' Default: %s' % (':'.join(self.asIscsiTargetsDef)));
496 reporter.log(' --tests <test1[:test2[:...]]>');
497 reporter.log(' Default: %s' % (':'.join(self.asTestsDef)));
498 reporter.log(' --test-sets <set1[:set2[:...]]>');
499 reporter.log(' Default: %s' % (':'.join(self.asTestSetsDef)));
500 reporter.log(' --diff-levels <number of diffs>');
501 reporter.log(' Default: %s' % (self.cDiffLvlsDef));
502 reporter.log(' --test-vms <vm1[:vm2[:...]]>');
503 reporter.log(' Test the specified VMs in the given order. Use this to change');
504 reporter.log(' the execution order or limit the choice of VMs');
505 reporter.log(' Default: %s (all)' % (':'.join(self.asTestVMsDef)));
506 reporter.log(' --skip-vms <vm1[:vm2[:...]]>');
507 reporter.log(' Skip the specified VMs when testing.');
508 reporter.log(' --test-host');
509 reporter.log(' Do all configured tests on the host first and report the results');
510 reporter.log(' to get a baseline');
511 reporter.log(' --use-scratch');
512 reporter.log(' Use the scratch directory for testing instead of setting up');
513 reporter.log(' fresh volumes on dedicated disks (for development)');
514 reporter.log(' --always-wipe-storage-cfg');
515 reporter.log(' Recreate the host storage config before each test');
516 reporter.log(' --dont-wipe-storage-cfg');
517 reporter.log(' Don\'t recreate the host storage config before each test');
518 reporter.log(' --report-benchmark-results');
519 reporter.log(' Report all benchmark results');
520 reporter.log(' --dont-report-benchmark-results');
521 reporter.log(' Don\'t report any benchmark results');
522 return rc;
523
524 def parseOption(self, asArgs, iArg): # pylint: disable=R0912,R0915
525 if asArgs[iArg] == '--virt-modes':
526 iArg += 1;
527 if iArg >= len(asArgs): raise base.InvalidOption('The "--virt-modes" takes a colon separated list of modes');
528 self.asVirtModes = asArgs[iArg].split(':');
529 for s in self.asVirtModes:
530 if s not in self.asVirtModesDef:
531 raise base.InvalidOption('The "--virt-modes" value "%s" is not valid; valid values are: %s' \
532 % (s, ' '.join(self.asVirtModesDef)));
533 elif asArgs[iArg] == '--cpu-counts':
534 iArg += 1;
535 if iArg >= len(asArgs): raise base.InvalidOption('The "--cpu-counts" takes a colon separated list of cpu counts');
536 self.acCpus = [];
537 for s in asArgs[iArg].split(':'):
538 try: c = int(s);
539 except: raise base.InvalidOption('The "--cpu-counts" value "%s" is not an integer' % (s,));
540 if c <= 0: raise base.InvalidOption('The "--cpu-counts" value "%s" is zero or negative' % (s,));
541 self.acCpus.append(c);
542 elif asArgs[iArg] == '--storage-ctrls':
543 iArg += 1;
544 if iArg >= len(asArgs):
545 raise base.InvalidOption('The "--storage-ctrls" takes a colon separated list of Storage controller types');
546 self.asStorageCtrls = asArgs[iArg].split(':');
547 elif asArgs[iArg] == '--host-io-cache':
548 iArg += 1;
549 if iArg >= len(asArgs):
550 raise base.InvalidOption('The "--host-io-cache" takes a colon separated list of I/O cache settings');
551 self.asHostIoCache = asArgs[iArg].split(':');
552 elif asArgs[iArg] == '--disk-formats':
553 iArg += 1;
554 if iArg >= len(asArgs): raise base.InvalidOption('The "--disk-formats" takes a colon separated list of disk formats');
555 self.asDiskFormats = asArgs[iArg].split(':');
556 elif asArgs[iArg] == '--disk-variants':
557 iArg += 1;
558 if iArg >= len(asArgs):
559 raise base.InvalidOption('The "--disk-variants" takes a colon separated list of disk variants');
560 self.asDiskVariants = asArgs[iArg].split(':');
561 elif asArgs[iArg] == '--iscsi-targets':
562 iArg += 1;
563 if iArg >= len(asArgs):
564 raise base.InvalidOption('The "--iscsi-targets" takes a colon separated list of iscsi targets');
565 self.asIscsiTargets = asArgs[iArg].split(':');
566 elif asArgs[iArg] == '--tests':
567 iArg += 1;
568 if iArg >= len(asArgs): raise base.InvalidOption('The "--tests" takes a colon separated list of tests to run');
569 self.asTests = asArgs[iArg].split(':');
570 elif asArgs[iArg] == '--test-sets':
571 iArg += 1;
572 if iArg >= len(asArgs): raise base.InvalidOption('The "--test-sets" takes a colon separated list of test sets');
573 self.asTestSets = asArgs[iArg].split(':');
574 elif asArgs[iArg] == '--diff-levels':
575 iArg += 1;
576 if iArg >= len(asArgs): raise base.InvalidOption('The "--diff-levels" takes an integer');
577 try: self.cDiffLvls = int(asArgs[iArg]);
578 except: raise base.InvalidOption('The "--diff-levels" value "%s" is not an integer' % (asArgs[iArg],));
579 elif asArgs[iArg] == '--test-vms':
580 iArg += 1;
581 if iArg >= len(asArgs): raise base.InvalidOption('The "--test-vms" takes colon separated list');
582 self.asTestVMs = asArgs[iArg].split(':');
583 for s in self.asTestVMs:
584 if s not in self.asTestVMsDef:
585 raise base.InvalidOption('The "--test-vms" value "%s" is not valid; valid values are: %s' \
586 % (s, ' '.join(self.asTestVMsDef)));
587 elif asArgs[iArg] == '--skip-vms':
588 iArg += 1;
589 if iArg >= len(asArgs): raise base.InvalidOption('The "--skip-vms" takes colon separated list');
590 self.asSkipVMs = asArgs[iArg].split(':');
591 for s in self.asSkipVMs:
592 if s not in self.asTestVMsDef:
593 reporter.log('warning: The "--test-vms" value "%s" does not specify any of our test VMs.' % (s));
594 elif asArgs[iArg] == '--test-host':
595 self.fTestHost = True;
596 elif asArgs[iArg] == '--use-scratch':
597 self.fUseScratch = True;
598 elif asArgs[iArg] == '--always-wipe-storage-cfg':
599 self.fRecreateStorCfg = True;
600 elif asArgs[iArg] == '--dont-wipe-storage-cfg':
601 self.fRecreateStorCfg = False;
602 elif asArgs[iArg] == '--report-benchmark-results':
603 self.fReportBenchmarkResults = True;
604 elif asArgs[iArg] == '--dont-report-benchmark-results':
605 self.fReportBenchmarkResults = False;
606 else:
607 return vbox.TestDriver.parseOption(self, asArgs, iArg);
608 return iArg + 1;
609
610 def completeOptions(self):
611 # Remove skipped VMs from the test list.
612 for sVM in self.asSkipVMs:
613 try: self.asTestVMs.remove(sVM);
614 except: pass;
615
616 return vbox.TestDriver.completeOptions(self);
617
618 def getResourceSet(self):
619 # Construct the resource list the first time it's queried.
620 if self.asRsrcs is None:
621 self.asRsrcs = [];
622 if 'tst-storage' in self.asTestVMs:
623 self.asRsrcs.append('5.0/storage/tst-storage.vdi');
624 if 'tst-storage32' in self.asTestVMs:
625 self.asRsrcs.append('5.0/storage/tst-storage32.vdi');
626
627 return self.asRsrcs;
628
629 def actionConfig(self):
630
631 # Make sure vboxapi has been imported so we can use the constants.
632 if not self.importVBoxApi():
633 return False;
634
635 #
636 # Configure the VMs we're going to use.
637 #
638
639 # Linux VMs
640 if 'tst-storage' in self.asTestVMs:
641 oVM = self.createTestVM('tst-storage', 1, '5.0/storage/tst-storage.vdi', sKind = 'ArchLinux_64', fIoApic = True, \
642 eNic0AttachType = vboxcon.NetworkAttachmentType_NAT, \
643 eNic0Type = vboxcon.NetworkAdapterType_Am79C973);
644 if oVM is None:
645 return False;
646
647 if 'tst-storage32' in self.asTestVMs:
648 oVM = self.createTestVM('tst-storage32', 1, '5.0/storage/tst-storage32.vdi', sKind = 'ArchLinux', fIoApic = True, \
649 eNic0AttachType = vboxcon.NetworkAttachmentType_NAT, \
650 eNic0Type = vboxcon.NetworkAdapterType_Am79C973);
651 if oVM is None:
652 return False;
653
654 return True;
655
656 def actionExecute(self):
657 """
658 Execute the testcase.
659 """
660 fRc = self.test1();
661 return fRc;
662
663
664 #
665 # Test execution helpers.
666 #
667
668 def prepareStorage(self, oStorCfg):
669 """
670 Prepares the host storage for disk images or direct testing on the host.
671 """
672 # Create a basic pool with the default configuration.
673 sMountPoint = None;
674 fRc, sPoolId = oStorCfg.createStoragePool();
675 if fRc:
676 fRc, sMountPoint = oStorCfg.createVolume(sPoolId);
677 if not fRc:
678 sMountPoint = None;
679 oStorCfg.cleanup();
680
681 return sMountPoint;
682
683 def cleanupStorage(self, oStorCfg):
684 """
685 Cleans up any created storage space for a test.
686 """
687 return oStorCfg.cleanup();
688
689 def getGuestDisk(self, oSession, oTxsSession, eStorageController):
690 """
691 Gets the path of the disk in the guest to use for testing.
692 """
693 lstDisks = None;
694
695 # The naming scheme for NVMe is different and we don't have
696 # to query the guest for unformatted disks here because the disk with the OS
697 # is not attached to a NVMe controller.
698 if eStorageController == vboxcon.StorageControllerType_NVMe:
699 lstDisks = [ '/dev/nvme0n1' ];
700 else:
701 # Find a unformatted disk (no partition).
702 # @todo: This is a hack because LIST and STAT are not yet implemented
703 # in TXS (get to this eventually)
704 lstBlkDev = [ '/dev/sda', '/dev/sdb' ];
705 for sBlkDev in lstBlkDev:
706 fRc = oTxsSession.syncExec('/usr/bin/ls', ('ls', sBlkDev + '1'));
707 if not fRc:
708 lstDisks = [ sBlkDev ];
709 break;
710
711 _ = oSession;
712 return lstDisks;
713
714 def getDiskFormatVariantsForTesting(self, sDiskFmt, asVariants):
715 """
716 Returns a list of disk variants for testing supported by the given
717 disk format and selected for testing.
718 """
719 lstDskFmts = self.oVBoxMgr.getArray(self.oVBox.systemProperties, 'mediumFormats');
720 for oDskFmt in lstDskFmts:
721 if oDskFmt.id == sDiskFmt:
722 lstDskVariants = [];
723 lstCaps = self.oVBoxMgr.getArray(oDskFmt, 'capabilities');
724
725 if vboxcon.MediumFormatCapabilities_CreateDynamic in lstCaps \
726 and 'Dynamic' in asVariants:
727 lstDskVariants.append('Dynamic');
728
729 if vboxcon.MediumFormatCapabilities_CreateFixed in lstCaps \
730 and 'Fixed' in asVariants:
731 lstDskVariants.append('Fixed');
732
733 if vboxcon.MediumFormatCapabilities_CreateSplit2G in lstCaps \
734 and vboxcon.MediumFormatCapabilities_CreateDynamic in lstCaps \
735 and 'DynamicSplit2G' in asVariants:
736 lstDskVariants.append('DynamicSplit2G');
737
738 if vboxcon.MediumFormatCapabilities_CreateSplit2G in lstCaps \
739 and vboxcon.MediumFormatCapabilities_CreateFixed in lstCaps \
740 and 'FixedSplit2G' in asVariants:
741 lstDskVariants.append('FixedSplit2G');
742
743 if vboxcon.MediumFormatCapabilities_TcpNetworking in lstCaps \
744 and 'Network' in asVariants:
745 lstDskVariants.append('Network'); # Solely for iSCSI to get a non empty list
746
747 return lstDskVariants;
748
749 return [];
750
751 def convDiskToMediumVariant(self, sDiskVariant):
752 """
753 Returns a tuple of medium variant flags matching the given disk variant.
754 """
755 tMediumVariant = None;
756 if sDiskVariant == 'Dynamic':
757 tMediumVariant = (vboxcon.MediumVariant_Standard, );
758 elif sDiskVariant == 'Fixed':
759 tMediumVariant = (vboxcon.MediumVariant_Fixed, );
760 elif sDiskVariant == 'DynamicSplit2G':
761 tMediumVariant = (vboxcon.MediumVariant_Standard, vboxcon.MediumVariant_VmdkSplit2G);
762 elif sDiskVariant == 'FixedSplit2G':
763 tMediumVariant = (vboxcon.MediumVariant_Fixed, vboxcon.MediumVariant_VmdkSplit2G);
764
765 return tMediumVariant;
766
767 def getStorageCtrlFromName(self, sStorageCtrl):
768 """
769 Resolves the storage controller string to the matching constant.
770 """
771 eStorageCtrl = None;
772
773 if sStorageCtrl == 'AHCI':
774 eStorageCtrl = vboxcon.StorageControllerType_IntelAhci;
775 elif sStorageCtrl == 'IDE':
776 eStorageCtrl = vboxcon.StorageControllerType_PIIX4;
777 elif sStorageCtrl == 'LsiLogicSAS':
778 eStorageCtrl = vboxcon.StorageControllerType_LsiLogicSas;
779 elif sStorageCtrl == 'LsiLogic':
780 eStorageCtrl = vboxcon.StorageControllerType_LsiLogic;
781 elif sStorageCtrl == 'BusLogic':
782 eStorageCtrl = vboxcon.StorageControllerType_BusLogic;
783 elif sStorageCtrl == 'NVMe':
784 eStorageCtrl = vboxcon.StorageControllerType_NVMe;
785
786 return eStorageCtrl;
787
788 def isTestCfgSupported(self, asTestCfg):
789 """
790 Returns whether a specific test config is supported.
791 """
792
793 # Check whether the disk variant is supported by the selected format.
794 asVariants = self.getDiskFormatVariantsForTesting(asTestCfg[self.kiDiskFmt], [ asTestCfg[self.kiDiskVar] ]);
795 if len(asVariants) == 0:
796 return False;
797
798 # For iSCSI check whether we have targets configured.
799 if asTestCfg[self.kiDiskFmt] == 'iSCSI' and len(self.asIscsiTargets) == 0:
800 return False;
801
802 # Check for virt mode, CPU count and selected VM.
803 if asTestCfg[self.kiVirtMode] == 'raw' \
804 and (asTestCfg[self.kiCpuCount] > 1 or asTestCfg[self.kiVmName] == 'tst-storage'):
805 return False;
806
807 # IDE does not support the no host I/O cache setting
808 if asTestCfg[self.kiHostIoCache] == 'no-hostiocache' \
809 and asTestCfg[self.kiStorageCtrl] == 'IDE':
810 return False;
811
812 return True;
813
814 def fnFormatCpuString(self, cCpus):
815 """
816 Formats the CPU count to be readable.
817 """
818 if cCpus == 1:
819 return '1 cpu';
820 else:
821 return '%u cpus' % (cCpus);
822
823 def fnFormatVirtMode(self, sVirtMode):
824 """
825 Formats the virtualization mode to be a little less cryptic for use in test
826 descriptions.
827 """
828 return self.kdVirtModeDescs[sVirtMode];
829
830 def fnFormatHostIoCache(self, sHostIoCache):
831 """
832 Formats the host I/O cache mode to be a little less cryptic for use in test
833 descriptions.
834 """
835 return self.kdHostIoCacheDescs[sHostIoCache];
836
837 def testBenchmark(self, sTargetOs, sBenchmark, sMountpoint, oExecutor, dTestSet, \
838 cMsTimeout = 3600000):
839 """
840 Runs the given benchmark on the test host.
841 """
842
843 dTestSet['FilePath'] = sMountpoint;
844 dTestSet['TargetOs'] = sTargetOs;
845
846 oTst = None;
847 if sBenchmark == 'iozone':
848 oTst = IozoneTest(oExecutor, dTestSet);
849 elif sBenchmark == 'fio':
850 oTst = FioTest(oExecutor, dTestSet); # pylint: disable=R0204
851
852 if oTst is not None:
853 fRc = oTst.prepare();
854 if fRc:
855 fRc = oTst.run(cMsTimeout);
856 if fRc:
857 if self.fReportBenchmarkResults:
858 fRc = oTst.reportResult();
859 else:
860 reporter.testFailure('Running the testcase failed');
861 else:
862 reporter.testFailure('Preparing the testcase failed');
863
864 oTst.cleanup();
865
866 return fRc;
867
868 def createHd(self, oSession, sDiskFormat, sDiskVariant, iDiffLvl, oHdParent, \
869 sDiskPath, cbDisk):
870 """
871 Creates a new disk with the given parameters returning the medium object
872 on success.
873 """
874
875 oHd = None;
876 if sDiskFormat == "iSCSI" and iDiffLvl == 0:
877 listNames = [];
878 listValues = [];
879 listValues = self.asIscsiTargets[0].split('|');
880 listNames.append('TargetAddress');
881 listNames.append('TargetName');
882 listNames.append('LUN');
883
884 if self.fpApiVer >= 5.0:
885 oHd = oSession.oVBox.createMedium(sDiskFormat, sDiskPath, vboxcon.AccessMode_ReadWrite, \
886 vboxcon.DeviceType_HardDisk);
887 else:
888 oHd = oSession.oVBox.createHardDisk(sDiskFormat, sDiskPath);
889 oHd.type = vboxcon.MediumType_Normal;
890 oHd.setProperties(listNames, listValues);
891 else:
892 if iDiffLvl == 0:
893 tMediumVariant = self.convDiskToMediumVariant(sDiskVariant);
894 oHd = oSession.createBaseHd(sDiskPath + '/base.disk', sDiskFormat, cbDisk, \
895 cMsTimeout = 3600 * 1000, tMediumVariant = tMediumVariant);
896 else:
897 sDiskPath = sDiskPath + '/diff_%u.disk' % (iDiffLvl);
898 oHd = oSession.createDiffHd(oHdParent, sDiskPath, None);
899
900 return oHd;
901
902 def testOneCfg(self, sVmName, eStorageController, sHostIoCache, sDiskFormat, # pylint: disable=R0913,R0915
903 sDiskVariant, sDiskPath, cCpus, sIoTest, sVirtMode, sTestSet):
904 """
905 Runs the specified VM thru test #1.
906
907 Returns a success indicator on the general test execution. This is not
908 the actual test result.
909 """
910 oVM = self.getVmByName(sVmName);
911
912 dTestSet = self.kdTestSets.get(sTestSet);
913 cbDisk = dTestSet.get('DiskSizeGb') * 1024*1024*1024;
914 fHwVirt = sVirtMode != 'raw';
915 fNestedPaging = sVirtMode == 'hwvirt-np';
916
917 fRc = True;
918 if sDiskFormat == 'iSCSI':
919 sDiskPath = self.asIscsiTargets[0];
920 elif self.fUseScratch:
921 sDiskPath = self.sScratchPath;
922 else:
923 # If requested recreate the storage space to start with a clean config
924 # for benchmarks
925 if self.fRecreateStorCfg:
926 sMountPoint = self.prepareStorage(self.oStorCfg);
927 if sMountPoint is not None:
928 # Create a directory where every normal user can write to.
929 self.oStorCfg.mkDirOnVolume(sMountPoint, 'test', 0777);
930 sDiskPath = sMountPoint + '/test';
931 else:
932 fRc = False;
933 reporter.testFailure('Failed to prepare storage for VM');
934
935 if not fRc:
936 return fRc;
937
938 lstDisks = []; # List of disks we have to delete afterwards.
939
940 for iDiffLvl in range(self.cDiffLvls + 1):
941
942 if iDiffLvl == 0:
943 reporter.testStart('Base');
944 else:
945 reporter.testStart('Diff %u' % (iDiffLvl));
946
947 # Reconfigure the VM
948 oSession = self.openSession(oVM);
949 if oSession is not None:
950 # Attach HD
951 fRc = oSession.ensureControllerAttached(_ControllerTypeToName(eStorageController));
952 fRc = fRc and oSession.setStorageControllerType(eStorageController, _ControllerTypeToName(eStorageController));
953
954 if sHostIoCache == 'hostiocache':
955 fRc = fRc and oSession.setStorageControllerHostIoCache(_ControllerTypeToName(eStorageController), True);
956 elif sHostIoCache == 'no-hostiocache':
957 fRc = fRc and oSession.setStorageControllerHostIoCache(_ControllerTypeToName(eStorageController), False);
958
959 iDevice = 0;
960 if eStorageController == vboxcon.StorageControllerType_PIIX3 or \
961 eStorageController == vboxcon.StorageControllerType_PIIX4:
962 iDevice = 1; # Master is for the OS.
963
964 oHdParent = None;
965 if iDiffLvl > 0:
966 oHdParent = lstDisks[0];
967 oHd = self.createHd(oSession, sDiskFormat, sDiskVariant, iDiffLvl, oHdParent, sDiskPath, cbDisk);
968 if oHd is not None:
969 lstDisks.insert(0, oHd);
970 try:
971 if oSession.fpApiVer >= 4.0:
972 oSession.o.machine.attachDevice(_ControllerTypeToName(eStorageController), \
973 0, iDevice, vboxcon.DeviceType_HardDisk, oHd);
974 else:
975 oSession.o.machine.attachDevice(_ControllerTypeToName(eStorageController), \
976 0, iDevice, vboxcon.DeviceType_HardDisk, oHd.id);
977 except:
978 reporter.errorXcpt('attachDevice("%s",%s,%s,HardDisk,"%s") failed on "%s"' \
979 % (_ControllerTypeToName(eStorageController), 1, 0, oHd.id, oSession.sName) );
980 fRc = False;
981 else:
982 reporter.log('attached "%s" to %s' % (sDiskPath, oSession.sName));
983 else:
984 fRc = False;
985 fRc = fRc and oSession.enableVirtEx(fHwVirt);
986 fRc = fRc and oSession.enableNestedPaging(fNestedPaging);
987 fRc = fRc and oSession.setCpuCount(cCpus);
988 fRc = fRc and oSession.saveSettings();
989 fRc = oSession.close() and fRc and True; # pychecker hack.
990 oSession = None;
991 else:
992 fRc = False;
993
994 # Start up.
995 if fRc is True:
996 self.logVmInfo(oVM);
997 oSession, oTxsSession = self.startVmAndConnectToTxsViaTcp(sVmName, fCdWait = False, fNatForwardingForTxs = True);
998 if oSession is not None:
999 self.addTask(oSession);
1000
1001 # Fudge factor - Allow the guest to finish starting up.
1002 self.sleep(5);
1003
1004 # Prepare the storage on the guest
1005 lstBinaryPaths = ['/bin', '/sbin', '/usr/bin', '/usr/sbin' ];
1006 oExecVm = remoteexecutor.RemoteExecutor(oTxsSession, lstBinaryPaths, '${SCRATCH}');
1007 oStorCfgVm = storagecfg.StorageCfg(oExecVm, 'linux', self.getGuestDisk(oSession, oTxsSession, \
1008 eStorageController));
1009
1010 sMountPoint = self.prepareStorage(oStorCfgVm);
1011 if sMountPoint is not None:
1012 self.testBenchmark('linux', sIoTest, sMountPoint, oExecVm, dTestSet, \
1013 cMsTimeout = 2 * 3600 * 1000); # 2 hours max (Benchmark and QCOW takes some time)
1014 self.cleanupStorage(oStorCfgVm);
1015 else:
1016 reporter.testFailure('Failed to prepare storage for the guest benchmark');
1017
1018 # cleanup.
1019 self.removeTask(oTxsSession);
1020 self.terminateVmBySession(oSession)
1021 else:
1022 fRc = False;
1023
1024 # Remove disk
1025 oSession = self.openSession(oVM);
1026 if oSession is not None:
1027 try:
1028 oSession.o.machine.detachDevice(_ControllerTypeToName(eStorageController), 0, iDevice);
1029
1030 # Remove storage controller if it is not an IDE controller.
1031 if eStorageController is not vboxcon.StorageControllerType_PIIX3 \
1032 and eStorageController is not vboxcon.StorageControllerType_PIIX4:
1033 oSession.o.machine.removeStorageController(_ControllerTypeToName(eStorageController));
1034
1035 oSession.saveSettings();
1036 oSession.saveSettings();
1037 oSession.close();
1038 oSession = None;
1039 except:
1040 reporter.errorXcpt('failed to detach/delete disk %s from storage controller' % (sDiskPath));
1041 else:
1042 fRc = False;
1043
1044 reporter.testDone();
1045
1046 # Delete all disks
1047 for oHd in lstDisks:
1048 self.oVBox.deleteHdByMedium(oHd);
1049
1050 # Cleanup storage area
1051 if sDiskFormat != 'iSCSI' and not self.fUseScratch and self.fRecreateStorCfg:
1052 self.cleanupStorage(self.oStorCfg);
1053
1054 return fRc;
1055
1056 def testStorage(self, sDiskPath = None):
1057 """
1058 Runs the storage testcase through the selected configurations
1059 """
1060
1061 aasTestCfgs = [];
1062 aasTestCfgs.insert(self.kiVmName, self.asTestVMs);
1063 aasTestCfgs.insert(self.kiStorageCtrl, self.asStorageCtrls);
1064 aasTestCfgs.insert(self.kiHostIoCache, (self.asHostIoCache, self.fnFormatHostIoCache));
1065 aasTestCfgs.insert(self.kiDiskFmt, self.asDiskFormats);
1066 aasTestCfgs.insert(self.kiDiskVar, self.asDiskVariants);
1067 aasTestCfgs.insert(self.kiCpuCount, (self.acCpus, self.fnFormatCpuString));
1068 aasTestCfgs.insert(self.kiVirtMode, (self.asVirtModes, self.fnFormatVirtMode));
1069 aasTestCfgs.insert(self.kiIoTest, self.asTests);
1070 aasTestCfgs.insert(self.kiTestSet, self.asTestSets);
1071
1072 aasTestsBlacklist = [];
1073 aasTestsBlacklist.append(['tst-storage', 'BusLogic']); # 64bit Linux is broken with BusLogic
1074
1075 oTstCfgMgr = StorTestCfgMgr(aasTestCfgs, aasTestsBlacklist, self.isTestCfgSupported);
1076
1077 fRc = True;
1078 asTestCfg = oTstCfgMgr.getCurrentTestCfg();
1079 while len(asTestCfg) > 0:
1080 fRc = self.testOneCfg(asTestCfg[self.kiVmName], self.getStorageCtrlFromName(asTestCfg[self.kiStorageCtrl]), \
1081 asTestCfg[self.kiHostIoCache], asTestCfg[self.kiDiskFmt], asTestCfg[self.kiDiskVar],
1082 sDiskPath, asTestCfg[self.kiCpuCount], asTestCfg[self.kiIoTest], \
1083 asTestCfg[self.kiVirtMode], asTestCfg[self.kiTestSet]) and fRc and True; # pychecker hack.
1084
1085 asTestCfg = oTstCfgMgr.getNextTestCfg();
1086
1087 return fRc;
1088
1089 def test1(self):
1090 """
1091 Executes test #1.
1092 """
1093
1094 fRc = True;
1095 oDiskCfg = self.kdStorageCfgs.get(socket.gethostname().lower());
1096
1097 # Test the host first if requested
1098 if oDiskCfg is not None or self.fUseScratch:
1099 lstBinaryPaths = ['/bin', '/sbin', '/usr/bin', '/usr/sbin', \
1100 '/opt/csw/bin', '/usr/ccs/bin', '/usr/sfw/bin'];
1101 oExecutor = remoteexecutor.RemoteExecutor(None, lstBinaryPaths, self.sScratchPath);
1102 if not self.fUseScratch:
1103 self.oStorCfg = storagecfg.StorageCfg(oExecutor, utils.getHostOs(), oDiskCfg);
1104
1105 # Try to cleanup any leftovers from a previous run first.
1106 fRc = self.oStorCfg.cleanupLeftovers();
1107 if not fRc:
1108 reporter.error('Failed to cleanup any leftovers from a previous run');
1109
1110 if self.fTestHost:
1111 reporter.testStart('Host');
1112 if self.fUseScratch:
1113 sMountPoint = self.sScratchPath;
1114 else:
1115 sMountPoint = self.prepareStorage(self.oStorCfg);
1116 if sMountPoint is not None:
1117 for sIoTest in self.asTests:
1118 reporter.testStart(sIoTest);
1119 for sTestSet in self.asTestSets:
1120 reporter.testStart(sTestSet);
1121 dTestSet = self.kdTestSets.get(sTestSet);
1122 self.testBenchmark(utils.getHostOs(), sIoTest, sMountPoint, oExecutor, dTestSet);
1123 reporter.testDone();
1124 reporter.testDone();
1125 self.cleanupStorage(self.oStorCfg);
1126 else:
1127 reporter.testFailure('Failed to prepare host storage');
1128 fRc = False;
1129 reporter.testDone();
1130 else:
1131 # Create the storage space first if it is not done before every test.
1132 sMountPoint = None;
1133 if self.fUseScratch:
1134 sMountPoint = self.sScratchPath;
1135 elif not self.fRecreateStorCfg:
1136 reporter.testStart('Create host storage');
1137 sMountPoint = self.prepareStorage(self.oStorCfg);
1138 if sMountPoint is None:
1139 reporter.testFailure('Failed to prepare host storage');
1140 fRc = False;
1141 self.oStorCfg.mkDirOnVolume(sMountPoint, 'test', 0777);
1142 sMountPoint = sMountPoint + '/test';
1143 reporter.testDone();
1144
1145 if fRc:
1146 # Run the storage tests.
1147 if not self.testStorage(sMountPoint):
1148 fRc = False;
1149
1150 if not self.fRecreateStorCfg and not self.fUseScratch:
1151 self.cleanupStorage(self.oStorCfg);
1152 else:
1153 fRc = False;
1154
1155 return fRc;
1156
1157if __name__ == '__main__':
1158 sys.exit(tdStorageBenchmark().main(sys.argv));
1159
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette