# Copyright (c) 2013~2015 The Chromium OS Authors. All rights reserved. # Use of this source code is governed by a BSD-style license that can be # found in the LICENSE file. """SuiteRunner defines the interface from crosperf to test script.""" from __future__ import print_function import os import time import shlex from cros_utils import command_executer import test_flag TEST_THAT_PATH = '/usr/bin/test_that' AUTOTEST_DIR = '~/trunk/src/third_party/autotest/files' CHROME_MOUNT_DIR = '/tmp/chrome_root' def GetProfilerArgs(profiler_args): # Remove "--" from in front of profiler args. args_list = shlex.split(profiler_args) new_list = [] for arg in args_list: if arg[0:2] == '--': arg = arg[2:] new_list.append(arg) args_list = new_list # Remove "perf_options=" from middle of profiler args. new_list = [] for arg in args_list: idx = arg.find('perf_options=') if idx != -1: prefix = arg[0:idx] suffix = arg[idx + len('perf_options=') + 1:-1] new_arg = prefix + "'" + suffix + "'" new_list.append(new_arg) else: new_list.append(arg) args_list = new_list return ' '.join(args_list) class SuiteRunner(object): """This defines the interface from crosperf to test script.""" def __init__(self, logger_to_use=None, log_level='verbose', cmd_exec=None, cmd_term=None): self.logger = logger_to_use self.log_level = log_level self._ce = cmd_exec or command_executer.GetCommandExecuter( self.logger, log_level=self.log_level) self._ct = cmd_term or command_executer.CommandTerminator() def Run(self, machine, label, benchmark, test_args, profiler_args): for i in range(0, benchmark.retries + 1): self.PinGovernorExecutionFrequencies(machine, label.chromeos_root) if benchmark.suite == 'telemetry': self.DecreaseWaitTime(machine, label.chromeos_root) ret_tup = self.Telemetry_Run(machine, label, benchmark, profiler_args) elif benchmark.suite == 'telemetry_Crosperf': self.DecreaseWaitTime(machine, label.chromeos_root) ret_tup = self.Telemetry_Crosperf_Run(machine, label, benchmark, test_args, profiler_args) else: ret_tup = self.Test_That_Run(machine, label, benchmark, test_args, profiler_args) if ret_tup[0] != 0: self.logger.LogOutput('benchmark %s failed. Retries left: %s' % (benchmark.name, benchmark.retries - i)) elif i > 0: self.logger.LogOutput('benchmark %s succeded after %s retries' % (benchmark.name, i)) break else: self.logger.LogOutput('benchmark %s succeded on first try' % benchmark.name) break return ret_tup def PinGovernorExecutionFrequencies(self, machine_name, chromeos_root): """Set min and max frequencies to max static frequency.""" # pyformat: disable set_cpu_freq = ( 'set -e && ' 'for f in /sys/devices/system/cpu/cpu*/cpufreq; do ' 'cd $f; ' 'val=0; ' 'if [[ -e scaling_available_frequencies ]]; then ' # pylint: disable=line-too-long ' val=`cat scaling_available_frequencies | tr " " "\\n" | sort -n -b -r`; ' 'else ' ' val=`cat scaling_max_freq | tr " " "\\n" | sort -n -b -r`; fi ; ' 'set -- $val; ' 'highest=$1; ' 'if [[ $# -gt 1 ]]; then ' ' case $highest in *1000) highest=$2;; esac; ' 'fi ;' 'echo $highest > scaling_max_freq; ' 'echo $highest > scaling_min_freq; ' 'echo performance > scaling_governor; ' 'done' ) # pyformat: enable if self.log_level == 'average': self.logger.LogOutput('Pinning governor execution frequencies for %s' % machine_name) ret = self._ce.CrosRunCommand( set_cpu_freq, machine=machine_name, chromeos_root=chromeos_root) self.logger.LogFatalIf(ret, 'Could not pin frequencies on machine: %s' % machine_name) def DecreaseWaitTime(self, machine_name, chromeos_root): """Change the ten seconds wait time for pagecycler to two seconds.""" FILE = '/usr/local/telemetry/src/tools/perf/page_sets/page_cycler_story.py' ret = self._ce.CrosRunCommand( 'ls ' + FILE, machine=machine_name, chromeos_root=chromeos_root) self.logger.LogFatalIf(ret, 'Could not find {} on machine: {}'.format( FILE, machine_name)) if not ret: sed_command = 'sed -i "s/_TTI_WAIT_TIME = 10/_TTI_WAIT_TIME = 2/g" ' ret = self._ce.CrosRunCommand( sed_command + FILE, machine=machine_name, chromeos_root=chromeos_root) self.logger.LogFatalIf(ret, 'Could not modify {} on machine: {}'.format( FILE, machine_name)) def RebootMachine(self, machine_name, chromeos_root): command = 'reboot && exit' self._ce.CrosRunCommand( command, machine=machine_name, chromeos_root=chromeos_root) time.sleep(60) # Whenever we reboot the machine, we need to restore the governor settings. self.PinGovernorExecutionFrequencies(machine_name, chromeos_root) def Test_That_Run(self, machine, label, benchmark, test_args, profiler_args): """Run the test_that test..""" options = '' if label.board: options += ' --board=%s' % label.board if test_args: options += ' %s' % test_args if profiler_args: self.logger.LogFatal('test_that does not support profiler.') command = 'rm -rf /usr/local/autotest/results/*' self._ce.CrosRunCommand( command, machine=machine, chromeos_root=label.chromeos_root) # We do this because some tests leave the machine in weird states. # Rebooting between iterations has proven to help with this. self.RebootMachine(machine, label.chromeos_root) autotest_dir = AUTOTEST_DIR if label.autotest_path != '': autotest_dir = label.autotest_path autotest_dir_arg = '--autotest_dir %s' % autotest_dir # For non-telemetry tests, specify an autotest directory only if the # specified directory is different from default (crosbug.com/679001). if autotest_dir == AUTOTEST_DIR: autotest_dir_arg = '' command = (('%s %s --fast ' '%s %s %s') % (TEST_THAT_PATH, autotest_dir_arg, options, machine, benchmark.test_name)) if self.log_level != 'verbose': self.logger.LogOutput('Running test.') self.logger.LogOutput('CMD: %s' % command) # Use --no-ns-pid so that cros_sdk does not create a different # process namespace and we can kill process created easily by # their process group. return self._ce.ChrootRunCommandWOutput( label.chromeos_root, command, command_terminator=self._ct, cros_sdk_options='--no-ns-pid') def RemoveTelemetryTempFile(self, machine, chromeos_root): filename = 'telemetry@%s' % machine fullname = os.path.join(chromeos_root, 'chroot', 'tmp', filename) if os.path.exists(fullname): os.remove(fullname) def Telemetry_Crosperf_Run(self, machine, label, benchmark, test_args, profiler_args): if not os.path.isdir(label.chrome_src): self.logger.LogFatal('Cannot find chrome src dir to' ' run telemetry: %s' % label.chrome_src) # Check for and remove temporary file that may have been left by # previous telemetry runs (and which might prevent this run from # working). self.RemoveTelemetryTempFile(machine, label.chromeos_root) # For telemetry runs, we can use the autotest copy from the source # location. No need to have one under /build/<board>. autotest_dir_arg = '--autotest_dir %s' % AUTOTEST_DIR if label.autotest_path != '': autotest_dir_arg = '--autotest_dir %s' % label.autotest_path profiler_args = GetProfilerArgs(profiler_args) fast_arg = '' if not profiler_args: # --fast works unless we are doing profiling (autotest limitation). # --fast avoids unnecessary copies of syslogs. fast_arg = '--fast' args_string = '' if test_args: # Strip double quotes off args (so we can wrap them in single # quotes, to pass through to Telemetry). if test_args[0] == '"' and test_args[-1] == '"': test_args = test_args[1:-1] args_string = "test_args='%s'" % test_args cmd = ('{} {} {} --board={} --args="{} run_local={} test={} ' '{}" {} telemetry_Crosperf'.format(TEST_THAT_PATH, autotest_dir_arg, fast_arg, label.board, args_string, benchmark.run_local, benchmark.test_name, profiler_args, machine)) # Use --no-ns-pid so that cros_sdk does not create a different # process namespace and we can kill process created easily by their # process group. chrome_root_options = ('--no-ns-pid ' '--chrome_root={} --chrome_root_mount={} ' "FEATURES=\"-usersandbox\" " 'CHROME_ROOT={}'.format(label.chrome_src, CHROME_MOUNT_DIR, CHROME_MOUNT_DIR)) if self.log_level != 'verbose': self.logger.LogOutput('Running test.') self.logger.LogOutput('CMD: %s' % cmd) return self._ce.ChrootRunCommandWOutput( label.chromeos_root, cmd, command_terminator=self._ct, cros_sdk_options=chrome_root_options) def Telemetry_Run(self, machine, label, benchmark, profiler_args): telemetry_run_path = '' if not os.path.isdir(label.chrome_src): self.logger.LogFatal('Cannot find chrome src dir to' ' run telemetry.') else: telemetry_run_path = os.path.join(label.chrome_src, 'src/tools/perf') if not os.path.exists(telemetry_run_path): self.logger.LogFatal('Cannot find %s directory.' % telemetry_run_path) if profiler_args: self.logger.LogFatal('Telemetry does not support the perf profiler.') # Check for and remove temporary file that may have been left by # previous telemetry runs (and which might prevent this run from # working). if not test_flag.GetTestMode(): self.RemoveTelemetryTempFile(machine, label.chromeos_root) rsa_key = os.path.join( label.chromeos_root, 'src/scripts/mod_for_test_scripts/ssh_keys/testing_rsa') cmd = ('cd {0} && ' './run_measurement ' '--browser=cros-chrome ' '--output-format=csv ' '--remote={1} ' '--identity {2} ' '{3} {4}'.format(telemetry_run_path, machine, rsa_key, benchmark.test_name, benchmark.test_args)) if self.log_level != 'verbose': self.logger.LogOutput('Running test.') self.logger.LogOutput('CMD: %s' % cmd) return self._ce.RunCommandWOutput(cmd, print_to_console=False) def CommandTerminator(self): return self._ct def Terminate(self): self._ct.Terminate() class MockSuiteRunner(object): """Mock suite runner for test.""" def __init__(self): self._true = True def Run(self, *_args): if self._true: return [0, '', ''] else: return [0, '', '']