Index: infra/bots/recipe_modules/skia/android_flavor.py |
diff --git a/infra/bots/recipe_modules/skia/android_flavor.py b/infra/bots/recipe_modules/skia/android_flavor.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..442604e74e97419eb836af7152067fb04febc3f5 |
--- /dev/null |
+++ b/infra/bots/recipe_modules/skia/android_flavor.py |
@@ -0,0 +1,316 @@ |
+# Copyright 2014 The Chromium Authors. All rights reserved. |
+# Use of this source code is governed by a BSD-style license that can be |
+# found in the LICENSE file. |
+ |
+ |
+# pylint: disable=W0201 |
+ |
+ |
+import android_devices |
+import copy |
+import default_flavor |
+ |
+ |
+"""Android flavor utils, used for building for and running tests on Android.""" |
+ |
+ |
+class _ADBWrapper(object): |
+ """Wrapper for the ADB recipe module. |
+ |
+ The ADB recipe module looks for the ADB binary at a path we don't have checked |
+ out on our bots. This wrapper ensures that we set a custom ADB path before |
+ attempting to use the module. |
+ """ |
+ def __init__(self, adb_api, path_to_adb, serial_args, android_flavor): |
+ self._adb = adb_api |
+ self._adb.set_adb_path(path_to_adb) |
+ self._has_root = False # This is set in install(). |
+ self._serial_args = serial_args |
+ self._wait_count = 0 |
+ self._android_flavor = android_flavor |
+ |
+ def wait_for_device(self): |
+ """Run 'adb wait-for-device'.""" |
+ self._wait_count += 1 |
+ cmd = [ |
+ self._android_flavor.android_bin.join('adb_wait_for_device') |
+ ] + self._serial_args |
+ self._android_flavor._skia_api.run( |
+ self._android_flavor._skia_api.m.step, |
+ name='wait for device (%d)' % self._wait_count, |
+ cmd=cmd, |
+ env=self._android_flavor._default_env, |
+ infra_step=True) |
+ |
+ cmd = [ |
+ self._android_flavor.android_bin.join('adb_wait_for_charge'), |
+ ] + self._serial_args |
+ self._android_flavor._skia_api.run( |
+ self._android_flavor._skia_api.m.step, |
+ name='wait for charge (%d)' % self._wait_count, |
+ cmd=cmd, |
+ env=self._android_flavor._default_env, |
+ infra_step=True) |
+ |
+ def maybe_wait_for_device(self): |
+ """Run 'adb wait-for-device' if it hasn't already been run.""" |
+ if self._wait_count == 0: |
+ self.wait_for_device() |
+ |
+ def __call__(self, *args, **kwargs): |
+ self.maybe_wait_for_device() |
+ return self._android_flavor._skia_api.run(self._adb, *args, **kwargs) |
+ |
+ |
+class AndroidFlavorUtils(default_flavor.DefaultFlavorUtils): |
+ def __init__(self, skia_api): |
+ super(AndroidFlavorUtils, self).__init__(skia_api) |
+ self.device = self._skia_api.builder_spec['device_cfg'] |
+ self.android_bin = self._skia_api.skia_dir.join( |
+ 'platform_tools', 'android', 'bin') |
+ self._android_sdk_root = self._skia_api.slave_dir.join( |
+ 'android_sdk', 'android-sdk') |
+ self.serial = None |
+ self.serial_args = [] |
+ try: |
+ path_to_adb = self._skia_api.m.step( |
+ 'which adb', |
+ cmd=['which', 'adb'], |
+ stdout=self._skia_api.m.raw_io.output(), |
+ infra_step=True).stdout.rstrip() |
+ except self._skia_api.m.step.StepFailure: |
+ path_to_adb = self._skia_api.m.path.join(self._android_sdk_root, |
+ 'platform-tools', 'adb') |
+ self._adb = _ADBWrapper( |
+ self._skia_api.m.adb, path_to_adb, self.serial_args, self) |
+ self._default_env = {'ANDROID_SDK_ROOT': self._android_sdk_root, |
+ 'ANDROID_HOME': self._android_sdk_root, |
+ 'SKIA_ANDROID_VERBOSE_SETUP': 1} |
+ |
+ def step(self, name, cmd, env=None, **kwargs): |
+ self._adb.maybe_wait_for_device() |
+ args = [ |
+ self.android_bin.join('android_run_skia'), |
+ '--verbose', |
+ '--logcat', |
+ '-d', self.device, |
+ ] + self.serial_args + [ |
+ '-t', self._skia_api.configuration, |
+ ] |
+ env = dict(env or {}) |
+ env.update(self._default_env) |
+ |
+ return self._skia_api.run(self._skia_api.m.step, name=name, cmd=args + cmd, |
+ env=env, **kwargs) |
+ |
+ def compile(self, target): |
+ """Build the given target.""" |
+ env = dict(self._default_env) |
+ ccache = self._skia_api.ccache() |
+ if ccache: |
+ env['ANDROID_MAKE_CCACHE'] = ccache |
+ |
+ cmd = [self.android_bin.join('android_ninja'), target, '-d', self.device] |
+ if 'Clang' in self._skia_api.builder_name: |
+ cmd.append('--clang') |
+ if 'GCC' in self._skia_api.builder_name: |
+ cmd.append('--gcc') |
+ if 'Vulkan' in self._skia_api.builder_name: |
+ cmd.append('--vulkan') |
+ self._skia_api.run(self._skia_api.m.step, 'build %s' % target, cmd=cmd, |
+ env=env, cwd=self._skia_api.m.path['checkout']) |
+ |
+ def device_path_join(self, *args): |
+ """Like os.path.join(), but for paths on a connected Android device.""" |
+ return '/'.join(args) |
+ |
+ def device_path_exists(self, path): |
+ """Like os.path.exists(), but for paths on a connected device.""" |
+ exists_str = 'FILE_EXISTS' |
+ return exists_str in self._adb( |
+ name='exists %s' % self._skia_api.m.path.basename(path), |
+ serial=self.serial, |
+ cmd=['shell', 'if', '[', '-e', path, '];', |
+ 'then', 'echo', exists_str + ';', 'fi'], |
+ stdout=self._skia_api.m.raw_io.output(), |
+ infra_step=True |
+ ).stdout |
+ |
+ def _remove_device_dir(self, path): |
+ """Remove the directory on the device.""" |
+ self._adb(name='rmdir %s' % self._skia_api.m.path.basename(path), |
+ serial=self.serial, |
+ cmd=['shell', 'rm', '-r', path], |
+ infra_step=True) |
+ # Sometimes the removal fails silently. Verify that it worked. |
+ if self.device_path_exists(path): |
+ raise Exception('Failed to remove %s!' % path) # pragma: no cover |
+ |
+ def _create_device_dir(self, path): |
+ """Create the directory on the device.""" |
+ self._adb(name='mkdir %s' % self._skia_api.m.path.basename(path), |
+ serial=self.serial, |
+ cmd=['shell', 'mkdir', '-p', path], |
+ infra_step=True) |
+ |
+ def copy_directory_contents_to_device(self, host_dir, device_dir): |
+ """Like shutil.copytree(), but for copying to a connected device.""" |
+ self._skia_api.run( |
+ self._skia_api.m.step, |
+ name='push %s' % self._skia_api.m.path.basename(host_dir), |
+ cmd=[ |
+ self.android_bin.join('adb_push_if_needed'), '--verbose', |
+ ] + self.serial_args + [ |
+ host_dir, device_dir, |
+ ], |
+ env=self._default_env, |
+ infra_step=True) |
+ |
+ def copy_directory_contents_to_host(self, device_dir, host_dir): |
+ """Like shutil.copytree(), but for copying from a connected device.""" |
+ self._skia_api.run( |
+ self._skia_api.m.step, |
+ name='pull %s' % self._skia_api.m.path.basename(device_dir), |
+ cmd=[ |
+ self.android_bin.join('adb_pull_if_needed'), '--verbose', |
+ ] + self.serial_args + [ |
+ device_dir, host_dir, |
+ ], |
+ env=self._default_env, |
+ infra_step=True) |
+ |
+ def copy_file_to_device(self, host_path, device_path): |
+ """Like shutil.copyfile, but for copying to a connected device.""" |
+ self._adb(name='push %s' % self._skia_api.m.path.basename(host_path), |
+ serial=self.serial, |
+ cmd=['push', host_path, device_path], |
+ infra_step=True) |
+ |
+ def create_clean_device_dir(self, path): |
+ """Like shutil.rmtree() + os.makedirs(), but on a connected device.""" |
+ self._remove_device_dir(path) |
+ self._create_device_dir(path) |
+ |
+ def has_root(self): |
+ """Determine if we have root access on this device.""" |
+ # Special case: GalaxyS3 hangs on `adb root`. Don't bother. |
+ if 'GalaxyS3' in self._skia_api.builder_name: |
+ return False |
+ |
+ # Determine if we have root access. |
+ has_root = False |
+ try: |
+ output = self._adb(name='adb root', |
+ serial=self.serial, |
+ cmd=['root'], |
+ stdout=self._skia_api.m.raw_io.output(), |
+ infra_step=True).stdout.rstrip() |
+ if ('restarting adbd as root' in output or |
+ 'adbd is already running as root' in output): |
+ has_root = True |
+ except self._skia_api.m.step.StepFailure: # pragma: nocover |
+ pass |
+ # Wait for the device to reconnect. |
+ self._skia_api.run( |
+ self._skia_api.m.step, |
+ name='wait', |
+ cmd=['sleep', '10'], |
+ infra_step=True) |
+ self._adb.wait_for_device() |
+ return has_root |
+ |
+ def install(self): |
+ """Run device-specific installation steps.""" |
+ self._has_root = self.has_root() |
+ self._skia_api.run(self._skia_api.m.step, |
+ name='kill skia', |
+ cmd=[ |
+ self.android_bin.join('android_kill_skia'), |
+ '--verbose', |
+ ] + self.serial_args, |
+ env=self._default_env, |
+ infra_step=True) |
+ if self._has_root: |
+ self._adb(name='stop shell', |
+ serial=self.serial, |
+ cmd=['shell', 'stop'], |
+ infra_step=True) |
+ |
+ # Print out battery stats. |
+ self._adb(name='starting battery stats', |
+ serial=self.serial, |
+ cmd=['shell', 'dumpsys', 'batteryproperties'], |
+ infra_step=True) |
+ |
+ # Print out CPU scale info. |
+ if self._has_root: |
+ self._adb(name='cat scaling_governor', |
+ serial=self.serial, |
+ cmd=['shell', 'cat', |
+ '/sys/devices/system/cpu/cpu0/cpufreq/scaling_governor'], |
+ infra_step=True) |
+ self._adb(name='cat cpu_freq', |
+ serial=self.serial, |
+ cmd=['shell', 'cat', |
+ '/sys/devices/system/cpu/cpu0/cpufreq/scaling_cur_freq'], |
+ infra_step=True) |
+ |
+ def cleanup_steps(self): |
+ """Run any device-specific cleanup steps.""" |
+ if self._skia_api.do_test_steps or self._skia_api.do_perf_steps: |
+ self._adb(name='final battery stats', |
+ serial=self.serial, |
+ cmd=['shell', 'dumpsys', 'batteryproperties'], |
+ infra_step=True) |
+ self._adb(name='reboot', |
+ serial=self.serial, |
+ cmd=['reboot'], |
+ infra_step=True) |
+ self._skia_api.run( |
+ self._skia_api.m.step, |
+ name='wait for reboot', |
+ cmd=['sleep', '10'], |
+ infra_step=True) |
+ self._adb.wait_for_device() |
+ # The ADB binary conflicts with py-adb used by swarming. Kill it |
+ # when finished to play nice. |
+ self._adb(name='kill-server', |
+ serial=self.serial, |
+ cmd=['kill-server'], |
+ infra_step=True) |
+ |
+ def read_file_on_device(self, path, *args, **kwargs): |
+ """Read the given file.""" |
+ return self._adb(name='read %s' % self._skia_api.m.path.basename(path), |
+ serial=self.serial, |
+ cmd=['shell', 'cat', path], |
+ stdout=self._skia_api.m.raw_io.output(), |
+ infra_step=True).stdout.rstrip() |
+ |
+ def remove_file_on_device(self, path, *args, **kwargs): |
+ """Delete the given file.""" |
+ return self._adb(name='rm %s' % self._skia_api.m.path.basename(path), |
+ serial=self.serial, |
+ cmd=['shell', 'rm', '-f', path], |
+ infra_step=True, |
+ *args, |
+ **kwargs) |
+ |
+ def get_device_dirs(self): |
+ """ Set the directories which will be used by the build steps.""" |
+ device_scratch_dir = self._adb( |
+ name='get EXTERNAL_STORAGE dir', |
+ serial=self.serial, |
+ cmd=['shell', 'echo', '$EXTERNAL_STORAGE'], |
+ stdout=self._skia_api.m.raw_io.output(), |
+ infra_step=True, |
+ ).stdout.rstrip() |
+ prefix = self.device_path_join(device_scratch_dir, 'skiabot', 'skia_') |
+ return default_flavor.DeviceDirs( |
+ dm_dir=prefix + 'dm', |
+ perf_data_dir=prefix + 'perf', |
+ resource_dir=prefix + 'resources', |
+ images_dir=prefix + 'images', |
+ skp_dir=prefix + 'skp/skps', |
+ tmp_dir=prefix + 'tmp_dir') |
+ |