| # Copyright 2016 Google Inc. |
| # |
| # Use of this source code is governed by a BSD-style license that can be |
| # found in the LICENSE file. |
| |
| from __future__ import print_function |
| from _hardware import Hardware |
| import sys |
| import time |
| |
| class HardwareAndroid(Hardware): |
| def __init__(self, adb): |
| Hardware.__init__(self) |
| self.warmup_time = 5 |
| self._adb = adb |
| self.desiredClock = 0.66 |
| |
| if self._adb.root(): |
| self._adb.remount() |
| |
| def __enter__(self): |
| Hardware.__enter__(self) |
| if not self._adb.is_root() and self._adb.root(): |
| self._adb.remount() |
| |
| self._adb.shell('\n'.join([ |
| # turn on airplane mode. |
| ''' |
| settings put global airplane_mode_on 1''', |
| |
| # disable GPS. |
| ''' |
| settings put secure location_providers_allowed -gps |
| settings put secure location_providers_allowed -wifi |
| settings put secure location_providers_allowed -network'''])) |
| |
| if self._adb.is_root(): |
| |
| # For explanation of variance reducing steps, see |
| # https://g3doc.corp.google.com/engedu/portal/android/g3doc/learn/develop/performance/content/best/reliable-startup-latency.md?cl=head |
| |
| self._adb.shell('\n'.join([ |
| # disable bluetooth, wifi, and mobile data. |
| ''' |
| service call bluetooth_manager 8 |
| svc wifi disable |
| svc data disable''', |
| |
| # kill the gui. |
| ''' |
| setprop ctl.stop media |
| setprop ctl.stop zygote |
| setprop ctl.stop surfaceflinger |
| setprop ctl.stop drm''', |
| |
| # disable ASLR |
| ''' |
| echo 0 > /proc/sys/kernel/randomize_va_space''', |
| ])) |
| |
| self.lock_top_three_cores() |
| |
| self.lock_adreno_gpu() |
| |
| else: |
| print("WARNING: no adb root access; results may be unreliable.", |
| file=sys.stderr) |
| |
| return self |
| |
| def __exit__(self, exception_type, exception_value, traceback): |
| Hardware.__exit__(self, exception_type, exception_value, traceback) |
| self._adb.reboot() # some devices struggle waking up; just hard reboot. |
| |
| def sanity_check(self): |
| Hardware.sanity_check(self) |
| |
| def print_debug_diagnostics(self): |
| # search for and print thermal trip points that may have been exceeded. |
| self._adb.shell('''\ |
| THERMALDIR=/sys/class/thermal |
| if [ ! -d $THERMALDIR ]; then |
| exit |
| fi |
| for ZONE in $(cd $THERMALDIR; echo thermal_zone*); do |
| cd $THERMALDIR/$ZONE |
| if [ ! -e mode ] || grep -Fxqv enabled mode || [ ! -e trip_point_0_temp ]; then |
| continue |
| fi |
| TEMP=$(cat temp) |
| TRIPPOINT=trip_point_0_temp |
| if [ $TEMP -le $(cat $TRIPPOINT) ]; then |
| echo "$ZONE ($(cat type)): temp=$TEMP <= $TRIPPOINT=$(cat $TRIPPOINT)" 1>&2 |
| else |
| let i=1 |
| while [ -e trip_point_${i}_temp ] && |
| [ $TEMP -gt $(cat trip_point_${i}_temp) ]; do |
| TRIPPOINT=trip_point_${i}_temp |
| let i=i+1 |
| done |
| echo "$ZONE ($(cat type)): temp=$TEMP > $TRIPPOINT=$(cat $TRIPPOINT)" 1>&2 |
| fi |
| done''') |
| |
| Hardware.print_debug_diagnostics(self) |
| |
| # expects a float between 0 and 100 representing where along the list of freqs to choose a value. |
| def setDesiredClock(self, c): |
| self.desiredClock = c / 100 |
| |
| def lock_top_three_cores(self): |
| # Lock the clocks of the fastest three cores and disable others. |
| # Assumes root privlidges |
| core_count = int(self._adb.check('cat /proc/cpuinfo | grep processor | wc -l')) |
| max_speeds = [] |
| for i in range(core_count): |
| khz = int(self._adb.check('cat /sys/devices/system/cpu/cpu%i/cpufreq/cpuinfo_max_freq' % i)) |
| max_speeds.append((khz, i)) # the tuple's first position and it will be the sort key |
| cores_in_desc_order_of_max_speed = [a[1] for a in sorted(max_speeds, reverse=True)] |
| top_cores = cores_in_desc_order_of_max_speed[:3] |
| disable_cores = cores_in_desc_order_of_max_speed[3:] |
| if disable_cores: |
| self._adb.shell('\n'.join([('echo 0 > /sys/devices/system/cpu/cpu%i/online' % i) for i in disable_cores])) |
| # since thermal-engine will be disabled, don't pick the max freq to lock these at, |
| # pick something lower, so it doesn't get too hot (it'd reboot) |
| # get a list of available scaling frequencies and pick one 2/3 of the way up. |
| for i in top_cores: |
| freqs = self._adb.check('cat /sys/devices/system/cpu/cpu%i/cpufreq/scaling_available_frequencies' % i).split() |
| speed = freqs[int((len(freqs)-1) * self.desiredClock)] |
| self._adb.shell('''echo 1 > /sys/devices/system/cpu/cpu{id}/online |
| echo userspace > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_governor |
| echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_max_freq |
| echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_min_freq |
| echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_setspeed'''.format(id=i, speed=speed)) |
| |
| def lock_adreno_gpu(self): |
| # Use presence of /sys/class/kgsl to indicate Adreno GPU |
| exists = self._adb.check('test -d /sys/class/kgsl && echo y') |
| if (exists.strip() != 'y'): |
| print('Not attempting Adreno GPU clock locking steps') |
| return |
| |
| # variance reducing changes |
| self._adb.shell(''' |
| echo 0 > /sys/class/kgsl/kgsl-3d0/bus_split |
| echo 1 > /sys/class/kgsl/kgsl-3d0/force_clk_on |
| echo 10000 > /sys/class/kgsl/kgsl-3d0/idle_timer''') |
| |
| freqs = self._adb.check('cat /sys/class/kgsl/kgsl-3d0/devfreq/available_frequencies').split() |
| speed = freqs[int((len(freqs)-1) * self.desiredClock)] |
| |
| # Set GPU to performance mode and lock clock |
| self._adb.shell(''' |
| echo performance > /sys/class/kgsl/kgsl-3d0/devfreq/governor |
| echo {speed} > /sys/class/kgsl/kgsl-3d0/devfreq/max_freq |
| echo {speed} > /sys/class/kgsl/kgsl-3d0/devfreq/min_freq'''.format(speed=speed)) |
| |
| # Set GPU power level |
| self._adb.shell(''' |
| echo 1 > /sys/class/kgsl/kgsl-3d0/max_pwrlevel |
| echo 1 > /sys/class/kgsl/kgsl-3d0/min_pwrlevel''') |