blob: 4cdf21ac8c72a52ac5e22b07e741f5536a510f10 [file] [log] [blame]
# Copyright 2017 Google Inc.
#
# Use of this source code is governed by a BSD-style license that can be
# found in the LICENSE file.
from _hardware import HardwareException, Expectation
from _hardware_android import HardwareAndroid
from collections import namedtuple
import itertools
CPU_CLOCK_RATE = 1670400
GPU_CLOCK_RATE = 510000000
DEVFREQ_DIRNAME = '/sys/class/devfreq'
DEVFREQ_THROTTLE = 0.74
DEVFREQ_BLACKLIST = ('b00000.qcom,kgsl-3d0', 'soc:qcom,kgsl-busmon',
'soc:qcom,m4m')
DevfreqKnobs = namedtuple('knobs',
('available_governors', 'available_frequencies',
'governor', 'min_freq', 'max_freq', 'cur_freq'))
class HardwarePixel(HardwareAndroid):
def __init__(self, adb):
HardwareAndroid.__init__(self, adb)
self._discover_devfreqs()
def __enter__(self):
HardwareAndroid.__enter__(self)
if not self._adb.is_root():
return self
self._adb.shell('\n'.join(['''\
stop thermal-engine
stop thermald
stop perfd
stop mpdecision''',
# enable and lock the two fast cores.
'''
for N in 3 2; do
echo 1 > /sys/devices/system/cpu/cpu$N/online
echo userspace > /sys/devices/system/cpu/cpu$N/cpufreq/scaling_governor
echo %i > /sys/devices/system/cpu/cpu$N/cpufreq/scaling_max_freq
echo %i > /sys/devices/system/cpu/cpu$N/cpufreq/scaling_min_freq
echo %i > /sys/devices/system/cpu/cpu$N/cpufreq/scaling_setspeed
done''' % tuple(CPU_CLOCK_RATE for _ in range(3)),
# turn off the two slow cores
'''
for N in 1 0; do
echo 0 > /sys/devices/system/cpu/cpu$N/online
done''',
# gpu perf commands from
# https://developer.qualcomm.com/qfile/28823/lm80-p0436-11_adb_commands.pdf
'''
echo 0 > /sys/class/kgsl/kgsl-3d0/bus_split
echo 1 > /sys/class/kgsl/kgsl-3d0/force_bus_on
echo 1 > /sys/class/kgsl/kgsl-3d0/force_rail_on
echo 1 > /sys/class/kgsl/kgsl-3d0/force_clk_on
echo 1000000 > /sys/class/kgsl/kgsl-3d0/idle_timer
echo userspace > /sys/class/kgsl/kgsl-3d0/devfreq/governor
echo 2 > /sys/class/kgsl/kgsl-3d0/max_pwrlevel
echo 2 > /sys/class/kgsl/kgsl-3d0/min_pwrlevel
echo 2 > /sys/class/kgsl/kgsl-3d0/thermal_pwrlevel
echo %i > /sys/class/kgsl/kgsl-3d0/devfreq/max_freq
echo %i > /sys/class/kgsl/kgsl-3d0/devfreq/min_freq
echo %i > /sys/class/kgsl/kgsl-3d0/max_gpuclk
echo %i > /sys/class/kgsl/kgsl-3d0/gpuclk''' %
tuple(GPU_CLOCK_RATE for _ in range(4))] + \
self._devfreq_lock_cmds))
return self
def sanity_check(self):
HardwareAndroid.sanity_check(self)
if not self._adb.is_root():
return
result = self._adb.check(' '.join(
['cat',
'/sys/class/power_supply/battery/capacity',
'/sys/devices/system/cpu/online'] + \
['/sys/devices/system/cpu/cpu%i/cpufreq/scaling_cur_freq' % i
for i in range(2, 4)] + \
['/sys/class/kgsl/kgsl-3d0/thermal_pwrlevel',
'/sys/kernel/debug/clk/gpu_gx_gfx3d_clk/measure',
'/sys/kernel/debug/clk/bimc_clk/measure',
'/sys/class/thermal/thermal_zone22/temp',
'/sys/class/thermal/thermal_zone23/temp'] + \
self._devfreq_sanity_knobs))
expectations = \
[Expectation(int, min_value=30, name='battery', sleeptime=30*60),
Expectation(str, exact_value='2-3', name='online cpus')] + \
[Expectation(int, exact_value=CPU_CLOCK_RATE, name='cpu_%i clock rate' %i)
for i in range(2, 4)] + \
[Expectation(int, exact_value=2, name='gpu thermal power level'),
Expectation(long, min_value=(GPU_CLOCK_RATE - 5000),
max_value=(GPU_CLOCK_RATE + 5000),
name='measured gpu clock'),
Expectation(long, min_value=902390000, max_value=902409999,
name='measured ddr clock', sleeptime=10),
Expectation(int, max_value=41000, name='pm8994_tz temperature'),
Expectation(int, max_value=40, name='msm_therm temperature')] + \
self._devfreq_sanity_expectations
Expectation.check_all(expectations, result.splitlines())
def _discover_devfreqs(self):
self._devfreq_lock_cmds = list()
self._devfreq_sanity_knobs = list()
self._devfreq_sanity_expectations = list()
results = iter(self._adb.check('''\
KNOBS='%s'
for DEVICE in %s/*; do
if cd $DEVICE && ls $KNOBS >/dev/null; then
basename $DEVICE
cat $KNOBS
fi
done 2>/dev/null''' %
(' '.join(DevfreqKnobs._fields), DEVFREQ_DIRNAME)).splitlines())
while True:
batch = tuple(itertools.islice(results, 1 + len(DevfreqKnobs._fields)))
if not batch:
break
devfreq = batch[0]
if devfreq in DEVFREQ_BLACKLIST:
continue
path = '%s/%s' % (DEVFREQ_DIRNAME, devfreq)
knobs = DevfreqKnobs(*batch[1:])
if not 'performance' in knobs.available_governors.split():
print('WARNING: devfreq %s does not have performance governor' % path)
continue
self._devfreq_lock_cmds.append('echo performance > %s/governor' % path)
frequencies = map(int, knobs.available_frequencies.split())
if frequencies:
# choose the lowest frequency that is >= DEVFREQ_THROTTLE * max.
frequencies.sort()
target = DEVFREQ_THROTTLE * frequencies[-1]
idx = len(frequencies) - 1
while idx > 0 and frequencies[idx - 1] >= target:
idx -= 1
bench_frequency = frequencies[idx]
self._devfreq_lock_cmds.append('echo %i > %s/min_freq' %
(bench_frequency, path))
self._devfreq_lock_cmds.append('echo %i > %s/max_freq' %
(bench_frequency, path))
self._devfreq_sanity_knobs.append('%s/cur_freq' % path)
self._devfreq_sanity_expectations.append(
Expectation(int, exact_value=bench_frequency,
name='%s/cur_freq' % path))