blob: 1fae63a9cc2c550a6390d9f0f06ccec123b76949 [file] [log] [blame]
csmartdaltond7a9db62016-09-22 05:10:02 -07001# Copyright 2016 Google Inc.
2#
3# Use of this source code is governed by a BSD-style license that can be
4# found in the LICENSE file.
5
6from __future__ import print_function
7from _hardware import Hardware
8import sys
9import time
10
11class HardwareAndroid(Hardware):
12 def __init__(self, adb):
13 Hardware.__init__(self)
csmartdalton5772eaa2016-10-11 18:28:54 -070014 self.warmup_time = 5
csmartdaltond7a9db62016-09-22 05:10:02 -070015 self._adb = adb
Nathaniel Nifongba4c3082021-02-09 08:58:00 -050016 self.desiredClock = 0.66
csmartdaltone4fd0782016-11-09 08:41:23 -080017
18 if self._adb.root():
csmartdaltond7a9db62016-09-22 05:10:02 -070019 self._adb.remount()
csmartdaltone4fd0782016-11-09 08:41:23 -080020
21 def __enter__(self):
Chris Dalton34d90552017-10-20 09:58:32 -060022 Hardware.__enter__(self)
23 if not self._adb.is_root() and self._adb.root():
24 self._adb.remount()
25
csmartdaltone4fd0782016-11-09 08:41:23 -080026 self._adb.shell('\n'.join([
27 # turn on airplane mode.
28 '''
29 settings put global airplane_mode_on 1''',
30
31 # disable GPS.
32 '''
Chris Dalton117d9722018-04-27 17:10:39 -060033 settings put secure location_providers_allowed -gps
34 settings put secure location_providers_allowed -wifi
35 settings put secure location_providers_allowed -network''']))
csmartdaltone4fd0782016-11-09 08:41:23 -080036
37 if self._adb.is_root():
Nathaniel Nifong9109dc12021-02-02 16:27:44 -050038
Nathaniel Nifongba4c3082021-02-09 08:58:00 -050039 # For explanation of variance reducing steps, see
40 # https://g3doc.corp.google.com/engedu/portal/android/g3doc/learn/develop/performance/content/best/reliable-startup-latency.md?cl=head
41
csmartdaltone4fd0782016-11-09 08:41:23 -080042 self._adb.shell('\n'.join([
43 # disable bluetooth, wifi, and mobile data.
44 '''
45 service call bluetooth_manager 8
46 svc wifi disable
47 svc data disable''',
48
49 # kill the gui.
50 '''
51 setprop ctl.stop media
52 setprop ctl.stop zygote
53 setprop ctl.stop surfaceflinger
54 setprop ctl.stop drm''',
55
56 # disable ASLR
57 '''
Nathaniel Nifong9109dc12021-02-02 16:27:44 -050058 echo 0 > /proc/sys/kernel/randomize_va_space''',
Nathaniel Nifongba4c3082021-02-09 08:58:00 -050059 ]))
Nathaniel Nifong9109dc12021-02-02 16:27:44 -050060
61 self.lock_top_three_cores()
62
Nathaniel Nifongba4c3082021-02-09 08:58:00 -050063 self.lock_adreno_gpu()
64
csmartdaltond7a9db62016-09-22 05:10:02 -070065 else:
66 print("WARNING: no adb root access; results may be unreliable.",
67 file=sys.stderr)
68
Chris Dalton34d90552017-10-20 09:58:32 -060069 return self
csmartdaltond7a9db62016-09-22 05:10:02 -070070
71 def __exit__(self, exception_type, exception_value, traceback):
72 Hardware.__exit__(self, exception_type, exception_value, traceback)
Chris Dalton49b7ed32017-10-23 17:19:37 -060073 self._adb.reboot() # some devices struggle waking up; just hard reboot.
csmartdaltond7a9db62016-09-22 05:10:02 -070074
75 def sanity_check(self):
76 Hardware.sanity_check(self)
77
csmartdalton2a961012016-10-11 12:15:13 -070078 def print_debug_diagnostics(self):
79 # search for and print thermal trip points that may have been exceeded.
80 self._adb.shell('''\
81 THERMALDIR=/sys/class/thermal
csmartdalton310d72c2016-10-18 09:19:50 -070082 if [ ! -d $THERMALDIR ]; then
83 exit
84 fi
85 for ZONE in $(cd $THERMALDIR; echo thermal_zone*); do
86 cd $THERMALDIR/$ZONE
87 if [ ! -e mode ] || grep -Fxqv enabled mode || [ ! -e trip_point_0_temp ]; then
88 continue
89 fi
90 TEMP=$(cat temp)
91 TRIPPOINT=trip_point_0_temp
92 if [ $TEMP -le $(cat $TRIPPOINT) ]; then
93 echo "$ZONE ($(cat type)): temp=$TEMP <= $TRIPPOINT=$(cat $TRIPPOINT)" 1>&2
94 else
95 let i=1
96 while [ -e trip_point_${i}_temp ] &&
97 [ $TEMP -gt $(cat trip_point_${i}_temp) ]; do
98 TRIPPOINT=trip_point_${i}_temp
99 let i=i+1
100 done
101 echo "$ZONE ($(cat type)): temp=$TEMP > $TRIPPOINT=$(cat $TRIPPOINT)" 1>&2
102 fi
103 done''')
csmartdalton2a961012016-10-11 12:15:13 -0700104
105 Hardware.print_debug_diagnostics(self)
Nathaniel Nifong9109dc12021-02-02 16:27:44 -0500106
Nathaniel Nifongba4c3082021-02-09 08:58:00 -0500107 # expects a float between 0 and 100 representing where along the list of freqs to choose a value.
108 def setDesiredClock(self, c):
109 self.desiredClock = c / 100
110
Nathaniel Nifong9109dc12021-02-02 16:27:44 -0500111 def lock_top_three_cores(self):
112 # Lock the clocks of the fastest three cores and disable others.
113 # Assumes root privlidges
114 core_count = int(self._adb.check('cat /proc/cpuinfo | grep processor | wc -l'))
115 max_speeds = []
116 for i in range(core_count):
117 khz = int(self._adb.check('cat /sys/devices/system/cpu/cpu%i/cpufreq/cpuinfo_max_freq' % i))
118 max_speeds.append((khz, i)) # the tuple's first position and it will be the sort key
119 cores_in_desc_order_of_max_speed = [a[1] for a in sorted(max_speeds, reverse=True)]
120 top_cores = cores_in_desc_order_of_max_speed[:3]
121 disable_cores = cores_in_desc_order_of_max_speed[3:]
122 if disable_cores:
123 self._adb.shell('\n'.join([('echo 0 > /sys/devices/system/cpu/cpu%i/online' % i) for i in disable_cores]))
124 # since thermal-engine will be disabled, don't pick the max freq to lock these at,
125 # pick something lower, so it doesn't get too hot (it'd reboot)
126 # get a list of available scaling frequencies and pick one 2/3 of the way up.
127 for i in top_cores:
128 freqs = self._adb.check('cat /sys/devices/system/cpu/cpu%i/cpufreq/scaling_available_frequencies' % i).split()
Nathaniel Nifongba4c3082021-02-09 08:58:00 -0500129 speed = freqs[int((len(freqs)-1) * self.desiredClock)]
Nathaniel Nifong9109dc12021-02-02 16:27:44 -0500130 self._adb.shell('''echo 1 > /sys/devices/system/cpu/cpu{id}/online
131 echo userspace > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_governor
132 echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_max_freq
133 echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_min_freq
134 echo {speed} > /sys/devices/system/cpu/cpu{id}/cpufreq/scaling_setspeed'''.format(id=i, speed=speed))
Nathaniel Nifongba4c3082021-02-09 08:58:00 -0500135
136 def lock_adreno_gpu(self):
137 # Use presence of /sys/class/kgsl to indicate Adreno GPU
138 exists = self._adb.check('test -d /sys/class/kgsl && echo y')
139 if (exists.strip() != 'y'):
140 print('Not attempting Adreno GPU clock locking steps')
141 return
142
143 # variance reducing changes
144 self._adb.shell('''
145 echo 0 > /sys/class/kgsl/kgsl-3d0/bus_split
146 echo 1 > /sys/class/kgsl/kgsl-3d0/force_clk_on
147 echo 10000 > /sys/class/kgsl/kgsl-3d0/idle_timer''')
148
149 freqs = self._adb.check('cat /sys/class/kgsl/kgsl-3d0/devfreq/available_frequencies').split()
150 speed = freqs[int((len(freqs)-1) * self.desiredClock)]
151
152 # Set GPU to performance mode and lock clock
153 self._adb.shell('''
154 echo performance > /sys/class/kgsl/kgsl-3d0/devfreq/governor
155 echo {speed} > /sys/class/kgsl/kgsl-3d0/devfreq/max_freq
156 echo {speed} > /sys/class/kgsl/kgsl-3d0/devfreq/min_freq'''.format(speed=speed))
157
158 # Set GPU power level
159 self._adb.shell('''
160 echo 1 > /sys/class/kgsl/kgsl-3d0/max_pwrlevel
161 echo 1 > /sys/class/kgsl/kgsl-3d0/min_pwrlevel''')