Loading...
1#!/usr/bin/env python3
2# SPDX-License-Identifier: GPL-2.0-only
3#
4# Tool for analyzing suspend/resume timing
5# Copyright (c) 2013, Intel Corporation.
6#
7# This program is free software; you can redistribute it and/or modify it
8# under the terms and conditions of the GNU General Public License,
9# version 2, as published by the Free Software Foundation.
10#
11# This program is distributed in the hope it will be useful, but WITHOUT
12# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14# more details.
15#
16# Authors:
17# Todd Brandt <todd.e.brandt@linux.intel.com>
18#
19# Links:
20# Home Page
21# https://www.intel.com/content/www/us/en/developer/topic-technology/open/pm-graph/overview.html
22# Source repo
23# git@github.com:intel/pm-graph
24#
25# Description:
26# This tool is designed to assist kernel and OS developers in optimizing
27# their linux stack's suspend/resume time. Using a kernel image built
28# with a few extra options enabled, the tool will execute a suspend and
29# will capture dmesg and ftrace data until resume is complete. This data
30# is transformed into a device timeline and a callgraph to give a quick
31# and detailed view of which devices and callbacks are taking the most
32# time in suspend/resume. The output is a single html file which can be
33# viewed in firefox or chrome.
34#
35# The following kernel build options are required:
36# CONFIG_DEVMEM=y
37# CONFIG_PM_DEBUG=y
38# CONFIG_PM_SLEEP_DEBUG=y
39# CONFIG_FTRACE=y
40# CONFIG_FUNCTION_TRACER=y
41# CONFIG_FUNCTION_GRAPH_TRACER=y
42# CONFIG_KPROBES=y
43# CONFIG_KPROBES_ON_FTRACE=y
44#
45# For kernel versions older than 3.15:
46# The following additional kernel parameters are required:
47# (e.g. in file /etc/default/grub)
48# GRUB_CMDLINE_LINUX_DEFAULT="... initcall_debug log_buf_len=16M ..."
49#
50
51# ----------------- LIBRARIES --------------------
52
53import sys
54import time
55import os
56import string
57import re
58import platform
59import signal
60import codecs
61from datetime import datetime, timedelta
62import struct
63import configparser
64import gzip
65from threading import Thread
66from subprocess import call, Popen, PIPE
67import base64
68import traceback
69
70debugtiming = False
71mystarttime = time.time()
72def pprint(msg):
73 if debugtiming:
74 print('[%09.3f] %s' % (time.time()-mystarttime, msg))
75 else:
76 print(msg)
77 sys.stdout.flush()
78
79def ascii(text):
80 return text.decode('ascii', 'ignore')
81
82# ----------------- CLASSES --------------------
83
84# Class: SystemValues
85# Description:
86# A global, single-instance container used to
87# store system values and test parameters
88class SystemValues:
89 title = 'SleepGraph'
90 version = '5.13'
91 ansi = False
92 rs = 0
93 display = ''
94 gzip = False
95 sync = False
96 wifi = False
97 netfix = False
98 verbose = False
99 testlog = True
100 dmesglog = True
101 ftracelog = False
102 acpidebug = True
103 tstat = True
104 wifitrace = False
105 mindevlen = 0.0001
106 mincglen = 0.0
107 cgphase = ''
108 cgtest = -1
109 cgskip = ''
110 maxfail = 0
111 multitest = {'run': False, 'count': 1000000, 'delay': 0}
112 max_graph_depth = 0
113 callloopmaxgap = 0.0001
114 callloopmaxlen = 0.005
115 bufsize = 0
116 cpucount = 0
117 memtotal = 204800
118 memfree = 204800
119 osversion = ''
120 srgap = 0
121 cgexp = False
122 testdir = ''
123 outdir = ''
124 tpath = '/sys/kernel/tracing/'
125 fpdtpath = '/sys/firmware/acpi/tables/FPDT'
126 epath = '/sys/kernel/tracing/events/power/'
127 pmdpath = '/sys/power/pm_debug_messages'
128 s0ixpath = '/sys/module/intel_pmc_core/parameters/warn_on_s0ix_failures'
129 s0ixres = '/sys/devices/system/cpu/cpuidle/low_power_idle_system_residency_us'
130 acpipath='/sys/module/acpi/parameters/debug_level'
131 traceevents = [
132 'suspend_resume',
133 'wakeup_source_activate',
134 'wakeup_source_deactivate',
135 'device_pm_callback_end',
136 'device_pm_callback_start'
137 ]
138 logmsg = ''
139 testcommand = ''
140 mempath = '/dev/mem'
141 powerfile = '/sys/power/state'
142 mempowerfile = '/sys/power/mem_sleep'
143 diskpowerfile = '/sys/power/disk'
144 suspendmode = 'mem'
145 memmode = ''
146 diskmode = ''
147 hostname = 'localhost'
148 prefix = 'test'
149 teststamp = ''
150 sysstamp = ''
151 dmesgstart = 0.0
152 dmesgfile = ''
153 ftracefile = ''
154 htmlfile = 'output.html'
155 result = ''
156 rtcwake = True
157 rtcwaketime = 15
158 rtcpath = ''
159 devicefilter = []
160 cgfilter = []
161 stamp = 0
162 execcount = 1
163 x2delay = 0
164 skiphtml = False
165 usecallgraph = False
166 ftopfunc = 'pm_suspend'
167 ftop = False
168 usetraceevents = False
169 usetracemarkers = True
170 useftrace = True
171 usekprobes = True
172 usedevsrc = False
173 useprocmon = False
174 notestrun = False
175 cgdump = False
176 devdump = False
177 mixedphaseheight = True
178 devprops = dict()
179 cfgdef = dict()
180 platinfo = []
181 predelay = 0
182 postdelay = 0
183 tmstart = 'SUSPEND START %Y%m%d-%H:%M:%S.%f'
184 tmend = 'RESUME COMPLETE %Y%m%d-%H:%M:%S.%f'
185 tracefuncs = {
186 'async_synchronize_full': {},
187 'sys_sync': {},
188 'ksys_sync': {},
189 '__pm_notifier_call_chain': {},
190 'pm_prepare_console': {},
191 'pm_notifier_call_chain': {},
192 'freeze_processes': {},
193 'freeze_kernel_threads': {},
194 'pm_restrict_gfp_mask': {},
195 'acpi_suspend_begin': {},
196 'acpi_hibernation_begin': {},
197 'acpi_hibernation_enter': {},
198 'acpi_hibernation_leave': {},
199 'acpi_pm_freeze': {},
200 'acpi_pm_thaw': {},
201 'acpi_s2idle_end': {},
202 'acpi_s2idle_sync': {},
203 'acpi_s2idle_begin': {},
204 'acpi_s2idle_prepare': {},
205 'acpi_s2idle_prepare_late': {},
206 'acpi_s2idle_wake': {},
207 'acpi_s2idle_wakeup': {},
208 'acpi_s2idle_restore': {},
209 'acpi_s2idle_restore_early': {},
210 'hibernate_preallocate_memory': {},
211 'create_basic_memory_bitmaps': {},
212 'swsusp_write': {},
213 'suspend_console': {},
214 'acpi_pm_prepare': {},
215 'syscore_suspend': {},
216 'arch_enable_nonboot_cpus_end': {},
217 'syscore_resume': {},
218 'acpi_pm_finish': {},
219 'resume_console': {},
220 'acpi_pm_end': {},
221 'pm_restore_gfp_mask': {},
222 'thaw_processes': {},
223 'pm_restore_console': {},
224 'CPU_OFF': {
225 'func':'_cpu_down',
226 'args_x86_64': {'cpu':'%di:s32'},
227 'format': 'CPU_OFF[{cpu}]'
228 },
229 'CPU_ON': {
230 'func':'_cpu_up',
231 'args_x86_64': {'cpu':'%di:s32'},
232 'format': 'CPU_ON[{cpu}]'
233 },
234 }
235 dev_tracefuncs = {
236 # general wait/delay/sleep
237 'msleep': { 'args_x86_64': {'time':'%di:s32'}, 'ub': 1 },
238 'schedule_timeout': { 'args_x86_64': {'timeout':'%di:s32'}, 'ub': 1 },
239 'udelay': { 'func':'__const_udelay', 'args_x86_64': {'loops':'%di:s32'}, 'ub': 1 },
240 'usleep_range': {
241 'func':'usleep_range_state',
242 'args_x86_64': {'min':'%di:s32', 'max':'%si:s32'},
243 'ub': 1
244 },
245 'mutex_lock_slowpath': { 'func':'__mutex_lock_slowpath', 'ub': 1 },
246 'acpi_os_stall': {'ub': 1},
247 'rt_mutex_slowlock': {'ub': 1},
248 # ACPI
249 'acpi_resume_power_resources': {},
250 'acpi_ps_execute_method': { 'args_x86_64': {
251 'fullpath':'+0(+40(%di)):string',
252 }},
253 # mei_me
254 'mei_reset': {},
255 # filesystem
256 'ext4_sync_fs': {},
257 # 80211
258 'ath10k_bmi_read_memory': { 'args_x86_64': {'length':'%cx:s32'} },
259 'ath10k_bmi_write_memory': { 'args_x86_64': {'length':'%cx:s32'} },
260 'ath10k_bmi_fast_download': { 'args_x86_64': {'length':'%cx:s32'} },
261 'iwlagn_mac_start': {},
262 'iwlagn_alloc_bcast_station': {},
263 'iwl_trans_pcie_start_hw': {},
264 'iwl_trans_pcie_start_fw': {},
265 'iwl_run_init_ucode': {},
266 'iwl_load_ucode_wait_alive': {},
267 'iwl_alive_start': {},
268 'iwlagn_mac_stop': {},
269 'iwlagn_mac_suspend': {},
270 'iwlagn_mac_resume': {},
271 'iwlagn_mac_add_interface': {},
272 'iwlagn_mac_remove_interface': {},
273 'iwlagn_mac_change_interface': {},
274 'iwlagn_mac_config': {},
275 'iwlagn_configure_filter': {},
276 'iwlagn_mac_hw_scan': {},
277 'iwlagn_bss_info_changed': {},
278 'iwlagn_mac_channel_switch': {},
279 'iwlagn_mac_flush': {},
280 # ATA
281 'ata_eh_recover': { 'args_x86_64': {'port':'+36(%di):s32'} },
282 # i915
283 'i915_gem_resume': {},
284 'i915_restore_state': {},
285 'intel_opregion_setup': {},
286 'g4x_pre_enable_dp': {},
287 'vlv_pre_enable_dp': {},
288 'chv_pre_enable_dp': {},
289 'g4x_enable_dp': {},
290 'vlv_enable_dp': {},
291 'intel_hpd_init': {},
292 'intel_opregion_register': {},
293 'intel_dp_detect': {},
294 'intel_hdmi_detect': {},
295 'intel_opregion_init': {},
296 'intel_fbdev_set_suspend': {},
297 }
298 infocmds = [
299 [0, 'sysinfo', 'uname', '-a'],
300 [0, 'cpuinfo', 'head', '-7', '/proc/cpuinfo'],
301 [0, 'kparams', 'cat', '/proc/cmdline'],
302 [0, 'mcelog', 'mcelog'],
303 [0, 'pcidevices', 'lspci', '-tv'],
304 [0, 'usbdevices', 'lsusb', '-tv'],
305 [0, 'acpidevices', 'sh', '-c', 'ls -l /sys/bus/acpi/devices/*/physical_node'],
306 [0, 's0ix_require', 'cat', '/sys/kernel/debug/pmc_core/substate_requirements'],
307 [0, 's0ix_debug', 'cat', '/sys/kernel/debug/pmc_core/slp_s0_debug_status'],
308 [0, 'ethtool', 'ethtool', '{ethdev}'],
309 [1, 's0ix_residency', 'cat', '/sys/kernel/debug/pmc_core/slp_s0_residency_usec'],
310 [1, 'interrupts', 'cat', '/proc/interrupts'],
311 [1, 'wakeups', 'cat', '/sys/kernel/debug/wakeup_sources'],
312 [2, 'gpecounts', 'sh', '-c', 'grep -v invalid /sys/firmware/acpi/interrupts/*'],
313 [2, 'suspendstats', 'sh', '-c', 'grep -v invalid /sys/power/suspend_stats/*'],
314 [2, 'cpuidle', 'sh', '-c', 'grep -v invalid /sys/devices/system/cpu/cpu*/cpuidle/state*/s2idle/*'],
315 [2, 'battery', 'sh', '-c', 'grep -v invalid /sys/class/power_supply/*/*'],
316 [2, 'thermal', 'sh', '-c', 'grep . /sys/class/thermal/thermal_zone*/temp'],
317 ]
318 cgblacklist = []
319 kprobes = dict()
320 timeformat = '%.3f'
321 cmdline = '%s %s' % \
322 (os.path.basename(sys.argv[0]), ' '.join(sys.argv[1:]))
323 sudouser = ''
324 def __init__(self):
325 self.archargs = 'args_'+platform.machine()
326 self.hostname = platform.node()
327 if(self.hostname == ''):
328 self.hostname = 'localhost'
329 rtc = "rtc0"
330 if os.path.exists('/dev/rtc'):
331 rtc = os.readlink('/dev/rtc')
332 rtc = '/sys/class/rtc/'+rtc
333 if os.path.exists(rtc) and os.path.exists(rtc+'/date') and \
334 os.path.exists(rtc+'/time') and os.path.exists(rtc+'/wakealarm'):
335 self.rtcpath = rtc
336 if (hasattr(sys.stdout, 'isatty') and sys.stdout.isatty()):
337 self.ansi = True
338 self.testdir = datetime.now().strftime('suspend-%y%m%d-%H%M%S')
339 if os.getuid() == 0 and 'SUDO_USER' in os.environ and \
340 os.environ['SUDO_USER']:
341 self.sudouser = os.environ['SUDO_USER']
342 def resetlog(self):
343 self.logmsg = ''
344 self.platinfo = []
345 def vprint(self, msg):
346 self.logmsg += msg+'\n'
347 if self.verbose or msg.startswith('WARNING:'):
348 pprint(msg)
349 def signalHandler(self, signum, frame):
350 signame = self.signames[signum] if signum in self.signames else 'UNKNOWN'
351 if signame in ['SIGUSR1', 'SIGUSR2', 'SIGSEGV']:
352 traceback.print_stack()
353 stack = traceback.format_list(traceback.extract_stack())
354 self.outputResult({'stack':stack})
355 if signame == 'SIGUSR1':
356 return
357 msg = '%s caused a tool exit, line %d' % (signame, frame.f_lineno)
358 pprint(msg)
359 self.outputResult({'error':msg})
360 os.kill(os.getpid(), signal.SIGKILL)
361 sys.exit(3)
362 def signalHandlerInit(self):
363 capture = ['BUS', 'SYS', 'XCPU', 'XFSZ', 'PWR', 'HUP', 'INT', 'QUIT',
364 'ILL', 'ABRT', 'FPE', 'SEGV', 'TERM', 'USR1', 'USR2']
365 self.signames = dict()
366 for i in capture:
367 s = 'SIG'+i
368 try:
369 signum = getattr(signal, s)
370 signal.signal(signum, self.signalHandler)
371 except:
372 continue
373 self.signames[signum] = s
374 def rootCheck(self, fatal=True):
375 if(os.access(self.powerfile, os.W_OK)):
376 return True
377 if fatal:
378 msg = 'This command requires sysfs mount and root access'
379 pprint('ERROR: %s\n' % msg)
380 self.outputResult({'error':msg})
381 sys.exit(1)
382 return False
383 def rootUser(self, fatal=False):
384 if 'USER' in os.environ and os.environ['USER'] == 'root':
385 return True
386 if fatal:
387 msg = 'This command must be run as root'
388 pprint('ERROR: %s\n' % msg)
389 self.outputResult({'error':msg})
390 sys.exit(1)
391 return False
392 def usable(self, file, ishtml=False):
393 if not os.path.exists(file) or os.path.getsize(file) < 1:
394 return False
395 if ishtml:
396 try:
397 fp = open(file, 'r')
398 res = fp.read(1000)
399 fp.close()
400 except:
401 return False
402 if '<html>' not in res:
403 return False
404 return True
405 def getExec(self, cmd):
406 try:
407 fp = Popen(['which', cmd], stdout=PIPE, stderr=PIPE).stdout
408 out = ascii(fp.read()).strip()
409 fp.close()
410 except:
411 out = ''
412 if out:
413 return out
414 for path in ['/sbin', '/bin', '/usr/sbin', '/usr/bin',
415 '/usr/local/sbin', '/usr/local/bin']:
416 cmdfull = os.path.join(path, cmd)
417 if os.path.exists(cmdfull):
418 return cmdfull
419 return out
420 def setPrecision(self, num):
421 if num < 0 or num > 6:
422 return
423 self.timeformat = '%.{0}f'.format(num)
424 def setOutputFolder(self, value):
425 args = dict()
426 n = datetime.now()
427 args['date'] = n.strftime('%y%m%d')
428 args['time'] = n.strftime('%H%M%S')
429 args['hostname'] = args['host'] = self.hostname
430 args['mode'] = self.suspendmode
431 return value.format(**args)
432 def setOutputFile(self):
433 if self.dmesgfile != '':
434 m = re.match(r'(?P<name>.*)_dmesg\.txt.*', self.dmesgfile)
435 if(m):
436 self.htmlfile = m.group('name')+'.html'
437 if self.ftracefile != '':
438 m = re.match(r'(?P<name>.*)_ftrace\.txt.*', self.ftracefile)
439 if(m):
440 self.htmlfile = m.group('name')+'.html'
441 def systemInfo(self, info):
442 p = m = ''
443 if 'baseboard-manufacturer' in info:
444 m = info['baseboard-manufacturer']
445 elif 'system-manufacturer' in info:
446 m = info['system-manufacturer']
447 if 'system-product-name' in info:
448 p = info['system-product-name']
449 elif 'baseboard-product-name' in info:
450 p = info['baseboard-product-name']
451 if m[:5].lower() == 'intel' and 'baseboard-product-name' in info:
452 p = info['baseboard-product-name']
453 c = info['processor-version'] if 'processor-version' in info else ''
454 b = info['bios-version'] if 'bios-version' in info else ''
455 r = info['bios-release-date'] if 'bios-release-date' in info else ''
456 self.sysstamp = '# sysinfo | man:%s | plat:%s | cpu:%s | bios:%s | biosdate:%s | numcpu:%d | memsz:%d | memfr:%d' % \
457 (m, p, c, b, r, self.cpucount, self.memtotal, self.memfree)
458 if self.osversion:
459 self.sysstamp += ' | os:%s' % self.osversion
460 def printSystemInfo(self, fatal=False):
461 self.rootCheck(True)
462 out = dmidecode(self.mempath, fatal)
463 if len(out) < 1:
464 return
465 fmt = '%-24s: %s'
466 if self.osversion:
467 print(fmt % ('os-version', self.osversion))
468 for name in sorted(out):
469 print(fmt % (name, out[name]))
470 print(fmt % ('cpucount', ('%d' % self.cpucount)))
471 print(fmt % ('memtotal', ('%d kB' % self.memtotal)))
472 print(fmt % ('memfree', ('%d kB' % self.memfree)))
473 def cpuInfo(self):
474 self.cpucount = 0
475 if os.path.exists('/proc/cpuinfo'):
476 with open('/proc/cpuinfo', 'r') as fp:
477 for line in fp:
478 if re.match(r'^processor[ \t]*:[ \t]*[0-9]*', line):
479 self.cpucount += 1
480 if os.path.exists('/proc/meminfo'):
481 with open('/proc/meminfo', 'r') as fp:
482 for line in fp:
483 m = re.match(r'^MemTotal:[ \t]*(?P<sz>[0-9]*) *kB', line)
484 if m:
485 self.memtotal = int(m.group('sz'))
486 m = re.match(r'^MemFree:[ \t]*(?P<sz>[0-9]*) *kB', line)
487 if m:
488 self.memfree = int(m.group('sz'))
489 if os.path.exists('/etc/os-release'):
490 with open('/etc/os-release', 'r') as fp:
491 for line in fp:
492 if line.startswith('PRETTY_NAME='):
493 self.osversion = line[12:].strip().replace('"', '')
494 def initTestOutput(self, name):
495 self.prefix = self.hostname
496 v = open('/proc/version', 'r').read().strip()
497 kver = v.split()[2]
498 fmt = name+'-%m%d%y-%H%M%S'
499 testtime = datetime.now().strftime(fmt)
500 self.teststamp = \
501 '# '+testtime+' '+self.prefix+' '+self.suspendmode+' '+kver
502 ext = ''
503 if self.gzip:
504 ext = '.gz'
505 self.dmesgfile = \
506 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'_dmesg.txt'+ext
507 self.ftracefile = \
508 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'_ftrace.txt'+ext
509 self.htmlfile = \
510 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'.html'
511 if not os.path.isdir(self.testdir):
512 os.makedirs(self.testdir)
513 self.sudoUserchown(self.testdir)
514 def getValueList(self, value):
515 out = []
516 for i in value.split(','):
517 if i.strip():
518 out.append(i.strip())
519 return out
520 def setDeviceFilter(self, value):
521 self.devicefilter = self.getValueList(value)
522 def setCallgraphFilter(self, value):
523 self.cgfilter = self.getValueList(value)
524 def skipKprobes(self, value):
525 for k in self.getValueList(value):
526 if k in self.tracefuncs:
527 del self.tracefuncs[k]
528 if k in self.dev_tracefuncs:
529 del self.dev_tracefuncs[k]
530 def setCallgraphBlacklist(self, file):
531 self.cgblacklist = self.listFromFile(file)
532 def rtcWakeAlarmOn(self):
533 call('echo 0 > '+self.rtcpath+'/wakealarm', shell=True)
534 nowtime = open(self.rtcpath+'/since_epoch', 'r').read().strip()
535 if nowtime:
536 nowtime = int(nowtime)
537 else:
538 # if hardware time fails, use the software time
539 nowtime = int(datetime.now().strftime('%s'))
540 alarm = nowtime + self.rtcwaketime
541 call('echo %d > %s/wakealarm' % (alarm, self.rtcpath), shell=True)
542 def rtcWakeAlarmOff(self):
543 call('echo 0 > %s/wakealarm' % self.rtcpath, shell=True)
544 def initdmesg(self):
545 # get the latest time stamp from the dmesg log
546 lines = Popen('dmesg', stdout=PIPE).stdout.readlines()
547 ktime = '0'
548 for line in reversed(lines):
549 line = ascii(line).replace('\r\n', '')
550 idx = line.find('[')
551 if idx > 1:
552 line = line[idx:]
553 m = re.match(r'[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
554 if(m):
555 ktime = m.group('ktime')
556 break
557 self.dmesgstart = float(ktime)
558 def getdmesg(self, testdata):
559 op = self.writeDatafileHeader(self.dmesgfile, testdata)
560 # store all new dmesg lines since initdmesg was called
561 fp = Popen('dmesg', stdout=PIPE).stdout
562 for line in fp:
563 line = ascii(line).replace('\r\n', '')
564 idx = line.find('[')
565 if idx > 1:
566 line = line[idx:]
567 m = re.match(r'[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
568 if(not m):
569 continue
570 ktime = float(m.group('ktime'))
571 if ktime > self.dmesgstart:
572 op.write(line)
573 fp.close()
574 op.close()
575 def listFromFile(self, file):
576 list = []
577 fp = open(file)
578 for i in fp.read().split('\n'):
579 i = i.strip()
580 if i and i[0] != '#':
581 list.append(i)
582 fp.close()
583 return list
584 def addFtraceFilterFunctions(self, file):
585 for i in self.listFromFile(file):
586 if len(i) < 2:
587 continue
588 self.tracefuncs[i] = dict()
589 def getFtraceFilterFunctions(self, current):
590 self.rootCheck(True)
591 if not current:
592 call('cat '+self.tpath+'available_filter_functions', shell=True)
593 return
594 master = self.listFromFile(self.tpath+'available_filter_functions')
595 for i in sorted(self.tracefuncs):
596 if 'func' in self.tracefuncs[i]:
597 i = self.tracefuncs[i]['func']
598 if i in master:
599 print(i)
600 else:
601 print(self.colorText(i))
602 def setFtraceFilterFunctions(self, list):
603 master = self.listFromFile(self.tpath+'available_filter_functions')
604 flist = ''
605 for i in list:
606 if i not in master:
607 continue
608 if ' [' in i:
609 flist += i.split(' ')[0]+'\n'
610 else:
611 flist += i+'\n'
612 fp = open(self.tpath+'set_graph_function', 'w')
613 fp.write(flist)
614 fp.close()
615 def basicKprobe(self, name):
616 self.kprobes[name] = {'name': name,'func': name,'args': dict(),'format': name}
617 def defaultKprobe(self, name, kdata):
618 k = kdata
619 for field in ['name', 'format', 'func']:
620 if field not in k:
621 k[field] = name
622 if self.archargs in k:
623 k['args'] = k[self.archargs]
624 else:
625 k['args'] = dict()
626 k['format'] = name
627 self.kprobes[name] = k
628 def kprobeColor(self, name):
629 if name not in self.kprobes or 'color' not in self.kprobes[name]:
630 return ''
631 return self.kprobes[name]['color']
632 def kprobeDisplayName(self, name, dataraw):
633 if name not in self.kprobes:
634 self.basicKprobe(name)
635 data = ''
636 quote=0
637 # first remvoe any spaces inside quotes, and the quotes
638 for c in dataraw:
639 if c == '"':
640 quote = (quote + 1) % 2
641 if quote and c == ' ':
642 data += '_'
643 elif c != '"':
644 data += c
645 fmt, args = self.kprobes[name]['format'], self.kprobes[name]['args']
646 arglist = dict()
647 # now process the args
648 for arg in sorted(args):
649 arglist[arg] = ''
650 m = re.match(r'.* '+arg+'=(?P<arg>.*) ', data);
651 if m:
652 arglist[arg] = m.group('arg')
653 else:
654 m = re.match(r'.* '+arg+'=(?P<arg>.*)', data);
655 if m:
656 arglist[arg] = m.group('arg')
657 out = fmt.format(**arglist)
658 out = out.replace(' ', '_').replace('"', '')
659 return out
660 def kprobeText(self, kname, kprobe):
661 name = fmt = func = kname
662 args = dict()
663 if 'name' in kprobe:
664 name = kprobe['name']
665 if 'format' in kprobe:
666 fmt = kprobe['format']
667 if 'func' in kprobe:
668 func = kprobe['func']
669 if self.archargs in kprobe:
670 args = kprobe[self.archargs]
671 if 'args' in kprobe:
672 args = kprobe['args']
673 if re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', func):
674 doError('Kprobe "%s" has format info in the function name "%s"' % (name, func))
675 for arg in re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', fmt):
676 if arg not in args:
677 doError('Kprobe "%s" is missing argument "%s"' % (name, arg))
678 val = 'p:%s_cal %s' % (name, func)
679 for i in sorted(args):
680 val += ' %s=%s' % (i, args[i])
681 val += '\nr:%s_ret %s $retval\n' % (name, func)
682 return val
683 def addKprobes(self, output=False):
684 if len(self.kprobes) < 1:
685 return
686 if output:
687 pprint(' kprobe functions in this kernel:')
688 # first test each kprobe
689 rejects = []
690 # sort kprobes: trace, ub-dev, custom, dev
691 kpl = [[], [], [], []]
692 linesout = len(self.kprobes)
693 for name in sorted(self.kprobes):
694 res = self.colorText('YES', 32)
695 if not self.testKprobe(name, self.kprobes[name]):
696 res = self.colorText('NO')
697 rejects.append(name)
698 else:
699 if name in self.tracefuncs:
700 kpl[0].append(name)
701 elif name in self.dev_tracefuncs:
702 if 'ub' in self.dev_tracefuncs[name]:
703 kpl[1].append(name)
704 else:
705 kpl[3].append(name)
706 else:
707 kpl[2].append(name)
708 if output:
709 pprint(' %s: %s' % (name, res))
710 kplist = kpl[0] + kpl[1] + kpl[2] + kpl[3]
711 # remove all failed ones from the list
712 for name in rejects:
713 self.kprobes.pop(name)
714 # set the kprobes all at once
715 self.fsetVal('', 'kprobe_events')
716 kprobeevents = ''
717 for kp in kplist:
718 kprobeevents += self.kprobeText(kp, self.kprobes[kp])
719 self.fsetVal(kprobeevents, 'kprobe_events')
720 if output:
721 check = self.fgetVal('kprobe_events')
722 linesack = (len(check.split('\n')) - 1) // 2
723 pprint(' kprobe functions enabled: %d/%d' % (linesack, linesout))
724 self.fsetVal('1', 'events/kprobes/enable')
725 def testKprobe(self, kname, kprobe):
726 self.fsetVal('0', 'events/kprobes/enable')
727 kprobeevents = self.kprobeText(kname, kprobe)
728 if not kprobeevents:
729 return False
730 try:
731 self.fsetVal(kprobeevents, 'kprobe_events')
732 check = self.fgetVal('kprobe_events')
733 except:
734 return False
735 linesout = len(kprobeevents.split('\n'))
736 linesack = len(check.split('\n'))
737 if linesack < linesout:
738 return False
739 return True
740 def setVal(self, val, file):
741 if not os.path.exists(file):
742 return False
743 try:
744 fp = open(file, 'wb', 0)
745 fp.write(val.encode())
746 fp.flush()
747 fp.close()
748 except:
749 return False
750 return True
751 def fsetVal(self, val, path):
752 if not self.useftrace:
753 return False
754 return self.setVal(val, self.tpath+path)
755 def getVal(self, file):
756 res = ''
757 if not os.path.exists(file):
758 return res
759 try:
760 fp = open(file, 'r')
761 res = fp.read()
762 fp.close()
763 except:
764 pass
765 return res
766 def fgetVal(self, path):
767 if not self.useftrace:
768 return ''
769 return self.getVal(self.tpath+path)
770 def cleanupFtrace(self):
771 if self.useftrace:
772 self.fsetVal('0', 'events/kprobes/enable')
773 self.fsetVal('', 'kprobe_events')
774 self.fsetVal('1024', 'buffer_size_kb')
775 def setupAllKprobes(self):
776 for name in self.tracefuncs:
777 self.defaultKprobe(name, self.tracefuncs[name])
778 for name in self.dev_tracefuncs:
779 self.defaultKprobe(name, self.dev_tracefuncs[name])
780 def isCallgraphFunc(self, name):
781 if len(self.tracefuncs) < 1 and self.suspendmode == 'command':
782 return True
783 for i in self.tracefuncs:
784 if 'func' in self.tracefuncs[i]:
785 f = self.tracefuncs[i]['func']
786 else:
787 f = i
788 if name == f:
789 return True
790 return False
791 def initFtrace(self, quiet=False):
792 if not self.useftrace:
793 return
794 if not quiet:
795 sysvals.printSystemInfo(False)
796 pprint('INITIALIZING FTRACE')
797 # turn trace off
798 self.fsetVal('0', 'tracing_on')
799 self.cleanupFtrace()
800 # set the trace clock to global
801 self.fsetVal('global', 'trace_clock')
802 self.fsetVal('nop', 'current_tracer')
803 # set trace buffer to an appropriate value
804 cpus = max(1, self.cpucount)
805 if self.bufsize > 0:
806 tgtsize = self.bufsize
807 elif self.usecallgraph or self.usedevsrc:
808 bmax = (1*1024*1024) if self.suspendmode in ['disk', 'command'] \
809 else (3*1024*1024)
810 tgtsize = min(self.memfree, bmax)
811 else:
812 tgtsize = 65536
813 while not self.fsetVal('%d' % (tgtsize // cpus), 'buffer_size_kb'):
814 # if the size failed to set, lower it and keep trying
815 tgtsize -= 65536
816 if tgtsize < 65536:
817 tgtsize = int(self.fgetVal('buffer_size_kb')) * cpus
818 break
819 self.vprint('Setting trace buffers to %d kB (%d kB per cpu)' % (tgtsize, tgtsize/cpus))
820 # initialize the callgraph trace
821 if(self.usecallgraph):
822 # set trace type
823 self.fsetVal('function_graph', 'current_tracer')
824 self.fsetVal('', 'set_ftrace_filter')
825 # temporary hack to fix https://bugzilla.kernel.org/show_bug.cgi?id=212761
826 fp = open(self.tpath+'set_ftrace_notrace', 'w')
827 fp.write('native_queued_spin_lock_slowpath\ndev_driver_string')
828 fp.close()
829 # set trace format options
830 self.fsetVal('print-parent', 'trace_options')
831 self.fsetVal('funcgraph-abstime', 'trace_options')
832 self.fsetVal('funcgraph-cpu', 'trace_options')
833 self.fsetVal('funcgraph-duration', 'trace_options')
834 self.fsetVal('funcgraph-proc', 'trace_options')
835 self.fsetVal('funcgraph-tail', 'trace_options')
836 self.fsetVal('nofuncgraph-overhead', 'trace_options')
837 self.fsetVal('context-info', 'trace_options')
838 self.fsetVal('graph-time', 'trace_options')
839 self.fsetVal('%d' % self.max_graph_depth, 'max_graph_depth')
840 cf = ['dpm_run_callback']
841 if(self.usetraceevents):
842 cf += ['dpm_prepare', 'dpm_complete']
843 for fn in self.tracefuncs:
844 if 'func' in self.tracefuncs[fn]:
845 cf.append(self.tracefuncs[fn]['func'])
846 else:
847 cf.append(fn)
848 if self.ftop:
849 self.setFtraceFilterFunctions([self.ftopfunc])
850 else:
851 self.setFtraceFilterFunctions(cf)
852 # initialize the kprobe trace
853 elif self.usekprobes:
854 for name in self.tracefuncs:
855 self.defaultKprobe(name, self.tracefuncs[name])
856 if self.usedevsrc:
857 for name in self.dev_tracefuncs:
858 self.defaultKprobe(name, self.dev_tracefuncs[name])
859 if not quiet:
860 pprint('INITIALIZING KPROBES')
861 self.addKprobes(self.verbose)
862 if(self.usetraceevents):
863 # turn trace events on
864 events = iter(self.traceevents)
865 for e in events:
866 self.fsetVal('1', 'events/power/'+e+'/enable')
867 # clear the trace buffer
868 self.fsetVal('', 'trace')
869 def verifyFtrace(self):
870 # files needed for any trace data
871 files = ['buffer_size_kb', 'current_tracer', 'trace', 'trace_clock',
872 'trace_marker', 'trace_options', 'tracing_on']
873 # legacy check for old systems
874 if not os.path.exists(self.tpath+'trace'):
875 self.tpath = '/sys/kernel/debug/tracing/'
876 if not os.path.exists(self.epath):
877 self.epath = '/sys/kernel/debug/tracing/events/power/'
878 # files needed for callgraph trace data
879 tp = self.tpath
880 if(self.usecallgraph):
881 files += [
882 'available_filter_functions',
883 'set_ftrace_filter',
884 'set_graph_function'
885 ]
886 for f in files:
887 if(os.path.exists(tp+f) == False):
888 return False
889 return True
890 def verifyKprobes(self):
891 # files needed for kprobes to work
892 files = ['kprobe_events', 'events']
893 tp = self.tpath
894 for f in files:
895 if(os.path.exists(tp+f) == False):
896 return False
897 return True
898 def colorText(self, str, color=31):
899 if not self.ansi:
900 return str
901 return '\x1B[%d;40m%s\x1B[m' % (color, str)
902 def writeDatafileHeader(self, filename, testdata):
903 fp = self.openlog(filename, 'w')
904 fp.write('%s\n%s\n# command | %s\n' % (self.teststamp, self.sysstamp, self.cmdline))
905 for test in testdata:
906 if 'fw' in test:
907 fw = test['fw']
908 if(fw):
909 fp.write('# fwsuspend %u fwresume %u\n' % (fw[0], fw[1]))
910 if 'turbo' in test:
911 fp.write('# turbostat %s\n' % test['turbo'])
912 if 'wifi' in test:
913 fp.write('# wifi %s\n' % test['wifi'])
914 if 'netfix' in test:
915 fp.write('# netfix %s\n' % test['netfix'])
916 if test['error'] or len(testdata) > 1:
917 fp.write('# enter_sleep_error %s\n' % test['error'])
918 return fp
919 def sudoUserchown(self, dir):
920 if os.path.exists(dir) and self.sudouser:
921 cmd = 'chown -R {0}:{0} {1} > /dev/null 2>&1'
922 call(cmd.format(self.sudouser, dir), shell=True)
923 def outputResult(self, testdata, num=0):
924 if not self.result:
925 return
926 n = ''
927 if num > 0:
928 n = '%d' % num
929 fp = open(self.result, 'a')
930 if 'stack' in testdata:
931 fp.write('Printing stack trace:\n')
932 for line in testdata['stack']:
933 fp.write(line)
934 fp.close()
935 self.sudoUserchown(self.result)
936 return
937 if 'error' in testdata:
938 fp.write('result%s: fail\n' % n)
939 fp.write('error%s: %s\n' % (n, testdata['error']))
940 else:
941 fp.write('result%s: pass\n' % n)
942 if 'mode' in testdata:
943 fp.write('mode%s: %s\n' % (n, testdata['mode']))
944 for v in ['suspend', 'resume', 'boot', 'lastinit']:
945 if v in testdata:
946 fp.write('%s%s: %.3f\n' % (v, n, testdata[v]))
947 for v in ['fwsuspend', 'fwresume']:
948 if v in testdata:
949 fp.write('%s%s: %.3f\n' % (v, n, testdata[v] / 1000000.0))
950 if 'bugurl' in testdata:
951 fp.write('url%s: %s\n' % (n, testdata['bugurl']))
952 fp.close()
953 self.sudoUserchown(self.result)
954 def configFile(self, file):
955 dir = os.path.dirname(os.path.realpath(__file__))
956 if os.path.exists(file):
957 return file
958 elif os.path.exists(dir+'/'+file):
959 return dir+'/'+file
960 elif os.path.exists(dir+'/config/'+file):
961 return dir+'/config/'+file
962 return ''
963 def openlog(self, filename, mode):
964 isgz = self.gzip
965 if mode == 'r':
966 try:
967 with gzip.open(filename, mode+'t') as fp:
968 test = fp.read(64)
969 isgz = True
970 except:
971 isgz = False
972 if isgz:
973 return gzip.open(filename, mode+'t')
974 return open(filename, mode)
975 def putlog(self, filename, text):
976 with self.openlog(filename, 'a') as fp:
977 fp.write(text)
978 fp.close()
979 def dlog(self, text):
980 if not self.dmesgfile:
981 return
982 self.putlog(self.dmesgfile, '# %s\n' % text)
983 def flog(self, text):
984 self.putlog(self.ftracefile, text)
985 def b64unzip(self, data):
986 try:
987 out = codecs.decode(base64.b64decode(data), 'zlib').decode()
988 except:
989 out = data
990 return out
991 def b64zip(self, data):
992 out = base64.b64encode(codecs.encode(data.encode(), 'zlib')).decode()
993 return out
994 def platforminfo(self, cmdafter):
995 # add platform info on to a completed ftrace file
996 if not os.path.exists(self.ftracefile):
997 return False
998 footer = '#\n'
999
1000 # add test command string line if need be
1001 if self.suspendmode == 'command' and self.testcommand:
1002 footer += '# platform-testcmd: %s\n' % (self.testcommand)
1003
1004 # get a list of target devices from the ftrace file
1005 props = dict()
1006 tp = TestProps()
1007 tf = self.openlog(self.ftracefile, 'r')
1008 for line in tf:
1009 if tp.stampInfo(line, self):
1010 continue
1011 # parse only valid lines, if this is not one move on
1012 m = re.match(tp.ftrace_line_fmt, line)
1013 if(not m or 'device_pm_callback_start' not in line):
1014 continue
1015 m = re.match(r'.*: (?P<drv>.*) (?P<d>.*), parent: *(?P<p>.*), .*', m.group('msg'));
1016 if(not m):
1017 continue
1018 dev = m.group('d')
1019 if dev not in props:
1020 props[dev] = DevProps()
1021 tf.close()
1022
1023 # now get the syspath for each target device
1024 for dirname, dirnames, filenames in os.walk('/sys/devices'):
1025 if(re.match(r'.*/power', dirname) and 'async' in filenames):
1026 dev = dirname.split('/')[-2]
1027 if dev in props and (not props[dev].syspath or len(dirname) < len(props[dev].syspath)):
1028 props[dev].syspath = dirname[:-6]
1029
1030 # now fill in the properties for our target devices
1031 for dev in sorted(props):
1032 dirname = props[dev].syspath
1033 if not dirname or not os.path.exists(dirname):
1034 continue
1035 props[dev].isasync = False
1036 if os.path.exists(dirname+'/power/async'):
1037 fp = open(dirname+'/power/async')
1038 if 'enabled' in fp.read():
1039 props[dev].isasync = True
1040 fp.close()
1041 fields = os.listdir(dirname)
1042 for file in ['product', 'name', 'model', 'description', 'id', 'idVendor']:
1043 if file not in fields:
1044 continue
1045 try:
1046 with open(os.path.join(dirname, file), 'rb') as fp:
1047 props[dev].altname = ascii(fp.read())
1048 except:
1049 continue
1050 if file == 'idVendor':
1051 idv, idp = props[dev].altname.strip(), ''
1052 try:
1053 with open(os.path.join(dirname, 'idProduct'), 'rb') as fp:
1054 idp = ascii(fp.read()).strip()
1055 except:
1056 props[dev].altname = ''
1057 break
1058 props[dev].altname = '%s:%s' % (idv, idp)
1059 break
1060 if props[dev].altname:
1061 out = props[dev].altname.strip().replace('\n', ' ')\
1062 .replace(',', ' ').replace(';', ' ')
1063 props[dev].altname = out
1064
1065 # add a devinfo line to the bottom of ftrace
1066 out = ''
1067 for dev in sorted(props):
1068 out += props[dev].out(dev)
1069 footer += '# platform-devinfo: %s\n' % self.b64zip(out)
1070
1071 # add a line for each of these commands with their outputs
1072 for name, cmdline, info in cmdafter:
1073 footer += '# platform-%s: %s | %s\n' % (name, cmdline, self.b64zip(info))
1074 self.flog(footer)
1075 return True
1076 def commonPrefix(self, list):
1077 if len(list) < 2:
1078 return ''
1079 prefix = list[0]
1080 for s in list[1:]:
1081 while s[:len(prefix)] != prefix and prefix:
1082 prefix = prefix[:len(prefix)-1]
1083 if not prefix:
1084 break
1085 if '/' in prefix and prefix[-1] != '/':
1086 prefix = prefix[0:prefix.rfind('/')+1]
1087 return prefix
1088 def dictify(self, text, format):
1089 out = dict()
1090 header = True if format == 1 else False
1091 delim = ' ' if format == 1 else ':'
1092 for line in text.split('\n'):
1093 if header:
1094 header, out['@'] = False, line
1095 continue
1096 line = line.strip()
1097 if delim in line:
1098 data = line.split(delim, 1)
1099 num = re.search(r'[\d]+', data[1])
1100 if format == 2 and num:
1101 out[data[0].strip()] = num.group()
1102 else:
1103 out[data[0].strip()] = data[1]
1104 return out
1105 def cmdinfovar(self, arg):
1106 if arg == 'ethdev':
1107 try:
1108 cmd = [self.getExec('ip'), '-4', '-o', '-br', 'addr']
1109 fp = Popen(cmd, stdout=PIPE, stderr=PIPE).stdout
1110 info = ascii(fp.read()).strip()
1111 fp.close()
1112 except:
1113 return 'iptoolcrash'
1114 for line in info.split('\n'):
1115 if line[0] == 'e' and 'UP' in line:
1116 return line.split()[0]
1117 return 'nodevicefound'
1118 return 'unknown'
1119 def cmdinfo(self, begin, debug=False):
1120 out = []
1121 if begin:
1122 self.cmd1 = dict()
1123 for cargs in self.infocmds:
1124 delta, name, args = cargs[0], cargs[1], cargs[2:]
1125 for i in range(len(args)):
1126 if args[i][0] == '{' and args[i][-1] == '}':
1127 args[i] = self.cmdinfovar(args[i][1:-1])
1128 cmdline, cmdpath = ' '.join(args[0:]), self.getExec(args[0])
1129 if not cmdpath or (begin and not delta):
1130 continue
1131 self.dlog('[%s]' % cmdline)
1132 try:
1133 fp = Popen([cmdpath]+args[1:], stdout=PIPE, stderr=PIPE).stdout
1134 info = ascii(fp.read()).strip()
1135 fp.close()
1136 except:
1137 continue
1138 if not debug and begin:
1139 self.cmd1[name] = self.dictify(info, delta)
1140 elif not debug and delta and name in self.cmd1:
1141 before, after = self.cmd1[name], self.dictify(info, delta)
1142 dinfo = ('\t%s\n' % before['@']) if '@' in before and len(before) > 1 else ''
1143 prefix = self.commonPrefix(list(before.keys()))
1144 for key in sorted(before):
1145 if key in after and before[key] != after[key]:
1146 title = key.replace(prefix, '')
1147 if delta == 2:
1148 dinfo += '\t%s : %s -> %s\n' % \
1149 (title, before[key].strip(), after[key].strip())
1150 else:
1151 dinfo += '%10s (start) : %s\n%10s (after) : %s\n' % \
1152 (title, before[key], title, after[key])
1153 dinfo = '\tnothing changed' if not dinfo else dinfo.rstrip()
1154 out.append((name, cmdline, dinfo))
1155 else:
1156 out.append((name, cmdline, '\tnothing' if not info else info))
1157 return out
1158 def testVal(self, file, fmt='basic', value=''):
1159 if file == 'restoreall':
1160 for f in self.cfgdef:
1161 if os.path.exists(f):
1162 fp = open(f, 'w')
1163 fp.write(self.cfgdef[f])
1164 fp.close()
1165 self.cfgdef = dict()
1166 elif value and os.path.exists(file):
1167 fp = open(file, 'r+')
1168 if fmt == 'radio':
1169 m = re.match(r'.*\[(?P<v>.*)\].*', fp.read())
1170 if m:
1171 self.cfgdef[file] = m.group('v')
1172 elif fmt == 'acpi':
1173 line = fp.read().strip().split('\n')[-1]
1174 m = re.match(r'.* (?P<v>[0-9A-Fx]*) .*', line)
1175 if m:
1176 self.cfgdef[file] = m.group('v')
1177 else:
1178 self.cfgdef[file] = fp.read().strip()
1179 fp.write(value)
1180 fp.close()
1181 def s0ixSupport(self):
1182 if not os.path.exists(self.s0ixres) or not os.path.exists(self.mempowerfile):
1183 return False
1184 fp = open(sysvals.mempowerfile, 'r')
1185 data = fp.read().strip()
1186 fp.close()
1187 if '[s2idle]' in data:
1188 return True
1189 return False
1190 def haveTurbostat(self):
1191 if not self.tstat:
1192 return False
1193 cmd = self.getExec('turbostat')
1194 if not cmd:
1195 return False
1196 fp = Popen([cmd, '-v'], stdout=PIPE, stderr=PIPE).stderr
1197 out = ascii(fp.read()).strip()
1198 fp.close()
1199 if re.match(r'turbostat version .*', out):
1200 self.vprint(out)
1201 return True
1202 return False
1203 def turbostat(self, s0ixready):
1204 cmd = self.getExec('turbostat')
1205 rawout = keyline = valline = ''
1206 fullcmd = '%s -q -S echo freeze > %s' % (cmd, self.powerfile)
1207 fp = Popen(['sh', '-c', fullcmd], stdout=PIPE, stderr=PIPE)
1208 for line in fp.stderr:
1209 line = ascii(line)
1210 rawout += line
1211 if keyline and valline:
1212 continue
1213 if re.match(r'(?i)Avg_MHz.*', line):
1214 keyline = line.strip().split()
1215 elif keyline:
1216 valline = line.strip().split()
1217 fp.wait()
1218 if not keyline or not valline or len(keyline) != len(valline):
1219 errmsg = 'unrecognized turbostat output:\n'+rawout.strip()
1220 self.vprint(errmsg)
1221 if not self.verbose:
1222 pprint(errmsg)
1223 return (fp.returncode, '')
1224 if self.verbose:
1225 pprint(rawout.strip())
1226 out = []
1227 for key in keyline:
1228 idx = keyline.index(key)
1229 val = valline[idx]
1230 if key == 'SYS%LPI' and not s0ixready and re.match(r'^[0\.]*$', val):
1231 continue
1232 out.append('%s=%s' % (key, val))
1233 return (fp.returncode, '|'.join(out))
1234 def netfixon(self, net='both'):
1235 cmd = self.getExec('netfix')
1236 if not cmd:
1237 return ''
1238 fp = Popen([cmd, '-s', net, 'on'], stdout=PIPE, stderr=PIPE).stdout
1239 out = ascii(fp.read()).strip()
1240 fp.close()
1241 return out
1242 def wifiDetails(self, dev):
1243 try:
1244 info = open('/sys/class/net/%s/device/uevent' % dev, 'r').read().strip()
1245 except:
1246 return dev
1247 vals = [dev]
1248 for prop in info.split('\n'):
1249 if prop.startswith('DRIVER=') or prop.startswith('PCI_ID='):
1250 vals.append(prop.split('=')[-1])
1251 return ':'.join(vals)
1252 def checkWifi(self, dev=''):
1253 try:
1254 w = open('/proc/net/wireless', 'r').read().strip()
1255 except:
1256 return ''
1257 for line in reversed(w.split('\n')):
1258 m = re.match(r' *(?P<dev>.*): (?P<stat>[0-9a-f]*) .*', line)
1259 if not m or (dev and dev != m.group('dev')):
1260 continue
1261 return m.group('dev')
1262 return ''
1263 def pollWifi(self, dev, timeout=10):
1264 start = time.time()
1265 while (time.time() - start) < timeout:
1266 w = self.checkWifi(dev)
1267 if w:
1268 return '%s reconnected %.2f' % \
1269 (self.wifiDetails(dev), max(0, time.time() - start))
1270 time.sleep(0.01)
1271 return '%s timeout %d' % (self.wifiDetails(dev), timeout)
1272 def errorSummary(self, errinfo, msg):
1273 found = False
1274 for entry in errinfo:
1275 if re.match(entry['match'], msg):
1276 entry['count'] += 1
1277 if self.hostname not in entry['urls']:
1278 entry['urls'][self.hostname] = [self.htmlfile]
1279 elif self.htmlfile not in entry['urls'][self.hostname]:
1280 entry['urls'][self.hostname].append(self.htmlfile)
1281 found = True
1282 break
1283 if found:
1284 return
1285 arr = msg.split()
1286 for j in range(len(arr)):
1287 if re.match(r'^[0-9,\-\.]*$', arr[j]):
1288 arr[j] = r'[0-9,\-\.]*'
1289 else:
1290 arr[j] = arr[j]\
1291 .replace('\\', r'\\\\').replace(']', r'\]').replace('[', r'\[')\
1292 .replace('.', r'\.').replace('+', r'\+').replace('*', r'\*')\
1293 .replace('(', r'\(').replace(')', r'\)').replace('}', r'\}')\
1294 .replace('{', r'\{')
1295 mstr = ' *'.join(arr)
1296 entry = {
1297 'line': msg,
1298 'match': mstr,
1299 'count': 1,
1300 'urls': {self.hostname: [self.htmlfile]}
1301 }
1302 errinfo.append(entry)
1303 def multistat(self, start, idx, finish):
1304 if 'time' in self.multitest:
1305 id = '%d Duration=%dmin' % (idx+1, self.multitest['time'])
1306 else:
1307 id = '%d/%d' % (idx+1, self.multitest['count'])
1308 t = time.time()
1309 if 'start' not in self.multitest:
1310 self.multitest['start'] = self.multitest['last'] = t
1311 self.multitest['total'] = 0.0
1312 pprint('TEST (%s) START' % id)
1313 return
1314 dt = t - self.multitest['last']
1315 if not start:
1316 if idx == 0 and self.multitest['delay'] > 0:
1317 self.multitest['total'] += self.multitest['delay']
1318 pprint('TEST (%s) COMPLETE -- Duration %.1fs' % (id, dt))
1319 return
1320 self.multitest['total'] += dt
1321 self.multitest['last'] = t
1322 avg = self.multitest['total'] / idx
1323 if 'time' in self.multitest:
1324 left = finish - datetime.now()
1325 left -= timedelta(microseconds=left.microseconds)
1326 else:
1327 left = timedelta(seconds=((self.multitest['count'] - idx) * int(avg)))
1328 pprint('TEST (%s) START - Avg Duration %.1fs, Time left %s' % \
1329 (id, avg, str(left)))
1330 def multiinit(self, c, d):
1331 sz, unit = 'count', 'm'
1332 if c.endswith('d') or c.endswith('h') or c.endswith('m'):
1333 sz, unit, c = 'time', c[-1], c[:-1]
1334 self.multitest['run'] = True
1335 self.multitest[sz] = getArgInt('multi: n d (exec count)', c, 1, 1000000, False)
1336 self.multitest['delay'] = getArgInt('multi: n d (delay between tests)', d, 0, 3600, False)
1337 if unit == 'd':
1338 self.multitest[sz] *= 1440
1339 elif unit == 'h':
1340 self.multitest[sz] *= 60
1341 def displayControl(self, cmd):
1342 xset, ret = 'timeout 10 xset -d :0.0 {0}', 0
1343 if self.sudouser:
1344 xset = 'sudo -u %s %s' % (self.sudouser, xset)
1345 if cmd == 'init':
1346 ret = call(xset.format('dpms 0 0 0'), shell=True)
1347 if not ret:
1348 ret = call(xset.format('s off'), shell=True)
1349 elif cmd == 'reset':
1350 ret = call(xset.format('s reset'), shell=True)
1351 elif cmd in ['on', 'off', 'standby', 'suspend']:
1352 b4 = self.displayControl('stat')
1353 ret = call(xset.format('dpms force %s' % cmd), shell=True)
1354 if not ret:
1355 curr = self.displayControl('stat')
1356 self.vprint('Display Switched: %s -> %s' % (b4, curr))
1357 if curr != cmd:
1358 self.vprint('WARNING: Display failed to change to %s' % cmd)
1359 if ret:
1360 self.vprint('WARNING: Display failed to change to %s with xset' % cmd)
1361 return ret
1362 elif cmd == 'stat':
1363 fp = Popen(xset.format('q').split(' '), stdout=PIPE).stdout
1364 ret = 'unknown'
1365 for line in fp:
1366 m = re.match(r'[\s]*Monitor is (?P<m>.*)', ascii(line))
1367 if(m and len(m.group('m')) >= 2):
1368 out = m.group('m').lower()
1369 ret = out[3:] if out[0:2] == 'in' else out
1370 break
1371 fp.close()
1372 return ret
1373 def setRuntimeSuspend(self, before=True):
1374 if before:
1375 # runtime suspend disable or enable
1376 if self.rs > 0:
1377 self.rstgt, self.rsval, self.rsdir = 'on', 'auto', 'enabled'
1378 else:
1379 self.rstgt, self.rsval, self.rsdir = 'auto', 'on', 'disabled'
1380 pprint('CONFIGURING RUNTIME SUSPEND...')
1381 self.rslist = deviceInfo(self.rstgt)
1382 for i in self.rslist:
1383 self.setVal(self.rsval, i)
1384 pprint('runtime suspend %s on all devices (%d changed)' % (self.rsdir, len(self.rslist)))
1385 pprint('waiting 5 seconds...')
1386 time.sleep(5)
1387 else:
1388 # runtime suspend re-enable or re-disable
1389 for i in self.rslist:
1390 self.setVal(self.rstgt, i)
1391 pprint('runtime suspend settings restored on %d devices' % len(self.rslist))
1392 def start(self, pm):
1393 if self.useftrace:
1394 self.dlog('start ftrace tracing')
1395 self.fsetVal('1', 'tracing_on')
1396 if self.useprocmon:
1397 self.dlog('start the process monitor')
1398 pm.start()
1399 def stop(self, pm):
1400 if self.useftrace:
1401 if self.useprocmon:
1402 self.dlog('stop the process monitor')
1403 pm.stop()
1404 self.dlog('stop ftrace tracing')
1405 self.fsetVal('0', 'tracing_on')
1406
1407sysvals = SystemValues()
1408switchvalues = ['enable', 'disable', 'on', 'off', 'true', 'false', '1', '0']
1409switchoff = ['disable', 'off', 'false', '0']
1410suspendmodename = {
1411 'standby': 'standby (S1)',
1412 'freeze': 'freeze (S2idle)',
1413 'mem': 'suspend (S3)',
1414 'disk': 'hibernate (S4)'
1415}
1416
1417# Class: DevProps
1418# Description:
1419# Simple class which holds property values collected
1420# for all the devices used in the timeline.
1421class DevProps:
1422 def __init__(self):
1423 self.syspath = ''
1424 self.altname = ''
1425 self.isasync = True
1426 self.xtraclass = ''
1427 self.xtrainfo = ''
1428 def out(self, dev):
1429 return '%s,%s,%d;' % (dev, self.altname, self.isasync)
1430 def debug(self, dev):
1431 pprint('%s:\n\taltname = %s\n\t async = %s' % (dev, self.altname, self.isasync))
1432 def altName(self, dev):
1433 if not self.altname or self.altname == dev:
1434 return dev
1435 return '%s [%s]' % (self.altname, dev)
1436 def xtraClass(self):
1437 if self.xtraclass:
1438 return ' '+self.xtraclass
1439 if not self.isasync:
1440 return ' sync'
1441 return ''
1442 def xtraInfo(self):
1443 if self.xtraclass:
1444 return ' '+self.xtraclass
1445 if self.isasync:
1446 return ' (async)'
1447 return ' (sync)'
1448
1449# Class: DeviceNode
1450# Description:
1451# A container used to create a device hierachy, with a single root node
1452# and a tree of child nodes. Used by Data.deviceTopology()
1453class DeviceNode:
1454 def __init__(self, nodename, nodedepth):
1455 self.name = nodename
1456 self.children = []
1457 self.depth = nodedepth
1458
1459# Class: Data
1460# Description:
1461# The primary container for suspend/resume test data. There is one for
1462# each test run. The data is organized into a cronological hierarchy:
1463# Data.dmesg {
1464# phases {
1465# 10 sequential, non-overlapping phases of S/R
1466# contents: times for phase start/end, order/color data for html
1467# devlist {
1468# device callback or action list for this phase
1469# device {
1470# a single device callback or generic action
1471# contents: start/stop times, pid/cpu/driver info
1472# parents/children, html id for timeline/callgraph
1473# optionally includes an ftrace callgraph
1474# optionally includes dev/ps data
1475# }
1476# }
1477# }
1478# }
1479#
1480class Data:
1481 phasedef = {
1482 'suspend_prepare': {'order': 0, 'color': '#CCFFCC'},
1483 'suspend': {'order': 1, 'color': '#88FF88'},
1484 'suspend_late': {'order': 2, 'color': '#00AA00'},
1485 'suspend_noirq': {'order': 3, 'color': '#008888'},
1486 'suspend_machine': {'order': 4, 'color': '#0000FF'},
1487 'resume_machine': {'order': 5, 'color': '#FF0000'},
1488 'resume_noirq': {'order': 6, 'color': '#FF9900'},
1489 'resume_early': {'order': 7, 'color': '#FFCC00'},
1490 'resume': {'order': 8, 'color': '#FFFF88'},
1491 'resume_complete': {'order': 9, 'color': '#FFFFCC'},
1492 }
1493 errlist = {
1494 'HWERROR' : r'.*\[ *Hardware Error *\].*',
1495 'FWBUG' : r'.*\[ *Firmware Bug *\].*',
1496 'TASKFAIL': r'.*Freezing .*after *.*',
1497 'BUG' : r'(?i).*\bBUG\b.*',
1498 'ERROR' : r'(?i).*\bERROR\b.*',
1499 'WARNING' : r'(?i).*\bWARNING\b.*',
1500 'FAULT' : r'(?i).*\bFAULT\b.*',
1501 'FAIL' : r'(?i).*\bFAILED\b.*',
1502 'INVALID' : r'(?i).*\bINVALID\b.*',
1503 'CRASH' : r'(?i).*\bCRASHED\b.*',
1504 'TIMEOUT' : r'(?i).*\bTIMEOUT\b.*',
1505 'ABORT' : r'(?i).*\bABORT\b.*',
1506 'IRQ' : r'.*\bgenirq: .*',
1507 'ACPI' : r'.*\bACPI *(?P<b>[A-Za-z]*) *Error[: ].*',
1508 'DISKFULL': r'.*\bNo space left on device.*',
1509 'USBERR' : r'.*usb .*device .*, error [0-9-]*',
1510 'ATAERR' : r' *ata[0-9\.]*: .*failed.*',
1511 'MEIERR' : r' *mei.*: .*failed.*',
1512 'TPMERR' : r'(?i) *tpm *tpm[0-9]*: .*error.*',
1513 }
1514 def __init__(self, num):
1515 idchar = 'abcdefghij'
1516 self.start = 0.0 # test start
1517 self.end = 0.0 # test end
1518 self.hwstart = 0 # rtc test start
1519 self.hwend = 0 # rtc test end
1520 self.tSuspended = 0.0 # low-level suspend start
1521 self.tResumed = 0.0 # low-level resume start
1522 self.tKernSus = 0.0 # kernel level suspend start
1523 self.tKernRes = 0.0 # kernel level resume end
1524 self.fwValid = False # is firmware data available
1525 self.fwSuspend = 0 # time spent in firmware suspend
1526 self.fwResume = 0 # time spent in firmware resume
1527 self.html_device_id = 0
1528 self.stamp = 0
1529 self.outfile = ''
1530 self.kerror = False
1531 self.wifi = dict()
1532 self.turbostat = 0
1533 self.enterfail = ''
1534 self.currphase = ''
1535 self.pstl = dict() # process timeline
1536 self.testnumber = num
1537 self.idstr = idchar[num]
1538 self.dmesgtext = [] # dmesg text file in memory
1539 self.dmesg = dict() # root data structure
1540 self.errorinfo = {'suspend':[],'resume':[]}
1541 self.tLow = [] # time spent in low-level suspends (standby/freeze)
1542 self.devpids = []
1543 self.devicegroups = 0
1544 def sortedPhases(self):
1545 return sorted(self.dmesg, key=lambda k:self.dmesg[k]['order'])
1546 def initDevicegroups(self):
1547 # called when phases are all finished being added
1548 for phase in sorted(self.dmesg.keys()):
1549 if '*' in phase:
1550 p = phase.split('*')
1551 pnew = '%s%d' % (p[0], len(p))
1552 self.dmesg[pnew] = self.dmesg.pop(phase)
1553 self.devicegroups = []
1554 for phase in self.sortedPhases():
1555 self.devicegroups.append([phase])
1556 def nextPhase(self, phase, offset):
1557 order = self.dmesg[phase]['order'] + offset
1558 for p in self.dmesg:
1559 if self.dmesg[p]['order'] == order:
1560 return p
1561 return ''
1562 def lastPhase(self, depth=1):
1563 plist = self.sortedPhases()
1564 if len(plist) < depth:
1565 return ''
1566 return plist[-1*depth]
1567 def turbostatInfo(self):
1568 tp = TestProps()
1569 out = {'syslpi':'N/A','pkgpc10':'N/A'}
1570 for line in self.dmesgtext:
1571 m = re.match(tp.tstatfmt, line)
1572 if not m:
1573 continue
1574 for i in m.group('t').split('|'):
1575 if 'SYS%LPI' in i:
1576 out['syslpi'] = i.split('=')[-1]+'%'
1577 elif 'pc10' in i:
1578 out['pkgpc10'] = i.split('=')[-1]+'%'
1579 break
1580 return out
1581 def extractErrorInfo(self):
1582 lf = self.dmesgtext
1583 if len(self.dmesgtext) < 1 and sysvals.dmesgfile:
1584 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
1585 i = 0
1586 tp = TestProps()
1587 list = []
1588 for line in lf:
1589 i += 1
1590 if tp.stampInfo(line, sysvals):
1591 continue
1592 m = re.match(r'[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
1593 if not m:
1594 continue
1595 t = float(m.group('ktime'))
1596 if t < self.start or t > self.end:
1597 continue
1598 dir = 'suspend' if t < self.tSuspended else 'resume'
1599 msg = m.group('msg')
1600 if re.match(r'capability: warning: .*', msg):
1601 continue
1602 for err in self.errlist:
1603 if re.match(self.errlist[err], msg):
1604 list.append((msg, err, dir, t, i, i))
1605 self.kerror = True
1606 break
1607 tp.msglist = []
1608 for msg, type, dir, t, idx1, idx2 in list:
1609 tp.msglist.append(msg)
1610 self.errorinfo[dir].append((type, t, idx1, idx2))
1611 if self.kerror:
1612 sysvals.dmesglog = True
1613 if len(self.dmesgtext) < 1 and sysvals.dmesgfile:
1614 lf.close()
1615 return tp
1616 def setStart(self, time, msg=''):
1617 self.start = time
1618 if msg:
1619 try:
1620 self.hwstart = datetime.strptime(msg, sysvals.tmstart)
1621 except:
1622 self.hwstart = 0
1623 def setEnd(self, time, msg=''):
1624 self.end = time
1625 if msg:
1626 try:
1627 self.hwend = datetime.strptime(msg, sysvals.tmend)
1628 except:
1629 self.hwend = 0
1630 def isTraceEventOutsideDeviceCalls(self, pid, time):
1631 for phase in self.sortedPhases():
1632 list = self.dmesg[phase]['list']
1633 for dev in list:
1634 d = list[dev]
1635 if(d['pid'] == pid and time >= d['start'] and
1636 time < d['end']):
1637 return False
1638 return True
1639 def sourcePhase(self, start):
1640 for phase in self.sortedPhases():
1641 if 'machine' in phase:
1642 continue
1643 pend = self.dmesg[phase]['end']
1644 if start <= pend:
1645 return phase
1646 return 'resume_complete' if 'resume_complete' in self.dmesg else ''
1647 def sourceDevice(self, phaselist, start, end, pid, type):
1648 tgtdev = ''
1649 for phase in phaselist:
1650 list = self.dmesg[phase]['list']
1651 for devname in list:
1652 dev = list[devname]
1653 # pid must match
1654 if dev['pid'] != pid:
1655 continue
1656 devS = dev['start']
1657 devE = dev['end']
1658 if type == 'device':
1659 # device target event is entirely inside the source boundary
1660 if(start < devS or start >= devE or end <= devS or end > devE):
1661 continue
1662 elif type == 'thread':
1663 # thread target event will expand the source boundary
1664 if start < devS:
1665 dev['start'] = start
1666 if end > devE:
1667 dev['end'] = end
1668 tgtdev = dev
1669 break
1670 return tgtdev
1671 def addDeviceFunctionCall(self, displayname, kprobename, proc, pid, start, end, cdata, rdata):
1672 # try to place the call in a device
1673 phases = self.sortedPhases()
1674 tgtdev = self.sourceDevice(phases, start, end, pid, 'device')
1675 # calls with device pids that occur outside device bounds are dropped
1676 # TODO: include these somehow
1677 if not tgtdev and pid in self.devpids:
1678 return False
1679 # try to place the call in a thread
1680 if not tgtdev:
1681 tgtdev = self.sourceDevice(phases, start, end, pid, 'thread')
1682 # create new thread blocks, expand as new calls are found
1683 if not tgtdev:
1684 if proc == '<...>':
1685 threadname = 'kthread-%d' % (pid)
1686 else:
1687 threadname = '%s-%d' % (proc, pid)
1688 tgtphase = self.sourcePhase(start)
1689 if not tgtphase:
1690 return False
1691 self.newAction(tgtphase, threadname, pid, '', start, end, '', ' kth', '')
1692 return self.addDeviceFunctionCall(displayname, kprobename, proc, pid, start, end, cdata, rdata)
1693 # this should not happen
1694 if not tgtdev:
1695 sysvals.vprint('[%f - %f] %s-%d %s %s %s' % \
1696 (start, end, proc, pid, kprobename, cdata, rdata))
1697 return False
1698 # place the call data inside the src element of the tgtdev
1699 if('src' not in tgtdev):
1700 tgtdev['src'] = []
1701 dtf = sysvals.dev_tracefuncs
1702 ubiquitous = False
1703 if kprobename in dtf and 'ub' in dtf[kprobename]:
1704 ubiquitous = True
1705 mc = re.match(r'\(.*\) *(?P<args>.*)', cdata)
1706 mr = re.match(r'\((?P<caller>\S*).* arg1=(?P<ret>.*)', rdata)
1707 if mc and mr:
1708 c = mr.group('caller').split('+')[0]
1709 a = mc.group('args').strip()
1710 r = mr.group('ret')
1711 if len(r) > 6:
1712 r = ''
1713 else:
1714 r = 'ret=%s ' % r
1715 if ubiquitous and c in dtf and 'ub' in dtf[c]:
1716 return False
1717 else:
1718 return False
1719 color = sysvals.kprobeColor(kprobename)
1720 e = DevFunction(displayname, a, c, r, start, end, ubiquitous, proc, pid, color)
1721 tgtdev['src'].append(e)
1722 return True
1723 def overflowDevices(self):
1724 # get a list of devices that extend beyond the end of this test run
1725 devlist = []
1726 for phase in self.sortedPhases():
1727 list = self.dmesg[phase]['list']
1728 for devname in list:
1729 dev = list[devname]
1730 if dev['end'] > self.end:
1731 devlist.append(dev)
1732 return devlist
1733 def mergeOverlapDevices(self, devlist):
1734 # merge any devices that overlap devlist
1735 for dev in devlist:
1736 devname = dev['name']
1737 for phase in self.sortedPhases():
1738 list = self.dmesg[phase]['list']
1739 if devname not in list:
1740 continue
1741 tdev = list[devname]
1742 o = min(dev['end'], tdev['end']) - max(dev['start'], tdev['start'])
1743 if o <= 0:
1744 continue
1745 dev['end'] = tdev['end']
1746 if 'src' not in dev or 'src' not in tdev:
1747 continue
1748 dev['src'] += tdev['src']
1749 del list[devname]
1750 def usurpTouchingThread(self, name, dev):
1751 # the caller test has priority of this thread, give it to him
1752 for phase in self.sortedPhases():
1753 list = self.dmesg[phase]['list']
1754 if name in list:
1755 tdev = list[name]
1756 if tdev['start'] - dev['end'] < 0.1:
1757 dev['end'] = tdev['end']
1758 if 'src' not in dev:
1759 dev['src'] = []
1760 if 'src' in tdev:
1761 dev['src'] += tdev['src']
1762 del list[name]
1763 break
1764 def stitchTouchingThreads(self, testlist):
1765 # merge any threads between tests that touch
1766 for phase in self.sortedPhases():
1767 list = self.dmesg[phase]['list']
1768 for devname in list:
1769 dev = list[devname]
1770 if 'htmlclass' not in dev or 'kth' not in dev['htmlclass']:
1771 continue
1772 for data in testlist:
1773 data.usurpTouchingThread(devname, dev)
1774 def optimizeDevSrc(self):
1775 # merge any src call loops to reduce timeline size
1776 for phase in self.sortedPhases():
1777 list = self.dmesg[phase]['list']
1778 for dev in list:
1779 if 'src' not in list[dev]:
1780 continue
1781 src = list[dev]['src']
1782 p = 0
1783 for e in sorted(src, key=lambda event: event.time):
1784 if not p or not e.repeat(p):
1785 p = e
1786 continue
1787 # e is another iteration of p, move it into p
1788 p.end = e.end
1789 p.length = p.end - p.time
1790 p.count += 1
1791 src.remove(e)
1792 def trimTimeVal(self, t, t0, dT, left):
1793 if left:
1794 if(t > t0):
1795 if(t - dT < t0):
1796 return t0
1797 return t - dT
1798 else:
1799 return t
1800 else:
1801 if(t < t0 + dT):
1802 if(t > t0):
1803 return t0 + dT
1804 return t + dT
1805 else:
1806 return t
1807 def trimTime(self, t0, dT, left):
1808 self.tSuspended = self.trimTimeVal(self.tSuspended, t0, dT, left)
1809 self.tResumed = self.trimTimeVal(self.tResumed, t0, dT, left)
1810 self.start = self.trimTimeVal(self.start, t0, dT, left)
1811 self.tKernSus = self.trimTimeVal(self.tKernSus, t0, dT, left)
1812 self.tKernRes = self.trimTimeVal(self.tKernRes, t0, dT, left)
1813 self.end = self.trimTimeVal(self.end, t0, dT, left)
1814 for phase in self.sortedPhases():
1815 p = self.dmesg[phase]
1816 p['start'] = self.trimTimeVal(p['start'], t0, dT, left)
1817 p['end'] = self.trimTimeVal(p['end'], t0, dT, left)
1818 list = p['list']
1819 for name in list:
1820 d = list[name]
1821 d['start'] = self.trimTimeVal(d['start'], t0, dT, left)
1822 d['end'] = self.trimTimeVal(d['end'], t0, dT, left)
1823 d['length'] = d['end'] - d['start']
1824 if('ftrace' in d):
1825 cg = d['ftrace']
1826 cg.start = self.trimTimeVal(cg.start, t0, dT, left)
1827 cg.end = self.trimTimeVal(cg.end, t0, dT, left)
1828 for line in cg.list:
1829 line.time = self.trimTimeVal(line.time, t0, dT, left)
1830 if('src' in d):
1831 for e in d['src']:
1832 e.time = self.trimTimeVal(e.time, t0, dT, left)
1833 e.end = self.trimTimeVal(e.end, t0, dT, left)
1834 e.length = e.end - e.time
1835 if('cpuexec' in d):
1836 cpuexec = dict()
1837 for e in d['cpuexec']:
1838 c0, cN = e
1839 c0 = self.trimTimeVal(c0, t0, dT, left)
1840 cN = self.trimTimeVal(cN, t0, dT, left)
1841 cpuexec[(c0, cN)] = d['cpuexec'][e]
1842 d['cpuexec'] = cpuexec
1843 for dir in ['suspend', 'resume']:
1844 list = []
1845 for e in self.errorinfo[dir]:
1846 type, tm, idx1, idx2 = e
1847 tm = self.trimTimeVal(tm, t0, dT, left)
1848 list.append((type, tm, idx1, idx2))
1849 self.errorinfo[dir] = list
1850 def trimFreezeTime(self, tZero):
1851 # trim out any standby or freeze clock time
1852 lp = ''
1853 for phase in self.sortedPhases():
1854 if 'resume_machine' in phase and 'suspend_machine' in lp:
1855 tS, tR = self.dmesg[lp]['end'], self.dmesg[phase]['start']
1856 tL = tR - tS
1857 if tL <= 0:
1858 continue
1859 left = True if tR > tZero else False
1860 self.trimTime(tS, tL, left)
1861 if 'waking' in self.dmesg[lp]:
1862 tCnt = self.dmesg[lp]['waking'][0]
1863 if self.dmesg[lp]['waking'][1] >= 0.001:
1864 tTry = '%.0f' % (round(self.dmesg[lp]['waking'][1] * 1000))
1865 else:
1866 tTry = '%.3f' % (self.dmesg[lp]['waking'][1] * 1000)
1867 text = '%.0f (%s ms waking %d times)' % (tL * 1000, tTry, tCnt)
1868 else:
1869 text = '%.0f' % (tL * 1000)
1870 self.tLow.append(text)
1871 lp = phase
1872 def getMemTime(self):
1873 if not self.hwstart or not self.hwend:
1874 return
1875 stime = (self.tSuspended - self.start) * 1000000
1876 rtime = (self.end - self.tResumed) * 1000000
1877 hws = self.hwstart + timedelta(microseconds=stime)
1878 hwr = self.hwend - timedelta(microseconds=rtime)
1879 self.tLow.append('%.0f'%((hwr - hws).total_seconds() * 1000))
1880 def getTimeValues(self):
1881 s = (self.tSuspended - self.tKernSus) * 1000
1882 r = (self.tKernRes - self.tResumed) * 1000
1883 return (max(s, 0), max(r, 0))
1884 def setPhase(self, phase, ktime, isbegin, order=-1):
1885 if(isbegin):
1886 # phase start over current phase
1887 if self.currphase:
1888 if 'resume_machine' not in self.currphase:
1889 sysvals.vprint('WARNING: phase %s failed to end' % self.currphase)
1890 self.dmesg[self.currphase]['end'] = ktime
1891 phases = self.dmesg.keys()
1892 color = self.phasedef[phase]['color']
1893 count = len(phases) if order < 0 else order
1894 # create unique name for every new phase
1895 while phase in phases:
1896 phase += '*'
1897 self.dmesg[phase] = {'list': dict(), 'start': -1.0, 'end': -1.0,
1898 'row': 0, 'color': color, 'order': count}
1899 self.dmesg[phase]['start'] = ktime
1900 self.currphase = phase
1901 else:
1902 # phase end without a start
1903 if phase not in self.currphase:
1904 if self.currphase:
1905 sysvals.vprint('WARNING: %s ended instead of %s, ftrace corruption?' % (phase, self.currphase))
1906 else:
1907 sysvals.vprint('WARNING: %s ended without a start, ftrace corruption?' % phase)
1908 return phase
1909 phase = self.currphase
1910 self.dmesg[phase]['end'] = ktime
1911 self.currphase = ''
1912 return phase
1913 def sortedDevices(self, phase):
1914 list = self.dmesg[phase]['list']
1915 return sorted(list, key=lambda k:list[k]['start'])
1916 def fixupInitcalls(self, phase):
1917 # if any calls never returned, clip them at system resume end
1918 phaselist = self.dmesg[phase]['list']
1919 for devname in phaselist:
1920 dev = phaselist[devname]
1921 if(dev['end'] < 0):
1922 for p in self.sortedPhases():
1923 if self.dmesg[p]['end'] > dev['start']:
1924 dev['end'] = self.dmesg[p]['end']
1925 break
1926 sysvals.vprint('%s (%s): callback didnt return' % (devname, phase))
1927 def deviceFilter(self, devicefilter):
1928 for phase in self.sortedPhases():
1929 list = self.dmesg[phase]['list']
1930 rmlist = []
1931 for name in list:
1932 keep = False
1933 for filter in devicefilter:
1934 if filter in name or \
1935 ('drv' in list[name] and filter in list[name]['drv']):
1936 keep = True
1937 if not keep:
1938 rmlist.append(name)
1939 for name in rmlist:
1940 del list[name]
1941 def fixupInitcallsThatDidntReturn(self):
1942 # if any calls never returned, clip them at system resume end
1943 for phase in self.sortedPhases():
1944 self.fixupInitcalls(phase)
1945 def phaseOverlap(self, phases):
1946 rmgroups = []
1947 newgroup = []
1948 for group in self.devicegroups:
1949 for phase in phases:
1950 if phase not in group:
1951 continue
1952 for p in group:
1953 if p not in newgroup:
1954 newgroup.append(p)
1955 if group not in rmgroups:
1956 rmgroups.append(group)
1957 for group in rmgroups:
1958 self.devicegroups.remove(group)
1959 self.devicegroups.append(newgroup)
1960 def newActionGlobal(self, name, start, end, pid=-1, color=''):
1961 # which phase is this device callback or action in
1962 phases = self.sortedPhases()
1963 targetphase = 'none'
1964 htmlclass = ''
1965 overlap = 0.0
1966 myphases = []
1967 for phase in phases:
1968 pstart = self.dmesg[phase]['start']
1969 pend = self.dmesg[phase]['end']
1970 # see if the action overlaps this phase
1971 o = max(0, min(end, pend) - max(start, pstart))
1972 if o > 0:
1973 myphases.append(phase)
1974 # set the target phase to the one that overlaps most
1975 if o > overlap:
1976 if overlap > 0 and phase == 'post_resume':
1977 continue
1978 targetphase = phase
1979 overlap = o
1980 # if no target phase was found, pin it to the edge
1981 if targetphase == 'none':
1982 p0start = self.dmesg[phases[0]]['start']
1983 if start <= p0start:
1984 targetphase = phases[0]
1985 else:
1986 targetphase = phases[-1]
1987 if pid == -2:
1988 htmlclass = ' bg'
1989 elif pid == -3:
1990 htmlclass = ' ps'
1991 if len(myphases) > 1:
1992 htmlclass = ' bg'
1993 self.phaseOverlap(myphases)
1994 if targetphase in phases:
1995 newname = self.newAction(targetphase, name, pid, '', start, end, '', htmlclass, color)
1996 return (targetphase, newname)
1997 return False
1998 def newAction(self, phase, name, pid, parent, start, end, drv, htmlclass='', color=''):
1999 # new device callback for a specific phase
2000 self.html_device_id += 1
2001 devid = '%s%d' % (self.idstr, self.html_device_id)
2002 list = self.dmesg[phase]['list']
2003 length = -1.0
2004 if(start >= 0 and end >= 0):
2005 length = end - start
2006 if pid >= -2:
2007 i = 2
2008 origname = name
2009 while(name in list):
2010 name = '%s[%d]' % (origname, i)
2011 i += 1
2012 list[name] = {'name': name, 'start': start, 'end': end, 'pid': pid,
2013 'par': parent, 'length': length, 'row': 0, 'id': devid, 'drv': drv }
2014 if htmlclass:
2015 list[name]['htmlclass'] = htmlclass
2016 if color:
2017 list[name]['color'] = color
2018 return name
2019 def findDevice(self, phase, name):
2020 list = self.dmesg[phase]['list']
2021 mydev = ''
2022 for devname in sorted(list):
2023 if name == devname or re.match(r'^%s\[(?P<num>[0-9]*)\]$' % name, devname):
2024 mydev = devname
2025 if mydev:
2026 return list[mydev]
2027 return False
2028 def deviceChildren(self, devname, phase):
2029 devlist = []
2030 list = self.dmesg[phase]['list']
2031 for child in list:
2032 if(list[child]['par'] == devname):
2033 devlist.append(child)
2034 return devlist
2035 def maxDeviceNameSize(self, phase):
2036 size = 0
2037 for name in self.dmesg[phase]['list']:
2038 if len(name) > size:
2039 size = len(name)
2040 return size
2041 def printDetails(self):
2042 sysvals.vprint('Timeline Details:')
2043 sysvals.vprint(' test start: %f' % self.start)
2044 sysvals.vprint('kernel suspend start: %f' % self.tKernSus)
2045 tS = tR = False
2046 for phase in self.sortedPhases():
2047 devlist = self.dmesg[phase]['list']
2048 dc, ps, pe = len(devlist), self.dmesg[phase]['start'], self.dmesg[phase]['end']
2049 if not tS and ps >= self.tSuspended:
2050 sysvals.vprint(' machine suspended: %f' % self.tSuspended)
2051 tS = True
2052 if not tR and ps >= self.tResumed:
2053 sysvals.vprint(' machine resumed: %f' % self.tResumed)
2054 tR = True
2055 sysvals.vprint('%20s: %f - %f (%d devices)' % (phase, ps, pe, dc))
2056 if sysvals.devdump:
2057 sysvals.vprint(''.join('-' for i in range(80)))
2058 maxname = '%d' % self.maxDeviceNameSize(phase)
2059 fmt = '%3d) %'+maxname+'s - %f - %f'
2060 c = 1
2061 for name in sorted(devlist):
2062 s = devlist[name]['start']
2063 e = devlist[name]['end']
2064 sysvals.vprint(fmt % (c, name, s, e))
2065 c += 1
2066 sysvals.vprint(''.join('-' for i in range(80)))
2067 sysvals.vprint(' kernel resume end: %f' % self.tKernRes)
2068 sysvals.vprint(' test end: %f' % self.end)
2069 def deviceChildrenAllPhases(self, devname):
2070 devlist = []
2071 for phase in self.sortedPhases():
2072 list = self.deviceChildren(devname, phase)
2073 for dev in sorted(list):
2074 if dev not in devlist:
2075 devlist.append(dev)
2076 return devlist
2077 def masterTopology(self, name, list, depth):
2078 node = DeviceNode(name, depth)
2079 for cname in list:
2080 # avoid recursions
2081 if name == cname:
2082 continue
2083 clist = self.deviceChildrenAllPhases(cname)
2084 cnode = self.masterTopology(cname, clist, depth+1)
2085 node.children.append(cnode)
2086 return node
2087 def printTopology(self, node):
2088 html = ''
2089 if node.name:
2090 info = ''
2091 drv = ''
2092 for phase in self.sortedPhases():
2093 list = self.dmesg[phase]['list']
2094 if node.name in list:
2095 s = list[node.name]['start']
2096 e = list[node.name]['end']
2097 if list[node.name]['drv']:
2098 drv = ' {'+list[node.name]['drv']+'}'
2099 info += ('<li>%s: %.3fms</li>' % (phase, (e-s)*1000))
2100 html += '<li><b>'+node.name+drv+'</b>'
2101 if info:
2102 html += '<ul>'+info+'</ul>'
2103 html += '</li>'
2104 if len(node.children) > 0:
2105 html += '<ul>'
2106 for cnode in node.children:
2107 html += self.printTopology(cnode)
2108 html += '</ul>'
2109 return html
2110 def rootDeviceList(self):
2111 # list of devices graphed
2112 real = []
2113 for phase in self.sortedPhases():
2114 list = self.dmesg[phase]['list']
2115 for dev in sorted(list):
2116 if list[dev]['pid'] >= 0 and dev not in real:
2117 real.append(dev)
2118 # list of top-most root devices
2119 rootlist = []
2120 for phase in self.sortedPhases():
2121 list = self.dmesg[phase]['list']
2122 for dev in sorted(list):
2123 pdev = list[dev]['par']
2124 pid = list[dev]['pid']
2125 if(pid < 0 or re.match(r'[0-9]*-[0-9]*\.[0-9]*[\.0-9]*\:[\.0-9]*$', pdev)):
2126 continue
2127 if pdev and pdev not in real and pdev not in rootlist:
2128 rootlist.append(pdev)
2129 return rootlist
2130 def deviceTopology(self):
2131 rootlist = self.rootDeviceList()
2132 master = self.masterTopology('', rootlist, 0)
2133 return self.printTopology(master)
2134 def selectTimelineDevices(self, widfmt, tTotal, mindevlen):
2135 # only select devices that will actually show up in html
2136 self.tdevlist = dict()
2137 for phase in self.dmesg:
2138 devlist = []
2139 list = self.dmesg[phase]['list']
2140 for dev in list:
2141 length = (list[dev]['end'] - list[dev]['start']) * 1000
2142 width = widfmt % (((list[dev]['end']-list[dev]['start'])*100)/tTotal)
2143 if length >= mindevlen:
2144 devlist.append(dev)
2145 self.tdevlist[phase] = devlist
2146 def addHorizontalDivider(self, devname, devend):
2147 phase = 'suspend_prepare'
2148 self.newAction(phase, devname, -2, '', \
2149 self.start, devend, '', ' sec', '')
2150 if phase not in self.tdevlist:
2151 self.tdevlist[phase] = []
2152 self.tdevlist[phase].append(devname)
2153 d = DevItem(0, phase, self.dmesg[phase]['list'][devname])
2154 return d
2155 def addProcessUsageEvent(self, name, times):
2156 # get the start and end times for this process
2157 cpuexec = dict()
2158 tlast = start = end = -1
2159 for t in sorted(times):
2160 if tlast < 0:
2161 tlast = t
2162 continue
2163 if name in self.pstl[t] and self.pstl[t][name] > 0:
2164 if start < 0:
2165 start = tlast
2166 end, key = t, (tlast, t)
2167 maxj = (t - tlast) * 1024.0
2168 cpuexec[key] = min(1.0, float(self.pstl[t][name]) / maxj)
2169 tlast = t
2170 if start < 0 or end < 0:
2171 return
2172 # add a new action for this process and get the object
2173 out = self.newActionGlobal(name, start, end, -3)
2174 if out:
2175 phase, devname = out
2176 dev = self.dmesg[phase]['list'][devname]
2177 dev['cpuexec'] = cpuexec
2178 def createProcessUsageEvents(self):
2179 # get an array of process names and times
2180 proclist = {'sus': dict(), 'res': dict()}
2181 tdata = {'sus': [], 'res': []}
2182 for t in sorted(self.pstl):
2183 dir = 'sus' if t < self.tSuspended else 'res'
2184 for ps in sorted(self.pstl[t]):
2185 if ps not in proclist[dir]:
2186 proclist[dir][ps] = 0
2187 tdata[dir].append(t)
2188 # process the events for suspend and resume
2189 if len(proclist['sus']) > 0 or len(proclist['res']) > 0:
2190 sysvals.vprint('Process Execution:')
2191 for dir in ['sus', 'res']:
2192 for ps in sorted(proclist[dir]):
2193 self.addProcessUsageEvent(ps, tdata[dir])
2194 def handleEndMarker(self, time, msg=''):
2195 dm = self.dmesg
2196 self.setEnd(time, msg)
2197 self.initDevicegroups()
2198 # give suspend_prepare an end if needed
2199 if 'suspend_prepare' in dm and dm['suspend_prepare']['end'] < 0:
2200 dm['suspend_prepare']['end'] = time
2201 # assume resume machine ends at next phase start
2202 if 'resume_machine' in dm and dm['resume_machine']['end'] < 0:
2203 np = self.nextPhase('resume_machine', 1)
2204 if np:
2205 dm['resume_machine']['end'] = dm[np]['start']
2206 # if kernel resume end not found, assume its the end marker
2207 if self.tKernRes == 0.0:
2208 self.tKernRes = time
2209 # if kernel suspend start not found, assume its the end marker
2210 if self.tKernSus == 0.0:
2211 self.tKernSus = time
2212 # set resume complete to end at end marker
2213 if 'resume_complete' in dm:
2214 dm['resume_complete']['end'] = time
2215 def initcall_debug_call(self, line, quick=False):
2216 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2217 r'PM: *calling .* @ (?P<n>.*), parent: (?P<p>.*)', line)
2218 if not m:
2219 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2220 r'calling .* @ (?P<n>.*), parent: (?P<p>.*)', line)
2221 if not m:
2222 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) calling '+\
2223 r'(?P<f>.*)\+ @ (?P<n>.*), parent: (?P<p>.*)', line)
2224 if m:
2225 return True if quick else m.group('t', 'f', 'n', 'p')
2226 return False if quick else ('', '', '', '')
2227 def initcall_debug_return(self, line, quick=False):
2228 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: PM: '+\
2229 r'.* returned (?P<r>[0-9]*) after (?P<dt>[0-9]*) usecs', line)
2230 if not m:
2231 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2232 r'.* returned (?P<r>[0-9]*) after (?P<dt>[0-9]*) usecs', line)
2233 if not m:
2234 m = re.match(r'.*(\[ *)(?P<t>[0-9\.]*)(\]) call '+\
2235 r'(?P<f>.*)\+ returned .* after (?P<dt>.*) usecs', line)
2236 if m:
2237 return True if quick else m.group('t', 'f', 'dt')
2238 return False if quick else ('', '', '')
2239 def debugPrint(self):
2240 for p in self.sortedPhases():
2241 list = self.dmesg[p]['list']
2242 for devname in sorted(list):
2243 dev = list[devname]
2244 if 'ftrace' in dev:
2245 dev['ftrace'].debugPrint(' [%s]' % devname)
2246
2247# Class: DevFunction
2248# Description:
2249# A container for kprobe function data we want in the dev timeline
2250class DevFunction:
2251 def __init__(self, name, args, caller, ret, start, end, u, proc, pid, color):
2252 self.row = 0
2253 self.count = 1
2254 self.name = name
2255 self.args = args
2256 self.caller = caller
2257 self.ret = ret
2258 self.time = start
2259 self.length = end - start
2260 self.end = end
2261 self.ubiquitous = u
2262 self.proc = proc
2263 self.pid = pid
2264 self.color = color
2265 def title(self):
2266 cnt = ''
2267 if self.count > 1:
2268 cnt = '(x%d)' % self.count
2269 l = '%0.3fms' % (self.length * 1000)
2270 if self.ubiquitous:
2271 title = '%s(%s)%s <- %s, %s(%s)' % \
2272 (self.name, self.args, cnt, self.caller, self.ret, l)
2273 else:
2274 title = '%s(%s) %s%s(%s)' % (self.name, self.args, self.ret, cnt, l)
2275 return title.replace('"', '')
2276 def text(self):
2277 if self.count > 1:
2278 text = '%s(x%d)' % (self.name, self.count)
2279 else:
2280 text = self.name
2281 return text
2282 def repeat(self, tgt):
2283 # is the tgt call just a repeat of this call (e.g. are we in a loop)
2284 dt = self.time - tgt.end
2285 # only combine calls if -all- attributes are identical
2286 if tgt.caller == self.caller and \
2287 tgt.name == self.name and tgt.args == self.args and \
2288 tgt.proc == self.proc and tgt.pid == self.pid and \
2289 tgt.ret == self.ret and dt >= 0 and \
2290 dt <= sysvals.callloopmaxgap and \
2291 self.length < sysvals.callloopmaxlen:
2292 return True
2293 return False
2294
2295# Class: FTraceLine
2296# Description:
2297# A container for a single line of ftrace data. There are six basic types:
2298# callgraph line:
2299# call: " dpm_run_callback() {"
2300# return: " }"
2301# leaf: " dpm_run_callback();"
2302# trace event:
2303# tracing_mark_write: SUSPEND START or RESUME COMPLETE
2304# suspend_resume: phase or custom exec block data
2305# device_pm_callback: device callback info
2306class FTraceLine:
2307 def __init__(self, t, m='', d=''):
2308 self.length = 0.0
2309 self.fcall = False
2310 self.freturn = False
2311 self.fevent = False
2312 self.fkprobe = False
2313 self.depth = 0
2314 self.name = ''
2315 self.type = ''
2316 self.time = float(t)
2317 if not m and not d:
2318 return
2319 # is this a trace event
2320 if(d == 'traceevent' or re.match(r'^ *\/\* *(?P<msg>.*) \*\/ *$', m)):
2321 if(d == 'traceevent'):
2322 # nop format trace event
2323 msg = m
2324 else:
2325 # function_graph format trace event
2326 em = re.match(r'^ *\/\* *(?P<msg>.*) \*\/ *$', m)
2327 msg = em.group('msg')
2328
2329 emm = re.match(r'^(?P<call>.*?): (?P<msg>.*)', msg)
2330 if(emm):
2331 self.name = emm.group('msg')
2332 self.type = emm.group('call')
2333 else:
2334 self.name = msg
2335 km = re.match(r'^(?P<n>.*)_cal$', self.type)
2336 if km:
2337 self.fcall = True
2338 self.fkprobe = True
2339 self.type = km.group('n')
2340 return
2341 km = re.match(r'^(?P<n>.*)_ret$', self.type)
2342 if km:
2343 self.freturn = True
2344 self.fkprobe = True
2345 self.type = km.group('n')
2346 return
2347 self.fevent = True
2348 return
2349 # convert the duration to seconds
2350 if(d):
2351 self.length = float(d)/1000000
2352 # the indentation determines the depth
2353 match = re.match(r'^(?P<d> *)(?P<o>.*)$', m)
2354 if(not match):
2355 return
2356 self.depth = self.getDepth(match.group('d'))
2357 m = match.group('o')
2358 # function return
2359 if(m[0] == '}'):
2360 self.freturn = True
2361 if(len(m) > 1):
2362 # includes comment with function name
2363 match = re.match(r'^} *\/\* *(?P<n>.*) *\*\/$', m)
2364 if(match):
2365 self.name = match.group('n').strip()
2366 # function call
2367 else:
2368 self.fcall = True
2369 # function call with children
2370 if(m[-1] == '{'):
2371 match = re.match(r'^(?P<n>.*) *\(.*', m)
2372 if(match):
2373 self.name = match.group('n').strip()
2374 # function call with no children (leaf)
2375 elif(m[-1] == ';'):
2376 self.freturn = True
2377 match = re.match(r'^(?P<n>.*) *\(.*', m)
2378 if(match):
2379 self.name = match.group('n').strip()
2380 # something else (possibly a trace marker)
2381 else:
2382 self.name = m
2383 def isCall(self):
2384 return self.fcall and not self.freturn
2385 def isReturn(self):
2386 return self.freturn and not self.fcall
2387 def isLeaf(self):
2388 return self.fcall and self.freturn
2389 def getDepth(self, str):
2390 return len(str)/2
2391 def debugPrint(self, info=''):
2392 if self.isLeaf():
2393 pprint(' -- %12.6f (depth=%02d): %s(); (%.3f us) %s' % (self.time, \
2394 self.depth, self.name, self.length*1000000, info))
2395 elif self.freturn:
2396 pprint(' -- %12.6f (depth=%02d): %s} (%.3f us) %s' % (self.time, \
2397 self.depth, self.name, self.length*1000000, info))
2398 else:
2399 pprint(' -- %12.6f (depth=%02d): %s() { (%.3f us) %s' % (self.time, \
2400 self.depth, self.name, self.length*1000000, info))
2401 def startMarker(self):
2402 # Is this the starting line of a suspend?
2403 if not self.fevent:
2404 return False
2405 if sysvals.usetracemarkers:
2406 if(self.name.startswith('SUSPEND START')):
2407 return True
2408 return False
2409 else:
2410 if(self.type == 'suspend_resume' and
2411 re.match(r'suspend_enter\[.*\] begin', self.name)):
2412 return True
2413 return False
2414 def endMarker(self):
2415 # Is this the ending line of a resume?
2416 if not self.fevent:
2417 return False
2418 if sysvals.usetracemarkers:
2419 if(self.name.startswith('RESUME COMPLETE')):
2420 return True
2421 return False
2422 else:
2423 if(self.type == 'suspend_resume' and
2424 re.match(r'thaw_processes\[.*\] end', self.name)):
2425 return True
2426 return False
2427
2428# Class: FTraceCallGraph
2429# Description:
2430# A container for the ftrace callgraph of a single recursive function.
2431# This can be a dpm_run_callback, dpm_prepare, or dpm_complete callgraph
2432# Each instance is tied to a single device in a single phase, and is
2433# comprised of an ordered list of FTraceLine objects
2434class FTraceCallGraph:
2435 vfname = 'missing_function_name'
2436 def __init__(self, pid, sv):
2437 self.id = ''
2438 self.invalid = False
2439 self.name = ''
2440 self.partial = False
2441 self.ignore = False
2442 self.start = -1.0
2443 self.end = -1.0
2444 self.list = []
2445 self.depth = 0
2446 self.pid = pid
2447 self.sv = sv
2448 def addLine(self, line):
2449 # if this is already invalid, just leave
2450 if(self.invalid):
2451 if(line.depth == 0 and line.freturn):
2452 return 1
2453 return 0
2454 # invalidate on bad depth
2455 if(self.depth < 0):
2456 self.invalidate(line)
2457 return 0
2458 # ignore data til we return to the current depth
2459 if self.ignore:
2460 if line.depth > self.depth:
2461 return 0
2462 else:
2463 self.list[-1].freturn = True
2464 self.list[-1].length = line.time - self.list[-1].time
2465 self.ignore = False
2466 # if this is a return at self.depth, no more work is needed
2467 if line.depth == self.depth and line.isReturn():
2468 if line.depth == 0:
2469 self.end = line.time
2470 return 1
2471 return 0
2472 # compare current depth with this lines pre-call depth
2473 prelinedep = line.depth
2474 if line.isReturn():
2475 prelinedep += 1
2476 last = 0
2477 lasttime = line.time
2478 if len(self.list) > 0:
2479 last = self.list[-1]
2480 lasttime = last.time
2481 if last.isLeaf():
2482 lasttime += last.length
2483 # handle low misalignments by inserting returns
2484 mismatch = prelinedep - self.depth
2485 warning = self.sv.verbose and abs(mismatch) > 1
2486 info = []
2487 if mismatch < 0:
2488 idx = 0
2489 # add return calls to get the depth down
2490 while prelinedep < self.depth:
2491 self.depth -= 1
2492 if idx == 0 and last and last.isCall():
2493 # special case, turn last call into a leaf
2494 last.depth = self.depth
2495 last.freturn = True
2496 last.length = line.time - last.time
2497 if warning:
2498 info.append(('[make leaf]', last))
2499 else:
2500 vline = FTraceLine(lasttime)
2501 vline.depth = self.depth
2502 vline.name = self.vfname
2503 vline.freturn = True
2504 self.list.append(vline)
2505 if warning:
2506 if idx == 0:
2507 info.append(('', last))
2508 info.append(('[add return]', vline))
2509 idx += 1
2510 if warning:
2511 info.append(('', line))
2512 # handle high misalignments by inserting calls
2513 elif mismatch > 0:
2514 idx = 0
2515 if warning:
2516 info.append(('', last))
2517 # add calls to get the depth up
2518 while prelinedep > self.depth:
2519 if idx == 0 and line.isReturn():
2520 # special case, turn this return into a leaf
2521 line.fcall = True
2522 prelinedep -= 1
2523 if warning:
2524 info.append(('[make leaf]', line))
2525 else:
2526 vline = FTraceLine(lasttime)
2527 vline.depth = self.depth
2528 vline.name = self.vfname
2529 vline.fcall = True
2530 self.list.append(vline)
2531 self.depth += 1
2532 if not last:
2533 self.start = vline.time
2534 if warning:
2535 info.append(('[add call]', vline))
2536 idx += 1
2537 if warning and ('[make leaf]', line) not in info:
2538 info.append(('', line))
2539 if warning:
2540 pprint('WARNING: ftrace data missing, corrections made:')
2541 for i in info:
2542 t, obj = i
2543 if obj:
2544 obj.debugPrint(t)
2545 # process the call and set the new depth
2546 skipadd = False
2547 md = self.sv.max_graph_depth
2548 if line.isCall():
2549 # ignore blacklisted/overdepth funcs
2550 if (md and self.depth >= md - 1) or (line.name in self.sv.cgblacklist):
2551 self.ignore = True
2552 else:
2553 self.depth += 1
2554 elif line.isReturn():
2555 self.depth -= 1
2556 # remove blacklisted/overdepth/empty funcs that slipped through
2557 if (last and last.isCall() and last.depth == line.depth) or \
2558 (md and last and last.depth >= md) or \
2559 (line.name in self.sv.cgblacklist):
2560 while len(self.list) > 0 and self.list[-1].depth > line.depth:
2561 self.list.pop(-1)
2562 if len(self.list) == 0:
2563 self.invalid = True
2564 return 1
2565 self.list[-1].freturn = True
2566 self.list[-1].length = line.time - self.list[-1].time
2567 self.list[-1].name = line.name
2568 skipadd = True
2569 if len(self.list) < 1:
2570 self.start = line.time
2571 # check for a mismatch that returned all the way to callgraph end
2572 res = 1
2573 if mismatch < 0 and self.list[-1].depth == 0 and self.list[-1].freturn:
2574 line = self.list[-1]
2575 skipadd = True
2576 res = -1
2577 if not skipadd:
2578 self.list.append(line)
2579 if(line.depth == 0 and line.freturn):
2580 if(self.start < 0):
2581 self.start = line.time
2582 self.end = line.time
2583 if line.fcall:
2584 self.end += line.length
2585 if self.list[0].name == self.vfname:
2586 self.invalid = True
2587 if res == -1:
2588 self.partial = True
2589 return res
2590 return 0
2591 def invalidate(self, line):
2592 if(len(self.list) > 0):
2593 first = self.list[0]
2594 self.list = []
2595 self.list.append(first)
2596 self.invalid = True
2597 id = 'task %s' % (self.pid)
2598 window = '(%f - %f)' % (self.start, line.time)
2599 if(self.depth < 0):
2600 pprint('Data misalignment for '+id+\
2601 ' (buffer overflow), ignoring this callback')
2602 else:
2603 pprint('Too much data for '+id+\
2604 ' '+window+', ignoring this callback')
2605 def slice(self, dev):
2606 minicg = FTraceCallGraph(dev['pid'], self.sv)
2607 minicg.name = self.name
2608 mydepth = -1
2609 good = False
2610 for l in self.list:
2611 if(l.time < dev['start'] or l.time > dev['end']):
2612 continue
2613 if mydepth < 0:
2614 if l.name == 'mutex_lock' and l.freturn:
2615 mydepth = l.depth
2616 continue
2617 elif l.depth == mydepth and l.name == 'mutex_unlock' and l.fcall:
2618 good = True
2619 break
2620 l.depth -= mydepth
2621 minicg.addLine(l)
2622 if not good or len(minicg.list) < 1:
2623 return 0
2624 return minicg
2625 def repair(self, enddepth):
2626 # bring the depth back to 0 with additional returns
2627 fixed = False
2628 last = self.list[-1]
2629 for i in reversed(range(enddepth)):
2630 t = FTraceLine(last.time)
2631 t.depth = i
2632 t.freturn = True
2633 fixed = self.addLine(t)
2634 if fixed != 0:
2635 self.end = last.time
2636 return True
2637 return False
2638 def postProcess(self):
2639 if len(self.list) > 0:
2640 self.name = self.list[0].name
2641 stack = dict()
2642 cnt = 0
2643 last = 0
2644 for l in self.list:
2645 # ftrace bug: reported duration is not reliable
2646 # check each leaf and clip it at max possible length
2647 if last and last.isLeaf():
2648 if last.length > l.time - last.time:
2649 last.length = l.time - last.time
2650 if l.isCall():
2651 stack[l.depth] = l
2652 cnt += 1
2653 elif l.isReturn():
2654 if(l.depth not in stack):
2655 if self.sv.verbose:
2656 pprint('Post Process Error: Depth missing')
2657 l.debugPrint()
2658 return False
2659 # calculate call length from call/return lines
2660 cl = stack[l.depth]
2661 cl.length = l.time - cl.time
2662 if cl.name == self.vfname:
2663 cl.name = l.name
2664 stack.pop(l.depth)
2665 l.length = 0
2666 cnt -= 1
2667 last = l
2668 if(cnt == 0):
2669 # trace caught the whole call tree
2670 return True
2671 elif(cnt < 0):
2672 if self.sv.verbose:
2673 pprint('Post Process Error: Depth is less than 0')
2674 return False
2675 # trace ended before call tree finished
2676 return self.repair(cnt)
2677 def deviceMatch(self, pid, data):
2678 found = ''
2679 # add the callgraph data to the device hierarchy
2680 borderphase = {
2681 'dpm_prepare': 'suspend_prepare',
2682 'dpm_complete': 'resume_complete'
2683 }
2684 if(self.name in borderphase):
2685 p = borderphase[self.name]
2686 list = data.dmesg[p]['list']
2687 for devname in list:
2688 dev = list[devname]
2689 if(pid == dev['pid'] and
2690 self.start <= dev['start'] and
2691 self.end >= dev['end']):
2692 cg = self.slice(dev)
2693 if cg:
2694 dev['ftrace'] = cg
2695 found = devname
2696 return found
2697 for p in data.sortedPhases():
2698 if(data.dmesg[p]['start'] <= self.start and
2699 self.start <= data.dmesg[p]['end']):
2700 list = data.dmesg[p]['list']
2701 for devname in sorted(list, key=lambda k:list[k]['start']):
2702 dev = list[devname]
2703 if(pid == dev['pid'] and
2704 self.start <= dev['start'] and
2705 self.end >= dev['end']):
2706 dev['ftrace'] = self
2707 found = devname
2708 break
2709 break
2710 return found
2711 def newActionFromFunction(self, data):
2712 name = self.name
2713 if name in ['dpm_run_callback', 'dpm_prepare', 'dpm_complete']:
2714 return
2715 fs = self.start
2716 fe = self.end
2717 if fs < data.start or fe > data.end:
2718 return
2719 phase = ''
2720 for p in data.sortedPhases():
2721 if(data.dmesg[p]['start'] <= self.start and
2722 self.start < data.dmesg[p]['end']):
2723 phase = p
2724 break
2725 if not phase:
2726 return
2727 out = data.newActionGlobal(name, fs, fe, -2)
2728 if out:
2729 phase, myname = out
2730 data.dmesg[phase]['list'][myname]['ftrace'] = self
2731 def debugPrint(self, info=''):
2732 pprint('%s pid=%d [%f - %f] %.3f us' % \
2733 (self.name, self.pid, self.start, self.end,
2734 (self.end - self.start)*1000000))
2735 for l in self.list:
2736 if l.isLeaf():
2737 pprint('%f (%02d): %s(); (%.3f us)%s' % (l.time, \
2738 l.depth, l.name, l.length*1000000, info))
2739 elif l.freturn:
2740 pprint('%f (%02d): %s} (%.3f us)%s' % (l.time, \
2741 l.depth, l.name, l.length*1000000, info))
2742 else:
2743 pprint('%f (%02d): %s() { (%.3f us)%s' % (l.time, \
2744 l.depth, l.name, l.length*1000000, info))
2745 pprint(' ')
2746
2747class DevItem:
2748 def __init__(self, test, phase, dev):
2749 self.test = test
2750 self.phase = phase
2751 self.dev = dev
2752 def isa(self, cls):
2753 if 'htmlclass' in self.dev and cls in self.dev['htmlclass']:
2754 return True
2755 return False
2756
2757# Class: Timeline
2758# Description:
2759# A container for a device timeline which calculates
2760# all the html properties to display it correctly
2761class Timeline:
2762 html_tblock = '<div id="block{0}" class="tblock" style="left:{1}%;width:{2}%;"><div class="tback" style="height:{3}px"></div>\n'
2763 html_device = '<div id="{0}" title="{1}" class="thread{7}" style="left:{2}%;top:{3}px;height:{4}px;width:{5}%;{8}">{6}</div>\n'
2764 html_phase = '<div class="phase" style="left:{0}%;width:{1}%;top:{2}px;height:{3}px;background:{4}">{5}</div>\n'
2765 html_phaselet = '<div id="{0}" class="phaselet" style="left:{1}%;width:{2}%;background:{3}"></div>\n'
2766 html_legend = '<div id="p{3}" class="square" style="left:{0}%;background:{1}"> {2}</div>\n'
2767 def __init__(self, rowheight, scaleheight):
2768 self.html = ''
2769 self.height = 0 # total timeline height
2770 self.scaleH = scaleheight # timescale (top) row height
2771 self.rowH = rowheight # device row height
2772 self.bodyH = 0 # body height
2773 self.rows = 0 # total timeline rows
2774 self.rowlines = dict()
2775 self.rowheight = dict()
2776 def createHeader(self, sv, stamp):
2777 if(not stamp['time']):
2778 return
2779 self.html += '<div class="version"><a href="https://www.intel.com/content/www/'+\
2780 'us/en/developer/topic-technology/open/pm-graph/overview.html">%s v%s</a></div>' \
2781 % (sv.title, sv.version)
2782 if sv.logmsg and sv.testlog:
2783 self.html += '<button id="showtest" class="logbtn btnfmt">log</button>'
2784 if sv.dmesglog:
2785 self.html += '<button id="showdmesg" class="logbtn btnfmt">dmesg</button>'
2786 if sv.ftracelog:
2787 self.html += '<button id="showftrace" class="logbtn btnfmt">ftrace</button>'
2788 headline_stamp = '<div class="stamp">{0} {1} {2} {3}</div>\n'
2789 self.html += headline_stamp.format(stamp['host'], stamp['kernel'],
2790 stamp['mode'], stamp['time'])
2791 if 'man' in stamp and 'plat' in stamp and 'cpu' in stamp and \
2792 stamp['man'] and stamp['plat'] and stamp['cpu']:
2793 headline_sysinfo = '<div class="stamp sysinfo">{0} {1} <i>with</i> {2}</div>\n'
2794 self.html += headline_sysinfo.format(stamp['man'], stamp['plat'], stamp['cpu'])
2795
2796 # Function: getDeviceRows
2797 # Description:
2798 # determine how may rows the device funcs will take
2799 # Arguments:
2800 # rawlist: the list of devices/actions for a single phase
2801 # Output:
2802 # The total number of rows needed to display this phase of the timeline
2803 def getDeviceRows(self, rawlist):
2804 # clear all rows and set them to undefined
2805 sortdict = dict()
2806 for item in rawlist:
2807 item.row = -1
2808 sortdict[item] = item.length
2809 sortlist = sorted(sortdict, key=sortdict.get, reverse=True)
2810 remaining = len(sortlist)
2811 rowdata = dict()
2812 row = 1
2813 # try to pack each row with as many ranges as possible
2814 while(remaining > 0):
2815 if(row not in rowdata):
2816 rowdata[row] = []
2817 for i in sortlist:
2818 if(i.row >= 0):
2819 continue
2820 s = i.time
2821 e = i.time + i.length
2822 valid = True
2823 for ritem in rowdata[row]:
2824 rs = ritem.time
2825 re = ritem.time + ritem.length
2826 if(not (((s <= rs) and (e <= rs)) or
2827 ((s >= re) and (e >= re)))):
2828 valid = False
2829 break
2830 if(valid):
2831 rowdata[row].append(i)
2832 i.row = row
2833 remaining -= 1
2834 row += 1
2835 return row
2836 # Function: getPhaseRows
2837 # Description:
2838 # Organize the timeline entries into the smallest
2839 # number of rows possible, with no entry overlapping
2840 # Arguments:
2841 # devlist: the list of devices/actions in a group of contiguous phases
2842 # Output:
2843 # The total number of rows needed to display this phase of the timeline
2844 def getPhaseRows(self, devlist, row=0, sortby='length'):
2845 # clear all rows and set them to undefined
2846 remaining = len(devlist)
2847 rowdata = dict()
2848 sortdict = dict()
2849 myphases = []
2850 # initialize all device rows to -1 and calculate devrows
2851 for item in devlist:
2852 dev = item.dev
2853 tp = (item.test, item.phase)
2854 if tp not in myphases:
2855 myphases.append(tp)
2856 dev['row'] = -1
2857 if sortby == 'start':
2858 # sort by start 1st, then length 2nd
2859 sortdict[item] = (-1*float(dev['start']), float(dev['end']) - float(dev['start']))
2860 else:
2861 # sort by length 1st, then name 2nd
2862 sortdict[item] = (float(dev['end']) - float(dev['start']), item.dev['name'])
2863 if 'src' in dev:
2864 dev['devrows'] = self.getDeviceRows(dev['src'])
2865 # sort the devlist by length so that large items graph on top
2866 sortlist = sorted(sortdict, key=sortdict.get, reverse=True)
2867 orderedlist = []
2868 for item in sortlist:
2869 if item.dev['pid'] == -2:
2870 orderedlist.append(item)
2871 for item in sortlist:
2872 if item not in orderedlist:
2873 orderedlist.append(item)
2874 # try to pack each row with as many devices as possible
2875 while(remaining > 0):
2876 rowheight = 1
2877 if(row not in rowdata):
2878 rowdata[row] = []
2879 for item in orderedlist:
2880 dev = item.dev
2881 if(dev['row'] < 0):
2882 s = dev['start']
2883 e = dev['end']
2884 valid = True
2885 for ritem in rowdata[row]:
2886 rs = ritem.dev['start']
2887 re = ritem.dev['end']
2888 if(not (((s <= rs) and (e <= rs)) or
2889 ((s >= re) and (e >= re)))):
2890 valid = False
2891 break
2892 if(valid):
2893 rowdata[row].append(item)
2894 dev['row'] = row
2895 remaining -= 1
2896 if 'devrows' in dev and dev['devrows'] > rowheight:
2897 rowheight = dev['devrows']
2898 for t, p in myphases:
2899 if t not in self.rowlines or t not in self.rowheight:
2900 self.rowlines[t] = dict()
2901 self.rowheight[t] = dict()
2902 if p not in self.rowlines[t] or p not in self.rowheight[t]:
2903 self.rowlines[t][p] = dict()
2904 self.rowheight[t][p] = dict()
2905 rh = self.rowH
2906 # section headers should use a different row height
2907 if len(rowdata[row]) == 1 and \
2908 'htmlclass' in rowdata[row][0].dev and \
2909 'sec' in rowdata[row][0].dev['htmlclass']:
2910 rh = 15
2911 self.rowlines[t][p][row] = rowheight
2912 self.rowheight[t][p][row] = rowheight * rh
2913 row += 1
2914 if(row > self.rows):
2915 self.rows = int(row)
2916 return row
2917 def phaseRowHeight(self, test, phase, row):
2918 return self.rowheight[test][phase][row]
2919 def phaseRowTop(self, test, phase, row):
2920 top = 0
2921 for i in sorted(self.rowheight[test][phase]):
2922 if i >= row:
2923 break
2924 top += self.rowheight[test][phase][i]
2925 return top
2926 def calcTotalRows(self):
2927 # Calculate the heights and offsets for the header and rows
2928 maxrows = 0
2929 standardphases = []
2930 for t in self.rowlines:
2931 for p in self.rowlines[t]:
2932 total = 0
2933 for i in sorted(self.rowlines[t][p]):
2934 total += self.rowlines[t][p][i]
2935 if total > maxrows:
2936 maxrows = total
2937 if total == len(self.rowlines[t][p]):
2938 standardphases.append((t, p))
2939 self.height = self.scaleH + (maxrows*self.rowH)
2940 self.bodyH = self.height - self.scaleH
2941 # if there is 1 line per row, draw them the standard way
2942 for t, p in standardphases:
2943 for i in sorted(self.rowheight[t][p]):
2944 self.rowheight[t][p][i] = float(self.bodyH)/len(self.rowlines[t][p])
2945 def createZoomBox(self, mode='command', testcount=1):
2946 # Create bounding box, add buttons
2947 html_zoombox = '<center><button id="zoomin">ZOOM IN +</button><button id="zoomout">ZOOM OUT -</button><button id="zoomdef">ZOOM 1:1</button></center>\n'
2948 html_timeline = '<div id="dmesgzoombox" class="zoombox">\n<div id="{0}" class="timeline" style="height:{1}px">\n'
2949 html_devlist1 = '<button id="devlist1" class="devlist" style="float:left;">Device Detail{0}</button>'
2950 html_devlist2 = '<button id="devlist2" class="devlist" style="float:right;">Device Detail2</button>\n'
2951 if mode != 'command':
2952 if testcount > 1:
2953 self.html += html_devlist2
2954 self.html += html_devlist1.format('1')
2955 else:
2956 self.html += html_devlist1.format('')
2957 self.html += html_zoombox
2958 self.html += html_timeline.format('dmesg', self.height)
2959 # Function: createTimeScale
2960 # Description:
2961 # Create the timescale for a timeline block
2962 # Arguments:
2963 # m0: start time (mode begin)
2964 # mMax: end time (mode end)
2965 # tTotal: total timeline time
2966 # mode: suspend or resume
2967 # Output:
2968 # The html code needed to display the time scale
2969 def createTimeScale(self, m0, mMax, tTotal, mode):
2970 timescale = '<div class="t" style="right:{0}%">{1}</div>\n'
2971 rline = '<div class="t" style="left:0;border-left:1px solid black;border-right:0;">{0}</div>\n'
2972 output = '<div class="timescale">\n'
2973 # set scale for timeline
2974 mTotal = mMax - m0
2975 tS = 0.1
2976 if(tTotal <= 0):
2977 return output+'</div>\n'
2978 if(tTotal > 4):
2979 tS = 1
2980 divTotal = int(mTotal/tS) + 1
2981 divEdge = (mTotal - tS*(divTotal-1))*100/mTotal
2982 for i in range(divTotal):
2983 htmlline = ''
2984 if(mode == 'suspend'):
2985 pos = '%0.3f' % (100 - ((float(i)*tS*100)/mTotal) - divEdge)
2986 val = '%0.fms' % (float(i-divTotal+1)*tS*1000)
2987 if(i == divTotal - 1):
2988 val = mode
2989 htmlline = timescale.format(pos, val)
2990 else:
2991 pos = '%0.3f' % (100 - ((float(i)*tS*100)/mTotal))
2992 val = '%0.fms' % (float(i)*tS*1000)
2993 htmlline = timescale.format(pos, val)
2994 if(i == 0):
2995 htmlline = rline.format(mode)
2996 output += htmlline
2997 self.html += output+'</div>\n'
2998
2999# Class: TestProps
3000# Description:
3001# A list of values describing the properties of these test runs
3002class TestProps:
3003 stampfmt = r'# [a-z]*-(?P<m>[0-9]{2})(?P<d>[0-9]{2})(?P<y>[0-9]{2})-'+\
3004 r'(?P<H>[0-9]{2})(?P<M>[0-9]{2})(?P<S>[0-9]{2})'+\
3005 r' (?P<host>.*) (?P<mode>.*) (?P<kernel>.*)$'
3006 wififmt = r'^# wifi *(?P<d>\S*) *(?P<s>\S*) *(?P<t>[0-9\.]+).*'
3007 tstatfmt = r'^# turbostat (?P<t>\S*)'
3008 testerrfmt = r'^# enter_sleep_error (?P<e>.*)'
3009 sysinfofmt = r'^# sysinfo .*'
3010 cmdlinefmt = r'^# command \| (?P<cmd>.*)'
3011 kparamsfmt = r'^# kparams \| (?P<kp>.*)'
3012 devpropfmt = r'# Device Properties: .*'
3013 pinfofmt = r'# platform-(?P<val>[a-z,A-Z,0-9,_]*): (?P<info>.*)'
3014 tracertypefmt = r'# tracer: (?P<t>.*)'
3015 firmwarefmt = r'# fwsuspend (?P<s>[0-9]*) fwresume (?P<r>[0-9]*)$'
3016 procexecfmt = r'ps - (?P<ps>.*)$'
3017 procmultifmt = r'@(?P<n>[0-9]*)\|(?P<ps>.*)$'
3018 ftrace_line_fmt_fg = \
3019 r'^ *(?P<time>[0-9\.]*) *\| *(?P<cpu>[0-9]*)\)'+\
3020 r' *(?P<proc>.*)-(?P<pid>[0-9]*) *\|'+\
3021 r'[ +!#\*@$]*(?P<dur>[0-9\.]*) .*\| (?P<msg>.*)'
3022 ftrace_line_fmt_nop = \
3023 r' *(?P<proc>.*)-(?P<pid>[0-9]*) *\[(?P<cpu>[0-9]*)\] *'+\
3024 r'(?P<flags>\S*) *(?P<time>[0-9\.]*): *'+\
3025 r'(?P<msg>.*)'
3026 machinesuspend = r'machine_suspend\[.*'
3027 multiproclist = dict()
3028 multiproctime = 0.0
3029 multiproccnt = 0
3030 def __init__(self):
3031 self.stamp = ''
3032 self.sysinfo = ''
3033 self.cmdline = ''
3034 self.testerror = []
3035 self.turbostat = []
3036 self.wifi = []
3037 self.fwdata = []
3038 self.ftrace_line_fmt = self.ftrace_line_fmt_nop
3039 self.cgformat = False
3040 self.data = 0
3041 self.ktemp = dict()
3042 def setTracerType(self, tracer):
3043 if(tracer == 'function_graph'):
3044 self.cgformat = True
3045 self.ftrace_line_fmt = self.ftrace_line_fmt_fg
3046 elif(tracer == 'nop'):
3047 self.ftrace_line_fmt = self.ftrace_line_fmt_nop
3048 else:
3049 doError('Invalid tracer format: [%s]' % tracer)
3050 def stampInfo(self, line, sv):
3051 if re.match(self.stampfmt, line):
3052 self.stamp = line
3053 return True
3054 elif re.match(self.sysinfofmt, line):
3055 self.sysinfo = line
3056 return True
3057 elif re.match(self.tstatfmt, line):
3058 self.turbostat.append(line)
3059 return True
3060 elif re.match(self.wififmt, line):
3061 self.wifi.append(line)
3062 return True
3063 elif re.match(self.testerrfmt, line):
3064 self.testerror.append(line)
3065 return True
3066 elif re.match(self.firmwarefmt, line):
3067 self.fwdata.append(line)
3068 return True
3069 elif(re.match(self.devpropfmt, line)):
3070 self.parseDevprops(line, sv)
3071 return True
3072 elif(re.match(self.pinfofmt, line)):
3073 self.parsePlatformInfo(line, sv)
3074 return True
3075 m = re.match(self.cmdlinefmt, line)
3076 if m:
3077 self.cmdline = m.group('cmd')
3078 return True
3079 m = re.match(self.tracertypefmt, line)
3080 if(m):
3081 self.setTracerType(m.group('t'))
3082 return True
3083 return False
3084 def parseStamp(self, data, sv):
3085 # global test data
3086 m = re.match(self.stampfmt, self.stamp)
3087 if not self.stamp or not m:
3088 doError('data does not include the expected stamp')
3089 data.stamp = {'time': '', 'host': '', 'mode': ''}
3090 dt = datetime(int(m.group('y'))+2000, int(m.group('m')),
3091 int(m.group('d')), int(m.group('H')), int(m.group('M')),
3092 int(m.group('S')))
3093 data.stamp['time'] = dt.strftime('%B %d %Y, %I:%M:%S %p')
3094 data.stamp['host'] = m.group('host')
3095 data.stamp['mode'] = m.group('mode')
3096 data.stamp['kernel'] = m.group('kernel')
3097 if re.match(self.sysinfofmt, self.sysinfo):
3098 for f in self.sysinfo.split('|'):
3099 if '#' in f:
3100 continue
3101 tmp = f.strip().split(':', 1)
3102 key = tmp[0]
3103 val = tmp[1]
3104 data.stamp[key] = val
3105 sv.hostname = data.stamp['host']
3106 sv.suspendmode = data.stamp['mode']
3107 if sv.suspendmode == 'freeze':
3108 self.machinesuspend = r'timekeeping_freeze\[.*'
3109 else:
3110 self.machinesuspend = r'machine_suspend\[.*'
3111 if sv.suspendmode == 'command' and sv.ftracefile != '':
3112 modes = ['on', 'freeze', 'standby', 'mem', 'disk']
3113 fp = sv.openlog(sv.ftracefile, 'r')
3114 for line in fp:
3115 m = re.match(r'.* machine_suspend\[(?P<mode>.*)\]', line)
3116 if m and m.group('mode') in ['1', '2', '3', '4']:
3117 sv.suspendmode = modes[int(m.group('mode'))]
3118 data.stamp['mode'] = sv.suspendmode
3119 break
3120 fp.close()
3121 sv.cmdline = self.cmdline
3122 if not sv.stamp:
3123 sv.stamp = data.stamp
3124 # firmware data
3125 if sv.suspendmode == 'mem' and len(self.fwdata) > data.testnumber:
3126 m = re.match(self.firmwarefmt, self.fwdata[data.testnumber])
3127 if m:
3128 data.fwSuspend, data.fwResume = int(m.group('s')), int(m.group('r'))
3129 if(data.fwSuspend > 0 or data.fwResume > 0):
3130 data.fwValid = True
3131 # turbostat data
3132 if len(self.turbostat) > data.testnumber:
3133 m = re.match(self.tstatfmt, self.turbostat[data.testnumber])
3134 if m:
3135 data.turbostat = m.group('t')
3136 # wifi data
3137 if len(self.wifi) > data.testnumber:
3138 m = re.match(self.wififmt, self.wifi[data.testnumber])
3139 if m:
3140 data.wifi = {'dev': m.group('d'), 'stat': m.group('s'),
3141 'time': float(m.group('t'))}
3142 data.stamp['wifi'] = m.group('d')
3143 # sleep mode enter errors
3144 if len(self.testerror) > data.testnumber:
3145 m = re.match(self.testerrfmt, self.testerror[data.testnumber])
3146 if m:
3147 data.enterfail = m.group('e')
3148 def devprops(self, data):
3149 props = dict()
3150 devlist = data.split(';')
3151 for dev in devlist:
3152 f = dev.split(',')
3153 if len(f) < 3:
3154 continue
3155 dev = f[0]
3156 props[dev] = DevProps()
3157 props[dev].altname = f[1]
3158 if int(f[2]):
3159 props[dev].isasync = True
3160 else:
3161 props[dev].isasync = False
3162 return props
3163 def parseDevprops(self, line, sv):
3164 idx = line.index(': ') + 2
3165 if idx >= len(line):
3166 return
3167 props = self.devprops(line[idx:])
3168 if sv.suspendmode == 'command' and 'testcommandstring' in props:
3169 sv.testcommand = props['testcommandstring'].altname
3170 sv.devprops = props
3171 def parsePlatformInfo(self, line, sv):
3172 m = re.match(self.pinfofmt, line)
3173 if not m:
3174 return
3175 name, info = m.group('val'), m.group('info')
3176 if name == 'devinfo':
3177 sv.devprops = self.devprops(sv.b64unzip(info))
3178 return
3179 elif name == 'testcmd':
3180 sv.testcommand = info
3181 return
3182 field = info.split('|')
3183 if len(field) < 2:
3184 return
3185 cmdline = field[0].strip()
3186 output = sv.b64unzip(field[1].strip())
3187 sv.platinfo.append([name, cmdline, output])
3188
3189# Class: TestRun
3190# Description:
3191# A container for a suspend/resume test run. This is necessary as
3192# there could be more than one, and they need to be separate.
3193class TestRun:
3194 def __init__(self, dataobj):
3195 self.data = dataobj
3196 self.ftemp = dict()
3197 self.ttemp = dict()
3198
3199class ProcessMonitor:
3200 maxchars = 512
3201 def __init__(self):
3202 self.proclist = dict()
3203 self.running = False
3204 def procstat(self):
3205 c = ['cat /proc/[1-9]*/stat 2>/dev/null']
3206 process = Popen(c, shell=True, stdout=PIPE)
3207 running = dict()
3208 for line in process.stdout:
3209 data = ascii(line).split()
3210 pid = data[0]
3211 name = re.sub('[()]', '', data[1])
3212 user = int(data[13])
3213 kern = int(data[14])
3214 kjiff = ujiff = 0
3215 if pid not in self.proclist:
3216 self.proclist[pid] = {'name' : name, 'user' : user, 'kern' : kern}
3217 else:
3218 val = self.proclist[pid]
3219 ujiff = user - val['user']
3220 kjiff = kern - val['kern']
3221 val['user'] = user
3222 val['kern'] = kern
3223 if ujiff > 0 or kjiff > 0:
3224 running[pid] = ujiff + kjiff
3225 process.wait()
3226 out = ['']
3227 for pid in running:
3228 jiffies = running[pid]
3229 val = self.proclist[pid]
3230 if len(out[-1]) > self.maxchars:
3231 out.append('')
3232 elif len(out[-1]) > 0:
3233 out[-1] += ','
3234 out[-1] += '%s-%s %d' % (val['name'], pid, jiffies)
3235 if len(out) > 1:
3236 for line in out:
3237 sysvals.fsetVal('ps - @%d|%s' % (len(out), line), 'trace_marker')
3238 else:
3239 sysvals.fsetVal('ps - %s' % out[0], 'trace_marker')
3240 def processMonitor(self, tid):
3241 while self.running:
3242 self.procstat()
3243 def start(self):
3244 self.thread = Thread(target=self.processMonitor, args=(0,))
3245 self.running = True
3246 self.thread.start()
3247 def stop(self):
3248 self.running = False
3249
3250# ----------------- FUNCTIONS --------------------
3251
3252# Function: doesTraceLogHaveTraceEvents
3253# Description:
3254# Quickly determine if the ftrace log has all of the trace events,
3255# markers, and/or kprobes required for primary parsing.
3256def doesTraceLogHaveTraceEvents():
3257 kpcheck = ['_cal: (', '_ret: (']
3258 techeck = ['suspend_resume', 'device_pm_callback', 'tracing_mark_write']
3259 tmcheck = ['SUSPEND START', 'RESUME COMPLETE']
3260 sysvals.usekprobes = False
3261 fp = sysvals.openlog(sysvals.ftracefile, 'r')
3262 for line in fp:
3263 # check for kprobes
3264 if not sysvals.usekprobes:
3265 for i in kpcheck:
3266 if i in line:
3267 sysvals.usekprobes = True
3268 # check for all necessary trace events
3269 check = techeck[:]
3270 for i in techeck:
3271 if i in line:
3272 check.remove(i)
3273 techeck = check
3274 # check for all necessary trace markers
3275 check = tmcheck[:]
3276 for i in tmcheck:
3277 if i in line:
3278 check.remove(i)
3279 tmcheck = check
3280 fp.close()
3281 sysvals.usetraceevents = True if len(techeck) < 3 else False
3282 sysvals.usetracemarkers = True if len(tmcheck) == 0 else False
3283
3284# Function: appendIncompleteTraceLog
3285# Description:
3286# Adds callgraph data which lacks trace event data. This is only
3287# for timelines generated from 3.15 or older
3288# Arguments:
3289# testruns: the array of Data objects obtained from parseKernelLog
3290def appendIncompleteTraceLog(testruns):
3291 # create TestRun vessels for ftrace parsing
3292 testcnt = len(testruns)
3293 testidx = 0
3294 testrun = []
3295 for data in testruns:
3296 testrun.append(TestRun(data))
3297
3298 # extract the callgraph and traceevent data
3299 sysvals.vprint('Analyzing the ftrace data (%s)...' % \
3300 os.path.basename(sysvals.ftracefile))
3301 tp = TestProps()
3302 tf = sysvals.openlog(sysvals.ftracefile, 'r')
3303 data = 0
3304 for line in tf:
3305 # remove any latent carriage returns
3306 line = line.replace('\r\n', '')
3307 if tp.stampInfo(line, sysvals):
3308 continue
3309 # parse only valid lines, if this is not one move on
3310 m = re.match(tp.ftrace_line_fmt, line)
3311 if(not m):
3312 continue
3313 # gather the basic message data from the line
3314 m_time = m.group('time')
3315 m_pid = m.group('pid')
3316 m_msg = m.group('msg')
3317 if(tp.cgformat):
3318 m_param3 = m.group('dur')
3319 else:
3320 m_param3 = 'traceevent'
3321 if(m_time and m_pid and m_msg):
3322 t = FTraceLine(m_time, m_msg, m_param3)
3323 pid = int(m_pid)
3324 else:
3325 continue
3326 # the line should be a call, return, or event
3327 if(not t.fcall and not t.freturn and not t.fevent):
3328 continue
3329 # look for the suspend start marker
3330 if(t.startMarker()):
3331 data = testrun[testidx].data
3332 tp.parseStamp(data, sysvals)
3333 data.setStart(t.time, t.name)
3334 continue
3335 if(not data):
3336 continue
3337 # find the end of resume
3338 if(t.endMarker()):
3339 data.setEnd(t.time, t.name)
3340 testidx += 1
3341 if(testidx >= testcnt):
3342 break
3343 continue
3344 # trace event processing
3345 if(t.fevent):
3346 continue
3347 # call/return processing
3348 elif sysvals.usecallgraph:
3349 # create a callgraph object for the data
3350 if(pid not in testrun[testidx].ftemp):
3351 testrun[testidx].ftemp[pid] = []
3352 testrun[testidx].ftemp[pid].append(FTraceCallGraph(pid, sysvals))
3353 # when the call is finished, see which device matches it
3354 cg = testrun[testidx].ftemp[pid][-1]
3355 res = cg.addLine(t)
3356 if(res != 0):
3357 testrun[testidx].ftemp[pid].append(FTraceCallGraph(pid, sysvals))
3358 if(res == -1):
3359 testrun[testidx].ftemp[pid][-1].addLine(t)
3360 tf.close()
3361
3362 for test in testrun:
3363 # add the callgraph data to the device hierarchy
3364 for pid in test.ftemp:
3365 for cg in test.ftemp[pid]:
3366 if len(cg.list) < 1 or cg.invalid or (cg.end - cg.start == 0):
3367 continue
3368 if(not cg.postProcess()):
3369 id = 'task %s cpu %s' % (pid, m.group('cpu'))
3370 sysvals.vprint('Sanity check failed for '+\
3371 id+', ignoring this callback')
3372 continue
3373 callstart = cg.start
3374 callend = cg.end
3375 for p in test.data.sortedPhases():
3376 if(test.data.dmesg[p]['start'] <= callstart and
3377 callstart <= test.data.dmesg[p]['end']):
3378 list = test.data.dmesg[p]['list']
3379 for devname in list:
3380 dev = list[devname]
3381 if(pid == dev['pid'] and
3382 callstart <= dev['start'] and
3383 callend >= dev['end']):
3384 dev['ftrace'] = cg
3385 break
3386
3387# Function: loadTraceLog
3388# Description:
3389# load the ftrace file into memory and fix up any ordering issues
3390# Output:
3391# TestProps instance and an array of lines in proper order
3392def loadTraceLog():
3393 tp, data, lines, trace = TestProps(), dict(), [], []
3394 tf = sysvals.openlog(sysvals.ftracefile, 'r')
3395 for line in tf:
3396 # remove any latent carriage returns
3397 line = line.replace('\r\n', '')
3398 if tp.stampInfo(line, sysvals):
3399 continue
3400 # ignore all other commented lines
3401 if line[0] == '#':
3402 continue
3403 # ftrace line: parse only valid lines
3404 m = re.match(tp.ftrace_line_fmt, line)
3405 if(not m):
3406 continue
3407 dur = m.group('dur') if tp.cgformat else 'traceevent'
3408 info = (m.group('time'), m.group('proc'), m.group('pid'),
3409 m.group('msg'), dur)
3410 # group the data by timestamp
3411 t = float(info[0])
3412 if t in data:
3413 data[t].append(info)
3414 else:
3415 data[t] = [info]
3416 # we only care about trace event ordering
3417 if (info[3].startswith('suspend_resume:') or \
3418 info[3].startswith('tracing_mark_write:')) and t not in trace:
3419 trace.append(t)
3420 tf.close()
3421 for t in sorted(data):
3422 first, last, blk = [], [], data[t]
3423 if len(blk) > 1 and t in trace:
3424 # move certain lines to the start or end of a timestamp block
3425 for i in range(len(blk)):
3426 if 'SUSPEND START' in blk[i][3]:
3427 first.append(i)
3428 elif re.match(r'.* timekeeping_freeze.*begin', blk[i][3]):
3429 last.append(i)
3430 elif re.match(r'.* timekeeping_freeze.*end', blk[i][3]):
3431 first.append(i)
3432 elif 'RESUME COMPLETE' in blk[i][3]:
3433 last.append(i)
3434 if len(first) == 1 and len(last) == 0:
3435 blk.insert(0, blk.pop(first[0]))
3436 elif len(last) == 1 and len(first) == 0:
3437 blk.append(blk.pop(last[0]))
3438 for info in blk:
3439 lines.append(info)
3440 return (tp, lines)
3441
3442# Function: parseTraceLog
3443# Description:
3444# Analyze an ftrace log output file generated from this app during
3445# the execution phase. Used when the ftrace log is the primary data source
3446# and includes the suspend_resume and device_pm_callback trace events
3447# The ftrace filename is taken from sysvals
3448# Output:
3449# An array of Data objects
3450def parseTraceLog(live=False):
3451 sysvals.vprint('Analyzing the ftrace data (%s)...' % \
3452 os.path.basename(sysvals.ftracefile))
3453 if(os.path.exists(sysvals.ftracefile) == False):
3454 doError('%s does not exist' % sysvals.ftracefile)
3455 if not live:
3456 sysvals.setupAllKprobes()
3457 ksuscalls = ['ksys_sync', 'pm_prepare_console']
3458 krescalls = ['pm_restore_console']
3459 tracewatch = ['irq_wakeup']
3460 if sysvals.usekprobes:
3461 tracewatch += ['sync_filesystems', 'freeze_processes', 'syscore_suspend',
3462 'syscore_resume', 'resume_console', 'thaw_processes', 'CPU_ON',
3463 'CPU_OFF', 'acpi_suspend']
3464
3465 # extract the callgraph and traceevent data
3466 s2idle_enter = hwsus = False
3467 testruns, testdata = [], []
3468 testrun, data, limbo = 0, 0, True
3469 phase = 'suspend_prepare'
3470 tp, tf = loadTraceLog()
3471 for m_time, m_proc, m_pid, m_msg, m_param3 in tf:
3472 # gather the basic message data from the line
3473 if(m_time and m_pid and m_msg):
3474 t = FTraceLine(m_time, m_msg, m_param3)
3475 pid = int(m_pid)
3476 else:
3477 continue
3478 # the line should be a call, return, or event
3479 if(not t.fcall and not t.freturn and not t.fevent):
3480 continue
3481 # find the start of suspend
3482 if(t.startMarker()):
3483 data, limbo = Data(len(testdata)), False
3484 testdata.append(data)
3485 testrun = TestRun(data)
3486 testruns.append(testrun)
3487 tp.parseStamp(data, sysvals)
3488 data.setStart(t.time, t.name)
3489 data.first_suspend_prepare = True
3490 phase = data.setPhase('suspend_prepare', t.time, True)
3491 continue
3492 if(not data or limbo):
3493 continue
3494 # process cpu exec line
3495 if t.type == 'tracing_mark_write':
3496 if t.name == 'CMD COMPLETE' and data.tKernRes == 0:
3497 data.tKernRes = t.time
3498 m = re.match(tp.procexecfmt, t.name)
3499 if(m):
3500 parts, msg = 1, m.group('ps')
3501 m = re.match(tp.procmultifmt, msg)
3502 if(m):
3503 parts, msg = int(m.group('n')), m.group('ps')
3504 if tp.multiproccnt == 0:
3505 tp.multiproctime = t.time
3506 tp.multiproclist = dict()
3507 proclist = tp.multiproclist
3508 tp.multiproccnt += 1
3509 else:
3510 proclist = dict()
3511 tp.multiproccnt = 0
3512 for ps in msg.split(','):
3513 val = ps.split()
3514 if not val or len(val) != 2:
3515 continue
3516 name = val[0].replace('--', '-')
3517 proclist[name] = int(val[1])
3518 if parts == 1:
3519 data.pstl[t.time] = proclist
3520 elif parts == tp.multiproccnt:
3521 data.pstl[tp.multiproctime] = proclist
3522 tp.multiproccnt = 0
3523 continue
3524 # find the end of resume
3525 if(t.endMarker()):
3526 if data.tKernRes == 0:
3527 data.tKernRes = t.time
3528 data.handleEndMarker(t.time, t.name)
3529 if(not sysvals.usetracemarkers):
3530 # no trace markers? then quit and be sure to finish recording
3531 # the event we used to trigger resume end
3532 if('thaw_processes' in testrun.ttemp and len(testrun.ttemp['thaw_processes']) > 0):
3533 # if an entry exists, assume this is its end
3534 testrun.ttemp['thaw_processes'][-1]['end'] = t.time
3535 limbo = True
3536 continue
3537 # trace event processing
3538 if(t.fevent):
3539 if(t.type == 'suspend_resume'):
3540 # suspend_resume trace events have two types, begin and end
3541 if(re.match(r'(?P<name>.*) begin$', t.name)):
3542 isbegin = True
3543 elif(re.match(r'(?P<name>.*) end$', t.name)):
3544 isbegin = False
3545 else:
3546 continue
3547 if '[' in t.name:
3548 m = re.match(r'(?P<name>.*)\[.*', t.name)
3549 else:
3550 m = re.match(r'(?P<name>.*) .*', t.name)
3551 name = m.group('name')
3552 # ignore these events
3553 if(name.split('[')[0] in tracewatch):
3554 continue
3555 # -- phase changes --
3556 # start of kernel suspend
3557 if(re.match(r'suspend_enter\[.*', t.name)):
3558 if(isbegin and data.tKernSus == 0):
3559 data.tKernSus = t.time
3560 continue
3561 # suspend_prepare start
3562 elif(re.match(r'dpm_prepare\[.*', t.name)):
3563 if isbegin and data.first_suspend_prepare:
3564 data.first_suspend_prepare = False
3565 if data.tKernSus == 0:
3566 data.tKernSus = t.time
3567 continue
3568 phase = data.setPhase('suspend_prepare', t.time, isbegin)
3569 continue
3570 # suspend start
3571 elif(re.match(r'dpm_suspend\[.*', t.name)):
3572 phase = data.setPhase('suspend', t.time, isbegin)
3573 continue
3574 # suspend_late start
3575 elif(re.match(r'dpm_suspend_late\[.*', t.name)):
3576 phase = data.setPhase('suspend_late', t.time, isbegin)
3577 continue
3578 # suspend_noirq start
3579 elif(re.match(r'dpm_suspend_noirq\[.*', t.name)):
3580 phase = data.setPhase('suspend_noirq', t.time, isbegin)
3581 continue
3582 # suspend_machine/resume_machine
3583 elif(re.match(tp.machinesuspend, t.name)):
3584 lp = data.lastPhase()
3585 if(isbegin):
3586 hwsus = True
3587 if lp.startswith('resume_machine'):
3588 # trim out s2idle loops, track time trying to freeze
3589 llp = data.lastPhase(2)
3590 if llp.startswith('suspend_machine'):
3591 if 'waking' not in data.dmesg[llp]:
3592 data.dmesg[llp]['waking'] = [0, 0.0]
3593 data.dmesg[llp]['waking'][0] += 1
3594 data.dmesg[llp]['waking'][1] += \
3595 t.time - data.dmesg[lp]['start']
3596 data.currphase = ''
3597 del data.dmesg[lp]
3598 continue
3599 phase = data.setPhase('suspend_machine', data.dmesg[lp]['end'], True)
3600 data.setPhase(phase, t.time, False)
3601 if data.tSuspended == 0:
3602 data.tSuspended = t.time
3603 else:
3604 if lp.startswith('resume_machine'):
3605 data.dmesg[lp]['end'] = t.time
3606 continue
3607 phase = data.setPhase('resume_machine', t.time, True)
3608 if(sysvals.suspendmode in ['mem', 'disk']):
3609 susp = phase.replace('resume', 'suspend')
3610 if susp in data.dmesg:
3611 data.dmesg[susp]['end'] = t.time
3612 data.tSuspended = t.time
3613 data.tResumed = t.time
3614 continue
3615 # resume_noirq start
3616 elif(re.match(r'dpm_resume_noirq\[.*', t.name)):
3617 phase = data.setPhase('resume_noirq', t.time, isbegin)
3618 continue
3619 # resume_early start
3620 elif(re.match(r'dpm_resume_early\[.*', t.name)):
3621 phase = data.setPhase('resume_early', t.time, isbegin)
3622 continue
3623 # resume start
3624 elif(re.match(r'dpm_resume\[.*', t.name)):
3625 phase = data.setPhase('resume', t.time, isbegin)
3626 continue
3627 # resume complete start
3628 elif(re.match(r'dpm_complete\[.*', t.name)):
3629 phase = data.setPhase('resume_complete', t.time, isbegin)
3630 continue
3631 # skip trace events inside devices calls
3632 if(not data.isTraceEventOutsideDeviceCalls(pid, t.time)):
3633 continue
3634 # global events (outside device calls) are graphed
3635 if(name not in testrun.ttemp):
3636 testrun.ttemp[name] = []
3637 # special handling for s2idle_enter
3638 if name == 'machine_suspend':
3639 if hwsus:
3640 s2idle_enter = hwsus = False
3641 elif s2idle_enter and not isbegin:
3642 if(len(testrun.ttemp[name]) > 0):
3643 testrun.ttemp[name][-1]['end'] = t.time
3644 testrun.ttemp[name][-1]['loop'] += 1
3645 elif not s2idle_enter and isbegin:
3646 s2idle_enter = True
3647 testrun.ttemp[name].append({'begin': t.time,
3648 'end': t.time, 'pid': pid, 'loop': 0})
3649 continue
3650 if(isbegin):
3651 # create a new list entry
3652 testrun.ttemp[name].append(\
3653 {'begin': t.time, 'end': t.time, 'pid': pid})
3654 else:
3655 if(len(testrun.ttemp[name]) > 0):
3656 # if an entry exists, assume this is its end
3657 testrun.ttemp[name][-1]['end'] = t.time
3658 # device callback start
3659 elif(t.type == 'device_pm_callback_start'):
3660 if phase not in data.dmesg:
3661 continue
3662 m = re.match(r'(?P<drv>.*) (?P<d>.*), parent: *(?P<p>.*), .*',\
3663 t.name);
3664 if(not m):
3665 continue
3666 drv = m.group('drv')
3667 n = m.group('d')
3668 p = m.group('p')
3669 if(n and p):
3670 data.newAction(phase, n, pid, p, t.time, -1, drv)
3671 if pid not in data.devpids:
3672 data.devpids.append(pid)
3673 # device callback finish
3674 elif(t.type == 'device_pm_callback_end'):
3675 if phase not in data.dmesg:
3676 continue
3677 m = re.match(r'(?P<drv>.*) (?P<d>.*), err.*', t.name);
3678 if(not m):
3679 continue
3680 n = m.group('d')
3681 dev = data.findDevice(phase, n)
3682 if dev:
3683 dev['length'] = t.time - dev['start']
3684 dev['end'] = t.time
3685 # kprobe event processing
3686 elif(t.fkprobe):
3687 kprobename = t.type
3688 kprobedata = t.name
3689 key = (kprobename, pid)
3690 # displayname is generated from kprobe data
3691 displayname = ''
3692 if(t.fcall):
3693 displayname = sysvals.kprobeDisplayName(kprobename, kprobedata)
3694 if not displayname:
3695 continue
3696 if(key not in tp.ktemp):
3697 tp.ktemp[key] = []
3698 tp.ktemp[key].append({
3699 'pid': pid,
3700 'begin': t.time,
3701 'end': -1,
3702 'name': displayname,
3703 'cdata': kprobedata,
3704 'proc': m_proc,
3705 })
3706 # start of kernel resume
3707 if(data.tKernSus == 0 and phase == 'suspend_prepare' \
3708 and kprobename in ksuscalls):
3709 data.tKernSus = t.time
3710 elif(t.freturn):
3711 if(key not in tp.ktemp) or len(tp.ktemp[key]) < 1:
3712 continue
3713 e = next((x for x in reversed(tp.ktemp[key]) if x['end'] < 0), 0)
3714 if not e:
3715 continue
3716 if (t.time - e['begin']) * 1000 < sysvals.mindevlen:
3717 tp.ktemp[key].pop()
3718 continue
3719 e['end'] = t.time
3720 e['rdata'] = kprobedata
3721 # end of kernel resume
3722 if(phase != 'suspend_prepare' and kprobename in krescalls):
3723 if phase in data.dmesg:
3724 data.dmesg[phase]['end'] = t.time
3725 data.tKernRes = t.time
3726
3727 # callgraph processing
3728 elif sysvals.usecallgraph:
3729 # create a callgraph object for the data
3730 key = (m_proc, pid)
3731 if(key not in testrun.ftemp):
3732 testrun.ftemp[key] = []
3733 testrun.ftemp[key].append(FTraceCallGraph(pid, sysvals))
3734 # when the call is finished, see which device matches it
3735 cg = testrun.ftemp[key][-1]
3736 res = cg.addLine(t)
3737 if(res != 0):
3738 testrun.ftemp[key].append(FTraceCallGraph(pid, sysvals))
3739 if(res == -1):
3740 testrun.ftemp[key][-1].addLine(t)
3741 if len(testdata) < 1:
3742 sysvals.vprint('WARNING: ftrace start marker is missing')
3743 if data and not data.devicegroups:
3744 sysvals.vprint('WARNING: ftrace end marker is missing')
3745 data.handleEndMarker(t.time, t.name)
3746
3747 if sysvals.suspendmode == 'command':
3748 for test in testruns:
3749 for p in test.data.sortedPhases():
3750 if p == 'suspend_prepare':
3751 test.data.dmesg[p]['start'] = test.data.start
3752 test.data.dmesg[p]['end'] = test.data.end
3753 else:
3754 test.data.dmesg[p]['start'] = test.data.end
3755 test.data.dmesg[p]['end'] = test.data.end
3756 test.data.tSuspended = test.data.end
3757 test.data.tResumed = test.data.end
3758 test.data.fwValid = False
3759
3760 # dev source and procmon events can be unreadable with mixed phase height
3761 if sysvals.usedevsrc or sysvals.useprocmon:
3762 sysvals.mixedphaseheight = False
3763
3764 # expand phase boundaries so there are no gaps
3765 for data in testdata:
3766 lp = data.sortedPhases()[0]
3767 for p in data.sortedPhases():
3768 if(p != lp and not ('machine' in p and 'machine' in lp)):
3769 data.dmesg[lp]['end'] = data.dmesg[p]['start']
3770 lp = p
3771
3772 for i in range(len(testruns)):
3773 test = testruns[i]
3774 data = test.data
3775 # find the total time range for this test (begin, end)
3776 tlb, tle = data.start, data.end
3777 if i < len(testruns) - 1:
3778 tle = testruns[i+1].data.start
3779 # add the process usage data to the timeline
3780 if sysvals.useprocmon:
3781 data.createProcessUsageEvents()
3782 # add the traceevent data to the device hierarchy
3783 if(sysvals.usetraceevents):
3784 # add actual trace funcs
3785 for name in sorted(test.ttemp):
3786 for event in test.ttemp[name]:
3787 if event['end'] - event['begin'] <= 0:
3788 continue
3789 title = name
3790 if name == 'machine_suspend' and 'loop' in event:
3791 title = 's2idle_enter_%dx' % event['loop']
3792 data.newActionGlobal(title, event['begin'], event['end'], event['pid'])
3793 # add the kprobe based virtual tracefuncs as actual devices
3794 for key in sorted(tp.ktemp):
3795 name, pid = key
3796 if name not in sysvals.tracefuncs:
3797 continue
3798 if pid not in data.devpids:
3799 data.devpids.append(pid)
3800 for e in tp.ktemp[key]:
3801 kb, ke = e['begin'], e['end']
3802 if ke - kb < 0.000001 or tlb > kb or tle <= kb:
3803 continue
3804 color = sysvals.kprobeColor(name)
3805 data.newActionGlobal(e['name'], kb, ke, pid, color)
3806 # add config base kprobes and dev kprobes
3807 if sysvals.usedevsrc:
3808 for key in sorted(tp.ktemp):
3809 name, pid = key
3810 if name in sysvals.tracefuncs or name not in sysvals.dev_tracefuncs:
3811 continue
3812 for e in tp.ktemp[key]:
3813 kb, ke = e['begin'], e['end']
3814 if ke - kb < 0.000001 or tlb > kb or tle <= kb:
3815 continue
3816 data.addDeviceFunctionCall(e['name'], name, e['proc'], pid, kb,
3817 ke, e['cdata'], e['rdata'])
3818 if sysvals.usecallgraph:
3819 # add the callgraph data to the device hierarchy
3820 sortlist = dict()
3821 for key in sorted(test.ftemp):
3822 proc, pid = key
3823 for cg in test.ftemp[key]:
3824 if len(cg.list) < 1 or cg.invalid or (cg.end - cg.start == 0):
3825 continue
3826 if(not cg.postProcess()):
3827 id = 'task %s' % (pid)
3828 sysvals.vprint('Sanity check failed for '+\
3829 id+', ignoring this callback')
3830 continue
3831 # match cg data to devices
3832 devname = ''
3833 if sysvals.suspendmode != 'command':
3834 devname = cg.deviceMatch(pid, data)
3835 if not devname:
3836 sortkey = '%f%f%d' % (cg.start, cg.end, pid)
3837 sortlist[sortkey] = cg
3838 elif len(cg.list) > 1000000 and cg.name != sysvals.ftopfunc:
3839 sysvals.vprint('WARNING: the callgraph for %s is massive (%d lines)' %\
3840 (devname, len(cg.list)))
3841 # create blocks for orphan cg data
3842 for sortkey in sorted(sortlist):
3843 cg = sortlist[sortkey]
3844 name = cg.name
3845 if sysvals.isCallgraphFunc(name):
3846 sysvals.vprint('Callgraph found for task %d: %.3fms, %s' % (cg.pid, (cg.end - cg.start)*1000, name))
3847 cg.newActionFromFunction(data)
3848 if sysvals.suspendmode == 'command':
3849 return (testdata, '')
3850
3851 # fill in any missing phases
3852 error = []
3853 for data in testdata:
3854 tn = '' if len(testdata) == 1 else ('%d' % (data.testnumber + 1))
3855 terr = ''
3856 phasedef = data.phasedef
3857 lp = 'suspend_prepare'
3858 for p in sorted(phasedef, key=lambda k:phasedef[k]['order']):
3859 if p not in data.dmesg:
3860 if not terr:
3861 ph = p if 'machine' in p else lp
3862 if p == 'suspend_machine':
3863 sm = sysvals.suspendmode
3864 if sm in suspendmodename:
3865 sm = suspendmodename[sm]
3866 terr = 'test%s did not enter %s power mode' % (tn, sm)
3867 else:
3868 terr = '%s%s failed in %s phase' % (sysvals.suspendmode, tn, ph)
3869 pprint('TEST%s FAILED: %s' % (tn, terr))
3870 error.append(terr)
3871 if data.tSuspended == 0:
3872 data.tSuspended = data.dmesg[lp]['end']
3873 if data.tResumed == 0:
3874 data.tResumed = data.dmesg[lp]['end']
3875 data.fwValid = False
3876 sysvals.vprint('WARNING: phase "%s" is missing!' % p)
3877 lp = p
3878 if not terr and 'dev' in data.wifi and data.wifi['stat'] == 'timeout':
3879 terr = '%s%s failed in wifi_resume <i>(%s %.0fs timeout)</i>' % \
3880 (sysvals.suspendmode, tn, data.wifi['dev'], data.wifi['time'])
3881 error.append(terr)
3882 if not terr and data.enterfail:
3883 pprint('test%s FAILED: enter %s failed with %s' % (tn, sysvals.suspendmode, data.enterfail))
3884 terr = 'test%s failed to enter %s mode' % (tn, sysvals.suspendmode)
3885 error.append(terr)
3886 if data.tSuspended == 0:
3887 data.tSuspended = data.tKernRes
3888 if data.tResumed == 0:
3889 data.tResumed = data.tSuspended
3890
3891 if(len(sysvals.devicefilter) > 0):
3892 data.deviceFilter(sysvals.devicefilter)
3893 data.fixupInitcallsThatDidntReturn()
3894 if sysvals.usedevsrc:
3895 data.optimizeDevSrc()
3896
3897 # x2: merge any overlapping devices between test runs
3898 if sysvals.usedevsrc and len(testdata) > 1:
3899 tc = len(testdata)
3900 for i in range(tc - 1):
3901 devlist = testdata[i].overflowDevices()
3902 for j in range(i + 1, tc):
3903 testdata[j].mergeOverlapDevices(devlist)
3904 testdata[0].stitchTouchingThreads(testdata[1:])
3905 return (testdata, ', '.join(error))
3906
3907# Function: loadKernelLog
3908# Description:
3909# load the dmesg file into memory and fix up any ordering issues
3910# Output:
3911# An array of empty Data objects with only their dmesgtext attributes set
3912def loadKernelLog():
3913 sysvals.vprint('Analyzing the dmesg data (%s)...' % \
3914 os.path.basename(sysvals.dmesgfile))
3915 if(os.path.exists(sysvals.dmesgfile) == False):
3916 doError('%s does not exist' % sysvals.dmesgfile)
3917
3918 # there can be multiple test runs in a single file
3919 tp = TestProps()
3920 tp.stamp = datetime.now().strftime('# suspend-%m%d%y-%H%M%S localhost mem unknown')
3921 testruns = []
3922 data = 0
3923 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
3924 for line in lf:
3925 line = line.replace('\r\n', '')
3926 idx = line.find('[')
3927 if idx > 1:
3928 line = line[idx:]
3929 if tp.stampInfo(line, sysvals):
3930 continue
3931 m = re.match(r'[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
3932 if(not m):
3933 continue
3934 msg = m.group("msg")
3935 if re.match(r'PM: Syncing filesystems.*', msg) or \
3936 re.match(r'PM: suspend entry.*', msg):
3937 if(data):
3938 testruns.append(data)
3939 data = Data(len(testruns))
3940 tp.parseStamp(data, sysvals)
3941 if(not data):
3942 continue
3943 m = re.match(r'.* *(?P<k>[0-9]\.[0-9]{2}\.[0-9]-.*) .*', msg)
3944 if(m):
3945 sysvals.stamp['kernel'] = m.group('k')
3946 m = re.match(r'PM: Preparing system for (?P<m>.*) sleep', msg)
3947 if not m:
3948 m = re.match(r'PM: Preparing system for sleep \((?P<m>.*)\)', msg)
3949 if m:
3950 sysvals.stamp['mode'] = sysvals.suspendmode = m.group('m')
3951 data.dmesgtext.append(line)
3952 lf.close()
3953
3954 if sysvals.suspendmode == 's2idle':
3955 sysvals.suspendmode = 'freeze'
3956 elif sysvals.suspendmode == 'deep':
3957 sysvals.suspendmode = 'mem'
3958 if data:
3959 testruns.append(data)
3960 if len(testruns) < 1:
3961 doError('dmesg log has no suspend/resume data: %s' \
3962 % sysvals.dmesgfile)
3963
3964 # fix lines with same timestamp/function with the call and return swapped
3965 for data in testruns:
3966 last = ''
3967 for line in data.dmesgtext:
3968 ct, cf, n, p = data.initcall_debug_call(line)
3969 rt, rf, l = data.initcall_debug_return(last)
3970 if ct and rt and ct == rt and cf == rf:
3971 i = data.dmesgtext.index(last)
3972 j = data.dmesgtext.index(line)
3973 data.dmesgtext[i] = line
3974 data.dmesgtext[j] = last
3975 last = line
3976 return testruns
3977
3978# Function: parseKernelLog
3979# Description:
3980# Analyse a dmesg log output file generated from this app during
3981# the execution phase. Create a set of device structures in memory
3982# for subsequent formatting in the html output file
3983# This call is only for legacy support on kernels where the ftrace
3984# data lacks the suspend_resume or device_pm_callbacks trace events.
3985# Arguments:
3986# data: an empty Data object (with dmesgtext) obtained from loadKernelLog
3987# Output:
3988# The filled Data object
3989def parseKernelLog(data):
3990 phase = 'suspend_runtime'
3991
3992 if(data.fwValid):
3993 sysvals.vprint('Firmware Suspend = %u ns, Firmware Resume = %u ns' % \
3994 (data.fwSuspend, data.fwResume))
3995
3996 # dmesg phase match table
3997 dm = {
3998 'suspend_prepare': ['PM: Syncing filesystems.*', 'PM: suspend entry.*'],
3999 'suspend': ['PM: Entering [a-z]* sleep.*', 'Suspending console.*',
4000 'PM: Suspending system .*'],
4001 'suspend_late': ['PM: suspend of devices complete after.*',
4002 'PM: freeze of devices complete after.*'],
4003 'suspend_noirq': ['PM: late suspend of devices complete after.*',
4004 'PM: late freeze of devices complete after.*'],
4005 'suspend_machine': ['PM: suspend-to-idle',
4006 'PM: noirq suspend of devices complete after.*',
4007 'PM: noirq freeze of devices complete after.*'],
4008 'resume_machine': ['[PM: ]*Timekeeping suspended for.*',
4009 'ACPI: Low-level resume complete.*',
4010 'ACPI: resume from mwait',
4011 r'Suspended for [0-9\.]* seconds'],
4012 'resume_noirq': ['PM: resume from suspend-to-idle',
4013 'ACPI: Waking up from system sleep state.*'],
4014 'resume_early': ['PM: noirq resume of devices complete after.*',
4015 'PM: noirq restore of devices complete after.*'],
4016 'resume': ['PM: early resume of devices complete after.*',
4017 'PM: early restore of devices complete after.*'],
4018 'resume_complete': ['PM: resume of devices complete after.*',
4019 'PM: restore of devices complete after.*'],
4020 'post_resume': [r'.*Restarting tasks \.\.\..*'],
4021 }
4022
4023 # action table (expected events that occur and show up in dmesg)
4024 at = {
4025 'sync_filesystems': {
4026 'smsg': '.*[Ff]+ilesystems.*',
4027 'emsg': 'PM: Preparing system for[a-z]* sleep.*' },
4028 'freeze_user_processes': {
4029 'smsg': 'Freezing user space processes.*',
4030 'emsg': 'Freezing remaining freezable tasks.*' },
4031 'freeze_tasks': {
4032 'smsg': 'Freezing remaining freezable tasks.*',
4033 'emsg': 'PM: Suspending system.*' },
4034 'ACPI prepare': {
4035 'smsg': 'ACPI: Preparing to enter system sleep state.*',
4036 'emsg': 'PM: Saving platform NVS memory.*' },
4037 'PM vns': {
4038 'smsg': 'PM: Saving platform NVS memory.*',
4039 'emsg': 'Disabling non-boot CPUs .*' },
4040 }
4041
4042 t0 = -1.0
4043 cpu_start = -1.0
4044 prevktime = -1.0
4045 actions = dict()
4046 for line in data.dmesgtext:
4047 # parse each dmesg line into the time and message
4048 m = re.match(r'[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
4049 if(m):
4050 val = m.group('ktime')
4051 try:
4052 ktime = float(val)
4053 except:
4054 continue
4055 msg = m.group('msg')
4056 # initialize data start to first line time
4057 if t0 < 0:
4058 data.setStart(ktime)
4059 t0 = ktime
4060 else:
4061 continue
4062
4063 # check for a phase change line
4064 phasechange = False
4065 for p in dm:
4066 for s in dm[p]:
4067 if(re.match(s, msg)):
4068 phasechange, phase = True, p
4069 dm[p] = [s]
4070 break
4071
4072 # hack for determining resume_machine end for freeze
4073 if(not sysvals.usetraceevents and sysvals.suspendmode == 'freeze' \
4074 and phase == 'resume_machine' and \
4075 data.initcall_debug_call(line, True)):
4076 data.setPhase(phase, ktime, False)
4077 phase = 'resume_noirq'
4078 data.setPhase(phase, ktime, True)
4079
4080 if phasechange:
4081 if phase == 'suspend_prepare':
4082 data.setPhase(phase, ktime, True)
4083 data.setStart(ktime)
4084 data.tKernSus = ktime
4085 elif phase == 'suspend':
4086 lp = data.lastPhase()
4087 if lp:
4088 data.setPhase(lp, ktime, False)
4089 data.setPhase(phase, ktime, True)
4090 elif phase == 'suspend_late':
4091 lp = data.lastPhase()
4092 if lp:
4093 data.setPhase(lp, ktime, False)
4094 data.setPhase(phase, ktime, True)
4095 elif phase == 'suspend_noirq':
4096 lp = data.lastPhase()
4097 if lp:
4098 data.setPhase(lp, ktime, False)
4099 data.setPhase(phase, ktime, True)
4100 elif phase == 'suspend_machine':
4101 lp = data.lastPhase()
4102 if lp:
4103 data.setPhase(lp, ktime, False)
4104 data.setPhase(phase, ktime, True)
4105 elif phase == 'resume_machine':
4106 lp = data.lastPhase()
4107 if(sysvals.suspendmode in ['freeze', 'standby']):
4108 data.tSuspended = prevktime
4109 if lp:
4110 data.setPhase(lp, prevktime, False)
4111 else:
4112 data.tSuspended = ktime
4113 if lp:
4114 data.setPhase(lp, prevktime, False)
4115 data.tResumed = ktime
4116 data.setPhase(phase, ktime, True)
4117 elif phase == 'resume_noirq':
4118 lp = data.lastPhase()
4119 if lp:
4120 data.setPhase(lp, ktime, False)
4121 data.setPhase(phase, ktime, True)
4122 elif phase == 'resume_early':
4123 lp = data.lastPhase()
4124 if lp:
4125 data.setPhase(lp, ktime, False)
4126 data.setPhase(phase, ktime, True)
4127 elif phase == 'resume':
4128 lp = data.lastPhase()
4129 if lp:
4130 data.setPhase(lp, ktime, False)
4131 data.setPhase(phase, ktime, True)
4132 elif phase == 'resume_complete':
4133 lp = data.lastPhase()
4134 if lp:
4135 data.setPhase(lp, ktime, False)
4136 data.setPhase(phase, ktime, True)
4137 elif phase == 'post_resume':
4138 lp = data.lastPhase()
4139 if lp:
4140 data.setPhase(lp, ktime, False)
4141 data.setEnd(ktime)
4142 data.tKernRes = ktime
4143 break
4144
4145 # -- device callbacks --
4146 if(phase in data.sortedPhases()):
4147 # device init call
4148 t, f, n, p = data.initcall_debug_call(line)
4149 if t and f and n and p:
4150 data.newAction(phase, f, int(n), p, ktime, -1, '')
4151 else:
4152 # device init return
4153 t, f, l = data.initcall_debug_return(line)
4154 if t and f and l:
4155 list = data.dmesg[phase]['list']
4156 if(f in list):
4157 dev = list[f]
4158 dev['length'] = int(l)
4159 dev['end'] = ktime
4160
4161 # if trace events are not available, these are better than nothing
4162 if(not sysvals.usetraceevents):
4163 # look for known actions
4164 for a in sorted(at):
4165 if(re.match(at[a]['smsg'], msg)):
4166 if(a not in actions):
4167 actions[a] = [{'begin': ktime, 'end': ktime}]
4168 if(re.match(at[a]['emsg'], msg)):
4169 if(a in actions and actions[a][-1]['begin'] == actions[a][-1]['end']):
4170 actions[a][-1]['end'] = ktime
4171 # now look for CPU on/off events
4172 if(re.match(r'Disabling non-boot CPUs .*', msg)):
4173 # start of first cpu suspend
4174 cpu_start = ktime
4175 elif(re.match(r'Enabling non-boot CPUs .*', msg)):
4176 # start of first cpu resume
4177 cpu_start = ktime
4178 elif(re.match(r'smpboot: CPU (?P<cpu>[0-9]*) is now offline', msg) \
4179 or re.match(r'psci: CPU(?P<cpu>[0-9]*) killed.*', msg)):
4180 # end of a cpu suspend, start of the next
4181 m = re.match(r'smpboot: CPU (?P<cpu>[0-9]*) is now offline', msg)
4182 if(not m):
4183 m = re.match(r'psci: CPU(?P<cpu>[0-9]*) killed.*', msg)
4184 cpu = 'CPU'+m.group('cpu')
4185 if(cpu not in actions):
4186 actions[cpu] = []
4187 actions[cpu].append({'begin': cpu_start, 'end': ktime})
4188 cpu_start = ktime
4189 elif(re.match(r'CPU(?P<cpu>[0-9]*) is up', msg)):
4190 # end of a cpu resume, start of the next
4191 m = re.match(r'CPU(?P<cpu>[0-9]*) is up', msg)
4192 cpu = 'CPU'+m.group('cpu')
4193 if(cpu not in actions):
4194 actions[cpu] = []
4195 actions[cpu].append({'begin': cpu_start, 'end': ktime})
4196 cpu_start = ktime
4197 prevktime = ktime
4198 data.initDevicegroups()
4199
4200 # fill in any missing phases
4201 phasedef = data.phasedef
4202 terr, lp = '', 'suspend_prepare'
4203 if lp not in data.dmesg:
4204 doError('dmesg log format has changed, could not find start of suspend')
4205 for p in sorted(phasedef, key=lambda k:phasedef[k]['order']):
4206 if p not in data.dmesg:
4207 if not terr:
4208 pprint('TEST FAILED: %s failed in %s phase' % (sysvals.suspendmode, lp))
4209 terr = '%s failed in %s phase' % (sysvals.suspendmode, lp)
4210 if data.tSuspended == 0:
4211 data.tSuspended = data.dmesg[lp]['end']
4212 if data.tResumed == 0:
4213 data.tResumed = data.dmesg[lp]['end']
4214 sysvals.vprint('WARNING: phase "%s" is missing!' % p)
4215 lp = p
4216 lp = data.sortedPhases()[0]
4217 for p in data.sortedPhases():
4218 if(p != lp and not ('machine' in p and 'machine' in lp)):
4219 data.dmesg[lp]['end'] = data.dmesg[p]['start']
4220 lp = p
4221 if data.tSuspended == 0:
4222 data.tSuspended = data.tKernRes
4223 if data.tResumed == 0:
4224 data.tResumed = data.tSuspended
4225
4226 # fill in any actions we've found
4227 for name in sorted(actions):
4228 for event in actions[name]:
4229 data.newActionGlobal(name, event['begin'], event['end'])
4230
4231 if(len(sysvals.devicefilter) > 0):
4232 data.deviceFilter(sysvals.devicefilter)
4233 data.fixupInitcallsThatDidntReturn()
4234 return True
4235
4236def callgraphHTML(sv, hf, num, cg, title, color, devid):
4237 html_func_top = '<article id="{0}" class="atop" style="background:{1}">\n<input type="checkbox" class="pf" id="f{2}" checked/><label for="f{2}">{3} {4}</label>\n'
4238 html_func_start = '<article>\n<input type="checkbox" class="pf" id="f{0}" checked/><label for="f{0}">{1} {2}</label>\n'
4239 html_func_end = '</article>\n'
4240 html_func_leaf = '<article>{0} {1}</article>\n'
4241
4242 cgid = devid
4243 if cg.id:
4244 cgid += cg.id
4245 cglen = (cg.end - cg.start) * 1000
4246 if cglen < sv.mincglen:
4247 return num
4248
4249 fmt = '<r>(%.3f ms @ '+sv.timeformat+' to '+sv.timeformat+')</r>'
4250 flen = fmt % (cglen, cg.start, cg.end)
4251 hf.write(html_func_top.format(cgid, color, num, title, flen))
4252 num += 1
4253 for line in cg.list:
4254 if(line.length < 0.000000001):
4255 flen = ''
4256 else:
4257 fmt = '<n>(%.3f ms @ '+sv.timeformat+')</n>'
4258 flen = fmt % (line.length*1000, line.time)
4259 if line.isLeaf():
4260 if line.length * 1000 < sv.mincglen:
4261 continue
4262 hf.write(html_func_leaf.format(line.name, flen))
4263 elif line.freturn:
4264 hf.write(html_func_end)
4265 else:
4266 hf.write(html_func_start.format(num, line.name, flen))
4267 num += 1
4268 hf.write(html_func_end)
4269 return num
4270
4271def addCallgraphs(sv, hf, data):
4272 hf.write('<section id="callgraphs" class="callgraph">\n')
4273 # write out the ftrace data converted to html
4274 num = 0
4275 for p in data.sortedPhases():
4276 if sv.cgphase and p != sv.cgphase:
4277 continue
4278 list = data.dmesg[p]['list']
4279 for d in data.sortedDevices(p):
4280 if len(sv.cgfilter) > 0 and d not in sv.cgfilter:
4281 continue
4282 dev = list[d]
4283 color = 'white'
4284 if 'color' in data.dmesg[p]:
4285 color = data.dmesg[p]['color']
4286 if 'color' in dev:
4287 color = dev['color']
4288 name = d if '[' not in d else d.split('[')[0]
4289 if(d in sv.devprops):
4290 name = sv.devprops[d].altName(d)
4291 if 'drv' in dev and dev['drv']:
4292 name += ' {%s}' % dev['drv']
4293 if sv.suspendmode in suspendmodename:
4294 name += ' '+p
4295 if('ftrace' in dev):
4296 cg = dev['ftrace']
4297 if cg.name == sv.ftopfunc:
4298 name = 'top level suspend/resume call'
4299 num = callgraphHTML(sv, hf, num, cg,
4300 name, color, dev['id'])
4301 if('ftraces' in dev):
4302 for cg in dev['ftraces']:
4303 num = callgraphHTML(sv, hf, num, cg,
4304 name+' → '+cg.name, color, dev['id'])
4305 hf.write('\n\n </section>\n')
4306
4307def summaryCSS(title, center=True):
4308 tdcenter = 'text-align:center;' if center else ''
4309 out = '<!DOCTYPE html>\n<html>\n<head>\n\
4310 <meta http-equiv="content-type" content="text/html; charset=UTF-8">\n\
4311 <title>'+title+'</title>\n\
4312 <style type=\'text/css\'>\n\
4313 .stamp {width: 100%;text-align:center;background:#888;line-height:30px;color:white;font: 25px Arial;}\n\
4314 table {width:100%;border-collapse: collapse;border:1px solid;}\n\
4315 th {border: 1px solid black;background:#222;color:white;}\n\
4316 td {font: 14px "Times New Roman";'+tdcenter+'}\n\
4317 tr.head td {border: 1px solid black;background:#aaa;}\n\
4318 tr.alt {background-color:#ddd;}\n\
4319 tr.notice {color:red;}\n\
4320 .minval {background-color:#BBFFBB;}\n\
4321 .medval {background-color:#BBBBFF;}\n\
4322 .maxval {background-color:#FFBBBB;}\n\
4323 .head a {color:#000;text-decoration: none;}\n\
4324 </style>\n</head>\n<body>\n'
4325 return out
4326
4327# Function: createHTMLSummarySimple
4328# Description:
4329# Create summary html file for a series of tests
4330# Arguments:
4331# testruns: array of Data objects from parseTraceLog
4332def createHTMLSummarySimple(testruns, htmlfile, title):
4333 # write the html header first (html head, css code, up to body start)
4334 html = summaryCSS('Summary - SleepGraph')
4335
4336 # extract the test data into list
4337 list = dict()
4338 tAvg, tMin, tMax, tMed = [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [dict(), dict()]
4339 iMin, iMed, iMax = [0, 0], [0, 0], [0, 0]
4340 num = 0
4341 useturbo = usewifi = False
4342 lastmode = ''
4343 cnt = dict()
4344 for data in sorted(testruns, key=lambda v:(v['mode'], v['host'], v['kernel'], v['time'])):
4345 mode = data['mode']
4346 if mode not in list:
4347 list[mode] = {'data': [], 'avg': [0,0], 'min': [0,0], 'max': [0,0], 'med': [0,0]}
4348 if lastmode and lastmode != mode and num > 0:
4349 for i in range(2):
4350 s = sorted(tMed[i])
4351 list[lastmode]['med'][i] = s[int(len(s)//2)]
4352 iMed[i] = tMed[i][list[lastmode]['med'][i]]
4353 list[lastmode]['avg'] = [tAvg[0] / num, tAvg[1] / num]
4354 list[lastmode]['min'] = tMin
4355 list[lastmode]['max'] = tMax
4356 list[lastmode]['idx'] = (iMin, iMed, iMax)
4357 tAvg, tMin, tMax, tMed = [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [dict(), dict()]
4358 iMin, iMed, iMax = [0, 0], [0, 0], [0, 0]
4359 num = 0
4360 pkgpc10 = syslpi = wifi = ''
4361 if 'pkgpc10' in data and 'syslpi' in data:
4362 pkgpc10, syslpi, useturbo = data['pkgpc10'], data['syslpi'], True
4363 if 'wifi' in data:
4364 wifi, usewifi = data['wifi'], True
4365 res = data['result']
4366 tVal = [float(data['suspend']), float(data['resume'])]
4367 list[mode]['data'].append([data['host'], data['kernel'],
4368 data['time'], tVal[0], tVal[1], data['url'], res,
4369 data['issues'], data['sus_worst'], data['sus_worsttime'],
4370 data['res_worst'], data['res_worsttime'], pkgpc10, syslpi, wifi,
4371 (data['fullmode'] if 'fullmode' in data else mode)])
4372 idx = len(list[mode]['data']) - 1
4373 if res.startswith('fail in'):
4374 res = 'fail'
4375 if res not in cnt:
4376 cnt[res] = 1
4377 else:
4378 cnt[res] += 1
4379 if res == 'pass':
4380 for i in range(2):
4381 tMed[i][tVal[i]] = idx
4382 tAvg[i] += tVal[i]
4383 if tMin[i] == 0 or tVal[i] < tMin[i]:
4384 iMin[i] = idx
4385 tMin[i] = tVal[i]
4386 if tMax[i] == 0 or tVal[i] > tMax[i]:
4387 iMax[i] = idx
4388 tMax[i] = tVal[i]
4389 num += 1
4390 lastmode = mode
4391 if lastmode and num > 0:
4392 for i in range(2):
4393 s = sorted(tMed[i])
4394 list[lastmode]['med'][i] = s[int(len(s)//2)]
4395 iMed[i] = tMed[i][list[lastmode]['med'][i]]
4396 list[lastmode]['avg'] = [tAvg[0] / num, tAvg[1] / num]
4397 list[lastmode]['min'] = tMin
4398 list[lastmode]['max'] = tMax
4399 list[lastmode]['idx'] = (iMin, iMed, iMax)
4400
4401 # group test header
4402 desc = []
4403 for ilk in sorted(cnt, reverse=True):
4404 if cnt[ilk] > 0:
4405 desc.append('%d %s' % (cnt[ilk], ilk))
4406 html += '<div class="stamp">%s (%d tests: %s)</div>\n' % (title, len(testruns), ', '.join(desc))
4407 th = '\t<th>{0}</th>\n'
4408 td = '\t<td>{0}</td>\n'
4409 tdh = '\t<td{1}>{0}</td>\n'
4410 tdlink = '\t<td><a href="{0}">html</a></td>\n'
4411 cols = 12
4412 if useturbo:
4413 cols += 2
4414 if usewifi:
4415 cols += 1
4416 colspan = '%d' % cols
4417
4418 # table header
4419 html += '<table>\n<tr>\n' + th.format('#') +\
4420 th.format('Mode') + th.format('Host') + th.format('Kernel') +\
4421 th.format('Test Time') + th.format('Result') + th.format('Issues') +\
4422 th.format('Suspend') + th.format('Resume') +\
4423 th.format('Worst Suspend Device') + th.format('SD Time') +\
4424 th.format('Worst Resume Device') + th.format('RD Time')
4425 if useturbo:
4426 html += th.format('PkgPC10') + th.format('SysLPI')
4427 if usewifi:
4428 html += th.format('Wifi')
4429 html += th.format('Detail')+'</tr>\n'
4430 # export list into html
4431 head = '<tr class="head"><td>{0}</td><td>{1}</td>'+\
4432 '<td colspan='+colspan+' class="sus">Suspend Avg={2} '+\
4433 '<span class=minval><a href="#s{10}min">Min={3}</a></span> '+\
4434 '<span class=medval><a href="#s{10}med">Med={4}</a></span> '+\
4435 '<span class=maxval><a href="#s{10}max">Max={5}</a></span> '+\
4436 'Resume Avg={6} '+\
4437 '<span class=minval><a href="#r{10}min">Min={7}</a></span> '+\
4438 '<span class=medval><a href="#r{10}med">Med={8}</a></span> '+\
4439 '<span class=maxval><a href="#r{10}max">Max={9}</a></span></td>'+\
4440 '</tr>\n'
4441 headnone = '<tr class="head"><td>{0}</td><td>{1}</td><td colspan='+\
4442 colspan+'></td></tr>\n'
4443 for mode in sorted(list):
4444 # header line for each suspend mode
4445 num = 0
4446 tAvg, tMin, tMax, tMed = list[mode]['avg'], list[mode]['min'],\
4447 list[mode]['max'], list[mode]['med']
4448 count = len(list[mode]['data'])
4449 if 'idx' in list[mode]:
4450 iMin, iMed, iMax = list[mode]['idx']
4451 html += head.format('%d' % count, mode.upper(),
4452 '%.3f' % tAvg[0], '%.3f' % tMin[0], '%.3f' % tMed[0], '%.3f' % tMax[0],
4453 '%.3f' % tAvg[1], '%.3f' % tMin[1], '%.3f' % tMed[1], '%.3f' % tMax[1],
4454 mode.lower()
4455 )
4456 else:
4457 iMin = iMed = iMax = [-1, -1, -1]
4458 html += headnone.format('%d' % count, mode.upper())
4459 for d in list[mode]['data']:
4460 # row classes - alternate row color
4461 rcls = ['alt'] if num % 2 == 1 else []
4462 if d[6] != 'pass':
4463 rcls.append('notice')
4464 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4465 # figure out if the line has sus or res highlighted
4466 idx = list[mode]['data'].index(d)
4467 tHigh = ['', '']
4468 for i in range(2):
4469 tag = 's%s' % mode if i == 0 else 'r%s' % mode
4470 if idx == iMin[i]:
4471 tHigh[i] = ' id="%smin" class=minval title="Minimum"' % tag
4472 elif idx == iMax[i]:
4473 tHigh[i] = ' id="%smax" class=maxval title="Maximum"' % tag
4474 elif idx == iMed[i]:
4475 tHigh[i] = ' id="%smed" class=medval title="Median"' % tag
4476 html += td.format("%d" % (list[mode]['data'].index(d) + 1)) # row
4477 html += td.format(d[15]) # mode
4478 html += td.format(d[0]) # host
4479 html += td.format(d[1]) # kernel
4480 html += td.format(d[2]) # time
4481 html += td.format(d[6]) # result
4482 html += td.format(d[7]) # issues
4483 html += tdh.format('%.3f ms' % d[3], tHigh[0]) if d[3] else td.format('') # suspend
4484 html += tdh.format('%.3f ms' % d[4], tHigh[1]) if d[4] else td.format('') # resume
4485 html += td.format(d[8]) # sus_worst
4486 html += td.format('%.3f ms' % d[9]) if d[9] else td.format('') # sus_worst time
4487 html += td.format(d[10]) # res_worst
4488 html += td.format('%.3f ms' % d[11]) if d[11] else td.format('') # res_worst time
4489 if useturbo:
4490 html += td.format(d[12]) # pkg_pc10
4491 html += td.format(d[13]) # syslpi
4492 if usewifi:
4493 html += td.format(d[14]) # wifi
4494 html += tdlink.format(d[5]) if d[5] else td.format('') # url
4495 html += '</tr>\n'
4496 num += 1
4497
4498 # flush the data to file
4499 hf = open(htmlfile, 'w')
4500 hf.write(html+'</table>\n</body>\n</html>\n')
4501 hf.close()
4502
4503def createHTMLDeviceSummary(testruns, htmlfile, title):
4504 html = summaryCSS('Device Summary - SleepGraph', False)
4505
4506 # create global device list from all tests
4507 devall = dict()
4508 for data in testruns:
4509 host, url, devlist = data['host'], data['url'], data['devlist']
4510 for type in devlist:
4511 if type not in devall:
4512 devall[type] = dict()
4513 mdevlist, devlist = devall[type], data['devlist'][type]
4514 for name in devlist:
4515 length = devlist[name]
4516 if name not in mdevlist:
4517 mdevlist[name] = {'name': name, 'host': host,
4518 'worst': length, 'total': length, 'count': 1,
4519 'url': url}
4520 else:
4521 if length > mdevlist[name]['worst']:
4522 mdevlist[name]['worst'] = length
4523 mdevlist[name]['url'] = url
4524 mdevlist[name]['host'] = host
4525 mdevlist[name]['total'] += length
4526 mdevlist[name]['count'] += 1
4527
4528 # generate the html
4529 th = '\t<th>{0}</th>\n'
4530 td = '\t<td align=center>{0}</td>\n'
4531 tdr = '\t<td align=right>{0}</td>\n'
4532 tdlink = '\t<td align=center><a href="{0}">html</a></td>\n'
4533 limit = 1
4534 for type in sorted(devall, reverse=True):
4535 num = 0
4536 devlist = devall[type]
4537 # table header
4538 html += '<div class="stamp">%s (%s devices > %d ms)</div><table>\n' % \
4539 (title, type.upper(), limit)
4540 html += '<tr>\n' + '<th align=right>Device Name</th>' +\
4541 th.format('Average Time') + th.format('Count') +\
4542 th.format('Worst Time') + th.format('Host (worst time)') +\
4543 th.format('Link (worst time)') + '</tr>\n'
4544 for name in sorted(devlist, key=lambda k:(devlist[k]['worst'], \
4545 devlist[k]['total'], devlist[k]['name']), reverse=True):
4546 data = devall[type][name]
4547 data['average'] = data['total'] / data['count']
4548 if data['average'] < limit:
4549 continue
4550 # row classes - alternate row color
4551 rcls = ['alt'] if num % 2 == 1 else []
4552 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4553 html += tdr.format(data['name']) # name
4554 html += td.format('%.3f ms' % data['average']) # average
4555 html += td.format(data['count']) # count
4556 html += td.format('%.3f ms' % data['worst']) # worst
4557 html += td.format(data['host']) # host
4558 html += tdlink.format(data['url']) # url
4559 html += '</tr>\n'
4560 num += 1
4561 html += '</table>\n'
4562
4563 # flush the data to file
4564 hf = open(htmlfile, 'w')
4565 hf.write(html+'</body>\n</html>\n')
4566 hf.close()
4567 return devall
4568
4569def createHTMLIssuesSummary(testruns, issues, htmlfile, title, extra=''):
4570 multihost = len([e for e in issues if len(e['urls']) > 1]) > 0
4571 html = summaryCSS('Issues Summary - SleepGraph', False)
4572 total = len(testruns)
4573
4574 # generate the html
4575 th = '\t<th>{0}</th>\n'
4576 td = '\t<td align={0}>{1}</td>\n'
4577 tdlink = '<a href="{1}">{0}</a>'
4578 subtitle = '%d issues' % len(issues) if len(issues) > 0 else 'no issues'
4579 html += '<div class="stamp">%s (%s)</div><table>\n' % (title, subtitle)
4580 html += '<tr>\n' + th.format('Issue') + th.format('Count')
4581 if multihost:
4582 html += th.format('Hosts')
4583 html += th.format('Tests') + th.format('Fail Rate') +\
4584 th.format('First Instance') + '</tr>\n'
4585
4586 num = 0
4587 for e in sorted(issues, key=lambda v:v['count'], reverse=True):
4588 testtotal = 0
4589 links = []
4590 for host in sorted(e['urls']):
4591 links.append(tdlink.format(host, e['urls'][host][0]))
4592 testtotal += len(e['urls'][host])
4593 rate = '%d/%d (%.2f%%)' % (testtotal, total, 100*float(testtotal)/float(total))
4594 # row classes - alternate row color
4595 rcls = ['alt'] if num % 2 == 1 else []
4596 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4597 html += td.format('left', e['line']) # issue
4598 html += td.format('center', e['count']) # count
4599 if multihost:
4600 html += td.format('center', len(e['urls'])) # hosts
4601 html += td.format('center', testtotal) # test count
4602 html += td.format('center', rate) # test rate
4603 html += td.format('center nowrap', '<br>'.join(links)) # links
4604 html += '</tr>\n'
4605 num += 1
4606
4607 # flush the data to file
4608 hf = open(htmlfile, 'w')
4609 hf.write(html+'</table>\n'+extra+'</body>\n</html>\n')
4610 hf.close()
4611 return issues
4612
4613def ordinal(value):
4614 suffix = 'th'
4615 if value < 10 or value > 19:
4616 if value % 10 == 1:
4617 suffix = 'st'
4618 elif value % 10 == 2:
4619 suffix = 'nd'
4620 elif value % 10 == 3:
4621 suffix = 'rd'
4622 return '%d%s' % (value, suffix)
4623
4624# Function: createHTML
4625# Description:
4626# Create the output html file from the resident test data
4627# Arguments:
4628# testruns: array of Data objects from parseKernelLog or parseTraceLog
4629# Output:
4630# True if the html file was created, false if it failed
4631def createHTML(testruns, testfail):
4632 if len(testruns) < 1:
4633 pprint('ERROR: Not enough test data to build a timeline')
4634 return
4635
4636 kerror = False
4637 for data in testruns:
4638 if data.kerror:
4639 kerror = True
4640 if(sysvals.suspendmode in ['freeze', 'standby']):
4641 data.trimFreezeTime(testruns[-1].tSuspended)
4642 else:
4643 data.getMemTime()
4644
4645 # html function templates
4646 html_error = '<div id="{1}" title="kernel error/warning" class="err" style="right:{0}%">{2}→</div>\n'
4647 html_traceevent = '<div title="{0}" class="traceevent{6}" style="left:{1}%;top:{2}px;height:{3}px;width:{4}%;line-height:{3}px;{7}">{5}</div>\n'
4648 html_cpuexec = '<div class="jiffie" style="left:{0}%;top:{1}px;height:{2}px;width:{3}%;background:{4};"></div>\n'
4649 html_timetotal = '<table class="time1">\n<tr>'\
4650 '<td class="green" title="{3}">{2} Suspend Time: <b>{0} ms</b></td>'\
4651 '<td class="yellow" title="{4}">{2} Resume Time: <b>{1} ms</b></td>'\
4652 '</tr>\n</table>\n'
4653 html_timetotal2 = '<table class="time1">\n<tr>'\
4654 '<td class="green" title="{4}">{3} Suspend Time: <b>{0} ms</b></td>'\
4655 '<td class="gray" title="time spent in low-power mode with clock running">'+sysvals.suspendmode+' time: <b>{1} ms</b></td>'\
4656 '<td class="yellow" title="{5}">{3} Resume Time: <b>{2} ms</b></td>'\
4657 '</tr>\n</table>\n'
4658 html_timetotal3 = '<table class="time1">\n<tr>'\
4659 '<td class="green">Execution Time: <b>{0} ms</b></td>'\
4660 '<td class="yellow">Command: <b>{1}</b></td>'\
4661 '</tr>\n</table>\n'
4662 html_fail = '<table class="testfail"><tr><td>{0}</td></tr></table>\n'
4663 html_kdesc = '<td class="{3}" title="time spent in kernel execution">{0}Kernel {2}: {1} ms</td>'
4664 html_fwdesc = '<td class="{3}" title="time spent in firmware">{0}Firmware {2}: {1} ms</td>'
4665 html_wifdesc = '<td class="yellow" title="time for wifi to reconnect after resume complete ({2})">{0}Wifi Resume: {1}</td>'
4666
4667 # html format variables
4668 scaleH = 20
4669 if kerror:
4670 scaleH = 40
4671
4672 # device timeline
4673 devtl = Timeline(30, scaleH)
4674
4675 # write the test title and general info header
4676 devtl.createHeader(sysvals, testruns[0].stamp)
4677
4678 # Generate the header for this timeline
4679 for data in testruns:
4680 tTotal = data.end - data.start
4681 if(tTotal == 0):
4682 doError('No timeline data')
4683 if sysvals.suspendmode == 'command':
4684 run_time = '%.0f' % (tTotal * 1000)
4685 if sysvals.testcommand:
4686 testdesc = sysvals.testcommand
4687 else:
4688 testdesc = 'unknown'
4689 if(len(testruns) > 1):
4690 testdesc = ordinal(data.testnumber+1)+' '+testdesc
4691 thtml = html_timetotal3.format(run_time, testdesc)
4692 devtl.html += thtml
4693 continue
4694 # typical full suspend/resume header
4695 stot, rtot = sktime, rktime = data.getTimeValues()
4696 ssrc, rsrc, testdesc, testdesc2 = ['kernel'], ['kernel'], 'Kernel', ''
4697 if data.fwValid:
4698 stot += (data.fwSuspend/1000000.0)
4699 rtot += (data.fwResume/1000000.0)
4700 ssrc.append('firmware')
4701 rsrc.append('firmware')
4702 testdesc = 'Total'
4703 if 'time' in data.wifi and data.wifi['stat'] != 'timeout':
4704 rtot += data.end - data.tKernRes + (data.wifi['time'] * 1000.0)
4705 rsrc.append('wifi')
4706 testdesc = 'Total'
4707 suspend_time, resume_time = '%.3f' % stot, '%.3f' % rtot
4708 stitle = 'time from kernel suspend start to %s mode [%s time]' % \
4709 (sysvals.suspendmode, ' & '.join(ssrc))
4710 rtitle = 'time from %s mode to kernel resume complete [%s time]' % \
4711 (sysvals.suspendmode, ' & '.join(rsrc))
4712 if(len(testruns) > 1):
4713 testdesc = testdesc2 = ordinal(data.testnumber+1)
4714 testdesc2 += ' '
4715 if(len(data.tLow) == 0):
4716 thtml = html_timetotal.format(suspend_time, \
4717 resume_time, testdesc, stitle, rtitle)
4718 else:
4719 low_time = '+'.join(data.tLow)
4720 thtml = html_timetotal2.format(suspend_time, low_time, \
4721 resume_time, testdesc, stitle, rtitle)
4722 devtl.html += thtml
4723 if not data.fwValid and 'dev' not in data.wifi:
4724 continue
4725 # extra detail when the times come from multiple sources
4726 thtml = '<table class="time2">\n<tr>'
4727 thtml += html_kdesc.format(testdesc2, '%.3f'%sktime, 'Suspend', 'green')
4728 if data.fwValid:
4729 sftime = '%.3f'%(data.fwSuspend / 1000000.0)
4730 rftime = '%.3f'%(data.fwResume / 1000000.0)
4731 thtml += html_fwdesc.format(testdesc2, sftime, 'Suspend', 'green')
4732 thtml += html_fwdesc.format(testdesc2, rftime, 'Resume', 'yellow')
4733 thtml += html_kdesc.format(testdesc2, '%.3f'%rktime, 'Resume', 'yellow')
4734 if 'time' in data.wifi:
4735 if data.wifi['stat'] != 'timeout':
4736 wtime = '%.0f ms'%(data.end - data.tKernRes + (data.wifi['time'] * 1000.0))
4737 else:
4738 wtime = 'TIMEOUT'
4739 thtml += html_wifdesc.format(testdesc2, wtime, data.wifi['dev'])
4740 thtml += '</tr>\n</table>\n'
4741 devtl.html += thtml
4742 if testfail:
4743 devtl.html += html_fail.format(testfail)
4744
4745 # time scale for potentially multiple datasets
4746 t0 = testruns[0].start
4747 tMax = testruns[-1].end
4748 tTotal = tMax - t0
4749
4750 # determine the maximum number of rows we need to draw
4751 fulllist = []
4752 threadlist = []
4753 pscnt = 0
4754 devcnt = 0
4755 for data in testruns:
4756 data.selectTimelineDevices('%f', tTotal, sysvals.mindevlen)
4757 for group in data.devicegroups:
4758 devlist = []
4759 for phase in group:
4760 for devname in sorted(data.tdevlist[phase]):
4761 d = DevItem(data.testnumber, phase, data.dmesg[phase]['list'][devname])
4762 devlist.append(d)
4763 if d.isa('kth'):
4764 threadlist.append(d)
4765 else:
4766 if d.isa('ps'):
4767 pscnt += 1
4768 else:
4769 devcnt += 1
4770 fulllist.append(d)
4771 if sysvals.mixedphaseheight:
4772 devtl.getPhaseRows(devlist)
4773 if not sysvals.mixedphaseheight:
4774 if len(threadlist) > 0 and len(fulllist) > 0:
4775 if pscnt > 0 and devcnt > 0:
4776 msg = 'user processes & device pm callbacks'
4777 elif pscnt > 0:
4778 msg = 'user processes'
4779 else:
4780 msg = 'device pm callbacks'
4781 d = testruns[0].addHorizontalDivider(msg, testruns[-1].end)
4782 fulllist.insert(0, d)
4783 devtl.getPhaseRows(fulllist)
4784 if len(threadlist) > 0:
4785 d = testruns[0].addHorizontalDivider('asynchronous kernel threads', testruns[-1].end)
4786 threadlist.insert(0, d)
4787 devtl.getPhaseRows(threadlist, devtl.rows)
4788 devtl.calcTotalRows()
4789
4790 # draw the full timeline
4791 devtl.createZoomBox(sysvals.suspendmode, len(testruns))
4792 for data in testruns:
4793 # draw each test run and block chronologically
4794 phases = {'suspend':[],'resume':[]}
4795 for phase in data.sortedPhases():
4796 if data.dmesg[phase]['start'] >= data.tSuspended:
4797 phases['resume'].append(phase)
4798 else:
4799 phases['suspend'].append(phase)
4800 # now draw the actual timeline blocks
4801 for dir in phases:
4802 # draw suspend and resume blocks separately
4803 bname = '%s%d' % (dir[0], data.testnumber)
4804 if dir == 'suspend':
4805 m0 = data.start
4806 mMax = data.tSuspended
4807 left = '%f' % (((m0-t0)*100.0)/tTotal)
4808 else:
4809 m0 = data.tSuspended
4810 mMax = data.end
4811 # in an x2 run, remove any gap between blocks
4812 if len(testruns) > 1 and data.testnumber == 0:
4813 mMax = testruns[1].start
4814 left = '%f' % ((((m0-t0)*100.0)+sysvals.srgap/2)/tTotal)
4815 mTotal = mMax - m0
4816 # if a timeline block is 0 length, skip altogether
4817 if mTotal == 0:
4818 continue
4819 width = '%f' % (((mTotal*100.0)-sysvals.srgap/2)/tTotal)
4820 devtl.html += devtl.html_tblock.format(bname, left, width, devtl.scaleH)
4821 for b in phases[dir]:
4822 # draw the phase color background
4823 phase = data.dmesg[b]
4824 length = phase['end']-phase['start']
4825 left = '%f' % (((phase['start']-m0)*100.0)/mTotal)
4826 width = '%f' % ((length*100.0)/mTotal)
4827 devtl.html += devtl.html_phase.format(left, width, \
4828 '%.3f'%devtl.scaleH, '%.3f'%devtl.bodyH, \
4829 data.dmesg[b]['color'], '')
4830 for e in data.errorinfo[dir]:
4831 # draw red lines for any kernel errors found
4832 type, t, idx1, idx2 = e
4833 id = '%d_%d' % (idx1, idx2)
4834 right = '%f' % (((mMax-t)*100.0)/mTotal)
4835 devtl.html += html_error.format(right, id, type)
4836 for b in phases[dir]:
4837 # draw the devices for this phase
4838 phaselist = data.dmesg[b]['list']
4839 for d in sorted(data.tdevlist[b]):
4840 dname = d if ('[' not in d or 'CPU' in d) else d.split('[')[0]
4841 name, dev = dname, phaselist[d]
4842 drv = xtraclass = xtrainfo = xtrastyle = ''
4843 if 'htmlclass' in dev:
4844 xtraclass = dev['htmlclass']
4845 if 'color' in dev:
4846 xtrastyle = 'background:%s;' % dev['color']
4847 if(d in sysvals.devprops):
4848 name = sysvals.devprops[d].altName(d)
4849 xtraclass = sysvals.devprops[d].xtraClass()
4850 xtrainfo = sysvals.devprops[d].xtraInfo()
4851 elif xtraclass == ' kth':
4852 xtrainfo = ' kernel_thread'
4853 if('drv' in dev and dev['drv']):
4854 drv = ' {%s}' % dev['drv']
4855 rowheight = devtl.phaseRowHeight(data.testnumber, b, dev['row'])
4856 rowtop = devtl.phaseRowTop(data.testnumber, b, dev['row'])
4857 top = '%.3f' % (rowtop + devtl.scaleH)
4858 left = '%f' % (((dev['start']-m0)*100)/mTotal)
4859 width = '%f' % (((dev['end']-dev['start'])*100)/mTotal)
4860 length = ' (%0.3f ms) ' % ((dev['end']-dev['start'])*1000)
4861 title = name+drv+xtrainfo+length
4862 if sysvals.suspendmode == 'command':
4863 title += sysvals.testcommand
4864 elif xtraclass == ' ps':
4865 if 'suspend' in b:
4866 title += 'pre_suspend_process'
4867 else:
4868 title += 'post_resume_process'
4869 else:
4870 title += b
4871 devtl.html += devtl.html_device.format(dev['id'], \
4872 title, left, top, '%.3f'%rowheight, width, \
4873 dname+drv, xtraclass, xtrastyle)
4874 if('cpuexec' in dev):
4875 for t in sorted(dev['cpuexec']):
4876 start, end = t
4877 height = '%.3f' % (rowheight/3)
4878 top = '%.3f' % (rowtop + devtl.scaleH + 2*rowheight/3)
4879 left = '%f' % (((start-m0)*100)/mTotal)
4880 width = '%f' % ((end-start)*100/mTotal)
4881 color = 'rgba(255, 0, 0, %f)' % dev['cpuexec'][t]
4882 devtl.html += \
4883 html_cpuexec.format(left, top, height, width, color)
4884 if('src' not in dev):
4885 continue
4886 # draw any trace events for this device
4887 for e in dev['src']:
4888 if e.length == 0:
4889 continue
4890 height = '%.3f' % devtl.rowH
4891 top = '%.3f' % (rowtop + devtl.scaleH + (e.row*devtl.rowH))
4892 left = '%f' % (((e.time-m0)*100)/mTotal)
4893 width = '%f' % (e.length*100/mTotal)
4894 xtrastyle = ''
4895 if e.color:
4896 xtrastyle = 'background:%s;' % e.color
4897 devtl.html += \
4898 html_traceevent.format(e.title(), \
4899 left, top, height, width, e.text(), '', xtrastyle)
4900 # draw the time scale, try to make the number of labels readable
4901 devtl.createTimeScale(m0, mMax, tTotal, dir)
4902 devtl.html += '</div>\n'
4903
4904 # timeline is finished
4905 devtl.html += '</div>\n</div>\n'
4906
4907 # draw a legend which describes the phases by color
4908 if sysvals.suspendmode != 'command':
4909 phasedef = testruns[-1].phasedef
4910 devtl.html += '<div class="legend">\n'
4911 pdelta = 100.0/len(phasedef.keys())
4912 pmargin = pdelta / 4.0
4913 for phase in sorted(phasedef, key=lambda k:phasedef[k]['order']):
4914 id, p = '', phasedef[phase]
4915 for word in phase.split('_'):
4916 id += word[0]
4917 order = '%.2f' % ((p['order'] * pdelta) + pmargin)
4918 name = phase.replace('_', ' ')
4919 devtl.html += devtl.html_legend.format(order, p['color'], name, id)
4920 devtl.html += '</div>\n'
4921
4922 hf = open(sysvals.htmlfile, 'w')
4923 addCSS(hf, sysvals, len(testruns), kerror)
4924
4925 # write the device timeline
4926 hf.write(devtl.html)
4927 hf.write('<div id="devicedetailtitle"></div>\n')
4928 hf.write('<div id="devicedetail" style="display:none;">\n')
4929 # draw the colored boxes for the device detail section
4930 for data in testruns:
4931 hf.write('<div id="devicedetail%d">\n' % data.testnumber)
4932 pscolor = 'linear-gradient(to top left, #ccc, #eee)'
4933 hf.write(devtl.html_phaselet.format('pre_suspend_process', \
4934 '0', '0', pscolor))
4935 for b in data.sortedPhases():
4936 phase = data.dmesg[b]
4937 length = phase['end']-phase['start']
4938 left = '%.3f' % (((phase['start']-t0)*100.0)/tTotal)
4939 width = '%.3f' % ((length*100.0)/tTotal)
4940 hf.write(devtl.html_phaselet.format(b, left, width, \
4941 data.dmesg[b]['color']))
4942 hf.write(devtl.html_phaselet.format('post_resume_process', \
4943 '0', '0', pscolor))
4944 if sysvals.suspendmode == 'command':
4945 hf.write(devtl.html_phaselet.format('cmdexec', '0', '0', pscolor))
4946 hf.write('</div>\n')
4947 hf.write('</div>\n')
4948
4949 # write the ftrace data (callgraph)
4950 if sysvals.cgtest >= 0 and len(testruns) > sysvals.cgtest:
4951 data = testruns[sysvals.cgtest]
4952 else:
4953 data = testruns[-1]
4954 if sysvals.usecallgraph:
4955 addCallgraphs(sysvals, hf, data)
4956
4957 # add the test log as a hidden div
4958 if sysvals.testlog and sysvals.logmsg:
4959 hf.write('<div id="testlog" style="display:none;">\n'+sysvals.logmsg+'</div>\n')
4960 # add the dmesg log as a hidden div
4961 if sysvals.dmesglog and sysvals.dmesgfile:
4962 hf.write('<div id="dmesglog" style="display:none;">\n')
4963 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
4964 for line in lf:
4965 line = line.replace('<', '<').replace('>', '>')
4966 hf.write(line)
4967 lf.close()
4968 hf.write('</div>\n')
4969 # add the ftrace log as a hidden div
4970 if sysvals.ftracelog and sysvals.ftracefile:
4971 hf.write('<div id="ftracelog" style="display:none;">\n')
4972 lf = sysvals.openlog(sysvals.ftracefile, 'r')
4973 for line in lf:
4974 hf.write(line)
4975 lf.close()
4976 hf.write('</div>\n')
4977
4978 # write the footer and close
4979 addScriptCode(hf, testruns)
4980 hf.write('</body>\n</html>\n')
4981 hf.close()
4982 return True
4983
4984def addCSS(hf, sv, testcount=1, kerror=False, extra=''):
4985 kernel = sv.stamp['kernel']
4986 host = sv.hostname[0].upper()+sv.hostname[1:]
4987 mode = sv.suspendmode
4988 if sv.suspendmode in suspendmodename:
4989 mode = suspendmodename[sv.suspendmode]
4990 title = host+' '+mode+' '+kernel
4991
4992 # various format changes by flags
4993 cgchk = 'checked'
4994 cgnchk = 'not(:checked)'
4995 if sv.cgexp:
4996 cgchk = 'not(:checked)'
4997 cgnchk = 'checked'
4998
4999 hoverZ = 'z-index:8;'
5000 if sv.usedevsrc:
5001 hoverZ = ''
5002
5003 devlistpos = 'absolute'
5004 if testcount > 1:
5005 devlistpos = 'relative'
5006
5007 scaleTH = 20
5008 if kerror:
5009 scaleTH = 60
5010
5011 # write the html header first (html head, css code, up to body start)
5012 html_header = '<!DOCTYPE html>\n<html>\n<head>\n\
5013 <meta http-equiv="content-type" content="text/html; charset=UTF-8">\n\
5014 <title>'+title+'</title>\n\
5015 <style type=\'text/css\'>\n\
5016 body {overflow-y:scroll;}\n\
5017 .stamp {width:100%;text-align:center;background:gray;line-height:30px;color:white;font:25px Arial;}\n\
5018 .stamp.sysinfo {font:10px Arial;}\n\
5019 .callgraph {margin-top:30px;box-shadow:5px 5px 20px black;}\n\
5020 .callgraph article * {padding-left:28px;}\n\
5021 h1 {color:black;font:bold 30px Times;}\n\
5022 t0 {color:black;font:bold 30px Times;}\n\
5023 t1 {color:black;font:30px Times;}\n\
5024 t2 {color:black;font:25px Times;}\n\
5025 t3 {color:black;font:20px Times;white-space:nowrap;}\n\
5026 t4 {color:black;font:bold 30px Times;line-height:60px;white-space:nowrap;}\n\
5027 cS {font:bold 13px Times;}\n\
5028 table {width:100%;}\n\
5029 .gray {background:rgba(80,80,80,0.1);}\n\
5030 .green {background:rgba(204,255,204,0.4);}\n\
5031 .purple {background:rgba(128,0,128,0.2);}\n\
5032 .yellow {background:rgba(255,255,204,0.4);}\n\
5033 .blue {background:rgba(169,208,245,0.4);}\n\
5034 .time1 {font:22px Arial;border:1px solid;}\n\
5035 .time2 {font:15px Arial;border-bottom:1px solid;border-left:1px solid;border-right:1px solid;}\n\
5036 .testfail {font:bold 22px Arial;color:red;border:1px dashed;}\n\
5037 td {text-align:center;}\n\
5038 r {color:#500000;font:15px Tahoma;}\n\
5039 n {color:#505050;font:15px Tahoma;}\n\
5040 .tdhl {color:red;}\n\
5041 .hide {display:none;}\n\
5042 .pf {display:none;}\n\
5043 .pf:'+cgchk+' + label {background:url(\'data:image/svg+xml;utf,<?xml version="1.0" standalone="no"?><svg xmlns="http://www.w3.org/2000/svg" height="18" width="18" version="1.1"><circle cx="9" cy="9" r="8" stroke="black" stroke-width="1" fill="white"/><rect x="4" y="8" width="10" height="2" style="fill:black;stroke-width:0"/><rect x="8" y="4" width="2" height="10" style="fill:black;stroke-width:0"/></svg>\') no-repeat left center;}\n\
5044 .pf:'+cgnchk+' ~ label {background:url(\'data:image/svg+xml;utf,<?xml version="1.0" standalone="no"?><svg xmlns="http://www.w3.org/2000/svg" height="18" width="18" version="1.1"><circle cx="9" cy="9" r="8" stroke="black" stroke-width="1" fill="white"/><rect x="4" y="8" width="10" height="2" style="fill:black;stroke-width:0"/></svg>\') no-repeat left center;}\n\
5045 .pf:'+cgchk+' ~ *:not(:nth-child(2)) {display:none;}\n\
5046 .zoombox {position:relative;width:100%;overflow-x:scroll;-webkit-user-select:none;-moz-user-select:none;user-select:none;}\n\
5047 .timeline {position:relative;font-size:14px;cursor:pointer;width:100%; overflow:hidden;background:linear-gradient(#cccccc, white);}\n\
5048 .thread {position:absolute;height:0%;overflow:hidden;z-index:7;line-height:30px;font-size:14px;border:1px solid;text-align:center;white-space:nowrap;}\n\
5049 .thread.ps {border-radius:3px;background:linear-gradient(to top, #ccc, #eee);}\n\
5050 .thread:hover {background:white;border:1px solid red;'+hoverZ+'}\n\
5051 .thread.sec,.thread.sec:hover {background:black;border:0;color:white;line-height:15px;font-size:10px;}\n\
5052 .hover {background:white;border:1px solid red;'+hoverZ+'}\n\
5053 .hover.sync {background:white;}\n\
5054 .hover.bg,.hover.kth,.hover.sync,.hover.ps {background:white;}\n\
5055 .jiffie {position:absolute;pointer-events: none;z-index:8;}\n\
5056 .traceevent {position:absolute;font-size:10px;z-index:7;overflow:hidden;color:black;text-align:center;white-space:nowrap;border-radius:5px;border:1px solid black;background:linear-gradient(to bottom right,#CCC,#969696);}\n\
5057 .traceevent:hover {color:white;font-weight:bold;border:1px solid white;}\n\
5058 .phase {position:absolute;overflow:hidden;border:0px;text-align:center;}\n\
5059 .phaselet {float:left;overflow:hidden;border:0px;text-align:center;min-height:100px;font-size:24px;}\n\
5060 .t {position:absolute;line-height:'+('%d'%scaleTH)+'px;pointer-events:none;top:0;height:100%;border-right:1px solid black;z-index:6;}\n\
5061 .err {position:absolute;top:0%;height:100%;border-right:3px solid red;color:red;font:bold 14px Times;line-height:18px;}\n\
5062 .legend {position:relative; width:100%; height:40px; text-align:center;margin-bottom:20px}\n\
5063 .legend .square {position:absolute;cursor:pointer;top:10px; width:0px;height:20px;border:1px solid;padding-left:20px;}\n\
5064 button {height:40px;width:200px;margin-bottom:20px;margin-top:20px;font-size:24px;}\n\
5065 .btnfmt {position:relative;float:right;height:25px;width:auto;margin-top:3px;margin-bottom:0;font-size:10px;text-align:center;}\n\
5066 .devlist {position:'+devlistpos+';width:190px;}\n\
5067 a:link {color:white;text-decoration:none;}\n\
5068 a:visited {color:white;}\n\
5069 a:hover {color:white;}\n\
5070 a:active {color:white;}\n\
5071 .version {position:relative;float:left;color:white;font-size:10px;line-height:30px;margin-left:10px;}\n\
5072 #devicedetail {min-height:100px;box-shadow:5px 5px 20px black;}\n\
5073 .tblock {position:absolute;height:100%;background:#ddd;}\n\
5074 .tback {position:absolute;width:100%;background:linear-gradient(#ccc, #ddd);}\n\
5075 .bg {z-index:1;}\n\
5076'+extra+'\
5077 </style>\n</head>\n<body>\n'
5078 hf.write(html_header)
5079
5080# Function: addScriptCode
5081# Description:
5082# Adds the javascript code to the output html
5083# Arguments:
5084# hf: the open html file pointer
5085# testruns: array of Data objects from parseKernelLog or parseTraceLog
5086def addScriptCode(hf, testruns):
5087 t0 = testruns[0].start * 1000
5088 tMax = testruns[-1].end * 1000
5089 hf.write('<script type="text/javascript">\n');
5090 # create an array in javascript memory with the device details
5091 detail = ' var devtable = [];\n'
5092 for data in testruns:
5093 topo = data.deviceTopology()
5094 detail += ' devtable[%d] = "%s";\n' % (data.testnumber, topo)
5095 detail += ' var bounds = [%f,%f];\n' % (t0, tMax)
5096 # add the code which will manipulate the data in the browser
5097 hf.write(detail);
5098 script_code = r""" var resolution = -1;
5099 var dragval = [0, 0];
5100 function redrawTimescale(t0, tMax, tS) {
5101 var rline = '<div class="t" style="left:0;border-left:1px solid black;border-right:0;">';
5102 var tTotal = tMax - t0;
5103 var list = document.getElementsByClassName("tblock");
5104 for (var i = 0; i < list.length; i++) {
5105 var timescale = list[i].getElementsByClassName("timescale")[0];
5106 var m0 = t0 + (tTotal*parseFloat(list[i].style.left)/100);
5107 var mTotal = tTotal*parseFloat(list[i].style.width)/100;
5108 var mMax = m0 + mTotal;
5109 var html = "";
5110 var divTotal = Math.floor(mTotal/tS) + 1;
5111 if(divTotal > 1000) continue;
5112 var divEdge = (mTotal - tS*(divTotal-1))*100/mTotal;
5113 var pos = 0.0, val = 0.0;
5114 for (var j = 0; j < divTotal; j++) {
5115 var htmlline = "";
5116 var mode = list[i].id[5];
5117 if(mode == "s") {
5118 pos = 100 - (((j)*tS*100)/mTotal) - divEdge;
5119 val = (j-divTotal+1)*tS;
5120 if(j == divTotal - 1)
5121 htmlline = '<div class="t" style="right:'+pos+'%"><cS>S→</cS></div>';
5122 else
5123 htmlline = '<div class="t" style="right:'+pos+'%">'+val+'ms</div>';
5124 } else {
5125 pos = 100 - (((j)*tS*100)/mTotal);
5126 val = (j)*tS;
5127 htmlline = '<div class="t" style="right:'+pos+'%">'+val+'ms</div>';
5128 if(j == 0)
5129 if(mode == "r")
5130 htmlline = rline+"<cS>←R</cS></div>";
5131 else
5132 htmlline = rline+"<cS>0ms</div>";
5133 }
5134 html += htmlline;
5135 }
5136 timescale.innerHTML = html;
5137 }
5138 }
5139 function zoomTimeline() {
5140 var dmesg = document.getElementById("dmesg");
5141 var zoombox = document.getElementById("dmesgzoombox");
5142 var left = zoombox.scrollLeft;
5143 var val = parseFloat(dmesg.style.width);
5144 var newval = 100;
5145 var sh = window.outerWidth / 2;
5146 if(this.id == "zoomin") {
5147 newval = val * 1.2;
5148 if(newval > 910034) newval = 910034;
5149 dmesg.style.width = newval+"%";
5150 zoombox.scrollLeft = ((left + sh) * newval / val) - sh;
5151 } else if (this.id == "zoomout") {
5152 newval = val / 1.2;
5153 if(newval < 100) newval = 100;
5154 dmesg.style.width = newval+"%";
5155 zoombox.scrollLeft = ((left + sh) * newval / val) - sh;
5156 } else {
5157 zoombox.scrollLeft = 0;
5158 dmesg.style.width = "100%";
5159 }
5160 var tS = [10000, 5000, 2000, 1000, 500, 200, 100, 50, 20, 10, 5, 2, 1];
5161 var t0 = bounds[0];
5162 var tMax = bounds[1];
5163 var tTotal = tMax - t0;
5164 var wTotal = tTotal * 100.0 / newval;
5165 var idx = 7*window.innerWidth/1100;
5166 for(var i = 0; (i < tS.length)&&((wTotal / tS[i]) < idx); i++);
5167 if(i >= tS.length) i = tS.length - 1;
5168 if(tS[i] == resolution) return;
5169 resolution = tS[i];
5170 redrawTimescale(t0, tMax, tS[i]);
5171 }
5172 function deviceName(title) {
5173 var name = title.slice(0, title.indexOf(" ("));
5174 return name;
5175 }
5176 function deviceHover() {
5177 var name = deviceName(this.title);
5178 var dmesg = document.getElementById("dmesg");
5179 var dev = dmesg.getElementsByClassName("thread");
5180 var cpu = -1;
5181 if(name.match("CPU_ON\[[0-9]*\]"))
5182 cpu = parseInt(name.slice(7));
5183 else if(name.match("CPU_OFF\[[0-9]*\]"))
5184 cpu = parseInt(name.slice(8));
5185 for (var i = 0; i < dev.length; i++) {
5186 dname = deviceName(dev[i].title);
5187 var cname = dev[i].className.slice(dev[i].className.indexOf("thread"));
5188 if((cpu >= 0 && dname.match("CPU_O[NF]*\\[*"+cpu+"\\]")) ||
5189 (name == dname))
5190 {
5191 dev[i].className = "hover "+cname;
5192 } else {
5193 dev[i].className = cname;
5194 }
5195 }
5196 }
5197 function deviceUnhover() {
5198 var dmesg = document.getElementById("dmesg");
5199 var dev = dmesg.getElementsByClassName("thread");
5200 for (var i = 0; i < dev.length; i++) {
5201 dev[i].className = dev[i].className.slice(dev[i].className.indexOf("thread"));
5202 }
5203 }
5204 function deviceTitle(title, total, cpu) {
5205 var prefix = "Total";
5206 if(total.length > 3) {
5207 prefix = "Average";
5208 total[1] = (total[1]+total[3])/2;
5209 total[2] = (total[2]+total[4])/2;
5210 }
5211 var devtitle = document.getElementById("devicedetailtitle");
5212 var name = deviceName(title);
5213 if(cpu >= 0) name = "CPU"+cpu;
5214 var driver = "";
5215 var tS = "<t2>(</t2>";
5216 var tR = "<t2>)</t2>";
5217 if(total[1] > 0)
5218 tS = "<t2>("+prefix+" Suspend:</t2><t0> "+total[1].toFixed(3)+" ms</t0> ";
5219 if(total[2] > 0)
5220 tR = " <t2>"+prefix+" Resume:</t2><t0> "+total[2].toFixed(3)+" ms<t2>)</t2></t0>";
5221 var s = title.indexOf("{");
5222 var e = title.indexOf("}");
5223 if((s >= 0) && (e >= 0))
5224 driver = title.slice(s+1, e) + " <t1>@</t1> ";
5225 if(total[1] > 0 && total[2] > 0)
5226 devtitle.innerHTML = "<t0>"+driver+name+"</t0> "+tS+tR;
5227 else
5228 devtitle.innerHTML = "<t0>"+title+"</t0>";
5229 return name;
5230 }
5231 function deviceDetail() {
5232 var devinfo = document.getElementById("devicedetail");
5233 devinfo.style.display = "block";
5234 var name = deviceName(this.title);
5235 var cpu = -1;
5236 if(name.match("CPU_ON\[[0-9]*\]"))
5237 cpu = parseInt(name.slice(7));
5238 else if(name.match("CPU_OFF\[[0-9]*\]"))
5239 cpu = parseInt(name.slice(8));
5240 var dmesg = document.getElementById("dmesg");
5241 var dev = dmesg.getElementsByClassName("thread");
5242 var idlist = [];
5243 var pdata = [[]];
5244 if(document.getElementById("devicedetail1"))
5245 pdata = [[], []];
5246 var pd = pdata[0];
5247 var total = [0.0, 0.0, 0.0];
5248 for (var i = 0; i < dev.length; i++) {
5249 dname = deviceName(dev[i].title);
5250 if((cpu >= 0 && dname.match("CPU_O[NF]*\\[*"+cpu+"\\]")) ||
5251 (name == dname))
5252 {
5253 idlist[idlist.length] = dev[i].id;
5254 var tidx = 1;
5255 if(dev[i].id[0] == "a") {
5256 pd = pdata[0];
5257 } else {
5258 if(pdata.length == 1) pdata[1] = [];
5259 if(total.length == 3) total[3]=total[4]=0.0;
5260 pd = pdata[1];
5261 tidx = 3;
5262 }
5263 var info = dev[i].title.split(" ");
5264 var pname = info[info.length-1];
5265 var length = parseFloat(info[info.length-3].slice(1));
5266 if (pname in pd)
5267 pd[pname] += length;
5268 else
5269 pd[pname] = length;
5270 total[0] += length;
5271 if(pname.indexOf("suspend") >= 0)
5272 total[tidx] += length;
5273 else
5274 total[tidx+1] += length;
5275 }
5276 }
5277 var devname = deviceTitle(this.title, total, cpu);
5278 var left = 0.0;
5279 for (var t = 0; t < pdata.length; t++) {
5280 pd = pdata[t];
5281 devinfo = document.getElementById("devicedetail"+t);
5282 var phases = devinfo.getElementsByClassName("phaselet");
5283 for (var i = 0; i < phases.length; i++) {
5284 if(phases[i].id in pd) {
5285 var w = 100.0*pd[phases[i].id]/total[0];
5286 var fs = 32;
5287 if(w < 8) fs = 4*w | 0;
5288 var fs2 = fs*3/4;
5289 phases[i].style.width = w+"%";
5290 phases[i].style.left = left+"%";
5291 phases[i].title = phases[i].id+" "+pd[phases[i].id]+" ms";
5292 left += w;
5293 var time = "<t4 style=\"font-size:"+fs+"px\">"+pd[phases[i].id].toFixed(3)+" ms<br></t4>";
5294 var pname = "<t3 style=\"font-size:"+fs2+"px\">"+phases[i].id.replace(new RegExp("_", "g"), " ")+"</t3>";
5295 phases[i].innerHTML = time+pname;
5296 } else {
5297 phases[i].style.width = "0%";
5298 phases[i].style.left = left+"%";
5299 }
5300 }
5301 }
5302 if(typeof devstats !== 'undefined')
5303 callDetail(this.id, this.title);
5304 var cglist = document.getElementById("callgraphs");
5305 if(!cglist) return;
5306 var cg = cglist.getElementsByClassName("atop");
5307 if(cg.length < 10) return;
5308 for (var i = 0; i < cg.length; i++) {
5309 cgid = cg[i].id.split("x")[0]
5310 if(idlist.indexOf(cgid) >= 0) {
5311 cg[i].style.display = "block";
5312 } else {
5313 cg[i].style.display = "none";
5314 }
5315 }
5316 }
5317 function callDetail(devid, devtitle) {
5318 if(!(devid in devstats) || devstats[devid].length < 1)
5319 return;
5320 var list = devstats[devid];
5321 var tmp = devtitle.split(" ");
5322 var name = tmp[0], phase = tmp[tmp.length-1];
5323 var dd = document.getElementById(phase);
5324 var total = parseFloat(tmp[1].slice(1));
5325 var mlist = [];
5326 var maxlen = 0;
5327 var info = []
5328 for(var i in list) {
5329 if(list[i][0] == "@") {
5330 info = list[i].split("|");
5331 continue;
5332 }
5333 var tmp = list[i].split("|");
5334 var t = parseFloat(tmp[0]), f = tmp[1], c = parseInt(tmp[2]);
5335 var p = (t*100.0/total).toFixed(2);
5336 mlist[mlist.length] = [f, c, t.toFixed(2), p+"%"];
5337 if(f.length > maxlen)
5338 maxlen = f.length;
5339 }
5340 var pad = 5;
5341 if(mlist.length == 0) pad = 30;
5342 var html = '<div style="padding-top:'+pad+'px"><t3> <b>'+name+':</b>';
5343 if(info.length > 2)
5344 html += " start=<b>"+info[1]+"</b>, end=<b>"+info[2]+"</b>";
5345 if(info.length > 3)
5346 html += ", length<i>(w/o overhead)</i>=<b>"+info[3]+" ms</b>";
5347 if(info.length > 4)
5348 html += ", return=<b>"+info[4]+"</b>";
5349 html += "</t3></div>";
5350 if(mlist.length > 0) {
5351 html += '<table class=fstat style="padding-top:'+(maxlen*5)+'px;"><tr><th>Function</th>';
5352 for(var i in mlist)
5353 html += "<td class=vt>"+mlist[i][0]+"</td>";
5354 html += "</tr><tr><th>Calls</th>";
5355 for(var i in mlist)
5356 html += "<td>"+mlist[i][1]+"</td>";
5357 html += "</tr><tr><th>Time(ms)</th>";
5358 for(var i in mlist)
5359 html += "<td>"+mlist[i][2]+"</td>";
5360 html += "</tr><tr><th>Percent</th>";
5361 for(var i in mlist)
5362 html += "<td>"+mlist[i][3]+"</td>";
5363 html += "</tr></table>";
5364 }
5365 dd.innerHTML = html;
5366 var height = (maxlen*5)+100;
5367 dd.style.height = height+"px";
5368 document.getElementById("devicedetail").style.height = height+"px";
5369 }
5370 function callSelect() {
5371 var cglist = document.getElementById("callgraphs");
5372 if(!cglist) return;
5373 var cg = cglist.getElementsByClassName("atop");
5374 for (var i = 0; i < cg.length; i++) {
5375 if(this.id == cg[i].id) {
5376 cg[i].style.display = "block";
5377 } else {
5378 cg[i].style.display = "none";
5379 }
5380 }
5381 }
5382 function devListWindow(e) {
5383 var win = window.open();
5384 var html = "<title>"+e.target.innerHTML+"</title>"+
5385 "<style type=\"text/css\">"+
5386 " ul {list-style-type:circle;padding-left:10px;margin-left:10px;}"+
5387 "</style>"
5388 var dt = devtable[0];
5389 if(e.target.id != "devlist1")
5390 dt = devtable[1];
5391 win.document.write(html+dt);
5392 }
5393 function errWindow() {
5394 var range = this.id.split("_");
5395 var idx1 = parseInt(range[0]);
5396 var idx2 = parseInt(range[1]);
5397 var win = window.open();
5398 var log = document.getElementById("dmesglog");
5399 var title = "<title>dmesg log</title>";
5400 var text = log.innerHTML.split("\n");
5401 var html = "";
5402 for(var i = 0; i < text.length; i++) {
5403 if(i == idx1) {
5404 html += "<e id=target>"+text[i]+"</e>\n";
5405 } else if(i > idx1 && i <= idx2) {
5406 html += "<e>"+text[i]+"</e>\n";
5407 } else {
5408 html += text[i]+"\n";
5409 }
5410 }
5411 win.document.write("<style>e{color:red}</style>"+title+"<pre>"+html+"</pre>");
5412 win.location.hash = "#target";
5413 win.document.close();
5414 }
5415 function logWindow(e) {
5416 var name = e.target.id.slice(4);
5417 var win = window.open();
5418 var log = document.getElementById(name+"log");
5419 var title = "<title>"+document.title.split(" ")[0]+" "+name+" log</title>";
5420 win.document.write(title+"<pre>"+log.innerHTML+"</pre>");
5421 win.document.close();
5422 }
5423 function onMouseDown(e) {
5424 dragval[0] = e.clientX;
5425 dragval[1] = document.getElementById("dmesgzoombox").scrollLeft;
5426 document.onmousemove = onMouseMove;
5427 }
5428 function onMouseMove(e) {
5429 var zoombox = document.getElementById("dmesgzoombox");
5430 zoombox.scrollLeft = dragval[1] + dragval[0] - e.clientX;
5431 }
5432 function onMouseUp(e) {
5433 document.onmousemove = null;
5434 }
5435 function onKeyPress(e) {
5436 var c = e.charCode;
5437 if(c != 42 && c != 43 && c != 45) return;
5438 var click = document.createEvent("Events");
5439 click.initEvent("click", true, false);
5440 if(c == 43)
5441 document.getElementById("zoomin").dispatchEvent(click);
5442 else if(c == 45)
5443 document.getElementById("zoomout").dispatchEvent(click);
5444 else if(c == 42)
5445 document.getElementById("zoomdef").dispatchEvent(click);
5446 }
5447 window.addEventListener("resize", function () {zoomTimeline();});
5448 window.addEventListener("load", function () {
5449 var dmesg = document.getElementById("dmesg");
5450 dmesg.style.width = "100%"
5451 dmesg.onmousedown = onMouseDown;
5452 document.onmouseup = onMouseUp;
5453 document.onkeypress = onKeyPress;
5454 document.getElementById("zoomin").onclick = zoomTimeline;
5455 document.getElementById("zoomout").onclick = zoomTimeline;
5456 document.getElementById("zoomdef").onclick = zoomTimeline;
5457 var list = document.getElementsByClassName("err");
5458 for (var i = 0; i < list.length; i++)
5459 list[i].onclick = errWindow;
5460 var list = document.getElementsByClassName("logbtn");
5461 for (var i = 0; i < list.length; i++)
5462 list[i].onclick = logWindow;
5463 list = document.getElementsByClassName("devlist");
5464 for (var i = 0; i < list.length; i++)
5465 list[i].onclick = devListWindow;
5466 var dev = dmesg.getElementsByClassName("thread");
5467 for (var i = 0; i < dev.length; i++) {
5468 dev[i].onclick = deviceDetail;
5469 dev[i].onmouseover = deviceHover;
5470 dev[i].onmouseout = deviceUnhover;
5471 }
5472 var dev = dmesg.getElementsByClassName("srccall");
5473 for (var i = 0; i < dev.length; i++)
5474 dev[i].onclick = callSelect;
5475 zoomTimeline();
5476 });
5477</script> """
5478 hf.write(script_code);
5479
5480# Function: executeSuspend
5481# Description:
5482# Execute system suspend through the sysfs interface, then copy the output
5483# dmesg and ftrace files to the test output directory.
5484def executeSuspend(quiet=False):
5485 sv, tp, pm = sysvals, sysvals.tpath, ProcessMonitor()
5486 if sv.wifi:
5487 wifi = sv.checkWifi()
5488 sv.dlog('wifi check, connected device is "%s"' % wifi)
5489 testdata = []
5490 # run these commands to prepare the system for suspend
5491 if sv.display:
5492 if not quiet:
5493 pprint('SET DISPLAY TO %s' % sv.display.upper())
5494 ret = sv.displayControl(sv.display)
5495 sv.dlog('xset display %s, ret = %d' % (sv.display, ret))
5496 time.sleep(1)
5497 if sv.sync:
5498 if not quiet:
5499 pprint('SYNCING FILESYSTEMS')
5500 sv.dlog('syncing filesystems')
5501 call('sync', shell=True)
5502 sv.dlog('read dmesg')
5503 sv.initdmesg()
5504 sv.dlog('cmdinfo before')
5505 sv.cmdinfo(True)
5506 sv.start(pm)
5507 # execute however many s/r runs requested
5508 for count in range(1,sv.execcount+1):
5509 # x2delay in between test runs
5510 if(count > 1 and sv.x2delay > 0):
5511 sv.fsetVal('WAIT %d' % sv.x2delay, 'trace_marker')
5512 time.sleep(sv.x2delay/1000.0)
5513 sv.fsetVal('WAIT END', 'trace_marker')
5514 # start message
5515 if sv.testcommand != '':
5516 pprint('COMMAND START')
5517 else:
5518 if(sv.rtcwake):
5519 pprint('SUSPEND START')
5520 else:
5521 pprint('SUSPEND START (press a key to resume)')
5522 # set rtcwake
5523 if(sv.rtcwake):
5524 if not quiet:
5525 pprint('will issue an rtcwake in %d seconds' % sv.rtcwaketime)
5526 sv.dlog('enable RTC wake alarm')
5527 sv.rtcWakeAlarmOn()
5528 # start of suspend trace marker
5529 sv.fsetVal(datetime.now().strftime(sv.tmstart), 'trace_marker')
5530 # predelay delay
5531 if(count == 1 and sv.predelay > 0):
5532 sv.fsetVal('WAIT %d' % sv.predelay, 'trace_marker')
5533 time.sleep(sv.predelay/1000.0)
5534 sv.fsetVal('WAIT END', 'trace_marker')
5535 # initiate suspend or command
5536 sv.dlog('system executing a suspend')
5537 tdata = {'error': ''}
5538 if sv.testcommand != '':
5539 res = call(sv.testcommand+' 2>&1', shell=True);
5540 if res != 0:
5541 tdata['error'] = 'cmd returned %d' % res
5542 else:
5543 s0ixready = sv.s0ixSupport()
5544 mode = sv.suspendmode
5545 if sv.memmode and os.path.exists(sv.mempowerfile):
5546 mode = 'mem'
5547 sv.testVal(sv.mempowerfile, 'radio', sv.memmode)
5548 if sv.diskmode and os.path.exists(sv.diskpowerfile):
5549 mode = 'disk'
5550 sv.testVal(sv.diskpowerfile, 'radio', sv.diskmode)
5551 if sv.acpidebug:
5552 sv.testVal(sv.acpipath, 'acpi', '0xe')
5553 if ((mode == 'freeze') or (sv.memmode == 's2idle')) \
5554 and sv.haveTurbostat():
5555 # execution will pause here
5556 retval, turbo = sv.turbostat(s0ixready)
5557 if retval != 0:
5558 tdata['error'] ='turbostat returned %d' % retval
5559 if turbo:
5560 tdata['turbo'] = turbo
5561 else:
5562 pf = open(sv.powerfile, 'w')
5563 pf.write(mode)
5564 # execution will pause here
5565 try:
5566 pf.flush()
5567 pf.close()
5568 except Exception as e:
5569 tdata['error'] = str(e)
5570 sv.fsetVal('CMD COMPLETE', 'trace_marker')
5571 sv.dlog('system returned')
5572 # reset everything
5573 sv.testVal('restoreall')
5574 if(sv.rtcwake):
5575 sv.dlog('disable RTC wake alarm')
5576 sv.rtcWakeAlarmOff()
5577 # postdelay delay
5578 if(count == sv.execcount and sv.postdelay > 0):
5579 sv.fsetVal('WAIT %d' % sv.postdelay, 'trace_marker')
5580 time.sleep(sv.postdelay/1000.0)
5581 sv.fsetVal('WAIT END', 'trace_marker')
5582 # return from suspend
5583 pprint('RESUME COMPLETE')
5584 if(count < sv.execcount):
5585 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5586 elif(not sv.wifitrace):
5587 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5588 sv.stop(pm)
5589 if sv.wifi and wifi:
5590 tdata['wifi'] = sv.pollWifi(wifi)
5591 sv.dlog('wifi check, %s' % tdata['wifi'])
5592 if(count == sv.execcount and sv.wifitrace):
5593 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5594 sv.stop(pm)
5595 if sv.netfix:
5596 tdata['netfix'] = sv.netfixon()
5597 sv.dlog('netfix, %s' % tdata['netfix'])
5598 if(sv.suspendmode == 'mem' or sv.suspendmode == 'command'):
5599 sv.dlog('read the ACPI FPDT')
5600 tdata['fw'] = getFPDT(False)
5601 testdata.append(tdata)
5602 sv.dlog('cmdinfo after')
5603 cmdafter = sv.cmdinfo(False)
5604 # grab a copy of the dmesg output
5605 if not quiet:
5606 pprint('CAPTURING DMESG')
5607 sv.getdmesg(testdata)
5608 # grab a copy of the ftrace output
5609 if sv.useftrace:
5610 if not quiet:
5611 pprint('CAPTURING TRACE')
5612 op = sv.writeDatafileHeader(sv.ftracefile, testdata)
5613 fp = open(tp+'trace', 'rb')
5614 op.write(ascii(fp.read()))
5615 op.close()
5616 sv.fsetVal('', 'trace')
5617 sv.platforminfo(cmdafter)
5618
5619def readFile(file):
5620 if os.path.islink(file):
5621 return os.readlink(file).split('/')[-1]
5622 else:
5623 return sysvals.getVal(file).strip()
5624
5625# Function: ms2nice
5626# Description:
5627# Print out a very concise time string in minutes and seconds
5628# Output:
5629# The time string, e.g. "1901m16s"
5630def ms2nice(val):
5631 val = int(val)
5632 h = val // 3600000
5633 m = (val // 60000) % 60
5634 s = (val // 1000) % 60
5635 if h > 0:
5636 return '%d:%02d:%02d' % (h, m, s)
5637 if m > 0:
5638 return '%02d:%02d' % (m, s)
5639 return '%ds' % s
5640
5641def yesno(val):
5642 list = {'enabled':'A', 'disabled':'S', 'auto':'E', 'on':'D',
5643 'active':'A', 'suspended':'S', 'suspending':'S'}
5644 if val not in list:
5645 return ' '
5646 return list[val]
5647
5648# Function: deviceInfo
5649# Description:
5650# Detect all the USB hosts and devices currently connected and add
5651# a list of USB device names to sysvals for better timeline readability
5652def deviceInfo(output=''):
5653 if not output:
5654 pprint('LEGEND\n'\
5655 '---------------------------------------------------------------------------------------------\n'\
5656 ' A = async/sync PM queue (A/S) C = runtime active children\n'\
5657 ' R = runtime suspend enabled/disabled (E/D) rACTIVE = runtime active (min/sec)\n'\
5658 ' S = runtime status active/suspended (A/S) rSUSPEND = runtime suspend (min/sec)\n'\
5659 ' U = runtime usage count\n'\
5660 '---------------------------------------------------------------------------------------------\n'\
5661 'DEVICE NAME A R S U C rACTIVE rSUSPEND\n'\
5662 '---------------------------------------------------------------------------------------------')
5663
5664 res = []
5665 tgtval = 'runtime_status'
5666 lines = dict()
5667 for dirname, dirnames, filenames in os.walk('/sys/devices'):
5668 if(not re.match(r'.*/power', dirname) or
5669 'control' not in filenames or
5670 tgtval not in filenames):
5671 continue
5672 name = ''
5673 dirname = dirname[:-6]
5674 device = dirname.split('/')[-1]
5675 power = dict()
5676 power[tgtval] = readFile('%s/power/%s' % (dirname, tgtval))
5677 # only list devices which support runtime suspend
5678 if power[tgtval] not in ['active', 'suspended', 'suspending']:
5679 continue
5680 for i in ['product', 'driver', 'subsystem']:
5681 file = '%s/%s' % (dirname, i)
5682 if os.path.exists(file):
5683 name = readFile(file)
5684 break
5685 for i in ['async', 'control', 'runtime_status', 'runtime_usage',
5686 'runtime_active_kids', 'runtime_active_time',
5687 'runtime_suspended_time']:
5688 if i in filenames:
5689 power[i] = readFile('%s/power/%s' % (dirname, i))
5690 if output:
5691 if power['control'] == output:
5692 res.append('%s/power/control' % dirname)
5693 continue
5694 lines[dirname] = '%-26s %-26s %1s %1s %1s %1s %1s %10s %10s' % \
5695 (device[:26], name[:26],
5696 yesno(power['async']), \
5697 yesno(power['control']), \
5698 yesno(power['runtime_status']), \
5699 power['runtime_usage'], \
5700 power['runtime_active_kids'], \
5701 ms2nice(power['runtime_active_time']), \
5702 ms2nice(power['runtime_suspended_time']))
5703 for i in sorted(lines):
5704 print(lines[i])
5705 return res
5706
5707# Function: getModes
5708# Description:
5709# Determine the supported power modes on this system
5710# Output:
5711# A string list of the available modes
5712def getModes():
5713 modes = []
5714 if(os.path.exists(sysvals.powerfile)):
5715 fp = open(sysvals.powerfile, 'r')
5716 modes = fp.read().split()
5717 fp.close()
5718 if(os.path.exists(sysvals.mempowerfile)):
5719 deep = False
5720 fp = open(sysvals.mempowerfile, 'r')
5721 for m in fp.read().split():
5722 memmode = m.strip('[]')
5723 if memmode == 'deep':
5724 deep = True
5725 else:
5726 modes.append('mem-%s' % memmode)
5727 fp.close()
5728 if 'mem' in modes and not deep:
5729 modes.remove('mem')
5730 if('disk' in modes and os.path.exists(sysvals.diskpowerfile)):
5731 fp = open(sysvals.diskpowerfile, 'r')
5732 for m in fp.read().split():
5733 modes.append('disk-%s' % m.strip('[]'))
5734 fp.close()
5735 return modes
5736
5737def dmidecode_backup(out, fatal=False):
5738 cpath, spath, info = '/proc/cpuinfo', '/sys/class/dmi/id', {
5739 'bios-vendor': 'bios_vendor',
5740 'bios-version': 'bios_version',
5741 'bios-release-date': 'bios_date',
5742 'system-manufacturer': 'sys_vendor',
5743 'system-product-name': 'product_name',
5744 'system-version': 'product_version',
5745 'system-serial-number': 'product_serial',
5746 'baseboard-manufacturer': 'board_vendor',
5747 'baseboard-product-name': 'board_name',
5748 'baseboard-version': 'board_version',
5749 'baseboard-serial-number': 'board_serial',
5750 'chassis-manufacturer': 'chassis_vendor',
5751 'chassis-version': 'chassis_version',
5752 'chassis-serial-number': 'chassis_serial',
5753 }
5754 for key in info:
5755 if key not in out:
5756 val = sysvals.getVal(os.path.join(spath, info[key])).strip()
5757 if val and val.lower() != 'to be filled by o.e.m.':
5758 out[key] = val
5759 if 'processor-version' not in out and os.path.exists(cpath):
5760 with open(cpath, 'r') as fp:
5761 for line in fp:
5762 m = re.match(r'^model\s*name\s*\:\s*(?P<c>.*)', line)
5763 if m:
5764 out['processor-version'] = m.group('c').strip()
5765 break
5766 if fatal and len(out) < 1:
5767 doError('dmidecode failed to get info from %s or %s' % \
5768 (sysvals.mempath, spath))
5769 return out
5770
5771# Function: dmidecode
5772# Description:
5773# Read the bios tables and pull out system info
5774# Arguments:
5775# mempath: /dev/mem or custom mem path
5776# fatal: True to exit on error, False to return empty dict
5777# Output:
5778# A dict object with all available key/values
5779def dmidecode(mempath, fatal=False):
5780 out = dict()
5781 if(not (os.path.exists(mempath) and os.access(mempath, os.R_OK))):
5782 return dmidecode_backup(out, fatal)
5783
5784 # the list of values to retrieve, with hardcoded (type, idx)
5785 info = {
5786 'bios-vendor': (0, 4),
5787 'bios-version': (0, 5),
5788 'bios-release-date': (0, 8),
5789 'system-manufacturer': (1, 4),
5790 'system-product-name': (1, 5),
5791 'system-version': (1, 6),
5792 'system-serial-number': (1, 7),
5793 'baseboard-manufacturer': (2, 4),
5794 'baseboard-product-name': (2, 5),
5795 'baseboard-version': (2, 6),
5796 'baseboard-serial-number': (2, 7),
5797 'chassis-manufacturer': (3, 4),
5798 'chassis-version': (3, 6),
5799 'chassis-serial-number': (3, 7),
5800 'processor-manufacturer': (4, 7),
5801 'processor-version': (4, 16),
5802 }
5803
5804 # by default use legacy scan, but try to use EFI first
5805 memaddr, memsize = 0xf0000, 0x10000
5806 for ep in ['/sys/firmware/efi/systab', '/proc/efi/systab']:
5807 if not os.path.exists(ep) or not os.access(ep, os.R_OK):
5808 continue
5809 fp = open(ep, 'r')
5810 buf = fp.read()
5811 fp.close()
5812 i = buf.find('SMBIOS=')
5813 if i >= 0:
5814 try:
5815 memaddr = int(buf[i+7:], 16)
5816 memsize = 0x20
5817 except:
5818 continue
5819
5820 # read in the memory for scanning
5821 try:
5822 fp = open(mempath, 'rb')
5823 fp.seek(memaddr)
5824 buf = fp.read(memsize)
5825 except:
5826 return dmidecode_backup(out, fatal)
5827 fp.close()
5828
5829 # search for either an SM table or DMI table
5830 i = base = length = num = 0
5831 while(i < memsize):
5832 if buf[i:i+4] == b'_SM_' and i < memsize - 16:
5833 length = struct.unpack('H', buf[i+22:i+24])[0]
5834 base, num = struct.unpack('IH', buf[i+24:i+30])
5835 break
5836 elif buf[i:i+5] == b'_DMI_':
5837 length = struct.unpack('H', buf[i+6:i+8])[0]
5838 base, num = struct.unpack('IH', buf[i+8:i+14])
5839 break
5840 i += 16
5841 if base == 0 and length == 0 and num == 0:
5842 return dmidecode_backup(out, fatal)
5843
5844 # read in the SM or DMI table
5845 try:
5846 fp = open(mempath, 'rb')
5847 fp.seek(base)
5848 buf = fp.read(length)
5849 except:
5850 return dmidecode_backup(out, fatal)
5851 fp.close()
5852
5853 # scan the table for the values we want
5854 count = i = 0
5855 while(count < num and i <= len(buf) - 4):
5856 type, size, handle = struct.unpack('BBH', buf[i:i+4])
5857 n = i + size
5858 while n < len(buf) - 1:
5859 if 0 == struct.unpack('H', buf[n:n+2])[0]:
5860 break
5861 n += 1
5862 data = buf[i+size:n+2].split(b'\0')
5863 for name in info:
5864 itype, idxadr = info[name]
5865 if itype == type:
5866 idx = struct.unpack('B', buf[i+idxadr:i+idxadr+1])[0]
5867 if idx > 0 and idx < len(data) - 1:
5868 s = data[idx-1].decode('utf-8')
5869 if s.strip() and s.strip().lower() != 'to be filled by o.e.m.':
5870 out[name] = s
5871 i = n + 2
5872 count += 1
5873 return out
5874
5875# Function: getFPDT
5876# Description:
5877# Read the acpi bios tables and pull out FPDT, the firmware data
5878# Arguments:
5879# output: True to output the info to stdout, False otherwise
5880def getFPDT(output):
5881 rectype = {}
5882 rectype[0] = 'Firmware Basic Boot Performance Record'
5883 rectype[1] = 'S3 Performance Table Record'
5884 prectype = {}
5885 prectype[0] = 'Basic S3 Resume Performance Record'
5886 prectype[1] = 'Basic S3 Suspend Performance Record'
5887
5888 sysvals.rootCheck(True)
5889 if(not os.path.exists(sysvals.fpdtpath)):
5890 if(output):
5891 doError('file does not exist: %s' % sysvals.fpdtpath)
5892 return False
5893 if(not os.access(sysvals.fpdtpath, os.R_OK)):
5894 if(output):
5895 doError('file is not readable: %s' % sysvals.fpdtpath)
5896 return False
5897 if(not os.path.exists(sysvals.mempath)):
5898 if(output):
5899 doError('file does not exist: %s' % sysvals.mempath)
5900 return False
5901 if(not os.access(sysvals.mempath, os.R_OK)):
5902 if(output):
5903 doError('file is not readable: %s' % sysvals.mempath)
5904 return False
5905
5906 fp = open(sysvals.fpdtpath, 'rb')
5907 buf = fp.read()
5908 fp.close()
5909
5910 if(len(buf) < 36):
5911 if(output):
5912 doError('Invalid FPDT table data, should '+\
5913 'be at least 36 bytes')
5914 return False
5915
5916 table = struct.unpack('4sIBB6s8sI4sI', buf[0:36])
5917 if(output):
5918 pprint('\n'\
5919 'Firmware Performance Data Table (%s)\n'\
5920 ' Signature : %s\n'\
5921 ' Table Length : %u\n'\
5922 ' Revision : %u\n'\
5923 ' Checksum : 0x%x\n'\
5924 ' OEM ID : %s\n'\
5925 ' OEM Table ID : %s\n'\
5926 ' OEM Revision : %u\n'\
5927 ' Creator ID : %s\n'\
5928 ' Creator Revision : 0x%x\n'\
5929 '' % (ascii(table[0]), ascii(table[0]), table[1], table[2],
5930 table[3], ascii(table[4]), ascii(table[5]), table[6],
5931 ascii(table[7]), table[8]))
5932
5933 if(table[0] != b'FPDT'):
5934 if(output):
5935 doError('Invalid FPDT table')
5936 return False
5937 if(len(buf) <= 36):
5938 return False
5939 i = 0
5940 fwData = [0, 0]
5941 records = buf[36:]
5942 try:
5943 fp = open(sysvals.mempath, 'rb')
5944 except:
5945 pprint('WARNING: /dev/mem is not readable, ignoring the FPDT data')
5946 return False
5947 while(i < len(records)):
5948 header = struct.unpack('HBB', records[i:i+4])
5949 if(header[0] not in rectype):
5950 i += header[1]
5951 continue
5952 if(header[1] != 16):
5953 i += header[1]
5954 continue
5955 addr = struct.unpack('Q', records[i+8:i+16])[0]
5956 try:
5957 fp.seek(addr)
5958 first = fp.read(8)
5959 except:
5960 if(output):
5961 pprint('Bad address 0x%x in %s' % (addr, sysvals.mempath))
5962 return [0, 0]
5963 rechead = struct.unpack('4sI', first)
5964 recdata = fp.read(rechead[1]-8)
5965 if(rechead[0] == b'FBPT'):
5966 record = struct.unpack('HBBIQQQQQ', recdata[:48])
5967 if(output):
5968 pprint('%s (%s)\n'\
5969 ' Reset END : %u ns\n'\
5970 ' OS Loader LoadImage Start : %u ns\n'\
5971 ' OS Loader StartImage Start : %u ns\n'\
5972 ' ExitBootServices Entry : %u ns\n'\
5973 ' ExitBootServices Exit : %u ns'\
5974 '' % (rectype[header[0]], ascii(rechead[0]), record[4], record[5],
5975 record[6], record[7], record[8]))
5976 elif(rechead[0] == b'S3PT'):
5977 if(output):
5978 pprint('%s (%s)' % (rectype[header[0]], ascii(rechead[0])))
5979 j = 0
5980 while(j < len(recdata)):
5981 prechead = struct.unpack('HBB', recdata[j:j+4])
5982 if(prechead[0] not in prectype):
5983 continue
5984 if(prechead[0] == 0):
5985 record = struct.unpack('IIQQ', recdata[j:j+prechead[1]])
5986 fwData[1] = record[2]
5987 if(output):
5988 pprint(' %s\n'\
5989 ' Resume Count : %u\n'\
5990 ' FullResume : %u ns\n'\
5991 ' AverageResume : %u ns'\
5992 '' % (prectype[prechead[0]], record[1],
5993 record[2], record[3]))
5994 elif(prechead[0] == 1):
5995 record = struct.unpack('QQ', recdata[j+4:j+prechead[1]])
5996 fwData[0] = record[1] - record[0]
5997 if(output):
5998 pprint(' %s\n'\
5999 ' SuspendStart : %u ns\n'\
6000 ' SuspendEnd : %u ns\n'\
6001 ' SuspendTime : %u ns'\
6002 '' % (prectype[prechead[0]], record[0],
6003 record[1], fwData[0]))
6004
6005 j += prechead[1]
6006 if(output):
6007 pprint('')
6008 i += header[1]
6009 fp.close()
6010 return fwData
6011
6012# Function: statusCheck
6013# Description:
6014# Verify that the requested command and options will work, and
6015# print the results to the terminal
6016# Output:
6017# True if the test will work, False if not
6018def statusCheck(probecheck=False):
6019 status = ''
6020
6021 pprint('Checking this system (%s)...' % platform.node())
6022
6023 # check we have root access
6024 res = sysvals.colorText('NO (No features of this tool will work!)')
6025 if(sysvals.rootCheck(False)):
6026 res = 'YES'
6027 pprint(' have root access: %s' % res)
6028 if(res != 'YES'):
6029 pprint(' Try running this script with sudo')
6030 return 'missing root access'
6031
6032 # check sysfs is mounted
6033 res = sysvals.colorText('NO (No features of this tool will work!)')
6034 if(os.path.exists(sysvals.powerfile)):
6035 res = 'YES'
6036 pprint(' is sysfs mounted: %s' % res)
6037 if(res != 'YES'):
6038 return 'sysfs is missing'
6039
6040 # check target mode is a valid mode
6041 if sysvals.suspendmode != 'command':
6042 res = sysvals.colorText('NO')
6043 modes = getModes()
6044 if(sysvals.suspendmode in modes):
6045 res = 'YES'
6046 else:
6047 status = '%s mode is not supported' % sysvals.suspendmode
6048 pprint(' is "%s" a valid power mode: %s' % (sysvals.suspendmode, res))
6049 if(res == 'NO'):
6050 pprint(' valid power modes are: %s' % modes)
6051 pprint(' please choose one with -m')
6052
6053 # check if ftrace is available
6054 if sysvals.useftrace:
6055 res = sysvals.colorText('NO')
6056 sysvals.useftrace = sysvals.verifyFtrace()
6057 efmt = '"{0}" uses ftrace, and it is not properly supported'
6058 if sysvals.useftrace:
6059 res = 'YES'
6060 elif sysvals.usecallgraph:
6061 status = efmt.format('-f')
6062 elif sysvals.usedevsrc:
6063 status = efmt.format('-dev')
6064 elif sysvals.useprocmon:
6065 status = efmt.format('-proc')
6066 pprint(' is ftrace supported: %s' % res)
6067
6068 # check if kprobes are available
6069 if sysvals.usekprobes:
6070 res = sysvals.colorText('NO')
6071 sysvals.usekprobes = sysvals.verifyKprobes()
6072 if(sysvals.usekprobes):
6073 res = 'YES'
6074 else:
6075 sysvals.usedevsrc = False
6076 pprint(' are kprobes supported: %s' % res)
6077
6078 # what data source are we using
6079 res = 'DMESG (very limited, ftrace is preferred)'
6080 if sysvals.useftrace:
6081 sysvals.usetraceevents = True
6082 for e in sysvals.traceevents:
6083 if not os.path.exists(sysvals.epath+e):
6084 sysvals.usetraceevents = False
6085 if(sysvals.usetraceevents):
6086 res = 'FTRACE (all trace events found)'
6087 pprint(' timeline data source: %s' % res)
6088
6089 # check if rtcwake
6090 res = sysvals.colorText('NO')
6091 if(sysvals.rtcpath != ''):
6092 res = 'YES'
6093 elif(sysvals.rtcwake):
6094 status = 'rtcwake is not properly supported'
6095 pprint(' is rtcwake supported: %s' % res)
6096
6097 # check info commands
6098 pprint(' optional commands this tool may use for info:')
6099 no = sysvals.colorText('MISSING')
6100 yes = sysvals.colorText('FOUND', 32)
6101 for c in ['turbostat', 'mcelog', 'lspci', 'lsusb', 'netfix']:
6102 if c == 'turbostat':
6103 res = yes if sysvals.haveTurbostat() else no
6104 else:
6105 res = yes if sysvals.getExec(c) else no
6106 pprint(' %s: %s' % (c, res))
6107
6108 if not probecheck:
6109 return status
6110
6111 # verify kprobes
6112 if sysvals.usekprobes:
6113 for name in sysvals.tracefuncs:
6114 sysvals.defaultKprobe(name, sysvals.tracefuncs[name])
6115 if sysvals.usedevsrc:
6116 for name in sysvals.dev_tracefuncs:
6117 sysvals.defaultKprobe(name, sysvals.dev_tracefuncs[name])
6118 sysvals.addKprobes(True)
6119
6120 return status
6121
6122# Function: doError
6123# Description:
6124# generic error function for catastrphic failures
6125# Arguments:
6126# msg: the error message to print
6127# help: True if printHelp should be called after, False otherwise
6128def doError(msg, help=False):
6129 if(help == True):
6130 printHelp()
6131 pprint('ERROR: %s\n' % msg)
6132 sysvals.outputResult({'error':msg})
6133 sys.exit(1)
6134
6135# Function: getArgInt
6136# Description:
6137# pull out an integer argument from the command line with checks
6138def getArgInt(name, args, min, max, main=True):
6139 if main:
6140 try:
6141 arg = next(args)
6142 except:
6143 doError(name+': no argument supplied', True)
6144 else:
6145 arg = args
6146 try:
6147 val = int(arg)
6148 except:
6149 doError(name+': non-integer value given', True)
6150 if(val < min or val > max):
6151 doError(name+': value should be between %d and %d' % (min, max), True)
6152 return val
6153
6154# Function: getArgFloat
6155# Description:
6156# pull out a float argument from the command line with checks
6157def getArgFloat(name, args, min, max, main=True):
6158 if main:
6159 try:
6160 arg = next(args)
6161 except:
6162 doError(name+': no argument supplied', True)
6163 else:
6164 arg = args
6165 try:
6166 val = float(arg)
6167 except:
6168 doError(name+': non-numerical value given', True)
6169 if(val < min or val > max):
6170 doError(name+': value should be between %f and %f' % (min, max), True)
6171 return val
6172
6173def processData(live=False, quiet=False):
6174 if not quiet:
6175 pprint('PROCESSING: %s' % sysvals.htmlfile)
6176 sysvals.vprint('usetraceevents=%s, usetracemarkers=%s, usekprobes=%s' % \
6177 (sysvals.usetraceevents, sysvals.usetracemarkers, sysvals.usekprobes))
6178 error = ''
6179 if(sysvals.usetraceevents):
6180 testruns, error = parseTraceLog(live)
6181 if sysvals.dmesgfile:
6182 for data in testruns:
6183 data.extractErrorInfo()
6184 else:
6185 testruns = loadKernelLog()
6186 for data in testruns:
6187 parseKernelLog(data)
6188 if(sysvals.ftracefile and (sysvals.usecallgraph or sysvals.usetraceevents)):
6189 appendIncompleteTraceLog(testruns)
6190 if not sysvals.stamp:
6191 pprint('ERROR: data does not include the expected stamp')
6192 return (testruns, {'error': 'timeline generation failed'})
6193 shown = ['os', 'bios', 'biosdate', 'cpu', 'host', 'kernel', 'man', 'memfr',
6194 'memsz', 'mode', 'numcpu', 'plat', 'time', 'wifi']
6195 sysvals.vprint('System Info:')
6196 for key in sorted(sysvals.stamp):
6197 if key in shown:
6198 sysvals.vprint(' %-8s : %s' % (key.upper(), sysvals.stamp[key]))
6199 sysvals.vprint('Command:\n %s' % sysvals.cmdline)
6200 for data in testruns:
6201 if data.turbostat:
6202 idx, s = 0, 'Turbostat:\n '
6203 for val in data.turbostat.split('|'):
6204 idx += len(val) + 1
6205 if idx >= 80:
6206 idx = 0
6207 s += '\n '
6208 s += val + ' '
6209 sysvals.vprint(s)
6210 data.printDetails()
6211 if len(sysvals.platinfo) > 0:
6212 sysvals.vprint('\nPlatform Info:')
6213 for info in sysvals.platinfo:
6214 sysvals.vprint('[%s - %s]' % (info[0], info[1]))
6215 sysvals.vprint(info[2])
6216 sysvals.vprint('')
6217 if sysvals.cgdump:
6218 for data in testruns:
6219 data.debugPrint()
6220 sys.exit(0)
6221 if len(testruns) < 1:
6222 pprint('ERROR: Not enough test data to build a timeline')
6223 return (testruns, {'error': 'timeline generation failed'})
6224 sysvals.vprint('Creating the html timeline (%s)...' % sysvals.htmlfile)
6225 createHTML(testruns, error)
6226 if not quiet:
6227 pprint('DONE: %s' % sysvals.htmlfile)
6228 data = testruns[0]
6229 stamp = data.stamp
6230 stamp['suspend'], stamp['resume'] = data.getTimeValues()
6231 if data.fwValid:
6232 stamp['fwsuspend'], stamp['fwresume'] = data.fwSuspend, data.fwResume
6233 if error:
6234 stamp['error'] = error
6235 return (testruns, stamp)
6236
6237# Function: rerunTest
6238# Description:
6239# generate an output from an existing set of ftrace/dmesg logs
6240def rerunTest(htmlfile=''):
6241 if sysvals.ftracefile:
6242 doesTraceLogHaveTraceEvents()
6243 if not sysvals.dmesgfile and not sysvals.usetraceevents:
6244 doError('recreating this html output requires a dmesg file')
6245 if htmlfile:
6246 sysvals.htmlfile = htmlfile
6247 else:
6248 sysvals.setOutputFile()
6249 if os.path.exists(sysvals.htmlfile):
6250 if not os.path.isfile(sysvals.htmlfile):
6251 doError('a directory already exists with this name: %s' % sysvals.htmlfile)
6252 elif not os.access(sysvals.htmlfile, os.W_OK):
6253 doError('missing permission to write to %s' % sysvals.htmlfile)
6254 testruns, stamp = processData()
6255 sysvals.resetlog()
6256 return stamp
6257
6258# Function: runTest
6259# Description:
6260# execute a suspend/resume, gather the logs, and generate the output
6261def runTest(n=0, quiet=False):
6262 # prepare for the test
6263 sysvals.initTestOutput('suspend')
6264 op = sysvals.writeDatafileHeader(sysvals.dmesgfile, [])
6265 op.write('# EXECUTION TRACE START\n')
6266 op.close()
6267 if n <= 1:
6268 if sysvals.rs != 0:
6269 sysvals.dlog('%sabling runtime suspend' % ('en' if sysvals.rs > 0 else 'dis'))
6270 sysvals.setRuntimeSuspend(True)
6271 if sysvals.display:
6272 ret = sysvals.displayControl('init')
6273 sysvals.dlog('xset display init, ret = %d' % ret)
6274 sysvals.testVal(sysvals.pmdpath, 'basic', '1')
6275 sysvals.testVal(sysvals.s0ixpath, 'basic', 'Y')
6276 sysvals.dlog('initialize ftrace')
6277 sysvals.initFtrace(quiet)
6278
6279 # execute the test
6280 executeSuspend(quiet)
6281 sysvals.cleanupFtrace()
6282 if sysvals.skiphtml:
6283 sysvals.outputResult({}, n)
6284 sysvals.sudoUserchown(sysvals.testdir)
6285 return
6286 testruns, stamp = processData(True, quiet)
6287 for data in testruns:
6288 del data
6289 sysvals.sudoUserchown(sysvals.testdir)
6290 sysvals.outputResult(stamp, n)
6291 if 'error' in stamp:
6292 return 2
6293 return 0
6294
6295def find_in_html(html, start, end, firstonly=True):
6296 cnt, out, list = len(html), [], []
6297 if firstonly:
6298 m = re.search(start, html)
6299 if m:
6300 list.append(m)
6301 else:
6302 list = re.finditer(start, html)
6303 for match in list:
6304 s = match.end()
6305 e = cnt if (len(out) < 1 or s + 10000 > cnt) else s + 10000
6306 m = re.search(end, html[s:e])
6307 if not m:
6308 break
6309 e = s + m.start()
6310 str = html[s:e]
6311 if end == 'ms':
6312 num = re.search(r'[-+]?\d*\.\d+|\d+', str)
6313 str = num.group() if num else 'NaN'
6314 if firstonly:
6315 return str
6316 out.append(str)
6317 if firstonly:
6318 return ''
6319 return out
6320
6321def data_from_html(file, outpath, issues, fulldetail=False):
6322 try:
6323 html = open(file, 'r').read()
6324 except:
6325 html = ascii(open(file, 'rb').read())
6326 sysvals.htmlfile = os.path.relpath(file, outpath)
6327 # extract general info
6328 suspend = find_in_html(html, 'Kernel Suspend', 'ms')
6329 resume = find_in_html(html, 'Kernel Resume', 'ms')
6330 sysinfo = find_in_html(html, '<div class="stamp sysinfo">', '</div>')
6331 line = find_in_html(html, '<div class="stamp">', '</div>')
6332 stmp = line.split()
6333 if not suspend or not resume or len(stmp) != 8:
6334 return False
6335 try:
6336 dt = datetime.strptime(' '.join(stmp[3:]), '%B %d %Y, %I:%M:%S %p')
6337 except:
6338 return False
6339 sysvals.hostname = stmp[0]
6340 tstr = dt.strftime('%Y/%m/%d %H:%M:%S')
6341 error = find_in_html(html, '<table class="testfail"><tr><td>', '</td>')
6342 if error:
6343 m = re.match(r'[a-z0-9]* failed in (?P<p>\S*).*', error)
6344 if m:
6345 result = 'fail in %s' % m.group('p')
6346 else:
6347 result = 'fail'
6348 else:
6349 result = 'pass'
6350 # extract error info
6351 tp, ilist = False, []
6352 extra = dict()
6353 log = find_in_html(html, '<div id="dmesglog" style="display:none;">',
6354 '</div>').strip()
6355 if log:
6356 d = Data(0)
6357 d.end = 999999999
6358 d.dmesgtext = log.split('\n')
6359 tp = d.extractErrorInfo()
6360 if len(issues) < 100:
6361 for msg in tp.msglist:
6362 sysvals.errorSummary(issues, msg)
6363 if stmp[2] == 'freeze':
6364 extra = d.turbostatInfo()
6365 elist = dict()
6366 for dir in d.errorinfo:
6367 for err in d.errorinfo[dir]:
6368 if err[0] not in elist:
6369 elist[err[0]] = 0
6370 elist[err[0]] += 1
6371 for i in elist:
6372 ilist.append('%sx%d' % (i, elist[i]) if elist[i] > 1 else i)
6373 line = find_in_html(log, '# wifi ', '\n')
6374 if line:
6375 extra['wifi'] = line
6376 line = find_in_html(log, '# netfix ', '\n')
6377 if line:
6378 extra['netfix'] = line
6379 line = find_in_html(log, '# command ', '\n')
6380 if line:
6381 m = re.match(r'.* -m (?P<m>\S*).*', line)
6382 if m:
6383 extra['fullmode'] = m.group('m')
6384 low = find_in_html(html, 'freeze time: <b>', ' ms</b>')
6385 for lowstr in ['waking', '+']:
6386 if not low:
6387 break
6388 if lowstr not in low:
6389 continue
6390 if lowstr == '+':
6391 issue = 'S2LOOPx%d' % len(low.split('+'))
6392 else:
6393 m = re.match(r'.*waking *(?P<n>[0-9]*) *times.*', low)
6394 issue = 'S2WAKEx%s' % m.group('n') if m else 'S2WAKExNaN'
6395 match = [i for i in issues if i['match'] == issue]
6396 if len(match) > 0:
6397 match[0]['count'] += 1
6398 if sysvals.hostname not in match[0]['urls']:
6399 match[0]['urls'][sysvals.hostname] = [sysvals.htmlfile]
6400 elif sysvals.htmlfile not in match[0]['urls'][sysvals.hostname]:
6401 match[0]['urls'][sysvals.hostname].append(sysvals.htmlfile)
6402 else:
6403 issues.append({
6404 'match': issue, 'count': 1, 'line': issue,
6405 'urls': {sysvals.hostname: [sysvals.htmlfile]},
6406 })
6407 ilist.append(issue)
6408 # extract device info
6409 devices = dict()
6410 for line in html.split('\n'):
6411 m = re.match(r' *<div id=\"[a,0-9]*\" *title=\"(?P<title>.*)\" class=\"thread.*', line)
6412 if not m or 'thread kth' in line or 'thread sec' in line:
6413 continue
6414 m = re.match(r'(?P<n>.*) \((?P<t>[0-9,\.]*) ms\) (?P<p>.*)', m.group('title'))
6415 if not m:
6416 continue
6417 name, time, phase = m.group('n'), m.group('t'), m.group('p')
6418 if name == 'async_synchronize_full':
6419 continue
6420 if ' async' in name or ' sync' in name:
6421 name = ' '.join(name.split(' ')[:-1])
6422 if phase.startswith('suspend'):
6423 d = 'suspend'
6424 elif phase.startswith('resume'):
6425 d = 'resume'
6426 else:
6427 continue
6428 if d not in devices:
6429 devices[d] = dict()
6430 if name not in devices[d]:
6431 devices[d][name] = 0.0
6432 devices[d][name] += float(time)
6433 # create worst device info
6434 worst = dict()
6435 for d in ['suspend', 'resume']:
6436 worst[d] = {'name':'', 'time': 0.0}
6437 dev = devices[d] if d in devices else 0
6438 if dev and len(dev.keys()) > 0:
6439 n = sorted(dev, key=lambda k:(dev[k], k), reverse=True)[0]
6440 worst[d]['name'], worst[d]['time'] = n, dev[n]
6441 data = {
6442 'mode': stmp[2],
6443 'host': stmp[0],
6444 'kernel': stmp[1],
6445 'sysinfo': sysinfo,
6446 'time': tstr,
6447 'result': result,
6448 'issues': ' '.join(ilist),
6449 'suspend': suspend,
6450 'resume': resume,
6451 'devlist': devices,
6452 'sus_worst': worst['suspend']['name'],
6453 'sus_worsttime': worst['suspend']['time'],
6454 'res_worst': worst['resume']['name'],
6455 'res_worsttime': worst['resume']['time'],
6456 'url': sysvals.htmlfile,
6457 }
6458 for key in extra:
6459 data[key] = extra[key]
6460 if fulldetail:
6461 data['funclist'] = find_in_html(html, '<div title="', '" class="traceevent"', False)
6462 if tp:
6463 for arg in ['-multi ', '-info ']:
6464 if arg in tp.cmdline:
6465 data['target'] = tp.cmdline[tp.cmdline.find(arg):].split()[1]
6466 break
6467 return data
6468
6469def genHtml(subdir, force=False):
6470 for dirname, dirnames, filenames in os.walk(subdir):
6471 sysvals.dmesgfile = sysvals.ftracefile = sysvals.htmlfile = ''
6472 for filename in filenames:
6473 file = os.path.join(dirname, filename)
6474 if sysvals.usable(file):
6475 if(re.match(r'.*_dmesg.txt', filename)):
6476 sysvals.dmesgfile = file
6477 elif(re.match(r'.*_ftrace.txt', filename)):
6478 sysvals.ftracefile = file
6479 sysvals.setOutputFile()
6480 if (sysvals.dmesgfile or sysvals.ftracefile) and sysvals.htmlfile and \
6481 (force or not sysvals.usable(sysvals.htmlfile, True)):
6482 pprint('FTRACE: %s' % sysvals.ftracefile)
6483 if sysvals.dmesgfile:
6484 pprint('DMESG : %s' % sysvals.dmesgfile)
6485 rerunTest()
6486
6487# Function: runSummary
6488# Description:
6489# create a summary of tests in a sub-directory
6490def runSummary(subdir, local=True, genhtml=False):
6491 inpath = os.path.abspath(subdir)
6492 outpath = os.path.abspath('.') if local else inpath
6493 pprint('Generating a summary of folder:\n %s' % inpath)
6494 if genhtml:
6495 genHtml(subdir)
6496 target, issues, testruns = '', [], []
6497 desc = {'host':[],'mode':[],'kernel':[]}
6498 for dirname, dirnames, filenames in os.walk(subdir):
6499 for filename in filenames:
6500 if(not re.match(r'.*.html', filename)):
6501 continue
6502 data = data_from_html(os.path.join(dirname, filename), outpath, issues)
6503 if(not data):
6504 continue
6505 if 'target' in data:
6506 target = data['target']
6507 testruns.append(data)
6508 for key in desc:
6509 if data[key] not in desc[key]:
6510 desc[key].append(data[key])
6511 pprint('Summary files:')
6512 if len(desc['host']) == len(desc['mode']) == len(desc['kernel']) == 1:
6513 title = '%s %s %s' % (desc['host'][0], desc['kernel'][0], desc['mode'][0])
6514 if target:
6515 title += ' %s' % target
6516 else:
6517 title = inpath
6518 createHTMLSummarySimple(testruns, os.path.join(outpath, 'summary.html'), title)
6519 pprint(' summary.html - tabular list of test data found')
6520 createHTMLDeviceSummary(testruns, os.path.join(outpath, 'summary-devices.html'), title)
6521 pprint(' summary-devices.html - kernel device list sorted by total execution time')
6522 createHTMLIssuesSummary(testruns, issues, os.path.join(outpath, 'summary-issues.html'), title)
6523 pprint(' summary-issues.html - kernel issues found sorted by frequency')
6524
6525# Function: checkArgBool
6526# Description:
6527# check if a boolean string value is true or false
6528def checkArgBool(name, value):
6529 if value in switchvalues:
6530 if value in switchoff:
6531 return False
6532 return True
6533 doError('invalid boolean --> (%s: %s), use "true/false" or "1/0"' % (name, value), True)
6534 return False
6535
6536# Function: configFromFile
6537# Description:
6538# Configure the script via the info in a config file
6539def configFromFile(file):
6540 Config = configparser.ConfigParser()
6541
6542 Config.read(file)
6543 sections = Config.sections()
6544 overridekprobes = False
6545 overridedevkprobes = False
6546 if 'Settings' in sections:
6547 for opt in Config.options('Settings'):
6548 value = Config.get('Settings', opt).lower()
6549 option = opt.lower()
6550 if(option == 'verbose'):
6551 sysvals.verbose = checkArgBool(option, value)
6552 elif(option == 'addlogs'):
6553 sysvals.dmesglog = sysvals.ftracelog = checkArgBool(option, value)
6554 elif(option == 'dev'):
6555 sysvals.usedevsrc = checkArgBool(option, value)
6556 elif(option == 'proc'):
6557 sysvals.useprocmon = checkArgBool(option, value)
6558 elif(option == 'x2'):
6559 if checkArgBool(option, value):
6560 sysvals.execcount = 2
6561 elif(option == 'callgraph'):
6562 sysvals.usecallgraph = checkArgBool(option, value)
6563 elif(option == 'override-timeline-functions'):
6564 overridekprobes = checkArgBool(option, value)
6565 elif(option == 'override-dev-timeline-functions'):
6566 overridedevkprobes = checkArgBool(option, value)
6567 elif(option == 'skiphtml'):
6568 sysvals.skiphtml = checkArgBool(option, value)
6569 elif(option == 'sync'):
6570 sysvals.sync = checkArgBool(option, value)
6571 elif(option == 'rs' or option == 'runtimesuspend'):
6572 if value in switchvalues:
6573 if value in switchoff:
6574 sysvals.rs = -1
6575 else:
6576 sysvals.rs = 1
6577 else:
6578 doError('invalid value --> (%s: %s), use "enable/disable"' % (option, value), True)
6579 elif(option == 'display'):
6580 disopt = ['on', 'off', 'standby', 'suspend']
6581 if value not in disopt:
6582 doError('invalid value --> (%s: %s), use %s' % (option, value, disopt), True)
6583 sysvals.display = value
6584 elif(option == 'gzip'):
6585 sysvals.gzip = checkArgBool(option, value)
6586 elif(option == 'cgfilter'):
6587 sysvals.setCallgraphFilter(value)
6588 elif(option == 'cgskip'):
6589 if value in switchoff:
6590 sysvals.cgskip = ''
6591 else:
6592 sysvals.cgskip = sysvals.configFile(val)
6593 if(not sysvals.cgskip):
6594 doError('%s does not exist' % sysvals.cgskip)
6595 elif(option == 'cgtest'):
6596 sysvals.cgtest = getArgInt('cgtest', value, 0, 1, False)
6597 elif(option == 'cgphase'):
6598 d = Data(0)
6599 if value not in d.phasedef:
6600 doError('invalid phase --> (%s: %s), valid phases are %s'\
6601 % (option, value, d.phasedef.keys()), True)
6602 sysvals.cgphase = value
6603 elif(option == 'fadd'):
6604 file = sysvals.configFile(value)
6605 if(not file):
6606 doError('%s does not exist' % value)
6607 sysvals.addFtraceFilterFunctions(file)
6608 elif(option == 'result'):
6609 sysvals.result = value
6610 elif(option == 'multi'):
6611 nums = value.split()
6612 if len(nums) != 2:
6613 doError('multi requires 2 integers (exec_count and delay)', True)
6614 sysvals.multiinit(nums[0], nums[1])
6615 elif(option == 'devicefilter'):
6616 sysvals.setDeviceFilter(value)
6617 elif(option == 'expandcg'):
6618 sysvals.cgexp = checkArgBool(option, value)
6619 elif(option == 'srgap'):
6620 if checkArgBool(option, value):
6621 sysvals.srgap = 5
6622 elif(option == 'mode'):
6623 sysvals.suspendmode = value
6624 elif(option == 'command' or option == 'cmd'):
6625 sysvals.testcommand = value
6626 elif(option == 'x2delay'):
6627 sysvals.x2delay = getArgInt('x2delay', value, 0, 60000, False)
6628 elif(option == 'predelay'):
6629 sysvals.predelay = getArgInt('predelay', value, 0, 60000, False)
6630 elif(option == 'postdelay'):
6631 sysvals.postdelay = getArgInt('postdelay', value, 0, 60000, False)
6632 elif(option == 'maxdepth'):
6633 sysvals.max_graph_depth = getArgInt('maxdepth', value, 0, 1000, False)
6634 elif(option == 'rtcwake'):
6635 if value in switchoff:
6636 sysvals.rtcwake = False
6637 else:
6638 sysvals.rtcwake = True
6639 sysvals.rtcwaketime = getArgInt('rtcwake', value, 0, 3600, False)
6640 elif(option == 'timeprec'):
6641 sysvals.setPrecision(getArgInt('timeprec', value, 0, 6, False))
6642 elif(option == 'mindev'):
6643 sysvals.mindevlen = getArgFloat('mindev', value, 0.0, 10000.0, False)
6644 elif(option == 'callloop-maxgap'):
6645 sysvals.callloopmaxgap = getArgFloat('callloop-maxgap', value, 0.0, 1.0, False)
6646 elif(option == 'callloop-maxlen'):
6647 sysvals.callloopmaxgap = getArgFloat('callloop-maxlen', value, 0.0, 1.0, False)
6648 elif(option == 'mincg'):
6649 sysvals.mincglen = getArgFloat('mincg', value, 0.0, 10000.0, False)
6650 elif(option == 'bufsize'):
6651 sysvals.bufsize = getArgInt('bufsize', value, 1, 1024*1024*8, False)
6652 elif(option == 'output-dir'):
6653 sysvals.outdir = sysvals.setOutputFolder(value)
6654
6655 if sysvals.suspendmode == 'command' and not sysvals.testcommand:
6656 doError('No command supplied for mode "command"')
6657
6658 # compatibility errors
6659 if sysvals.usedevsrc and sysvals.usecallgraph:
6660 doError('-dev is not compatible with -f')
6661 if sysvals.usecallgraph and sysvals.useprocmon:
6662 doError('-proc is not compatible with -f')
6663
6664 if overridekprobes:
6665 sysvals.tracefuncs = dict()
6666 if overridedevkprobes:
6667 sysvals.dev_tracefuncs = dict()
6668
6669 kprobes = dict()
6670 kprobesec = 'dev_timeline_functions_'+platform.machine()
6671 if kprobesec in sections:
6672 for name in Config.options(kprobesec):
6673 text = Config.get(kprobesec, name)
6674 kprobes[name] = (text, True)
6675 kprobesec = 'timeline_functions_'+platform.machine()
6676 if kprobesec in sections:
6677 for name in Config.options(kprobesec):
6678 if name in kprobes:
6679 doError('Duplicate timeline function found "%s"' % (name))
6680 text = Config.get(kprobesec, name)
6681 kprobes[name] = (text, False)
6682
6683 for name in kprobes:
6684 function = name
6685 format = name
6686 color = ''
6687 args = dict()
6688 text, dev = kprobes[name]
6689 data = text.split()
6690 i = 0
6691 for val in data:
6692 # bracketted strings are special formatting, read them separately
6693 if val[0] == '[' and val[-1] == ']':
6694 for prop in val[1:-1].split(','):
6695 p = prop.split('=')
6696 if p[0] == 'color':
6697 try:
6698 color = int(p[1], 16)
6699 color = '#'+p[1]
6700 except:
6701 color = p[1]
6702 continue
6703 # first real arg should be the format string
6704 if i == 0:
6705 format = val
6706 # all other args are actual function args
6707 else:
6708 d = val.split('=')
6709 args[d[0]] = d[1]
6710 i += 1
6711 if not function or not format:
6712 doError('Invalid kprobe: %s' % name)
6713 for arg in re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', format):
6714 if arg not in args:
6715 doError('Kprobe "%s" is missing argument "%s"' % (name, arg))
6716 if (dev and name in sysvals.dev_tracefuncs) or (not dev and name in sysvals.tracefuncs):
6717 doError('Duplicate timeline function found "%s"' % (name))
6718
6719 kp = {
6720 'name': name,
6721 'func': function,
6722 'format': format,
6723 sysvals.archargs: args
6724 }
6725 if color:
6726 kp['color'] = color
6727 if dev:
6728 sysvals.dev_tracefuncs[name] = kp
6729 else:
6730 sysvals.tracefuncs[name] = kp
6731
6732# Function: printHelp
6733# Description:
6734# print out the help text
6735def printHelp():
6736 pprint('\n%s v%s\n'\
6737 'Usage: sudo sleepgraph <options> <commands>\n'\
6738 '\n'\
6739 'Description:\n'\
6740 ' This tool is designed to assist kernel and OS developers in optimizing\n'\
6741 ' their linux stack\'s suspend/resume time. Using a kernel image built\n'\
6742 ' with a few extra options enabled, the tool will execute a suspend and\n'\
6743 ' capture dmesg and ftrace data until resume is complete. This data is\n'\
6744 ' transformed into a device timeline and an optional callgraph to give\n'\
6745 ' a detailed view of which devices/subsystems are taking the most\n'\
6746 ' time in suspend/resume.\n'\
6747 '\n'\
6748 ' If no specific command is given, the default behavior is to initiate\n'\
6749 ' a suspend/resume and capture the dmesg/ftrace output as an html timeline.\n'\
6750 '\n'\
6751 ' Generates output files in subdirectory: suspend-yymmdd-HHMMSS\n'\
6752 ' HTML output: <hostname>_<mode>.html\n'\
6753 ' raw dmesg output: <hostname>_<mode>_dmesg.txt\n'\
6754 ' raw ftrace output: <hostname>_<mode>_ftrace.txt\n'\
6755 '\n'\
6756 'Options:\n'\
6757 ' -h Print this help text\n'\
6758 ' -v Print the current tool version\n'\
6759 ' -config fn Pull arguments and config options from file fn\n'\
6760 ' -verbose Print extra information during execution and analysis\n'\
6761 ' -m mode Mode to initiate for suspend (default: %s)\n'\
6762 ' -o name Overrides the output subdirectory name when running a new test\n'\
6763 ' default: suspend-{date}-{time}\n'\
6764 ' -rtcwake t Wakeup t seconds after suspend, set t to "off" to disable (default: 15)\n'\
6765 ' -addlogs Add the dmesg and ftrace logs to the html output\n'\
6766 ' -noturbostat Dont use turbostat in freeze mode (default: disabled)\n'\
6767 ' -srgap Add a visible gap in the timeline between sus/res (default: disabled)\n'\
6768 ' -skiphtml Run the test and capture the trace logs, but skip the timeline (default: disabled)\n'\
6769 ' -result fn Export a results table to a text file for parsing.\n'\
6770 ' -wifi If a wifi connection is available, check that it reconnects after resume.\n'\
6771 ' -wifitrace Trace kernel execution through wifi reconnect.\n'\
6772 ' -netfix Use netfix to reset the network in the event it fails to resume.\n'\
6773 ' -debugtiming Add timestamp to each printed line\n'\
6774 ' [testprep]\n'\
6775 ' -sync Sync the filesystems before starting the test\n'\
6776 ' -rs on/off Enable/disable runtime suspend for all devices, restore all after test\n'\
6777 ' -display m Change the display mode to m for the test (on/off/standby/suspend)\n'\
6778 ' [advanced]\n'\
6779 ' -gzip Gzip the trace and dmesg logs to save space\n'\
6780 ' -cmd {s} Run the timeline over a custom command, e.g. "sync -d"\n'\
6781 ' -proc Add usermode process info into the timeline (default: disabled)\n'\
6782 ' -dev Add kernel function calls and threads to the timeline (default: disabled)\n'\
6783 ' -x2 Run two suspend/resumes back to back (default: disabled)\n'\
6784 ' -x2delay t Include t ms delay between multiple test runs (default: 0 ms)\n'\
6785 ' -predelay t Include t ms delay before 1st suspend (default: 0 ms)\n'\
6786 ' -postdelay t Include t ms delay after last resume (default: 0 ms)\n'\
6787 ' -mindev ms Discard all device blocks shorter than ms milliseconds (e.g. 0.001 for us)\n'\
6788 ' -multi n d Execute <n> consecutive tests at <d> seconds intervals. If <n> is followed\n'\
6789 ' by a "d", "h", or "m" execute for <n> days, hours, or mins instead.\n'\
6790 ' The outputs will be created in a new subdirectory with a summary page.\n'\
6791 ' -maxfail n Abort a -multi run after n consecutive fails (default is 0 = never abort)\n'\
6792 ' [debug]\n'\
6793 ' -f Use ftrace to create device callgraphs (default: disabled)\n'\
6794 ' -ftop Use ftrace on the top level call: "%s" (default: disabled)\n'\
6795 ' -maxdepth N limit the callgraph data to N call levels (default: 0=all)\n'\
6796 ' -expandcg pre-expand the callgraph data in the html output (default: disabled)\n'\
6797 ' -fadd file Add functions to be graphed in the timeline from a list in a text file\n'\
6798 ' -filter "d1,d2,..." Filter out all but this comma-delimited list of device names\n'\
6799 ' -mincg ms Discard all callgraphs shorter than ms milliseconds (e.g. 0.001 for us)\n'\
6800 ' -cgphase P Only show callgraph data for phase P (e.g. suspend_late)\n'\
6801 ' -cgtest N Only show callgraph data for test N (e.g. 0 or 1 in an x2 run)\n'\
6802 ' -timeprec N Number of significant digits in timestamps (0:S, [3:ms], 6:us)\n'\
6803 ' -cgfilter S Filter the callgraph output in the timeline\n'\
6804 ' -cgskip file Callgraph functions to skip, off to disable (default: cgskip.txt)\n'\
6805 ' -bufsize N Set trace buffer size to N kilo-bytes (default: all of free memory)\n'\
6806 ' -devdump Print out all the raw device data for each phase\n'\
6807 ' -cgdump Print out all the raw callgraph data\n'\
6808 '\n'\
6809 'Other commands:\n'\
6810 ' -modes List available suspend modes\n'\
6811 ' -status Test to see if the system is enabled to run this tool\n'\
6812 ' -fpdt Print out the contents of the ACPI Firmware Performance Data Table\n'\
6813 ' -wificheck Print out wifi connection info\n'\
6814 ' -x<mode> Test xset by toggling the given mode (on/off/standby/suspend)\n'\
6815 ' -sysinfo Print out system info extracted from BIOS\n'\
6816 ' -devinfo Print out the pm settings of all devices which support runtime suspend\n'\
6817 ' -cmdinfo Print out all the platform info collected before and after suspend/resume\n'\
6818 ' -flist Print the list of functions currently being captured in ftrace\n'\
6819 ' -flistall Print all functions capable of being captured in ftrace\n'\
6820 ' -summary dir Create a summary of tests in this dir [-genhtml builds missing html]\n'\
6821 ' [redo]\n'\
6822 ' -ftrace ftracefile Create HTML output using ftrace input (used with -dmesg)\n'\
6823 ' -dmesg dmesgfile Create HTML output using dmesg (used with -ftrace)\n'\
6824 '' % (sysvals.title, sysvals.version, sysvals.suspendmode, sysvals.ftopfunc))
6825 return True
6826
6827# ----------------- MAIN --------------------
6828# exec start (skipped if script is loaded as library)
6829if __name__ == '__main__':
6830 genhtml = False
6831 cmd = ''
6832 simplecmds = ['-sysinfo', '-modes', '-fpdt', '-flist', '-flistall',
6833 '-devinfo', '-status', '-xon', '-xoff', '-xstandby', '-xsuspend',
6834 '-xinit', '-xreset', '-xstat', '-wificheck', '-cmdinfo']
6835 if '-f' in sys.argv:
6836 sysvals.cgskip = sysvals.configFile('cgskip.txt')
6837 # loop through the command line arguments
6838 args = iter(sys.argv[1:])
6839 for arg in args:
6840 if(arg == '-m'):
6841 try:
6842 val = next(args)
6843 except:
6844 doError('No mode supplied', True)
6845 if val == 'command' and not sysvals.testcommand:
6846 doError('No command supplied for mode "command"', True)
6847 sysvals.suspendmode = val
6848 elif(arg in simplecmds):
6849 cmd = arg[1:]
6850 elif(arg == '-h'):
6851 printHelp()
6852 sys.exit(0)
6853 elif(arg == '-v'):
6854 pprint("Version %s" % sysvals.version)
6855 sys.exit(0)
6856 elif(arg == '-debugtiming'):
6857 debugtiming = True
6858 elif(arg == '-x2'):
6859 sysvals.execcount = 2
6860 elif(arg == '-x2delay'):
6861 sysvals.x2delay = getArgInt('-x2delay', args, 0, 60000)
6862 elif(arg == '-predelay'):
6863 sysvals.predelay = getArgInt('-predelay', args, 0, 60000)
6864 elif(arg == '-postdelay'):
6865 sysvals.postdelay = getArgInt('-postdelay', args, 0, 60000)
6866 elif(arg == '-f'):
6867 sysvals.usecallgraph = True
6868 elif(arg == '-ftop'):
6869 sysvals.usecallgraph = True
6870 sysvals.ftop = True
6871 sysvals.usekprobes = False
6872 elif(arg == '-skiphtml'):
6873 sysvals.skiphtml = True
6874 elif(arg == '-cgdump'):
6875 sysvals.cgdump = True
6876 elif(arg == '-devdump'):
6877 sysvals.devdump = True
6878 elif(arg == '-genhtml'):
6879 genhtml = True
6880 elif(arg == '-addlogs'):
6881 sysvals.dmesglog = sysvals.ftracelog = True
6882 elif(arg == '-nologs'):
6883 sysvals.dmesglog = sysvals.ftracelog = False
6884 elif(arg == '-addlogdmesg'):
6885 sysvals.dmesglog = True
6886 elif(arg == '-addlogftrace'):
6887 sysvals.ftracelog = True
6888 elif(arg == '-noturbostat'):
6889 sysvals.tstat = False
6890 elif(arg == '-verbose'):
6891 sysvals.verbose = True
6892 elif(arg == '-proc'):
6893 sysvals.useprocmon = True
6894 elif(arg == '-dev'):
6895 sysvals.usedevsrc = True
6896 elif(arg == '-sync'):
6897 sysvals.sync = True
6898 elif(arg == '-wifi'):
6899 sysvals.wifi = True
6900 elif(arg == '-wifitrace'):
6901 sysvals.wifitrace = True
6902 elif(arg == '-netfix'):
6903 sysvals.netfix = True
6904 elif(arg == '-gzip'):
6905 sysvals.gzip = True
6906 elif(arg == '-info'):
6907 try:
6908 val = next(args)
6909 except:
6910 doError('-info requires one string argument', True)
6911 elif(arg == '-desc'):
6912 try:
6913 val = next(args)
6914 except:
6915 doError('-desc requires one string argument', True)
6916 elif(arg == '-rs'):
6917 try:
6918 val = next(args)
6919 except:
6920 doError('-rs requires "enable" or "disable"', True)
6921 if val.lower() in switchvalues:
6922 if val.lower() in switchoff:
6923 sysvals.rs = -1
6924 else:
6925 sysvals.rs = 1
6926 else:
6927 doError('invalid option: %s, use "enable/disable" or "on/off"' % val, True)
6928 elif(arg == '-display'):
6929 try:
6930 val = next(args)
6931 except:
6932 doError('-display requires an mode value', True)
6933 disopt = ['on', 'off', 'standby', 'suspend']
6934 if val.lower() not in disopt:
6935 doError('valid display mode values are %s' % disopt, True)
6936 sysvals.display = val.lower()
6937 elif(arg == '-maxdepth'):
6938 sysvals.max_graph_depth = getArgInt('-maxdepth', args, 0, 1000)
6939 elif(arg == '-rtcwake'):
6940 try:
6941 val = next(args)
6942 except:
6943 doError('No rtcwake time supplied', True)
6944 if val.lower() in switchoff:
6945 sysvals.rtcwake = False
6946 else:
6947 sysvals.rtcwake = True
6948 sysvals.rtcwaketime = getArgInt('-rtcwake', val, 0, 3600, False)
6949 elif(arg == '-timeprec'):
6950 sysvals.setPrecision(getArgInt('-timeprec', args, 0, 6))
6951 elif(arg == '-mindev'):
6952 sysvals.mindevlen = getArgFloat('-mindev', args, 0.0, 10000.0)
6953 elif(arg == '-mincg'):
6954 sysvals.mincglen = getArgFloat('-mincg', args, 0.0, 10000.0)
6955 elif(arg == '-bufsize'):
6956 sysvals.bufsize = getArgInt('-bufsize', args, 1, 1024*1024*8)
6957 elif(arg == '-cgtest'):
6958 sysvals.cgtest = getArgInt('-cgtest', args, 0, 1)
6959 elif(arg == '-cgphase'):
6960 try:
6961 val = next(args)
6962 except:
6963 doError('No phase name supplied', True)
6964 d = Data(0)
6965 if val not in d.phasedef:
6966 doError('invalid phase --> (%s: %s), valid phases are %s'\
6967 % (arg, val, d.phasedef.keys()), True)
6968 sysvals.cgphase = val
6969 elif(arg == '-cgfilter'):
6970 try:
6971 val = next(args)
6972 except:
6973 doError('No callgraph functions supplied', True)
6974 sysvals.setCallgraphFilter(val)
6975 elif(arg == '-skipkprobe'):
6976 try:
6977 val = next(args)
6978 except:
6979 doError('No kprobe functions supplied', True)
6980 sysvals.skipKprobes(val)
6981 elif(arg == '-cgskip'):
6982 try:
6983 val = next(args)
6984 except:
6985 doError('No file supplied', True)
6986 if val.lower() in switchoff:
6987 sysvals.cgskip = ''
6988 else:
6989 sysvals.cgskip = sysvals.configFile(val)
6990 if(not sysvals.cgskip):
6991 doError('%s does not exist' % sysvals.cgskip)
6992 elif(arg == '-callloop-maxgap'):
6993 sysvals.callloopmaxgap = getArgFloat('-callloop-maxgap', args, 0.0, 1.0)
6994 elif(arg == '-callloop-maxlen'):
6995 sysvals.callloopmaxlen = getArgFloat('-callloop-maxlen', args, 0.0, 1.0)
6996 elif(arg == '-cmd'):
6997 try:
6998 val = next(args)
6999 except:
7000 doError('No command string supplied', True)
7001 sysvals.testcommand = val
7002 sysvals.suspendmode = 'command'
7003 elif(arg == '-expandcg'):
7004 sysvals.cgexp = True
7005 elif(arg == '-srgap'):
7006 sysvals.srgap = 5
7007 elif(arg == '-maxfail'):
7008 sysvals.maxfail = getArgInt('-maxfail', args, 0, 1000000)
7009 elif(arg == '-multi'):
7010 try:
7011 c, d = next(args), next(args)
7012 except:
7013 doError('-multi requires two values', True)
7014 sysvals.multiinit(c, d)
7015 elif(arg == '-o'):
7016 try:
7017 val = next(args)
7018 except:
7019 doError('No subdirectory name supplied', True)
7020 sysvals.outdir = sysvals.setOutputFolder(val)
7021 elif(arg == '-config'):
7022 try:
7023 val = next(args)
7024 except:
7025 doError('No text file supplied', True)
7026 file = sysvals.configFile(val)
7027 if(not file):
7028 doError('%s does not exist' % val)
7029 configFromFile(file)
7030 elif(arg == '-fadd'):
7031 try:
7032 val = next(args)
7033 except:
7034 doError('No text file supplied', True)
7035 file = sysvals.configFile(val)
7036 if(not file):
7037 doError('%s does not exist' % val)
7038 sysvals.addFtraceFilterFunctions(file)
7039 elif(arg == '-dmesg'):
7040 try:
7041 val = next(args)
7042 except:
7043 doError('No dmesg file supplied', True)
7044 sysvals.notestrun = True
7045 sysvals.dmesgfile = val
7046 if(os.path.exists(sysvals.dmesgfile) == False):
7047 doError('%s does not exist' % sysvals.dmesgfile)
7048 elif(arg == '-ftrace'):
7049 try:
7050 val = next(args)
7051 except:
7052 doError('No ftrace file supplied', True)
7053 sysvals.notestrun = True
7054 sysvals.ftracefile = val
7055 if(os.path.exists(sysvals.ftracefile) == False):
7056 doError('%s does not exist' % sysvals.ftracefile)
7057 elif(arg == '-summary'):
7058 try:
7059 val = next(args)
7060 except:
7061 doError('No directory supplied', True)
7062 cmd = 'summary'
7063 sysvals.outdir = val
7064 sysvals.notestrun = True
7065 if(os.path.isdir(val) == False):
7066 doError('%s is not accesible' % val)
7067 elif(arg == '-filter'):
7068 try:
7069 val = next(args)
7070 except:
7071 doError('No devnames supplied', True)
7072 sysvals.setDeviceFilter(val)
7073 elif(arg == '-result'):
7074 try:
7075 val = next(args)
7076 except:
7077 doError('No result file supplied', True)
7078 sysvals.result = val
7079 else:
7080 doError('Invalid argument: '+arg, True)
7081
7082 # compatibility errors
7083 if(sysvals.usecallgraph and sysvals.usedevsrc):
7084 doError('-dev is not compatible with -f')
7085 if(sysvals.usecallgraph and sysvals.useprocmon):
7086 doError('-proc is not compatible with -f')
7087
7088 sysvals.signalHandlerInit()
7089 if sysvals.usecallgraph and sysvals.cgskip:
7090 sysvals.vprint('Using cgskip file: %s' % sysvals.cgskip)
7091 sysvals.setCallgraphBlacklist(sysvals.cgskip)
7092
7093 # callgraph size cannot exceed device size
7094 if sysvals.mincglen < sysvals.mindevlen:
7095 sysvals.mincglen = sysvals.mindevlen
7096
7097 # remove existing buffers before calculating memory
7098 if(sysvals.usecallgraph or sysvals.usedevsrc):
7099 sysvals.fsetVal('16', 'buffer_size_kb')
7100 sysvals.cpuInfo()
7101
7102 # just run a utility command and exit
7103 if(cmd != ''):
7104 ret = 0
7105 if(cmd == 'status'):
7106 if not statusCheck(True):
7107 ret = 1
7108 elif(cmd == 'fpdt'):
7109 if not getFPDT(True):
7110 ret = 1
7111 elif(cmd == 'sysinfo'):
7112 sysvals.printSystemInfo(True)
7113 elif(cmd == 'devinfo'):
7114 deviceInfo()
7115 elif(cmd == 'modes'):
7116 pprint(getModes())
7117 elif(cmd == 'flist'):
7118 sysvals.getFtraceFilterFunctions(True)
7119 elif(cmd == 'flistall'):
7120 sysvals.getFtraceFilterFunctions(False)
7121 elif(cmd == 'summary'):
7122 runSummary(sysvals.outdir, True, genhtml)
7123 elif(cmd in ['xon', 'xoff', 'xstandby', 'xsuspend', 'xinit', 'xreset']):
7124 sysvals.verbose = True
7125 ret = sysvals.displayControl(cmd[1:])
7126 elif(cmd == 'xstat'):
7127 pprint('Display Status: %s' % sysvals.displayControl('stat').upper())
7128 elif(cmd == 'wificheck'):
7129 dev = sysvals.checkWifi()
7130 if dev:
7131 print('%s is connected' % sysvals.wifiDetails(dev))
7132 else:
7133 print('No wifi connection found')
7134 elif(cmd == 'cmdinfo'):
7135 for out in sysvals.cmdinfo(False, True):
7136 print('[%s - %s]\n%s\n' % out)
7137 sys.exit(ret)
7138
7139 # if instructed, re-analyze existing data files
7140 if(sysvals.notestrun):
7141 stamp = rerunTest(sysvals.outdir)
7142 sysvals.outputResult(stamp)
7143 sys.exit(0)
7144
7145 # verify that we can run a test
7146 error = statusCheck()
7147 if(error):
7148 doError(error)
7149
7150 # extract mem/disk extra modes and convert
7151 mode = sysvals.suspendmode
7152 if mode.startswith('mem'):
7153 memmode = mode.split('-', 1)[-1] if '-' in mode else 'deep'
7154 if memmode == 'shallow':
7155 mode = 'standby'
7156 elif memmode == 's2idle':
7157 mode = 'freeze'
7158 else:
7159 mode = 'mem'
7160 sysvals.memmode = memmode
7161 sysvals.suspendmode = mode
7162 if mode.startswith('disk-'):
7163 sysvals.diskmode = mode.split('-', 1)[-1]
7164 sysvals.suspendmode = 'disk'
7165 sysvals.systemInfo(dmidecode(sysvals.mempath))
7166
7167 failcnt, ret = 0, 0
7168 if sysvals.multitest['run']:
7169 # run multiple tests in a separate subdirectory
7170 if not sysvals.outdir:
7171 if 'time' in sysvals.multitest:
7172 s = '-%dm' % sysvals.multitest['time']
7173 else:
7174 s = '-x%d' % sysvals.multitest['count']
7175 sysvals.outdir = datetime.now().strftime('suspend-%y%m%d-%H%M%S'+s)
7176 if not os.path.isdir(sysvals.outdir):
7177 os.makedirs(sysvals.outdir)
7178 sysvals.sudoUserchown(sysvals.outdir)
7179 finish = datetime.now()
7180 if 'time' in sysvals.multitest:
7181 finish += timedelta(minutes=sysvals.multitest['time'])
7182 for i in range(sysvals.multitest['count']):
7183 sysvals.multistat(True, i, finish)
7184 if i != 0 and sysvals.multitest['delay'] > 0:
7185 pprint('Waiting %d seconds...' % (sysvals.multitest['delay']))
7186 time.sleep(sysvals.multitest['delay'])
7187 fmt = 'suspend-%y%m%d-%H%M%S'
7188 sysvals.testdir = os.path.join(sysvals.outdir, datetime.now().strftime(fmt))
7189 ret = runTest(i+1, not sysvals.verbose)
7190 failcnt = 0 if not ret else failcnt + 1
7191 if sysvals.maxfail > 0 and failcnt >= sysvals.maxfail:
7192 pprint('Maximum fail count of %d reached, aborting multitest' % (sysvals.maxfail))
7193 break
7194 sysvals.resetlog()
7195 sysvals.multistat(False, i, finish)
7196 if 'time' in sysvals.multitest and datetime.now() >= finish:
7197 break
7198 if not sysvals.skiphtml:
7199 runSummary(sysvals.outdir, False, False)
7200 sysvals.sudoUserchown(sysvals.outdir)
7201 else:
7202 if sysvals.outdir:
7203 sysvals.testdir = sysvals.outdir
7204 # run the test in the current directory
7205 ret = runTest()
7206
7207 # reset to default values after testing
7208 if sysvals.display:
7209 sysvals.displayControl('reset')
7210 if sysvals.rs != 0:
7211 sysvals.setRuntimeSuspend(False)
7212 sys.exit(ret)
1#!/usr/bin/env python3
2# SPDX-License-Identifier: GPL-2.0-only
3#
4# Tool for analyzing suspend/resume timing
5# Copyright (c) 2013, Intel Corporation.
6#
7# This program is free software; you can redistribute it and/or modify it
8# under the terms and conditions of the GNU General Public License,
9# version 2, as published by the Free Software Foundation.
10#
11# This program is distributed in the hope it will be useful, but WITHOUT
12# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
14# more details.
15#
16# Authors:
17# Todd Brandt <todd.e.brandt@linux.intel.com>
18#
19# Links:
20# Home Page
21# https://01.org/pm-graph
22# Source repo
23# git@github.com:intel/pm-graph
24#
25# Description:
26# This tool is designed to assist kernel and OS developers in optimizing
27# their linux stack's suspend/resume time. Using a kernel image built
28# with a few extra options enabled, the tool will execute a suspend and
29# will capture dmesg and ftrace data until resume is complete. This data
30# is transformed into a device timeline and a callgraph to give a quick
31# and detailed view of which devices and callbacks are taking the most
32# time in suspend/resume. The output is a single html file which can be
33# viewed in firefox or chrome.
34#
35# The following kernel build options are required:
36# CONFIG_DEVMEM=y
37# CONFIG_PM_DEBUG=y
38# CONFIG_PM_SLEEP_DEBUG=y
39# CONFIG_FTRACE=y
40# CONFIG_FUNCTION_TRACER=y
41# CONFIG_FUNCTION_GRAPH_TRACER=y
42# CONFIG_KPROBES=y
43# CONFIG_KPROBES_ON_FTRACE=y
44#
45# For kernel versions older than 3.15:
46# The following additional kernel parameters are required:
47# (e.g. in file /etc/default/grub)
48# GRUB_CMDLINE_LINUX_DEFAULT="... initcall_debug log_buf_len=16M ..."
49#
50
51# ----------------- LIBRARIES --------------------
52
53import sys
54import time
55import os
56import string
57import re
58import platform
59import signal
60import codecs
61from datetime import datetime, timedelta
62import struct
63import configparser
64import gzip
65from threading import Thread
66from subprocess import call, Popen, PIPE
67import base64
68
69debugtiming = False
70mystarttime = time.time()
71def pprint(msg):
72 if debugtiming:
73 print('[%09.3f] %s' % (time.time()-mystarttime, msg))
74 else:
75 print(msg)
76 sys.stdout.flush()
77
78def ascii(text):
79 return text.decode('ascii', 'ignore')
80
81# ----------------- CLASSES --------------------
82
83# Class: SystemValues
84# Description:
85# A global, single-instance container used to
86# store system values and test parameters
87class SystemValues:
88 title = 'SleepGraph'
89 version = '5.10'
90 ansi = False
91 rs = 0
92 display = ''
93 gzip = False
94 sync = False
95 wifi = False
96 netfix = False
97 verbose = False
98 testlog = True
99 dmesglog = True
100 ftracelog = False
101 acpidebug = True
102 tstat = True
103 wifitrace = False
104 mindevlen = 0.0001
105 mincglen = 0.0
106 cgphase = ''
107 cgtest = -1
108 cgskip = ''
109 maxfail = 0
110 multitest = {'run': False, 'count': 1000000, 'delay': 0}
111 max_graph_depth = 0
112 callloopmaxgap = 0.0001
113 callloopmaxlen = 0.005
114 bufsize = 0
115 cpucount = 0
116 memtotal = 204800
117 memfree = 204800
118 osversion = ''
119 srgap = 0
120 cgexp = False
121 testdir = ''
122 outdir = ''
123 tpath = '/sys/kernel/debug/tracing/'
124 fpdtpath = '/sys/firmware/acpi/tables/FPDT'
125 epath = '/sys/kernel/debug/tracing/events/power/'
126 pmdpath = '/sys/power/pm_debug_messages'
127 s0ixpath = '/sys/module/intel_pmc_core/parameters/warn_on_s0ix_failures'
128 s0ixres = '/sys/devices/system/cpu/cpuidle/low_power_idle_system_residency_us'
129 acpipath='/sys/module/acpi/parameters/debug_level'
130 traceevents = [
131 'suspend_resume',
132 'wakeup_source_activate',
133 'wakeup_source_deactivate',
134 'device_pm_callback_end',
135 'device_pm_callback_start'
136 ]
137 logmsg = ''
138 testcommand = ''
139 mempath = '/dev/mem'
140 powerfile = '/sys/power/state'
141 mempowerfile = '/sys/power/mem_sleep'
142 diskpowerfile = '/sys/power/disk'
143 suspendmode = 'mem'
144 memmode = ''
145 diskmode = ''
146 hostname = 'localhost'
147 prefix = 'test'
148 teststamp = ''
149 sysstamp = ''
150 dmesgstart = 0.0
151 dmesgfile = ''
152 ftracefile = ''
153 htmlfile = 'output.html'
154 result = ''
155 rtcwake = True
156 rtcwaketime = 15
157 rtcpath = ''
158 devicefilter = []
159 cgfilter = []
160 stamp = 0
161 execcount = 1
162 x2delay = 0
163 skiphtml = False
164 usecallgraph = False
165 ftopfunc = 'pm_suspend'
166 ftop = False
167 usetraceevents = False
168 usetracemarkers = True
169 useftrace = True
170 usekprobes = True
171 usedevsrc = False
172 useprocmon = False
173 notestrun = False
174 cgdump = False
175 devdump = False
176 mixedphaseheight = True
177 devprops = dict()
178 cfgdef = dict()
179 platinfo = []
180 predelay = 0
181 postdelay = 0
182 tmstart = 'SUSPEND START %Y%m%d-%H:%M:%S.%f'
183 tmend = 'RESUME COMPLETE %Y%m%d-%H:%M:%S.%f'
184 tracefuncs = {
185 'async_synchronize_full': {},
186 'sys_sync': {},
187 'ksys_sync': {},
188 '__pm_notifier_call_chain': {},
189 'pm_prepare_console': {},
190 'pm_notifier_call_chain': {},
191 'freeze_processes': {},
192 'freeze_kernel_threads': {},
193 'pm_restrict_gfp_mask': {},
194 'acpi_suspend_begin': {},
195 'acpi_hibernation_begin': {},
196 'acpi_hibernation_enter': {},
197 'acpi_hibernation_leave': {},
198 'acpi_pm_freeze': {},
199 'acpi_pm_thaw': {},
200 'acpi_s2idle_end': {},
201 'acpi_s2idle_sync': {},
202 'acpi_s2idle_begin': {},
203 'acpi_s2idle_prepare': {},
204 'acpi_s2idle_prepare_late': {},
205 'acpi_s2idle_wake': {},
206 'acpi_s2idle_wakeup': {},
207 'acpi_s2idle_restore': {},
208 'acpi_s2idle_restore_early': {},
209 'hibernate_preallocate_memory': {},
210 'create_basic_memory_bitmaps': {},
211 'swsusp_write': {},
212 'suspend_console': {},
213 'acpi_pm_prepare': {},
214 'syscore_suspend': {},
215 'arch_enable_nonboot_cpus_end': {},
216 'syscore_resume': {},
217 'acpi_pm_finish': {},
218 'resume_console': {},
219 'acpi_pm_end': {},
220 'pm_restore_gfp_mask': {},
221 'thaw_processes': {},
222 'pm_restore_console': {},
223 'CPU_OFF': {
224 'func':'_cpu_down',
225 'args_x86_64': {'cpu':'%di:s32'},
226 'format': 'CPU_OFF[{cpu}]'
227 },
228 'CPU_ON': {
229 'func':'_cpu_up',
230 'args_x86_64': {'cpu':'%di:s32'},
231 'format': 'CPU_ON[{cpu}]'
232 },
233 }
234 dev_tracefuncs = {
235 # general wait/delay/sleep
236 'msleep': { 'args_x86_64': {'time':'%di:s32'}, 'ub': 1 },
237 'schedule_timeout': { 'args_x86_64': {'timeout':'%di:s32'}, 'ub': 1 },
238 'udelay': { 'func':'__const_udelay', 'args_x86_64': {'loops':'%di:s32'}, 'ub': 1 },
239 'usleep_range': { 'args_x86_64': {'min':'%di:s32', 'max':'%si:s32'}, 'ub': 1 },
240 'mutex_lock_slowpath': { 'func':'__mutex_lock_slowpath', 'ub': 1 },
241 'acpi_os_stall': {'ub': 1},
242 'rt_mutex_slowlock': {'ub': 1},
243 # ACPI
244 'acpi_resume_power_resources': {},
245 'acpi_ps_execute_method': { 'args_x86_64': {
246 'fullpath':'+0(+40(%di)):string',
247 }},
248 # mei_me
249 'mei_reset': {},
250 # filesystem
251 'ext4_sync_fs': {},
252 # 80211
253 'ath10k_bmi_read_memory': { 'args_x86_64': {'length':'%cx:s32'} },
254 'ath10k_bmi_write_memory': { 'args_x86_64': {'length':'%cx:s32'} },
255 'ath10k_bmi_fast_download': { 'args_x86_64': {'length':'%cx:s32'} },
256 'iwlagn_mac_start': {},
257 'iwlagn_alloc_bcast_station': {},
258 'iwl_trans_pcie_start_hw': {},
259 'iwl_trans_pcie_start_fw': {},
260 'iwl_run_init_ucode': {},
261 'iwl_load_ucode_wait_alive': {},
262 'iwl_alive_start': {},
263 'iwlagn_mac_stop': {},
264 'iwlagn_mac_suspend': {},
265 'iwlagn_mac_resume': {},
266 'iwlagn_mac_add_interface': {},
267 'iwlagn_mac_remove_interface': {},
268 'iwlagn_mac_change_interface': {},
269 'iwlagn_mac_config': {},
270 'iwlagn_configure_filter': {},
271 'iwlagn_mac_hw_scan': {},
272 'iwlagn_bss_info_changed': {},
273 'iwlagn_mac_channel_switch': {},
274 'iwlagn_mac_flush': {},
275 # ATA
276 'ata_eh_recover': { 'args_x86_64': {'port':'+36(%di):s32'} },
277 # i915
278 'i915_gem_resume': {},
279 'i915_restore_state': {},
280 'intel_opregion_setup': {},
281 'g4x_pre_enable_dp': {},
282 'vlv_pre_enable_dp': {},
283 'chv_pre_enable_dp': {},
284 'g4x_enable_dp': {},
285 'vlv_enable_dp': {},
286 'intel_hpd_init': {},
287 'intel_opregion_register': {},
288 'intel_dp_detect': {},
289 'intel_hdmi_detect': {},
290 'intel_opregion_init': {},
291 'intel_fbdev_set_suspend': {},
292 }
293 infocmds = [
294 [0, 'sysinfo', 'uname', '-a'],
295 [0, 'cpuinfo', 'head', '-7', '/proc/cpuinfo'],
296 [0, 'kparams', 'cat', '/proc/cmdline'],
297 [0, 'mcelog', 'mcelog'],
298 [0, 'pcidevices', 'lspci', '-tv'],
299 [0, 'usbdevices', 'lsusb', '-tv'],
300 [0, 'acpidevices', 'sh', '-c', 'ls -l /sys/bus/acpi/devices/*/physical_node'],
301 [0, 's0ix_require', 'cat', '/sys/kernel/debug/pmc_core/substate_requirements'],
302 [0, 's0ix_debug', 'cat', '/sys/kernel/debug/pmc_core/slp_s0_debug_status'],
303 [1, 's0ix_residency', 'cat', '/sys/kernel/debug/pmc_core/slp_s0_residency_usec'],
304 [1, 'interrupts', 'cat', '/proc/interrupts'],
305 [1, 'wakeups', 'cat', '/sys/kernel/debug/wakeup_sources'],
306 [2, 'gpecounts', 'sh', '-c', 'grep -v invalid /sys/firmware/acpi/interrupts/*'],
307 [2, 'suspendstats', 'sh', '-c', 'grep -v invalid /sys/power/suspend_stats/*'],
308 [2, 'cpuidle', 'sh', '-c', 'grep -v invalid /sys/devices/system/cpu/cpu*/cpuidle/state*/s2idle/*'],
309 [2, 'battery', 'sh', '-c', 'grep -v invalid /sys/class/power_supply/*/*'],
310 [2, 'thermal', 'sh', '-c', 'grep . /sys/class/thermal/thermal_zone*/temp'],
311 ]
312 cgblacklist = []
313 kprobes = dict()
314 timeformat = '%.3f'
315 cmdline = '%s %s' % \
316 (os.path.basename(sys.argv[0]), ' '.join(sys.argv[1:]))
317 sudouser = ''
318 def __init__(self):
319 self.archargs = 'args_'+platform.machine()
320 self.hostname = platform.node()
321 if(self.hostname == ''):
322 self.hostname = 'localhost'
323 rtc = "rtc0"
324 if os.path.exists('/dev/rtc'):
325 rtc = os.readlink('/dev/rtc')
326 rtc = '/sys/class/rtc/'+rtc
327 if os.path.exists(rtc) and os.path.exists(rtc+'/date') and \
328 os.path.exists(rtc+'/time') and os.path.exists(rtc+'/wakealarm'):
329 self.rtcpath = rtc
330 if (hasattr(sys.stdout, 'isatty') and sys.stdout.isatty()):
331 self.ansi = True
332 self.testdir = datetime.now().strftime('suspend-%y%m%d-%H%M%S')
333 if os.getuid() == 0 and 'SUDO_USER' in os.environ and \
334 os.environ['SUDO_USER']:
335 self.sudouser = os.environ['SUDO_USER']
336 def resetlog(self):
337 self.logmsg = ''
338 self.platinfo = []
339 def vprint(self, msg):
340 self.logmsg += msg+'\n'
341 if self.verbose or msg.startswith('WARNING:'):
342 pprint(msg)
343 def signalHandler(self, signum, frame):
344 if not self.result:
345 return
346 signame = self.signames[signum] if signum in self.signames else 'UNKNOWN'
347 msg = 'Signal %s caused a tool exit, line %d' % (signame, frame.f_lineno)
348 self.outputResult({'error':msg})
349 sys.exit(3)
350 def signalHandlerInit(self):
351 capture = ['BUS', 'SYS', 'XCPU', 'XFSZ', 'PWR', 'HUP', 'INT', 'QUIT',
352 'ILL', 'ABRT', 'FPE', 'SEGV', 'TERM']
353 self.signames = dict()
354 for i in capture:
355 s = 'SIG'+i
356 try:
357 signum = getattr(signal, s)
358 signal.signal(signum, self.signalHandler)
359 except:
360 continue
361 self.signames[signum] = s
362 def rootCheck(self, fatal=True):
363 if(os.access(self.powerfile, os.W_OK)):
364 return True
365 if fatal:
366 msg = 'This command requires sysfs mount and root access'
367 pprint('ERROR: %s\n' % msg)
368 self.outputResult({'error':msg})
369 sys.exit(1)
370 return False
371 def rootUser(self, fatal=False):
372 if 'USER' in os.environ and os.environ['USER'] == 'root':
373 return True
374 if fatal:
375 msg = 'This command must be run as root'
376 pprint('ERROR: %s\n' % msg)
377 self.outputResult({'error':msg})
378 sys.exit(1)
379 return False
380 def usable(self, file, ishtml=False):
381 if not os.path.exists(file) or os.path.getsize(file) < 1:
382 return False
383 if ishtml:
384 try:
385 fp = open(file, 'r')
386 res = fp.read(1000)
387 fp.close()
388 except:
389 return False
390 if '<html>' not in res:
391 return False
392 return True
393 def getExec(self, cmd):
394 try:
395 fp = Popen(['which', cmd], stdout=PIPE, stderr=PIPE).stdout
396 out = ascii(fp.read()).strip()
397 fp.close()
398 except:
399 out = ''
400 if out:
401 return out
402 for path in ['/sbin', '/bin', '/usr/sbin', '/usr/bin',
403 '/usr/local/sbin', '/usr/local/bin']:
404 cmdfull = os.path.join(path, cmd)
405 if os.path.exists(cmdfull):
406 return cmdfull
407 return out
408 def setPrecision(self, num):
409 if num < 0 or num > 6:
410 return
411 self.timeformat = '%.{0}f'.format(num)
412 def setOutputFolder(self, value):
413 args = dict()
414 n = datetime.now()
415 args['date'] = n.strftime('%y%m%d')
416 args['time'] = n.strftime('%H%M%S')
417 args['hostname'] = args['host'] = self.hostname
418 args['mode'] = self.suspendmode
419 return value.format(**args)
420 def setOutputFile(self):
421 if self.dmesgfile != '':
422 m = re.match('(?P<name>.*)_dmesg\.txt.*', self.dmesgfile)
423 if(m):
424 self.htmlfile = m.group('name')+'.html'
425 if self.ftracefile != '':
426 m = re.match('(?P<name>.*)_ftrace\.txt.*', self.ftracefile)
427 if(m):
428 self.htmlfile = m.group('name')+'.html'
429 def systemInfo(self, info):
430 p = m = ''
431 if 'baseboard-manufacturer' in info:
432 m = info['baseboard-manufacturer']
433 elif 'system-manufacturer' in info:
434 m = info['system-manufacturer']
435 if 'system-product-name' in info:
436 p = info['system-product-name']
437 elif 'baseboard-product-name' in info:
438 p = info['baseboard-product-name']
439 if m[:5].lower() == 'intel' and 'baseboard-product-name' in info:
440 p = info['baseboard-product-name']
441 c = info['processor-version'] if 'processor-version' in info else ''
442 b = info['bios-version'] if 'bios-version' in info else ''
443 r = info['bios-release-date'] if 'bios-release-date' in info else ''
444 self.sysstamp = '# sysinfo | man:%s | plat:%s | cpu:%s | bios:%s | biosdate:%s | numcpu:%d | memsz:%d | memfr:%d' % \
445 (m, p, c, b, r, self.cpucount, self.memtotal, self.memfree)
446 if self.osversion:
447 self.sysstamp += ' | os:%s' % self.osversion
448 def printSystemInfo(self, fatal=False):
449 self.rootCheck(True)
450 out = dmidecode(self.mempath, fatal)
451 if len(out) < 1:
452 return
453 fmt = '%-24s: %s'
454 if self.osversion:
455 print(fmt % ('os-version', self.osversion))
456 for name in sorted(out):
457 print(fmt % (name, out[name]))
458 print(fmt % ('cpucount', ('%d' % self.cpucount)))
459 print(fmt % ('memtotal', ('%d kB' % self.memtotal)))
460 print(fmt % ('memfree', ('%d kB' % self.memfree)))
461 def cpuInfo(self):
462 self.cpucount = 0
463 if os.path.exists('/proc/cpuinfo'):
464 with open('/proc/cpuinfo', 'r') as fp:
465 for line in fp:
466 if re.match('^processor[ \t]*:[ \t]*[0-9]*', line):
467 self.cpucount += 1
468 if os.path.exists('/proc/meminfo'):
469 with open('/proc/meminfo', 'r') as fp:
470 for line in fp:
471 m = re.match('^MemTotal:[ \t]*(?P<sz>[0-9]*) *kB', line)
472 if m:
473 self.memtotal = int(m.group('sz'))
474 m = re.match('^MemFree:[ \t]*(?P<sz>[0-9]*) *kB', line)
475 if m:
476 self.memfree = int(m.group('sz'))
477 if os.path.exists('/etc/os-release'):
478 with open('/etc/os-release', 'r') as fp:
479 for line in fp:
480 if line.startswith('PRETTY_NAME='):
481 self.osversion = line[12:].strip().replace('"', '')
482 def initTestOutput(self, name):
483 self.prefix = self.hostname
484 v = open('/proc/version', 'r').read().strip()
485 kver = v.split()[2]
486 fmt = name+'-%m%d%y-%H%M%S'
487 testtime = datetime.now().strftime(fmt)
488 self.teststamp = \
489 '# '+testtime+' '+self.prefix+' '+self.suspendmode+' '+kver
490 ext = ''
491 if self.gzip:
492 ext = '.gz'
493 self.dmesgfile = \
494 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'_dmesg.txt'+ext
495 self.ftracefile = \
496 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'_ftrace.txt'+ext
497 self.htmlfile = \
498 self.testdir+'/'+self.prefix+'_'+self.suspendmode+'.html'
499 if not os.path.isdir(self.testdir):
500 os.makedirs(self.testdir)
501 self.sudoUserchown(self.testdir)
502 def getValueList(self, value):
503 out = []
504 for i in value.split(','):
505 if i.strip():
506 out.append(i.strip())
507 return out
508 def setDeviceFilter(self, value):
509 self.devicefilter = self.getValueList(value)
510 def setCallgraphFilter(self, value):
511 self.cgfilter = self.getValueList(value)
512 def skipKprobes(self, value):
513 for k in self.getValueList(value):
514 if k in self.tracefuncs:
515 del self.tracefuncs[k]
516 if k in self.dev_tracefuncs:
517 del self.dev_tracefuncs[k]
518 def setCallgraphBlacklist(self, file):
519 self.cgblacklist = self.listFromFile(file)
520 def rtcWakeAlarmOn(self):
521 call('echo 0 > '+self.rtcpath+'/wakealarm', shell=True)
522 nowtime = open(self.rtcpath+'/since_epoch', 'r').read().strip()
523 if nowtime:
524 nowtime = int(nowtime)
525 else:
526 # if hardware time fails, use the software time
527 nowtime = int(datetime.now().strftime('%s'))
528 alarm = nowtime + self.rtcwaketime
529 call('echo %d > %s/wakealarm' % (alarm, self.rtcpath), shell=True)
530 def rtcWakeAlarmOff(self):
531 call('echo 0 > %s/wakealarm' % self.rtcpath, shell=True)
532 def initdmesg(self):
533 # get the latest time stamp from the dmesg log
534 lines = Popen('dmesg', stdout=PIPE).stdout.readlines()
535 ktime = '0'
536 for line in reversed(lines):
537 line = ascii(line).replace('\r\n', '')
538 idx = line.find('[')
539 if idx > 1:
540 line = line[idx:]
541 m = re.match('[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
542 if(m):
543 ktime = m.group('ktime')
544 break
545 self.dmesgstart = float(ktime)
546 def getdmesg(self, testdata):
547 op = self.writeDatafileHeader(self.dmesgfile, testdata)
548 # store all new dmesg lines since initdmesg was called
549 fp = Popen('dmesg', stdout=PIPE).stdout
550 for line in fp:
551 line = ascii(line).replace('\r\n', '')
552 idx = line.find('[')
553 if idx > 1:
554 line = line[idx:]
555 m = re.match('[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
556 if(not m):
557 continue
558 ktime = float(m.group('ktime'))
559 if ktime > self.dmesgstart:
560 op.write(line)
561 fp.close()
562 op.close()
563 def listFromFile(self, file):
564 list = []
565 fp = open(file)
566 for i in fp.read().split('\n'):
567 i = i.strip()
568 if i and i[0] != '#':
569 list.append(i)
570 fp.close()
571 return list
572 def addFtraceFilterFunctions(self, file):
573 for i in self.listFromFile(file):
574 if len(i) < 2:
575 continue
576 self.tracefuncs[i] = dict()
577 def getFtraceFilterFunctions(self, current):
578 self.rootCheck(True)
579 if not current:
580 call('cat '+self.tpath+'available_filter_functions', shell=True)
581 return
582 master = self.listFromFile(self.tpath+'available_filter_functions')
583 for i in sorted(self.tracefuncs):
584 if 'func' in self.tracefuncs[i]:
585 i = self.tracefuncs[i]['func']
586 if i in master:
587 print(i)
588 else:
589 print(self.colorText(i))
590 def setFtraceFilterFunctions(self, list):
591 master = self.listFromFile(self.tpath+'available_filter_functions')
592 flist = ''
593 for i in list:
594 if i not in master:
595 continue
596 if ' [' in i:
597 flist += i.split(' ')[0]+'\n'
598 else:
599 flist += i+'\n'
600 fp = open(self.tpath+'set_graph_function', 'w')
601 fp.write(flist)
602 fp.close()
603 def basicKprobe(self, name):
604 self.kprobes[name] = {'name': name,'func': name,'args': dict(),'format': name}
605 def defaultKprobe(self, name, kdata):
606 k = kdata
607 for field in ['name', 'format', 'func']:
608 if field not in k:
609 k[field] = name
610 if self.archargs in k:
611 k['args'] = k[self.archargs]
612 else:
613 k['args'] = dict()
614 k['format'] = name
615 self.kprobes[name] = k
616 def kprobeColor(self, name):
617 if name not in self.kprobes or 'color' not in self.kprobes[name]:
618 return ''
619 return self.kprobes[name]['color']
620 def kprobeDisplayName(self, name, dataraw):
621 if name not in self.kprobes:
622 self.basicKprobe(name)
623 data = ''
624 quote=0
625 # first remvoe any spaces inside quotes, and the quotes
626 for c in dataraw:
627 if c == '"':
628 quote = (quote + 1) % 2
629 if quote and c == ' ':
630 data += '_'
631 elif c != '"':
632 data += c
633 fmt, args = self.kprobes[name]['format'], self.kprobes[name]['args']
634 arglist = dict()
635 # now process the args
636 for arg in sorted(args):
637 arglist[arg] = ''
638 m = re.match('.* '+arg+'=(?P<arg>.*) ', data);
639 if m:
640 arglist[arg] = m.group('arg')
641 else:
642 m = re.match('.* '+arg+'=(?P<arg>.*)', data);
643 if m:
644 arglist[arg] = m.group('arg')
645 out = fmt.format(**arglist)
646 out = out.replace(' ', '_').replace('"', '')
647 return out
648 def kprobeText(self, kname, kprobe):
649 name = fmt = func = kname
650 args = dict()
651 if 'name' in kprobe:
652 name = kprobe['name']
653 if 'format' in kprobe:
654 fmt = kprobe['format']
655 if 'func' in kprobe:
656 func = kprobe['func']
657 if self.archargs in kprobe:
658 args = kprobe[self.archargs]
659 if 'args' in kprobe:
660 args = kprobe['args']
661 if re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', func):
662 doError('Kprobe "%s" has format info in the function name "%s"' % (name, func))
663 for arg in re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', fmt):
664 if arg not in args:
665 doError('Kprobe "%s" is missing argument "%s"' % (name, arg))
666 val = 'p:%s_cal %s' % (name, func)
667 for i in sorted(args):
668 val += ' %s=%s' % (i, args[i])
669 val += '\nr:%s_ret %s $retval\n' % (name, func)
670 return val
671 def addKprobes(self, output=False):
672 if len(self.kprobes) < 1:
673 return
674 if output:
675 pprint(' kprobe functions in this kernel:')
676 # first test each kprobe
677 rejects = []
678 # sort kprobes: trace, ub-dev, custom, dev
679 kpl = [[], [], [], []]
680 linesout = len(self.kprobes)
681 for name in sorted(self.kprobes):
682 res = self.colorText('YES', 32)
683 if not self.testKprobe(name, self.kprobes[name]):
684 res = self.colorText('NO')
685 rejects.append(name)
686 else:
687 if name in self.tracefuncs:
688 kpl[0].append(name)
689 elif name in self.dev_tracefuncs:
690 if 'ub' in self.dev_tracefuncs[name]:
691 kpl[1].append(name)
692 else:
693 kpl[3].append(name)
694 else:
695 kpl[2].append(name)
696 if output:
697 pprint(' %s: %s' % (name, res))
698 kplist = kpl[0] + kpl[1] + kpl[2] + kpl[3]
699 # remove all failed ones from the list
700 for name in rejects:
701 self.kprobes.pop(name)
702 # set the kprobes all at once
703 self.fsetVal('', 'kprobe_events')
704 kprobeevents = ''
705 for kp in kplist:
706 kprobeevents += self.kprobeText(kp, self.kprobes[kp])
707 self.fsetVal(kprobeevents, 'kprobe_events')
708 if output:
709 check = self.fgetVal('kprobe_events')
710 linesack = (len(check.split('\n')) - 1) // 2
711 pprint(' kprobe functions enabled: %d/%d' % (linesack, linesout))
712 self.fsetVal('1', 'events/kprobes/enable')
713 def testKprobe(self, kname, kprobe):
714 self.fsetVal('0', 'events/kprobes/enable')
715 kprobeevents = self.kprobeText(kname, kprobe)
716 if not kprobeevents:
717 return False
718 try:
719 self.fsetVal(kprobeevents, 'kprobe_events')
720 check = self.fgetVal('kprobe_events')
721 except:
722 return False
723 linesout = len(kprobeevents.split('\n'))
724 linesack = len(check.split('\n'))
725 if linesack < linesout:
726 return False
727 return True
728 def setVal(self, val, file):
729 if not os.path.exists(file):
730 return False
731 try:
732 fp = open(file, 'wb', 0)
733 fp.write(val.encode())
734 fp.flush()
735 fp.close()
736 except:
737 return False
738 return True
739 def fsetVal(self, val, path):
740 if not self.useftrace:
741 return False
742 return self.setVal(val, self.tpath+path)
743 def getVal(self, file):
744 res = ''
745 if not os.path.exists(file):
746 return res
747 try:
748 fp = open(file, 'r')
749 res = fp.read()
750 fp.close()
751 except:
752 pass
753 return res
754 def fgetVal(self, path):
755 if not self.useftrace:
756 return ''
757 return self.getVal(self.tpath+path)
758 def cleanupFtrace(self):
759 if self.useftrace:
760 self.fsetVal('0', 'events/kprobes/enable')
761 self.fsetVal('', 'kprobe_events')
762 self.fsetVal('1024', 'buffer_size_kb')
763 def setupAllKprobes(self):
764 for name in self.tracefuncs:
765 self.defaultKprobe(name, self.tracefuncs[name])
766 for name in self.dev_tracefuncs:
767 self.defaultKprobe(name, self.dev_tracefuncs[name])
768 def isCallgraphFunc(self, name):
769 if len(self.tracefuncs) < 1 and self.suspendmode == 'command':
770 return True
771 for i in self.tracefuncs:
772 if 'func' in self.tracefuncs[i]:
773 f = self.tracefuncs[i]['func']
774 else:
775 f = i
776 if name == f:
777 return True
778 return False
779 def initFtrace(self, quiet=False):
780 if not self.useftrace:
781 return
782 if not quiet:
783 sysvals.printSystemInfo(False)
784 pprint('INITIALIZING FTRACE')
785 # turn trace off
786 self.fsetVal('0', 'tracing_on')
787 self.cleanupFtrace()
788 # set the trace clock to global
789 self.fsetVal('global', 'trace_clock')
790 self.fsetVal('nop', 'current_tracer')
791 # set trace buffer to an appropriate value
792 cpus = max(1, self.cpucount)
793 if self.bufsize > 0:
794 tgtsize = self.bufsize
795 elif self.usecallgraph or self.usedevsrc:
796 bmax = (1*1024*1024) if self.suspendmode in ['disk', 'command'] \
797 else (3*1024*1024)
798 tgtsize = min(self.memfree, bmax)
799 else:
800 tgtsize = 65536
801 while not self.fsetVal('%d' % (tgtsize // cpus), 'buffer_size_kb'):
802 # if the size failed to set, lower it and keep trying
803 tgtsize -= 65536
804 if tgtsize < 65536:
805 tgtsize = int(self.fgetVal('buffer_size_kb')) * cpus
806 break
807 self.vprint('Setting trace buffers to %d kB (%d kB per cpu)' % (tgtsize, tgtsize/cpus))
808 # initialize the callgraph trace
809 if(self.usecallgraph):
810 # set trace type
811 self.fsetVal('function_graph', 'current_tracer')
812 self.fsetVal('', 'set_ftrace_filter')
813 # temporary hack to fix https://bugzilla.kernel.org/show_bug.cgi?id=212761
814 fp = open(self.tpath+'set_ftrace_notrace', 'w')
815 fp.write('native_queued_spin_lock_slowpath\ndev_driver_string')
816 fp.close()
817 # set trace format options
818 self.fsetVal('print-parent', 'trace_options')
819 self.fsetVal('funcgraph-abstime', 'trace_options')
820 self.fsetVal('funcgraph-cpu', 'trace_options')
821 self.fsetVal('funcgraph-duration', 'trace_options')
822 self.fsetVal('funcgraph-proc', 'trace_options')
823 self.fsetVal('funcgraph-tail', 'trace_options')
824 self.fsetVal('nofuncgraph-overhead', 'trace_options')
825 self.fsetVal('context-info', 'trace_options')
826 self.fsetVal('graph-time', 'trace_options')
827 self.fsetVal('%d' % self.max_graph_depth, 'max_graph_depth')
828 cf = ['dpm_run_callback']
829 if(self.usetraceevents):
830 cf += ['dpm_prepare', 'dpm_complete']
831 for fn in self.tracefuncs:
832 if 'func' in self.tracefuncs[fn]:
833 cf.append(self.tracefuncs[fn]['func'])
834 else:
835 cf.append(fn)
836 if self.ftop:
837 self.setFtraceFilterFunctions([self.ftopfunc])
838 else:
839 self.setFtraceFilterFunctions(cf)
840 # initialize the kprobe trace
841 elif self.usekprobes:
842 for name in self.tracefuncs:
843 self.defaultKprobe(name, self.tracefuncs[name])
844 if self.usedevsrc:
845 for name in self.dev_tracefuncs:
846 self.defaultKprobe(name, self.dev_tracefuncs[name])
847 if not quiet:
848 pprint('INITIALIZING KPROBES')
849 self.addKprobes(self.verbose)
850 if(self.usetraceevents):
851 # turn trace events on
852 events = iter(self.traceevents)
853 for e in events:
854 self.fsetVal('1', 'events/power/'+e+'/enable')
855 # clear the trace buffer
856 self.fsetVal('', 'trace')
857 def verifyFtrace(self):
858 # files needed for any trace data
859 files = ['buffer_size_kb', 'current_tracer', 'trace', 'trace_clock',
860 'trace_marker', 'trace_options', 'tracing_on']
861 # files needed for callgraph trace data
862 tp = self.tpath
863 if(self.usecallgraph):
864 files += [
865 'available_filter_functions',
866 'set_ftrace_filter',
867 'set_graph_function'
868 ]
869 for f in files:
870 if(os.path.exists(tp+f) == False):
871 return False
872 return True
873 def verifyKprobes(self):
874 # files needed for kprobes to work
875 files = ['kprobe_events', 'events']
876 tp = self.tpath
877 for f in files:
878 if(os.path.exists(tp+f) == False):
879 return False
880 return True
881 def colorText(self, str, color=31):
882 if not self.ansi:
883 return str
884 return '\x1B[%d;40m%s\x1B[m' % (color, str)
885 def writeDatafileHeader(self, filename, testdata):
886 fp = self.openlog(filename, 'w')
887 fp.write('%s\n%s\n# command | %s\n' % (self.teststamp, self.sysstamp, self.cmdline))
888 for test in testdata:
889 if 'fw' in test:
890 fw = test['fw']
891 if(fw):
892 fp.write('# fwsuspend %u fwresume %u\n' % (fw[0], fw[1]))
893 if 'turbo' in test:
894 fp.write('# turbostat %s\n' % test['turbo'])
895 if 'wifi' in test:
896 fp.write('# wifi %s\n' % test['wifi'])
897 if 'netfix' in test:
898 fp.write('# netfix %s\n' % test['netfix'])
899 if test['error'] or len(testdata) > 1:
900 fp.write('# enter_sleep_error %s\n' % test['error'])
901 return fp
902 def sudoUserchown(self, dir):
903 if os.path.exists(dir) and self.sudouser:
904 cmd = 'chown -R {0}:{0} {1} > /dev/null 2>&1'
905 call(cmd.format(self.sudouser, dir), shell=True)
906 def outputResult(self, testdata, num=0):
907 if not self.result:
908 return
909 n = ''
910 if num > 0:
911 n = '%d' % num
912 fp = open(self.result, 'a')
913 if 'error' in testdata:
914 fp.write('result%s: fail\n' % n)
915 fp.write('error%s: %s\n' % (n, testdata['error']))
916 else:
917 fp.write('result%s: pass\n' % n)
918 if 'mode' in testdata:
919 fp.write('mode%s: %s\n' % (n, testdata['mode']))
920 for v in ['suspend', 'resume', 'boot', 'lastinit']:
921 if v in testdata:
922 fp.write('%s%s: %.3f\n' % (v, n, testdata[v]))
923 for v in ['fwsuspend', 'fwresume']:
924 if v in testdata:
925 fp.write('%s%s: %.3f\n' % (v, n, testdata[v] / 1000000.0))
926 if 'bugurl' in testdata:
927 fp.write('url%s: %s\n' % (n, testdata['bugurl']))
928 fp.close()
929 self.sudoUserchown(self.result)
930 def configFile(self, file):
931 dir = os.path.dirname(os.path.realpath(__file__))
932 if os.path.exists(file):
933 return file
934 elif os.path.exists(dir+'/'+file):
935 return dir+'/'+file
936 elif os.path.exists(dir+'/config/'+file):
937 return dir+'/config/'+file
938 return ''
939 def openlog(self, filename, mode):
940 isgz = self.gzip
941 if mode == 'r':
942 try:
943 with gzip.open(filename, mode+'t') as fp:
944 test = fp.read(64)
945 isgz = True
946 except:
947 isgz = False
948 if isgz:
949 return gzip.open(filename, mode+'t')
950 return open(filename, mode)
951 def putlog(self, filename, text):
952 with self.openlog(filename, 'a') as fp:
953 fp.write(text)
954 fp.close()
955 def dlog(self, text):
956 if not self.dmesgfile:
957 return
958 self.putlog(self.dmesgfile, '# %s\n' % text)
959 def flog(self, text):
960 self.putlog(self.ftracefile, text)
961 def b64unzip(self, data):
962 try:
963 out = codecs.decode(base64.b64decode(data), 'zlib').decode()
964 except:
965 out = data
966 return out
967 def b64zip(self, data):
968 out = base64.b64encode(codecs.encode(data.encode(), 'zlib')).decode()
969 return out
970 def platforminfo(self, cmdafter):
971 # add platform info on to a completed ftrace file
972 if not os.path.exists(self.ftracefile):
973 return False
974 footer = '#\n'
975
976 # add test command string line if need be
977 if self.suspendmode == 'command' and self.testcommand:
978 footer += '# platform-testcmd: %s\n' % (self.testcommand)
979
980 # get a list of target devices from the ftrace file
981 props = dict()
982 tp = TestProps()
983 tf = self.openlog(self.ftracefile, 'r')
984 for line in tf:
985 if tp.stampInfo(line, self):
986 continue
987 # parse only valid lines, if this is not one move on
988 m = re.match(tp.ftrace_line_fmt, line)
989 if(not m or 'device_pm_callback_start' not in line):
990 continue
991 m = re.match('.*: (?P<drv>.*) (?P<d>.*), parent: *(?P<p>.*), .*', m.group('msg'));
992 if(not m):
993 continue
994 dev = m.group('d')
995 if dev not in props:
996 props[dev] = DevProps()
997 tf.close()
998
999 # now get the syspath for each target device
1000 for dirname, dirnames, filenames in os.walk('/sys/devices'):
1001 if(re.match('.*/power', dirname) and 'async' in filenames):
1002 dev = dirname.split('/')[-2]
1003 if dev in props and (not props[dev].syspath or len(dirname) < len(props[dev].syspath)):
1004 props[dev].syspath = dirname[:-6]
1005
1006 # now fill in the properties for our target devices
1007 for dev in sorted(props):
1008 dirname = props[dev].syspath
1009 if not dirname or not os.path.exists(dirname):
1010 continue
1011 props[dev].isasync = False
1012 if os.path.exists(dirname+'/power/async'):
1013 fp = open(dirname+'/power/async')
1014 if 'enabled' in fp.read():
1015 props[dev].isasync = True
1016 fp.close()
1017 fields = os.listdir(dirname)
1018 for file in ['product', 'name', 'model', 'description', 'id', 'idVendor']:
1019 if file not in fields:
1020 continue
1021 try:
1022 with open(os.path.join(dirname, file), 'rb') as fp:
1023 props[dev].altname = ascii(fp.read())
1024 except:
1025 continue
1026 if file == 'idVendor':
1027 idv, idp = props[dev].altname.strip(), ''
1028 try:
1029 with open(os.path.join(dirname, 'idProduct'), 'rb') as fp:
1030 idp = ascii(fp.read()).strip()
1031 except:
1032 props[dev].altname = ''
1033 break
1034 props[dev].altname = '%s:%s' % (idv, idp)
1035 break
1036 if props[dev].altname:
1037 out = props[dev].altname.strip().replace('\n', ' ')\
1038 .replace(',', ' ').replace(';', ' ')
1039 props[dev].altname = out
1040
1041 # add a devinfo line to the bottom of ftrace
1042 out = ''
1043 for dev in sorted(props):
1044 out += props[dev].out(dev)
1045 footer += '# platform-devinfo: %s\n' % self.b64zip(out)
1046
1047 # add a line for each of these commands with their outputs
1048 for name, cmdline, info in cmdafter:
1049 footer += '# platform-%s: %s | %s\n' % (name, cmdline, self.b64zip(info))
1050 self.flog(footer)
1051 return True
1052 def commonPrefix(self, list):
1053 if len(list) < 2:
1054 return ''
1055 prefix = list[0]
1056 for s in list[1:]:
1057 while s[:len(prefix)] != prefix and prefix:
1058 prefix = prefix[:len(prefix)-1]
1059 if not prefix:
1060 break
1061 if '/' in prefix and prefix[-1] != '/':
1062 prefix = prefix[0:prefix.rfind('/')+1]
1063 return prefix
1064 def dictify(self, text, format):
1065 out = dict()
1066 header = True if format == 1 else False
1067 delim = ' ' if format == 1 else ':'
1068 for line in text.split('\n'):
1069 if header:
1070 header, out['@'] = False, line
1071 continue
1072 line = line.strip()
1073 if delim in line:
1074 data = line.split(delim, 1)
1075 num = re.search(r'[\d]+', data[1])
1076 if format == 2 and num:
1077 out[data[0].strip()] = num.group()
1078 else:
1079 out[data[0].strip()] = data[1]
1080 return out
1081 def cmdinfo(self, begin, debug=False):
1082 out = []
1083 if begin:
1084 self.cmd1 = dict()
1085 for cargs in self.infocmds:
1086 delta, name = cargs[0], cargs[1]
1087 cmdline, cmdpath = ' '.join(cargs[2:]), self.getExec(cargs[2])
1088 if not cmdpath or (begin and not delta):
1089 continue
1090 self.dlog('[%s]' % cmdline)
1091 try:
1092 fp = Popen([cmdpath]+cargs[3:], stdout=PIPE, stderr=PIPE).stdout
1093 info = ascii(fp.read()).strip()
1094 fp.close()
1095 except:
1096 continue
1097 if not debug and begin:
1098 self.cmd1[name] = self.dictify(info, delta)
1099 elif not debug and delta and name in self.cmd1:
1100 before, after = self.cmd1[name], self.dictify(info, delta)
1101 dinfo = ('\t%s\n' % before['@']) if '@' in before and len(before) > 1 else ''
1102 prefix = self.commonPrefix(list(before.keys()))
1103 for key in sorted(before):
1104 if key in after and before[key] != after[key]:
1105 title = key.replace(prefix, '')
1106 if delta == 2:
1107 dinfo += '\t%s : %s -> %s\n' % \
1108 (title, before[key].strip(), after[key].strip())
1109 else:
1110 dinfo += '%10s (start) : %s\n%10s (after) : %s\n' % \
1111 (title, before[key], title, after[key])
1112 dinfo = '\tnothing changed' if not dinfo else dinfo.rstrip()
1113 out.append((name, cmdline, dinfo))
1114 else:
1115 out.append((name, cmdline, '\tnothing' if not info else info))
1116 return out
1117 def testVal(self, file, fmt='basic', value=''):
1118 if file == 'restoreall':
1119 for f in self.cfgdef:
1120 if os.path.exists(f):
1121 fp = open(f, 'w')
1122 fp.write(self.cfgdef[f])
1123 fp.close()
1124 self.cfgdef = dict()
1125 elif value and os.path.exists(file):
1126 fp = open(file, 'r+')
1127 if fmt == 'radio':
1128 m = re.match('.*\[(?P<v>.*)\].*', fp.read())
1129 if m:
1130 self.cfgdef[file] = m.group('v')
1131 elif fmt == 'acpi':
1132 line = fp.read().strip().split('\n')[-1]
1133 m = re.match('.* (?P<v>[0-9A-Fx]*) .*', line)
1134 if m:
1135 self.cfgdef[file] = m.group('v')
1136 else:
1137 self.cfgdef[file] = fp.read().strip()
1138 fp.write(value)
1139 fp.close()
1140 def s0ixSupport(self):
1141 if not os.path.exists(self.s0ixres) or not os.path.exists(self.mempowerfile):
1142 return False
1143 fp = open(sysvals.mempowerfile, 'r')
1144 data = fp.read().strip()
1145 fp.close()
1146 if '[s2idle]' in data:
1147 return True
1148 return False
1149 def haveTurbostat(self):
1150 if not self.tstat:
1151 return False
1152 cmd = self.getExec('turbostat')
1153 if not cmd:
1154 return False
1155 fp = Popen([cmd, '-v'], stdout=PIPE, stderr=PIPE).stderr
1156 out = ascii(fp.read()).strip()
1157 fp.close()
1158 if re.match('turbostat version .*', out):
1159 self.vprint(out)
1160 return True
1161 return False
1162 def turbostat(self, s0ixready):
1163 cmd = self.getExec('turbostat')
1164 rawout = keyline = valline = ''
1165 fullcmd = '%s -q -S echo freeze > %s' % (cmd, self.powerfile)
1166 fp = Popen(['sh', '-c', fullcmd], stdout=PIPE, stderr=PIPE).stderr
1167 for line in fp:
1168 line = ascii(line)
1169 rawout += line
1170 if keyline and valline:
1171 continue
1172 if re.match('(?i)Avg_MHz.*', line):
1173 keyline = line.strip().split()
1174 elif keyline:
1175 valline = line.strip().split()
1176 fp.close()
1177 if not keyline or not valline or len(keyline) != len(valline):
1178 errmsg = 'unrecognized turbostat output:\n'+rawout.strip()
1179 self.vprint(errmsg)
1180 if not self.verbose:
1181 pprint(errmsg)
1182 return ''
1183 if self.verbose:
1184 pprint(rawout.strip())
1185 out = []
1186 for key in keyline:
1187 idx = keyline.index(key)
1188 val = valline[idx]
1189 if key == 'SYS%LPI' and not s0ixready and re.match('^[0\.]*$', val):
1190 continue
1191 out.append('%s=%s' % (key, val))
1192 return '|'.join(out)
1193 def netfixon(self, net='both'):
1194 cmd = self.getExec('netfix')
1195 if not cmd:
1196 return ''
1197 fp = Popen([cmd, '-s', net, 'on'], stdout=PIPE, stderr=PIPE).stdout
1198 out = ascii(fp.read()).strip()
1199 fp.close()
1200 return out
1201 def wifiDetails(self, dev):
1202 try:
1203 info = open('/sys/class/net/%s/device/uevent' % dev, 'r').read().strip()
1204 except:
1205 return dev
1206 vals = [dev]
1207 for prop in info.split('\n'):
1208 if prop.startswith('DRIVER=') or prop.startswith('PCI_ID='):
1209 vals.append(prop.split('=')[-1])
1210 return ':'.join(vals)
1211 def checkWifi(self, dev=''):
1212 try:
1213 w = open('/proc/net/wireless', 'r').read().strip()
1214 except:
1215 return ''
1216 for line in reversed(w.split('\n')):
1217 m = re.match(' *(?P<dev>.*): (?P<stat>[0-9a-f]*) .*', line)
1218 if not m or (dev and dev != m.group('dev')):
1219 continue
1220 return m.group('dev')
1221 return ''
1222 def pollWifi(self, dev, timeout=10):
1223 start = time.time()
1224 while (time.time() - start) < timeout:
1225 w = self.checkWifi(dev)
1226 if w:
1227 return '%s reconnected %.2f' % \
1228 (self.wifiDetails(dev), max(0, time.time() - start))
1229 time.sleep(0.01)
1230 return '%s timeout %d' % (self.wifiDetails(dev), timeout)
1231 def errorSummary(self, errinfo, msg):
1232 found = False
1233 for entry in errinfo:
1234 if re.match(entry['match'], msg):
1235 entry['count'] += 1
1236 if self.hostname not in entry['urls']:
1237 entry['urls'][self.hostname] = [self.htmlfile]
1238 elif self.htmlfile not in entry['urls'][self.hostname]:
1239 entry['urls'][self.hostname].append(self.htmlfile)
1240 found = True
1241 break
1242 if found:
1243 return
1244 arr = msg.split()
1245 for j in range(len(arr)):
1246 if re.match('^[0-9,\-\.]*$', arr[j]):
1247 arr[j] = '[0-9,\-\.]*'
1248 else:
1249 arr[j] = arr[j]\
1250 .replace('\\', '\\\\').replace(']', '\]').replace('[', '\[')\
1251 .replace('.', '\.').replace('+', '\+').replace('*', '\*')\
1252 .replace('(', '\(').replace(')', '\)').replace('}', '\}')\
1253 .replace('{', '\{')
1254 mstr = ' *'.join(arr)
1255 entry = {
1256 'line': msg,
1257 'match': mstr,
1258 'count': 1,
1259 'urls': {self.hostname: [self.htmlfile]}
1260 }
1261 errinfo.append(entry)
1262 def multistat(self, start, idx, finish):
1263 if 'time' in self.multitest:
1264 id = '%d Duration=%dmin' % (idx+1, self.multitest['time'])
1265 else:
1266 id = '%d/%d' % (idx+1, self.multitest['count'])
1267 t = time.time()
1268 if 'start' not in self.multitest:
1269 self.multitest['start'] = self.multitest['last'] = t
1270 self.multitest['total'] = 0.0
1271 pprint('TEST (%s) START' % id)
1272 return
1273 dt = t - self.multitest['last']
1274 if not start:
1275 if idx == 0 and self.multitest['delay'] > 0:
1276 self.multitest['total'] += self.multitest['delay']
1277 pprint('TEST (%s) COMPLETE -- Duration %.1fs' % (id, dt))
1278 return
1279 self.multitest['total'] += dt
1280 self.multitest['last'] = t
1281 avg = self.multitest['total'] / idx
1282 if 'time' in self.multitest:
1283 left = finish - datetime.now()
1284 left -= timedelta(microseconds=left.microseconds)
1285 else:
1286 left = timedelta(seconds=((self.multitest['count'] - idx) * int(avg)))
1287 pprint('TEST (%s) START - Avg Duration %.1fs, Time left %s' % \
1288 (id, avg, str(left)))
1289 def multiinit(self, c, d):
1290 sz, unit = 'count', 'm'
1291 if c.endswith('d') or c.endswith('h') or c.endswith('m'):
1292 sz, unit, c = 'time', c[-1], c[:-1]
1293 self.multitest['run'] = True
1294 self.multitest[sz] = getArgInt('multi: n d (exec count)', c, 1, 1000000, False)
1295 self.multitest['delay'] = getArgInt('multi: n d (delay between tests)', d, 0, 3600, False)
1296 if unit == 'd':
1297 self.multitest[sz] *= 1440
1298 elif unit == 'h':
1299 self.multitest[sz] *= 60
1300 def displayControl(self, cmd):
1301 xset, ret = 'timeout 10 xset -d :0.0 {0}', 0
1302 if self.sudouser:
1303 xset = 'sudo -u %s %s' % (self.sudouser, xset)
1304 if cmd == 'init':
1305 ret = call(xset.format('dpms 0 0 0'), shell=True)
1306 if not ret:
1307 ret = call(xset.format('s off'), shell=True)
1308 elif cmd == 'reset':
1309 ret = call(xset.format('s reset'), shell=True)
1310 elif cmd in ['on', 'off', 'standby', 'suspend']:
1311 b4 = self.displayControl('stat')
1312 ret = call(xset.format('dpms force %s' % cmd), shell=True)
1313 if not ret:
1314 curr = self.displayControl('stat')
1315 self.vprint('Display Switched: %s -> %s' % (b4, curr))
1316 if curr != cmd:
1317 self.vprint('WARNING: Display failed to change to %s' % cmd)
1318 if ret:
1319 self.vprint('WARNING: Display failed to change to %s with xset' % cmd)
1320 return ret
1321 elif cmd == 'stat':
1322 fp = Popen(xset.format('q').split(' '), stdout=PIPE).stdout
1323 ret = 'unknown'
1324 for line in fp:
1325 m = re.match('[\s]*Monitor is (?P<m>.*)', ascii(line))
1326 if(m and len(m.group('m')) >= 2):
1327 out = m.group('m').lower()
1328 ret = out[3:] if out[0:2] == 'in' else out
1329 break
1330 fp.close()
1331 return ret
1332 def setRuntimeSuspend(self, before=True):
1333 if before:
1334 # runtime suspend disable or enable
1335 if self.rs > 0:
1336 self.rstgt, self.rsval, self.rsdir = 'on', 'auto', 'enabled'
1337 else:
1338 self.rstgt, self.rsval, self.rsdir = 'auto', 'on', 'disabled'
1339 pprint('CONFIGURING RUNTIME SUSPEND...')
1340 self.rslist = deviceInfo(self.rstgt)
1341 for i in self.rslist:
1342 self.setVal(self.rsval, i)
1343 pprint('runtime suspend %s on all devices (%d changed)' % (self.rsdir, len(self.rslist)))
1344 pprint('waiting 5 seconds...')
1345 time.sleep(5)
1346 else:
1347 # runtime suspend re-enable or re-disable
1348 for i in self.rslist:
1349 self.setVal(self.rstgt, i)
1350 pprint('runtime suspend settings restored on %d devices' % len(self.rslist))
1351 def start(self, pm):
1352 if self.useftrace:
1353 self.dlog('start ftrace tracing')
1354 self.fsetVal('1', 'tracing_on')
1355 if self.useprocmon:
1356 self.dlog('start the process monitor')
1357 pm.start()
1358 def stop(self, pm):
1359 if self.useftrace:
1360 if self.useprocmon:
1361 self.dlog('stop the process monitor')
1362 pm.stop()
1363 self.dlog('stop ftrace tracing')
1364 self.fsetVal('0', 'tracing_on')
1365
1366sysvals = SystemValues()
1367switchvalues = ['enable', 'disable', 'on', 'off', 'true', 'false', '1', '0']
1368switchoff = ['disable', 'off', 'false', '0']
1369suspendmodename = {
1370 'standby': 'standby (S1)',
1371 'freeze': 'freeze (S2idle)',
1372 'mem': 'suspend (S3)',
1373 'disk': 'hibernate (S4)'
1374}
1375
1376# Class: DevProps
1377# Description:
1378# Simple class which holds property values collected
1379# for all the devices used in the timeline.
1380class DevProps:
1381 def __init__(self):
1382 self.syspath = ''
1383 self.altname = ''
1384 self.isasync = True
1385 self.xtraclass = ''
1386 self.xtrainfo = ''
1387 def out(self, dev):
1388 return '%s,%s,%d;' % (dev, self.altname, self.isasync)
1389 def debug(self, dev):
1390 pprint('%s:\n\taltname = %s\n\t async = %s' % (dev, self.altname, self.isasync))
1391 def altName(self, dev):
1392 if not self.altname or self.altname == dev:
1393 return dev
1394 return '%s [%s]' % (self.altname, dev)
1395 def xtraClass(self):
1396 if self.xtraclass:
1397 return ' '+self.xtraclass
1398 if not self.isasync:
1399 return ' sync'
1400 return ''
1401 def xtraInfo(self):
1402 if self.xtraclass:
1403 return ' '+self.xtraclass
1404 if self.isasync:
1405 return ' (async)'
1406 return ' (sync)'
1407
1408# Class: DeviceNode
1409# Description:
1410# A container used to create a device hierachy, with a single root node
1411# and a tree of child nodes. Used by Data.deviceTopology()
1412class DeviceNode:
1413 def __init__(self, nodename, nodedepth):
1414 self.name = nodename
1415 self.children = []
1416 self.depth = nodedepth
1417
1418# Class: Data
1419# Description:
1420# The primary container for suspend/resume test data. There is one for
1421# each test run. The data is organized into a cronological hierarchy:
1422# Data.dmesg {
1423# phases {
1424# 10 sequential, non-overlapping phases of S/R
1425# contents: times for phase start/end, order/color data for html
1426# devlist {
1427# device callback or action list for this phase
1428# device {
1429# a single device callback or generic action
1430# contents: start/stop times, pid/cpu/driver info
1431# parents/children, html id for timeline/callgraph
1432# optionally includes an ftrace callgraph
1433# optionally includes dev/ps data
1434# }
1435# }
1436# }
1437# }
1438#
1439class Data:
1440 phasedef = {
1441 'suspend_prepare': {'order': 0, 'color': '#CCFFCC'},
1442 'suspend': {'order': 1, 'color': '#88FF88'},
1443 'suspend_late': {'order': 2, 'color': '#00AA00'},
1444 'suspend_noirq': {'order': 3, 'color': '#008888'},
1445 'suspend_machine': {'order': 4, 'color': '#0000FF'},
1446 'resume_machine': {'order': 5, 'color': '#FF0000'},
1447 'resume_noirq': {'order': 6, 'color': '#FF9900'},
1448 'resume_early': {'order': 7, 'color': '#FFCC00'},
1449 'resume': {'order': 8, 'color': '#FFFF88'},
1450 'resume_complete': {'order': 9, 'color': '#FFFFCC'},
1451 }
1452 errlist = {
1453 'HWERROR' : r'.*\[ *Hardware Error *\].*',
1454 'FWBUG' : r'.*\[ *Firmware Bug *\].*',
1455 'BUG' : r'(?i).*\bBUG\b.*',
1456 'ERROR' : r'(?i).*\bERROR\b.*',
1457 'WARNING' : r'(?i).*\bWARNING\b.*',
1458 'FAULT' : r'(?i).*\bFAULT\b.*',
1459 'FAIL' : r'(?i).*\bFAILED\b.*',
1460 'INVALID' : r'(?i).*\bINVALID\b.*',
1461 'CRASH' : r'(?i).*\bCRASHED\b.*',
1462 'TIMEOUT' : r'(?i).*\bTIMEOUT\b.*',
1463 'ABORT' : r'(?i).*\bABORT\b.*',
1464 'IRQ' : r'.*\bgenirq: .*',
1465 'TASKFAIL': r'.*Freezing .*after *.*',
1466 'ACPI' : r'.*\bACPI *(?P<b>[A-Za-z]*) *Error[: ].*',
1467 'DISKFULL': r'.*\bNo space left on device.*',
1468 'USBERR' : r'.*usb .*device .*, error [0-9-]*',
1469 'ATAERR' : r' *ata[0-9\.]*: .*failed.*',
1470 'MEIERR' : r' *mei.*: .*failed.*',
1471 'TPMERR' : r'(?i) *tpm *tpm[0-9]*: .*error.*',
1472 }
1473 def __init__(self, num):
1474 idchar = 'abcdefghij'
1475 self.start = 0.0 # test start
1476 self.end = 0.0 # test end
1477 self.hwstart = 0 # rtc test start
1478 self.hwend = 0 # rtc test end
1479 self.tSuspended = 0.0 # low-level suspend start
1480 self.tResumed = 0.0 # low-level resume start
1481 self.tKernSus = 0.0 # kernel level suspend start
1482 self.tKernRes = 0.0 # kernel level resume end
1483 self.fwValid = False # is firmware data available
1484 self.fwSuspend = 0 # time spent in firmware suspend
1485 self.fwResume = 0 # time spent in firmware resume
1486 self.html_device_id = 0
1487 self.stamp = 0
1488 self.outfile = ''
1489 self.kerror = False
1490 self.wifi = dict()
1491 self.turbostat = 0
1492 self.enterfail = ''
1493 self.currphase = ''
1494 self.pstl = dict() # process timeline
1495 self.testnumber = num
1496 self.idstr = idchar[num]
1497 self.dmesgtext = [] # dmesg text file in memory
1498 self.dmesg = dict() # root data structure
1499 self.errorinfo = {'suspend':[],'resume':[]}
1500 self.tLow = [] # time spent in low-level suspends (standby/freeze)
1501 self.devpids = []
1502 self.devicegroups = 0
1503 def sortedPhases(self):
1504 return sorted(self.dmesg, key=lambda k:self.dmesg[k]['order'])
1505 def initDevicegroups(self):
1506 # called when phases are all finished being added
1507 for phase in sorted(self.dmesg.keys()):
1508 if '*' in phase:
1509 p = phase.split('*')
1510 pnew = '%s%d' % (p[0], len(p))
1511 self.dmesg[pnew] = self.dmesg.pop(phase)
1512 self.devicegroups = []
1513 for phase in self.sortedPhases():
1514 self.devicegroups.append([phase])
1515 def nextPhase(self, phase, offset):
1516 order = self.dmesg[phase]['order'] + offset
1517 for p in self.dmesg:
1518 if self.dmesg[p]['order'] == order:
1519 return p
1520 return ''
1521 def lastPhase(self, depth=1):
1522 plist = self.sortedPhases()
1523 if len(plist) < depth:
1524 return ''
1525 return plist[-1*depth]
1526 def turbostatInfo(self):
1527 tp = TestProps()
1528 out = {'syslpi':'N/A','pkgpc10':'N/A'}
1529 for line in self.dmesgtext:
1530 m = re.match(tp.tstatfmt, line)
1531 if not m:
1532 continue
1533 for i in m.group('t').split('|'):
1534 if 'SYS%LPI' in i:
1535 out['syslpi'] = i.split('=')[-1]+'%'
1536 elif 'pc10' in i:
1537 out['pkgpc10'] = i.split('=')[-1]+'%'
1538 break
1539 return out
1540 def extractErrorInfo(self):
1541 lf = self.dmesgtext
1542 if len(self.dmesgtext) < 1 and sysvals.dmesgfile:
1543 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
1544 i = 0
1545 tp = TestProps()
1546 list = []
1547 for line in lf:
1548 i += 1
1549 if tp.stampInfo(line, sysvals):
1550 continue
1551 m = re.match('[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
1552 if not m:
1553 continue
1554 t = float(m.group('ktime'))
1555 if t < self.start or t > self.end:
1556 continue
1557 dir = 'suspend' if t < self.tSuspended else 'resume'
1558 msg = m.group('msg')
1559 if re.match('capability: warning: .*', msg):
1560 continue
1561 for err in self.errlist:
1562 if re.match(self.errlist[err], msg):
1563 list.append((msg, err, dir, t, i, i))
1564 self.kerror = True
1565 break
1566 tp.msglist = []
1567 for msg, type, dir, t, idx1, idx2 in list:
1568 tp.msglist.append(msg)
1569 self.errorinfo[dir].append((type, t, idx1, idx2))
1570 if self.kerror:
1571 sysvals.dmesglog = True
1572 if len(self.dmesgtext) < 1 and sysvals.dmesgfile:
1573 lf.close()
1574 return tp
1575 def setStart(self, time, msg=''):
1576 self.start = time
1577 if msg:
1578 try:
1579 self.hwstart = datetime.strptime(msg, sysvals.tmstart)
1580 except:
1581 self.hwstart = 0
1582 def setEnd(self, time, msg=''):
1583 self.end = time
1584 if msg:
1585 try:
1586 self.hwend = datetime.strptime(msg, sysvals.tmend)
1587 except:
1588 self.hwend = 0
1589 def isTraceEventOutsideDeviceCalls(self, pid, time):
1590 for phase in self.sortedPhases():
1591 list = self.dmesg[phase]['list']
1592 for dev in list:
1593 d = list[dev]
1594 if(d['pid'] == pid and time >= d['start'] and
1595 time < d['end']):
1596 return False
1597 return True
1598 def sourcePhase(self, start):
1599 for phase in self.sortedPhases():
1600 if 'machine' in phase:
1601 continue
1602 pend = self.dmesg[phase]['end']
1603 if start <= pend:
1604 return phase
1605 return 'resume_complete'
1606 def sourceDevice(self, phaselist, start, end, pid, type):
1607 tgtdev = ''
1608 for phase in phaselist:
1609 list = self.dmesg[phase]['list']
1610 for devname in list:
1611 dev = list[devname]
1612 # pid must match
1613 if dev['pid'] != pid:
1614 continue
1615 devS = dev['start']
1616 devE = dev['end']
1617 if type == 'device':
1618 # device target event is entirely inside the source boundary
1619 if(start < devS or start >= devE or end <= devS or end > devE):
1620 continue
1621 elif type == 'thread':
1622 # thread target event will expand the source boundary
1623 if start < devS:
1624 dev['start'] = start
1625 if end > devE:
1626 dev['end'] = end
1627 tgtdev = dev
1628 break
1629 return tgtdev
1630 def addDeviceFunctionCall(self, displayname, kprobename, proc, pid, start, end, cdata, rdata):
1631 # try to place the call in a device
1632 phases = self.sortedPhases()
1633 tgtdev = self.sourceDevice(phases, start, end, pid, 'device')
1634 # calls with device pids that occur outside device bounds are dropped
1635 # TODO: include these somehow
1636 if not tgtdev and pid in self.devpids:
1637 return False
1638 # try to place the call in a thread
1639 if not tgtdev:
1640 tgtdev = self.sourceDevice(phases, start, end, pid, 'thread')
1641 # create new thread blocks, expand as new calls are found
1642 if not tgtdev:
1643 if proc == '<...>':
1644 threadname = 'kthread-%d' % (pid)
1645 else:
1646 threadname = '%s-%d' % (proc, pid)
1647 tgtphase = self.sourcePhase(start)
1648 self.newAction(tgtphase, threadname, pid, '', start, end, '', ' kth', '')
1649 return self.addDeviceFunctionCall(displayname, kprobename, proc, pid, start, end, cdata, rdata)
1650 # this should not happen
1651 if not tgtdev:
1652 sysvals.vprint('[%f - %f] %s-%d %s %s %s' % \
1653 (start, end, proc, pid, kprobename, cdata, rdata))
1654 return False
1655 # place the call data inside the src element of the tgtdev
1656 if('src' not in tgtdev):
1657 tgtdev['src'] = []
1658 dtf = sysvals.dev_tracefuncs
1659 ubiquitous = False
1660 if kprobename in dtf and 'ub' in dtf[kprobename]:
1661 ubiquitous = True
1662 mc = re.match('\(.*\) *(?P<args>.*)', cdata)
1663 mr = re.match('\((?P<caller>\S*).* arg1=(?P<ret>.*)', rdata)
1664 if mc and mr:
1665 c = mr.group('caller').split('+')[0]
1666 a = mc.group('args').strip()
1667 r = mr.group('ret')
1668 if len(r) > 6:
1669 r = ''
1670 else:
1671 r = 'ret=%s ' % r
1672 if ubiquitous and c in dtf and 'ub' in dtf[c]:
1673 return False
1674 else:
1675 return False
1676 color = sysvals.kprobeColor(kprobename)
1677 e = DevFunction(displayname, a, c, r, start, end, ubiquitous, proc, pid, color)
1678 tgtdev['src'].append(e)
1679 return True
1680 def overflowDevices(self):
1681 # get a list of devices that extend beyond the end of this test run
1682 devlist = []
1683 for phase in self.sortedPhases():
1684 list = self.dmesg[phase]['list']
1685 for devname in list:
1686 dev = list[devname]
1687 if dev['end'] > self.end:
1688 devlist.append(dev)
1689 return devlist
1690 def mergeOverlapDevices(self, devlist):
1691 # merge any devices that overlap devlist
1692 for dev in devlist:
1693 devname = dev['name']
1694 for phase in self.sortedPhases():
1695 list = self.dmesg[phase]['list']
1696 if devname not in list:
1697 continue
1698 tdev = list[devname]
1699 o = min(dev['end'], tdev['end']) - max(dev['start'], tdev['start'])
1700 if o <= 0:
1701 continue
1702 dev['end'] = tdev['end']
1703 if 'src' not in dev or 'src' not in tdev:
1704 continue
1705 dev['src'] += tdev['src']
1706 del list[devname]
1707 def usurpTouchingThread(self, name, dev):
1708 # the caller test has priority of this thread, give it to him
1709 for phase in self.sortedPhases():
1710 list = self.dmesg[phase]['list']
1711 if name in list:
1712 tdev = list[name]
1713 if tdev['start'] - dev['end'] < 0.1:
1714 dev['end'] = tdev['end']
1715 if 'src' not in dev:
1716 dev['src'] = []
1717 if 'src' in tdev:
1718 dev['src'] += tdev['src']
1719 del list[name]
1720 break
1721 def stitchTouchingThreads(self, testlist):
1722 # merge any threads between tests that touch
1723 for phase in self.sortedPhases():
1724 list = self.dmesg[phase]['list']
1725 for devname in list:
1726 dev = list[devname]
1727 if 'htmlclass' not in dev or 'kth' not in dev['htmlclass']:
1728 continue
1729 for data in testlist:
1730 data.usurpTouchingThread(devname, dev)
1731 def optimizeDevSrc(self):
1732 # merge any src call loops to reduce timeline size
1733 for phase in self.sortedPhases():
1734 list = self.dmesg[phase]['list']
1735 for dev in list:
1736 if 'src' not in list[dev]:
1737 continue
1738 src = list[dev]['src']
1739 p = 0
1740 for e in sorted(src, key=lambda event: event.time):
1741 if not p or not e.repeat(p):
1742 p = e
1743 continue
1744 # e is another iteration of p, move it into p
1745 p.end = e.end
1746 p.length = p.end - p.time
1747 p.count += 1
1748 src.remove(e)
1749 def trimTimeVal(self, t, t0, dT, left):
1750 if left:
1751 if(t > t0):
1752 if(t - dT < t0):
1753 return t0
1754 return t - dT
1755 else:
1756 return t
1757 else:
1758 if(t < t0 + dT):
1759 if(t > t0):
1760 return t0 + dT
1761 return t + dT
1762 else:
1763 return t
1764 def trimTime(self, t0, dT, left):
1765 self.tSuspended = self.trimTimeVal(self.tSuspended, t0, dT, left)
1766 self.tResumed = self.trimTimeVal(self.tResumed, t0, dT, left)
1767 self.start = self.trimTimeVal(self.start, t0, dT, left)
1768 self.tKernSus = self.trimTimeVal(self.tKernSus, t0, dT, left)
1769 self.tKernRes = self.trimTimeVal(self.tKernRes, t0, dT, left)
1770 self.end = self.trimTimeVal(self.end, t0, dT, left)
1771 for phase in self.sortedPhases():
1772 p = self.dmesg[phase]
1773 p['start'] = self.trimTimeVal(p['start'], t0, dT, left)
1774 p['end'] = self.trimTimeVal(p['end'], t0, dT, left)
1775 list = p['list']
1776 for name in list:
1777 d = list[name]
1778 d['start'] = self.trimTimeVal(d['start'], t0, dT, left)
1779 d['end'] = self.trimTimeVal(d['end'], t0, dT, left)
1780 d['length'] = d['end'] - d['start']
1781 if('ftrace' in d):
1782 cg = d['ftrace']
1783 cg.start = self.trimTimeVal(cg.start, t0, dT, left)
1784 cg.end = self.trimTimeVal(cg.end, t0, dT, left)
1785 for line in cg.list:
1786 line.time = self.trimTimeVal(line.time, t0, dT, left)
1787 if('src' in d):
1788 for e in d['src']:
1789 e.time = self.trimTimeVal(e.time, t0, dT, left)
1790 e.end = self.trimTimeVal(e.end, t0, dT, left)
1791 e.length = e.end - e.time
1792 if('cpuexec' in d):
1793 cpuexec = dict()
1794 for e in d['cpuexec']:
1795 c0, cN = e
1796 c0 = self.trimTimeVal(c0, t0, dT, left)
1797 cN = self.trimTimeVal(cN, t0, dT, left)
1798 cpuexec[(c0, cN)] = d['cpuexec'][e]
1799 d['cpuexec'] = cpuexec
1800 for dir in ['suspend', 'resume']:
1801 list = []
1802 for e in self.errorinfo[dir]:
1803 type, tm, idx1, idx2 = e
1804 tm = self.trimTimeVal(tm, t0, dT, left)
1805 list.append((type, tm, idx1, idx2))
1806 self.errorinfo[dir] = list
1807 def trimFreezeTime(self, tZero):
1808 # trim out any standby or freeze clock time
1809 lp = ''
1810 for phase in self.sortedPhases():
1811 if 'resume_machine' in phase and 'suspend_machine' in lp:
1812 tS, tR = self.dmesg[lp]['end'], self.dmesg[phase]['start']
1813 tL = tR - tS
1814 if tL <= 0:
1815 continue
1816 left = True if tR > tZero else False
1817 self.trimTime(tS, tL, left)
1818 if 'waking' in self.dmesg[lp]:
1819 tCnt = self.dmesg[lp]['waking'][0]
1820 if self.dmesg[lp]['waking'][1] >= 0.001:
1821 tTry = '%.0f' % (round(self.dmesg[lp]['waking'][1] * 1000))
1822 else:
1823 tTry = '%.3f' % (self.dmesg[lp]['waking'][1] * 1000)
1824 text = '%.0f (%s ms waking %d times)' % (tL * 1000, tTry, tCnt)
1825 else:
1826 text = '%.0f' % (tL * 1000)
1827 self.tLow.append(text)
1828 lp = phase
1829 def getMemTime(self):
1830 if not self.hwstart or not self.hwend:
1831 return
1832 stime = (self.tSuspended - self.start) * 1000000
1833 rtime = (self.end - self.tResumed) * 1000000
1834 hws = self.hwstart + timedelta(microseconds=stime)
1835 hwr = self.hwend - timedelta(microseconds=rtime)
1836 self.tLow.append('%.0f'%((hwr - hws).total_seconds() * 1000))
1837 def getTimeValues(self):
1838 sktime = (self.tSuspended - self.tKernSus) * 1000
1839 rktime = (self.tKernRes - self.tResumed) * 1000
1840 return (sktime, rktime)
1841 def setPhase(self, phase, ktime, isbegin, order=-1):
1842 if(isbegin):
1843 # phase start over current phase
1844 if self.currphase:
1845 if 'resume_machine' not in self.currphase:
1846 sysvals.vprint('WARNING: phase %s failed to end' % self.currphase)
1847 self.dmesg[self.currphase]['end'] = ktime
1848 phases = self.dmesg.keys()
1849 color = self.phasedef[phase]['color']
1850 count = len(phases) if order < 0 else order
1851 # create unique name for every new phase
1852 while phase in phases:
1853 phase += '*'
1854 self.dmesg[phase] = {'list': dict(), 'start': -1.0, 'end': -1.0,
1855 'row': 0, 'color': color, 'order': count}
1856 self.dmesg[phase]['start'] = ktime
1857 self.currphase = phase
1858 else:
1859 # phase end without a start
1860 if phase not in self.currphase:
1861 if self.currphase:
1862 sysvals.vprint('WARNING: %s ended instead of %s, ftrace corruption?' % (phase, self.currphase))
1863 else:
1864 sysvals.vprint('WARNING: %s ended without a start, ftrace corruption?' % phase)
1865 return phase
1866 phase = self.currphase
1867 self.dmesg[phase]['end'] = ktime
1868 self.currphase = ''
1869 return phase
1870 def sortedDevices(self, phase):
1871 list = self.dmesg[phase]['list']
1872 return sorted(list, key=lambda k:list[k]['start'])
1873 def fixupInitcalls(self, phase):
1874 # if any calls never returned, clip them at system resume end
1875 phaselist = self.dmesg[phase]['list']
1876 for devname in phaselist:
1877 dev = phaselist[devname]
1878 if(dev['end'] < 0):
1879 for p in self.sortedPhases():
1880 if self.dmesg[p]['end'] > dev['start']:
1881 dev['end'] = self.dmesg[p]['end']
1882 break
1883 sysvals.vprint('%s (%s): callback didnt return' % (devname, phase))
1884 def deviceFilter(self, devicefilter):
1885 for phase in self.sortedPhases():
1886 list = self.dmesg[phase]['list']
1887 rmlist = []
1888 for name in list:
1889 keep = False
1890 for filter in devicefilter:
1891 if filter in name or \
1892 ('drv' in list[name] and filter in list[name]['drv']):
1893 keep = True
1894 if not keep:
1895 rmlist.append(name)
1896 for name in rmlist:
1897 del list[name]
1898 def fixupInitcallsThatDidntReturn(self):
1899 # if any calls never returned, clip them at system resume end
1900 for phase in self.sortedPhases():
1901 self.fixupInitcalls(phase)
1902 def phaseOverlap(self, phases):
1903 rmgroups = []
1904 newgroup = []
1905 for group in self.devicegroups:
1906 for phase in phases:
1907 if phase not in group:
1908 continue
1909 for p in group:
1910 if p not in newgroup:
1911 newgroup.append(p)
1912 if group not in rmgroups:
1913 rmgroups.append(group)
1914 for group in rmgroups:
1915 self.devicegroups.remove(group)
1916 self.devicegroups.append(newgroup)
1917 def newActionGlobal(self, name, start, end, pid=-1, color=''):
1918 # which phase is this device callback or action in
1919 phases = self.sortedPhases()
1920 targetphase = 'none'
1921 htmlclass = ''
1922 overlap = 0.0
1923 myphases = []
1924 for phase in phases:
1925 pstart = self.dmesg[phase]['start']
1926 pend = self.dmesg[phase]['end']
1927 # see if the action overlaps this phase
1928 o = max(0, min(end, pend) - max(start, pstart))
1929 if o > 0:
1930 myphases.append(phase)
1931 # set the target phase to the one that overlaps most
1932 if o > overlap:
1933 if overlap > 0 and phase == 'post_resume':
1934 continue
1935 targetphase = phase
1936 overlap = o
1937 # if no target phase was found, pin it to the edge
1938 if targetphase == 'none':
1939 p0start = self.dmesg[phases[0]]['start']
1940 if start <= p0start:
1941 targetphase = phases[0]
1942 else:
1943 targetphase = phases[-1]
1944 if pid == -2:
1945 htmlclass = ' bg'
1946 elif pid == -3:
1947 htmlclass = ' ps'
1948 if len(myphases) > 1:
1949 htmlclass = ' bg'
1950 self.phaseOverlap(myphases)
1951 if targetphase in phases:
1952 newname = self.newAction(targetphase, name, pid, '', start, end, '', htmlclass, color)
1953 return (targetphase, newname)
1954 return False
1955 def newAction(self, phase, name, pid, parent, start, end, drv, htmlclass='', color=''):
1956 # new device callback for a specific phase
1957 self.html_device_id += 1
1958 devid = '%s%d' % (self.idstr, self.html_device_id)
1959 list = self.dmesg[phase]['list']
1960 length = -1.0
1961 if(start >= 0 and end >= 0):
1962 length = end - start
1963 if pid == -2 or name not in sysvals.tracefuncs.keys():
1964 i = 2
1965 origname = name
1966 while(name in list):
1967 name = '%s[%d]' % (origname, i)
1968 i += 1
1969 list[name] = {'name': name, 'start': start, 'end': end, 'pid': pid,
1970 'par': parent, 'length': length, 'row': 0, 'id': devid, 'drv': drv }
1971 if htmlclass:
1972 list[name]['htmlclass'] = htmlclass
1973 if color:
1974 list[name]['color'] = color
1975 return name
1976 def findDevice(self, phase, name):
1977 list = self.dmesg[phase]['list']
1978 mydev = ''
1979 for devname in sorted(list):
1980 if name == devname or re.match('^%s\[(?P<num>[0-9]*)\]$' % name, devname):
1981 mydev = devname
1982 if mydev:
1983 return list[mydev]
1984 return False
1985 def deviceChildren(self, devname, phase):
1986 devlist = []
1987 list = self.dmesg[phase]['list']
1988 for child in list:
1989 if(list[child]['par'] == devname):
1990 devlist.append(child)
1991 return devlist
1992 def maxDeviceNameSize(self, phase):
1993 size = 0
1994 for name in self.dmesg[phase]['list']:
1995 if len(name) > size:
1996 size = len(name)
1997 return size
1998 def printDetails(self):
1999 sysvals.vprint('Timeline Details:')
2000 sysvals.vprint(' test start: %f' % self.start)
2001 sysvals.vprint('kernel suspend start: %f' % self.tKernSus)
2002 tS = tR = False
2003 for phase in self.sortedPhases():
2004 devlist = self.dmesg[phase]['list']
2005 dc, ps, pe = len(devlist), self.dmesg[phase]['start'], self.dmesg[phase]['end']
2006 if not tS and ps >= self.tSuspended:
2007 sysvals.vprint(' machine suspended: %f' % self.tSuspended)
2008 tS = True
2009 if not tR and ps >= self.tResumed:
2010 sysvals.vprint(' machine resumed: %f' % self.tResumed)
2011 tR = True
2012 sysvals.vprint('%20s: %f - %f (%d devices)' % (phase, ps, pe, dc))
2013 if sysvals.devdump:
2014 sysvals.vprint(''.join('-' for i in range(80)))
2015 maxname = '%d' % self.maxDeviceNameSize(phase)
2016 fmt = '%3d) %'+maxname+'s - %f - %f'
2017 c = 1
2018 for name in sorted(devlist):
2019 s = devlist[name]['start']
2020 e = devlist[name]['end']
2021 sysvals.vprint(fmt % (c, name, s, e))
2022 c += 1
2023 sysvals.vprint(''.join('-' for i in range(80)))
2024 sysvals.vprint(' kernel resume end: %f' % self.tKernRes)
2025 sysvals.vprint(' test end: %f' % self.end)
2026 def deviceChildrenAllPhases(self, devname):
2027 devlist = []
2028 for phase in self.sortedPhases():
2029 list = self.deviceChildren(devname, phase)
2030 for dev in sorted(list):
2031 if dev not in devlist:
2032 devlist.append(dev)
2033 return devlist
2034 def masterTopology(self, name, list, depth):
2035 node = DeviceNode(name, depth)
2036 for cname in list:
2037 # avoid recursions
2038 if name == cname:
2039 continue
2040 clist = self.deviceChildrenAllPhases(cname)
2041 cnode = self.masterTopology(cname, clist, depth+1)
2042 node.children.append(cnode)
2043 return node
2044 def printTopology(self, node):
2045 html = ''
2046 if node.name:
2047 info = ''
2048 drv = ''
2049 for phase in self.sortedPhases():
2050 list = self.dmesg[phase]['list']
2051 if node.name in list:
2052 s = list[node.name]['start']
2053 e = list[node.name]['end']
2054 if list[node.name]['drv']:
2055 drv = ' {'+list[node.name]['drv']+'}'
2056 info += ('<li>%s: %.3fms</li>' % (phase, (e-s)*1000))
2057 html += '<li><b>'+node.name+drv+'</b>'
2058 if info:
2059 html += '<ul>'+info+'</ul>'
2060 html += '</li>'
2061 if len(node.children) > 0:
2062 html += '<ul>'
2063 for cnode in node.children:
2064 html += self.printTopology(cnode)
2065 html += '</ul>'
2066 return html
2067 def rootDeviceList(self):
2068 # list of devices graphed
2069 real = []
2070 for phase in self.sortedPhases():
2071 list = self.dmesg[phase]['list']
2072 for dev in sorted(list):
2073 if list[dev]['pid'] >= 0 and dev not in real:
2074 real.append(dev)
2075 # list of top-most root devices
2076 rootlist = []
2077 for phase in self.sortedPhases():
2078 list = self.dmesg[phase]['list']
2079 for dev in sorted(list):
2080 pdev = list[dev]['par']
2081 pid = list[dev]['pid']
2082 if(pid < 0 or re.match('[0-9]*-[0-9]*\.[0-9]*[\.0-9]*\:[\.0-9]*$', pdev)):
2083 continue
2084 if pdev and pdev not in real and pdev not in rootlist:
2085 rootlist.append(pdev)
2086 return rootlist
2087 def deviceTopology(self):
2088 rootlist = self.rootDeviceList()
2089 master = self.masterTopology('', rootlist, 0)
2090 return self.printTopology(master)
2091 def selectTimelineDevices(self, widfmt, tTotal, mindevlen):
2092 # only select devices that will actually show up in html
2093 self.tdevlist = dict()
2094 for phase in self.dmesg:
2095 devlist = []
2096 list = self.dmesg[phase]['list']
2097 for dev in list:
2098 length = (list[dev]['end'] - list[dev]['start']) * 1000
2099 width = widfmt % (((list[dev]['end']-list[dev]['start'])*100)/tTotal)
2100 if length >= mindevlen:
2101 devlist.append(dev)
2102 self.tdevlist[phase] = devlist
2103 def addHorizontalDivider(self, devname, devend):
2104 phase = 'suspend_prepare'
2105 self.newAction(phase, devname, -2, '', \
2106 self.start, devend, '', ' sec', '')
2107 if phase not in self.tdevlist:
2108 self.tdevlist[phase] = []
2109 self.tdevlist[phase].append(devname)
2110 d = DevItem(0, phase, self.dmesg[phase]['list'][devname])
2111 return d
2112 def addProcessUsageEvent(self, name, times):
2113 # get the start and end times for this process
2114 cpuexec = dict()
2115 tlast = start = end = -1
2116 for t in sorted(times):
2117 if tlast < 0:
2118 tlast = t
2119 continue
2120 if name in self.pstl[t] and self.pstl[t][name] > 0:
2121 if start < 0:
2122 start = tlast
2123 end, key = t, (tlast, t)
2124 maxj = (t - tlast) * 1024.0
2125 cpuexec[key] = min(1.0, float(self.pstl[t][name]) / maxj)
2126 tlast = t
2127 if start < 0 or end < 0:
2128 return
2129 # add a new action for this process and get the object
2130 out = self.newActionGlobal(name, start, end, -3)
2131 if out:
2132 phase, devname = out
2133 dev = self.dmesg[phase]['list'][devname]
2134 dev['cpuexec'] = cpuexec
2135 def createProcessUsageEvents(self):
2136 # get an array of process names and times
2137 proclist = {'sus': dict(), 'res': dict()}
2138 tdata = {'sus': [], 'res': []}
2139 for t in sorted(self.pstl):
2140 dir = 'sus' if t < self.tSuspended else 'res'
2141 for ps in sorted(self.pstl[t]):
2142 if ps not in proclist[dir]:
2143 proclist[dir][ps] = 0
2144 tdata[dir].append(t)
2145 # process the events for suspend and resume
2146 if len(proclist['sus']) > 0 or len(proclist['res']) > 0:
2147 sysvals.vprint('Process Execution:')
2148 for dir in ['sus', 'res']:
2149 for ps in sorted(proclist[dir]):
2150 self.addProcessUsageEvent(ps, tdata[dir])
2151 def handleEndMarker(self, time, msg=''):
2152 dm = self.dmesg
2153 self.setEnd(time, msg)
2154 self.initDevicegroups()
2155 # give suspend_prepare an end if needed
2156 if 'suspend_prepare' in dm and dm['suspend_prepare']['end'] < 0:
2157 dm['suspend_prepare']['end'] = time
2158 # assume resume machine ends at next phase start
2159 if 'resume_machine' in dm and dm['resume_machine']['end'] < 0:
2160 np = self.nextPhase('resume_machine', 1)
2161 if np:
2162 dm['resume_machine']['end'] = dm[np]['start']
2163 # if kernel resume end not found, assume its the end marker
2164 if self.tKernRes == 0.0:
2165 self.tKernRes = time
2166 # if kernel suspend start not found, assume its the end marker
2167 if self.tKernSus == 0.0:
2168 self.tKernSus = time
2169 # set resume complete to end at end marker
2170 if 'resume_complete' in dm:
2171 dm['resume_complete']['end'] = time
2172 def initcall_debug_call(self, line, quick=False):
2173 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2174 'PM: *calling .* @ (?P<n>.*), parent: (?P<p>.*)', line)
2175 if not m:
2176 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2177 'calling .* @ (?P<n>.*), parent: (?P<p>.*)', line)
2178 if not m:
2179 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) calling '+\
2180 '(?P<f>.*)\+ @ (?P<n>.*), parent: (?P<p>.*)', line)
2181 if m:
2182 return True if quick else m.group('t', 'f', 'n', 'p')
2183 return False if quick else ('', '', '', '')
2184 def initcall_debug_return(self, line, quick=False):
2185 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: PM: '+\
2186 '.* returned (?P<r>[0-9]*) after (?P<dt>[0-9]*) usecs', line)
2187 if not m:
2188 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) .* (?P<f>.*)\: '+\
2189 '.* returned (?P<r>[0-9]*) after (?P<dt>[0-9]*) usecs', line)
2190 if not m:
2191 m = re.match('.*(\[ *)(?P<t>[0-9\.]*)(\]) call '+\
2192 '(?P<f>.*)\+ returned .* after (?P<dt>.*) usecs', line)
2193 if m:
2194 return True if quick else m.group('t', 'f', 'dt')
2195 return False if quick else ('', '', '')
2196 def debugPrint(self):
2197 for p in self.sortedPhases():
2198 list = self.dmesg[p]['list']
2199 for devname in sorted(list):
2200 dev = list[devname]
2201 if 'ftrace' in dev:
2202 dev['ftrace'].debugPrint(' [%s]' % devname)
2203
2204# Class: DevFunction
2205# Description:
2206# A container for kprobe function data we want in the dev timeline
2207class DevFunction:
2208 def __init__(self, name, args, caller, ret, start, end, u, proc, pid, color):
2209 self.row = 0
2210 self.count = 1
2211 self.name = name
2212 self.args = args
2213 self.caller = caller
2214 self.ret = ret
2215 self.time = start
2216 self.length = end - start
2217 self.end = end
2218 self.ubiquitous = u
2219 self.proc = proc
2220 self.pid = pid
2221 self.color = color
2222 def title(self):
2223 cnt = ''
2224 if self.count > 1:
2225 cnt = '(x%d)' % self.count
2226 l = '%0.3fms' % (self.length * 1000)
2227 if self.ubiquitous:
2228 title = '%s(%s)%s <- %s, %s(%s)' % \
2229 (self.name, self.args, cnt, self.caller, self.ret, l)
2230 else:
2231 title = '%s(%s) %s%s(%s)' % (self.name, self.args, self.ret, cnt, l)
2232 return title.replace('"', '')
2233 def text(self):
2234 if self.count > 1:
2235 text = '%s(x%d)' % (self.name, self.count)
2236 else:
2237 text = self.name
2238 return text
2239 def repeat(self, tgt):
2240 # is the tgt call just a repeat of this call (e.g. are we in a loop)
2241 dt = self.time - tgt.end
2242 # only combine calls if -all- attributes are identical
2243 if tgt.caller == self.caller and \
2244 tgt.name == self.name and tgt.args == self.args and \
2245 tgt.proc == self.proc and tgt.pid == self.pid and \
2246 tgt.ret == self.ret and dt >= 0 and \
2247 dt <= sysvals.callloopmaxgap and \
2248 self.length < sysvals.callloopmaxlen:
2249 return True
2250 return False
2251
2252# Class: FTraceLine
2253# Description:
2254# A container for a single line of ftrace data. There are six basic types:
2255# callgraph line:
2256# call: " dpm_run_callback() {"
2257# return: " }"
2258# leaf: " dpm_run_callback();"
2259# trace event:
2260# tracing_mark_write: SUSPEND START or RESUME COMPLETE
2261# suspend_resume: phase or custom exec block data
2262# device_pm_callback: device callback info
2263class FTraceLine:
2264 def __init__(self, t, m='', d=''):
2265 self.length = 0.0
2266 self.fcall = False
2267 self.freturn = False
2268 self.fevent = False
2269 self.fkprobe = False
2270 self.depth = 0
2271 self.name = ''
2272 self.type = ''
2273 self.time = float(t)
2274 if not m and not d:
2275 return
2276 # is this a trace event
2277 if(d == 'traceevent' or re.match('^ *\/\* *(?P<msg>.*) \*\/ *$', m)):
2278 if(d == 'traceevent'):
2279 # nop format trace event
2280 msg = m
2281 else:
2282 # function_graph format trace event
2283 em = re.match('^ *\/\* *(?P<msg>.*) \*\/ *$', m)
2284 msg = em.group('msg')
2285
2286 emm = re.match('^(?P<call>.*?): (?P<msg>.*)', msg)
2287 if(emm):
2288 self.name = emm.group('msg')
2289 self.type = emm.group('call')
2290 else:
2291 self.name = msg
2292 km = re.match('^(?P<n>.*)_cal$', self.type)
2293 if km:
2294 self.fcall = True
2295 self.fkprobe = True
2296 self.type = km.group('n')
2297 return
2298 km = re.match('^(?P<n>.*)_ret$', self.type)
2299 if km:
2300 self.freturn = True
2301 self.fkprobe = True
2302 self.type = km.group('n')
2303 return
2304 self.fevent = True
2305 return
2306 # convert the duration to seconds
2307 if(d):
2308 self.length = float(d)/1000000
2309 # the indentation determines the depth
2310 match = re.match('^(?P<d> *)(?P<o>.*)$', m)
2311 if(not match):
2312 return
2313 self.depth = self.getDepth(match.group('d'))
2314 m = match.group('o')
2315 # function return
2316 if(m[0] == '}'):
2317 self.freturn = True
2318 if(len(m) > 1):
2319 # includes comment with function name
2320 match = re.match('^} *\/\* *(?P<n>.*) *\*\/$', m)
2321 if(match):
2322 self.name = match.group('n').strip()
2323 # function call
2324 else:
2325 self.fcall = True
2326 # function call with children
2327 if(m[-1] == '{'):
2328 match = re.match('^(?P<n>.*) *\(.*', m)
2329 if(match):
2330 self.name = match.group('n').strip()
2331 # function call with no children (leaf)
2332 elif(m[-1] == ';'):
2333 self.freturn = True
2334 match = re.match('^(?P<n>.*) *\(.*', m)
2335 if(match):
2336 self.name = match.group('n').strip()
2337 # something else (possibly a trace marker)
2338 else:
2339 self.name = m
2340 def isCall(self):
2341 return self.fcall and not self.freturn
2342 def isReturn(self):
2343 return self.freturn and not self.fcall
2344 def isLeaf(self):
2345 return self.fcall and self.freturn
2346 def getDepth(self, str):
2347 return len(str)/2
2348 def debugPrint(self, info=''):
2349 if self.isLeaf():
2350 pprint(' -- %12.6f (depth=%02d): %s(); (%.3f us) %s' % (self.time, \
2351 self.depth, self.name, self.length*1000000, info))
2352 elif self.freturn:
2353 pprint(' -- %12.6f (depth=%02d): %s} (%.3f us) %s' % (self.time, \
2354 self.depth, self.name, self.length*1000000, info))
2355 else:
2356 pprint(' -- %12.6f (depth=%02d): %s() { (%.3f us) %s' % (self.time, \
2357 self.depth, self.name, self.length*1000000, info))
2358 def startMarker(self):
2359 # Is this the starting line of a suspend?
2360 if not self.fevent:
2361 return False
2362 if sysvals.usetracemarkers:
2363 if(self.name.startswith('SUSPEND START')):
2364 return True
2365 return False
2366 else:
2367 if(self.type == 'suspend_resume' and
2368 re.match('suspend_enter\[.*\] begin', self.name)):
2369 return True
2370 return False
2371 def endMarker(self):
2372 # Is this the ending line of a resume?
2373 if not self.fevent:
2374 return False
2375 if sysvals.usetracemarkers:
2376 if(self.name.startswith('RESUME COMPLETE')):
2377 return True
2378 return False
2379 else:
2380 if(self.type == 'suspend_resume' and
2381 re.match('thaw_processes\[.*\] end', self.name)):
2382 return True
2383 return False
2384
2385# Class: FTraceCallGraph
2386# Description:
2387# A container for the ftrace callgraph of a single recursive function.
2388# This can be a dpm_run_callback, dpm_prepare, or dpm_complete callgraph
2389# Each instance is tied to a single device in a single phase, and is
2390# comprised of an ordered list of FTraceLine objects
2391class FTraceCallGraph:
2392 vfname = 'missing_function_name'
2393 def __init__(self, pid, sv):
2394 self.id = ''
2395 self.invalid = False
2396 self.name = ''
2397 self.partial = False
2398 self.ignore = False
2399 self.start = -1.0
2400 self.end = -1.0
2401 self.list = []
2402 self.depth = 0
2403 self.pid = pid
2404 self.sv = sv
2405 def addLine(self, line):
2406 # if this is already invalid, just leave
2407 if(self.invalid):
2408 if(line.depth == 0 and line.freturn):
2409 return 1
2410 return 0
2411 # invalidate on bad depth
2412 if(self.depth < 0):
2413 self.invalidate(line)
2414 return 0
2415 # ignore data til we return to the current depth
2416 if self.ignore:
2417 if line.depth > self.depth:
2418 return 0
2419 else:
2420 self.list[-1].freturn = True
2421 self.list[-1].length = line.time - self.list[-1].time
2422 self.ignore = False
2423 # if this is a return at self.depth, no more work is needed
2424 if line.depth == self.depth and line.isReturn():
2425 if line.depth == 0:
2426 self.end = line.time
2427 return 1
2428 return 0
2429 # compare current depth with this lines pre-call depth
2430 prelinedep = line.depth
2431 if line.isReturn():
2432 prelinedep += 1
2433 last = 0
2434 lasttime = line.time
2435 if len(self.list) > 0:
2436 last = self.list[-1]
2437 lasttime = last.time
2438 if last.isLeaf():
2439 lasttime += last.length
2440 # handle low misalignments by inserting returns
2441 mismatch = prelinedep - self.depth
2442 warning = self.sv.verbose and abs(mismatch) > 1
2443 info = []
2444 if mismatch < 0:
2445 idx = 0
2446 # add return calls to get the depth down
2447 while prelinedep < self.depth:
2448 self.depth -= 1
2449 if idx == 0 and last and last.isCall():
2450 # special case, turn last call into a leaf
2451 last.depth = self.depth
2452 last.freturn = True
2453 last.length = line.time - last.time
2454 if warning:
2455 info.append(('[make leaf]', last))
2456 else:
2457 vline = FTraceLine(lasttime)
2458 vline.depth = self.depth
2459 vline.name = self.vfname
2460 vline.freturn = True
2461 self.list.append(vline)
2462 if warning:
2463 if idx == 0:
2464 info.append(('', last))
2465 info.append(('[add return]', vline))
2466 idx += 1
2467 if warning:
2468 info.append(('', line))
2469 # handle high misalignments by inserting calls
2470 elif mismatch > 0:
2471 idx = 0
2472 if warning:
2473 info.append(('', last))
2474 # add calls to get the depth up
2475 while prelinedep > self.depth:
2476 if idx == 0 and line.isReturn():
2477 # special case, turn this return into a leaf
2478 line.fcall = True
2479 prelinedep -= 1
2480 if warning:
2481 info.append(('[make leaf]', line))
2482 else:
2483 vline = FTraceLine(lasttime)
2484 vline.depth = self.depth
2485 vline.name = self.vfname
2486 vline.fcall = True
2487 self.list.append(vline)
2488 self.depth += 1
2489 if not last:
2490 self.start = vline.time
2491 if warning:
2492 info.append(('[add call]', vline))
2493 idx += 1
2494 if warning and ('[make leaf]', line) not in info:
2495 info.append(('', line))
2496 if warning:
2497 pprint('WARNING: ftrace data missing, corrections made:')
2498 for i in info:
2499 t, obj = i
2500 if obj:
2501 obj.debugPrint(t)
2502 # process the call and set the new depth
2503 skipadd = False
2504 md = self.sv.max_graph_depth
2505 if line.isCall():
2506 # ignore blacklisted/overdepth funcs
2507 if (md and self.depth >= md - 1) or (line.name in self.sv.cgblacklist):
2508 self.ignore = True
2509 else:
2510 self.depth += 1
2511 elif line.isReturn():
2512 self.depth -= 1
2513 # remove blacklisted/overdepth/empty funcs that slipped through
2514 if (last and last.isCall() and last.depth == line.depth) or \
2515 (md and last and last.depth >= md) or \
2516 (line.name in self.sv.cgblacklist):
2517 while len(self.list) > 0 and self.list[-1].depth > line.depth:
2518 self.list.pop(-1)
2519 if len(self.list) == 0:
2520 self.invalid = True
2521 return 1
2522 self.list[-1].freturn = True
2523 self.list[-1].length = line.time - self.list[-1].time
2524 self.list[-1].name = line.name
2525 skipadd = True
2526 if len(self.list) < 1:
2527 self.start = line.time
2528 # check for a mismatch that returned all the way to callgraph end
2529 res = 1
2530 if mismatch < 0 and self.list[-1].depth == 0 and self.list[-1].freturn:
2531 line = self.list[-1]
2532 skipadd = True
2533 res = -1
2534 if not skipadd:
2535 self.list.append(line)
2536 if(line.depth == 0 and line.freturn):
2537 if(self.start < 0):
2538 self.start = line.time
2539 self.end = line.time
2540 if line.fcall:
2541 self.end += line.length
2542 if self.list[0].name == self.vfname:
2543 self.invalid = True
2544 if res == -1:
2545 self.partial = True
2546 return res
2547 return 0
2548 def invalidate(self, line):
2549 if(len(self.list) > 0):
2550 first = self.list[0]
2551 self.list = []
2552 self.list.append(first)
2553 self.invalid = True
2554 id = 'task %s' % (self.pid)
2555 window = '(%f - %f)' % (self.start, line.time)
2556 if(self.depth < 0):
2557 pprint('Data misalignment for '+id+\
2558 ' (buffer overflow), ignoring this callback')
2559 else:
2560 pprint('Too much data for '+id+\
2561 ' '+window+', ignoring this callback')
2562 def slice(self, dev):
2563 minicg = FTraceCallGraph(dev['pid'], self.sv)
2564 minicg.name = self.name
2565 mydepth = -1
2566 good = False
2567 for l in self.list:
2568 if(l.time < dev['start'] or l.time > dev['end']):
2569 continue
2570 if mydepth < 0:
2571 if l.name == 'mutex_lock' and l.freturn:
2572 mydepth = l.depth
2573 continue
2574 elif l.depth == mydepth and l.name == 'mutex_unlock' and l.fcall:
2575 good = True
2576 break
2577 l.depth -= mydepth
2578 minicg.addLine(l)
2579 if not good or len(minicg.list) < 1:
2580 return 0
2581 return minicg
2582 def repair(self, enddepth):
2583 # bring the depth back to 0 with additional returns
2584 fixed = False
2585 last = self.list[-1]
2586 for i in reversed(range(enddepth)):
2587 t = FTraceLine(last.time)
2588 t.depth = i
2589 t.freturn = True
2590 fixed = self.addLine(t)
2591 if fixed != 0:
2592 self.end = last.time
2593 return True
2594 return False
2595 def postProcess(self):
2596 if len(self.list) > 0:
2597 self.name = self.list[0].name
2598 stack = dict()
2599 cnt = 0
2600 last = 0
2601 for l in self.list:
2602 # ftrace bug: reported duration is not reliable
2603 # check each leaf and clip it at max possible length
2604 if last and last.isLeaf():
2605 if last.length > l.time - last.time:
2606 last.length = l.time - last.time
2607 if l.isCall():
2608 stack[l.depth] = l
2609 cnt += 1
2610 elif l.isReturn():
2611 if(l.depth not in stack):
2612 if self.sv.verbose:
2613 pprint('Post Process Error: Depth missing')
2614 l.debugPrint()
2615 return False
2616 # calculate call length from call/return lines
2617 cl = stack[l.depth]
2618 cl.length = l.time - cl.time
2619 if cl.name == self.vfname:
2620 cl.name = l.name
2621 stack.pop(l.depth)
2622 l.length = 0
2623 cnt -= 1
2624 last = l
2625 if(cnt == 0):
2626 # trace caught the whole call tree
2627 return True
2628 elif(cnt < 0):
2629 if self.sv.verbose:
2630 pprint('Post Process Error: Depth is less than 0')
2631 return False
2632 # trace ended before call tree finished
2633 return self.repair(cnt)
2634 def deviceMatch(self, pid, data):
2635 found = ''
2636 # add the callgraph data to the device hierarchy
2637 borderphase = {
2638 'dpm_prepare': 'suspend_prepare',
2639 'dpm_complete': 'resume_complete'
2640 }
2641 if(self.name in borderphase):
2642 p = borderphase[self.name]
2643 list = data.dmesg[p]['list']
2644 for devname in list:
2645 dev = list[devname]
2646 if(pid == dev['pid'] and
2647 self.start <= dev['start'] and
2648 self.end >= dev['end']):
2649 cg = self.slice(dev)
2650 if cg:
2651 dev['ftrace'] = cg
2652 found = devname
2653 return found
2654 for p in data.sortedPhases():
2655 if(data.dmesg[p]['start'] <= self.start and
2656 self.start <= data.dmesg[p]['end']):
2657 list = data.dmesg[p]['list']
2658 for devname in sorted(list, key=lambda k:list[k]['start']):
2659 dev = list[devname]
2660 if(pid == dev['pid'] and
2661 self.start <= dev['start'] and
2662 self.end >= dev['end']):
2663 dev['ftrace'] = self
2664 found = devname
2665 break
2666 break
2667 return found
2668 def newActionFromFunction(self, data):
2669 name = self.name
2670 if name in ['dpm_run_callback', 'dpm_prepare', 'dpm_complete']:
2671 return
2672 fs = self.start
2673 fe = self.end
2674 if fs < data.start or fe > data.end:
2675 return
2676 phase = ''
2677 for p in data.sortedPhases():
2678 if(data.dmesg[p]['start'] <= self.start and
2679 self.start < data.dmesg[p]['end']):
2680 phase = p
2681 break
2682 if not phase:
2683 return
2684 out = data.newActionGlobal(name, fs, fe, -2)
2685 if out:
2686 phase, myname = out
2687 data.dmesg[phase]['list'][myname]['ftrace'] = self
2688 def debugPrint(self, info=''):
2689 pprint('%s pid=%d [%f - %f] %.3f us' % \
2690 (self.name, self.pid, self.start, self.end,
2691 (self.end - self.start)*1000000))
2692 for l in self.list:
2693 if l.isLeaf():
2694 pprint('%f (%02d): %s(); (%.3f us)%s' % (l.time, \
2695 l.depth, l.name, l.length*1000000, info))
2696 elif l.freturn:
2697 pprint('%f (%02d): %s} (%.3f us)%s' % (l.time, \
2698 l.depth, l.name, l.length*1000000, info))
2699 else:
2700 pprint('%f (%02d): %s() { (%.3f us)%s' % (l.time, \
2701 l.depth, l.name, l.length*1000000, info))
2702 pprint(' ')
2703
2704class DevItem:
2705 def __init__(self, test, phase, dev):
2706 self.test = test
2707 self.phase = phase
2708 self.dev = dev
2709 def isa(self, cls):
2710 if 'htmlclass' in self.dev and cls in self.dev['htmlclass']:
2711 return True
2712 return False
2713
2714# Class: Timeline
2715# Description:
2716# A container for a device timeline which calculates
2717# all the html properties to display it correctly
2718class Timeline:
2719 html_tblock = '<div id="block{0}" class="tblock" style="left:{1}%;width:{2}%;"><div class="tback" style="height:{3}px"></div>\n'
2720 html_device = '<div id="{0}" title="{1}" class="thread{7}" style="left:{2}%;top:{3}px;height:{4}px;width:{5}%;{8}">{6}</div>\n'
2721 html_phase = '<div class="phase" style="left:{0}%;width:{1}%;top:{2}px;height:{3}px;background:{4}">{5}</div>\n'
2722 html_phaselet = '<div id="{0}" class="phaselet" style="left:{1}%;width:{2}%;background:{3}"></div>\n'
2723 html_legend = '<div id="p{3}" class="square" style="left:{0}%;background:{1}"> {2}</div>\n'
2724 def __init__(self, rowheight, scaleheight):
2725 self.html = ''
2726 self.height = 0 # total timeline height
2727 self.scaleH = scaleheight # timescale (top) row height
2728 self.rowH = rowheight # device row height
2729 self.bodyH = 0 # body height
2730 self.rows = 0 # total timeline rows
2731 self.rowlines = dict()
2732 self.rowheight = dict()
2733 def createHeader(self, sv, stamp):
2734 if(not stamp['time']):
2735 return
2736 self.html += '<div class="version"><a href="https://01.org/pm-graph">%s v%s</a></div>' \
2737 % (sv.title, sv.version)
2738 if sv.logmsg and sv.testlog:
2739 self.html += '<button id="showtest" class="logbtn btnfmt">log</button>'
2740 if sv.dmesglog:
2741 self.html += '<button id="showdmesg" class="logbtn btnfmt">dmesg</button>'
2742 if sv.ftracelog:
2743 self.html += '<button id="showftrace" class="logbtn btnfmt">ftrace</button>'
2744 headline_stamp = '<div class="stamp">{0} {1} {2} {3}</div>\n'
2745 self.html += headline_stamp.format(stamp['host'], stamp['kernel'],
2746 stamp['mode'], stamp['time'])
2747 if 'man' in stamp and 'plat' in stamp and 'cpu' in stamp and \
2748 stamp['man'] and stamp['plat'] and stamp['cpu']:
2749 headline_sysinfo = '<div class="stamp sysinfo">{0} {1} <i>with</i> {2}</div>\n'
2750 self.html += headline_sysinfo.format(stamp['man'], stamp['plat'], stamp['cpu'])
2751
2752 # Function: getDeviceRows
2753 # Description:
2754 # determine how may rows the device funcs will take
2755 # Arguments:
2756 # rawlist: the list of devices/actions for a single phase
2757 # Output:
2758 # The total number of rows needed to display this phase of the timeline
2759 def getDeviceRows(self, rawlist):
2760 # clear all rows and set them to undefined
2761 sortdict = dict()
2762 for item in rawlist:
2763 item.row = -1
2764 sortdict[item] = item.length
2765 sortlist = sorted(sortdict, key=sortdict.get, reverse=True)
2766 remaining = len(sortlist)
2767 rowdata = dict()
2768 row = 1
2769 # try to pack each row with as many ranges as possible
2770 while(remaining > 0):
2771 if(row not in rowdata):
2772 rowdata[row] = []
2773 for i in sortlist:
2774 if(i.row >= 0):
2775 continue
2776 s = i.time
2777 e = i.time + i.length
2778 valid = True
2779 for ritem in rowdata[row]:
2780 rs = ritem.time
2781 re = ritem.time + ritem.length
2782 if(not (((s <= rs) and (e <= rs)) or
2783 ((s >= re) and (e >= re)))):
2784 valid = False
2785 break
2786 if(valid):
2787 rowdata[row].append(i)
2788 i.row = row
2789 remaining -= 1
2790 row += 1
2791 return row
2792 # Function: getPhaseRows
2793 # Description:
2794 # Organize the timeline entries into the smallest
2795 # number of rows possible, with no entry overlapping
2796 # Arguments:
2797 # devlist: the list of devices/actions in a group of contiguous phases
2798 # Output:
2799 # The total number of rows needed to display this phase of the timeline
2800 def getPhaseRows(self, devlist, row=0, sortby='length'):
2801 # clear all rows and set them to undefined
2802 remaining = len(devlist)
2803 rowdata = dict()
2804 sortdict = dict()
2805 myphases = []
2806 # initialize all device rows to -1 and calculate devrows
2807 for item in devlist:
2808 dev = item.dev
2809 tp = (item.test, item.phase)
2810 if tp not in myphases:
2811 myphases.append(tp)
2812 dev['row'] = -1
2813 if sortby == 'start':
2814 # sort by start 1st, then length 2nd
2815 sortdict[item] = (-1*float(dev['start']), float(dev['end']) - float(dev['start']))
2816 else:
2817 # sort by length 1st, then name 2nd
2818 sortdict[item] = (float(dev['end']) - float(dev['start']), item.dev['name'])
2819 if 'src' in dev:
2820 dev['devrows'] = self.getDeviceRows(dev['src'])
2821 # sort the devlist by length so that large items graph on top
2822 sortlist = sorted(sortdict, key=sortdict.get, reverse=True)
2823 orderedlist = []
2824 for item in sortlist:
2825 if item.dev['pid'] == -2:
2826 orderedlist.append(item)
2827 for item in sortlist:
2828 if item not in orderedlist:
2829 orderedlist.append(item)
2830 # try to pack each row with as many devices as possible
2831 while(remaining > 0):
2832 rowheight = 1
2833 if(row not in rowdata):
2834 rowdata[row] = []
2835 for item in orderedlist:
2836 dev = item.dev
2837 if(dev['row'] < 0):
2838 s = dev['start']
2839 e = dev['end']
2840 valid = True
2841 for ritem in rowdata[row]:
2842 rs = ritem.dev['start']
2843 re = ritem.dev['end']
2844 if(not (((s <= rs) and (e <= rs)) or
2845 ((s >= re) and (e >= re)))):
2846 valid = False
2847 break
2848 if(valid):
2849 rowdata[row].append(item)
2850 dev['row'] = row
2851 remaining -= 1
2852 if 'devrows' in dev and dev['devrows'] > rowheight:
2853 rowheight = dev['devrows']
2854 for t, p in myphases:
2855 if t not in self.rowlines or t not in self.rowheight:
2856 self.rowlines[t] = dict()
2857 self.rowheight[t] = dict()
2858 if p not in self.rowlines[t] or p not in self.rowheight[t]:
2859 self.rowlines[t][p] = dict()
2860 self.rowheight[t][p] = dict()
2861 rh = self.rowH
2862 # section headers should use a different row height
2863 if len(rowdata[row]) == 1 and \
2864 'htmlclass' in rowdata[row][0].dev and \
2865 'sec' in rowdata[row][0].dev['htmlclass']:
2866 rh = 15
2867 self.rowlines[t][p][row] = rowheight
2868 self.rowheight[t][p][row] = rowheight * rh
2869 row += 1
2870 if(row > self.rows):
2871 self.rows = int(row)
2872 return row
2873 def phaseRowHeight(self, test, phase, row):
2874 return self.rowheight[test][phase][row]
2875 def phaseRowTop(self, test, phase, row):
2876 top = 0
2877 for i in sorted(self.rowheight[test][phase]):
2878 if i >= row:
2879 break
2880 top += self.rowheight[test][phase][i]
2881 return top
2882 def calcTotalRows(self):
2883 # Calculate the heights and offsets for the header and rows
2884 maxrows = 0
2885 standardphases = []
2886 for t in self.rowlines:
2887 for p in self.rowlines[t]:
2888 total = 0
2889 for i in sorted(self.rowlines[t][p]):
2890 total += self.rowlines[t][p][i]
2891 if total > maxrows:
2892 maxrows = total
2893 if total == len(self.rowlines[t][p]):
2894 standardphases.append((t, p))
2895 self.height = self.scaleH + (maxrows*self.rowH)
2896 self.bodyH = self.height - self.scaleH
2897 # if there is 1 line per row, draw them the standard way
2898 for t, p in standardphases:
2899 for i in sorted(self.rowheight[t][p]):
2900 self.rowheight[t][p][i] = float(self.bodyH)/len(self.rowlines[t][p])
2901 def createZoomBox(self, mode='command', testcount=1):
2902 # Create bounding box, add buttons
2903 html_zoombox = '<center><button id="zoomin">ZOOM IN +</button><button id="zoomout">ZOOM OUT -</button><button id="zoomdef">ZOOM 1:1</button></center>\n'
2904 html_timeline = '<div id="dmesgzoombox" class="zoombox">\n<div id="{0}" class="timeline" style="height:{1}px">\n'
2905 html_devlist1 = '<button id="devlist1" class="devlist" style="float:left;">Device Detail{0}</button>'
2906 html_devlist2 = '<button id="devlist2" class="devlist" style="float:right;">Device Detail2</button>\n'
2907 if mode != 'command':
2908 if testcount > 1:
2909 self.html += html_devlist2
2910 self.html += html_devlist1.format('1')
2911 else:
2912 self.html += html_devlist1.format('')
2913 self.html += html_zoombox
2914 self.html += html_timeline.format('dmesg', self.height)
2915 # Function: createTimeScale
2916 # Description:
2917 # Create the timescale for a timeline block
2918 # Arguments:
2919 # m0: start time (mode begin)
2920 # mMax: end time (mode end)
2921 # tTotal: total timeline time
2922 # mode: suspend or resume
2923 # Output:
2924 # The html code needed to display the time scale
2925 def createTimeScale(self, m0, mMax, tTotal, mode):
2926 timescale = '<div class="t" style="right:{0}%">{1}</div>\n'
2927 rline = '<div class="t" style="left:0;border-left:1px solid black;border-right:0;">{0}</div>\n'
2928 output = '<div class="timescale">\n'
2929 # set scale for timeline
2930 mTotal = mMax - m0
2931 tS = 0.1
2932 if(tTotal <= 0):
2933 return output+'</div>\n'
2934 if(tTotal > 4):
2935 tS = 1
2936 divTotal = int(mTotal/tS) + 1
2937 divEdge = (mTotal - tS*(divTotal-1))*100/mTotal
2938 for i in range(divTotal):
2939 htmlline = ''
2940 if(mode == 'suspend'):
2941 pos = '%0.3f' % (100 - ((float(i)*tS*100)/mTotal) - divEdge)
2942 val = '%0.fms' % (float(i-divTotal+1)*tS*1000)
2943 if(i == divTotal - 1):
2944 val = mode
2945 htmlline = timescale.format(pos, val)
2946 else:
2947 pos = '%0.3f' % (100 - ((float(i)*tS*100)/mTotal))
2948 val = '%0.fms' % (float(i)*tS*1000)
2949 htmlline = timescale.format(pos, val)
2950 if(i == 0):
2951 htmlline = rline.format(mode)
2952 output += htmlline
2953 self.html += output+'</div>\n'
2954
2955# Class: TestProps
2956# Description:
2957# A list of values describing the properties of these test runs
2958class TestProps:
2959 stampfmt = '# [a-z]*-(?P<m>[0-9]{2})(?P<d>[0-9]{2})(?P<y>[0-9]{2})-'+\
2960 '(?P<H>[0-9]{2})(?P<M>[0-9]{2})(?P<S>[0-9]{2})'+\
2961 ' (?P<host>.*) (?P<mode>.*) (?P<kernel>.*)$'
2962 wififmt = '^# wifi *(?P<d>\S*) *(?P<s>\S*) *(?P<t>[0-9\.]+).*'
2963 tstatfmt = '^# turbostat (?P<t>\S*)'
2964 testerrfmt = '^# enter_sleep_error (?P<e>.*)'
2965 sysinfofmt = '^# sysinfo .*'
2966 cmdlinefmt = '^# command \| (?P<cmd>.*)'
2967 kparamsfmt = '^# kparams \| (?P<kp>.*)'
2968 devpropfmt = '# Device Properties: .*'
2969 pinfofmt = '# platform-(?P<val>[a-z,A-Z,0-9,_]*): (?P<info>.*)'
2970 tracertypefmt = '# tracer: (?P<t>.*)'
2971 firmwarefmt = '# fwsuspend (?P<s>[0-9]*) fwresume (?P<r>[0-9]*)$'
2972 procexecfmt = 'ps - (?P<ps>.*)$'
2973 procmultifmt = '@(?P<n>[0-9]*)\|(?P<ps>.*)$'
2974 ftrace_line_fmt_fg = \
2975 '^ *(?P<time>[0-9\.]*) *\| *(?P<cpu>[0-9]*)\)'+\
2976 ' *(?P<proc>.*)-(?P<pid>[0-9]*) *\|'+\
2977 '[ +!#\*@$]*(?P<dur>[0-9\.]*) .*\| (?P<msg>.*)'
2978 ftrace_line_fmt_nop = \
2979 ' *(?P<proc>.*)-(?P<pid>[0-9]*) *\[(?P<cpu>[0-9]*)\] *'+\
2980 '(?P<flags>\S*) *(?P<time>[0-9\.]*): *'+\
2981 '(?P<msg>.*)'
2982 machinesuspend = 'machine_suspend\[.*'
2983 multiproclist = dict()
2984 multiproctime = 0.0
2985 multiproccnt = 0
2986 def __init__(self):
2987 self.stamp = ''
2988 self.sysinfo = ''
2989 self.cmdline = ''
2990 self.testerror = []
2991 self.turbostat = []
2992 self.wifi = []
2993 self.fwdata = []
2994 self.ftrace_line_fmt = self.ftrace_line_fmt_nop
2995 self.cgformat = False
2996 self.data = 0
2997 self.ktemp = dict()
2998 def setTracerType(self, tracer):
2999 if(tracer == 'function_graph'):
3000 self.cgformat = True
3001 self.ftrace_line_fmt = self.ftrace_line_fmt_fg
3002 elif(tracer == 'nop'):
3003 self.ftrace_line_fmt = self.ftrace_line_fmt_nop
3004 else:
3005 doError('Invalid tracer format: [%s]' % tracer)
3006 def stampInfo(self, line, sv):
3007 if re.match(self.stampfmt, line):
3008 self.stamp = line
3009 return True
3010 elif re.match(self.sysinfofmt, line):
3011 self.sysinfo = line
3012 return True
3013 elif re.match(self.tstatfmt, line):
3014 self.turbostat.append(line)
3015 return True
3016 elif re.match(self.wififmt, line):
3017 self.wifi.append(line)
3018 return True
3019 elif re.match(self.testerrfmt, line):
3020 self.testerror.append(line)
3021 return True
3022 elif re.match(self.firmwarefmt, line):
3023 self.fwdata.append(line)
3024 return True
3025 elif(re.match(self.devpropfmt, line)):
3026 self.parseDevprops(line, sv)
3027 return True
3028 elif(re.match(self.pinfofmt, line)):
3029 self.parsePlatformInfo(line, sv)
3030 return True
3031 m = re.match(self.cmdlinefmt, line)
3032 if m:
3033 self.cmdline = m.group('cmd')
3034 return True
3035 m = re.match(self.tracertypefmt, line)
3036 if(m):
3037 self.setTracerType(m.group('t'))
3038 return True
3039 return False
3040 def parseStamp(self, data, sv):
3041 # global test data
3042 m = re.match(self.stampfmt, self.stamp)
3043 if not self.stamp or not m:
3044 doError('data does not include the expected stamp')
3045 data.stamp = {'time': '', 'host': '', 'mode': ''}
3046 dt = datetime(int(m.group('y'))+2000, int(m.group('m')),
3047 int(m.group('d')), int(m.group('H')), int(m.group('M')),
3048 int(m.group('S')))
3049 data.stamp['time'] = dt.strftime('%B %d %Y, %I:%M:%S %p')
3050 data.stamp['host'] = m.group('host')
3051 data.stamp['mode'] = m.group('mode')
3052 data.stamp['kernel'] = m.group('kernel')
3053 if re.match(self.sysinfofmt, self.sysinfo):
3054 for f in self.sysinfo.split('|'):
3055 if '#' in f:
3056 continue
3057 tmp = f.strip().split(':', 1)
3058 key = tmp[0]
3059 val = tmp[1]
3060 data.stamp[key] = val
3061 sv.hostname = data.stamp['host']
3062 sv.suspendmode = data.stamp['mode']
3063 if sv.suspendmode == 'freeze':
3064 self.machinesuspend = 'timekeeping_freeze\[.*'
3065 else:
3066 self.machinesuspend = 'machine_suspend\[.*'
3067 if sv.suspendmode == 'command' and sv.ftracefile != '':
3068 modes = ['on', 'freeze', 'standby', 'mem', 'disk']
3069 fp = sv.openlog(sv.ftracefile, 'r')
3070 for line in fp:
3071 m = re.match('.* machine_suspend\[(?P<mode>.*)\]', line)
3072 if m and m.group('mode') in ['1', '2', '3', '4']:
3073 sv.suspendmode = modes[int(m.group('mode'))]
3074 data.stamp['mode'] = sv.suspendmode
3075 break
3076 fp.close()
3077 sv.cmdline = self.cmdline
3078 if not sv.stamp:
3079 sv.stamp = data.stamp
3080 # firmware data
3081 if sv.suspendmode == 'mem' and len(self.fwdata) > data.testnumber:
3082 m = re.match(self.firmwarefmt, self.fwdata[data.testnumber])
3083 if m:
3084 data.fwSuspend, data.fwResume = int(m.group('s')), int(m.group('r'))
3085 if(data.fwSuspend > 0 or data.fwResume > 0):
3086 data.fwValid = True
3087 # turbostat data
3088 if len(self.turbostat) > data.testnumber:
3089 m = re.match(self.tstatfmt, self.turbostat[data.testnumber])
3090 if m:
3091 data.turbostat = m.group('t')
3092 # wifi data
3093 if len(self.wifi) > data.testnumber:
3094 m = re.match(self.wififmt, self.wifi[data.testnumber])
3095 if m:
3096 data.wifi = {'dev': m.group('d'), 'stat': m.group('s'),
3097 'time': float(m.group('t'))}
3098 data.stamp['wifi'] = m.group('d')
3099 # sleep mode enter errors
3100 if len(self.testerror) > data.testnumber:
3101 m = re.match(self.testerrfmt, self.testerror[data.testnumber])
3102 if m:
3103 data.enterfail = m.group('e')
3104 def devprops(self, data):
3105 props = dict()
3106 devlist = data.split(';')
3107 for dev in devlist:
3108 f = dev.split(',')
3109 if len(f) < 3:
3110 continue
3111 dev = f[0]
3112 props[dev] = DevProps()
3113 props[dev].altname = f[1]
3114 if int(f[2]):
3115 props[dev].isasync = True
3116 else:
3117 props[dev].isasync = False
3118 return props
3119 def parseDevprops(self, line, sv):
3120 idx = line.index(': ') + 2
3121 if idx >= len(line):
3122 return
3123 props = self.devprops(line[idx:])
3124 if sv.suspendmode == 'command' and 'testcommandstring' in props:
3125 sv.testcommand = props['testcommandstring'].altname
3126 sv.devprops = props
3127 def parsePlatformInfo(self, line, sv):
3128 m = re.match(self.pinfofmt, line)
3129 if not m:
3130 return
3131 name, info = m.group('val'), m.group('info')
3132 if name == 'devinfo':
3133 sv.devprops = self.devprops(sv.b64unzip(info))
3134 return
3135 elif name == 'testcmd':
3136 sv.testcommand = info
3137 return
3138 field = info.split('|')
3139 if len(field) < 2:
3140 return
3141 cmdline = field[0].strip()
3142 output = sv.b64unzip(field[1].strip())
3143 sv.platinfo.append([name, cmdline, output])
3144
3145# Class: TestRun
3146# Description:
3147# A container for a suspend/resume test run. This is necessary as
3148# there could be more than one, and they need to be separate.
3149class TestRun:
3150 def __init__(self, dataobj):
3151 self.data = dataobj
3152 self.ftemp = dict()
3153 self.ttemp = dict()
3154
3155class ProcessMonitor:
3156 maxchars = 512
3157 def __init__(self):
3158 self.proclist = dict()
3159 self.running = False
3160 def procstat(self):
3161 c = ['cat /proc/[1-9]*/stat 2>/dev/null']
3162 process = Popen(c, shell=True, stdout=PIPE)
3163 running = dict()
3164 for line in process.stdout:
3165 data = ascii(line).split()
3166 pid = data[0]
3167 name = re.sub('[()]', '', data[1])
3168 user = int(data[13])
3169 kern = int(data[14])
3170 kjiff = ujiff = 0
3171 if pid not in self.proclist:
3172 self.proclist[pid] = {'name' : name, 'user' : user, 'kern' : kern}
3173 else:
3174 val = self.proclist[pid]
3175 ujiff = user - val['user']
3176 kjiff = kern - val['kern']
3177 val['user'] = user
3178 val['kern'] = kern
3179 if ujiff > 0 or kjiff > 0:
3180 running[pid] = ujiff + kjiff
3181 process.wait()
3182 out = ['']
3183 for pid in running:
3184 jiffies = running[pid]
3185 val = self.proclist[pid]
3186 if len(out[-1]) > self.maxchars:
3187 out.append('')
3188 elif len(out[-1]) > 0:
3189 out[-1] += ','
3190 out[-1] += '%s-%s %d' % (val['name'], pid, jiffies)
3191 if len(out) > 1:
3192 for line in out:
3193 sysvals.fsetVal('ps - @%d|%s' % (len(out), line), 'trace_marker')
3194 else:
3195 sysvals.fsetVal('ps - %s' % out[0], 'trace_marker')
3196 def processMonitor(self, tid):
3197 while self.running:
3198 self.procstat()
3199 def start(self):
3200 self.thread = Thread(target=self.processMonitor, args=(0,))
3201 self.running = True
3202 self.thread.start()
3203 def stop(self):
3204 self.running = False
3205
3206# ----------------- FUNCTIONS --------------------
3207
3208# Function: doesTraceLogHaveTraceEvents
3209# Description:
3210# Quickly determine if the ftrace log has all of the trace events,
3211# markers, and/or kprobes required for primary parsing.
3212def doesTraceLogHaveTraceEvents():
3213 kpcheck = ['_cal: (', '_ret: (']
3214 techeck = ['suspend_resume', 'device_pm_callback', 'tracing_mark_write']
3215 tmcheck = ['SUSPEND START', 'RESUME COMPLETE']
3216 sysvals.usekprobes = False
3217 fp = sysvals.openlog(sysvals.ftracefile, 'r')
3218 for line in fp:
3219 # check for kprobes
3220 if not sysvals.usekprobes:
3221 for i in kpcheck:
3222 if i in line:
3223 sysvals.usekprobes = True
3224 # check for all necessary trace events
3225 check = techeck[:]
3226 for i in techeck:
3227 if i in line:
3228 check.remove(i)
3229 techeck = check
3230 # check for all necessary trace markers
3231 check = tmcheck[:]
3232 for i in tmcheck:
3233 if i in line:
3234 check.remove(i)
3235 tmcheck = check
3236 fp.close()
3237 sysvals.usetraceevents = True if len(techeck) < 3 else False
3238 sysvals.usetracemarkers = True if len(tmcheck) == 0 else False
3239
3240# Function: appendIncompleteTraceLog
3241# Description:
3242# Adds callgraph data which lacks trace event data. This is only
3243# for timelines generated from 3.15 or older
3244# Arguments:
3245# testruns: the array of Data objects obtained from parseKernelLog
3246def appendIncompleteTraceLog(testruns):
3247 # create TestRun vessels for ftrace parsing
3248 testcnt = len(testruns)
3249 testidx = 0
3250 testrun = []
3251 for data in testruns:
3252 testrun.append(TestRun(data))
3253
3254 # extract the callgraph and traceevent data
3255 sysvals.vprint('Analyzing the ftrace data (%s)...' % \
3256 os.path.basename(sysvals.ftracefile))
3257 tp = TestProps()
3258 tf = sysvals.openlog(sysvals.ftracefile, 'r')
3259 data = 0
3260 for line in tf:
3261 # remove any latent carriage returns
3262 line = line.replace('\r\n', '')
3263 if tp.stampInfo(line, sysvals):
3264 continue
3265 # parse only valid lines, if this is not one move on
3266 m = re.match(tp.ftrace_line_fmt, line)
3267 if(not m):
3268 continue
3269 # gather the basic message data from the line
3270 m_time = m.group('time')
3271 m_pid = m.group('pid')
3272 m_msg = m.group('msg')
3273 if(tp.cgformat):
3274 m_param3 = m.group('dur')
3275 else:
3276 m_param3 = 'traceevent'
3277 if(m_time and m_pid and m_msg):
3278 t = FTraceLine(m_time, m_msg, m_param3)
3279 pid = int(m_pid)
3280 else:
3281 continue
3282 # the line should be a call, return, or event
3283 if(not t.fcall and not t.freturn and not t.fevent):
3284 continue
3285 # look for the suspend start marker
3286 if(t.startMarker()):
3287 data = testrun[testidx].data
3288 tp.parseStamp(data, sysvals)
3289 data.setStart(t.time, t.name)
3290 continue
3291 if(not data):
3292 continue
3293 # find the end of resume
3294 if(t.endMarker()):
3295 data.setEnd(t.time, t.name)
3296 testidx += 1
3297 if(testidx >= testcnt):
3298 break
3299 continue
3300 # trace event processing
3301 if(t.fevent):
3302 continue
3303 # call/return processing
3304 elif sysvals.usecallgraph:
3305 # create a callgraph object for the data
3306 if(pid not in testrun[testidx].ftemp):
3307 testrun[testidx].ftemp[pid] = []
3308 testrun[testidx].ftemp[pid].append(FTraceCallGraph(pid, sysvals))
3309 # when the call is finished, see which device matches it
3310 cg = testrun[testidx].ftemp[pid][-1]
3311 res = cg.addLine(t)
3312 if(res != 0):
3313 testrun[testidx].ftemp[pid].append(FTraceCallGraph(pid, sysvals))
3314 if(res == -1):
3315 testrun[testidx].ftemp[pid][-1].addLine(t)
3316 tf.close()
3317
3318 for test in testrun:
3319 # add the callgraph data to the device hierarchy
3320 for pid in test.ftemp:
3321 for cg in test.ftemp[pid]:
3322 if len(cg.list) < 1 or cg.invalid or (cg.end - cg.start == 0):
3323 continue
3324 if(not cg.postProcess()):
3325 id = 'task %s cpu %s' % (pid, m.group('cpu'))
3326 sysvals.vprint('Sanity check failed for '+\
3327 id+', ignoring this callback')
3328 continue
3329 callstart = cg.start
3330 callend = cg.end
3331 for p in test.data.sortedPhases():
3332 if(test.data.dmesg[p]['start'] <= callstart and
3333 callstart <= test.data.dmesg[p]['end']):
3334 list = test.data.dmesg[p]['list']
3335 for devname in list:
3336 dev = list[devname]
3337 if(pid == dev['pid'] and
3338 callstart <= dev['start'] and
3339 callend >= dev['end']):
3340 dev['ftrace'] = cg
3341 break
3342
3343# Function: loadTraceLog
3344# Description:
3345# load the ftrace file into memory and fix up any ordering issues
3346# Output:
3347# TestProps instance and an array of lines in proper order
3348def loadTraceLog():
3349 tp, data, lines, trace = TestProps(), dict(), [], []
3350 tf = sysvals.openlog(sysvals.ftracefile, 'r')
3351 for line in tf:
3352 # remove any latent carriage returns
3353 line = line.replace('\r\n', '')
3354 if tp.stampInfo(line, sysvals):
3355 continue
3356 # ignore all other commented lines
3357 if line[0] == '#':
3358 continue
3359 # ftrace line: parse only valid lines
3360 m = re.match(tp.ftrace_line_fmt, line)
3361 if(not m):
3362 continue
3363 dur = m.group('dur') if tp.cgformat else 'traceevent'
3364 info = (m.group('time'), m.group('proc'), m.group('pid'),
3365 m.group('msg'), dur)
3366 # group the data by timestamp
3367 t = float(info[0])
3368 if t in data:
3369 data[t].append(info)
3370 else:
3371 data[t] = [info]
3372 # we only care about trace event ordering
3373 if (info[3].startswith('suspend_resume:') or \
3374 info[3].startswith('tracing_mark_write:')) and t not in trace:
3375 trace.append(t)
3376 tf.close()
3377 for t in sorted(data):
3378 first, last, blk = [], [], data[t]
3379 if len(blk) > 1 and t in trace:
3380 # move certain lines to the start or end of a timestamp block
3381 for i in range(len(blk)):
3382 if 'SUSPEND START' in blk[i][3]:
3383 first.append(i)
3384 elif re.match('.* timekeeping_freeze.*begin', blk[i][3]):
3385 last.append(i)
3386 elif re.match('.* timekeeping_freeze.*end', blk[i][3]):
3387 first.append(i)
3388 elif 'RESUME COMPLETE' in blk[i][3]:
3389 last.append(i)
3390 if len(first) == 1 and len(last) == 0:
3391 blk.insert(0, blk.pop(first[0]))
3392 elif len(last) == 1 and len(first) == 0:
3393 blk.append(blk.pop(last[0]))
3394 for info in blk:
3395 lines.append(info)
3396 return (tp, lines)
3397
3398# Function: parseTraceLog
3399# Description:
3400# Analyze an ftrace log output file generated from this app during
3401# the execution phase. Used when the ftrace log is the primary data source
3402# and includes the suspend_resume and device_pm_callback trace events
3403# The ftrace filename is taken from sysvals
3404# Output:
3405# An array of Data objects
3406def parseTraceLog(live=False):
3407 sysvals.vprint('Analyzing the ftrace data (%s)...' % \
3408 os.path.basename(sysvals.ftracefile))
3409 if(os.path.exists(sysvals.ftracefile) == False):
3410 doError('%s does not exist' % sysvals.ftracefile)
3411 if not live:
3412 sysvals.setupAllKprobes()
3413 ksuscalls = ['ksys_sync', 'pm_prepare_console']
3414 krescalls = ['pm_restore_console']
3415 tracewatch = ['irq_wakeup']
3416 if sysvals.usekprobes:
3417 tracewatch += ['sync_filesystems', 'freeze_processes', 'syscore_suspend',
3418 'syscore_resume', 'resume_console', 'thaw_processes', 'CPU_ON',
3419 'CPU_OFF', 'acpi_suspend']
3420
3421 # extract the callgraph and traceevent data
3422 s2idle_enter = hwsus = False
3423 testruns, testdata = [], []
3424 testrun, data, limbo = 0, 0, True
3425 phase = 'suspend_prepare'
3426 tp, tf = loadTraceLog()
3427 for m_time, m_proc, m_pid, m_msg, m_param3 in tf:
3428 # gather the basic message data from the line
3429 if(m_time and m_pid and m_msg):
3430 t = FTraceLine(m_time, m_msg, m_param3)
3431 pid = int(m_pid)
3432 else:
3433 continue
3434 # the line should be a call, return, or event
3435 if(not t.fcall and not t.freturn and not t.fevent):
3436 continue
3437 # find the start of suspend
3438 if(t.startMarker()):
3439 data, limbo = Data(len(testdata)), False
3440 testdata.append(data)
3441 testrun = TestRun(data)
3442 testruns.append(testrun)
3443 tp.parseStamp(data, sysvals)
3444 data.setStart(t.time, t.name)
3445 data.first_suspend_prepare = True
3446 phase = data.setPhase('suspend_prepare', t.time, True)
3447 continue
3448 if(not data or limbo):
3449 continue
3450 # process cpu exec line
3451 if t.type == 'tracing_mark_write':
3452 if t.name == 'CMD COMPLETE' and data.tKernRes == 0:
3453 data.tKernRes = t.time
3454 m = re.match(tp.procexecfmt, t.name)
3455 if(m):
3456 parts, msg = 1, m.group('ps')
3457 m = re.match(tp.procmultifmt, msg)
3458 if(m):
3459 parts, msg = int(m.group('n')), m.group('ps')
3460 if tp.multiproccnt == 0:
3461 tp.multiproctime = t.time
3462 tp.multiproclist = dict()
3463 proclist = tp.multiproclist
3464 tp.multiproccnt += 1
3465 else:
3466 proclist = dict()
3467 tp.multiproccnt = 0
3468 for ps in msg.split(','):
3469 val = ps.split()
3470 if not val or len(val) != 2:
3471 continue
3472 name = val[0].replace('--', '-')
3473 proclist[name] = int(val[1])
3474 if parts == 1:
3475 data.pstl[t.time] = proclist
3476 elif parts == tp.multiproccnt:
3477 data.pstl[tp.multiproctime] = proclist
3478 tp.multiproccnt = 0
3479 continue
3480 # find the end of resume
3481 if(t.endMarker()):
3482 if data.tKernRes == 0:
3483 data.tKernRes = t.time
3484 data.handleEndMarker(t.time, t.name)
3485 if(not sysvals.usetracemarkers):
3486 # no trace markers? then quit and be sure to finish recording
3487 # the event we used to trigger resume end
3488 if('thaw_processes' in testrun.ttemp and len(testrun.ttemp['thaw_processes']) > 0):
3489 # if an entry exists, assume this is its end
3490 testrun.ttemp['thaw_processes'][-1]['end'] = t.time
3491 limbo = True
3492 continue
3493 # trace event processing
3494 if(t.fevent):
3495 if(t.type == 'suspend_resume'):
3496 # suspend_resume trace events have two types, begin and end
3497 if(re.match('(?P<name>.*) begin$', t.name)):
3498 isbegin = True
3499 elif(re.match('(?P<name>.*) end$', t.name)):
3500 isbegin = False
3501 else:
3502 continue
3503 if '[' in t.name:
3504 m = re.match('(?P<name>.*)\[.*', t.name)
3505 else:
3506 m = re.match('(?P<name>.*) .*', t.name)
3507 name = m.group('name')
3508 # ignore these events
3509 if(name.split('[')[0] in tracewatch):
3510 continue
3511 # -- phase changes --
3512 # start of kernel suspend
3513 if(re.match('suspend_enter\[.*', t.name)):
3514 if(isbegin and data.tKernSus == 0):
3515 data.tKernSus = t.time
3516 continue
3517 # suspend_prepare start
3518 elif(re.match('dpm_prepare\[.*', t.name)):
3519 if isbegin and data.first_suspend_prepare:
3520 data.first_suspend_prepare = False
3521 if data.tKernSus == 0:
3522 data.tKernSus = t.time
3523 continue
3524 phase = data.setPhase('suspend_prepare', t.time, isbegin)
3525 continue
3526 # suspend start
3527 elif(re.match('dpm_suspend\[.*', t.name)):
3528 phase = data.setPhase('suspend', t.time, isbegin)
3529 continue
3530 # suspend_late start
3531 elif(re.match('dpm_suspend_late\[.*', t.name)):
3532 phase = data.setPhase('suspend_late', t.time, isbegin)
3533 continue
3534 # suspend_noirq start
3535 elif(re.match('dpm_suspend_noirq\[.*', t.name)):
3536 phase = data.setPhase('suspend_noirq', t.time, isbegin)
3537 continue
3538 # suspend_machine/resume_machine
3539 elif(re.match(tp.machinesuspend, t.name)):
3540 lp = data.lastPhase()
3541 if(isbegin):
3542 hwsus = True
3543 if lp.startswith('resume_machine'):
3544 # trim out s2idle loops, track time trying to freeze
3545 llp = data.lastPhase(2)
3546 if llp.startswith('suspend_machine'):
3547 if 'waking' not in data.dmesg[llp]:
3548 data.dmesg[llp]['waking'] = [0, 0.0]
3549 data.dmesg[llp]['waking'][0] += 1
3550 data.dmesg[llp]['waking'][1] += \
3551 t.time - data.dmesg[lp]['start']
3552 data.currphase = ''
3553 del data.dmesg[lp]
3554 continue
3555 phase = data.setPhase('suspend_machine', data.dmesg[lp]['end'], True)
3556 data.setPhase(phase, t.time, False)
3557 if data.tSuspended == 0:
3558 data.tSuspended = t.time
3559 else:
3560 if lp.startswith('resume_machine'):
3561 data.dmesg[lp]['end'] = t.time
3562 continue
3563 phase = data.setPhase('resume_machine', t.time, True)
3564 if(sysvals.suspendmode in ['mem', 'disk']):
3565 susp = phase.replace('resume', 'suspend')
3566 if susp in data.dmesg:
3567 data.dmesg[susp]['end'] = t.time
3568 data.tSuspended = t.time
3569 data.tResumed = t.time
3570 continue
3571 # resume_noirq start
3572 elif(re.match('dpm_resume_noirq\[.*', t.name)):
3573 phase = data.setPhase('resume_noirq', t.time, isbegin)
3574 continue
3575 # resume_early start
3576 elif(re.match('dpm_resume_early\[.*', t.name)):
3577 phase = data.setPhase('resume_early', t.time, isbegin)
3578 continue
3579 # resume start
3580 elif(re.match('dpm_resume\[.*', t.name)):
3581 phase = data.setPhase('resume', t.time, isbegin)
3582 continue
3583 # resume complete start
3584 elif(re.match('dpm_complete\[.*', t.name)):
3585 phase = data.setPhase('resume_complete', t.time, isbegin)
3586 continue
3587 # skip trace events inside devices calls
3588 if(not data.isTraceEventOutsideDeviceCalls(pid, t.time)):
3589 continue
3590 # global events (outside device calls) are graphed
3591 if(name not in testrun.ttemp):
3592 testrun.ttemp[name] = []
3593 # special handling for s2idle_enter
3594 if name == 'machine_suspend':
3595 if hwsus:
3596 s2idle_enter = hwsus = False
3597 elif s2idle_enter and not isbegin:
3598 if(len(testrun.ttemp[name]) > 0):
3599 testrun.ttemp[name][-1]['end'] = t.time
3600 testrun.ttemp[name][-1]['loop'] += 1
3601 elif not s2idle_enter and isbegin:
3602 s2idle_enter = True
3603 testrun.ttemp[name].append({'begin': t.time,
3604 'end': t.time, 'pid': pid, 'loop': 0})
3605 continue
3606 if(isbegin):
3607 # create a new list entry
3608 testrun.ttemp[name].append(\
3609 {'begin': t.time, 'end': t.time, 'pid': pid})
3610 else:
3611 if(len(testrun.ttemp[name]) > 0):
3612 # if an entry exists, assume this is its end
3613 testrun.ttemp[name][-1]['end'] = t.time
3614 # device callback start
3615 elif(t.type == 'device_pm_callback_start'):
3616 if phase not in data.dmesg:
3617 continue
3618 m = re.match('(?P<drv>.*) (?P<d>.*), parent: *(?P<p>.*), .*',\
3619 t.name);
3620 if(not m):
3621 continue
3622 drv = m.group('drv')
3623 n = m.group('d')
3624 p = m.group('p')
3625 if(n and p):
3626 data.newAction(phase, n, pid, p, t.time, -1, drv)
3627 if pid not in data.devpids:
3628 data.devpids.append(pid)
3629 # device callback finish
3630 elif(t.type == 'device_pm_callback_end'):
3631 if phase not in data.dmesg:
3632 continue
3633 m = re.match('(?P<drv>.*) (?P<d>.*), err.*', t.name);
3634 if(not m):
3635 continue
3636 n = m.group('d')
3637 dev = data.findDevice(phase, n)
3638 if dev:
3639 dev['length'] = t.time - dev['start']
3640 dev['end'] = t.time
3641 # kprobe event processing
3642 elif(t.fkprobe):
3643 kprobename = t.type
3644 kprobedata = t.name
3645 key = (kprobename, pid)
3646 # displayname is generated from kprobe data
3647 displayname = ''
3648 if(t.fcall):
3649 displayname = sysvals.kprobeDisplayName(kprobename, kprobedata)
3650 if not displayname:
3651 continue
3652 if(key not in tp.ktemp):
3653 tp.ktemp[key] = []
3654 tp.ktemp[key].append({
3655 'pid': pid,
3656 'begin': t.time,
3657 'end': -1,
3658 'name': displayname,
3659 'cdata': kprobedata,
3660 'proc': m_proc,
3661 })
3662 # start of kernel resume
3663 if(data.tKernSus == 0 and phase == 'suspend_prepare' \
3664 and kprobename in ksuscalls):
3665 data.tKernSus = t.time
3666 elif(t.freturn):
3667 if(key not in tp.ktemp) or len(tp.ktemp[key]) < 1:
3668 continue
3669 e = next((x for x in reversed(tp.ktemp[key]) if x['end'] < 0), 0)
3670 if not e:
3671 continue
3672 if (t.time - e['begin']) * 1000 < sysvals.mindevlen:
3673 tp.ktemp[key].pop()
3674 continue
3675 e['end'] = t.time
3676 e['rdata'] = kprobedata
3677 # end of kernel resume
3678 if(phase != 'suspend_prepare' and kprobename in krescalls):
3679 if phase in data.dmesg:
3680 data.dmesg[phase]['end'] = t.time
3681 data.tKernRes = t.time
3682
3683 # callgraph processing
3684 elif sysvals.usecallgraph:
3685 # create a callgraph object for the data
3686 key = (m_proc, pid)
3687 if(key not in testrun.ftemp):
3688 testrun.ftemp[key] = []
3689 testrun.ftemp[key].append(FTraceCallGraph(pid, sysvals))
3690 # when the call is finished, see which device matches it
3691 cg = testrun.ftemp[key][-1]
3692 res = cg.addLine(t)
3693 if(res != 0):
3694 testrun.ftemp[key].append(FTraceCallGraph(pid, sysvals))
3695 if(res == -1):
3696 testrun.ftemp[key][-1].addLine(t)
3697 if len(testdata) < 1:
3698 sysvals.vprint('WARNING: ftrace start marker is missing')
3699 if data and not data.devicegroups:
3700 sysvals.vprint('WARNING: ftrace end marker is missing')
3701 data.handleEndMarker(t.time, t.name)
3702
3703 if sysvals.suspendmode == 'command':
3704 for test in testruns:
3705 for p in test.data.sortedPhases():
3706 if p == 'suspend_prepare':
3707 test.data.dmesg[p]['start'] = test.data.start
3708 test.data.dmesg[p]['end'] = test.data.end
3709 else:
3710 test.data.dmesg[p]['start'] = test.data.end
3711 test.data.dmesg[p]['end'] = test.data.end
3712 test.data.tSuspended = test.data.end
3713 test.data.tResumed = test.data.end
3714 test.data.fwValid = False
3715
3716 # dev source and procmon events can be unreadable with mixed phase height
3717 if sysvals.usedevsrc or sysvals.useprocmon:
3718 sysvals.mixedphaseheight = False
3719
3720 # expand phase boundaries so there are no gaps
3721 for data in testdata:
3722 lp = data.sortedPhases()[0]
3723 for p in data.sortedPhases():
3724 if(p != lp and not ('machine' in p and 'machine' in lp)):
3725 data.dmesg[lp]['end'] = data.dmesg[p]['start']
3726 lp = p
3727
3728 for i in range(len(testruns)):
3729 test = testruns[i]
3730 data = test.data
3731 # find the total time range for this test (begin, end)
3732 tlb, tle = data.start, data.end
3733 if i < len(testruns) - 1:
3734 tle = testruns[i+1].data.start
3735 # add the process usage data to the timeline
3736 if sysvals.useprocmon:
3737 data.createProcessUsageEvents()
3738 # add the traceevent data to the device hierarchy
3739 if(sysvals.usetraceevents):
3740 # add actual trace funcs
3741 for name in sorted(test.ttemp):
3742 for event in test.ttemp[name]:
3743 if event['end'] - event['begin'] <= 0:
3744 continue
3745 title = name
3746 if name == 'machine_suspend' and 'loop' in event:
3747 title = 's2idle_enter_%dx' % event['loop']
3748 data.newActionGlobal(title, event['begin'], event['end'], event['pid'])
3749 # add the kprobe based virtual tracefuncs as actual devices
3750 for key in sorted(tp.ktemp):
3751 name, pid = key
3752 if name not in sysvals.tracefuncs:
3753 continue
3754 if pid not in data.devpids:
3755 data.devpids.append(pid)
3756 for e in tp.ktemp[key]:
3757 kb, ke = e['begin'], e['end']
3758 if ke - kb < 0.000001 or tlb > kb or tle <= kb:
3759 continue
3760 color = sysvals.kprobeColor(name)
3761 data.newActionGlobal(e['name'], kb, ke, pid, color)
3762 # add config base kprobes and dev kprobes
3763 if sysvals.usedevsrc:
3764 for key in sorted(tp.ktemp):
3765 name, pid = key
3766 if name in sysvals.tracefuncs or name not in sysvals.dev_tracefuncs:
3767 continue
3768 for e in tp.ktemp[key]:
3769 kb, ke = e['begin'], e['end']
3770 if ke - kb < 0.000001 or tlb > kb or tle <= kb:
3771 continue
3772 data.addDeviceFunctionCall(e['name'], name, e['proc'], pid, kb,
3773 ke, e['cdata'], e['rdata'])
3774 if sysvals.usecallgraph:
3775 # add the callgraph data to the device hierarchy
3776 sortlist = dict()
3777 for key in sorted(test.ftemp):
3778 proc, pid = key
3779 for cg in test.ftemp[key]:
3780 if len(cg.list) < 1 or cg.invalid or (cg.end - cg.start == 0):
3781 continue
3782 if(not cg.postProcess()):
3783 id = 'task %s' % (pid)
3784 sysvals.vprint('Sanity check failed for '+\
3785 id+', ignoring this callback')
3786 continue
3787 # match cg data to devices
3788 devname = ''
3789 if sysvals.suspendmode != 'command':
3790 devname = cg.deviceMatch(pid, data)
3791 if not devname:
3792 sortkey = '%f%f%d' % (cg.start, cg.end, pid)
3793 sortlist[sortkey] = cg
3794 elif len(cg.list) > 1000000 and cg.name != sysvals.ftopfunc:
3795 sysvals.vprint('WARNING: the callgraph for %s is massive (%d lines)' %\
3796 (devname, len(cg.list)))
3797 # create blocks for orphan cg data
3798 for sortkey in sorted(sortlist):
3799 cg = sortlist[sortkey]
3800 name = cg.name
3801 if sysvals.isCallgraphFunc(name):
3802 sysvals.vprint('Callgraph found for task %d: %.3fms, %s' % (cg.pid, (cg.end - cg.start)*1000, name))
3803 cg.newActionFromFunction(data)
3804 if sysvals.suspendmode == 'command':
3805 return (testdata, '')
3806
3807 # fill in any missing phases
3808 error = []
3809 for data in testdata:
3810 tn = '' if len(testdata) == 1 else ('%d' % (data.testnumber + 1))
3811 terr = ''
3812 phasedef = data.phasedef
3813 lp = 'suspend_prepare'
3814 for p in sorted(phasedef, key=lambda k:phasedef[k]['order']):
3815 if p not in data.dmesg:
3816 if not terr:
3817 ph = p if 'machine' in p else lp
3818 if p == 'suspend_machine':
3819 sm = sysvals.suspendmode
3820 if sm in suspendmodename:
3821 sm = suspendmodename[sm]
3822 terr = 'test%s did not enter %s power mode' % (tn, sm)
3823 else:
3824 terr = '%s%s failed in %s phase' % (sysvals.suspendmode, tn, ph)
3825 pprint('TEST%s FAILED: %s' % (tn, terr))
3826 error.append(terr)
3827 if data.tSuspended == 0:
3828 data.tSuspended = data.dmesg[lp]['end']
3829 if data.tResumed == 0:
3830 data.tResumed = data.dmesg[lp]['end']
3831 data.fwValid = False
3832 sysvals.vprint('WARNING: phase "%s" is missing!' % p)
3833 lp = p
3834 if not terr and 'dev' in data.wifi and data.wifi['stat'] == 'timeout':
3835 terr = '%s%s failed in wifi_resume <i>(%s %.0fs timeout)</i>' % \
3836 (sysvals.suspendmode, tn, data.wifi['dev'], data.wifi['time'])
3837 error.append(terr)
3838 if not terr and data.enterfail:
3839 pprint('test%s FAILED: enter %s failed with %s' % (tn, sysvals.suspendmode, data.enterfail))
3840 terr = 'test%s failed to enter %s mode' % (tn, sysvals.suspendmode)
3841 error.append(terr)
3842 if data.tSuspended == 0:
3843 data.tSuspended = data.tKernRes
3844 if data.tResumed == 0:
3845 data.tResumed = data.tSuspended
3846
3847 if(len(sysvals.devicefilter) > 0):
3848 data.deviceFilter(sysvals.devicefilter)
3849 data.fixupInitcallsThatDidntReturn()
3850 if sysvals.usedevsrc:
3851 data.optimizeDevSrc()
3852
3853 # x2: merge any overlapping devices between test runs
3854 if sysvals.usedevsrc and len(testdata) > 1:
3855 tc = len(testdata)
3856 for i in range(tc - 1):
3857 devlist = testdata[i].overflowDevices()
3858 for j in range(i + 1, tc):
3859 testdata[j].mergeOverlapDevices(devlist)
3860 testdata[0].stitchTouchingThreads(testdata[1:])
3861 return (testdata, ', '.join(error))
3862
3863# Function: loadKernelLog
3864# Description:
3865# load the dmesg file into memory and fix up any ordering issues
3866# Output:
3867# An array of empty Data objects with only their dmesgtext attributes set
3868def loadKernelLog():
3869 sysvals.vprint('Analyzing the dmesg data (%s)...' % \
3870 os.path.basename(sysvals.dmesgfile))
3871 if(os.path.exists(sysvals.dmesgfile) == False):
3872 doError('%s does not exist' % sysvals.dmesgfile)
3873
3874 # there can be multiple test runs in a single file
3875 tp = TestProps()
3876 tp.stamp = datetime.now().strftime('# suspend-%m%d%y-%H%M%S localhost mem unknown')
3877 testruns = []
3878 data = 0
3879 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
3880 for line in lf:
3881 line = line.replace('\r\n', '')
3882 idx = line.find('[')
3883 if idx > 1:
3884 line = line[idx:]
3885 if tp.stampInfo(line, sysvals):
3886 continue
3887 m = re.match('[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
3888 if(not m):
3889 continue
3890 msg = m.group("msg")
3891 if re.match('PM: Syncing filesystems.*', msg) or \
3892 re.match('PM: suspend entry.*', msg):
3893 if(data):
3894 testruns.append(data)
3895 data = Data(len(testruns))
3896 tp.parseStamp(data, sysvals)
3897 if(not data):
3898 continue
3899 m = re.match('.* *(?P<k>[0-9]\.[0-9]{2}\.[0-9]-.*) .*', msg)
3900 if(m):
3901 sysvals.stamp['kernel'] = m.group('k')
3902 m = re.match('PM: Preparing system for (?P<m>.*) sleep', msg)
3903 if not m:
3904 m = re.match('PM: Preparing system for sleep \((?P<m>.*)\)', msg)
3905 if m:
3906 sysvals.stamp['mode'] = sysvals.suspendmode = m.group('m')
3907 data.dmesgtext.append(line)
3908 lf.close()
3909
3910 if sysvals.suspendmode == 's2idle':
3911 sysvals.suspendmode = 'freeze'
3912 elif sysvals.suspendmode == 'deep':
3913 sysvals.suspendmode = 'mem'
3914 if data:
3915 testruns.append(data)
3916 if len(testruns) < 1:
3917 doError('dmesg log has no suspend/resume data: %s' \
3918 % sysvals.dmesgfile)
3919
3920 # fix lines with same timestamp/function with the call and return swapped
3921 for data in testruns:
3922 last = ''
3923 for line in data.dmesgtext:
3924 ct, cf, n, p = data.initcall_debug_call(line)
3925 rt, rf, l = data.initcall_debug_return(last)
3926 if ct and rt and ct == rt and cf == rf:
3927 i = data.dmesgtext.index(last)
3928 j = data.dmesgtext.index(line)
3929 data.dmesgtext[i] = line
3930 data.dmesgtext[j] = last
3931 last = line
3932 return testruns
3933
3934# Function: parseKernelLog
3935# Description:
3936# Analyse a dmesg log output file generated from this app during
3937# the execution phase. Create a set of device structures in memory
3938# for subsequent formatting in the html output file
3939# This call is only for legacy support on kernels where the ftrace
3940# data lacks the suspend_resume or device_pm_callbacks trace events.
3941# Arguments:
3942# data: an empty Data object (with dmesgtext) obtained from loadKernelLog
3943# Output:
3944# The filled Data object
3945def parseKernelLog(data):
3946 phase = 'suspend_runtime'
3947
3948 if(data.fwValid):
3949 sysvals.vprint('Firmware Suspend = %u ns, Firmware Resume = %u ns' % \
3950 (data.fwSuspend, data.fwResume))
3951
3952 # dmesg phase match table
3953 dm = {
3954 'suspend_prepare': ['PM: Syncing filesystems.*', 'PM: suspend entry.*'],
3955 'suspend': ['PM: Entering [a-z]* sleep.*', 'Suspending console.*',
3956 'PM: Suspending system .*'],
3957 'suspend_late': ['PM: suspend of devices complete after.*',
3958 'PM: freeze of devices complete after.*'],
3959 'suspend_noirq': ['PM: late suspend of devices complete after.*',
3960 'PM: late freeze of devices complete after.*'],
3961 'suspend_machine': ['PM: suspend-to-idle',
3962 'PM: noirq suspend of devices complete after.*',
3963 'PM: noirq freeze of devices complete after.*'],
3964 'resume_machine': ['PM: Timekeeping suspended for.*',
3965 'ACPI: Low-level resume complete.*',
3966 'ACPI: resume from mwait',
3967 'Suspended for [0-9\.]* seconds'],
3968 'resume_noirq': ['PM: resume from suspend-to-idle',
3969 'ACPI: Waking up from system sleep state.*'],
3970 'resume_early': ['PM: noirq resume of devices complete after.*',
3971 'PM: noirq restore of devices complete after.*'],
3972 'resume': ['PM: early resume of devices complete after.*',
3973 'PM: early restore of devices complete after.*'],
3974 'resume_complete': ['PM: resume of devices complete after.*',
3975 'PM: restore of devices complete after.*'],
3976 'post_resume': ['.*Restarting tasks \.\.\..*'],
3977 }
3978
3979 # action table (expected events that occur and show up in dmesg)
3980 at = {
3981 'sync_filesystems': {
3982 'smsg': 'PM: Syncing filesystems.*',
3983 'emsg': 'PM: Preparing system for mem sleep.*' },
3984 'freeze_user_processes': {
3985 'smsg': 'Freezing user space processes .*',
3986 'emsg': 'Freezing remaining freezable tasks.*' },
3987 'freeze_tasks': {
3988 'smsg': 'Freezing remaining freezable tasks.*',
3989 'emsg': 'PM: Entering (?P<mode>[a-z,A-Z]*) sleep.*' },
3990 'ACPI prepare': {
3991 'smsg': 'ACPI: Preparing to enter system sleep state.*',
3992 'emsg': 'PM: Saving platform NVS memory.*' },
3993 'PM vns': {
3994 'smsg': 'PM: Saving platform NVS memory.*',
3995 'emsg': 'Disabling non-boot CPUs .*' },
3996 }
3997
3998 t0 = -1.0
3999 cpu_start = -1.0
4000 prevktime = -1.0
4001 actions = dict()
4002 for line in data.dmesgtext:
4003 # parse each dmesg line into the time and message
4004 m = re.match('[ \t]*(\[ *)(?P<ktime>[0-9\.]*)(\]) (?P<msg>.*)', line)
4005 if(m):
4006 val = m.group('ktime')
4007 try:
4008 ktime = float(val)
4009 except:
4010 continue
4011 msg = m.group('msg')
4012 # initialize data start to first line time
4013 if t0 < 0:
4014 data.setStart(ktime)
4015 t0 = ktime
4016 else:
4017 continue
4018
4019 # check for a phase change line
4020 phasechange = False
4021 for p in dm:
4022 for s in dm[p]:
4023 if(re.match(s, msg)):
4024 phasechange, phase = True, p
4025 dm[p] = [s]
4026 break
4027
4028 # hack for determining resume_machine end for freeze
4029 if(not sysvals.usetraceevents and sysvals.suspendmode == 'freeze' \
4030 and phase == 'resume_machine' and \
4031 data.initcall_debug_call(line, True)):
4032 data.setPhase(phase, ktime, False)
4033 phase = 'resume_noirq'
4034 data.setPhase(phase, ktime, True)
4035
4036 if phasechange:
4037 if phase == 'suspend_prepare':
4038 data.setPhase(phase, ktime, True)
4039 data.setStart(ktime)
4040 data.tKernSus = ktime
4041 elif phase == 'suspend':
4042 lp = data.lastPhase()
4043 if lp:
4044 data.setPhase(lp, ktime, False)
4045 data.setPhase(phase, ktime, True)
4046 elif phase == 'suspend_late':
4047 lp = data.lastPhase()
4048 if lp:
4049 data.setPhase(lp, ktime, False)
4050 data.setPhase(phase, ktime, True)
4051 elif phase == 'suspend_noirq':
4052 lp = data.lastPhase()
4053 if lp:
4054 data.setPhase(lp, ktime, False)
4055 data.setPhase(phase, ktime, True)
4056 elif phase == 'suspend_machine':
4057 lp = data.lastPhase()
4058 if lp:
4059 data.setPhase(lp, ktime, False)
4060 data.setPhase(phase, ktime, True)
4061 elif phase == 'resume_machine':
4062 lp = data.lastPhase()
4063 if(sysvals.suspendmode in ['freeze', 'standby']):
4064 data.tSuspended = prevktime
4065 if lp:
4066 data.setPhase(lp, prevktime, False)
4067 else:
4068 data.tSuspended = ktime
4069 if lp:
4070 data.setPhase(lp, prevktime, False)
4071 data.tResumed = ktime
4072 data.setPhase(phase, ktime, True)
4073 elif phase == 'resume_noirq':
4074 lp = data.lastPhase()
4075 if lp:
4076 data.setPhase(lp, ktime, False)
4077 data.setPhase(phase, ktime, True)
4078 elif phase == 'resume_early':
4079 lp = data.lastPhase()
4080 if lp:
4081 data.setPhase(lp, ktime, False)
4082 data.setPhase(phase, ktime, True)
4083 elif phase == 'resume':
4084 lp = data.lastPhase()
4085 if lp:
4086 data.setPhase(lp, ktime, False)
4087 data.setPhase(phase, ktime, True)
4088 elif phase == 'resume_complete':
4089 lp = data.lastPhase()
4090 if lp:
4091 data.setPhase(lp, ktime, False)
4092 data.setPhase(phase, ktime, True)
4093 elif phase == 'post_resume':
4094 lp = data.lastPhase()
4095 if lp:
4096 data.setPhase(lp, ktime, False)
4097 data.setEnd(ktime)
4098 data.tKernRes = ktime
4099 break
4100
4101 # -- device callbacks --
4102 if(phase in data.sortedPhases()):
4103 # device init call
4104 t, f, n, p = data.initcall_debug_call(line)
4105 if t and f and n and p:
4106 data.newAction(phase, f, int(n), p, ktime, -1, '')
4107 else:
4108 # device init return
4109 t, f, l = data.initcall_debug_return(line)
4110 if t and f and l:
4111 list = data.dmesg[phase]['list']
4112 if(f in list):
4113 dev = list[f]
4114 dev['length'] = int(l)
4115 dev['end'] = ktime
4116
4117 # if trace events are not available, these are better than nothing
4118 if(not sysvals.usetraceevents):
4119 # look for known actions
4120 for a in sorted(at):
4121 if(re.match(at[a]['smsg'], msg)):
4122 if(a not in actions):
4123 actions[a] = []
4124 actions[a].append({'begin': ktime, 'end': ktime})
4125 if(re.match(at[a]['emsg'], msg)):
4126 if(a in actions):
4127 actions[a][-1]['end'] = ktime
4128 # now look for CPU on/off events
4129 if(re.match('Disabling non-boot CPUs .*', msg)):
4130 # start of first cpu suspend
4131 cpu_start = ktime
4132 elif(re.match('Enabling non-boot CPUs .*', msg)):
4133 # start of first cpu resume
4134 cpu_start = ktime
4135 elif(re.match('smpboot: CPU (?P<cpu>[0-9]*) is now offline', msg)):
4136 # end of a cpu suspend, start of the next
4137 m = re.match('smpboot: CPU (?P<cpu>[0-9]*) is now offline', msg)
4138 cpu = 'CPU'+m.group('cpu')
4139 if(cpu not in actions):
4140 actions[cpu] = []
4141 actions[cpu].append({'begin': cpu_start, 'end': ktime})
4142 cpu_start = ktime
4143 elif(re.match('CPU(?P<cpu>[0-9]*) is up', msg)):
4144 # end of a cpu resume, start of the next
4145 m = re.match('CPU(?P<cpu>[0-9]*) is up', msg)
4146 cpu = 'CPU'+m.group('cpu')
4147 if(cpu not in actions):
4148 actions[cpu] = []
4149 actions[cpu].append({'begin': cpu_start, 'end': ktime})
4150 cpu_start = ktime
4151 prevktime = ktime
4152 data.initDevicegroups()
4153
4154 # fill in any missing phases
4155 phasedef = data.phasedef
4156 terr, lp = '', 'suspend_prepare'
4157 if lp not in data.dmesg:
4158 doError('dmesg log format has changed, could not find start of suspend')
4159 for p in sorted(phasedef, key=lambda k:phasedef[k]['order']):
4160 if p not in data.dmesg:
4161 if not terr:
4162 pprint('TEST FAILED: %s failed in %s phase' % (sysvals.suspendmode, lp))
4163 terr = '%s failed in %s phase' % (sysvals.suspendmode, lp)
4164 if data.tSuspended == 0:
4165 data.tSuspended = data.dmesg[lp]['end']
4166 if data.tResumed == 0:
4167 data.tResumed = data.dmesg[lp]['end']
4168 sysvals.vprint('WARNING: phase "%s" is missing!' % p)
4169 lp = p
4170 lp = data.sortedPhases()[0]
4171 for p in data.sortedPhases():
4172 if(p != lp and not ('machine' in p and 'machine' in lp)):
4173 data.dmesg[lp]['end'] = data.dmesg[p]['start']
4174 lp = p
4175 if data.tSuspended == 0:
4176 data.tSuspended = data.tKernRes
4177 if data.tResumed == 0:
4178 data.tResumed = data.tSuspended
4179
4180 # fill in any actions we've found
4181 for name in sorted(actions):
4182 for event in actions[name]:
4183 data.newActionGlobal(name, event['begin'], event['end'])
4184
4185 if(len(sysvals.devicefilter) > 0):
4186 data.deviceFilter(sysvals.devicefilter)
4187 data.fixupInitcallsThatDidntReturn()
4188 return True
4189
4190def callgraphHTML(sv, hf, num, cg, title, color, devid):
4191 html_func_top = '<article id="{0}" class="atop" style="background:{1}">\n<input type="checkbox" class="pf" id="f{2}" checked/><label for="f{2}">{3} {4}</label>\n'
4192 html_func_start = '<article>\n<input type="checkbox" class="pf" id="f{0}" checked/><label for="f{0}">{1} {2}</label>\n'
4193 html_func_end = '</article>\n'
4194 html_func_leaf = '<article>{0} {1}</article>\n'
4195
4196 cgid = devid
4197 if cg.id:
4198 cgid += cg.id
4199 cglen = (cg.end - cg.start) * 1000
4200 if cglen < sv.mincglen:
4201 return num
4202
4203 fmt = '<r>(%.3f ms @ '+sv.timeformat+' to '+sv.timeformat+')</r>'
4204 flen = fmt % (cglen, cg.start, cg.end)
4205 hf.write(html_func_top.format(cgid, color, num, title, flen))
4206 num += 1
4207 for line in cg.list:
4208 if(line.length < 0.000000001):
4209 flen = ''
4210 else:
4211 fmt = '<n>(%.3f ms @ '+sv.timeformat+')</n>'
4212 flen = fmt % (line.length*1000, line.time)
4213 if line.isLeaf():
4214 if line.length * 1000 < sv.mincglen:
4215 continue
4216 hf.write(html_func_leaf.format(line.name, flen))
4217 elif line.freturn:
4218 hf.write(html_func_end)
4219 else:
4220 hf.write(html_func_start.format(num, line.name, flen))
4221 num += 1
4222 hf.write(html_func_end)
4223 return num
4224
4225def addCallgraphs(sv, hf, data):
4226 hf.write('<section id="callgraphs" class="callgraph">\n')
4227 # write out the ftrace data converted to html
4228 num = 0
4229 for p in data.sortedPhases():
4230 if sv.cgphase and p != sv.cgphase:
4231 continue
4232 list = data.dmesg[p]['list']
4233 for d in data.sortedDevices(p):
4234 if len(sv.cgfilter) > 0 and d not in sv.cgfilter:
4235 continue
4236 dev = list[d]
4237 color = 'white'
4238 if 'color' in data.dmesg[p]:
4239 color = data.dmesg[p]['color']
4240 if 'color' in dev:
4241 color = dev['color']
4242 name = d if '[' not in d else d.split('[')[0]
4243 if(d in sv.devprops):
4244 name = sv.devprops[d].altName(d)
4245 if 'drv' in dev and dev['drv']:
4246 name += ' {%s}' % dev['drv']
4247 if sv.suspendmode in suspendmodename:
4248 name += ' '+p
4249 if('ftrace' in dev):
4250 cg = dev['ftrace']
4251 if cg.name == sv.ftopfunc:
4252 name = 'top level suspend/resume call'
4253 num = callgraphHTML(sv, hf, num, cg,
4254 name, color, dev['id'])
4255 if('ftraces' in dev):
4256 for cg in dev['ftraces']:
4257 num = callgraphHTML(sv, hf, num, cg,
4258 name+' → '+cg.name, color, dev['id'])
4259 hf.write('\n\n </section>\n')
4260
4261def summaryCSS(title, center=True):
4262 tdcenter = 'text-align:center;' if center else ''
4263 out = '<!DOCTYPE html>\n<html>\n<head>\n\
4264 <meta http-equiv="content-type" content="text/html; charset=UTF-8">\n\
4265 <title>'+title+'</title>\n\
4266 <style type=\'text/css\'>\n\
4267 .stamp {width: 100%;text-align:center;background:#888;line-height:30px;color:white;font: 25px Arial;}\n\
4268 table {width:100%;border-collapse: collapse;border:1px solid;}\n\
4269 th {border: 1px solid black;background:#222;color:white;}\n\
4270 td {font: 14px "Times New Roman";'+tdcenter+'}\n\
4271 tr.head td {border: 1px solid black;background:#aaa;}\n\
4272 tr.alt {background-color:#ddd;}\n\
4273 tr.notice {color:red;}\n\
4274 .minval {background-color:#BBFFBB;}\n\
4275 .medval {background-color:#BBBBFF;}\n\
4276 .maxval {background-color:#FFBBBB;}\n\
4277 .head a {color:#000;text-decoration: none;}\n\
4278 </style>\n</head>\n<body>\n'
4279 return out
4280
4281# Function: createHTMLSummarySimple
4282# Description:
4283# Create summary html file for a series of tests
4284# Arguments:
4285# testruns: array of Data objects from parseTraceLog
4286def createHTMLSummarySimple(testruns, htmlfile, title):
4287 # write the html header first (html head, css code, up to body start)
4288 html = summaryCSS('Summary - SleepGraph')
4289
4290 # extract the test data into list
4291 list = dict()
4292 tAvg, tMin, tMax, tMed = [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [dict(), dict()]
4293 iMin, iMed, iMax = [0, 0], [0, 0], [0, 0]
4294 num = 0
4295 useturbo = usewifi = False
4296 lastmode = ''
4297 cnt = dict()
4298 for data in sorted(testruns, key=lambda v:(v['mode'], v['host'], v['kernel'], v['time'])):
4299 mode = data['mode']
4300 if mode not in list:
4301 list[mode] = {'data': [], 'avg': [0,0], 'min': [0,0], 'max': [0,0], 'med': [0,0]}
4302 if lastmode and lastmode != mode and num > 0:
4303 for i in range(2):
4304 s = sorted(tMed[i])
4305 list[lastmode]['med'][i] = s[int(len(s)//2)]
4306 iMed[i] = tMed[i][list[lastmode]['med'][i]]
4307 list[lastmode]['avg'] = [tAvg[0] / num, tAvg[1] / num]
4308 list[lastmode]['min'] = tMin
4309 list[lastmode]['max'] = tMax
4310 list[lastmode]['idx'] = (iMin, iMed, iMax)
4311 tAvg, tMin, tMax, tMed = [0.0, 0.0], [0.0, 0.0], [0.0, 0.0], [dict(), dict()]
4312 iMin, iMed, iMax = [0, 0], [0, 0], [0, 0]
4313 num = 0
4314 pkgpc10 = syslpi = wifi = ''
4315 if 'pkgpc10' in data and 'syslpi' in data:
4316 pkgpc10, syslpi, useturbo = data['pkgpc10'], data['syslpi'], True
4317 if 'wifi' in data:
4318 wifi, usewifi = data['wifi'], True
4319 res = data['result']
4320 tVal = [float(data['suspend']), float(data['resume'])]
4321 list[mode]['data'].append([data['host'], data['kernel'],
4322 data['time'], tVal[0], tVal[1], data['url'], res,
4323 data['issues'], data['sus_worst'], data['sus_worsttime'],
4324 data['res_worst'], data['res_worsttime'], pkgpc10, syslpi, wifi])
4325 idx = len(list[mode]['data']) - 1
4326 if res.startswith('fail in'):
4327 res = 'fail'
4328 if res not in cnt:
4329 cnt[res] = 1
4330 else:
4331 cnt[res] += 1
4332 if res == 'pass':
4333 for i in range(2):
4334 tMed[i][tVal[i]] = idx
4335 tAvg[i] += tVal[i]
4336 if tMin[i] == 0 or tVal[i] < tMin[i]:
4337 iMin[i] = idx
4338 tMin[i] = tVal[i]
4339 if tMax[i] == 0 or tVal[i] > tMax[i]:
4340 iMax[i] = idx
4341 tMax[i] = tVal[i]
4342 num += 1
4343 lastmode = mode
4344 if lastmode and num > 0:
4345 for i in range(2):
4346 s = sorted(tMed[i])
4347 list[lastmode]['med'][i] = s[int(len(s)//2)]
4348 iMed[i] = tMed[i][list[lastmode]['med'][i]]
4349 list[lastmode]['avg'] = [tAvg[0] / num, tAvg[1] / num]
4350 list[lastmode]['min'] = tMin
4351 list[lastmode]['max'] = tMax
4352 list[lastmode]['idx'] = (iMin, iMed, iMax)
4353
4354 # group test header
4355 desc = []
4356 for ilk in sorted(cnt, reverse=True):
4357 if cnt[ilk] > 0:
4358 desc.append('%d %s' % (cnt[ilk], ilk))
4359 html += '<div class="stamp">%s (%d tests: %s)</div>\n' % (title, len(testruns), ', '.join(desc))
4360 th = '\t<th>{0}</th>\n'
4361 td = '\t<td>{0}</td>\n'
4362 tdh = '\t<td{1}>{0}</td>\n'
4363 tdlink = '\t<td><a href="{0}">html</a></td>\n'
4364 cols = 12
4365 if useturbo:
4366 cols += 2
4367 if usewifi:
4368 cols += 1
4369 colspan = '%d' % cols
4370
4371 # table header
4372 html += '<table>\n<tr>\n' + th.format('#') +\
4373 th.format('Mode') + th.format('Host') + th.format('Kernel') +\
4374 th.format('Test Time') + th.format('Result') + th.format('Issues') +\
4375 th.format('Suspend') + th.format('Resume') +\
4376 th.format('Worst Suspend Device') + th.format('SD Time') +\
4377 th.format('Worst Resume Device') + th.format('RD Time')
4378 if useturbo:
4379 html += th.format('PkgPC10') + th.format('SysLPI')
4380 if usewifi:
4381 html += th.format('Wifi')
4382 html += th.format('Detail')+'</tr>\n'
4383 # export list into html
4384 head = '<tr class="head"><td>{0}</td><td>{1}</td>'+\
4385 '<td colspan='+colspan+' class="sus">Suspend Avg={2} '+\
4386 '<span class=minval><a href="#s{10}min">Min={3}</a></span> '+\
4387 '<span class=medval><a href="#s{10}med">Med={4}</a></span> '+\
4388 '<span class=maxval><a href="#s{10}max">Max={5}</a></span> '+\
4389 'Resume Avg={6} '+\
4390 '<span class=minval><a href="#r{10}min">Min={7}</a></span> '+\
4391 '<span class=medval><a href="#r{10}med">Med={8}</a></span> '+\
4392 '<span class=maxval><a href="#r{10}max">Max={9}</a></span></td>'+\
4393 '</tr>\n'
4394 headnone = '<tr class="head"><td>{0}</td><td>{1}</td><td colspan='+\
4395 colspan+'></td></tr>\n'
4396 for mode in sorted(list):
4397 # header line for each suspend mode
4398 num = 0
4399 tAvg, tMin, tMax, tMed = list[mode]['avg'], list[mode]['min'],\
4400 list[mode]['max'], list[mode]['med']
4401 count = len(list[mode]['data'])
4402 if 'idx' in list[mode]:
4403 iMin, iMed, iMax = list[mode]['idx']
4404 html += head.format('%d' % count, mode.upper(),
4405 '%.3f' % tAvg[0], '%.3f' % tMin[0], '%.3f' % tMed[0], '%.3f' % tMax[0],
4406 '%.3f' % tAvg[1], '%.3f' % tMin[1], '%.3f' % tMed[1], '%.3f' % tMax[1],
4407 mode.lower()
4408 )
4409 else:
4410 iMin = iMed = iMax = [-1, -1, -1]
4411 html += headnone.format('%d' % count, mode.upper())
4412 for d in list[mode]['data']:
4413 # row classes - alternate row color
4414 rcls = ['alt'] if num % 2 == 1 else []
4415 if d[6] != 'pass':
4416 rcls.append('notice')
4417 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4418 # figure out if the line has sus or res highlighted
4419 idx = list[mode]['data'].index(d)
4420 tHigh = ['', '']
4421 for i in range(2):
4422 tag = 's%s' % mode if i == 0 else 'r%s' % mode
4423 if idx == iMin[i]:
4424 tHigh[i] = ' id="%smin" class=minval title="Minimum"' % tag
4425 elif idx == iMax[i]:
4426 tHigh[i] = ' id="%smax" class=maxval title="Maximum"' % tag
4427 elif idx == iMed[i]:
4428 tHigh[i] = ' id="%smed" class=medval title="Median"' % tag
4429 html += td.format("%d" % (list[mode]['data'].index(d) + 1)) # row
4430 html += td.format(mode) # mode
4431 html += td.format(d[0]) # host
4432 html += td.format(d[1]) # kernel
4433 html += td.format(d[2]) # time
4434 html += td.format(d[6]) # result
4435 html += td.format(d[7]) # issues
4436 html += tdh.format('%.3f ms' % d[3], tHigh[0]) if d[3] else td.format('') # suspend
4437 html += tdh.format('%.3f ms' % d[4], tHigh[1]) if d[4] else td.format('') # resume
4438 html += td.format(d[8]) # sus_worst
4439 html += td.format('%.3f ms' % d[9]) if d[9] else td.format('') # sus_worst time
4440 html += td.format(d[10]) # res_worst
4441 html += td.format('%.3f ms' % d[11]) if d[11] else td.format('') # res_worst time
4442 if useturbo:
4443 html += td.format(d[12]) # pkg_pc10
4444 html += td.format(d[13]) # syslpi
4445 if usewifi:
4446 html += td.format(d[14]) # wifi
4447 html += tdlink.format(d[5]) if d[5] else td.format('') # url
4448 html += '</tr>\n'
4449 num += 1
4450
4451 # flush the data to file
4452 hf = open(htmlfile, 'w')
4453 hf.write(html+'</table>\n</body>\n</html>\n')
4454 hf.close()
4455
4456def createHTMLDeviceSummary(testruns, htmlfile, title):
4457 html = summaryCSS('Device Summary - SleepGraph', False)
4458
4459 # create global device list from all tests
4460 devall = dict()
4461 for data in testruns:
4462 host, url, devlist = data['host'], data['url'], data['devlist']
4463 for type in devlist:
4464 if type not in devall:
4465 devall[type] = dict()
4466 mdevlist, devlist = devall[type], data['devlist'][type]
4467 for name in devlist:
4468 length = devlist[name]
4469 if name not in mdevlist:
4470 mdevlist[name] = {'name': name, 'host': host,
4471 'worst': length, 'total': length, 'count': 1,
4472 'url': url}
4473 else:
4474 if length > mdevlist[name]['worst']:
4475 mdevlist[name]['worst'] = length
4476 mdevlist[name]['url'] = url
4477 mdevlist[name]['host'] = host
4478 mdevlist[name]['total'] += length
4479 mdevlist[name]['count'] += 1
4480
4481 # generate the html
4482 th = '\t<th>{0}</th>\n'
4483 td = '\t<td align=center>{0}</td>\n'
4484 tdr = '\t<td align=right>{0}</td>\n'
4485 tdlink = '\t<td align=center><a href="{0}">html</a></td>\n'
4486 limit = 1
4487 for type in sorted(devall, reverse=True):
4488 num = 0
4489 devlist = devall[type]
4490 # table header
4491 html += '<div class="stamp">%s (%s devices > %d ms)</div><table>\n' % \
4492 (title, type.upper(), limit)
4493 html += '<tr>\n' + '<th align=right>Device Name</th>' +\
4494 th.format('Average Time') + th.format('Count') +\
4495 th.format('Worst Time') + th.format('Host (worst time)') +\
4496 th.format('Link (worst time)') + '</tr>\n'
4497 for name in sorted(devlist, key=lambda k:(devlist[k]['worst'], \
4498 devlist[k]['total'], devlist[k]['name']), reverse=True):
4499 data = devall[type][name]
4500 data['average'] = data['total'] / data['count']
4501 if data['average'] < limit:
4502 continue
4503 # row classes - alternate row color
4504 rcls = ['alt'] if num % 2 == 1 else []
4505 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4506 html += tdr.format(data['name']) # name
4507 html += td.format('%.3f ms' % data['average']) # average
4508 html += td.format(data['count']) # count
4509 html += td.format('%.3f ms' % data['worst']) # worst
4510 html += td.format(data['host']) # host
4511 html += tdlink.format(data['url']) # url
4512 html += '</tr>\n'
4513 num += 1
4514 html += '</table>\n'
4515
4516 # flush the data to file
4517 hf = open(htmlfile, 'w')
4518 hf.write(html+'</body>\n</html>\n')
4519 hf.close()
4520 return devall
4521
4522def createHTMLIssuesSummary(testruns, issues, htmlfile, title, extra=''):
4523 multihost = len([e for e in issues if len(e['urls']) > 1]) > 0
4524 html = summaryCSS('Issues Summary - SleepGraph', False)
4525 total = len(testruns)
4526
4527 # generate the html
4528 th = '\t<th>{0}</th>\n'
4529 td = '\t<td align={0}>{1}</td>\n'
4530 tdlink = '<a href="{1}">{0}</a>'
4531 subtitle = '%d issues' % len(issues) if len(issues) > 0 else 'no issues'
4532 html += '<div class="stamp">%s (%s)</div><table>\n' % (title, subtitle)
4533 html += '<tr>\n' + th.format('Issue') + th.format('Count')
4534 if multihost:
4535 html += th.format('Hosts')
4536 html += th.format('Tests') + th.format('Fail Rate') +\
4537 th.format('First Instance') + '</tr>\n'
4538
4539 num = 0
4540 for e in sorted(issues, key=lambda v:v['count'], reverse=True):
4541 testtotal = 0
4542 links = []
4543 for host in sorted(e['urls']):
4544 links.append(tdlink.format(host, e['urls'][host][0]))
4545 testtotal += len(e['urls'][host])
4546 rate = '%d/%d (%.2f%%)' % (testtotal, total, 100*float(testtotal)/float(total))
4547 # row classes - alternate row color
4548 rcls = ['alt'] if num % 2 == 1 else []
4549 html += '<tr class="'+(' '.join(rcls))+'">\n' if len(rcls) > 0 else '<tr>\n'
4550 html += td.format('left', e['line']) # issue
4551 html += td.format('center', e['count']) # count
4552 if multihost:
4553 html += td.format('center', len(e['urls'])) # hosts
4554 html += td.format('center', testtotal) # test count
4555 html += td.format('center', rate) # test rate
4556 html += td.format('center nowrap', '<br>'.join(links)) # links
4557 html += '</tr>\n'
4558 num += 1
4559
4560 # flush the data to file
4561 hf = open(htmlfile, 'w')
4562 hf.write(html+'</table>\n'+extra+'</body>\n</html>\n')
4563 hf.close()
4564 return issues
4565
4566def ordinal(value):
4567 suffix = 'th'
4568 if value < 10 or value > 19:
4569 if value % 10 == 1:
4570 suffix = 'st'
4571 elif value % 10 == 2:
4572 suffix = 'nd'
4573 elif value % 10 == 3:
4574 suffix = 'rd'
4575 return '%d%s' % (value, suffix)
4576
4577# Function: createHTML
4578# Description:
4579# Create the output html file from the resident test data
4580# Arguments:
4581# testruns: array of Data objects from parseKernelLog or parseTraceLog
4582# Output:
4583# True if the html file was created, false if it failed
4584def createHTML(testruns, testfail):
4585 if len(testruns) < 1:
4586 pprint('ERROR: Not enough test data to build a timeline')
4587 return
4588
4589 kerror = False
4590 for data in testruns:
4591 if data.kerror:
4592 kerror = True
4593 if(sysvals.suspendmode in ['freeze', 'standby']):
4594 data.trimFreezeTime(testruns[-1].tSuspended)
4595 else:
4596 data.getMemTime()
4597
4598 # html function templates
4599 html_error = '<div id="{1}" title="kernel error/warning" class="err" style="right:{0}%">{2}→</div>\n'
4600 html_traceevent = '<div title="{0}" class="traceevent{6}" style="left:{1}%;top:{2}px;height:{3}px;width:{4}%;line-height:{3}px;{7}">{5}</div>\n'
4601 html_cpuexec = '<div class="jiffie" style="left:{0}%;top:{1}px;height:{2}px;width:{3}%;background:{4};"></div>\n'
4602 html_timetotal = '<table class="time1">\n<tr>'\
4603 '<td class="green" title="{3}">{2} Suspend Time: <b>{0} ms</b></td>'\
4604 '<td class="yellow" title="{4}">{2} Resume Time: <b>{1} ms</b></td>'\
4605 '</tr>\n</table>\n'
4606 html_timetotal2 = '<table class="time1">\n<tr>'\
4607 '<td class="green" title="{4}">{3} Suspend Time: <b>{0} ms</b></td>'\
4608 '<td class="gray" title="time spent in low-power mode with clock running">'+sysvals.suspendmode+' time: <b>{1} ms</b></td>'\
4609 '<td class="yellow" title="{5}">{3} Resume Time: <b>{2} ms</b></td>'\
4610 '</tr>\n</table>\n'
4611 html_timetotal3 = '<table class="time1">\n<tr>'\
4612 '<td class="green">Execution Time: <b>{0} ms</b></td>'\
4613 '<td class="yellow">Command: <b>{1}</b></td>'\
4614 '</tr>\n</table>\n'
4615 html_fail = '<table class="testfail"><tr><td>{0}</td></tr></table>\n'
4616 html_kdesc = '<td class="{3}" title="time spent in kernel execution">{0}Kernel {2}: {1} ms</td>'
4617 html_fwdesc = '<td class="{3}" title="time spent in firmware">{0}Firmware {2}: {1} ms</td>'
4618 html_wifdesc = '<td class="yellow" title="time for wifi to reconnect after resume complete ({2})">{0}Wifi Resume: {1}</td>'
4619
4620 # html format variables
4621 scaleH = 20
4622 if kerror:
4623 scaleH = 40
4624
4625 # device timeline
4626 devtl = Timeline(30, scaleH)
4627
4628 # write the test title and general info header
4629 devtl.createHeader(sysvals, testruns[0].stamp)
4630
4631 # Generate the header for this timeline
4632 for data in testruns:
4633 tTotal = data.end - data.start
4634 if(tTotal == 0):
4635 doError('No timeline data')
4636 if sysvals.suspendmode == 'command':
4637 run_time = '%.0f' % (tTotal * 1000)
4638 if sysvals.testcommand:
4639 testdesc = sysvals.testcommand
4640 else:
4641 testdesc = 'unknown'
4642 if(len(testruns) > 1):
4643 testdesc = ordinal(data.testnumber+1)+' '+testdesc
4644 thtml = html_timetotal3.format(run_time, testdesc)
4645 devtl.html += thtml
4646 continue
4647 # typical full suspend/resume header
4648 stot, rtot = sktime, rktime = data.getTimeValues()
4649 ssrc, rsrc, testdesc, testdesc2 = ['kernel'], ['kernel'], 'Kernel', ''
4650 if data.fwValid:
4651 stot += (data.fwSuspend/1000000.0)
4652 rtot += (data.fwResume/1000000.0)
4653 ssrc.append('firmware')
4654 rsrc.append('firmware')
4655 testdesc = 'Total'
4656 if 'time' in data.wifi and data.wifi['stat'] != 'timeout':
4657 rtot += data.end - data.tKernRes + (data.wifi['time'] * 1000.0)
4658 rsrc.append('wifi')
4659 testdesc = 'Total'
4660 suspend_time, resume_time = '%.3f' % stot, '%.3f' % rtot
4661 stitle = 'time from kernel suspend start to %s mode [%s time]' % \
4662 (sysvals.suspendmode, ' & '.join(ssrc))
4663 rtitle = 'time from %s mode to kernel resume complete [%s time]' % \
4664 (sysvals.suspendmode, ' & '.join(rsrc))
4665 if(len(testruns) > 1):
4666 testdesc = testdesc2 = ordinal(data.testnumber+1)
4667 testdesc2 += ' '
4668 if(len(data.tLow) == 0):
4669 thtml = html_timetotal.format(suspend_time, \
4670 resume_time, testdesc, stitle, rtitle)
4671 else:
4672 low_time = '+'.join(data.tLow)
4673 thtml = html_timetotal2.format(suspend_time, low_time, \
4674 resume_time, testdesc, stitle, rtitle)
4675 devtl.html += thtml
4676 if not data.fwValid and 'dev' not in data.wifi:
4677 continue
4678 # extra detail when the times come from multiple sources
4679 thtml = '<table class="time2">\n<tr>'
4680 thtml += html_kdesc.format(testdesc2, '%.3f'%sktime, 'Suspend', 'green')
4681 if data.fwValid:
4682 sftime = '%.3f'%(data.fwSuspend / 1000000.0)
4683 rftime = '%.3f'%(data.fwResume / 1000000.0)
4684 thtml += html_fwdesc.format(testdesc2, sftime, 'Suspend', 'green')
4685 thtml += html_fwdesc.format(testdesc2, rftime, 'Resume', 'yellow')
4686 thtml += html_kdesc.format(testdesc2, '%.3f'%rktime, 'Resume', 'yellow')
4687 if 'time' in data.wifi:
4688 if data.wifi['stat'] != 'timeout':
4689 wtime = '%.0f ms'%(data.end - data.tKernRes + (data.wifi['time'] * 1000.0))
4690 else:
4691 wtime = 'TIMEOUT'
4692 thtml += html_wifdesc.format(testdesc2, wtime, data.wifi['dev'])
4693 thtml += '</tr>\n</table>\n'
4694 devtl.html += thtml
4695 if testfail:
4696 devtl.html += html_fail.format(testfail)
4697
4698 # time scale for potentially multiple datasets
4699 t0 = testruns[0].start
4700 tMax = testruns[-1].end
4701 tTotal = tMax - t0
4702
4703 # determine the maximum number of rows we need to draw
4704 fulllist = []
4705 threadlist = []
4706 pscnt = 0
4707 devcnt = 0
4708 for data in testruns:
4709 data.selectTimelineDevices('%f', tTotal, sysvals.mindevlen)
4710 for group in data.devicegroups:
4711 devlist = []
4712 for phase in group:
4713 for devname in sorted(data.tdevlist[phase]):
4714 d = DevItem(data.testnumber, phase, data.dmesg[phase]['list'][devname])
4715 devlist.append(d)
4716 if d.isa('kth'):
4717 threadlist.append(d)
4718 else:
4719 if d.isa('ps'):
4720 pscnt += 1
4721 else:
4722 devcnt += 1
4723 fulllist.append(d)
4724 if sysvals.mixedphaseheight:
4725 devtl.getPhaseRows(devlist)
4726 if not sysvals.mixedphaseheight:
4727 if len(threadlist) > 0 and len(fulllist) > 0:
4728 if pscnt > 0 and devcnt > 0:
4729 msg = 'user processes & device pm callbacks'
4730 elif pscnt > 0:
4731 msg = 'user processes'
4732 else:
4733 msg = 'device pm callbacks'
4734 d = testruns[0].addHorizontalDivider(msg, testruns[-1].end)
4735 fulllist.insert(0, d)
4736 devtl.getPhaseRows(fulllist)
4737 if len(threadlist) > 0:
4738 d = testruns[0].addHorizontalDivider('asynchronous kernel threads', testruns[-1].end)
4739 threadlist.insert(0, d)
4740 devtl.getPhaseRows(threadlist, devtl.rows)
4741 devtl.calcTotalRows()
4742
4743 # draw the full timeline
4744 devtl.createZoomBox(sysvals.suspendmode, len(testruns))
4745 for data in testruns:
4746 # draw each test run and block chronologically
4747 phases = {'suspend':[],'resume':[]}
4748 for phase in data.sortedPhases():
4749 if data.dmesg[phase]['start'] >= data.tSuspended:
4750 phases['resume'].append(phase)
4751 else:
4752 phases['suspend'].append(phase)
4753 # now draw the actual timeline blocks
4754 for dir in phases:
4755 # draw suspend and resume blocks separately
4756 bname = '%s%d' % (dir[0], data.testnumber)
4757 if dir == 'suspend':
4758 m0 = data.start
4759 mMax = data.tSuspended
4760 left = '%f' % (((m0-t0)*100.0)/tTotal)
4761 else:
4762 m0 = data.tSuspended
4763 mMax = data.end
4764 # in an x2 run, remove any gap between blocks
4765 if len(testruns) > 1 and data.testnumber == 0:
4766 mMax = testruns[1].start
4767 left = '%f' % ((((m0-t0)*100.0)+sysvals.srgap/2)/tTotal)
4768 mTotal = mMax - m0
4769 # if a timeline block is 0 length, skip altogether
4770 if mTotal == 0:
4771 continue
4772 width = '%f' % (((mTotal*100.0)-sysvals.srgap/2)/tTotal)
4773 devtl.html += devtl.html_tblock.format(bname, left, width, devtl.scaleH)
4774 for b in phases[dir]:
4775 # draw the phase color background
4776 phase = data.dmesg[b]
4777 length = phase['end']-phase['start']
4778 left = '%f' % (((phase['start']-m0)*100.0)/mTotal)
4779 width = '%f' % ((length*100.0)/mTotal)
4780 devtl.html += devtl.html_phase.format(left, width, \
4781 '%.3f'%devtl.scaleH, '%.3f'%devtl.bodyH, \
4782 data.dmesg[b]['color'], '')
4783 for e in data.errorinfo[dir]:
4784 # draw red lines for any kernel errors found
4785 type, t, idx1, idx2 = e
4786 id = '%d_%d' % (idx1, idx2)
4787 right = '%f' % (((mMax-t)*100.0)/mTotal)
4788 devtl.html += html_error.format(right, id, type)
4789 for b in phases[dir]:
4790 # draw the devices for this phase
4791 phaselist = data.dmesg[b]['list']
4792 for d in sorted(data.tdevlist[b]):
4793 dname = d if ('[' not in d or 'CPU' in d) else d.split('[')[0]
4794 name, dev = dname, phaselist[d]
4795 drv = xtraclass = xtrainfo = xtrastyle = ''
4796 if 'htmlclass' in dev:
4797 xtraclass = dev['htmlclass']
4798 if 'color' in dev:
4799 xtrastyle = 'background:%s;' % dev['color']
4800 if(d in sysvals.devprops):
4801 name = sysvals.devprops[d].altName(d)
4802 xtraclass = sysvals.devprops[d].xtraClass()
4803 xtrainfo = sysvals.devprops[d].xtraInfo()
4804 elif xtraclass == ' kth':
4805 xtrainfo = ' kernel_thread'
4806 if('drv' in dev and dev['drv']):
4807 drv = ' {%s}' % dev['drv']
4808 rowheight = devtl.phaseRowHeight(data.testnumber, b, dev['row'])
4809 rowtop = devtl.phaseRowTop(data.testnumber, b, dev['row'])
4810 top = '%.3f' % (rowtop + devtl.scaleH)
4811 left = '%f' % (((dev['start']-m0)*100)/mTotal)
4812 width = '%f' % (((dev['end']-dev['start'])*100)/mTotal)
4813 length = ' (%0.3f ms) ' % ((dev['end']-dev['start'])*1000)
4814 title = name+drv+xtrainfo+length
4815 if sysvals.suspendmode == 'command':
4816 title += sysvals.testcommand
4817 elif xtraclass == ' ps':
4818 if 'suspend' in b:
4819 title += 'pre_suspend_process'
4820 else:
4821 title += 'post_resume_process'
4822 else:
4823 title += b
4824 devtl.html += devtl.html_device.format(dev['id'], \
4825 title, left, top, '%.3f'%rowheight, width, \
4826 dname+drv, xtraclass, xtrastyle)
4827 if('cpuexec' in dev):
4828 for t in sorted(dev['cpuexec']):
4829 start, end = t
4830 height = '%.3f' % (rowheight/3)
4831 top = '%.3f' % (rowtop + devtl.scaleH + 2*rowheight/3)
4832 left = '%f' % (((start-m0)*100)/mTotal)
4833 width = '%f' % ((end-start)*100/mTotal)
4834 color = 'rgba(255, 0, 0, %f)' % dev['cpuexec'][t]
4835 devtl.html += \
4836 html_cpuexec.format(left, top, height, width, color)
4837 if('src' not in dev):
4838 continue
4839 # draw any trace events for this device
4840 for e in dev['src']:
4841 if e.length == 0:
4842 continue
4843 height = '%.3f' % devtl.rowH
4844 top = '%.3f' % (rowtop + devtl.scaleH + (e.row*devtl.rowH))
4845 left = '%f' % (((e.time-m0)*100)/mTotal)
4846 width = '%f' % (e.length*100/mTotal)
4847 xtrastyle = ''
4848 if e.color:
4849 xtrastyle = 'background:%s;' % e.color
4850 devtl.html += \
4851 html_traceevent.format(e.title(), \
4852 left, top, height, width, e.text(), '', xtrastyle)
4853 # draw the time scale, try to make the number of labels readable
4854 devtl.createTimeScale(m0, mMax, tTotal, dir)
4855 devtl.html += '</div>\n'
4856
4857 # timeline is finished
4858 devtl.html += '</div>\n</div>\n'
4859
4860 # draw a legend which describes the phases by color
4861 if sysvals.suspendmode != 'command':
4862 phasedef = testruns[-1].phasedef
4863 devtl.html += '<div class="legend">\n'
4864 pdelta = 100.0/len(phasedef.keys())
4865 pmargin = pdelta / 4.0
4866 for phase in sorted(phasedef, key=lambda k:phasedef[k]['order']):
4867 id, p = '', phasedef[phase]
4868 for word in phase.split('_'):
4869 id += word[0]
4870 order = '%.2f' % ((p['order'] * pdelta) + pmargin)
4871 name = phase.replace('_', ' ')
4872 devtl.html += devtl.html_legend.format(order, p['color'], name, id)
4873 devtl.html += '</div>\n'
4874
4875 hf = open(sysvals.htmlfile, 'w')
4876 addCSS(hf, sysvals, len(testruns), kerror)
4877
4878 # write the device timeline
4879 hf.write(devtl.html)
4880 hf.write('<div id="devicedetailtitle"></div>\n')
4881 hf.write('<div id="devicedetail" style="display:none;">\n')
4882 # draw the colored boxes for the device detail section
4883 for data in testruns:
4884 hf.write('<div id="devicedetail%d">\n' % data.testnumber)
4885 pscolor = 'linear-gradient(to top left, #ccc, #eee)'
4886 hf.write(devtl.html_phaselet.format('pre_suspend_process', \
4887 '0', '0', pscolor))
4888 for b in data.sortedPhases():
4889 phase = data.dmesg[b]
4890 length = phase['end']-phase['start']
4891 left = '%.3f' % (((phase['start']-t0)*100.0)/tTotal)
4892 width = '%.3f' % ((length*100.0)/tTotal)
4893 hf.write(devtl.html_phaselet.format(b, left, width, \
4894 data.dmesg[b]['color']))
4895 hf.write(devtl.html_phaselet.format('post_resume_process', \
4896 '0', '0', pscolor))
4897 if sysvals.suspendmode == 'command':
4898 hf.write(devtl.html_phaselet.format('cmdexec', '0', '0', pscolor))
4899 hf.write('</div>\n')
4900 hf.write('</div>\n')
4901
4902 # write the ftrace data (callgraph)
4903 if sysvals.cgtest >= 0 and len(testruns) > sysvals.cgtest:
4904 data = testruns[sysvals.cgtest]
4905 else:
4906 data = testruns[-1]
4907 if sysvals.usecallgraph:
4908 addCallgraphs(sysvals, hf, data)
4909
4910 # add the test log as a hidden div
4911 if sysvals.testlog and sysvals.logmsg:
4912 hf.write('<div id="testlog" style="display:none;">\n'+sysvals.logmsg+'</div>\n')
4913 # add the dmesg log as a hidden div
4914 if sysvals.dmesglog and sysvals.dmesgfile:
4915 hf.write('<div id="dmesglog" style="display:none;">\n')
4916 lf = sysvals.openlog(sysvals.dmesgfile, 'r')
4917 for line in lf:
4918 line = line.replace('<', '<').replace('>', '>')
4919 hf.write(line)
4920 lf.close()
4921 hf.write('</div>\n')
4922 # add the ftrace log as a hidden div
4923 if sysvals.ftracelog and sysvals.ftracefile:
4924 hf.write('<div id="ftracelog" style="display:none;">\n')
4925 lf = sysvals.openlog(sysvals.ftracefile, 'r')
4926 for line in lf:
4927 hf.write(line)
4928 lf.close()
4929 hf.write('</div>\n')
4930
4931 # write the footer and close
4932 addScriptCode(hf, testruns)
4933 hf.write('</body>\n</html>\n')
4934 hf.close()
4935 return True
4936
4937def addCSS(hf, sv, testcount=1, kerror=False, extra=''):
4938 kernel = sv.stamp['kernel']
4939 host = sv.hostname[0].upper()+sv.hostname[1:]
4940 mode = sv.suspendmode
4941 if sv.suspendmode in suspendmodename:
4942 mode = suspendmodename[sv.suspendmode]
4943 title = host+' '+mode+' '+kernel
4944
4945 # various format changes by flags
4946 cgchk = 'checked'
4947 cgnchk = 'not(:checked)'
4948 if sv.cgexp:
4949 cgchk = 'not(:checked)'
4950 cgnchk = 'checked'
4951
4952 hoverZ = 'z-index:8;'
4953 if sv.usedevsrc:
4954 hoverZ = ''
4955
4956 devlistpos = 'absolute'
4957 if testcount > 1:
4958 devlistpos = 'relative'
4959
4960 scaleTH = 20
4961 if kerror:
4962 scaleTH = 60
4963
4964 # write the html header first (html head, css code, up to body start)
4965 html_header = '<!DOCTYPE html>\n<html>\n<head>\n\
4966 <meta http-equiv="content-type" content="text/html; charset=UTF-8">\n\
4967 <title>'+title+'</title>\n\
4968 <style type=\'text/css\'>\n\
4969 body {overflow-y:scroll;}\n\
4970 .stamp {width:100%;text-align:center;background:gray;line-height:30px;color:white;font:25px Arial;}\n\
4971 .stamp.sysinfo {font:10px Arial;}\n\
4972 .callgraph {margin-top:30px;box-shadow:5px 5px 20px black;}\n\
4973 .callgraph article * {padding-left:28px;}\n\
4974 h1 {color:black;font:bold 30px Times;}\n\
4975 t0 {color:black;font:bold 30px Times;}\n\
4976 t1 {color:black;font:30px Times;}\n\
4977 t2 {color:black;font:25px Times;}\n\
4978 t3 {color:black;font:20px Times;white-space:nowrap;}\n\
4979 t4 {color:black;font:bold 30px Times;line-height:60px;white-space:nowrap;}\n\
4980 cS {font:bold 13px Times;}\n\
4981 table {width:100%;}\n\
4982 .gray {background:rgba(80,80,80,0.1);}\n\
4983 .green {background:rgba(204,255,204,0.4);}\n\
4984 .purple {background:rgba(128,0,128,0.2);}\n\
4985 .yellow {background:rgba(255,255,204,0.4);}\n\
4986 .blue {background:rgba(169,208,245,0.4);}\n\
4987 .time1 {font:22px Arial;border:1px solid;}\n\
4988 .time2 {font:15px Arial;border-bottom:1px solid;border-left:1px solid;border-right:1px solid;}\n\
4989 .testfail {font:bold 22px Arial;color:red;border:1px dashed;}\n\
4990 td {text-align:center;}\n\
4991 r {color:#500000;font:15px Tahoma;}\n\
4992 n {color:#505050;font:15px Tahoma;}\n\
4993 .tdhl {color:red;}\n\
4994 .hide {display:none;}\n\
4995 .pf {display:none;}\n\
4996 .pf:'+cgchk+' + label {background:url(\'data:image/svg+xml;utf,<?xml version="1.0" standalone="no"?><svg xmlns="http://www.w3.org/2000/svg" height="18" width="18" version="1.1"><circle cx="9" cy="9" r="8" stroke="black" stroke-width="1" fill="white"/><rect x="4" y="8" width="10" height="2" style="fill:black;stroke-width:0"/><rect x="8" y="4" width="2" height="10" style="fill:black;stroke-width:0"/></svg>\') no-repeat left center;}\n\
4997 .pf:'+cgnchk+' ~ label {background:url(\'data:image/svg+xml;utf,<?xml version="1.0" standalone="no"?><svg xmlns="http://www.w3.org/2000/svg" height="18" width="18" version="1.1"><circle cx="9" cy="9" r="8" stroke="black" stroke-width="1" fill="white"/><rect x="4" y="8" width="10" height="2" style="fill:black;stroke-width:0"/></svg>\') no-repeat left center;}\n\
4998 .pf:'+cgchk+' ~ *:not(:nth-child(2)) {display:none;}\n\
4999 .zoombox {position:relative;width:100%;overflow-x:scroll;-webkit-user-select:none;-moz-user-select:none;user-select:none;}\n\
5000 .timeline {position:relative;font-size:14px;cursor:pointer;width:100%; overflow:hidden;background:linear-gradient(#cccccc, white);}\n\
5001 .thread {position:absolute;height:0%;overflow:hidden;z-index:7;line-height:30px;font-size:14px;border:1px solid;text-align:center;white-space:nowrap;}\n\
5002 .thread.ps {border-radius:3px;background:linear-gradient(to top, #ccc, #eee);}\n\
5003 .thread:hover {background:white;border:1px solid red;'+hoverZ+'}\n\
5004 .thread.sec,.thread.sec:hover {background:black;border:0;color:white;line-height:15px;font-size:10px;}\n\
5005 .hover {background:white;border:1px solid red;'+hoverZ+'}\n\
5006 .hover.sync {background:white;}\n\
5007 .hover.bg,.hover.kth,.hover.sync,.hover.ps {background:white;}\n\
5008 .jiffie {position:absolute;pointer-events: none;z-index:8;}\n\
5009 .traceevent {position:absolute;font-size:10px;z-index:7;overflow:hidden;color:black;text-align:center;white-space:nowrap;border-radius:5px;border:1px solid black;background:linear-gradient(to bottom right,#CCC,#969696);}\n\
5010 .traceevent:hover {color:white;font-weight:bold;border:1px solid white;}\n\
5011 .phase {position:absolute;overflow:hidden;border:0px;text-align:center;}\n\
5012 .phaselet {float:left;overflow:hidden;border:0px;text-align:center;min-height:100px;font-size:24px;}\n\
5013 .t {position:absolute;line-height:'+('%d'%scaleTH)+'px;pointer-events:none;top:0;height:100%;border-right:1px solid black;z-index:6;}\n\
5014 .err {position:absolute;top:0%;height:100%;border-right:3px solid red;color:red;font:bold 14px Times;line-height:18px;}\n\
5015 .legend {position:relative; width:100%; height:40px; text-align:center;margin-bottom:20px}\n\
5016 .legend .square {position:absolute;cursor:pointer;top:10px; width:0px;height:20px;border:1px solid;padding-left:20px;}\n\
5017 button {height:40px;width:200px;margin-bottom:20px;margin-top:20px;font-size:24px;}\n\
5018 .btnfmt {position:relative;float:right;height:25px;width:auto;margin-top:3px;margin-bottom:0;font-size:10px;text-align:center;}\n\
5019 .devlist {position:'+devlistpos+';width:190px;}\n\
5020 a:link {color:white;text-decoration:none;}\n\
5021 a:visited {color:white;}\n\
5022 a:hover {color:white;}\n\
5023 a:active {color:white;}\n\
5024 .version {position:relative;float:left;color:white;font-size:10px;line-height:30px;margin-left:10px;}\n\
5025 #devicedetail {min-height:100px;box-shadow:5px 5px 20px black;}\n\
5026 .tblock {position:absolute;height:100%;background:#ddd;}\n\
5027 .tback {position:absolute;width:100%;background:linear-gradient(#ccc, #ddd);}\n\
5028 .bg {z-index:1;}\n\
5029'+extra+'\
5030 </style>\n</head>\n<body>\n'
5031 hf.write(html_header)
5032
5033# Function: addScriptCode
5034# Description:
5035# Adds the javascript code to the output html
5036# Arguments:
5037# hf: the open html file pointer
5038# testruns: array of Data objects from parseKernelLog or parseTraceLog
5039def addScriptCode(hf, testruns):
5040 t0 = testruns[0].start * 1000
5041 tMax = testruns[-1].end * 1000
5042 # create an array in javascript memory with the device details
5043 detail = ' var devtable = [];\n'
5044 for data in testruns:
5045 topo = data.deviceTopology()
5046 detail += ' devtable[%d] = "%s";\n' % (data.testnumber, topo)
5047 detail += ' var bounds = [%f,%f];\n' % (t0, tMax)
5048 # add the code which will manipulate the data in the browser
5049 script_code = \
5050 '<script type="text/javascript">\n'+detail+\
5051 ' var resolution = -1;\n'\
5052 ' var dragval = [0, 0];\n'\
5053 ' function redrawTimescale(t0, tMax, tS) {\n'\
5054 ' var rline = \'<div class="t" style="left:0;border-left:1px solid black;border-right:0;">\';\n'\
5055 ' var tTotal = tMax - t0;\n'\
5056 ' var list = document.getElementsByClassName("tblock");\n'\
5057 ' for (var i = 0; i < list.length; i++) {\n'\
5058 ' var timescale = list[i].getElementsByClassName("timescale")[0];\n'\
5059 ' var m0 = t0 + (tTotal*parseFloat(list[i].style.left)/100);\n'\
5060 ' var mTotal = tTotal*parseFloat(list[i].style.width)/100;\n'\
5061 ' var mMax = m0 + mTotal;\n'\
5062 ' var html = "";\n'\
5063 ' var divTotal = Math.floor(mTotal/tS) + 1;\n'\
5064 ' if(divTotal > 1000) continue;\n'\
5065 ' var divEdge = (mTotal - tS*(divTotal-1))*100/mTotal;\n'\
5066 ' var pos = 0.0, val = 0.0;\n'\
5067 ' for (var j = 0; j < divTotal; j++) {\n'\
5068 ' var htmlline = "";\n'\
5069 ' var mode = list[i].id[5];\n'\
5070 ' if(mode == "s") {\n'\
5071 ' pos = 100 - (((j)*tS*100)/mTotal) - divEdge;\n'\
5072 ' val = (j-divTotal+1)*tS;\n'\
5073 ' if(j == divTotal - 1)\n'\
5074 ' htmlline = \'<div class="t" style="right:\'+pos+\'%"><cS>S→</cS></div>\';\n'\
5075 ' else\n'\
5076 ' htmlline = \'<div class="t" style="right:\'+pos+\'%">\'+val+\'ms</div>\';\n'\
5077 ' } else {\n'\
5078 ' pos = 100 - (((j)*tS*100)/mTotal);\n'\
5079 ' val = (j)*tS;\n'\
5080 ' htmlline = \'<div class="t" style="right:\'+pos+\'%">\'+val+\'ms</div>\';\n'\
5081 ' if(j == 0)\n'\
5082 ' if(mode == "r")\n'\
5083 ' htmlline = rline+"<cS>←R</cS></div>";\n'\
5084 ' else\n'\
5085 ' htmlline = rline+"<cS>0ms</div>";\n'\
5086 ' }\n'\
5087 ' html += htmlline;\n'\
5088 ' }\n'\
5089 ' timescale.innerHTML = html;\n'\
5090 ' }\n'\
5091 ' }\n'\
5092 ' function zoomTimeline() {\n'\
5093 ' var dmesg = document.getElementById("dmesg");\n'\
5094 ' var zoombox = document.getElementById("dmesgzoombox");\n'\
5095 ' var left = zoombox.scrollLeft;\n'\
5096 ' var val = parseFloat(dmesg.style.width);\n'\
5097 ' var newval = 100;\n'\
5098 ' var sh = window.outerWidth / 2;\n'\
5099 ' if(this.id == "zoomin") {\n'\
5100 ' newval = val * 1.2;\n'\
5101 ' if(newval > 910034) newval = 910034;\n'\
5102 ' dmesg.style.width = newval+"%";\n'\
5103 ' zoombox.scrollLeft = ((left + sh) * newval / val) - sh;\n'\
5104 ' } else if (this.id == "zoomout") {\n'\
5105 ' newval = val / 1.2;\n'\
5106 ' if(newval < 100) newval = 100;\n'\
5107 ' dmesg.style.width = newval+"%";\n'\
5108 ' zoombox.scrollLeft = ((left + sh) * newval / val) - sh;\n'\
5109 ' } else {\n'\
5110 ' zoombox.scrollLeft = 0;\n'\
5111 ' dmesg.style.width = "100%";\n'\
5112 ' }\n'\
5113 ' var tS = [10000, 5000, 2000, 1000, 500, 200, 100, 50, 20, 10, 5, 2, 1];\n'\
5114 ' var t0 = bounds[0];\n'\
5115 ' var tMax = bounds[1];\n'\
5116 ' var tTotal = tMax - t0;\n'\
5117 ' var wTotal = tTotal * 100.0 / newval;\n'\
5118 ' var idx = 7*window.innerWidth/1100;\n'\
5119 ' for(var i = 0; (i < tS.length)&&((wTotal / tS[i]) < idx); i++);\n'\
5120 ' if(i >= tS.length) i = tS.length - 1;\n'\
5121 ' if(tS[i] == resolution) return;\n'\
5122 ' resolution = tS[i];\n'\
5123 ' redrawTimescale(t0, tMax, tS[i]);\n'\
5124 ' }\n'\
5125 ' function deviceName(title) {\n'\
5126 ' var name = title.slice(0, title.indexOf(" ("));\n'\
5127 ' return name;\n'\
5128 ' }\n'\
5129 ' function deviceHover() {\n'\
5130 ' var name = deviceName(this.title);\n'\
5131 ' var dmesg = document.getElementById("dmesg");\n'\
5132 ' var dev = dmesg.getElementsByClassName("thread");\n'\
5133 ' var cpu = -1;\n'\
5134 ' if(name.match("CPU_ON\[[0-9]*\]"))\n'\
5135 ' cpu = parseInt(name.slice(7));\n'\
5136 ' else if(name.match("CPU_OFF\[[0-9]*\]"))\n'\
5137 ' cpu = parseInt(name.slice(8));\n'\
5138 ' for (var i = 0; i < dev.length; i++) {\n'\
5139 ' dname = deviceName(dev[i].title);\n'\
5140 ' var cname = dev[i].className.slice(dev[i].className.indexOf("thread"));\n'\
5141 ' if((cpu >= 0 && dname.match("CPU_O[NF]*\\\[*"+cpu+"\\\]")) ||\n'\
5142 ' (name == dname))\n'\
5143 ' {\n'\
5144 ' dev[i].className = "hover "+cname;\n'\
5145 ' } else {\n'\
5146 ' dev[i].className = cname;\n'\
5147 ' }\n'\
5148 ' }\n'\
5149 ' }\n'\
5150 ' function deviceUnhover() {\n'\
5151 ' var dmesg = document.getElementById("dmesg");\n'\
5152 ' var dev = dmesg.getElementsByClassName("thread");\n'\
5153 ' for (var i = 0; i < dev.length; i++) {\n'\
5154 ' dev[i].className = dev[i].className.slice(dev[i].className.indexOf("thread"));\n'\
5155 ' }\n'\
5156 ' }\n'\
5157 ' function deviceTitle(title, total, cpu) {\n'\
5158 ' var prefix = "Total";\n'\
5159 ' if(total.length > 3) {\n'\
5160 ' prefix = "Average";\n'\
5161 ' total[1] = (total[1]+total[3])/2;\n'\
5162 ' total[2] = (total[2]+total[4])/2;\n'\
5163 ' }\n'\
5164 ' var devtitle = document.getElementById("devicedetailtitle");\n'\
5165 ' var name = deviceName(title);\n'\
5166 ' if(cpu >= 0) name = "CPU"+cpu;\n'\
5167 ' var driver = "";\n'\
5168 ' var tS = "<t2>(</t2>";\n'\
5169 ' var tR = "<t2>)</t2>";\n'\
5170 ' if(total[1] > 0)\n'\
5171 ' tS = "<t2>("+prefix+" Suspend:</t2><t0> "+total[1].toFixed(3)+" ms</t0> ";\n'\
5172 ' if(total[2] > 0)\n'\
5173 ' tR = " <t2>"+prefix+" Resume:</t2><t0> "+total[2].toFixed(3)+" ms<t2>)</t2></t0>";\n'\
5174 ' var s = title.indexOf("{");\n'\
5175 ' var e = title.indexOf("}");\n'\
5176 ' if((s >= 0) && (e >= 0))\n'\
5177 ' driver = title.slice(s+1, e) + " <t1>@</t1> ";\n'\
5178 ' if(total[1] > 0 && total[2] > 0)\n'\
5179 ' devtitle.innerHTML = "<t0>"+driver+name+"</t0> "+tS+tR;\n'\
5180 ' else\n'\
5181 ' devtitle.innerHTML = "<t0>"+title+"</t0>";\n'\
5182 ' return name;\n'\
5183 ' }\n'\
5184 ' function deviceDetail() {\n'\
5185 ' var devinfo = document.getElementById("devicedetail");\n'\
5186 ' devinfo.style.display = "block";\n'\
5187 ' var name = deviceName(this.title);\n'\
5188 ' var cpu = -1;\n'\
5189 ' if(name.match("CPU_ON\[[0-9]*\]"))\n'\
5190 ' cpu = parseInt(name.slice(7));\n'\
5191 ' else if(name.match("CPU_OFF\[[0-9]*\]"))\n'\
5192 ' cpu = parseInt(name.slice(8));\n'\
5193 ' var dmesg = document.getElementById("dmesg");\n'\
5194 ' var dev = dmesg.getElementsByClassName("thread");\n'\
5195 ' var idlist = [];\n'\
5196 ' var pdata = [[]];\n'\
5197 ' if(document.getElementById("devicedetail1"))\n'\
5198 ' pdata = [[], []];\n'\
5199 ' var pd = pdata[0];\n'\
5200 ' var total = [0.0, 0.0, 0.0];\n'\
5201 ' for (var i = 0; i < dev.length; i++) {\n'\
5202 ' dname = deviceName(dev[i].title);\n'\
5203 ' if((cpu >= 0 && dname.match("CPU_O[NF]*\\\[*"+cpu+"\\\]")) ||\n'\
5204 ' (name == dname))\n'\
5205 ' {\n'\
5206 ' idlist[idlist.length] = dev[i].id;\n'\
5207 ' var tidx = 1;\n'\
5208 ' if(dev[i].id[0] == "a") {\n'\
5209 ' pd = pdata[0];\n'\
5210 ' } else {\n'\
5211 ' if(pdata.length == 1) pdata[1] = [];\n'\
5212 ' if(total.length == 3) total[3]=total[4]=0.0;\n'\
5213 ' pd = pdata[1];\n'\
5214 ' tidx = 3;\n'\
5215 ' }\n'\
5216 ' var info = dev[i].title.split(" ");\n'\
5217 ' var pname = info[info.length-1];\n'\
5218 ' pd[pname] = parseFloat(info[info.length-3].slice(1));\n'\
5219 ' total[0] += pd[pname];\n'\
5220 ' if(pname.indexOf("suspend") >= 0)\n'\
5221 ' total[tidx] += pd[pname];\n'\
5222 ' else\n'\
5223 ' total[tidx+1] += pd[pname];\n'\
5224 ' }\n'\
5225 ' }\n'\
5226 ' var devname = deviceTitle(this.title, total, cpu);\n'\
5227 ' var left = 0.0;\n'\
5228 ' for (var t = 0; t < pdata.length; t++) {\n'\
5229 ' pd = pdata[t];\n'\
5230 ' devinfo = document.getElementById("devicedetail"+t);\n'\
5231 ' var phases = devinfo.getElementsByClassName("phaselet");\n'\
5232 ' for (var i = 0; i < phases.length; i++) {\n'\
5233 ' if(phases[i].id in pd) {\n'\
5234 ' var w = 100.0*pd[phases[i].id]/total[0];\n'\
5235 ' var fs = 32;\n'\
5236 ' if(w < 8) fs = 4*w | 0;\n'\
5237 ' var fs2 = fs*3/4;\n'\
5238 ' phases[i].style.width = w+"%";\n'\
5239 ' phases[i].style.left = left+"%";\n'\
5240 ' phases[i].title = phases[i].id+" "+pd[phases[i].id]+" ms";\n'\
5241 ' left += w;\n'\
5242 ' var time = "<t4 style=\\"font-size:"+fs+"px\\">"+pd[phases[i].id]+" ms<br></t4>";\n'\
5243 ' var pname = "<t3 style=\\"font-size:"+fs2+"px\\">"+phases[i].id.replace(new RegExp("_", "g"), " ")+"</t3>";\n'\
5244 ' phases[i].innerHTML = time+pname;\n'\
5245 ' } else {\n'\
5246 ' phases[i].style.width = "0%";\n'\
5247 ' phases[i].style.left = left+"%";\n'\
5248 ' }\n'\
5249 ' }\n'\
5250 ' }\n'\
5251 ' if(typeof devstats !== \'undefined\')\n'\
5252 ' callDetail(this.id, this.title);\n'\
5253 ' var cglist = document.getElementById("callgraphs");\n'\
5254 ' if(!cglist) return;\n'\
5255 ' var cg = cglist.getElementsByClassName("atop");\n'\
5256 ' if(cg.length < 10) return;\n'\
5257 ' for (var i = 0; i < cg.length; i++) {\n'\
5258 ' cgid = cg[i].id.split("x")[0]\n'\
5259 ' if(idlist.indexOf(cgid) >= 0) {\n'\
5260 ' cg[i].style.display = "block";\n'\
5261 ' } else {\n'\
5262 ' cg[i].style.display = "none";\n'\
5263 ' }\n'\
5264 ' }\n'\
5265 ' }\n'\
5266 ' function callDetail(devid, devtitle) {\n'\
5267 ' if(!(devid in devstats) || devstats[devid].length < 1)\n'\
5268 ' return;\n'\
5269 ' var list = devstats[devid];\n'\
5270 ' var tmp = devtitle.split(" ");\n'\
5271 ' var name = tmp[0], phase = tmp[tmp.length-1];\n'\
5272 ' var dd = document.getElementById(phase);\n'\
5273 ' var total = parseFloat(tmp[1].slice(1));\n'\
5274 ' var mlist = [];\n'\
5275 ' var maxlen = 0;\n'\
5276 ' var info = []\n'\
5277 ' for(var i in list) {\n'\
5278 ' if(list[i][0] == "@") {\n'\
5279 ' info = list[i].split("|");\n'\
5280 ' continue;\n'\
5281 ' }\n'\
5282 ' var tmp = list[i].split("|");\n'\
5283 ' var t = parseFloat(tmp[0]), f = tmp[1], c = parseInt(tmp[2]);\n'\
5284 ' var p = (t*100.0/total).toFixed(2);\n'\
5285 ' mlist[mlist.length] = [f, c, t.toFixed(2), p+"%"];\n'\
5286 ' if(f.length > maxlen)\n'\
5287 ' maxlen = f.length;\n'\
5288 ' }\n'\
5289 ' var pad = 5;\n'\
5290 ' if(mlist.length == 0) pad = 30;\n'\
5291 ' var html = \'<div style="padding-top:\'+pad+\'px"><t3> <b>\'+name+\':</b>\';\n'\
5292 ' if(info.length > 2)\n'\
5293 ' html += " start=<b>"+info[1]+"</b>, end=<b>"+info[2]+"</b>";\n'\
5294 ' if(info.length > 3)\n'\
5295 ' html += ", length<i>(w/o overhead)</i>=<b>"+info[3]+" ms</b>";\n'\
5296 ' if(info.length > 4)\n'\
5297 ' html += ", return=<b>"+info[4]+"</b>";\n'\
5298 ' html += "</t3></div>";\n'\
5299 ' if(mlist.length > 0) {\n'\
5300 ' html += \'<table class=fstat style="padding-top:\'+(maxlen*5)+\'px;"><tr><th>Function</th>\';\n'\
5301 ' for(var i in mlist)\n'\
5302 ' html += "<td class=vt>"+mlist[i][0]+"</td>";\n'\
5303 ' html += "</tr><tr><th>Calls</th>";\n'\
5304 ' for(var i in mlist)\n'\
5305 ' html += "<td>"+mlist[i][1]+"</td>";\n'\
5306 ' html += "</tr><tr><th>Time(ms)</th>";\n'\
5307 ' for(var i in mlist)\n'\
5308 ' html += "<td>"+mlist[i][2]+"</td>";\n'\
5309 ' html += "</tr><tr><th>Percent</th>";\n'\
5310 ' for(var i in mlist)\n'\
5311 ' html += "<td>"+mlist[i][3]+"</td>";\n'\
5312 ' html += "</tr></table>";\n'\
5313 ' }\n'\
5314 ' dd.innerHTML = html;\n'\
5315 ' var height = (maxlen*5)+100;\n'\
5316 ' dd.style.height = height+"px";\n'\
5317 ' document.getElementById("devicedetail").style.height = height+"px";\n'\
5318 ' }\n'\
5319 ' function callSelect() {\n'\
5320 ' var cglist = document.getElementById("callgraphs");\n'\
5321 ' if(!cglist) return;\n'\
5322 ' var cg = cglist.getElementsByClassName("atop");\n'\
5323 ' for (var i = 0; i < cg.length; i++) {\n'\
5324 ' if(this.id == cg[i].id) {\n'\
5325 ' cg[i].style.display = "block";\n'\
5326 ' } else {\n'\
5327 ' cg[i].style.display = "none";\n'\
5328 ' }\n'\
5329 ' }\n'\
5330 ' }\n'\
5331 ' function devListWindow(e) {\n'\
5332 ' var win = window.open();\n'\
5333 ' var html = "<title>"+e.target.innerHTML+"</title>"+\n'\
5334 ' "<style type=\\"text/css\\">"+\n'\
5335 ' " ul {list-style-type:circle;padding-left:10px;margin-left:10px;}"+\n'\
5336 ' "</style>"\n'\
5337 ' var dt = devtable[0];\n'\
5338 ' if(e.target.id != "devlist1")\n'\
5339 ' dt = devtable[1];\n'\
5340 ' win.document.write(html+dt);\n'\
5341 ' }\n'\
5342 ' function errWindow() {\n'\
5343 ' var range = this.id.split("_");\n'\
5344 ' var idx1 = parseInt(range[0]);\n'\
5345 ' var idx2 = parseInt(range[1]);\n'\
5346 ' var win = window.open();\n'\
5347 ' var log = document.getElementById("dmesglog");\n'\
5348 ' var title = "<title>dmesg log</title>";\n'\
5349 ' var text = log.innerHTML.split("\\n");\n'\
5350 ' var html = "";\n'\
5351 ' for(var i = 0; i < text.length; i++) {\n'\
5352 ' if(i == idx1) {\n'\
5353 ' html += "<e id=target>"+text[i]+"</e>\\n";\n'\
5354 ' } else if(i > idx1 && i <= idx2) {\n'\
5355 ' html += "<e>"+text[i]+"</e>\\n";\n'\
5356 ' } else {\n'\
5357 ' html += text[i]+"\\n";\n'\
5358 ' }\n'\
5359 ' }\n'\
5360 ' win.document.write("<style>e{color:red}</style>"+title+"<pre>"+html+"</pre>");\n'\
5361 ' win.location.hash = "#target";\n'\
5362 ' win.document.close();\n'\
5363 ' }\n'\
5364 ' function logWindow(e) {\n'\
5365 ' var name = e.target.id.slice(4);\n'\
5366 ' var win = window.open();\n'\
5367 ' var log = document.getElementById(name+"log");\n'\
5368 ' var title = "<title>"+document.title.split(" ")[0]+" "+name+" log</title>";\n'\
5369 ' win.document.write(title+"<pre>"+log.innerHTML+"</pre>");\n'\
5370 ' win.document.close();\n'\
5371 ' }\n'\
5372 ' function onMouseDown(e) {\n'\
5373 ' dragval[0] = e.clientX;\n'\
5374 ' dragval[1] = document.getElementById("dmesgzoombox").scrollLeft;\n'\
5375 ' document.onmousemove = onMouseMove;\n'\
5376 ' }\n'\
5377 ' function onMouseMove(e) {\n'\
5378 ' var zoombox = document.getElementById("dmesgzoombox");\n'\
5379 ' zoombox.scrollLeft = dragval[1] + dragval[0] - e.clientX;\n'\
5380 ' }\n'\
5381 ' function onMouseUp(e) {\n'\
5382 ' document.onmousemove = null;\n'\
5383 ' }\n'\
5384 ' function onKeyPress(e) {\n'\
5385 ' var c = e.charCode;\n'\
5386 ' if(c != 42 && c != 43 && c != 45) return;\n'\
5387 ' var click = document.createEvent("Events");\n'\
5388 ' click.initEvent("click", true, false);\n'\
5389 ' if(c == 43) \n'\
5390 ' document.getElementById("zoomin").dispatchEvent(click);\n'\
5391 ' else if(c == 45)\n'\
5392 ' document.getElementById("zoomout").dispatchEvent(click);\n'\
5393 ' else if(c == 42)\n'\
5394 ' document.getElementById("zoomdef").dispatchEvent(click);\n'\
5395 ' }\n'\
5396 ' window.addEventListener("resize", function () {zoomTimeline();});\n'\
5397 ' window.addEventListener("load", function () {\n'\
5398 ' var dmesg = document.getElementById("dmesg");\n'\
5399 ' dmesg.style.width = "100%"\n'\
5400 ' dmesg.onmousedown = onMouseDown;\n'\
5401 ' document.onmouseup = onMouseUp;\n'\
5402 ' document.onkeypress = onKeyPress;\n'\
5403 ' document.getElementById("zoomin").onclick = zoomTimeline;\n'\
5404 ' document.getElementById("zoomout").onclick = zoomTimeline;\n'\
5405 ' document.getElementById("zoomdef").onclick = zoomTimeline;\n'\
5406 ' var list = document.getElementsByClassName("err");\n'\
5407 ' for (var i = 0; i < list.length; i++)\n'\
5408 ' list[i].onclick = errWindow;\n'\
5409 ' var list = document.getElementsByClassName("logbtn");\n'\
5410 ' for (var i = 0; i < list.length; i++)\n'\
5411 ' list[i].onclick = logWindow;\n'\
5412 ' list = document.getElementsByClassName("devlist");\n'\
5413 ' for (var i = 0; i < list.length; i++)\n'\
5414 ' list[i].onclick = devListWindow;\n'\
5415 ' var dev = dmesg.getElementsByClassName("thread");\n'\
5416 ' for (var i = 0; i < dev.length; i++) {\n'\
5417 ' dev[i].onclick = deviceDetail;\n'\
5418 ' dev[i].onmouseover = deviceHover;\n'\
5419 ' dev[i].onmouseout = deviceUnhover;\n'\
5420 ' }\n'\
5421 ' var dev = dmesg.getElementsByClassName("srccall");\n'\
5422 ' for (var i = 0; i < dev.length; i++)\n'\
5423 ' dev[i].onclick = callSelect;\n'\
5424 ' zoomTimeline();\n'\
5425 ' });\n'\
5426 '</script>\n'
5427 hf.write(script_code);
5428
5429# Function: executeSuspend
5430# Description:
5431# Execute system suspend through the sysfs interface, then copy the output
5432# dmesg and ftrace files to the test output directory.
5433def executeSuspend(quiet=False):
5434 sv, tp, pm = sysvals, sysvals.tpath, ProcessMonitor()
5435 if sv.wifi:
5436 wifi = sv.checkWifi()
5437 sv.dlog('wifi check, connected device is "%s"' % wifi)
5438 testdata = []
5439 # run these commands to prepare the system for suspend
5440 if sv.display:
5441 if not quiet:
5442 pprint('SET DISPLAY TO %s' % sv.display.upper())
5443 ret = sv.displayControl(sv.display)
5444 sv.dlog('xset display %s, ret = %d' % (sv.display, ret))
5445 time.sleep(1)
5446 if sv.sync:
5447 if not quiet:
5448 pprint('SYNCING FILESYSTEMS')
5449 sv.dlog('syncing filesystems')
5450 call('sync', shell=True)
5451 sv.dlog('read dmesg')
5452 sv.initdmesg()
5453 sv.dlog('cmdinfo before')
5454 sv.cmdinfo(True)
5455 sv.start(pm)
5456 # execute however many s/r runs requested
5457 for count in range(1,sv.execcount+1):
5458 # x2delay in between test runs
5459 if(count > 1 and sv.x2delay > 0):
5460 sv.fsetVal('WAIT %d' % sv.x2delay, 'trace_marker')
5461 time.sleep(sv.x2delay/1000.0)
5462 sv.fsetVal('WAIT END', 'trace_marker')
5463 # start message
5464 if sv.testcommand != '':
5465 pprint('COMMAND START')
5466 else:
5467 if(sv.rtcwake):
5468 pprint('SUSPEND START')
5469 else:
5470 pprint('SUSPEND START (press a key to resume)')
5471 # set rtcwake
5472 if(sv.rtcwake):
5473 if not quiet:
5474 pprint('will issue an rtcwake in %d seconds' % sv.rtcwaketime)
5475 sv.dlog('enable RTC wake alarm')
5476 sv.rtcWakeAlarmOn()
5477 # start of suspend trace marker
5478 sv.fsetVal(datetime.now().strftime(sv.tmstart), 'trace_marker')
5479 # predelay delay
5480 if(count == 1 and sv.predelay > 0):
5481 sv.fsetVal('WAIT %d' % sv.predelay, 'trace_marker')
5482 time.sleep(sv.predelay/1000.0)
5483 sv.fsetVal('WAIT END', 'trace_marker')
5484 # initiate suspend or command
5485 sv.dlog('system executing a suspend')
5486 tdata = {'error': ''}
5487 if sv.testcommand != '':
5488 res = call(sv.testcommand+' 2>&1', shell=True);
5489 if res != 0:
5490 tdata['error'] = 'cmd returned %d' % res
5491 else:
5492 s0ixready = sv.s0ixSupport()
5493 mode = sv.suspendmode
5494 if sv.memmode and os.path.exists(sv.mempowerfile):
5495 mode = 'mem'
5496 sv.testVal(sv.mempowerfile, 'radio', sv.memmode)
5497 if sv.diskmode and os.path.exists(sv.diskpowerfile):
5498 mode = 'disk'
5499 sv.testVal(sv.diskpowerfile, 'radio', sv.diskmode)
5500 if sv.acpidebug:
5501 sv.testVal(sv.acpipath, 'acpi', '0xe')
5502 if ((mode == 'freeze') or (sv.memmode == 's2idle')) \
5503 and sv.haveTurbostat():
5504 # execution will pause here
5505 turbo = sv.turbostat(s0ixready)
5506 if turbo:
5507 tdata['turbo'] = turbo
5508 else:
5509 pf = open(sv.powerfile, 'w')
5510 pf.write(mode)
5511 # execution will pause here
5512 try:
5513 pf.close()
5514 except Exception as e:
5515 tdata['error'] = str(e)
5516 sv.fsetVal('CMD COMPLETE', 'trace_marker')
5517 sv.dlog('system returned')
5518 # reset everything
5519 sv.testVal('restoreall')
5520 if(sv.rtcwake):
5521 sv.dlog('disable RTC wake alarm')
5522 sv.rtcWakeAlarmOff()
5523 # postdelay delay
5524 if(count == sv.execcount and sv.postdelay > 0):
5525 sv.fsetVal('WAIT %d' % sv.postdelay, 'trace_marker')
5526 time.sleep(sv.postdelay/1000.0)
5527 sv.fsetVal('WAIT END', 'trace_marker')
5528 # return from suspend
5529 pprint('RESUME COMPLETE')
5530 if(count < sv.execcount):
5531 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5532 elif(not sv.wifitrace):
5533 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5534 sv.stop(pm)
5535 if sv.wifi and wifi:
5536 tdata['wifi'] = sv.pollWifi(wifi)
5537 sv.dlog('wifi check, %s' % tdata['wifi'])
5538 if(count == sv.execcount and sv.wifitrace):
5539 sv.fsetVal(datetime.now().strftime(sv.tmend), 'trace_marker')
5540 sv.stop(pm)
5541 if sv.netfix:
5542 tdata['netfix'] = sv.netfixon()
5543 sv.dlog('netfix, %s' % tdata['netfix'])
5544 if(sv.suspendmode == 'mem' or sv.suspendmode == 'command'):
5545 sv.dlog('read the ACPI FPDT')
5546 tdata['fw'] = getFPDT(False)
5547 testdata.append(tdata)
5548 sv.dlog('cmdinfo after')
5549 cmdafter = sv.cmdinfo(False)
5550 # grab a copy of the dmesg output
5551 if not quiet:
5552 pprint('CAPTURING DMESG')
5553 sv.getdmesg(testdata)
5554 # grab a copy of the ftrace output
5555 if sv.useftrace:
5556 if not quiet:
5557 pprint('CAPTURING TRACE')
5558 op = sv.writeDatafileHeader(sv.ftracefile, testdata)
5559 fp = open(tp+'trace', 'r')
5560 for line in fp:
5561 op.write(line)
5562 op.close()
5563 sv.fsetVal('', 'trace')
5564 sv.platforminfo(cmdafter)
5565
5566def readFile(file):
5567 if os.path.islink(file):
5568 return os.readlink(file).split('/')[-1]
5569 else:
5570 return sysvals.getVal(file).strip()
5571
5572# Function: ms2nice
5573# Description:
5574# Print out a very concise time string in minutes and seconds
5575# Output:
5576# The time string, e.g. "1901m16s"
5577def ms2nice(val):
5578 val = int(val)
5579 h = val // 3600000
5580 m = (val // 60000) % 60
5581 s = (val // 1000) % 60
5582 if h > 0:
5583 return '%d:%02d:%02d' % (h, m, s)
5584 if m > 0:
5585 return '%02d:%02d' % (m, s)
5586 return '%ds' % s
5587
5588def yesno(val):
5589 list = {'enabled':'A', 'disabled':'S', 'auto':'E', 'on':'D',
5590 'active':'A', 'suspended':'S', 'suspending':'S'}
5591 if val not in list:
5592 return ' '
5593 return list[val]
5594
5595# Function: deviceInfo
5596# Description:
5597# Detect all the USB hosts and devices currently connected and add
5598# a list of USB device names to sysvals for better timeline readability
5599def deviceInfo(output=''):
5600 if not output:
5601 pprint('LEGEND\n'\
5602 '---------------------------------------------------------------------------------------------\n'\
5603 ' A = async/sync PM queue (A/S) C = runtime active children\n'\
5604 ' R = runtime suspend enabled/disabled (E/D) rACTIVE = runtime active (min/sec)\n'\
5605 ' S = runtime status active/suspended (A/S) rSUSPEND = runtime suspend (min/sec)\n'\
5606 ' U = runtime usage count\n'\
5607 '---------------------------------------------------------------------------------------------\n'\
5608 'DEVICE NAME A R S U C rACTIVE rSUSPEND\n'\
5609 '---------------------------------------------------------------------------------------------')
5610
5611 res = []
5612 tgtval = 'runtime_status'
5613 lines = dict()
5614 for dirname, dirnames, filenames in os.walk('/sys/devices'):
5615 if(not re.match('.*/power', dirname) or
5616 'control' not in filenames or
5617 tgtval not in filenames):
5618 continue
5619 name = ''
5620 dirname = dirname[:-6]
5621 device = dirname.split('/')[-1]
5622 power = dict()
5623 power[tgtval] = readFile('%s/power/%s' % (dirname, tgtval))
5624 # only list devices which support runtime suspend
5625 if power[tgtval] not in ['active', 'suspended', 'suspending']:
5626 continue
5627 for i in ['product', 'driver', 'subsystem']:
5628 file = '%s/%s' % (dirname, i)
5629 if os.path.exists(file):
5630 name = readFile(file)
5631 break
5632 for i in ['async', 'control', 'runtime_status', 'runtime_usage',
5633 'runtime_active_kids', 'runtime_active_time',
5634 'runtime_suspended_time']:
5635 if i in filenames:
5636 power[i] = readFile('%s/power/%s' % (dirname, i))
5637 if output:
5638 if power['control'] == output:
5639 res.append('%s/power/control' % dirname)
5640 continue
5641 lines[dirname] = '%-26s %-26s %1s %1s %1s %1s %1s %10s %10s' % \
5642 (device[:26], name[:26],
5643 yesno(power['async']), \
5644 yesno(power['control']), \
5645 yesno(power['runtime_status']), \
5646 power['runtime_usage'], \
5647 power['runtime_active_kids'], \
5648 ms2nice(power['runtime_active_time']), \
5649 ms2nice(power['runtime_suspended_time']))
5650 for i in sorted(lines):
5651 print(lines[i])
5652 return res
5653
5654# Function: getModes
5655# Description:
5656# Determine the supported power modes on this system
5657# Output:
5658# A string list of the available modes
5659def getModes():
5660 modes = []
5661 if(os.path.exists(sysvals.powerfile)):
5662 fp = open(sysvals.powerfile, 'r')
5663 modes = fp.read().split()
5664 fp.close()
5665 if(os.path.exists(sysvals.mempowerfile)):
5666 deep = False
5667 fp = open(sysvals.mempowerfile, 'r')
5668 for m in fp.read().split():
5669 memmode = m.strip('[]')
5670 if memmode == 'deep':
5671 deep = True
5672 else:
5673 modes.append('mem-%s' % memmode)
5674 fp.close()
5675 if 'mem' in modes and not deep:
5676 modes.remove('mem')
5677 if('disk' in modes and os.path.exists(sysvals.diskpowerfile)):
5678 fp = open(sysvals.diskpowerfile, 'r')
5679 for m in fp.read().split():
5680 modes.append('disk-%s' % m.strip('[]'))
5681 fp.close()
5682 return modes
5683
5684# Function: dmidecode
5685# Description:
5686# Read the bios tables and pull out system info
5687# Arguments:
5688# mempath: /dev/mem or custom mem path
5689# fatal: True to exit on error, False to return empty dict
5690# Output:
5691# A dict object with all available key/values
5692def dmidecode(mempath, fatal=False):
5693 out = dict()
5694
5695 # the list of values to retrieve, with hardcoded (type, idx)
5696 info = {
5697 'bios-vendor': (0, 4),
5698 'bios-version': (0, 5),
5699 'bios-release-date': (0, 8),
5700 'system-manufacturer': (1, 4),
5701 'system-product-name': (1, 5),
5702 'system-version': (1, 6),
5703 'system-serial-number': (1, 7),
5704 'baseboard-manufacturer': (2, 4),
5705 'baseboard-product-name': (2, 5),
5706 'baseboard-version': (2, 6),
5707 'baseboard-serial-number': (2, 7),
5708 'chassis-manufacturer': (3, 4),
5709 'chassis-type': (3, 5),
5710 'chassis-version': (3, 6),
5711 'chassis-serial-number': (3, 7),
5712 'processor-manufacturer': (4, 7),
5713 'processor-version': (4, 16),
5714 }
5715 if(not os.path.exists(mempath)):
5716 if(fatal):
5717 doError('file does not exist: %s' % mempath)
5718 return out
5719 if(not os.access(mempath, os.R_OK)):
5720 if(fatal):
5721 doError('file is not readable: %s' % mempath)
5722 return out
5723
5724 # by default use legacy scan, but try to use EFI first
5725 memaddr = 0xf0000
5726 memsize = 0x10000
5727 for ep in ['/sys/firmware/efi/systab', '/proc/efi/systab']:
5728 if not os.path.exists(ep) or not os.access(ep, os.R_OK):
5729 continue
5730 fp = open(ep, 'r')
5731 buf = fp.read()
5732 fp.close()
5733 i = buf.find('SMBIOS=')
5734 if i >= 0:
5735 try:
5736 memaddr = int(buf[i+7:], 16)
5737 memsize = 0x20
5738 except:
5739 continue
5740
5741 # read in the memory for scanning
5742 try:
5743 fp = open(mempath, 'rb')
5744 fp.seek(memaddr)
5745 buf = fp.read(memsize)
5746 except:
5747 if(fatal):
5748 doError('DMI table is unreachable, sorry')
5749 else:
5750 pprint('WARNING: /dev/mem is not readable, ignoring DMI data')
5751 return out
5752 fp.close()
5753
5754 # search for either an SM table or DMI table
5755 i = base = length = num = 0
5756 while(i < memsize):
5757 if buf[i:i+4] == b'_SM_' and i < memsize - 16:
5758 length = struct.unpack('H', buf[i+22:i+24])[0]
5759 base, num = struct.unpack('IH', buf[i+24:i+30])
5760 break
5761 elif buf[i:i+5] == b'_DMI_':
5762 length = struct.unpack('H', buf[i+6:i+8])[0]
5763 base, num = struct.unpack('IH', buf[i+8:i+14])
5764 break
5765 i += 16
5766 if base == 0 and length == 0 and num == 0:
5767 if(fatal):
5768 doError('Neither SMBIOS nor DMI were found')
5769 else:
5770 return out
5771
5772 # read in the SM or DMI table
5773 try:
5774 fp = open(mempath, 'rb')
5775 fp.seek(base)
5776 buf = fp.read(length)
5777 except:
5778 if(fatal):
5779 doError('DMI table is unreachable, sorry')
5780 else:
5781 pprint('WARNING: /dev/mem is not readable, ignoring DMI data')
5782 return out
5783 fp.close()
5784
5785 # scan the table for the values we want
5786 count = i = 0
5787 while(count < num and i <= len(buf) - 4):
5788 type, size, handle = struct.unpack('BBH', buf[i:i+4])
5789 n = i + size
5790 while n < len(buf) - 1:
5791 if 0 == struct.unpack('H', buf[n:n+2])[0]:
5792 break
5793 n += 1
5794 data = buf[i+size:n+2].split(b'\0')
5795 for name in info:
5796 itype, idxadr = info[name]
5797 if itype == type:
5798 idx = struct.unpack('B', buf[i+idxadr:i+idxadr+1])[0]
5799 if idx > 0 and idx < len(data) - 1:
5800 s = data[idx-1].decode('utf-8')
5801 if s.strip() and s.strip().lower() != 'to be filled by o.e.m.':
5802 out[name] = s
5803 i = n + 2
5804 count += 1
5805 return out
5806
5807# Function: getFPDT
5808# Description:
5809# Read the acpi bios tables and pull out FPDT, the firmware data
5810# Arguments:
5811# output: True to output the info to stdout, False otherwise
5812def getFPDT(output):
5813 rectype = {}
5814 rectype[0] = 'Firmware Basic Boot Performance Record'
5815 rectype[1] = 'S3 Performance Table Record'
5816 prectype = {}
5817 prectype[0] = 'Basic S3 Resume Performance Record'
5818 prectype[1] = 'Basic S3 Suspend Performance Record'
5819
5820 sysvals.rootCheck(True)
5821 if(not os.path.exists(sysvals.fpdtpath)):
5822 if(output):
5823 doError('file does not exist: %s' % sysvals.fpdtpath)
5824 return False
5825 if(not os.access(sysvals.fpdtpath, os.R_OK)):
5826 if(output):
5827 doError('file is not readable: %s' % sysvals.fpdtpath)
5828 return False
5829 if(not os.path.exists(sysvals.mempath)):
5830 if(output):
5831 doError('file does not exist: %s' % sysvals.mempath)
5832 return False
5833 if(not os.access(sysvals.mempath, os.R_OK)):
5834 if(output):
5835 doError('file is not readable: %s' % sysvals.mempath)
5836 return False
5837
5838 fp = open(sysvals.fpdtpath, 'rb')
5839 buf = fp.read()
5840 fp.close()
5841
5842 if(len(buf) < 36):
5843 if(output):
5844 doError('Invalid FPDT table data, should '+\
5845 'be at least 36 bytes')
5846 return False
5847
5848 table = struct.unpack('4sIBB6s8sI4sI', buf[0:36])
5849 if(output):
5850 pprint('\n'\
5851 'Firmware Performance Data Table (%s)\n'\
5852 ' Signature : %s\n'\
5853 ' Table Length : %u\n'\
5854 ' Revision : %u\n'\
5855 ' Checksum : 0x%x\n'\
5856 ' OEM ID : %s\n'\
5857 ' OEM Table ID : %s\n'\
5858 ' OEM Revision : %u\n'\
5859 ' Creator ID : %s\n'\
5860 ' Creator Revision : 0x%x\n'\
5861 '' % (ascii(table[0]), ascii(table[0]), table[1], table[2],
5862 table[3], ascii(table[4]), ascii(table[5]), table[6],
5863 ascii(table[7]), table[8]))
5864
5865 if(table[0] != b'FPDT'):
5866 if(output):
5867 doError('Invalid FPDT table')
5868 return False
5869 if(len(buf) <= 36):
5870 return False
5871 i = 0
5872 fwData = [0, 0]
5873 records = buf[36:]
5874 try:
5875 fp = open(sysvals.mempath, 'rb')
5876 except:
5877 pprint('WARNING: /dev/mem is not readable, ignoring the FPDT data')
5878 return False
5879 while(i < len(records)):
5880 header = struct.unpack('HBB', records[i:i+4])
5881 if(header[0] not in rectype):
5882 i += header[1]
5883 continue
5884 if(header[1] != 16):
5885 i += header[1]
5886 continue
5887 addr = struct.unpack('Q', records[i+8:i+16])[0]
5888 try:
5889 fp.seek(addr)
5890 first = fp.read(8)
5891 except:
5892 if(output):
5893 pprint('Bad address 0x%x in %s' % (addr, sysvals.mempath))
5894 return [0, 0]
5895 rechead = struct.unpack('4sI', first)
5896 recdata = fp.read(rechead[1]-8)
5897 if(rechead[0] == b'FBPT'):
5898 record = struct.unpack('HBBIQQQQQ', recdata[:48])
5899 if(output):
5900 pprint('%s (%s)\n'\
5901 ' Reset END : %u ns\n'\
5902 ' OS Loader LoadImage Start : %u ns\n'\
5903 ' OS Loader StartImage Start : %u ns\n'\
5904 ' ExitBootServices Entry : %u ns\n'\
5905 ' ExitBootServices Exit : %u ns'\
5906 '' % (rectype[header[0]], ascii(rechead[0]), record[4], record[5],
5907 record[6], record[7], record[8]))
5908 elif(rechead[0] == b'S3PT'):
5909 if(output):
5910 pprint('%s (%s)' % (rectype[header[0]], ascii(rechead[0])))
5911 j = 0
5912 while(j < len(recdata)):
5913 prechead = struct.unpack('HBB', recdata[j:j+4])
5914 if(prechead[0] not in prectype):
5915 continue
5916 if(prechead[0] == 0):
5917 record = struct.unpack('IIQQ', recdata[j:j+prechead[1]])
5918 fwData[1] = record[2]
5919 if(output):
5920 pprint(' %s\n'\
5921 ' Resume Count : %u\n'\
5922 ' FullResume : %u ns\n'\
5923 ' AverageResume : %u ns'\
5924 '' % (prectype[prechead[0]], record[1],
5925 record[2], record[3]))
5926 elif(prechead[0] == 1):
5927 record = struct.unpack('QQ', recdata[j+4:j+prechead[1]])
5928 fwData[0] = record[1] - record[0]
5929 if(output):
5930 pprint(' %s\n'\
5931 ' SuspendStart : %u ns\n'\
5932 ' SuspendEnd : %u ns\n'\
5933 ' SuspendTime : %u ns'\
5934 '' % (prectype[prechead[0]], record[0],
5935 record[1], fwData[0]))
5936
5937 j += prechead[1]
5938 if(output):
5939 pprint('')
5940 i += header[1]
5941 fp.close()
5942 return fwData
5943
5944# Function: statusCheck
5945# Description:
5946# Verify that the requested command and options will work, and
5947# print the results to the terminal
5948# Output:
5949# True if the test will work, False if not
5950def statusCheck(probecheck=False):
5951 status = ''
5952
5953 pprint('Checking this system (%s)...' % platform.node())
5954
5955 # check we have root access
5956 res = sysvals.colorText('NO (No features of this tool will work!)')
5957 if(sysvals.rootCheck(False)):
5958 res = 'YES'
5959 pprint(' have root access: %s' % res)
5960 if(res != 'YES'):
5961 pprint(' Try running this script with sudo')
5962 return 'missing root access'
5963
5964 # check sysfs is mounted
5965 res = sysvals.colorText('NO (No features of this tool will work!)')
5966 if(os.path.exists(sysvals.powerfile)):
5967 res = 'YES'
5968 pprint(' is sysfs mounted: %s' % res)
5969 if(res != 'YES'):
5970 return 'sysfs is missing'
5971
5972 # check target mode is a valid mode
5973 if sysvals.suspendmode != 'command':
5974 res = sysvals.colorText('NO')
5975 modes = getModes()
5976 if(sysvals.suspendmode in modes):
5977 res = 'YES'
5978 else:
5979 status = '%s mode is not supported' % sysvals.suspendmode
5980 pprint(' is "%s" a valid power mode: %s' % (sysvals.suspendmode, res))
5981 if(res == 'NO'):
5982 pprint(' valid power modes are: %s' % modes)
5983 pprint(' please choose one with -m')
5984
5985 # check if ftrace is available
5986 if sysvals.useftrace:
5987 res = sysvals.colorText('NO')
5988 sysvals.useftrace = sysvals.verifyFtrace()
5989 efmt = '"{0}" uses ftrace, and it is not properly supported'
5990 if sysvals.useftrace:
5991 res = 'YES'
5992 elif sysvals.usecallgraph:
5993 status = efmt.format('-f')
5994 elif sysvals.usedevsrc:
5995 status = efmt.format('-dev')
5996 elif sysvals.useprocmon:
5997 status = efmt.format('-proc')
5998 pprint(' is ftrace supported: %s' % res)
5999
6000 # check if kprobes are available
6001 if sysvals.usekprobes:
6002 res = sysvals.colorText('NO')
6003 sysvals.usekprobes = sysvals.verifyKprobes()
6004 if(sysvals.usekprobes):
6005 res = 'YES'
6006 else:
6007 sysvals.usedevsrc = False
6008 pprint(' are kprobes supported: %s' % res)
6009
6010 # what data source are we using
6011 res = 'DMESG (very limited, ftrace is preferred)'
6012 if sysvals.useftrace:
6013 sysvals.usetraceevents = True
6014 for e in sysvals.traceevents:
6015 if not os.path.exists(sysvals.epath+e):
6016 sysvals.usetraceevents = False
6017 if(sysvals.usetraceevents):
6018 res = 'FTRACE (all trace events found)'
6019 pprint(' timeline data source: %s' % res)
6020
6021 # check if rtcwake
6022 res = sysvals.colorText('NO')
6023 if(sysvals.rtcpath != ''):
6024 res = 'YES'
6025 elif(sysvals.rtcwake):
6026 status = 'rtcwake is not properly supported'
6027 pprint(' is rtcwake supported: %s' % res)
6028
6029 # check info commands
6030 pprint(' optional commands this tool may use for info:')
6031 no = sysvals.colorText('MISSING')
6032 yes = sysvals.colorText('FOUND', 32)
6033 for c in ['turbostat', 'mcelog', 'lspci', 'lsusb', 'netfix']:
6034 if c == 'turbostat':
6035 res = yes if sysvals.haveTurbostat() else no
6036 else:
6037 res = yes if sysvals.getExec(c) else no
6038 pprint(' %s: %s' % (c, res))
6039
6040 if not probecheck:
6041 return status
6042
6043 # verify kprobes
6044 if sysvals.usekprobes:
6045 for name in sysvals.tracefuncs:
6046 sysvals.defaultKprobe(name, sysvals.tracefuncs[name])
6047 if sysvals.usedevsrc:
6048 for name in sysvals.dev_tracefuncs:
6049 sysvals.defaultKprobe(name, sysvals.dev_tracefuncs[name])
6050 sysvals.addKprobes(True)
6051
6052 return status
6053
6054# Function: doError
6055# Description:
6056# generic error function for catastrphic failures
6057# Arguments:
6058# msg: the error message to print
6059# help: True if printHelp should be called after, False otherwise
6060def doError(msg, help=False):
6061 if(help == True):
6062 printHelp()
6063 pprint('ERROR: %s\n' % msg)
6064 sysvals.outputResult({'error':msg})
6065 sys.exit(1)
6066
6067# Function: getArgInt
6068# Description:
6069# pull out an integer argument from the command line with checks
6070def getArgInt(name, args, min, max, main=True):
6071 if main:
6072 try:
6073 arg = next(args)
6074 except:
6075 doError(name+': no argument supplied', True)
6076 else:
6077 arg = args
6078 try:
6079 val = int(arg)
6080 except:
6081 doError(name+': non-integer value given', True)
6082 if(val < min or val > max):
6083 doError(name+': value should be between %d and %d' % (min, max), True)
6084 return val
6085
6086# Function: getArgFloat
6087# Description:
6088# pull out a float argument from the command line with checks
6089def getArgFloat(name, args, min, max, main=True):
6090 if main:
6091 try:
6092 arg = next(args)
6093 except:
6094 doError(name+': no argument supplied', True)
6095 else:
6096 arg = args
6097 try:
6098 val = float(arg)
6099 except:
6100 doError(name+': non-numerical value given', True)
6101 if(val < min or val > max):
6102 doError(name+': value should be between %f and %f' % (min, max), True)
6103 return val
6104
6105def processData(live=False, quiet=False):
6106 if not quiet:
6107 pprint('PROCESSING: %s' % sysvals.htmlfile)
6108 sysvals.vprint('usetraceevents=%s, usetracemarkers=%s, usekprobes=%s' % \
6109 (sysvals.usetraceevents, sysvals.usetracemarkers, sysvals.usekprobes))
6110 error = ''
6111 if(sysvals.usetraceevents):
6112 testruns, error = parseTraceLog(live)
6113 if sysvals.dmesgfile:
6114 for data in testruns:
6115 data.extractErrorInfo()
6116 else:
6117 testruns = loadKernelLog()
6118 for data in testruns:
6119 parseKernelLog(data)
6120 if(sysvals.ftracefile and (sysvals.usecallgraph or sysvals.usetraceevents)):
6121 appendIncompleteTraceLog(testruns)
6122 if not sysvals.stamp:
6123 pprint('ERROR: data does not include the expected stamp')
6124 return (testruns, {'error': 'timeline generation failed'})
6125 shown = ['os', 'bios', 'biosdate', 'cpu', 'host', 'kernel', 'man', 'memfr',
6126 'memsz', 'mode', 'numcpu', 'plat', 'time', 'wifi']
6127 sysvals.vprint('System Info:')
6128 for key in sorted(sysvals.stamp):
6129 if key in shown:
6130 sysvals.vprint(' %-8s : %s' % (key.upper(), sysvals.stamp[key]))
6131 sysvals.vprint('Command:\n %s' % sysvals.cmdline)
6132 for data in testruns:
6133 if data.turbostat:
6134 idx, s = 0, 'Turbostat:\n '
6135 for val in data.turbostat.split('|'):
6136 idx += len(val) + 1
6137 if idx >= 80:
6138 idx = 0
6139 s += '\n '
6140 s += val + ' '
6141 sysvals.vprint(s)
6142 data.printDetails()
6143 if len(sysvals.platinfo) > 0:
6144 sysvals.vprint('\nPlatform Info:')
6145 for info in sysvals.platinfo:
6146 sysvals.vprint('[%s - %s]' % (info[0], info[1]))
6147 sysvals.vprint(info[2])
6148 sysvals.vprint('')
6149 if sysvals.cgdump:
6150 for data in testruns:
6151 data.debugPrint()
6152 sys.exit(0)
6153 if len(testruns) < 1:
6154 pprint('ERROR: Not enough test data to build a timeline')
6155 return (testruns, {'error': 'timeline generation failed'})
6156 sysvals.vprint('Creating the html timeline (%s)...' % sysvals.htmlfile)
6157 createHTML(testruns, error)
6158 if not quiet:
6159 pprint('DONE: %s' % sysvals.htmlfile)
6160 data = testruns[0]
6161 stamp = data.stamp
6162 stamp['suspend'], stamp['resume'] = data.getTimeValues()
6163 if data.fwValid:
6164 stamp['fwsuspend'], stamp['fwresume'] = data.fwSuspend, data.fwResume
6165 if error:
6166 stamp['error'] = error
6167 return (testruns, stamp)
6168
6169# Function: rerunTest
6170# Description:
6171# generate an output from an existing set of ftrace/dmesg logs
6172def rerunTest(htmlfile=''):
6173 if sysvals.ftracefile:
6174 doesTraceLogHaveTraceEvents()
6175 if not sysvals.dmesgfile and not sysvals.usetraceevents:
6176 doError('recreating this html output requires a dmesg file')
6177 if htmlfile:
6178 sysvals.htmlfile = htmlfile
6179 else:
6180 sysvals.setOutputFile()
6181 if os.path.exists(sysvals.htmlfile):
6182 if not os.path.isfile(sysvals.htmlfile):
6183 doError('a directory already exists with this name: %s' % sysvals.htmlfile)
6184 elif not os.access(sysvals.htmlfile, os.W_OK):
6185 doError('missing permission to write to %s' % sysvals.htmlfile)
6186 testruns, stamp = processData()
6187 sysvals.resetlog()
6188 return stamp
6189
6190# Function: runTest
6191# Description:
6192# execute a suspend/resume, gather the logs, and generate the output
6193def runTest(n=0, quiet=False):
6194 # prepare for the test
6195 sysvals.initTestOutput('suspend')
6196 op = sysvals.writeDatafileHeader(sysvals.dmesgfile, [])
6197 op.write('# EXECUTION TRACE START\n')
6198 op.close()
6199 if n <= 1:
6200 if sysvals.rs != 0:
6201 sysvals.dlog('%sabling runtime suspend' % ('en' if sysvals.rs > 0 else 'dis'))
6202 sysvals.setRuntimeSuspend(True)
6203 if sysvals.display:
6204 ret = sysvals.displayControl('init')
6205 sysvals.dlog('xset display init, ret = %d' % ret)
6206 sysvals.testVal(sysvals.pmdpath, 'basic', '1')
6207 sysvals.testVal(sysvals.s0ixpath, 'basic', 'Y')
6208 sysvals.dlog('initialize ftrace')
6209 sysvals.initFtrace(quiet)
6210
6211 # execute the test
6212 executeSuspend(quiet)
6213 sysvals.cleanupFtrace()
6214 if sysvals.skiphtml:
6215 sysvals.outputResult({}, n)
6216 sysvals.sudoUserchown(sysvals.testdir)
6217 return
6218 testruns, stamp = processData(True, quiet)
6219 for data in testruns:
6220 del data
6221 sysvals.sudoUserchown(sysvals.testdir)
6222 sysvals.outputResult(stamp, n)
6223 if 'error' in stamp:
6224 return 2
6225 return 0
6226
6227def find_in_html(html, start, end, firstonly=True):
6228 cnt, out, list = len(html), [], []
6229 if firstonly:
6230 m = re.search(start, html)
6231 if m:
6232 list.append(m)
6233 else:
6234 list = re.finditer(start, html)
6235 for match in list:
6236 s = match.end()
6237 e = cnt if (len(out) < 1 or s + 10000 > cnt) else s + 10000
6238 m = re.search(end, html[s:e])
6239 if not m:
6240 break
6241 e = s + m.start()
6242 str = html[s:e]
6243 if end == 'ms':
6244 num = re.search(r'[-+]?\d*\.\d+|\d+', str)
6245 str = num.group() if num else 'NaN'
6246 if firstonly:
6247 return str
6248 out.append(str)
6249 if firstonly:
6250 return ''
6251 return out
6252
6253def data_from_html(file, outpath, issues, fulldetail=False):
6254 html = open(file, 'r').read()
6255 sysvals.htmlfile = os.path.relpath(file, outpath)
6256 # extract general info
6257 suspend = find_in_html(html, 'Kernel Suspend', 'ms')
6258 resume = find_in_html(html, 'Kernel Resume', 'ms')
6259 sysinfo = find_in_html(html, '<div class="stamp sysinfo">', '</div>')
6260 line = find_in_html(html, '<div class="stamp">', '</div>')
6261 stmp = line.split()
6262 if not suspend or not resume or len(stmp) != 8:
6263 return False
6264 try:
6265 dt = datetime.strptime(' '.join(stmp[3:]), '%B %d %Y, %I:%M:%S %p')
6266 except:
6267 return False
6268 sysvals.hostname = stmp[0]
6269 tstr = dt.strftime('%Y/%m/%d %H:%M:%S')
6270 error = find_in_html(html, '<table class="testfail"><tr><td>', '</td>')
6271 if error:
6272 m = re.match('[a-z0-9]* failed in (?P<p>\S*).*', error)
6273 if m:
6274 result = 'fail in %s' % m.group('p')
6275 else:
6276 result = 'fail'
6277 else:
6278 result = 'pass'
6279 # extract error info
6280 tp, ilist = False, []
6281 extra = dict()
6282 log = find_in_html(html, '<div id="dmesglog" style="display:none;">',
6283 '</div>').strip()
6284 if log:
6285 d = Data(0)
6286 d.end = 999999999
6287 d.dmesgtext = log.split('\n')
6288 tp = d.extractErrorInfo()
6289 for msg in tp.msglist:
6290 sysvals.errorSummary(issues, msg)
6291 if stmp[2] == 'freeze':
6292 extra = d.turbostatInfo()
6293 elist = dict()
6294 for dir in d.errorinfo:
6295 for err in d.errorinfo[dir]:
6296 if err[0] not in elist:
6297 elist[err[0]] = 0
6298 elist[err[0]] += 1
6299 for i in elist:
6300 ilist.append('%sx%d' % (i, elist[i]) if elist[i] > 1 else i)
6301 line = find_in_html(log, '# wifi ', '\n')
6302 if line:
6303 extra['wifi'] = line
6304 line = find_in_html(log, '# netfix ', '\n')
6305 if line:
6306 extra['netfix'] = line
6307 low = find_in_html(html, 'freeze time: <b>', ' ms</b>')
6308 for lowstr in ['waking', '+']:
6309 if not low:
6310 break
6311 if lowstr not in low:
6312 continue
6313 if lowstr == '+':
6314 issue = 'S2LOOPx%d' % len(low.split('+'))
6315 else:
6316 m = re.match('.*waking *(?P<n>[0-9]*) *times.*', low)
6317 issue = 'S2WAKEx%s' % m.group('n') if m else 'S2WAKExNaN'
6318 match = [i for i in issues if i['match'] == issue]
6319 if len(match) > 0:
6320 match[0]['count'] += 1
6321 if sysvals.hostname not in match[0]['urls']:
6322 match[0]['urls'][sysvals.hostname] = [sysvals.htmlfile]
6323 elif sysvals.htmlfile not in match[0]['urls'][sysvals.hostname]:
6324 match[0]['urls'][sysvals.hostname].append(sysvals.htmlfile)
6325 else:
6326 issues.append({
6327 'match': issue, 'count': 1, 'line': issue,
6328 'urls': {sysvals.hostname: [sysvals.htmlfile]},
6329 })
6330 ilist.append(issue)
6331 # extract device info
6332 devices = dict()
6333 for line in html.split('\n'):
6334 m = re.match(' *<div id=\"[a,0-9]*\" *title=\"(?P<title>.*)\" class=\"thread.*', line)
6335 if not m or 'thread kth' in line or 'thread sec' in line:
6336 continue
6337 m = re.match('(?P<n>.*) \((?P<t>[0-9,\.]*) ms\) (?P<p>.*)', m.group('title'))
6338 if not m:
6339 continue
6340 name, time, phase = m.group('n'), m.group('t'), m.group('p')
6341 if name == 'async_synchronize_full':
6342 continue
6343 if ' async' in name or ' sync' in name:
6344 name = ' '.join(name.split(' ')[:-1])
6345 if phase.startswith('suspend'):
6346 d = 'suspend'
6347 elif phase.startswith('resume'):
6348 d = 'resume'
6349 else:
6350 continue
6351 if d not in devices:
6352 devices[d] = dict()
6353 if name not in devices[d]:
6354 devices[d][name] = 0.0
6355 devices[d][name] += float(time)
6356 # create worst device info
6357 worst = dict()
6358 for d in ['suspend', 'resume']:
6359 worst[d] = {'name':'', 'time': 0.0}
6360 dev = devices[d] if d in devices else 0
6361 if dev and len(dev.keys()) > 0:
6362 n = sorted(dev, key=lambda k:(dev[k], k), reverse=True)[0]
6363 worst[d]['name'], worst[d]['time'] = n, dev[n]
6364 data = {
6365 'mode': stmp[2],
6366 'host': stmp[0],
6367 'kernel': stmp[1],
6368 'sysinfo': sysinfo,
6369 'time': tstr,
6370 'result': result,
6371 'issues': ' '.join(ilist),
6372 'suspend': suspend,
6373 'resume': resume,
6374 'devlist': devices,
6375 'sus_worst': worst['suspend']['name'],
6376 'sus_worsttime': worst['suspend']['time'],
6377 'res_worst': worst['resume']['name'],
6378 'res_worsttime': worst['resume']['time'],
6379 'url': sysvals.htmlfile,
6380 }
6381 for key in extra:
6382 data[key] = extra[key]
6383 if fulldetail:
6384 data['funclist'] = find_in_html(html, '<div title="', '" class="traceevent"', False)
6385 if tp:
6386 for arg in ['-multi ', '-info ']:
6387 if arg in tp.cmdline:
6388 data['target'] = tp.cmdline[tp.cmdline.find(arg):].split()[1]
6389 break
6390 return data
6391
6392def genHtml(subdir, force=False):
6393 for dirname, dirnames, filenames in os.walk(subdir):
6394 sysvals.dmesgfile = sysvals.ftracefile = sysvals.htmlfile = ''
6395 for filename in filenames:
6396 file = os.path.join(dirname, filename)
6397 if sysvals.usable(file):
6398 if(re.match('.*_dmesg.txt', filename)):
6399 sysvals.dmesgfile = file
6400 elif(re.match('.*_ftrace.txt', filename)):
6401 sysvals.ftracefile = file
6402 sysvals.setOutputFile()
6403 if (sysvals.dmesgfile or sysvals.ftracefile) and sysvals.htmlfile and \
6404 (force or not sysvals.usable(sysvals.htmlfile, True)):
6405 pprint('FTRACE: %s' % sysvals.ftracefile)
6406 if sysvals.dmesgfile:
6407 pprint('DMESG : %s' % sysvals.dmesgfile)
6408 rerunTest()
6409
6410# Function: runSummary
6411# Description:
6412# create a summary of tests in a sub-directory
6413def runSummary(subdir, local=True, genhtml=False):
6414 inpath = os.path.abspath(subdir)
6415 outpath = os.path.abspath('.') if local else inpath
6416 pprint('Generating a summary of folder:\n %s' % inpath)
6417 if genhtml:
6418 genHtml(subdir)
6419 target, issues, testruns = '', [], []
6420 desc = {'host':[],'mode':[],'kernel':[]}
6421 for dirname, dirnames, filenames in os.walk(subdir):
6422 for filename in filenames:
6423 if(not re.match('.*.html', filename)):
6424 continue
6425 data = data_from_html(os.path.join(dirname, filename), outpath, issues)
6426 if(not data):
6427 continue
6428 if 'target' in data:
6429 target = data['target']
6430 testruns.append(data)
6431 for key in desc:
6432 if data[key] not in desc[key]:
6433 desc[key].append(data[key])
6434 pprint('Summary files:')
6435 if len(desc['host']) == len(desc['mode']) == len(desc['kernel']) == 1:
6436 title = '%s %s %s' % (desc['host'][0], desc['kernel'][0], desc['mode'][0])
6437 if target:
6438 title += ' %s' % target
6439 else:
6440 title = inpath
6441 createHTMLSummarySimple(testruns, os.path.join(outpath, 'summary.html'), title)
6442 pprint(' summary.html - tabular list of test data found')
6443 createHTMLDeviceSummary(testruns, os.path.join(outpath, 'summary-devices.html'), title)
6444 pprint(' summary-devices.html - kernel device list sorted by total execution time')
6445 createHTMLIssuesSummary(testruns, issues, os.path.join(outpath, 'summary-issues.html'), title)
6446 pprint(' summary-issues.html - kernel issues found sorted by frequency')
6447
6448# Function: checkArgBool
6449# Description:
6450# check if a boolean string value is true or false
6451def checkArgBool(name, value):
6452 if value in switchvalues:
6453 if value in switchoff:
6454 return False
6455 return True
6456 doError('invalid boolean --> (%s: %s), use "true/false" or "1/0"' % (name, value), True)
6457 return False
6458
6459# Function: configFromFile
6460# Description:
6461# Configure the script via the info in a config file
6462def configFromFile(file):
6463 Config = configparser.ConfigParser()
6464
6465 Config.read(file)
6466 sections = Config.sections()
6467 overridekprobes = False
6468 overridedevkprobes = False
6469 if 'Settings' in sections:
6470 for opt in Config.options('Settings'):
6471 value = Config.get('Settings', opt).lower()
6472 option = opt.lower()
6473 if(option == 'verbose'):
6474 sysvals.verbose = checkArgBool(option, value)
6475 elif(option == 'addlogs'):
6476 sysvals.dmesglog = sysvals.ftracelog = checkArgBool(option, value)
6477 elif(option == 'dev'):
6478 sysvals.usedevsrc = checkArgBool(option, value)
6479 elif(option == 'proc'):
6480 sysvals.useprocmon = checkArgBool(option, value)
6481 elif(option == 'x2'):
6482 if checkArgBool(option, value):
6483 sysvals.execcount = 2
6484 elif(option == 'callgraph'):
6485 sysvals.usecallgraph = checkArgBool(option, value)
6486 elif(option == 'override-timeline-functions'):
6487 overridekprobes = checkArgBool(option, value)
6488 elif(option == 'override-dev-timeline-functions'):
6489 overridedevkprobes = checkArgBool(option, value)
6490 elif(option == 'skiphtml'):
6491 sysvals.skiphtml = checkArgBool(option, value)
6492 elif(option == 'sync'):
6493 sysvals.sync = checkArgBool(option, value)
6494 elif(option == 'rs' or option == 'runtimesuspend'):
6495 if value in switchvalues:
6496 if value in switchoff:
6497 sysvals.rs = -1
6498 else:
6499 sysvals.rs = 1
6500 else:
6501 doError('invalid value --> (%s: %s), use "enable/disable"' % (option, value), True)
6502 elif(option == 'display'):
6503 disopt = ['on', 'off', 'standby', 'suspend']
6504 if value not in disopt:
6505 doError('invalid value --> (%s: %s), use %s' % (option, value, disopt), True)
6506 sysvals.display = value
6507 elif(option == 'gzip'):
6508 sysvals.gzip = checkArgBool(option, value)
6509 elif(option == 'cgfilter'):
6510 sysvals.setCallgraphFilter(value)
6511 elif(option == 'cgskip'):
6512 if value in switchoff:
6513 sysvals.cgskip = ''
6514 else:
6515 sysvals.cgskip = sysvals.configFile(val)
6516 if(not sysvals.cgskip):
6517 doError('%s does not exist' % sysvals.cgskip)
6518 elif(option == 'cgtest'):
6519 sysvals.cgtest = getArgInt('cgtest', value, 0, 1, False)
6520 elif(option == 'cgphase'):
6521 d = Data(0)
6522 if value not in d.phasedef:
6523 doError('invalid phase --> (%s: %s), valid phases are %s'\
6524 % (option, value, d.phasedef.keys()), True)
6525 sysvals.cgphase = value
6526 elif(option == 'fadd'):
6527 file = sysvals.configFile(value)
6528 if(not file):
6529 doError('%s does not exist' % value)
6530 sysvals.addFtraceFilterFunctions(file)
6531 elif(option == 'result'):
6532 sysvals.result = value
6533 elif(option == 'multi'):
6534 nums = value.split()
6535 if len(nums) != 2:
6536 doError('multi requires 2 integers (exec_count and delay)', True)
6537 sysvals.multiinit(nums[0], nums[1])
6538 elif(option == 'devicefilter'):
6539 sysvals.setDeviceFilter(value)
6540 elif(option == 'expandcg'):
6541 sysvals.cgexp = checkArgBool(option, value)
6542 elif(option == 'srgap'):
6543 if checkArgBool(option, value):
6544 sysvals.srgap = 5
6545 elif(option == 'mode'):
6546 sysvals.suspendmode = value
6547 elif(option == 'command' or option == 'cmd'):
6548 sysvals.testcommand = value
6549 elif(option == 'x2delay'):
6550 sysvals.x2delay = getArgInt('x2delay', value, 0, 60000, False)
6551 elif(option == 'predelay'):
6552 sysvals.predelay = getArgInt('predelay', value, 0, 60000, False)
6553 elif(option == 'postdelay'):
6554 sysvals.postdelay = getArgInt('postdelay', value, 0, 60000, False)
6555 elif(option == 'maxdepth'):
6556 sysvals.max_graph_depth = getArgInt('maxdepth', value, 0, 1000, False)
6557 elif(option == 'rtcwake'):
6558 if value in switchoff:
6559 sysvals.rtcwake = False
6560 else:
6561 sysvals.rtcwake = True
6562 sysvals.rtcwaketime = getArgInt('rtcwake', value, 0, 3600, False)
6563 elif(option == 'timeprec'):
6564 sysvals.setPrecision(getArgInt('timeprec', value, 0, 6, False))
6565 elif(option == 'mindev'):
6566 sysvals.mindevlen = getArgFloat('mindev', value, 0.0, 10000.0, False)
6567 elif(option == 'callloop-maxgap'):
6568 sysvals.callloopmaxgap = getArgFloat('callloop-maxgap', value, 0.0, 1.0, False)
6569 elif(option == 'callloop-maxlen'):
6570 sysvals.callloopmaxgap = getArgFloat('callloop-maxlen', value, 0.0, 1.0, False)
6571 elif(option == 'mincg'):
6572 sysvals.mincglen = getArgFloat('mincg', value, 0.0, 10000.0, False)
6573 elif(option == 'bufsize'):
6574 sysvals.bufsize = getArgInt('bufsize', value, 1, 1024*1024*8, False)
6575 elif(option == 'output-dir'):
6576 sysvals.outdir = sysvals.setOutputFolder(value)
6577
6578 if sysvals.suspendmode == 'command' and not sysvals.testcommand:
6579 doError('No command supplied for mode "command"')
6580
6581 # compatibility errors
6582 if sysvals.usedevsrc and sysvals.usecallgraph:
6583 doError('-dev is not compatible with -f')
6584 if sysvals.usecallgraph and sysvals.useprocmon:
6585 doError('-proc is not compatible with -f')
6586
6587 if overridekprobes:
6588 sysvals.tracefuncs = dict()
6589 if overridedevkprobes:
6590 sysvals.dev_tracefuncs = dict()
6591
6592 kprobes = dict()
6593 kprobesec = 'dev_timeline_functions_'+platform.machine()
6594 if kprobesec in sections:
6595 for name in Config.options(kprobesec):
6596 text = Config.get(kprobesec, name)
6597 kprobes[name] = (text, True)
6598 kprobesec = 'timeline_functions_'+platform.machine()
6599 if kprobesec in sections:
6600 for name in Config.options(kprobesec):
6601 if name in kprobes:
6602 doError('Duplicate timeline function found "%s"' % (name))
6603 text = Config.get(kprobesec, name)
6604 kprobes[name] = (text, False)
6605
6606 for name in kprobes:
6607 function = name
6608 format = name
6609 color = ''
6610 args = dict()
6611 text, dev = kprobes[name]
6612 data = text.split()
6613 i = 0
6614 for val in data:
6615 # bracketted strings are special formatting, read them separately
6616 if val[0] == '[' and val[-1] == ']':
6617 for prop in val[1:-1].split(','):
6618 p = prop.split('=')
6619 if p[0] == 'color':
6620 try:
6621 color = int(p[1], 16)
6622 color = '#'+p[1]
6623 except:
6624 color = p[1]
6625 continue
6626 # first real arg should be the format string
6627 if i == 0:
6628 format = val
6629 # all other args are actual function args
6630 else:
6631 d = val.split('=')
6632 args[d[0]] = d[1]
6633 i += 1
6634 if not function or not format:
6635 doError('Invalid kprobe: %s' % name)
6636 for arg in re.findall('{(?P<n>[a-z,A-Z,0-9]*)}', format):
6637 if arg not in args:
6638 doError('Kprobe "%s" is missing argument "%s"' % (name, arg))
6639 if (dev and name in sysvals.dev_tracefuncs) or (not dev and name in sysvals.tracefuncs):
6640 doError('Duplicate timeline function found "%s"' % (name))
6641
6642 kp = {
6643 'name': name,
6644 'func': function,
6645 'format': format,
6646 sysvals.archargs: args
6647 }
6648 if color:
6649 kp['color'] = color
6650 if dev:
6651 sysvals.dev_tracefuncs[name] = kp
6652 else:
6653 sysvals.tracefuncs[name] = kp
6654
6655# Function: printHelp
6656# Description:
6657# print out the help text
6658def printHelp():
6659 pprint('\n%s v%s\n'\
6660 'Usage: sudo sleepgraph <options> <commands>\n'\
6661 '\n'\
6662 'Description:\n'\
6663 ' This tool is designed to assist kernel and OS developers in optimizing\n'\
6664 ' their linux stack\'s suspend/resume time. Using a kernel image built\n'\
6665 ' with a few extra options enabled, the tool will execute a suspend and\n'\
6666 ' capture dmesg and ftrace data until resume is complete. This data is\n'\
6667 ' transformed into a device timeline and an optional callgraph to give\n'\
6668 ' a detailed view of which devices/subsystems are taking the most\n'\
6669 ' time in suspend/resume.\n'\
6670 '\n'\
6671 ' If no specific command is given, the default behavior is to initiate\n'\
6672 ' a suspend/resume and capture the dmesg/ftrace output as an html timeline.\n'\
6673 '\n'\
6674 ' Generates output files in subdirectory: suspend-yymmdd-HHMMSS\n'\
6675 ' HTML output: <hostname>_<mode>.html\n'\
6676 ' raw dmesg output: <hostname>_<mode>_dmesg.txt\n'\
6677 ' raw ftrace output: <hostname>_<mode>_ftrace.txt\n'\
6678 '\n'\
6679 'Options:\n'\
6680 ' -h Print this help text\n'\
6681 ' -v Print the current tool version\n'\
6682 ' -config fn Pull arguments and config options from file fn\n'\
6683 ' -verbose Print extra information during execution and analysis\n'\
6684 ' -m mode Mode to initiate for suspend (default: %s)\n'\
6685 ' -o name Overrides the output subdirectory name when running a new test\n'\
6686 ' default: suspend-{date}-{time}\n'\
6687 ' -rtcwake t Wakeup t seconds after suspend, set t to "off" to disable (default: 15)\n'\
6688 ' -addlogs Add the dmesg and ftrace logs to the html output\n'\
6689 ' -noturbostat Dont use turbostat in freeze mode (default: disabled)\n'\
6690 ' -srgap Add a visible gap in the timeline between sus/res (default: disabled)\n'\
6691 ' -skiphtml Run the test and capture the trace logs, but skip the timeline (default: disabled)\n'\
6692 ' -result fn Export a results table to a text file for parsing.\n'\
6693 ' -wifi If a wifi connection is available, check that it reconnects after resume.\n'\
6694 ' -wifitrace Trace kernel execution through wifi reconnect.\n'\
6695 ' -netfix Use netfix to reset the network in the event it fails to resume.\n'\
6696 ' [testprep]\n'\
6697 ' -sync Sync the filesystems before starting the test\n'\
6698 ' -rs on/off Enable/disable runtime suspend for all devices, restore all after test\n'\
6699 ' -display m Change the display mode to m for the test (on/off/standby/suspend)\n'\
6700 ' [advanced]\n'\
6701 ' -gzip Gzip the trace and dmesg logs to save space\n'\
6702 ' -cmd {s} Run the timeline over a custom command, e.g. "sync -d"\n'\
6703 ' -proc Add usermode process info into the timeline (default: disabled)\n'\
6704 ' -dev Add kernel function calls and threads to the timeline (default: disabled)\n'\
6705 ' -x2 Run two suspend/resumes back to back (default: disabled)\n'\
6706 ' -x2delay t Include t ms delay between multiple test runs (default: 0 ms)\n'\
6707 ' -predelay t Include t ms delay before 1st suspend (default: 0 ms)\n'\
6708 ' -postdelay t Include t ms delay after last resume (default: 0 ms)\n'\
6709 ' -mindev ms Discard all device blocks shorter than ms milliseconds (e.g. 0.001 for us)\n'\
6710 ' -multi n d Execute <n> consecutive tests at <d> seconds intervals. If <n> is followed\n'\
6711 ' by a "d", "h", or "m" execute for <n> days, hours, or mins instead.\n'\
6712 ' The outputs will be created in a new subdirectory with a summary page.\n'\
6713 ' -maxfail n Abort a -multi run after n consecutive fails (default is 0 = never abort)\n'\
6714 ' [debug]\n'\
6715 ' -f Use ftrace to create device callgraphs (default: disabled)\n'\
6716 ' -ftop Use ftrace on the top level call: "%s" (default: disabled)\n'\
6717 ' -maxdepth N limit the callgraph data to N call levels (default: 0=all)\n'\
6718 ' -expandcg pre-expand the callgraph data in the html output (default: disabled)\n'\
6719 ' -fadd file Add functions to be graphed in the timeline from a list in a text file\n'\
6720 ' -filter "d1,d2,..." Filter out all but this comma-delimited list of device names\n'\
6721 ' -mincg ms Discard all callgraphs shorter than ms milliseconds (e.g. 0.001 for us)\n'\
6722 ' -cgphase P Only show callgraph data for phase P (e.g. suspend_late)\n'\
6723 ' -cgtest N Only show callgraph data for test N (e.g. 0 or 1 in an x2 run)\n'\
6724 ' -timeprec N Number of significant digits in timestamps (0:S, [3:ms], 6:us)\n'\
6725 ' -cgfilter S Filter the callgraph output in the timeline\n'\
6726 ' -cgskip file Callgraph functions to skip, off to disable (default: cgskip.txt)\n'\
6727 ' -bufsize N Set trace buffer size to N kilo-bytes (default: all of free memory)\n'\
6728 ' -devdump Print out all the raw device data for each phase\n'\
6729 ' -cgdump Print out all the raw callgraph data\n'\
6730 '\n'\
6731 'Other commands:\n'\
6732 ' -modes List available suspend modes\n'\
6733 ' -status Test to see if the system is enabled to run this tool\n'\
6734 ' -fpdt Print out the contents of the ACPI Firmware Performance Data Table\n'\
6735 ' -wificheck Print out wifi connection info\n'\
6736 ' -x<mode> Test xset by toggling the given mode (on/off/standby/suspend)\n'\
6737 ' -sysinfo Print out system info extracted from BIOS\n'\
6738 ' -devinfo Print out the pm settings of all devices which support runtime suspend\n'\
6739 ' -cmdinfo Print out all the platform info collected before and after suspend/resume\n'\
6740 ' -flist Print the list of functions currently being captured in ftrace\n'\
6741 ' -flistall Print all functions capable of being captured in ftrace\n'\
6742 ' -summary dir Create a summary of tests in this dir [-genhtml builds missing html]\n'\
6743 ' [redo]\n'\
6744 ' -ftrace ftracefile Create HTML output using ftrace input (used with -dmesg)\n'\
6745 ' -dmesg dmesgfile Create HTML output using dmesg (used with -ftrace)\n'\
6746 '' % (sysvals.title, sysvals.version, sysvals.suspendmode, sysvals.ftopfunc))
6747 return True
6748
6749# ----------------- MAIN --------------------
6750# exec start (skipped if script is loaded as library)
6751if __name__ == '__main__':
6752 genhtml = False
6753 cmd = ''
6754 simplecmds = ['-sysinfo', '-modes', '-fpdt', '-flist', '-flistall',
6755 '-devinfo', '-status', '-xon', '-xoff', '-xstandby', '-xsuspend',
6756 '-xinit', '-xreset', '-xstat', '-wificheck', '-cmdinfo']
6757 if '-f' in sys.argv:
6758 sysvals.cgskip = sysvals.configFile('cgskip.txt')
6759 # loop through the command line arguments
6760 args = iter(sys.argv[1:])
6761 for arg in args:
6762 if(arg == '-m'):
6763 try:
6764 val = next(args)
6765 except:
6766 doError('No mode supplied', True)
6767 if val == 'command' and not sysvals.testcommand:
6768 doError('No command supplied for mode "command"', True)
6769 sysvals.suspendmode = val
6770 elif(arg in simplecmds):
6771 cmd = arg[1:]
6772 elif(arg == '-h'):
6773 printHelp()
6774 sys.exit(0)
6775 elif(arg == '-v'):
6776 pprint("Version %s" % sysvals.version)
6777 sys.exit(0)
6778 elif(arg == '-debugtiming'):
6779 debugtiming = True
6780 elif(arg == '-x2'):
6781 sysvals.execcount = 2
6782 elif(arg == '-x2delay'):
6783 sysvals.x2delay = getArgInt('-x2delay', args, 0, 60000)
6784 elif(arg == '-predelay'):
6785 sysvals.predelay = getArgInt('-predelay', args, 0, 60000)
6786 elif(arg == '-postdelay'):
6787 sysvals.postdelay = getArgInt('-postdelay', args, 0, 60000)
6788 elif(arg == '-f'):
6789 sysvals.usecallgraph = True
6790 elif(arg == '-ftop'):
6791 sysvals.usecallgraph = True
6792 sysvals.ftop = True
6793 sysvals.usekprobes = False
6794 elif(arg == '-skiphtml'):
6795 sysvals.skiphtml = True
6796 elif(arg == '-cgdump'):
6797 sysvals.cgdump = True
6798 elif(arg == '-devdump'):
6799 sysvals.devdump = True
6800 elif(arg == '-genhtml'):
6801 genhtml = True
6802 elif(arg == '-addlogs'):
6803 sysvals.dmesglog = sysvals.ftracelog = True
6804 elif(arg == '-nologs'):
6805 sysvals.dmesglog = sysvals.ftracelog = False
6806 elif(arg == '-addlogdmesg'):
6807 sysvals.dmesglog = True
6808 elif(arg == '-addlogftrace'):
6809 sysvals.ftracelog = True
6810 elif(arg == '-noturbostat'):
6811 sysvals.tstat = False
6812 elif(arg == '-verbose'):
6813 sysvals.verbose = True
6814 elif(arg == '-proc'):
6815 sysvals.useprocmon = True
6816 elif(arg == '-dev'):
6817 sysvals.usedevsrc = True
6818 elif(arg == '-sync'):
6819 sysvals.sync = True
6820 elif(arg == '-wifi'):
6821 sysvals.wifi = True
6822 elif(arg == '-wifitrace'):
6823 sysvals.wifitrace = True
6824 elif(arg == '-netfix'):
6825 sysvals.netfix = True
6826 elif(arg == '-gzip'):
6827 sysvals.gzip = True
6828 elif(arg == '-info'):
6829 try:
6830 val = next(args)
6831 except:
6832 doError('-info requires one string argument', True)
6833 elif(arg == '-desc'):
6834 try:
6835 val = next(args)
6836 except:
6837 doError('-desc requires one string argument', True)
6838 elif(arg == '-rs'):
6839 try:
6840 val = next(args)
6841 except:
6842 doError('-rs requires "enable" or "disable"', True)
6843 if val.lower() in switchvalues:
6844 if val.lower() in switchoff:
6845 sysvals.rs = -1
6846 else:
6847 sysvals.rs = 1
6848 else:
6849 doError('invalid option: %s, use "enable/disable" or "on/off"' % val, True)
6850 elif(arg == '-display'):
6851 try:
6852 val = next(args)
6853 except:
6854 doError('-display requires an mode value', True)
6855 disopt = ['on', 'off', 'standby', 'suspend']
6856 if val.lower() not in disopt:
6857 doError('valid display mode values are %s' % disopt, True)
6858 sysvals.display = val.lower()
6859 elif(arg == '-maxdepth'):
6860 sysvals.max_graph_depth = getArgInt('-maxdepth', args, 0, 1000)
6861 elif(arg == '-rtcwake'):
6862 try:
6863 val = next(args)
6864 except:
6865 doError('No rtcwake time supplied', True)
6866 if val.lower() in switchoff:
6867 sysvals.rtcwake = False
6868 else:
6869 sysvals.rtcwake = True
6870 sysvals.rtcwaketime = getArgInt('-rtcwake', val, 0, 3600, False)
6871 elif(arg == '-timeprec'):
6872 sysvals.setPrecision(getArgInt('-timeprec', args, 0, 6))
6873 elif(arg == '-mindev'):
6874 sysvals.mindevlen = getArgFloat('-mindev', args, 0.0, 10000.0)
6875 elif(arg == '-mincg'):
6876 sysvals.mincglen = getArgFloat('-mincg', args, 0.0, 10000.0)
6877 elif(arg == '-bufsize'):
6878 sysvals.bufsize = getArgInt('-bufsize', args, 1, 1024*1024*8)
6879 elif(arg == '-cgtest'):
6880 sysvals.cgtest = getArgInt('-cgtest', args, 0, 1)
6881 elif(arg == '-cgphase'):
6882 try:
6883 val = next(args)
6884 except:
6885 doError('No phase name supplied', True)
6886 d = Data(0)
6887 if val not in d.phasedef:
6888 doError('invalid phase --> (%s: %s), valid phases are %s'\
6889 % (arg, val, d.phasedef.keys()), True)
6890 sysvals.cgphase = val
6891 elif(arg == '-cgfilter'):
6892 try:
6893 val = next(args)
6894 except:
6895 doError('No callgraph functions supplied', True)
6896 sysvals.setCallgraphFilter(val)
6897 elif(arg == '-skipkprobe'):
6898 try:
6899 val = next(args)
6900 except:
6901 doError('No kprobe functions supplied', True)
6902 sysvals.skipKprobes(val)
6903 elif(arg == '-cgskip'):
6904 try:
6905 val = next(args)
6906 except:
6907 doError('No file supplied', True)
6908 if val.lower() in switchoff:
6909 sysvals.cgskip = ''
6910 else:
6911 sysvals.cgskip = sysvals.configFile(val)
6912 if(not sysvals.cgskip):
6913 doError('%s does not exist' % sysvals.cgskip)
6914 elif(arg == '-callloop-maxgap'):
6915 sysvals.callloopmaxgap = getArgFloat('-callloop-maxgap', args, 0.0, 1.0)
6916 elif(arg == '-callloop-maxlen'):
6917 sysvals.callloopmaxlen = getArgFloat('-callloop-maxlen', args, 0.0, 1.0)
6918 elif(arg == '-cmd'):
6919 try:
6920 val = next(args)
6921 except:
6922 doError('No command string supplied', True)
6923 sysvals.testcommand = val
6924 sysvals.suspendmode = 'command'
6925 elif(arg == '-expandcg'):
6926 sysvals.cgexp = True
6927 elif(arg == '-srgap'):
6928 sysvals.srgap = 5
6929 elif(arg == '-maxfail'):
6930 sysvals.maxfail = getArgInt('-maxfail', args, 0, 1000000)
6931 elif(arg == '-multi'):
6932 try:
6933 c, d = next(args), next(args)
6934 except:
6935 doError('-multi requires two values', True)
6936 sysvals.multiinit(c, d)
6937 elif(arg == '-o'):
6938 try:
6939 val = next(args)
6940 except:
6941 doError('No subdirectory name supplied', True)
6942 sysvals.outdir = sysvals.setOutputFolder(val)
6943 elif(arg == '-config'):
6944 try:
6945 val = next(args)
6946 except:
6947 doError('No text file supplied', True)
6948 file = sysvals.configFile(val)
6949 if(not file):
6950 doError('%s does not exist' % val)
6951 configFromFile(file)
6952 elif(arg == '-fadd'):
6953 try:
6954 val = next(args)
6955 except:
6956 doError('No text file supplied', True)
6957 file = sysvals.configFile(val)
6958 if(not file):
6959 doError('%s does not exist' % val)
6960 sysvals.addFtraceFilterFunctions(file)
6961 elif(arg == '-dmesg'):
6962 try:
6963 val = next(args)
6964 except:
6965 doError('No dmesg file supplied', True)
6966 sysvals.notestrun = True
6967 sysvals.dmesgfile = val
6968 if(os.path.exists(sysvals.dmesgfile) == False):
6969 doError('%s does not exist' % sysvals.dmesgfile)
6970 elif(arg == '-ftrace'):
6971 try:
6972 val = next(args)
6973 except:
6974 doError('No ftrace file supplied', True)
6975 sysvals.notestrun = True
6976 sysvals.ftracefile = val
6977 if(os.path.exists(sysvals.ftracefile) == False):
6978 doError('%s does not exist' % sysvals.ftracefile)
6979 elif(arg == '-summary'):
6980 try:
6981 val = next(args)
6982 except:
6983 doError('No directory supplied', True)
6984 cmd = 'summary'
6985 sysvals.outdir = val
6986 sysvals.notestrun = True
6987 if(os.path.isdir(val) == False):
6988 doError('%s is not accesible' % val)
6989 elif(arg == '-filter'):
6990 try:
6991 val = next(args)
6992 except:
6993 doError('No devnames supplied', True)
6994 sysvals.setDeviceFilter(val)
6995 elif(arg == '-result'):
6996 try:
6997 val = next(args)
6998 except:
6999 doError('No result file supplied', True)
7000 sysvals.result = val
7001 sysvals.signalHandlerInit()
7002 else:
7003 doError('Invalid argument: '+arg, True)
7004
7005 # compatibility errors
7006 if(sysvals.usecallgraph and sysvals.usedevsrc):
7007 doError('-dev is not compatible with -f')
7008 if(sysvals.usecallgraph and sysvals.useprocmon):
7009 doError('-proc is not compatible with -f')
7010
7011 if sysvals.usecallgraph and sysvals.cgskip:
7012 sysvals.vprint('Using cgskip file: %s' % sysvals.cgskip)
7013 sysvals.setCallgraphBlacklist(sysvals.cgskip)
7014
7015 # callgraph size cannot exceed device size
7016 if sysvals.mincglen < sysvals.mindevlen:
7017 sysvals.mincglen = sysvals.mindevlen
7018
7019 # remove existing buffers before calculating memory
7020 if(sysvals.usecallgraph or sysvals.usedevsrc):
7021 sysvals.fsetVal('16', 'buffer_size_kb')
7022 sysvals.cpuInfo()
7023
7024 # just run a utility command and exit
7025 if(cmd != ''):
7026 ret = 0
7027 if(cmd == 'status'):
7028 if not statusCheck(True):
7029 ret = 1
7030 elif(cmd == 'fpdt'):
7031 if not getFPDT(True):
7032 ret = 1
7033 elif(cmd == 'sysinfo'):
7034 sysvals.printSystemInfo(True)
7035 elif(cmd == 'devinfo'):
7036 deviceInfo()
7037 elif(cmd == 'modes'):
7038 pprint(getModes())
7039 elif(cmd == 'flist'):
7040 sysvals.getFtraceFilterFunctions(True)
7041 elif(cmd == 'flistall'):
7042 sysvals.getFtraceFilterFunctions(False)
7043 elif(cmd == 'summary'):
7044 runSummary(sysvals.outdir, True, genhtml)
7045 elif(cmd in ['xon', 'xoff', 'xstandby', 'xsuspend', 'xinit', 'xreset']):
7046 sysvals.verbose = True
7047 ret = sysvals.displayControl(cmd[1:])
7048 elif(cmd == 'xstat'):
7049 pprint('Display Status: %s' % sysvals.displayControl('stat').upper())
7050 elif(cmd == 'wificheck'):
7051 dev = sysvals.checkWifi()
7052 if dev:
7053 print('%s is connected' % sysvals.wifiDetails(dev))
7054 else:
7055 print('No wifi connection found')
7056 elif(cmd == 'cmdinfo'):
7057 for out in sysvals.cmdinfo(False, True):
7058 print('[%s - %s]\n%s\n' % out)
7059 sys.exit(ret)
7060
7061 # if instructed, re-analyze existing data files
7062 if(sysvals.notestrun):
7063 stamp = rerunTest(sysvals.outdir)
7064 sysvals.outputResult(stamp)
7065 sys.exit(0)
7066
7067 # verify that we can run a test
7068 error = statusCheck()
7069 if(error):
7070 doError(error)
7071
7072 # extract mem/disk extra modes and convert
7073 mode = sysvals.suspendmode
7074 if mode.startswith('mem'):
7075 memmode = mode.split('-', 1)[-1] if '-' in mode else 'deep'
7076 if memmode == 'shallow':
7077 mode = 'standby'
7078 elif memmode == 's2idle':
7079 mode = 'freeze'
7080 else:
7081 mode = 'mem'
7082 sysvals.memmode = memmode
7083 sysvals.suspendmode = mode
7084 if mode.startswith('disk-'):
7085 sysvals.diskmode = mode.split('-', 1)[-1]
7086 sysvals.suspendmode = 'disk'
7087 sysvals.systemInfo(dmidecode(sysvals.mempath))
7088
7089 failcnt, ret = 0, 0
7090 if sysvals.multitest['run']:
7091 # run multiple tests in a separate subdirectory
7092 if not sysvals.outdir:
7093 if 'time' in sysvals.multitest:
7094 s = '-%dm' % sysvals.multitest['time']
7095 else:
7096 s = '-x%d' % sysvals.multitest['count']
7097 sysvals.outdir = datetime.now().strftime('suspend-%y%m%d-%H%M%S'+s)
7098 if not os.path.isdir(sysvals.outdir):
7099 os.makedirs(sysvals.outdir)
7100 sysvals.sudoUserchown(sysvals.outdir)
7101 finish = datetime.now()
7102 if 'time' in sysvals.multitest:
7103 finish += timedelta(minutes=sysvals.multitest['time'])
7104 for i in range(sysvals.multitest['count']):
7105 sysvals.multistat(True, i, finish)
7106 if i != 0 and sysvals.multitest['delay'] > 0:
7107 pprint('Waiting %d seconds...' % (sysvals.multitest['delay']))
7108 time.sleep(sysvals.multitest['delay'])
7109 fmt = 'suspend-%y%m%d-%H%M%S'
7110 sysvals.testdir = os.path.join(sysvals.outdir, datetime.now().strftime(fmt))
7111 ret = runTest(i+1, not sysvals.verbose)
7112 failcnt = 0 if not ret else failcnt + 1
7113 if sysvals.maxfail > 0 and failcnt >= sysvals.maxfail:
7114 pprint('Maximum fail count of %d reached, aborting multitest' % (sysvals.maxfail))
7115 break
7116 sysvals.resetlog()
7117 sysvals.multistat(False, i, finish)
7118 if 'time' in sysvals.multitest and datetime.now() >= finish:
7119 break
7120 if not sysvals.skiphtml:
7121 runSummary(sysvals.outdir, False, False)
7122 sysvals.sudoUserchown(sysvals.outdir)
7123 else:
7124 if sysvals.outdir:
7125 sysvals.testdir = sysvals.outdir
7126 # run the test in the current directory
7127 ret = runTest()
7128
7129 # reset to default values after testing
7130 if sysvals.display:
7131 sysvals.displayControl('reset')
7132 if sysvals.rs != 0:
7133 sysvals.setRuntimeSuspend(False)
7134 sys.exit(ret)