VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/tests/unittests/tdUnitTest1.py@ 64247

Last change on this file since 64247 was 62484, checked in by vboxsync, 8 years ago

(C) 2016

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 31.4 KB
Line 
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3# $Id: tdUnitTest1.py 62484 2016-07-22 18:35:33Z vboxsync $
4
5"""
6VirtualBox Validation Kit - Unit Tests.
7"""
8
9__copyright__ = \
10"""
11Copyright (C) 2010-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: 62484 $"
31
32
33# Standard Python imports.
34import os
35import sys
36import re
37import subprocess
38
39
40# Only the main script needs to modify the path.
41try: __file__
42except: __file__ = sys.argv[0];
43g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
44sys.path.append(g_ksValidationKitDir)
45
46# Validation Kit imports.
47from common import utils;
48from testdriver import vbox
49from testdriver import reporter
50
51
52class tdUnitTest1(vbox.TestDriver):
53 """
54 Unit Tests.
55 """
56
57 ## The temporary exclude list.
58 ## @note This shall be empty before we release 4.3!
59 kdTestCasesBuggyPerOs = {
60 'darwin': {
61 'testcase/tstX86-1': '', # 'FSTP M32R, ST0' fails; no idea why.
62 },
63 'linux': {
64 'testcase/tstRTFileAio': '', # See xTracker #8035.
65 },
66 'solaris': {
67 'testcase/tstIntNet-1': '', # Fails opening rge0, probably a generic issue figuring which nic to use.
68 'testcase/tstIprtList': '', # Crashes in the multithreaded test, I think.
69 'testcase/tstRTCritSect': '', # Fairness/whatever issue here.
70 'testcase/tstRTR0MemUserKernelDriver': '', # Failes when kernel to kernel buffers.
71 'testcase/tstRTSemRW': '', # line 338: RTSemRWReleaseRead(hSemRW): got VERR_ACCESS_DENIED
72 'testcase/tstRTStrAlloc': '', # VERR_NO_STR_MEMORY!
73 'testcase/tstRTFileGetSize-1': '', # VERR_DEV_IO_ERROR on /dev/null!
74 },
75 'win': {
76 'testcase/tstFile': '', # ??
77 'testcase/tstIntNet-1': '', # possibly same issue as solaris.
78 'testcase/tstMouseImpl': '', # STATUS_ACCESS_VIOLATION
79 'testcase/tstRTR0ThreadPreemptionDriver': '', # ??
80 'testcase/tstRTPath': '<4.3.51r89894',
81 'testcase/tstRTPipe': '', # ??
82 'testcase/tstRTR0MemUserKernelDriver': '', # ??
83 'testcase/tstRTR0SemMutexDriver': '', # ??
84 'testcase/tstRTStrAlloc': '', # ??
85 'testcase/tstRTStrFormat': '', # ??
86 'testcase/tstRTSystemQueryOsInfo': '', # ??
87 'testcase/tstRTTemp': '', # ??
88 'testcase/tstRTTime': '', # ??
89 'testcase/tstTime-2': '', # Total time differs too much! ... delta=-10859859
90 'testcase/tstUtf8': '', # ??
91 'testcase/tstVMMR0CallHost-2': '', # STATUS_STACK_OVERFLOW
92 'testcase/tstX86-1': '', # Fails on win.x86.
93 'tscpasswd': '', # ??
94 'tstVMREQ': '', # ?? Same as darwin.x86?
95 },
96 'win.x86': {
97 'testcase/tstRTR0TimerDriver': '', # See xTracker #8041.
98 }
99 };
100
101 kdTestCasesBuggy = {
102 'testcase/tstGuestPropSvc': '', # GET_NOTIFICATION fails on testboxlin5.de.oracle.com and others.
103 'testcase/tstRTProcCreateEx': '', # Seen failing on wei01-b6ka-9.de.oracle.com.
104 'testcase/tstTimer': '', # Sometimes fails on linux, not important atm.
105 'testcase/tstGIP-2': '', # 2015-09-10: Fails regularly. E.g. TestSetID 2744205 (testboxsh2),
106 # 2743961 (wei01-b6kc-6). The responsible engineer should reenable
107 # it once it has been fixed.
108 };
109
110 ## The permanent exclude list.
111 # @note Stripped of extensions!
112 kdTestCasesBlackList = {
113 'testcase/tstClipboardX11Smoke': '',
114 'testcase/tstFileLock': '',
115 'testcase/tstDisasm-2': '', # without parameters it will disassembler 1GB starting from 0
116 'testcase/tstFileAppendWin-1': '',
117 'testcase/tstDir': '', # useless without parameters
118 'testcase/tstDir-2': '', # useless without parameters
119 'testcase/tstGlobalConfig': '',
120 'testcase/tstHostHardwareLinux': '', # must be killed with CTRL-C
121 'testcase/tstHttp': '', # Talks to outside servers.
122 'testcase/tstRTHttp': '', # parameters required
123 'testcase/tstLdr-2': '', # parameters required
124 'testcase/tstLdr-3': '', # parameters required
125 'testcase/tstLdr': '', # parameters required
126 'testcase/tstLdrLoad': '', # parameters required
127 'testcase/tstMove': '', # parameters required
128 'testcase/tstRTR0Timer': '', # loads 'tstRTR0Timer.r0'
129 'testcase/tstRTR0ThreadDriver': '', # loads 'tstRTR0Thread.r0'
130 'testcase/tstRunTestcases': '', # that's a script like this one
131 'testcase/tstRTReqPool': '', # fails sometimes, testcase buggy
132 'testcase/tstRTS3': '', # parameters required
133 'testcase/tstSDL': '', # graphics test
134 'testcase/tstSupLoadModule': '', # Needs parameters and vboxdrv access. Covered elsewhere.
135 'testcase/tstSeamlessX11': '', # graphics test
136 'testcase/tstTime-3': '', # parameters required
137 'testcase/tstVBoxControl': '', # works only inside a guest
138 'testcase/tstVDCopy': '', # parameters required
139 'testcase/tstVDFill': '', # parameters required
140 'tstAnimate': '', # parameters required
141 'testcase/tstAPI': '', # user interaction required
142 'tstCollector': '', # takes forever
143 'testcase/tstHeadless': '', # parameters required
144 'tstHeadless': '', # parameters required
145 'tstMicroRC': '', # required by tstMicro
146 'tstVBoxDbg': '', # interactive test
147 'testcase/tstTestServMgr': '', # some strange xpcom18a4 test, does not work
148 'tstTestServMgr': '', # some strange xpcom18a4 test, does not work
149 'tstPDMAsyncCompletion': '', # parameters required
150 'testcase/tstXptDump': '', # parameters required
151 'tstXptDump': '', # parameters required
152 'testcase/tstnsIFileEnumerator': '', # some strange xpcom18a4 test, does not work
153 'tstnsIFileEnumerator': '', # some strange xpcom18a4 test, does not work
154 'testcase/tstSimpleTypeLib': '', # parameters required
155 'tstSimpleTypeLib': '', # parameters required
156 'testcase/tstTestAtoms': '', # additional test file (words.txt) required
157 'tstTestAtoms': '', # additional test file (words.txt) required
158 'testcase/tstXptLink': '', # parameters required
159 'tstXptLink': '', # parameters required
160 'tstXPCOMCGlue': '', # user interaction required
161 'testcase/tstXPCOMCGlue': '', # user interaction required
162 'testcase/tstCAPIGlue': '', # user interaction required
163 'testcase/tstTestCallTemplates': '', # some strange xpcom18a4 test, segfaults
164 'tstTestCallTemplates': '', # some strange xpcom18a4 test, segfaults
165 'testcase/tstRTFilesystem': '', # parameters required
166 'testcase/tstRTDvm': '', # parameters required
167 # later
168 'testcase/tstIntNetR0': '', # RTSPINLOCK_FLAGS_INTERRUPT_SAFE == RTSPINLOCK_FLAGS_INTERRUPT_UNSAFE
169 # slow stuff
170 'testcase/tstAvl': '', # SLOW!
171 'testcase/tstRTAvl': '', # SLOW! (new name)
172 'testcase/tstVD': '', # 8GB fixed-sized vmdk
173 # failed or hang
174 'testcase/tstCryptoPkcs7Verify': '', # hang
175 'tstOVF': '', # hang (only ancient version, now in new place)
176 'testcase/tstRTLockValidator': '', # Lock validation is not enabled for critical sections
177 'testcase/tstGuestControlSvc': '', # failed: line 288: testHost(&svcTable): expected VINF_SUCCESS, got VERR_NOT_FOUND
178 'testcase/tstRTMemEf': '', # failed w/o error message
179 'testcase/tstSupSem': '', # failed: SRE Timeout Accuracy (ms) : FAILED (1 errors)
180 'testcase/tstCryptoPkcs7Sign': '',# failed: 29330:error:02001002:lib(2):func(1):reason(2):NA:0:fopen('server.pem': '','r')
181 'testcase/tstCompressionBenchmark': '', # failed: error: RTZipBlockCompress failed
182 # for 'RTZipBlock/LZJB' (#4): VERR_NOT_SUPPORTED
183 'testcase/tstLdr-4': '', # failed: Failed to get bits for '/home/vbox/test/tmp/bin/testcase/tstLdrObjR0.r0'/0,
184 # rc=VERR_SYMBOL_VALUE_TOO_BIG. aborting test
185 'tstPDMAsyncCompletionStress': '', # VERR_INVALID_PARAMETER (cbSize = 0)
186 'tstMicro': '', # doesn't work on solaris, fix later if we care.
187 'tstVMM-HwAccm': '', # failed: Only checked AMD-V on linux
188 'tstVMM-HM': '', # failed: Only checked AMD-V on linux
189 'tstVMMFork': '', # failed: xtracker 6171
190 'tstTestFactory': '', # some strange xpcom18a4 test, does not work
191 'testcase/tstRTSemXRoads': '', # sporadically failed: Traffic - 8 threads per direction, 10 sec : FAILED (8 errors)
192 'tstVBoxAPILinux': '', # creates VirtualBox directories for root user because of sudo (should be in vbox)
193 'testcase/tstVMStructDTrace': '', # This is a D-script generator.
194 'tstVMStructRC': '', # This is a C-code generator.
195 'tstDeviceStructSizeRC': '', # This is a C-code generator.
196 'testcase/tstTSC': '', # Doesn't test anything and might fail with HT or/and too many cores.
197 'testcase/tstOpenUSBDev': '', # Not a useful testcase.
198 'testcase/tstX86-1': '', # Really more guest side.
199 'testcase/tstX86-FpuSaveRestore': '', # Experiments, could be useful for the guest not the host.
200 'tstAsmStructsRC': '', # Testcase run during build time (fails to find libstdc++.so.6 on some
201 # Solaris testboxes).
202 };
203
204 # Suffix exclude list.
205 kasSuffixBlackList = [
206 '.r0',
207 '.gc',
208 '.debug',
209 '.rel',
210 '.sys',
211 '.ko',
212 '.o',
213 '.obj',
214 '.lib',
215 '.a',
216 '.so',
217 '.dll',
218 '.dylib',
219 '.tmp',
220 '.log',
221 '.py',
222 '.pyc',
223 '.pyo',
224 '.pdb',
225 '.dSYM',
226 '.sym',
227 ];
228
229 ## The exclude list.
230 # @note Stripped extensions!
231 kasHardened = [
232 "testcase/tstIntNet-1",
233 "testcase/tstR0ThreadPreemptionDriver", # VBox 4.3
234 "testcase/tstRTR0ThreadPreemptionDriver",
235 "testcase/tstRTR0MemUserKernelDriver",
236 "testcase/tstRTR0SemMutexDriver",
237 "testcase/tstRTR0TimerDriver",
238 "testcase/tstRTR0ThreadDriver",
239 'testcase/tstRTR0DbgKrnlInfoDriver',
240 "tstInt",
241 "tstVMM",
242 "tstVMMFork",
243 "tstVMREQ",
244 'testcase/tstCFGM',
245 'testcase/tstContiguous',
246 'testcase/tstGetPagingMode',
247 'testcase/tstGIP-2',
248 'testcase/tstInit',
249 'testcase/tstLow',
250 'testcase/tstMMHyperHeap',
251 'testcase/tstPage',
252 'testcase/tstPin',
253 'testcase/tstRTTime', 'testcase/tstTime', # GIP test case.
254 'testcase/tstSSM',
255 'testcase/tstSupSem-Zombie',
256 ]
257
258 ## Argument lists
259 kdArguments = {
260 'testcase/tstbntest': [ '-out', os.devnull, ], # Very noisy.
261 };
262
263
264 ## Status code translations.
265 ## @{
266 kdExitCodeNames = {
267 0: 'RTEXITCODE_SUCCESS',
268 1: 'RTEXITCODE_FAILURE',
269 2: 'RTEXITCODE_SYNTAX',
270 3: 'RTEXITCODE_INIT',
271 4: 'RTEXITCODE_SKIPPED',
272 };
273 kdExitCodeNamesWin = {
274 -1073741515: 'STATUS_DLL_NOT_FOUND',
275 -1073741512: 'STATUS_ORDINAL_NOT_FOUND',
276 -1073741511: 'STATUS_ENTRYPOINT_NOT_FOUND',
277 -1073741502: 'STATUS_DLL_INIT_FAILED',
278 -1073741500: 'STATUS_UNHANDLED_EXCEPTION',
279 -1073741499: 'STATUS_APP_INIT_FAILURE',
280 -1073741819: 'STATUS_ACCESS_VIOLATION',
281 -1073741571: 'STATUS_STACK_OVERFLOW',
282 };
283 ## @}
284
285 def __init__(self):
286 """
287 Reinitialize child class instance.
288 """
289 vbox.TestDriver.__init__(self)
290 self.oTestVmSet = None;
291
292 self.sVBoxInstallRoot = None
293
294 self.cSkipped = 0
295 self.cPassed = 0
296 self.cFailed = 0
297
298 self.sUnitTestsPathBase = None
299 self.sExeSuff = '.exe' if utils.getHostOs() in [ 'win', 'dos', 'os2' ] else '';
300
301 self.aiVBoxVer = (4, 3, 0, 0);
302
303 # For testing testcase logic.
304 self.fDryRun = False;
305
306
307 def _detectPaths(self):
308 """
309 Internal worker for actionVerify and actionExecute that detects paths.
310
311 This sets sVBoxInstallRoot and sUnitTestsPathBase and returns True/False.
312 """
313
314 #
315 # We need a VBox install (/ build) to test.
316 #
317 if False is True:
318 if not self.importVBoxApi():
319 reporter.error('Unabled to import the VBox Python API.')
320 return False
321 else:
322 self._detectBuild();
323 if self.oBuild is None:
324 reporter.error('Unabled to detect the VBox build.');
325 return False;
326
327 #
328 # Where are the files installed?
329 # Solaris requires special handling because of it's multi arch subdirs.
330 #
331 self.sVBoxInstallRoot = self.oBuild.sInstallPath
332 if not self.oBuild.isDevBuild() and utils.getHostOs() == 'solaris':
333 sArchDir = utils.getHostArch();
334 if sArchDir == 'x86': sArchDir = 'i386';
335 self.sVBoxInstallRoot = os.path.join(self.sVBoxInstallRoot, sArchDir);
336
337 # Add the installation root to the PATH on windows so we can get DLLs from it.
338 if utils.getHostOs() == 'win':
339 sPathName = 'PATH';
340 if not sPathName in os.environ:
341 sPathName = 'Path';
342 sPath = os.environ.get(sPathName, '.');
343 if len(sPath) > 0 and sPath[-1] != ';':
344 sPath += ';';
345 os.environ[sPathName] = sPath + self.sVBoxInstallRoot + ';';
346
347 #
348 # The unittests are generally not installed, so look for them.
349 #
350 sBinOrDist = 'dist' if utils.getHostOs() in [ 'darwin', ] else 'bin';
351 asCandidates = [
352 self.oBuild.sInstallPath,
353 os.path.join(self.sScratchPath, sBinOrDist + '.' + utils.getHostArch()),
354 os.path.join(self.sScratchPath, sBinOrDist, utils.getHostArch()),
355 os.path.join(self.sScratchPath, sBinOrDist),
356 ];
357 if utils.getHostOs() == 'darwin':
358 for i in range(1, len(asCandidates)):
359 asCandidates[i] = os.path.join(asCandidates[i], 'VirtualBox.app', 'Contents', 'MacOS');
360
361 for sCandidat in asCandidates:
362 if os.path.exists(os.path.join(sCandidat, 'testcase', 'tstVMStructSize' + self.sExeSuff)):
363 self.sUnitTestsPathBase = sCandidat;
364 return True;
365
366 reporter.error('Unable to find unit test dir. Candidates: %s' % (asCandidates,))
367 return False;
368
369 #
370 # Overridden methods.
371 #
372
373 def actionVerify(self):
374 return self._detectPaths();
375
376 def actionExecute(self):
377
378 if self.sUnitTestsPathBase is None and self._detectPaths():
379 return False;
380
381 self._figureVersion();
382 self._makeEnvironmentChanges();
383
384 self.testRunUnitTestsSet(r'^tst*', 'testcase')
385 self.testRunUnitTestsSet(r'^tst*', '.')
386
387 reporter.log('')
388 reporter.log('********************')
389 reporter.log('*** PASSED: %d' % self.cPassed)
390 reporter.log('*** FAILED: %d' % self.cFailed)
391 reporter.log('*** SKIPPED: %d' % self.cSkipped)
392 reporter.log('*** TOTAL: %d' % (self.cPassed + self.cFailed + self.cSkipped))
393
394 return self.cFailed == 0
395
396 #
397 # Test execution helpers.
398 #
399
400 def _figureVersion(self):
401 """ Tries to figure which VBox version this is, setting self.aiVBoxVer. """
402 try:
403 sVer = utils.processOutputChecked(['VBoxManage', '--version'])
404
405 sVer = sVer.strip();
406 sVer = re.sub(r'_BETA.*r', '.', sVer);
407 sVer = re.sub(r'_ALPHA.*r', '.', sVer);
408 sVer = re.sub(r'_RC.*r', '.', sVer);
409 sVer = sVer.replace('r', '.');
410
411 self.aiVBoxVer = [int(sComp) for sComp in sVer.split('.')];
412
413 reporter.log('VBox version: %s' % (self.aiVBoxVer,));
414 except:
415 reporter.logXcpt();
416 return False;
417 return True;
418
419 def _compareVersion(self, aiVer):
420 """
421 Compares the give version string with the vbox version string,
422 returning a result similar to C strcmp(). aiVer is on the right side.
423 """
424 cComponents = min(len(self.aiVBoxVer), len(aiVer));
425 for i in range(cComponents):
426 if self.aiVBoxVer[i] < aiVer[i]:
427 return -1;
428 if self.aiVBoxVer[i] > aiVer[i]:
429 return 1;
430 return len(self.aiVBoxVer) - len(aiVer);
431
432 def _isExcluded(self, sTest, dExclList):
433 """ Checks if the testcase is excluded or not. """
434 if sTest in dExclList:
435 sFullExpr = dExclList[sTest].replace(' ', '').strip();
436 if sFullExpr == '':
437 return True;
438
439 # Consider each exclusion expression. These are generally ranges,
440 # either open ended or closed: "<4.3.51r12345", ">=4.3.0 && <=4.3.4".
441 asExprs = sFullExpr.split(';');
442 for sExpr in asExprs:
443
444 # Split it on the and operator and process each sub expression.
445 fResult = True;
446 for sSubExpr in sExpr.split('&&'):
447 # Split out the comparison operator and the version value.
448 if sSubExpr.startswith('<=') or sSubExpr.startswith('>='):
449 sOp = sSubExpr[:2];
450 sValue = sSubExpr[2:];
451 elif sSubExpr.startswith('<') or sSubExpr.startswith('>') or sSubExpr.startswith('='):
452 sOp = sSubExpr[:1];
453 sValue = sSubExpr[1:];
454 else:
455 sOp = sValue = '';
456
457 # Convert the version value, making sure we've got a valid one.
458 try: aiValue = [int(sComp) for sComp in sValue.replace('r', '.').split('.')];
459 except: aiValue = ();
460 if len(aiValue) == 0 or len(aiValue) > 4:
461 reporter.error('Invalid exclusion expression for %s: "%s" [%s]' % (sTest, sSubExpr, dExclList[sTest]));
462 return True;
463
464 # Do the compare.
465 iCmp = self._compareVersion(aiValue);
466 if sOp == '>=' and iCmp < 0:
467 fResult = False;
468 elif sOp == '>' and iCmp <= 0:
469 fResult = False;
470 elif sOp == '<' and iCmp >= 0:
471 fResult = False;
472 elif sOp == '>=' and iCmp < 0:
473 fResult = False;
474 reporter.log2('iCmp=%s; %s %s %s -> %s' % (iCmp, self.aiVBoxVer, sOp, aiValue, fResult));
475
476 # Did the expression match?
477 if fResult:
478 return True;
479
480 return False;
481
482 def _sudoExecuteSync(self, asArgs):
483 """
484 Executes a sudo child process synchronously.
485 Returns True if the process executed successfully and returned 0,
486 otherwise False is returned.
487 """
488 reporter.log('Executing [sudo]: %s' % (asArgs, ));
489 try:
490 iRc = utils.sudoProcessCall(asArgs, shell = False, close_fds = False);
491 except:
492 reporter.errorXcpt();
493 return False;
494 reporter.log('Exit code [sudo]: %s (%s)' % (iRc, asArgs));
495 return iRc is 0;
496
497 def _hardenedMkDir(self, sPath):
498 """
499 Creates the directory specified sPath (including parents).
500 """
501 reporter.log('_hardenedMkDir: %s' % (sPath,));
502 if utils.getHostOs() in [ 'win', 'os2' ]:
503 os.makedirs(sPath, 0755);
504 else:
505 fRc = self._sudoExecuteSync(['/bin/mkdir', '-p', '-m', '0755', sPath]);
506 if fRc is not True:
507 raise Exception('Failed to create dir "%s".' % (sPath,));
508 return True;
509
510 def _hardenedCopyFile(self, sSrc, sDst, iMode):
511 """
512 Copies a file.
513 """
514 reporter.log('_hardenedCopyFile: %s -> %s (mode: %o)' % (sSrc, sDst, iMode,));
515 if utils.getHostOs() in [ 'win', 'os2' ]:
516 utils.copyFileSimple(sSrc, sDst);
517 os.chmod(sDst, iMode);
518 else:
519 fRc = self._sudoExecuteSync(['/bin/cp', sSrc, sDst]);
520 if fRc is not True:
521 raise Exception('Failed to copy "%s" to "%s".' % (sSrc, sDst,));
522 fRc = self._sudoExecuteSync(['/bin/chmod', '%o' % (iMode,), sDst]);
523 if fRc is not True:
524 raise Exception('Failed to chmod "%s".' % (sDst,));
525 return True;
526
527 def _hardenedDeleteFile(self, sPath):
528 """
529 Deletes a file.
530 """
531 reporter.log('_hardenedDeleteFile: %s' % (sPath,));
532 if os.path.exists(sPath):
533 if utils.getHostOs() in [ 'win', 'os2' ]:
534 os.remove(sPath);
535 else:
536 fRc = self._sudoExecuteSync(['/bin/rm', sPath]);
537 if fRc is not True:
538 raise Exception('Failed to remove "%s".' % (sPath,));
539 return True;
540
541 def _hardenedRemoveDir(self, sPath):
542 """
543 Removes a directory.
544 """
545 reporter.log('_hardenedRemoveDir: %s' % (sPath,));
546 if os.path.exists(sPath):
547 if utils.getHostOs() in [ 'win', 'os2' ]:
548 os.rmdir(sPath);
549 else:
550 fRc = self._sudoExecuteSync(['/bin/rmdir', sPath]);
551 if fRc is not True:
552 raise Exception('Failed to remove "%s".' % (sPath,));
553 return True;
554
555 def _executeTestCase(self, sName, sFullPath, sTestCaseSubDir, oDevNull): # pylint: disable=R0914
556 """
557 Executes a test case.
558 """
559
560 fSkipped = False;
561
562 #
563 # If hardening is enabled, some test cases and their dependencies
564 # needs to be copied to and execute from the sVBoxInstallRoot
565 # directory in order to work. They also have to be executed as
566 # root, i.e. via sudo.
567 #
568 fHardened = False;
569 asFilesToRemove = []; # Stuff to clean up.
570 asDirsToRemove = []; # Ditto.
571 if sName in self.kasHardened \
572 and self.sUnitTestsPathBase != self.sVBoxInstallRoot:
573
574 sDstDir = os.path.join(self.sVBoxInstallRoot, sTestCaseSubDir);
575 if not os.path.exists(sDstDir):
576 self._hardenedMkDir(sDstDir);
577 asDirsToRemove.append(sDstDir);
578
579 sDst = os.path.join(sDstDir, os.path.basename(sFullPath));
580 self._hardenedCopyFile(sFullPath, sDst, 0755);
581 asFilesToRemove.append(sDst);
582
583 # Copy any associated .dll/.so/.dylib.
584 for sSuff in [ '.dll', '.so', '.dylib' ]:
585 sSrc = os.path.splitext(sFullPath)[0] + sSuff;
586 if os.path.exists(sSrc):
587 sDst = os.path.join(sDstDir, os.path.basename(sSrc));
588 self._hardenedCopyFile(sSrc, sDst, 0644);
589 asFilesToRemove.append(sDst);
590
591 # Copy any associated .r0, .rc and .gc modules.
592 offDriver = sFullPath.rfind('Driver')
593 if offDriver > 0:
594 for sSuff in [ '.r0', 'RC.rc', 'RC.gc' ]:
595 sSrc = sFullPath[:offDriver] + sSuff;
596 if os.path.exists(sSrc):
597 sDst = os.path.join(sDstDir, os.path.basename(sSrc));
598 self._hardenedCopyFile(sSrc, sDst, 0644);
599 asFilesToRemove.append(sDst);
600
601 sFullPath = os.path.join(sDstDir, os.path.basename(sFullPath));
602 fHardened = True;
603
604 #
605 # Set up arguments and environment.
606 #
607 asArgs = [sFullPath,]
608 if sName in self.kdArguments:
609 asArgs.extend(self.kdArguments[sName]);
610
611 os.environ['IPRT_TEST_OMIT_TOP_TEST'] = '1';
612 os.environ['IPRT_TEST_FILE'] = sXmlFile = os.path.join(self.sScratchPath, 'result.xml');
613 if os.path.exists(sXmlFile):
614 try: os.unlink(sXmlFile);
615 except: self._hardenedDeleteFile(sXmlFile);
616
617 #
618 # Execute the test case.
619 #
620 # Windows is confusing output. Trying a few things to get rid of this.
621 # First, flush both stderr and stdout before running the child. Second,
622 # assign the child stderr to stdout. If this doesn't help, we'll have
623 # to capture the child output.
624 #
625 reporter.log('*** Executing %s%s...' % (asArgs, ' [hardened]' if fHardened else ''));
626 try: sys.stdout.flush();
627 except: pass;
628 try: sys.stderr.flush();
629 except: pass;
630 if not self.fDryRun:
631 try:
632 if fHardened:
633 oChild = utils.sudoProcessPopen(asArgs, stdin = oDevNull, stdout = sys.stdout, stderr = sys.stdout);
634 else:
635 oChild = subprocess.Popen( asArgs, stdin = oDevNull, stdout = sys.stdout, stderr = sys.stdout);
636 except:
637 if sName in [ 'tstAsmStructsRC', # 32-bit, may fail to start on 64-bit linux. Just ignore.
638 ]:
639 reporter.logXcpt();
640 fSkipped = True;
641 else:
642 reporter.errorXcpt();
643 iRc = 1023;
644 oChild = None;
645
646 if oChild is not None:
647 self.pidFileAdd(oChild.pid, fSudo = fHardened);
648 iRc = oChild.wait();
649 self.pidFileRemove(oChild.pid);
650 else:
651 iRc = 0;
652
653 #
654 # Clean up
655 #
656 for sPath in asFilesToRemove:
657 self._hardenedDeleteFile(sPath);
658 for sPath in asDirsToRemove:
659 self._hardenedRemoveDir(sPath);
660
661 #
662 # Report.
663 #
664 if os.path.exists(sXmlFile):
665 reporter.addSubXmlFile(sXmlFile);
666 if fHardened:
667 self._hardenedDeleteFile(sXmlFile);
668 else:
669 os.unlink(sXmlFile);
670
671 if iRc == 0:
672 reporter.log('*** %s: exit code %d' % (sFullPath, iRc));
673 self.cPassed += 1
674
675 elif iRc == 4: # RTEXITCODE_SKIPPED
676 reporter.log('*** %s: exit code %d (RTEXITCODE_SKIPPED)' % (sFullPath, iRc));
677 fSkipped = True;
678 self.cSkipped += 1;
679
680 elif fSkipped:
681 reporter.log('*** %s: exit code %d (Skipped)' % (sFullPath, iRc));
682 self.cSkipped += 1;
683
684 else:
685 sName = self.kdExitCodeNames.get(iRc, '');
686 if iRc in self.kdExitCodeNamesWin and utils.getHostOs() == 'win':
687 sName = self.kdExitCodeNamesWin[iRc];
688 if sName != '':
689 sName = ' (%s)' % (sName);
690
691 if iRc != 1:
692 reporter.testFailure('Exit status: %d%s' % (iRc, sName));
693 reporter.log( '!*! %s: exit code %d%s' % (sFullPath, iRc, sName));
694 else:
695 reporter.error('!*! %s: exit code %d%s' % (sFullPath, iRc, sName));
696 self.cFailed += 1
697
698 return fSkipped;
699
700 def testRunUnitTestsSet(self, sTestCasePattern, sTestCaseSubDir):
701 """
702 Run subset of the unit tests set.
703 """
704
705 # Open /dev/null for use as stdin further down.
706 try:
707 oDevNull = open(os.path.devnull, 'w+');
708 except:
709 oDevNull = None;
710
711 # Determin the host OS specific exclusion lists.
712 dTestCasesBuggyForHostOs = self.kdTestCasesBuggyPerOs.get(utils.getHostOs(), []);
713 dTestCasesBuggyForHostOs.update(self.kdTestCasesBuggyPerOs.get(utils.getHostOsDotArch(), []));
714
715 #
716 # Process the file list and run everything looking like a testcase.
717 #
718 for sFilename in sorted(os.listdir(os.path.join(self.sUnitTestsPathBase, sTestCaseSubDir))):
719 # Separate base and suffix and morph the base into something we
720 # can use for reporting and array lookups.
721 sName, sSuffix = os.path.splitext(sFilename);
722 if sTestCaseSubDir != '.':
723 sName = sTestCaseSubDir + '/' + sName;
724
725 # Basic exclusion.
726 if not re.match(sTestCasePattern, sFilename) \
727 or sSuffix in self.kasSuffixBlackList:
728 reporter.log('"%s" is not a test case.' % (sFilename,))
729 continue
730
731 # Check if the testcase is black listed or buggy before executing it.
732 if self._isExcluded(sName, self.kdTestCasesBlackList):
733 # (No testStart/Done or accounting here!)
734 reporter.log('%s: SKIPPED (blacklisted)' % (sName,));
735
736 elif self._isExcluded(sName, self.kdTestCasesBuggy):
737 reporter.testStart(sName);
738 reporter.log('%s: Skipping, buggy in general.' % (sName,));
739 reporter.testDone(fSkipped = True);
740 self.cSkipped += 1;
741
742 elif self._isExcluded(sName, dTestCasesBuggyForHostOs):
743 reporter.testStart(sName);
744 reporter.log('%s: Skipping, buggy on %s.' % (sName, utils.getHostOs(),));
745 reporter.testDone(fSkipped = True);
746 self.cSkipped += 1;
747
748 else:
749 sFullPath = os.path.normpath(os.path.join(self.sUnitTestsPathBase, os.path.join(sTestCaseSubDir, sFilename)));
750 reporter.testStart(sName);
751 try:
752 fSkipped = self._executeTestCase(sName, sFullPath, sTestCaseSubDir, oDevNull);
753 except:
754 reporter.errorXcpt('!*!');
755 self.cFailed += 1;
756 fSkipped = False;
757 reporter.testDone(fSkipped);
758
759
760
761if __name__ == '__main__':
762 sys.exit(tdUnitTest1().main(sys.argv))
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