#!/usr/bin/env python # -*- coding: utf-8 -*- # $Id: tdSerial1.py 106061 2024-09-16 14:03:52Z vboxsync $ """ VirtualBox Validation Kit - Serial port testing #1. """ __copyright__ = \ """ Copyright (C) 2018-2024 Oracle and/or its affiliates. This file is part of VirtualBox base platform packages, as available from https://www.virtualbox.org. 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, in version 3 of the License. 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 the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, see . The contents of this file may alternatively be used under the terms of the Common Development and Distribution License Version 1.0 (CDDL), a copy of it is provided in the "COPYING.CDDL" file included in the VirtualBox distribution, in which case the provisions of the CDDL are applicable instead of those of the GPL. You may elect to license modified versions of this file under the terms and conditions of either the GPL or the CDDL or both. SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0 """ __version__ = "$Revision: 106061 $" # Standard Python imports. import os; import random; import string; import struct; import sys; # Only the main script needs to modify the path. try: __file__ # pylint: disable=used-before-assignment except: __file__ = sys.argv[0]; g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))); sys.path.append(g_ksValidationKitDir); # Validation Kit imports. from testdriver import base; from testdriver import reporter; from testdriver import vbox; from testdriver import vboxcon; import loopback; # Python 3 hacks: if sys.version_info[0] >= 3: xrange = range; # pylint: disable=redefined-builtin,invalid-name class tdSerial1(vbox.TestDriver): """ VBox serial port testing #1. """ def __init__(self): vbox.TestDriver.__init__(self); self.asRsrcs = None; self.oTestVmSet = self.oTestVmManager.selectSet(self.oTestVmManager.kfGrpStdSmoke); self.asSerialModesDef = ['RawFile', 'Tcp', 'TcpServ', 'NamedPipe', 'NamedPipeServ', 'HostDev']; self.asSerialModes = self.asSerialModesDef; self.asSerialTestsDef = ['Write', 'ReadWrite']; self.asSerialTests = self.asSerialTestsDef; self.asUartsDef = ['16450', '16550A', '16750']; self.asUarts = self.asUartsDef; self.oLoopback = None; self.sLocation = None; self.fVerboseTest = False; # # Overridden methods. # def showUsage(self): rc = vbox.TestDriver.showUsage(self); reporter.log(''); reporter.log('tdSerial1 Options:'); reporter.log(' --serial-modes = len(asArgs): raise base.InvalidOption('The "--serial-modes" takes a colon separated list of serial port modes to test'); self.asSerialModes = asArgs[iArg].split(':'); for s in self.asSerialModes: if s not in self.asSerialModesDef: reporter.log('warning: The "--serial-modes" value "%s" is not a valid serial port mode.' % (s)); elif asArgs[iArg] == '--serial-tests': iArg += 1; if iArg >= len(asArgs): raise base.InvalidOption('The "--serial-tests" takes a colon separated list of serial port tests'); self.asSerialTests = asArgs[iArg].split(':'); for s in self.asSerialTests: if s not in self.asSerialTestsDef: reporter.log('warning: The "--serial-tests" value "%s" is not a valid serial port test.' % (s)); elif asArgs[iArg] == '--uarts': iArg += 1; if iArg >= len(asArgs): raise base.InvalidOption('The "--uarts" takes a colon separated list of uarts to test'); self.asUarts = asArgs[iArg].split(':'); for s in self.asUarts: if s not in self.asUartsDef: reporter.log('warning: The "--uarts" value "%s" is not a valid uart.' % (s)); elif asArgs[iArg] == '--verbose-test': iArg += 1; self.fVerboseTest = True; else: return vbox.TestDriver.parseOption(self, asArgs, iArg); return iArg + 1; def actionVerify(self): if self.sVBoxValidationKitIso is None or not os.path.isfile(self.sVBoxValidationKitIso): reporter.error('Cannot find the VBoxValidationKit.iso! (%s)' 'Please unzip a Validation Kit build in the current directory or in some parent one.' % (self.sVBoxValidationKitIso,) ); return False; return vbox.TestDriver.actionVerify(self); def actionConfig(self): # Make sure vboxapi has been imported so we can use the constants. if not self.importVBoxApi(): return False; assert self.sVBoxValidationKitIso is not None; return self.oTestVmSet.actionConfig(self, sDvdImage = self.sVBoxValidationKitIso); def actionExecute(self): """ Execute the testcase. """ return self.oTestVmSet.actionExecute(self, self.testOneVmConfig) # # Test execution helpers. # def _generateRawPortFilename(self, oTestDrv, oTestVm, sInfix, sSuffix): """ Generates a raw port filename. """ random.seed(); sRandom = ''.join(random.choice(string.ascii_lowercase + string.digits) for _ in range(10)); return os.path.join(oTestDrv.sScratchPath, oTestVm.sVmName + sInfix + sRandom + sSuffix); def setupSerialMode(self, oSession, oTestVm, sMode): """ Sets up the serial mode. """ fRc = True; fServer = False; sLocation = None; ePortMode = vboxcon.PortMode_Disconnected; if sMode == 'RawFile': sLocation = self._generateRawPortFilename(self, oTestVm, '-com1-', '.out'); ePortMode = vboxcon.PortMode_RawFile; elif sMode == 'Tcp': sLocation = '127.0.0.1:1234'; self.oLoopback = loopback.SerialLoopback(loopback.g_ksLoopbackTcpServ, sLocation); ePortMode = vboxcon.PortMode_TCP; elif sMode == 'TcpServ': fServer = True; sLocation = '1234'; ePortMode = vboxcon.PortMode_TCP; self.oLoopback = loopback.SerialLoopback(loopback.g_ksLoopbackTcpClient, '127.0.0.1:1234'); elif sMode == 'NamedPipe': sLocation = self._generateRawPortFilename(self, oTestVm, '-com1-', '.out'); ePortMode = vboxcon.PortMode_HostPipe; self.oLoopback = loopback.SerialLoopback(loopback.g_ksLoopbackNamedPipeServ, sLocation); elif sMode == 'NamedPipeServ': fServer = True; sLocation = self._generateRawPortFilename(self, oTestVm, '-com1-', '.out'); ePortMode = vboxcon.PortMode_HostPipe; self.oLoopback = loopback.SerialLoopback(loopback.g_ksLoopbackNamedPipeClient, sLocation); elif sMode == 'HostDev': sLocation = '/dev/ttyUSB0'; ePortMode = vboxcon.PortMode_HostDevice; else: reporter.log('warning, invalid mode %s given' % (sMode, )); fRc = False; if fRc: fRc = oSession.changeSerialPortAttachment(0, ePortMode, sLocation, fServer); if fRc and sMode in ('TcpServ', 'NamedPipeServ',): self.sleep(2); # Fudge to allow the TCP server to get started. fRc = self.oLoopback.connect(); if not fRc: reporter.log('Failed to connect to %s' % (sLocation, )); self.sLocation = sLocation; return fRc; def testWrite(self, oSession, oTxsSession, oTestVm, sMode): """ Does a simple write test verifying the output. """ _ = oSession; reporter.testStart('Write'); tupCmdLine = ('SerialTest', '--tests', 'write', '--txbytes', '1048576'); if self.fVerboseTest: tupCmdLine += ('--verbose',); if oTestVm.isWindows(): tupCmdLine += ('--device', r'\\.\COM1',); elif oTestVm.isLinux(): tupCmdLine += ('--device', r'/dev/ttyS0',); fRc = self.txsRunTest(oTxsSession, 'SerialTest', 3600 * 1000, \ '${CDROM}/${OS/ARCH}/SerialTest${EXESUFF}', tupCmdLine); if not fRc: reporter.testFailure('Running serial test utility failed'); elif sMode == 'RawFile': # Open serial port and verify cLast = 0; try: with open(self.sLocation, 'rb') as oFile: sFmt = '=I'; cBytes = 4; for i in xrange(1048576 // 4): _ = i; sData = oFile.read(cBytes); tupUnpacked = struct.unpack(sFmt, sData); cLast = cLast + 1; if tupUnpacked[0] != cLast: reporter.testFailure('Corruption detected, expected counter value %s, got %s' % (cLast + 1, tupUnpacked[0],)); break; except: reporter.logXcpt(); reporter.testFailure('Verifying the written data failed'); reporter.testDone(); return fRc; def testReadWrite(self, oSession, oTxsSession, oTestVm): """ Does a simple write test verifying the output. """ _ = oSession; reporter.testStart('ReadWrite'); tupCmdLine = ('SerialTest', '--tests', 'readwrite', '--txbytes', '1048576'); if self.fVerboseTest: tupCmdLine += ('--verbose',); if oTestVm.isWindows(): tupCmdLine += ('--device', r'\\.\COM1',); elif oTestVm.isLinux(): tupCmdLine += ('--device', r'/dev/ttyS0',); fRc = self.txsRunTest(oTxsSession, 'SerialTest', 600 * 1000, \ '${CDROM}/${OS/ARCH}/SerialTest${EXESUFF}', tupCmdLine); if not fRc: reporter.testFailure('Running serial test utility failed'); reporter.testDone(); return fRc; def isModeCompatibleWithTest(self, sMode, sTest): """ Returns whether the given port mode and test combination is supported for testing. """ if sMode == 'RawFile' and sTest == 'ReadWrite': return False; if sMode != 'RawFile' and sTest == 'Write': return False; return True; def testOneVmConfig(self, oVM, oTestVm): """ Runs the specified VM thru test #1. """ for sUart in self.asUarts: reporter.testStart(sUart); # Reconfigure the VM fRc = True; oSession = self.openSession(oVM); if oSession is not None: fRc = oSession.enableSerialPort(0); fRc = fRc and oSession.setExtraData("VBoxInternal/Devices/serial/0/Config/UartType", "string:" + sUart); fRc = fRc and oSession.saveSettings(); fRc = oSession.close() and fRc; oSession = None; else: fRc = False; if fRc is True: self.logVmInfo(oVM); oSession, oTxsSession = self.startVmAndConnectToTxsViaTcp(oTestVm.sVmName, fCdWait = True); if oSession is not None: self.addTask(oTxsSession); for sMode in self.asSerialModes: reporter.testStart(sMode); fRc = self.setupSerialMode(oSession, oTestVm, sMode); if fRc: for sTest in self.asSerialTests: # Skip tests which don't work with the current mode. if self.isModeCompatibleWithTest(sMode, sTest): if sTest == 'Write': fRc = self.testWrite(oSession, oTxsSession, oTestVm, sMode); if sTest == 'ReadWrite': fRc = self.testReadWrite(oSession, oTxsSession, oTestVm); if self.oLoopback is not None: self.oLoopback.shutdown(); self.oLoopback = None; reporter.testDone(); self.removeTask(oTxsSession); self.terminateVmBySession(oSession); reporter.testDone(); return fRc; if __name__ == '__main__': sys.exit(tdSerial1().main(sys.argv));