diff options
Diffstat (limited to 'tools/perf/scripts/python')
20 files changed, 931 insertions, 491 deletions
| diff --git a/tools/perf/scripts/python/Perf-Trace-Util/Build b/tools/perf/scripts/python/Perf-Trace-Util/Build index aefc15c9444a..7d0e33ce6aba 100644 --- a/tools/perf/scripts/python/Perf-Trace-Util/Build +++ b/tools/perf/scripts/python/Perf-Trace-Util/Build @@ -1,3 +1,3 @@ -libperf-y += Context.o +perf-y += Context.o  CFLAGS_Context.o += $(PYTHON_EMBED_CCOPTS) -Wno-redundant-decls -Wno-strict-prototypes -Wno-unused-parameter -Wno-nested-externs diff --git a/tools/perf/scripts/python/check-perf-trace.py b/tools/perf/scripts/python/check-perf-trace.py index 334599c6032c..d2c22954800d 100644 --- a/tools/perf/scripts/python/check-perf-trace.py +++ b/tools/perf/scripts/python/check-perf-trace.py @@ -7,6 +7,8 @@  # events, etc.  Basically, if this script runs successfully and  # displays expected results, Python scripting support should be ok. +from __future__ import print_function +  import os  import sys @@ -19,64 +21,64 @@ from perf_trace_context import *  unhandled = autodict()  def trace_begin(): -	print "trace_begin" +	print("trace_begin")  	pass  def trace_end(): -        print_unhandled() +	print_unhandled()  def irq__softirq_entry(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	common_callchain, vec): -		print_header(event_name, common_cpu, common_secs, common_nsecs, -			common_pid, common_comm) +		       common_secs, common_nsecs, common_pid, common_comm, +		       common_callchain, vec): +	print_header(event_name, common_cpu, common_secs, common_nsecs, +		common_pid, common_comm) -                print_uncommon(context) +	print_uncommon(context) -		print "vec=%s\n" % \ -		(symbol_str("irq__softirq_entry", "vec", vec)), +	print("vec=%s" % (symbol_str("irq__softirq_entry", "vec", vec)))  def kmem__kmalloc(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	common_callchain, call_site, ptr, bytes_req, bytes_alloc, -	gfp_flags): -		print_header(event_name, common_cpu, common_secs, common_nsecs, -			common_pid, common_comm) +		  common_secs, common_nsecs, common_pid, common_comm, +		  common_callchain, call_site, ptr, bytes_req, bytes_alloc, +		  gfp_flags): +	print_header(event_name, common_cpu, common_secs, common_nsecs, +		common_pid, common_comm) -                print_uncommon(context) +	print_uncommon(context) -		print "call_site=%u, ptr=%u, bytes_req=%u, " \ -		"bytes_alloc=%u, gfp_flags=%s\n" % \ +	print("call_site=%u, ptr=%u, bytes_req=%u, " +		"bytes_alloc=%u, gfp_flags=%s" %  		(call_site, ptr, bytes_req, bytes_alloc, - -		flag_str("kmem__kmalloc", "gfp_flags", gfp_flags)), +		flag_str("kmem__kmalloc", "gfp_flags", gfp_flags)))  def trace_unhandled(event_name, context, event_fields_dict): -    try: -        unhandled[event_name] += 1 -    except TypeError: -        unhandled[event_name] = 1 +	try: +		unhandled[event_name] += 1 +	except TypeError: +		unhandled[event_name] = 1  def print_header(event_name, cpu, secs, nsecs, pid, comm): -	print "%-20s %5u %05u.%09u %8u %-20s " % \ -	(event_name, cpu, secs, nsecs, pid, comm), +	print("%-20s %5u %05u.%09u %8u %-20s " % +		(event_name, cpu, secs, nsecs, pid, comm), +		end=' ')  # print trace fields not included in handler args  def print_uncommon(context): -    print "common_preempt_count=%d, common_flags=%s, common_lock_depth=%d, " \ -        % (common_pc(context), trace_flag_str(common_flags(context)), \ -               common_lock_depth(context)) +	print("common_preempt_count=%d, common_flags=%s, " +		"common_lock_depth=%d, " % +		(common_pc(context), trace_flag_str(common_flags(context)), +		common_lock_depth(context)))  def print_unhandled(): -    keys = unhandled.keys() -    if not keys: -        return +	keys = unhandled.keys() +	if not keys: +		return -    print "\nunhandled events:\n\n", +	print("\nunhandled events:\n") -    print "%-40s  %10s\n" % ("event", "count"), -    print "%-40s  %10s\n" % ("----------------------------------------", \ -                                 "-----------"), +	print("%-40s  %10s" % ("event", "count")) +	print("%-40s  %10s" % ("----------------------------------------", +				"-----------")) -    for event_name in keys: -	print "%-40s  %10d\n" % (event_name, unhandled[event_name]) +	for event_name in keys: +		print("%-40s  %10d\n" % (event_name, unhandled[event_name])) diff --git a/tools/perf/scripts/python/compaction-times.py b/tools/perf/scripts/python/compaction-times.py index 239cb0568ec3..2560a042dc6f 100644 --- a/tools/perf/scripts/python/compaction-times.py +++ b/tools/perf/scripts/python/compaction-times.py @@ -216,15 +216,15 @@ def compaction__mm_compaction_migratepages(event_name, context, common_cpu,  		pair(nr_migrated, nr_failed), None, None)  def compaction__mm_compaction_isolate_freepages(event_name, context, common_cpu, -        common_secs, common_nsecs, common_pid, common_comm, -        common_callchain, start_pfn, end_pfn, nr_scanned, nr_taken): +	common_secs, common_nsecs, common_pid, common_comm, +	common_callchain, start_pfn, end_pfn, nr_scanned, nr_taken):  	chead.increment_pending(common_pid,  		None, pair(nr_scanned, nr_taken), None)  def compaction__mm_compaction_isolate_migratepages(event_name, context, common_cpu, -        common_secs, common_nsecs, common_pid, common_comm, -        common_callchain, start_pfn, end_pfn, nr_scanned, nr_taken): +	common_secs, common_nsecs, common_pid, common_comm, +	common_callchain, start_pfn, end_pfn, nr_scanned, nr_taken):  	chead.increment_pending(common_pid,  		None, None, pair(nr_scanned, nr_taken)) diff --git a/tools/perf/scripts/python/event_analyzing_sample.py b/tools/perf/scripts/python/event_analyzing_sample.py index 4e843b9864ec..aa1e2cfa26a6 100644 --- a/tools/perf/scripts/python/event_analyzing_sample.py +++ b/tools/perf/scripts/python/event_analyzing_sample.py @@ -15,6 +15,8 @@  # for a x86 HW PMU event: PEBS with load latency data.  # +from __future__ import print_function +  import os  import sys  import math @@ -37,7 +39,7 @@ con = sqlite3.connect("/dev/shm/perf.db")  con.isolation_level = None  def trace_begin(): -	print "In trace_begin:\n" +        print("In trace_begin:\n")          #          # Will create several tables at the start, pebs_ll is for PEBS data with @@ -76,12 +78,12 @@ def process_event(param_dict):          name       = param_dict["ev_name"]          # Symbol and dso info are not always resolved -        if (param_dict.has_key("dso")): +        if ("dso" in param_dict):                  dso = param_dict["dso"]          else:                  dso = "Unknown_dso" -        if (param_dict.has_key("symbol")): +        if ("symbol" in param_dict):                  symbol = param_dict["symbol"]          else:                  symbol = "Unknown_symbol" @@ -102,7 +104,7 @@ def insert_db(event):                                  event.ip, event.status, event.dse, event.dla, event.lat))  def trace_end(): -	print "In trace_end:\n" +        print("In trace_end:\n")          # We show the basic info for the 2 type of event classes          show_general_events()          show_pebs_ll() @@ -123,29 +125,29 @@ def show_general_events():          # Check the total record number in the table          count = con.execute("select count(*) from gen_events")          for t in count: -                print "There is %d records in gen_events table" % t[0] +                print("There is %d records in gen_events table" % t[0])                  if t[0] == 0:                          return -        print "Statistics about the general events grouped by thread/symbol/dso: \n" +        print("Statistics about the general events grouped by thread/symbol/dso: \n")           # Group by thread          commq = con.execute("select comm, count(comm) from gen_events group by comm order by -count(comm)") -        print "\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42) +        print("\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42))          for row in commq: -             print "%16s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%16s %8d     %s" % (row[0], row[1], num2sym(row[1])))          # Group by symbol -        print "\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58) +        print("\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58))          symbolq = con.execute("select symbol, count(symbol) from gen_events group by symbol order by -count(symbol)")          for row in symbolq: -             print "%32s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%32s %8d     %s" % (row[0], row[1], num2sym(row[1])))          # Group by dso -        print "\n%40s %8s %16s\n%s" % ("dso", "number", "histogram", "="*74) +        print("\n%40s %8s %16s\n%s" % ("dso", "number", "histogram", "="*74))          dsoq = con.execute("select dso, count(dso) from gen_events group by dso order by -count(dso)")          for row in dsoq: -             print "%40s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%40s %8d     %s" % (row[0], row[1], num2sym(row[1])))  #  # This function just shows the basic info, and we could do more with the @@ -156,35 +158,35 @@ def show_pebs_ll():          count = con.execute("select count(*) from pebs_ll")          for t in count: -                print "There is %d records in pebs_ll table" % t[0] +                print("There is %d records in pebs_ll table" % t[0])                  if t[0] == 0:                          return -        print "Statistics about the PEBS Load Latency events grouped by thread/symbol/dse/latency: \n" +        print("Statistics about the PEBS Load Latency events grouped by thread/symbol/dse/latency: \n")          # Group by thread          commq = con.execute("select comm, count(comm) from pebs_ll group by comm order by -count(comm)") -        print "\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42) +        print("\n%16s %8s %16s\n%s" % ("comm", "number", "histogram", "="*42))          for row in commq: -             print "%16s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%16s %8d     %s" % (row[0], row[1], num2sym(row[1])))          # Group by symbol -        print "\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58) +        print("\n%32s %8s %16s\n%s" % ("symbol", "number", "histogram", "="*58))          symbolq = con.execute("select symbol, count(symbol) from pebs_ll group by symbol order by -count(symbol)")          for row in symbolq: -             print "%32s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%32s %8d     %s" % (row[0], row[1], num2sym(row[1])))          # Group by dse          dseq = con.execute("select dse, count(dse) from pebs_ll group by dse order by -count(dse)") -        print "\n%32s %8s %16s\n%s" % ("dse", "number", "histogram", "="*58) +        print("\n%32s %8s %16s\n%s" % ("dse", "number", "histogram", "="*58))          for row in dseq: -             print "%32s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%32s %8d     %s" % (row[0], row[1], num2sym(row[1])))          # Group by latency          latq = con.execute("select lat, count(lat) from pebs_ll group by lat order by lat") -        print "\n%32s %8s %16s\n%s" % ("latency", "number", "histogram", "="*58) +        print("\n%32s %8s %16s\n%s" % ("latency", "number", "histogram", "="*58))          for row in latq: -             print "%32s %8d     %s" % (row[0], row[1], num2sym(row[1])) +             print("%32s %8d     %s" % (row[0], row[1], num2sym(row[1])))  def trace_unhandled(event_name, context, event_fields_dict): -		print ' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]) +        print (' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())])) diff --git a/tools/perf/scripts/python/export-to-postgresql.py b/tools/perf/scripts/python/export-to-postgresql.py index 0564dd7377f2..390a351d15ea 100644 --- a/tools/perf/scripts/python/export-to-postgresql.py +++ b/tools/perf/scripts/python/export-to-postgresql.py @@ -394,7 +394,8 @@ if perf_db_export_calls:  		'call_id	bigint,'  		'return_id	bigint,'  		'parent_call_path_id	bigint,' -		'flags		integer)') +		'flags		integer,' +		'parent_id	bigint)')  do_query(query, 'CREATE VIEW machines_view AS '  	'SELECT ' @@ -478,8 +479,9 @@ if perf_db_export_calls:  			'branch_count,'  			'call_id,'  			'return_id,' -			'CASE WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' ELSE \'\' END AS flags,' -			'parent_call_path_id' +			'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE CAST ( flags AS VARCHAR(6) ) END AS flags,' +			'parent_call_path_id,' +			'calls.parent_id'  		' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id')  do_query(query, 'CREATE VIEW samples_view AS ' @@ -575,6 +577,7 @@ def trace_begin():  	sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)  	if perf_db_export_calls or perf_db_export_callchains:  		call_path_table(0, 0, 0, 0) +		call_return_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)  unhandled_count = 0 @@ -657,6 +660,7 @@ def trace_end():  					'ADD CONSTRAINT returnfk    FOREIGN KEY (return_id)    REFERENCES samples    (id),'  					'ADD CONSTRAINT parent_call_pathfk FOREIGN KEY (parent_call_path_id) REFERENCES call_paths (id)')  		do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)') +		do_query(query, 'CREATE INDEX pid_idx ON calls (parent_id)')  	if (unhandled_count):  		print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events" @@ -728,7 +732,7 @@ def call_path_table(cp_id, parent_id, symbol_id, ip, *x):  	value = struct.pack(fmt, 4, 8, cp_id, 8, parent_id, 8, symbol_id, 8, ip)  	call_path_file.write(value) -def call_return_table(cr_id, thread_id, comm_id, call_path_id, call_time, return_time, branch_count, call_id, return_id, parent_call_path_id, flags, *x): -	fmt = "!hiqiqiqiqiqiqiqiqiqiqii" -	value = struct.pack(fmt, 11, 8, cr_id, 8, thread_id, 8, comm_id, 8, call_path_id, 8, call_time, 8, return_time, 8, branch_count, 8, call_id, 8, return_id, 8, parent_call_path_id, 4, flags) +def call_return_table(cr_id, thread_id, comm_id, call_path_id, call_time, return_time, branch_count, call_id, return_id, parent_call_path_id, flags, parent_id, *x): +	fmt = "!hiqiqiqiqiqiqiqiqiqiqiiiq" +	value = struct.pack(fmt, 12, 8, cr_id, 8, thread_id, 8, comm_id, 8, call_path_id, 8, call_time, 8, return_time, 8, branch_count, 8, call_id, 8, return_id, 8, parent_call_path_id, 4, flags, 8, parent_id)  	call_file.write(value) diff --git a/tools/perf/scripts/python/export-to-sqlite.py b/tools/perf/scripts/python/export-to-sqlite.py index 245caf2643ed..eb63e6c7107f 100644 --- a/tools/perf/scripts/python/export-to-sqlite.py +++ b/tools/perf/scripts/python/export-to-sqlite.py @@ -222,7 +222,8 @@ if perf_db_export_calls:  		'call_id	bigint,'  		'return_id	bigint,'  		'parent_call_path_id	bigint,' -		'flags		integer)') +		'flags		integer,' +		'parent_id	bigint)')  # printf was added to sqlite in version 3.8.3  sqlite_has_printf = False @@ -320,8 +321,9 @@ if perf_db_export_calls:  			'branch_count,'  			'call_id,'  			'return_id,' -			'CASE WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' ELSE \'\' END AS flags,' -			'parent_call_path_id' +			'CASE WHEN flags=0 THEN \'\' WHEN flags=1 THEN \'no call\' WHEN flags=2 THEN \'no return\' WHEN flags=3 THEN \'no call/return\' WHEN flags=6 THEN \'jump\' ELSE flags END AS flags,' +			'parent_call_path_id,' +			'parent_id'  		' FROM calls INNER JOIN call_paths ON call_paths.id = call_path_id')  do_query(query, 'CREATE VIEW samples_view AS ' @@ -373,7 +375,7 @@ if perf_db_export_calls or perf_db_export_callchains:  	call_path_query.prepare("INSERT INTO call_paths VALUES (?, ?, ?, ?)")  if perf_db_export_calls:  	call_query = QSqlQuery(db) -	call_query.prepare("INSERT INTO calls VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)") +	call_query.prepare("INSERT INTO calls VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)")  def trace_begin():  	print datetime.datetime.today(), "Writing records..." @@ -388,6 +390,7 @@ def trace_begin():  	sample_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)  	if perf_db_export_calls or perf_db_export_callchains:  		call_path_table(0, 0, 0, 0) +		call_return_table(0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0)  unhandled_count = 0 @@ -397,6 +400,7 @@ def trace_end():  	print datetime.datetime.today(), "Adding indexes"  	if perf_db_export_calls:  		do_query(query, 'CREATE INDEX pcpid_idx ON calls (parent_call_path_id)') +		do_query(query, 'CREATE INDEX pid_idx ON calls (parent_id)')  	if (unhandled_count):  		print datetime.datetime.today(), "Warning: ", unhandled_count, " unhandled events" @@ -452,4 +456,4 @@ def call_path_table(*x):  	bind_exec(call_path_query, 4, x)  def call_return_table(*x): -	bind_exec(call_query, 11, x) +	bind_exec(call_query, 12, x) diff --git a/tools/perf/scripts/python/exported-sql-viewer.py b/tools/perf/scripts/python/exported-sql-viewer.py index f278ce5ebab7..afec9479ca7f 100755 --- a/tools/perf/scripts/python/exported-sql-viewer.py +++ b/tools/perf/scripts/python/exported-sql-viewer.py @@ -1,4 +1,4 @@ -#!/usr/bin/python2 +#!/usr/bin/env python2  # SPDX-License-Identifier: GPL-2.0  # exported-sql-viewer.py: view data from sql database  # Copyright (c) 2014-2018, Intel Corporation. @@ -167,9 +167,10 @@ class Thread(QThread):  class TreeModel(QAbstractItemModel): -	def __init__(self, root, parent=None): +	def __init__(self, glb, parent=None):  		super(TreeModel, self).__init__(parent) -		self.root = root +		self.glb = glb +		self.root = self.GetRoot()  		self.last_row_read = 0  	def Item(self, parent): @@ -557,24 +558,12 @@ class CallGraphRootItem(CallGraphLevelItemBase):  			self.child_items.append(child_item)  			self.child_count += 1 -# Context-sensitive call graph data model +# Context-sensitive call graph data model base -class CallGraphModel(TreeModel): +class CallGraphModelBase(TreeModel):  	def __init__(self, glb, parent=None): -		super(CallGraphModel, self).__init__(CallGraphRootItem(glb), parent) -		self.glb = glb - -	def columnCount(self, parent=None): -		return 7 - -	def columnHeader(self, column): -		headers = ["Call Path", "Object", "Count ", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "] -		return headers[column] - -	def columnAlignment(self, column): -		alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ] -		return alignment[column] +		super(CallGraphModelBase, self).__init__(glb, parent)  	def FindSelect(self, value, pattern, query):  		if pattern: @@ -594,34 +583,7 @@ class CallGraphModel(TreeModel):  				match = " GLOB '" + str(value) + "'"  		else:  			match = " = '" + str(value) + "'" -		QueryExec(query, "SELECT call_path_id, comm_id, thread_id" -						" FROM calls" -						" INNER JOIN call_paths ON calls.call_path_id = call_paths.id" -						" INNER JOIN symbols ON call_paths.symbol_id = symbols.id" -						" WHERE symbols.name" + match + -						" GROUP BY comm_id, thread_id, call_path_id" -						" ORDER BY comm_id, thread_id, call_path_id") - -	def FindPath(self, query): -		# Turn the query result into a list of ids that the tree view can walk -		# to open the tree at the right place. -		ids = [] -		parent_id = query.value(0) -		while parent_id: -			ids.insert(0, parent_id) -			q2 = QSqlQuery(self.glb.db) -			QueryExec(q2, "SELECT parent_id" -					" FROM call_paths" -					" WHERE id = " + str(parent_id)) -			if not q2.next(): -				break -			parent_id = q2.value(0) -		# The call path root is not used -		if ids[0] == 1: -			del ids[0] -		ids.insert(0, query.value(2)) -		ids.insert(0, query.value(1)) -		return ids +		self.DoFindSelect(query, match)  	def Found(self, query, found):  		if found: @@ -675,6 +637,201 @@ class CallGraphModel(TreeModel):  	def FindDone(self, thread, callback, ids):  		callback(ids) +# Context-sensitive call graph data model + +class CallGraphModel(CallGraphModelBase): + +	def __init__(self, glb, parent=None): +		super(CallGraphModel, self).__init__(glb, parent) + +	def GetRoot(self): +		return CallGraphRootItem(self.glb) + +	def columnCount(self, parent=None): +		return 7 + +	def columnHeader(self, column): +		headers = ["Call Path", "Object", "Count ", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "] +		return headers[column] + +	def columnAlignment(self, column): +		alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ] +		return alignment[column] + +	def DoFindSelect(self, query, match): +		QueryExec(query, "SELECT call_path_id, comm_id, thread_id" +						" FROM calls" +						" INNER JOIN call_paths ON calls.call_path_id = call_paths.id" +						" INNER JOIN symbols ON call_paths.symbol_id = symbols.id" +						" WHERE symbols.name" + match + +						" GROUP BY comm_id, thread_id, call_path_id" +						" ORDER BY comm_id, thread_id, call_path_id") + +	def FindPath(self, query): +		# Turn the query result into a list of ids that the tree view can walk +		# to open the tree at the right place. +		ids = [] +		parent_id = query.value(0) +		while parent_id: +			ids.insert(0, parent_id) +			q2 = QSqlQuery(self.glb.db) +			QueryExec(q2, "SELECT parent_id" +					" FROM call_paths" +					" WHERE id = " + str(parent_id)) +			if not q2.next(): +				break +			parent_id = q2.value(0) +		# The call path root is not used +		if ids[0] == 1: +			del ids[0] +		ids.insert(0, query.value(2)) +		ids.insert(0, query.value(1)) +		return ids + +# Call tree data model level 2+ item base + +class CallTreeLevelTwoPlusItemBase(CallGraphLevelItemBase): + +	def __init__(self, glb, row, comm_id, thread_id, calls_id, time, branch_count, parent_item): +		super(CallTreeLevelTwoPlusItemBase, self).__init__(glb, row, parent_item) +		self.comm_id = comm_id +		self.thread_id = thread_id +		self.calls_id = calls_id +		self.branch_count = branch_count +		self.time = time + +	def Select(self): +		self.query_done = True; +		if self.calls_id == 0: +			comm_thread = " AND comm_id = " + str(self.comm_id) + " AND thread_id = " + str(self.thread_id) +		else: +			comm_thread = "" +		query = QSqlQuery(self.glb.db) +		QueryExec(query, "SELECT calls.id, name, short_name, call_time, return_time - call_time, branch_count" +					" FROM calls" +					" INNER JOIN call_paths ON calls.call_path_id = call_paths.id" +					" INNER JOIN symbols ON call_paths.symbol_id = symbols.id" +					" INNER JOIN dsos ON symbols.dso_id = dsos.id" +					" WHERE calls.parent_id = " + str(self.calls_id) + comm_thread + +					" ORDER BY call_time, calls.id") +		while query.next(): +			child_item = CallTreeLevelThreeItem(self.glb, self.child_count, self.comm_id, self.thread_id, query.value(0), query.value(1), query.value(2), query.value(3), int(query.value(4)), int(query.value(5)), self) +			self.child_items.append(child_item) +			self.child_count += 1 + +# Call tree data model level three item + +class CallTreeLevelThreeItem(CallTreeLevelTwoPlusItemBase): + +	def __init__(self, glb, row, comm_id, thread_id, calls_id, name, dso, count, time, branch_count, parent_item): +		super(CallTreeLevelThreeItem, self).__init__(glb, row, comm_id, thread_id, calls_id, time, branch_count, parent_item) +		dso = dsoname(dso) +		self.data = [ name, dso, str(count), str(time), PercentToOneDP(time, parent_item.time), str(branch_count), PercentToOneDP(branch_count, parent_item.branch_count) ] +		self.dbid = calls_id + +# Call tree data model level two item + +class CallTreeLevelTwoItem(CallTreeLevelTwoPlusItemBase): + +	def __init__(self, glb, row, comm_id, thread_id, pid, tid, parent_item): +		super(CallTreeLevelTwoItem, self).__init__(glb, row, comm_id, thread_id, 0, 0, 0, parent_item) +		self.data = [str(pid) + ":" + str(tid), "", "", "", "", "", ""] +		self.dbid = thread_id + +	def Select(self): +		super(CallTreeLevelTwoItem, self).Select() +		for child_item in self.child_items: +			self.time += child_item.time +			self.branch_count += child_item.branch_count +		for child_item in self.child_items: +			child_item.data[4] = PercentToOneDP(child_item.time, self.time) +			child_item.data[6] = PercentToOneDP(child_item.branch_count, self.branch_count) + +# Call tree data model level one item + +class CallTreeLevelOneItem(CallGraphLevelItemBase): + +	def __init__(self, glb, row, comm_id, comm, parent_item): +		super(CallTreeLevelOneItem, self).__init__(glb, row, parent_item) +		self.data = [comm, "", "", "", "", "", ""] +		self.dbid = comm_id + +	def Select(self): +		self.query_done = True; +		query = QSqlQuery(self.glb.db) +		QueryExec(query, "SELECT thread_id, pid, tid" +					" FROM comm_threads" +					" INNER JOIN threads ON thread_id = threads.id" +					" WHERE comm_id = " + str(self.dbid)) +		while query.next(): +			child_item = CallTreeLevelTwoItem(self.glb, self.child_count, self.dbid, query.value(0), query.value(1), query.value(2), self) +			self.child_items.append(child_item) +			self.child_count += 1 + +# Call tree data model root item + +class CallTreeRootItem(CallGraphLevelItemBase): + +	def __init__(self, glb): +		super(CallTreeRootItem, self).__init__(glb, 0, None) +		self.dbid = 0 +		self.query_done = True; +		query = QSqlQuery(glb.db) +		QueryExec(query, "SELECT id, comm FROM comms") +		while query.next(): +			if not query.value(0): +				continue +			child_item = CallTreeLevelOneItem(glb, self.child_count, query.value(0), query.value(1), self) +			self.child_items.append(child_item) +			self.child_count += 1 + +# Call Tree data model + +class CallTreeModel(CallGraphModelBase): + +	def __init__(self, glb, parent=None): +		super(CallTreeModel, self).__init__(glb, parent) + +	def GetRoot(self): +		return CallTreeRootItem(self.glb) + +	def columnCount(self, parent=None): +		return 7 + +	def columnHeader(self, column): +		headers = ["Call Path", "Object", "Call Time", "Time (ns) ", "Time (%) ", "Branch Count ", "Branch Count (%) "] +		return headers[column] + +	def columnAlignment(self, column): +		alignment = [ Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight, Qt.AlignRight ] +		return alignment[column] + +	def DoFindSelect(self, query, match): +		QueryExec(query, "SELECT calls.id, comm_id, thread_id" +						" FROM calls" +						" INNER JOIN call_paths ON calls.call_path_id = call_paths.id" +						" INNER JOIN symbols ON call_paths.symbol_id = symbols.id" +						" WHERE symbols.name" + match + +						" ORDER BY comm_id, thread_id, call_time, calls.id") + +	def FindPath(self, query): +		# Turn the query result into a list of ids that the tree view can walk +		# to open the tree at the right place. +		ids = [] +		parent_id = query.value(0) +		while parent_id: +			ids.insert(0, parent_id) +			q2 = QSqlQuery(self.glb.db) +			QueryExec(q2, "SELECT parent_id" +					" FROM calls" +					" WHERE id = " + str(parent_id)) +			if not q2.next(): +				break +			parent_id = q2.value(0) +		ids.insert(0, query.value(2)) +		ids.insert(0, query.value(1)) +		return ids +  # Vertical widget layout  class VBox(): @@ -693,28 +850,16 @@ class VBox():  	def Widget(self):  		return self.vbox -# Context-sensitive call graph window - -class CallGraphWindow(QMdiSubWindow): - -	def __init__(self, glb, parent=None): -		super(CallGraphWindow, self).__init__(parent) - -		self.model = LookupCreateModel("Context-Sensitive Call Graph", lambda x=glb: CallGraphModel(x)) - -		self.view = QTreeView() -		self.view.setModel(self.model) - -		for c, w in ((0, 250), (1, 100), (2, 60), (3, 70), (4, 70), (5, 100)): -			self.view.setColumnWidth(c, w) - -		self.find_bar = FindBar(self, self) +# Tree window base -		self.vbox = VBox(self.view, self.find_bar.Widget()) +class TreeWindowBase(QMdiSubWindow): -		self.setWidget(self.vbox.Widget()) +	def __init__(self, parent=None): +		super(TreeWindowBase, self).__init__(parent) -		AddSubWindow(glb.mainwindow.mdi_area, self, "Context-Sensitive Call Graph") +		self.model = None +		self.view = None +		self.find_bar = None  	def DisplayFound(self, ids):  		if not len(ids): @@ -747,6 +892,53 @@ class CallGraphWindow(QMdiSubWindow):  		if not found:  			self.find_bar.NotFound() + +# Context-sensitive call graph window + +class CallGraphWindow(TreeWindowBase): + +	def __init__(self, glb, parent=None): +		super(CallGraphWindow, self).__init__(parent) + +		self.model = LookupCreateModel("Context-Sensitive Call Graph", lambda x=glb: CallGraphModel(x)) + +		self.view = QTreeView() +		self.view.setModel(self.model) + +		for c, w in ((0, 250), (1, 100), (2, 60), (3, 70), (4, 70), (5, 100)): +			self.view.setColumnWidth(c, w) + +		self.find_bar = FindBar(self, self) + +		self.vbox = VBox(self.view, self.find_bar.Widget()) + +		self.setWidget(self.vbox.Widget()) + +		AddSubWindow(glb.mainwindow.mdi_area, self, "Context-Sensitive Call Graph") + +# Call tree window + +class CallTreeWindow(TreeWindowBase): + +	def __init__(self, glb, parent=None): +		super(CallTreeWindow, self).__init__(parent) + +		self.model = LookupCreateModel("Call Tree", lambda x=glb: CallTreeModel(x)) + +		self.view = QTreeView() +		self.view.setModel(self.model) + +		for c, w in ((0, 230), (1, 100), (2, 100), (3, 70), (4, 70), (5, 100)): +			self.view.setColumnWidth(c, w) + +		self.find_bar = FindBar(self, self) + +		self.vbox = VBox(self.view, self.find_bar.Widget()) + +		self.setWidget(self.vbox.Widget()) + +		AddSubWindow(glb.mainwindow.mdi_area, self, "Call Tree") +  # Child data item  finder  class ChildDataItemFinder(): @@ -1327,8 +1519,7 @@ class BranchModel(TreeModel):  	progress = Signal(object)  	def __init__(self, glb, event_id, where_clause, parent=None): -		super(BranchModel, self).__init__(BranchRootItem(), parent) -		self.glb = glb +		super(BranchModel, self).__init__(glb, parent)  		self.event_id = event_id  		self.more = True  		self.populated = 0 @@ -1352,6 +1543,9 @@ class BranchModel(TreeModel):  		self.fetcher.done.connect(self.Update)  		self.fetcher.Fetch(glb_chunk_sz) +	def GetRoot(self): +		return BranchRootItem() +  	def columnCount(self, parent=None):  		return 8 @@ -1398,18 +1592,28 @@ class BranchModel(TreeModel):  	def HasMoreRecords(self):  		return self.more +# Report Variables + +class ReportVars(): + +	def __init__(self, name = "", where_clause = "", limit = ""): +		self.name = name +		self.where_clause = where_clause +		self.limit = limit + +	def UniqueId(self): +		return str(self.where_clause + ";" + self.limit) +  # Branch window  class BranchWindow(QMdiSubWindow): -	def __init__(self, glb, event_id, name, where_clause, parent=None): +	def __init__(self, glb, event_id, report_vars, parent=None):  		super(BranchWindow, self).__init__(parent) -		model_name = "Branch Events " + str(event_id) -		if len(where_clause): -			model_name = where_clause + " " + model_name +		model_name = "Branch Events " + str(event_id) +  " " + report_vars.UniqueId() -		self.model = LookupCreateModel(model_name, lambda: BranchModel(glb, event_id, where_clause)) +		self.model = LookupCreateModel(model_name, lambda: BranchModel(glb, event_id, report_vars.where_clause))  		self.view = QTreeView()  		self.view.setUniformRowHeights(True) @@ -1427,7 +1631,7 @@ class BranchWindow(QMdiSubWindow):  		self.setWidget(self.vbox.Widget()) -		AddSubWindow(glb.mainwindow.mdi_area, self, name + " Branch Events") +		AddSubWindow(glb.mainwindow.mdi_area, self, report_vars.name + " Branch Events")  	def ResizeColumnToContents(self, column, n):  		# Using the view's resizeColumnToContents() here is extrememly slow @@ -1472,47 +1676,134 @@ class BranchWindow(QMdiSubWindow):  		else:  			self.find_bar.NotFound() -# Dialog data item converted and validated using a SQL table +# Line edit data item -class SQLTableDialogDataItem(): +class LineEditDataItem(object): -	def __init__(self, glb, label, placeholder_text, table_name, match_column, column_name1, column_name2, parent): +	def __init__(self, glb, label, placeholder_text, parent, id = "", default = ""):  		self.glb = glb  		self.label = label  		self.placeholder_text = placeholder_text -		self.table_name = table_name -		self.match_column = match_column -		self.column_name1 = column_name1 -		self.column_name2 = column_name2  		self.parent = parent +		self.id = id -		self.value = "" +		self.value = default -		self.widget = QLineEdit() +		self.widget = QLineEdit(default)  		self.widget.editingFinished.connect(self.Validate)  		self.widget.textChanged.connect(self.Invalidate)  		self.red = False  		self.error = ""  		self.validated = True -		self.last_id = 0 -		self.first_time = 0 -		self.last_time = 2 ** 64 -		if self.table_name == "<timeranges>": -			query = QSqlQuery(self.glb.db) -			QueryExec(query, "SELECT id, time FROM samples ORDER BY id DESC LIMIT 1") -			if query.next(): -				self.last_id = int(query.value(0)) -				self.last_time = int(query.value(1)) -			QueryExec(query, "SELECT time FROM samples WHERE time != 0 ORDER BY id LIMIT 1") -			if query.next(): -				self.first_time = int(query.value(0)) -			if placeholder_text: -				placeholder_text += ", between " + str(self.first_time) + " and " + str(self.last_time) -  		if placeholder_text:  			self.widget.setPlaceholderText(placeholder_text) +	def TurnTextRed(self): +		if not self.red: +			palette = QPalette() +			palette.setColor(QPalette.Text,Qt.red) +			self.widget.setPalette(palette) +			self.red = True + +	def TurnTextNormal(self): +		if self.red: +			palette = QPalette() +			self.widget.setPalette(palette) +			self.red = False + +	def InvalidValue(self, value): +		self.value = "" +		self.TurnTextRed() +		self.error = self.label + " invalid value '" + value + "'" +		self.parent.ShowMessage(self.error) + +	def Invalidate(self): +		self.validated = False + +	def DoValidate(self, input_string): +		self.value = input_string.strip() + +	def Validate(self): +		self.validated = True +		self.error = "" +		self.TurnTextNormal() +		self.parent.ClearMessage() +		input_string = self.widget.text() +		if not len(input_string.strip()): +			self.value = "" +			return +		self.DoValidate(input_string) + +	def IsValid(self): +		if not self.validated: +			self.Validate() +		if len(self.error): +			self.parent.ShowMessage(self.error) +			return False +		return True + +	def IsNumber(self, value): +		try: +			x = int(value) +		except: +			x = 0 +		return str(x) == value + +# Non-negative integer ranges dialog data item + +class NonNegativeIntegerRangesDataItem(LineEditDataItem): + +	def __init__(self, glb, label, placeholder_text, column_name, parent): +		super(NonNegativeIntegerRangesDataItem, self).__init__(glb, label, placeholder_text, parent) + +		self.column_name = column_name + +	def DoValidate(self, input_string): +		singles = [] +		ranges = [] +		for value in [x.strip() for x in input_string.split(",")]: +			if "-" in value: +				vrange = value.split("-") +				if len(vrange) != 2 or not self.IsNumber(vrange[0]) or not self.IsNumber(vrange[1]): +					return self.InvalidValue(value) +				ranges.append(vrange) +			else: +				if not self.IsNumber(value): +					return self.InvalidValue(value) +				singles.append(value) +		ranges = [("(" + self.column_name + " >= " + r[0] + " AND " + self.column_name + " <= " + r[1] + ")") for r in ranges] +		if len(singles): +			ranges.append(self.column_name + " IN (" + ",".join(singles) + ")") +		self.value = " OR ".join(ranges) + +# Positive integer dialog data item + +class PositiveIntegerDataItem(LineEditDataItem): + +	def __init__(self, glb, label, placeholder_text, parent, id = "", default = ""): +		super(PositiveIntegerDataItem, self).__init__(glb, label, placeholder_text, parent, id, default) + +	def DoValidate(self, input_string): +		if not self.IsNumber(input_string.strip()): +			return self.InvalidValue(input_string) +		value = int(input_string.strip()) +		if value <= 0: +			return self.InvalidValue(input_string) +		self.value = str(value) + +# Dialog data item converted and validated using a SQL table + +class SQLTableDataItem(LineEditDataItem): + +	def __init__(self, glb, label, placeholder_text, table_name, match_column, column_name1, column_name2, parent): +		super(SQLTableDataItem, self).__init__(glb, label, placeholder_text, parent) + +		self.table_name = table_name +		self.match_column = match_column +		self.column_name1 = column_name1 +		self.column_name2 = column_name2 +  	def ValueToIds(self, value):  		ids = []  		query = QSqlQuery(self.glb.db) @@ -1523,6 +1814,42 @@ class SQLTableDialogDataItem():  				ids.append(str(query.value(0)))  		return ids +	def DoValidate(self, input_string): +		all_ids = [] +		for value in [x.strip() for x in input_string.split(",")]: +			ids = self.ValueToIds(value) +			if len(ids): +				all_ids.extend(ids) +			else: +				return self.InvalidValue(value) +		self.value = self.column_name1 + " IN (" + ",".join(all_ids) + ")" +		if self.column_name2: +			self.value = "( " + self.value + " OR " + self.column_name2 + " IN (" + ",".join(all_ids) + ") )" + +# Sample time ranges dialog data item converted and validated using 'samples' SQL table + +class SampleTimeRangesDataItem(LineEditDataItem): + +	def __init__(self, glb, label, placeholder_text, column_name, parent): +		self.column_name = column_name + +		self.last_id = 0 +		self.first_time = 0 +		self.last_time = 2 ** 64 + +		query = QSqlQuery(glb.db) +		QueryExec(query, "SELECT id, time FROM samples ORDER BY id DESC LIMIT 1") +		if query.next(): +			self.last_id = int(query.value(0)) +			self.last_time = int(query.value(1)) +		QueryExec(query, "SELECT time FROM samples WHERE time != 0 ORDER BY id LIMIT 1") +		if query.next(): +			self.first_time = int(query.value(0)) +		if placeholder_text: +			placeholder_text += ", between " + str(self.first_time) + " and " + str(self.last_time) + +		super(SampleTimeRangesDataItem, self).__init__(glb, label, placeholder_text, parent) +  	def IdBetween(self, query, lower_id, higher_id, order):  		QueryExec(query, "SELECT id FROM samples WHERE id > " + str(lower_id) + " AND id < " + str(higher_id) + " ORDER BY id " + order + " LIMIT 1")  		if query.next(): @@ -1560,7 +1887,6 @@ class SQLTableDialogDataItem():  					return str(lower_id)  	def ConvertRelativeTime(self, val): -		print "val ", val  		mult = 1  		suffix = val[-2:]  		if suffix == "ms": @@ -1582,29 +1908,23 @@ class SQLTableDialogDataItem():  		return str(val)  	def ConvertTimeRange(self, vrange): -		print "vrange ", vrange  		if vrange[0] == "":  			vrange[0] = str(self.first_time)  		if vrange[1] == "":  			vrange[1] = str(self.last_time)  		vrange[0] = self.ConvertRelativeTime(vrange[0])  		vrange[1] = self.ConvertRelativeTime(vrange[1]) -		print "vrange2 ", vrange  		if not self.IsNumber(vrange[0]) or not self.IsNumber(vrange[1]):  			return False -		print "ok1"  		beg_range = max(int(vrange[0]), self.first_time)  		end_range = min(int(vrange[1]), self.last_time)  		if beg_range > self.last_time or end_range < self.first_time:  			return False -		print "ok2"  		vrange[0] = self.BinarySearchTime(0, self.last_id, beg_range, True)  		vrange[1] = self.BinarySearchTime(1, self.last_id + 1, end_range, False) -		print "vrange3 ", vrange  		return True  	def AddTimeRange(self, value, ranges): -		print "value ", value  		n = value.count("-")  		if n == 1:  			pass @@ -1622,111 +1942,31 @@ class SQLTableDialogDataItem():  			return True  		return False -	def InvalidValue(self, value): -		self.value = "" -		palette = QPalette() -		palette.setColor(QPalette.Text,Qt.red) -		self.widget.setPalette(palette) -		self.red = True -		self.error = self.label + " invalid value '" + value + "'" -		self.parent.ShowMessage(self.error) - -	def IsNumber(self, value): -		try: -			x = int(value) -		except: -			x = 0 -		return str(x) == value +	def DoValidate(self, input_string): +		ranges = [] +		for value in [x.strip() for x in input_string.split(",")]: +			if not self.AddTimeRange(value, ranges): +				return self.InvalidValue(value) +		ranges = [("(" + self.column_name + " >= " + r[0] + " AND " + self.column_name + " <= " + r[1] + ")") for r in ranges] +		self.value = " OR ".join(ranges) -	def Invalidate(self): -		self.validated = False +# Report Dialog Base -	def Validate(self): -		input_string = self.widget.text() -		self.validated = True -		if self.red: -			palette = QPalette() -			self.widget.setPalette(palette) -			self.red = False -		if not len(input_string.strip()): -			self.error = "" -			self.value = "" -			return -		if self.table_name == "<timeranges>": -			ranges = [] -			for value in [x.strip() for x in input_string.split(",")]: -				if not self.AddTimeRange(value, ranges): -					return self.InvalidValue(value) -			ranges = [("(" + self.column_name1 + " >= " + r[0] + " AND " + self.column_name1 + " <= " + r[1] + ")") for r in ranges] -			self.value = " OR ".join(ranges) -		elif self.table_name == "<ranges>": -			singles = [] -			ranges = [] -			for value in [x.strip() for x in input_string.split(",")]: -				if "-" in value: -					vrange = value.split("-") -					if len(vrange) != 2 or not self.IsNumber(vrange[0]) or not self.IsNumber(vrange[1]): -						return self.InvalidValue(value) -					ranges.append(vrange) -				else: -					if not self.IsNumber(value): -						return self.InvalidValue(value) -					singles.append(value) -			ranges = [("(" + self.column_name1 + " >= " + r[0] + " AND " + self.column_name1 + " <= " + r[1] + ")") for r in ranges] -			if len(singles): -				ranges.append(self.column_name1 + " IN (" + ",".join(singles) + ")") -			self.value = " OR ".join(ranges) -		elif self.table_name: -			all_ids = [] -			for value in [x.strip() for x in input_string.split(",")]: -				ids = self.ValueToIds(value) -				if len(ids): -					all_ids.extend(ids) -				else: -					return self.InvalidValue(value) -			self.value = self.column_name1 + " IN (" + ",".join(all_ids) + ")" -			if self.column_name2: -				self.value = "( " + self.value + " OR " + self.column_name2 + " IN (" + ",".join(all_ids) + ") )" -		else: -			self.value = input_string.strip() -		self.error = "" -		self.parent.ClearMessage() +class ReportDialogBase(QDialog): -	def IsValid(self): -		if not self.validated: -			self.Validate() -		if len(self.error): -			self.parent.ShowMessage(self.error) -			return False -		return True - -# Selected branch report creation dialog - -class SelectedBranchDialog(QDialog): - -	def __init__(self, glb, parent=None): -		super(SelectedBranchDialog, self).__init__(parent) +	def __init__(self, glb, title, items, partial, parent=None): +		super(ReportDialogBase, self).__init__(parent)  		self.glb = glb -		self.name = "" -		self.where_clause = "" +		self.report_vars = ReportVars() -		self.setWindowTitle("Selected Branches") +		self.setWindowTitle(title)  		self.setMinimumWidth(600) -		items = ( -			("Report name:", "Enter a name to appear in the window title bar", "", "", "", ""), -			("Time ranges:", "Enter time ranges", "<timeranges>", "", "samples.id", ""), -			("CPUs:", "Enter CPUs or ranges e.g. 0,5-6", "<ranges>", "", "cpu", ""), -			("Commands:", "Only branches with these commands will be included", "comms", "comm", "comm_id", ""), -			("PIDs:", "Only branches with these process IDs will be included", "threads", "pid", "thread_id", ""), -			("TIDs:", "Only branches with these thread IDs will be included", "threads", "tid", "thread_id", ""), -			("DSOs:", "Only branches with these DSOs will be included", "dsos", "short_name", "samples.dso_id", "to_dso_id"), -			("Symbols:", "Only branches with these symbols will be included", "symbols", "name", "symbol_id", "to_symbol_id"), -			("Raw SQL clause: ", "Enter a raw SQL WHERE clause", "", "", "", ""), -			) -		self.data_items = [SQLTableDialogDataItem(glb, *x, parent=self) for x in items] +		self.data_items = [x(glb, self) for x in items] + +		self.partial = partial  		self.grid = QGridLayout() @@ -1758,23 +1998,28 @@ class SelectedBranchDialog(QDialog):  		self.setLayout(self.vbox);  	def Ok(self): -		self.name = self.data_items[0].value -		if not self.name: +		vars = self.report_vars +		for d in self.data_items: +			if d.id == "REPORTNAME": +				vars.name = d.value +		if not vars.name:  			self.ShowMessage("Report name is required")  			return  		for d in self.data_items:  			if not d.IsValid():  				return  		for d in self.data_items[1:]: -			if len(d.value): -				if len(self.where_clause): -					self.where_clause += " AND " -				self.where_clause += d.value -		if len(self.where_clause): -			self.where_clause = " AND ( " + self.where_clause + " ) " -		else: -			self.ShowMessage("No selection") -			return +			if d.id == "LIMIT": +				vars.limit = d.value +			elif len(d.value): +				if len(vars.where_clause): +					vars.where_clause += " AND " +				vars.where_clause += d.value +		if len(vars.where_clause): +			if self.partial: +				vars.where_clause = " AND ( " + vars.where_clause + " ) " +			else: +				vars.where_clause = " WHERE " + vars.where_clause + " "  		self.accept()  	def ShowMessage(self, msg): @@ -1783,6 +2028,23 @@ class SelectedBranchDialog(QDialog):  	def ClearMessage(self):  		self.status.setText("") +# Selected branch report creation dialog + +class SelectedBranchDialog(ReportDialogBase): + +	def __init__(self, glb, parent=None): +		title = "Selected Branches" +		items = (lambda g, p: LineEditDataItem(g, "Report name:", "Enter a name to appear in the window title bar", p, "REPORTNAME"), +			 lambda g, p: SampleTimeRangesDataItem(g, "Time ranges:", "Enter time ranges", "samples.id", p), +			 lambda g, p: NonNegativeIntegerRangesDataItem(g, "CPUs:", "Enter CPUs or ranges e.g. 0,5-6", "cpu", p), +			 lambda g, p: SQLTableDataItem(g, "Commands:", "Only branches with these commands will be included", "comms", "comm", "comm_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "PIDs:", "Only branches with these process IDs will be included", "threads", "pid", "thread_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "TIDs:", "Only branches with these thread IDs will be included", "threads", "tid", "thread_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "DSOs:", "Only branches with these DSOs will be included", "dsos", "short_name", "samples.dso_id", "to_dso_id", p), +			 lambda g, p: SQLTableDataItem(g, "Symbols:", "Only branches with these symbols will be included", "symbols", "name", "symbol_id", "to_symbol_id", p), +			 lambda g, p: LineEditDataItem(g, "Raw SQL clause: ", "Enter a raw SQL WHERE clause", p)) +		super(SelectedBranchDialog, self).__init__(glb, title, items, True, parent) +  # Event list  def GetEventList(db): @@ -1793,6 +2055,16 @@ def GetEventList(db):  		events.append(query.value(0))  	return events +# Is a table selectable + +def IsSelectable(db, table, sql = ""): +	query = QSqlQuery(db) +	try: +		QueryExec(query, "SELECT * FROM " + table + " " + sql + " LIMIT 1") +	except: +		return False +	return True +  # SQL data preparation  def SQLTableDataPrep(query, count): @@ -1818,12 +2090,13 @@ class SQLTableModel(TableModel):  	progress = Signal(object) -	def __init__(self, glb, sql, column_count, parent=None): +	def __init__(self, glb, sql, column_headers, parent=None):  		super(SQLTableModel, self).__init__(parent)  		self.glb = glb  		self.more = True  		self.populated = 0 -		self.fetcher = SQLFetcher(glb, sql, lambda x, y=column_count: SQLTableDataPrep(x, y), self.AddSample) +		self.column_headers = column_headers +		self.fetcher = SQLFetcher(glb, sql, lambda x, y=len(column_headers): SQLTableDataPrep(x, y), self.AddSample)  		self.fetcher.done.connect(self.Update)  		self.fetcher.Fetch(glb_chunk_sz) @@ -1861,6 +2134,12 @@ class SQLTableModel(TableModel):  	def HasMoreRecords(self):  		return self.more +	def columnCount(self, parent=None): +		return len(self.column_headers) + +	def columnHeader(self, column): +		return self.column_headers[column] +  # SQL automatic table data model  class SQLAutoTableModel(SQLTableModel): @@ -1870,12 +2149,12 @@ class SQLAutoTableModel(SQLTableModel):  		if table_name == "comm_threads_view":  			# For now, comm_threads_view has no id column  			sql = "SELECT * FROM " + table_name + " WHERE comm_id > $$last_id$$ ORDER BY comm_id LIMIT " + str(glb_chunk_sz) -		self.column_headers = [] +		column_headers = []  		query = QSqlQuery(glb.db)  		if glb.dbref.is_sqlite3:  			QueryExec(query, "PRAGMA table_info(" + table_name + ")")  			while query.next(): -				self.column_headers.append(query.value(1)) +				column_headers.append(query.value(1))  			if table_name == "sqlite_master":  				sql = "SELECT * FROM " + table_name  		else: @@ -1888,14 +2167,8 @@ class SQLAutoTableModel(SQLTableModel):  				schema = "public"  			QueryExec(query, "SELECT column_name FROM information_schema.columns WHERE table_schema = '" + schema + "' and table_name = '" + select_table_name + "'")  			while query.next(): -				self.column_headers.append(query.value(0)) -		super(SQLAutoTableModel, self).__init__(glb, sql, len(self.column_headers), parent) - -	def columnCount(self, parent=None): -		return len(self.column_headers) - -	def columnHeader(self, column): -		return self.column_headers[column] +				column_headers.append(query.value(0)) +		super(SQLAutoTableModel, self).__init__(glb, sql, column_headers, parent)  # Base class for custom ResizeColumnsToContents @@ -1998,6 +2271,103 @@ def GetTableList(glb):  		tables.append("information_schema.columns")  	return tables +# Top Calls data model + +class TopCallsModel(SQLTableModel): + +	def __init__(self, glb, report_vars, parent=None): +		text = "" +		if not glb.dbref.is_sqlite3: +			text = "::text" +		limit = "" +		if len(report_vars.limit): +			limit = " LIMIT " + report_vars.limit +		sql = ("SELECT comm, pid, tid, name," +			" CASE" +			" WHEN (short_name = '[kernel.kallsyms]') THEN '[kernel]'" + text + +			" ELSE short_name" +			" END AS dso," +			" call_time, return_time, (return_time - call_time) AS elapsed_time, branch_count, " +			" CASE" +			" WHEN (calls.flags = 1) THEN 'no call'" + text + +			" WHEN (calls.flags = 2) THEN 'no return'" + text + +			" WHEN (calls.flags = 3) THEN 'no call/return'" + text + +			" ELSE ''" + text + +			" END AS flags" +			" FROM calls" +			" INNER JOIN call_paths ON calls.call_path_id = call_paths.id" +			" INNER JOIN symbols ON call_paths.symbol_id = symbols.id" +			" INNER JOIN dsos ON symbols.dso_id = dsos.id" +			" INNER JOIN comms ON calls.comm_id = comms.id" +			" INNER JOIN threads ON calls.thread_id = threads.id" + +			report_vars.where_clause + +			" ORDER BY elapsed_time DESC" + +			limit +			) +		column_headers = ("Command", "PID", "TID", "Symbol", "Object", "Call Time", "Return Time", "Elapsed Time (ns)", "Branch Count", "Flags") +		self.alignment = (Qt.AlignLeft, Qt.AlignLeft, Qt.AlignLeft, Qt.AlignLeft, Qt.AlignLeft, Qt.AlignLeft, Qt.AlignLeft, Qt.AlignRight, Qt.AlignRight, Qt.AlignLeft) +		super(TopCallsModel, self).__init__(glb, sql, column_headers, parent) + +	def columnAlignment(self, column): +		return self.alignment[column] + +# Top Calls report creation dialog + +class TopCallsDialog(ReportDialogBase): + +	def __init__(self, glb, parent=None): +		title = "Top Calls by Elapsed Time" +		items = (lambda g, p: LineEditDataItem(g, "Report name:", "Enter a name to appear in the window title bar", p, "REPORTNAME"), +			 lambda g, p: SQLTableDataItem(g, "Commands:", "Only calls with these commands will be included", "comms", "comm", "comm_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "PIDs:", "Only calls with these process IDs will be included", "threads", "pid", "thread_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "TIDs:", "Only calls with these thread IDs will be included", "threads", "tid", "thread_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "DSOs:", "Only calls with these DSOs will be included", "dsos", "short_name", "dso_id", "", p), +			 lambda g, p: SQLTableDataItem(g, "Symbols:", "Only calls with these symbols will be included", "symbols", "name", "symbol_id", "", p), +			 lambda g, p: LineEditDataItem(g, "Raw SQL clause: ", "Enter a raw SQL WHERE clause", p), +			 lambda g, p: PositiveIntegerDataItem(g, "Record limit:", "Limit selection to this number of records", p, "LIMIT", "100")) +		super(TopCallsDialog, self).__init__(glb, title, items, False, parent) + +# Top Calls window + +class TopCallsWindow(QMdiSubWindow, ResizeColumnsToContentsBase): + +	def __init__(self, glb, report_vars, parent=None): +		super(TopCallsWindow, self).__init__(parent) + +		self.data_model = LookupCreateModel("Top Calls " + report_vars.UniqueId(), lambda: TopCallsModel(glb, report_vars)) +		self.model = self.data_model + +		self.view = QTableView() +		self.view.setModel(self.model) +		self.view.setEditTriggers(QAbstractItemView.NoEditTriggers) +		self.view.verticalHeader().setVisible(False) + +		self.ResizeColumnsToContents() + +		self.find_bar = FindBar(self, self, True) + +		self.finder = ChildDataItemFinder(self.model) + +		self.fetch_bar = FetchMoreRecordsBar(self.data_model, self) + +		self.vbox = VBox(self.view, self.find_bar.Widget(), self.fetch_bar.Widget()) + +		self.setWidget(self.vbox.Widget()) + +		AddSubWindow(glb.mainwindow.mdi_area, self, report_vars.name) + +	def Find(self, value, direction, pattern, context): +		self.view.setFocus() +		self.find_bar.Busy() +		self.finder.Find(value, direction, pattern, context, self.FindDone) + +	def FindDone(self, row): +		self.find_bar.Idle() +		if row >= 0: +			self.view.setCurrentIndex(self.model.index(row, 0, QModelIndex())) +		else: +			self.find_bar.NotFound() +  # Action Definition  def CreateAction(label, tip, callback, parent=None, shortcut=None): @@ -2099,8 +2469,10 @@ p.c2 {  </style>  <p class=c1><a href=#reports>1. Reports</a></p>  <p class=c2><a href=#callgraph>1.1 Context-Sensitive Call Graph</a></p> -<p class=c2><a href=#allbranches>1.2 All branches</a></p> -<p class=c2><a href=#selectedbranches>1.3 Selected branches</a></p> +<p class=c2><a href=#calltree>1.2 Call Tree</a></p> +<p class=c2><a href=#allbranches>1.3 All branches</a></p> +<p class=c2><a href=#selectedbranches>1.4 Selected branches</a></p> +<p class=c2><a href=#topcallsbyelapsedtime>1.5 Top calls by elapsed time</a></p>  <p class=c1><a href=#tables>2. Tables</a></p>  <h1 id=reports>1. Reports</h1>  <h2 id=callgraph>1.1 Context-Sensitive Call Graph</h2> @@ -2136,7 +2508,10 @@ v- ls  <h3>Find</h3>  Ctrl-F displays a Find bar which finds function names by either an exact match or a pattern match.  The pattern matching symbols are ? for any character and * for zero or more characters. -<h2 id=allbranches>1.2 All branches</h2> +<h2 id=calltree>1.2 Call Tree</h2> +The Call Tree report is very similar to the Context-Sensitive Call Graph, but the data is not aggregated. +Also the 'Count' column, which would be always 1, is replaced by the 'Call Time'. +<h2 id=allbranches>1.3 All branches</h2>  The All branches report displays all branches in chronological order.  Not all data is fetched immediately. More records can be fetched using the Fetch bar provided.  <h3>Disassembly</h3> @@ -2162,10 +2537,10 @@ sudo ldconfig  Ctrl-F displays a Find bar which finds substrings by either an exact match or a regular expression match.  Refer to Python documentation for the regular expression syntax.  All columns are searched, but only currently fetched rows are searched. -<h2 id=selectedbranches>1.3 Selected branches</h2> +<h2 id=selectedbranches>1.4 Selected branches</h2>  This is the same as the <a href=#allbranches>All branches</a> report but with the data reduced  by various selection criteria. A dialog box displays available criteria which are AND'ed together. -<h3>1.3.1 Time ranges</h3> +<h3>1.4.1 Time ranges</h3>  The time ranges hint text shows the total time range. Relative time ranges can also be entered in  ms, us or ns. Also, negative values are relative to the end of trace.  Examples:  <pre> @@ -2176,6 +2551,10 @@ ms, us or ns. Also, negative values are relative to the end of trace.  Examples:  	-10ms-			The last 10ms  </pre>  N.B. Due to the granularity of timestamps, there could be no branches in any given time range. +<h2 id=topcallsbyelapsedtime>1.5 Top calls by elapsed time</h2> +The Top calls by elapsed time report displays calls in descending order of time elapsed between when the function was called and when it returned. +The data is reduced by various selection criteria. A dialog box displays available criteria which are AND'ed together. +If not all data is fetched, a Fetch bar is provided. Ctrl-F displays a Find bar.  <h1 id=tables>2. Tables</h1>  The Tables menu shows all tables and views in the database. Most tables have an associated view  which displays the information in a more friendly way. Not all data for large tables is fetched @@ -2305,10 +2684,17 @@ class MainWindow(QMainWindow):  		edit_menu.addAction(CreateAction("&Enlarge Font", "Make text bigger", self.EnlargeFont, self, [QKeySequence("Ctrl++")]))  		reports_menu = menu.addMenu("&Reports") -		reports_menu.addAction(CreateAction("Context-Sensitive Call &Graph", "Create a new window containing a context-sensitive call graph", self.NewCallGraph, self)) +		if IsSelectable(glb.db, "calls"): +			reports_menu.addAction(CreateAction("Context-Sensitive Call &Graph", "Create a new window containing a context-sensitive call graph", self.NewCallGraph, self)) + +		if IsSelectable(glb.db, "calls", "WHERE parent_id >= 0"): +			reports_menu.addAction(CreateAction("Call &Tree", "Create a new window containing a call tree", self.NewCallTree, self))  		self.EventMenu(GetEventList(glb.db), reports_menu) +		if IsSelectable(glb.db, "calls"): +			reports_menu.addAction(CreateAction("&Top calls by elapsed time", "Create a new window displaying top calls by elapsed time", self.NewTopCalls, self)) +  		self.TableMenu(GetTableList(glb), menu)  		self.window_menu = WindowMenu(self.mdi_area, menu) @@ -2364,14 +2750,23 @@ class MainWindow(QMainWindow):  	def NewCallGraph(self):  		CallGraphWindow(self.glb, self) +	def NewCallTree(self): +		CallTreeWindow(self.glb, self) + +	def NewTopCalls(self): +		dialog = TopCallsDialog(self.glb, self) +		ret = dialog.exec_() +		if ret: +			TopCallsWindow(self.glb, dialog.report_vars, self) +  	def NewBranchView(self, event_id): -		BranchWindow(self.glb, event_id, "", "", self) +		BranchWindow(self.glb, event_id, ReportVars(), self)  	def NewSelectedBranchView(self, event_id):  		dialog = SelectedBranchDialog(self.glb, self)  		ret = dialog.exec_()  		if ret: -			BranchWindow(self.glb, event_id, dialog.name, dialog.where_clause, self) +			BranchWindow(self.glb, event_id, dialog.report_vars, self)  	def NewTableView(self, table_name):  		TableWindow(self.glb, table_name, self) diff --git a/tools/perf/scripts/python/failed-syscalls-by-pid.py b/tools/perf/scripts/python/failed-syscalls-by-pid.py index cafeff3d74db..310efe5e7e23 100644 --- a/tools/perf/scripts/python/failed-syscalls-by-pid.py +++ b/tools/perf/scripts/python/failed-syscalls-by-pid.py @@ -5,6 +5,8 @@  # Displays system-wide failed system call totals, broken down by pid.  # If a [comm] arg is specified, only syscalls called by [comm] are displayed. +from __future__ import print_function +  import os  import sys @@ -32,7 +34,7 @@ if len(sys.argv) > 1:  syscalls = autodict()  def trace_begin(): -	print "Press control+C to stop and show the summary" +	print("Press control+C to stop and show the summary")  def trace_end():  	print_error_totals() @@ -56,23 +58,22 @@ def syscalls__sys_exit(event_name, context, common_cpu,  	raw_syscalls__sys_exit(**locals())  def print_error_totals(): -    if for_comm is not None: -	    print "\nsyscall errors for %s:\n\n" % (for_comm), -    else: -	    print "\nsyscall errors:\n\n", - -    print "%-30s  %10s\n" % ("comm [pid]", "count"), -    print "%-30s  %10s\n" % ("------------------------------", \ -                                 "----------"), - -    comm_keys = syscalls.keys() -    for comm in comm_keys: -	    pid_keys = syscalls[comm].keys() -	    for pid in pid_keys: -		    print "\n%s [%d]\n" % (comm, pid), -		    id_keys = syscalls[comm][pid].keys() -		    for id in id_keys: -			    print "  syscall: %-16s\n" % syscall_name(id), -			    ret_keys = syscalls[comm][pid][id].keys() -			    for ret, val in sorted(syscalls[comm][pid][id].iteritems(), key = lambda(k, v): (v, k),  reverse = True): -				    print "    err = %-20s  %10d\n" % (strerror(ret), val), +	if for_comm is not None: +		print("\nsyscall errors for %s:\n" % (for_comm)) +	else: +		print("\nsyscall errors:\n") + +	print("%-30s  %10s" % ("comm [pid]", "count")) +	print("%-30s  %10s" % ("------------------------------", "----------")) + +	comm_keys = syscalls.keys() +	for comm in comm_keys: +		pid_keys = syscalls[comm].keys() +		for pid in pid_keys: +			print("\n%s [%d]" % (comm, pid)) +			id_keys = syscalls[comm][pid].keys() +			for id in id_keys: +				print("  syscall: %-16s" % syscall_name(id)) +				ret_keys = syscalls[comm][pid][id].keys() +				for ret, val in sorted(syscalls[comm][pid][id].items(), key = lambda kv: (kv[1], kv[0]), reverse = True): +					print("    err = %-20s  %10d" % (strerror(ret), val)) diff --git a/tools/perf/scripts/python/futex-contention.py b/tools/perf/scripts/python/futex-contention.py index 0f5cf437b602..0c4841acf75d 100644 --- a/tools/perf/scripts/python/futex-contention.py +++ b/tools/perf/scripts/python/futex-contention.py @@ -10,6 +10,8 @@  #  # Measures futex contention +from __future__ import print_function +  import os, sys  sys.path.append(os.environ['PERF_EXEC_PATH'] + '/scripts/python/Perf-Trace-Util/lib/Perf/Trace')  from Util import * @@ -33,18 +35,18 @@ def syscalls__sys_enter_futex(event, ctxt, cpu, s, ns, tid, comm, callchain,  def syscalls__sys_exit_futex(event, ctxt, cpu, s, ns, tid, comm, callchain,  			     nr, ret): -	if thread_blocktime.has_key(tid): +	if tid in thread_blocktime:  		elapsed = nsecs(s, ns) - thread_blocktime[tid]  		add_stats(lock_waits, (tid, thread_thislock[tid]), elapsed)  		del thread_blocktime[tid]  		del thread_thislock[tid]  def trace_begin(): -	print "Press control+C to stop and show the summary" +	print("Press control+C to stop and show the summary")  def trace_end():  	for (tid, lock) in lock_waits:  		min, max, avg, count = lock_waits[tid, lock] -		print "%s[%d] lock %x contended %d times, %d avg ns" % \ -		      (process_names[tid], tid, lock, count, avg) +		print("%s[%d] lock %x contended %d times, %d avg ns" % +			(process_names[tid], tid, lock, count, avg)) diff --git a/tools/perf/scripts/python/intel-pt-events.py b/tools/perf/scripts/python/intel-pt-events.py index b19172d673af..a73847c8f548 100644 --- a/tools/perf/scripts/python/intel-pt-events.py +++ b/tools/perf/scripts/python/intel-pt-events.py @@ -10,6 +10,8 @@  # FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for  # more details. +from __future__ import print_function +  import os  import sys  import struct @@ -22,34 +24,34 @@ sys.path.append(os.environ['PERF_EXEC_PATH'] + \  #from Core import *  def trace_begin(): -	print "Intel PT Power Events and PTWRITE" +	print("Intel PT Power Events and PTWRITE")  def trace_end(): -	print "End" +	print("End")  def trace_unhandled(event_name, context, event_fields_dict): -		print ' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]) +		print(' '.join(['%s=%s'%(k,str(v))for k,v in sorted(event_fields_dict.items())]))  def print_ptwrite(raw_buf):  	data = struct.unpack_from("<IQ", raw_buf)  	flags = data[0]  	payload = data[1]  	exact_ip = flags & 1 -	print "IP: %u payload: %#x" % (exact_ip, payload), +	print("IP: %u payload: %#x" % (exact_ip, payload), end=' ')  def print_cbr(raw_buf):  	data = struct.unpack_from("<BBBBII", raw_buf)  	cbr = data[0]  	f = (data[4] + 500) / 1000  	p = ((cbr * 1000 / data[2]) + 5) / 10 -	print "%3u  freq: %4u MHz  (%3u%%)" % (cbr, f, p), +	print("%3u  freq: %4u MHz  (%3u%%)" % (cbr, f, p), end=' ')  def print_mwait(raw_buf):  	data = struct.unpack_from("<IQ", raw_buf)  	payload = data[1]  	hints = payload & 0xff  	extensions = (payload >> 32) & 0x3 -	print "hints: %#x extensions: %#x" % (hints, extensions), +	print("hints: %#x extensions: %#x" % (hints, extensions), end=' ')  def print_pwre(raw_buf):  	data = struct.unpack_from("<IQ", raw_buf) @@ -57,13 +59,14 @@ def print_pwre(raw_buf):  	hw = (payload >> 7) & 1  	cstate = (payload >> 12) & 0xf  	subcstate = (payload >> 8) & 0xf -	print "hw: %u cstate: %u sub-cstate: %u" % (hw, cstate, subcstate), +	print("hw: %u cstate: %u sub-cstate: %u" % (hw, cstate, subcstate), +		end=' ')  def print_exstop(raw_buf):  	data = struct.unpack_from("<I", raw_buf)  	flags = data[0]  	exact_ip = flags & 1 -	print "IP: %u" % (exact_ip), +	print("IP: %u" % (exact_ip), end=' ')  def print_pwrx(raw_buf):  	data = struct.unpack_from("<IQ", raw_buf) @@ -71,36 +74,39 @@ def print_pwrx(raw_buf):  	deepest_cstate = payload & 0xf  	last_cstate = (payload >> 4) & 0xf  	wake_reason = (payload >> 8) & 0xf -	print "deepest cstate: %u last cstate: %u wake reason: %#x" % (deepest_cstate, last_cstate, wake_reason), +	print("deepest cstate: %u last cstate: %u wake reason: %#x" % +		(deepest_cstate, last_cstate, wake_reason), end=' ')  def print_common_start(comm, sample, name):  	ts = sample["time"]  	cpu = sample["cpu"]  	pid = sample["pid"]  	tid = sample["tid"] -	print "%16s %5u/%-5u [%03u] %9u.%09u %7s:" % (comm, pid, tid, cpu, ts / 1000000000, ts %1000000000, name), +	print("%16s %5u/%-5u [%03u] %9u.%09u %7s:" % +		(comm, pid, tid, cpu, ts / 1000000000, ts %1000000000, name), +		end=' ')  def print_common_ip(sample, symbol, dso):  	ip = sample["ip"] -	print "%16x %s (%s)" % (ip, symbol, dso) +	print("%16x %s (%s)" % (ip, symbol, dso))  def process_event(param_dict): -        event_attr = param_dict["attr"] -        sample     = param_dict["sample"] -        raw_buf    = param_dict["raw_buf"] -        comm       = param_dict["comm"] -        name       = param_dict["ev_name"] - -        # Symbol and dso info are not always resolved -        if (param_dict.has_key("dso")): -                dso = param_dict["dso"] -        else: -                dso = "[unknown]" - -        if (param_dict.has_key("symbol")): -                symbol = param_dict["symbol"] -        else: -                symbol = "[unknown]" +	event_attr = param_dict["attr"] +	sample	 = param_dict["sample"] +	raw_buf	= param_dict["raw_buf"] +	comm	   = param_dict["comm"] +	name	   = param_dict["ev_name"] + +	# Symbol and dso info are not always resolved +	if "dso" in param_dict: +		dso = param_dict["dso"] +	else: +		dso = "[unknown]" + +	if "symbol" in param_dict: +		symbol = param_dict["symbol"] +	else: +		symbol = "[unknown]"  	if name == "ptwrite":  		print_common_start(comm, sample, name) diff --git a/tools/perf/scripts/python/mem-phys-addr.py b/tools/perf/scripts/python/mem-phys-addr.py index ebee2c5ae496..1f332e72b9b0 100644 --- a/tools/perf/scripts/python/mem-phys-addr.py +++ b/tools/perf/scripts/python/mem-phys-addr.py @@ -4,6 +4,8 @@  # Copyright (c) 2018, Intel Corporation.  from __future__ import division +from __future__ import print_function +  import os  import sys  import struct @@ -31,21 +33,24 @@ def parse_iomem():  	for i, j in enumerate(f):  		m = re.split('-|:',j,2)  		if m[2].strip() == 'System RAM': -			system_ram.append(long(m[0], 16)) -			system_ram.append(long(m[1], 16)) +			system_ram.append(int(m[0], 16)) +			system_ram.append(int(m[1], 16))  		if m[2].strip() == 'Persistent Memory': -			pmem.append(long(m[0], 16)) -			pmem.append(long(m[1], 16)) +			pmem.append(int(m[0], 16)) +			pmem.append(int(m[1], 16))  def print_memory_type(): -	print "Event: %s" % (event_name) -	print "%-40s  %10s  %10s\n" % ("Memory type", "count", "percentage"), -	print "%-40s  %10s  %10s\n" % ("----------------------------------------", \ +	print("Event: %s" % (event_name)) +	print("%-40s  %10s  %10s\n" % ("Memory type", "count", "percentage"), end='') +	print("%-40s  %10s  %10s\n" % ("----------------------------------------",  					"-----------", "-----------"), +					end='');  	total = sum(load_mem_type_cnt.values())  	for mem_type, count in sorted(load_mem_type_cnt.most_common(), \ -					key = lambda(k, v): (v, k), reverse = True): -		print "%-40s  %10d  %10.1f%%\n" % (mem_type, count, 100 * count / total), +					key = lambda kv: (kv[1], kv[0]), reverse = True): +		print("%-40s  %10d  %10.1f%%\n" % +			(mem_type, count, 100 * count / total), +			end='')  def trace_begin():  	parse_iomem() @@ -80,7 +85,7 @@ def find_memory_type(phys_addr):  	f.seek(0, 0)  	for j in f:  		m = re.split('-|:',j,2) -		if long(m[0], 16) <= phys_addr <= long(m[1], 16): +		if int(m[0], 16) <= phys_addr <= int(m[1], 16):  			return m[2]  	return "N/A" diff --git a/tools/perf/scripts/python/net_dropmonitor.py b/tools/perf/scripts/python/net_dropmonitor.py index a150164b44a3..101059971738 100755 --- a/tools/perf/scripts/python/net_dropmonitor.py +++ b/tools/perf/scripts/python/net_dropmonitor.py @@ -1,11 +1,13 @@  # Monitor the system for dropped packets and proudce a report of drop locations and counts  # SPDX-License-Identifier: GPL-2.0 +from __future__ import print_function +  import os  import sys  sys.path.append(os.environ['PERF_EXEC_PATH'] + \ -		'/scripts/python/Perf-Trace-Util/lib/Perf/Trace') +	'/scripts/python/Perf-Trace-Util/lib/Perf/Trace')  from perf_trace_context import *  from Core import * @@ -50,19 +52,19 @@ def get_sym(sloc):  		return (None, 0)  def print_drop_table(): -	print "%25s %25s %25s" % ("LOCATION", "OFFSET", "COUNT") +	print("%25s %25s %25s" % ("LOCATION", "OFFSET", "COUNT"))  	for i in drop_log.keys():  		(sym, off) = get_sym(i)  		if sym == None:  			sym = i -		print "%25s %25s %25s" % (sym, off, drop_log[i]) +		print("%25s %25s %25s" % (sym, off, drop_log[i]))  def trace_begin(): -	print "Starting trace (Ctrl-C to dump results)" +	print("Starting trace (Ctrl-C to dump results)")  def trace_end(): -	print "Gathering kallsyms data" +	print("Gathering kallsyms data")  	get_kallsyms_table()  	print_drop_table() diff --git a/tools/perf/scripts/python/netdev-times.py b/tools/perf/scripts/python/netdev-times.py index 9b2050f778f1..ea0c8b90a783 100644 --- a/tools/perf/scripts/python/netdev-times.py +++ b/tools/perf/scripts/python/netdev-times.py @@ -8,6 +8,8 @@  # dev=: show only thing related to specified device  # debug: work with debug mode. It shows buffer status. +from __future__ import print_function +  import os  import sys @@ -17,6 +19,7 @@ sys.path.append(os.environ['PERF_EXEC_PATH'] + \  from perf_trace_context import *  from Core import *  from Util import * +from functools import cmp_to_key  all_event_list = []; # insert all tracepoint event related with this script  irq_dic = {}; # key is cpu and value is a list which stacks irqs @@ -61,12 +64,12 @@ def diff_msec(src, dst):  def print_transmit(hunk):  	if dev != 0 and hunk['dev'].find(dev) < 0:  		return -	print "%7s %5d %6d.%06dsec %12.3fmsec      %12.3fmsec" % \ +	print("%7s %5d %6d.%06dsec %12.3fmsec      %12.3fmsec" %  		(hunk['dev'], hunk['len'],  		nsecs_secs(hunk['queue_t']),  		nsecs_nsecs(hunk['queue_t'])/1000,  		diff_msec(hunk['queue_t'], hunk['xmit_t']), -		diff_msec(hunk['xmit_t'], hunk['free_t'])) +		diff_msec(hunk['xmit_t'], hunk['free_t'])))  # Format for displaying rx packet processing  PF_IRQ_ENTRY= "  irq_entry(+%.3fmsec irq=%d:%s)" @@ -98,55 +101,57 @@ def print_receive(hunk):  	if show_hunk == 0:  		return -	print "%d.%06dsec cpu=%d" % \ -		(nsecs_secs(base_t), nsecs_nsecs(base_t)/1000, cpu) +	print("%d.%06dsec cpu=%d" % +		(nsecs_secs(base_t), nsecs_nsecs(base_t)/1000, cpu))  	for i in range(len(irq_list)): -		print PF_IRQ_ENTRY % \ +		print(PF_IRQ_ENTRY %  			(diff_msec(base_t, irq_list[i]['irq_ent_t']), -			irq_list[i]['irq'], irq_list[i]['name']) -		print PF_JOINT +			irq_list[i]['irq'], irq_list[i]['name'])) +		print(PF_JOINT)  		irq_event_list = irq_list[i]['event_list']  		for j in range(len(irq_event_list)):  			irq_event = irq_event_list[j]  			if irq_event['event'] == 'netif_rx': -				print PF_NET_RX % \ +				print(PF_NET_RX %  					(diff_msec(base_t, irq_event['time']), -					irq_event['skbaddr']) -				print PF_JOINT -	print PF_SOFT_ENTRY % \ -		diff_msec(base_t, hunk['sirq_ent_t']) -	print PF_JOINT +					irq_event['skbaddr'])) +				print(PF_JOINT) +	print(PF_SOFT_ENTRY % +		diff_msec(base_t, hunk['sirq_ent_t'])) +	print(PF_JOINT)  	event_list = hunk['event_list']  	for i in range(len(event_list)):  		event = event_list[i]  		if event['event_name'] == 'napi_poll': -			print PF_NAPI_POLL % \ -			    (diff_msec(base_t, event['event_t']), event['dev']) +			print(PF_NAPI_POLL % +				(diff_msec(base_t, event['event_t']), +				event['dev']))  			if i == len(event_list) - 1: -				print "" +				print("")  			else: -				print PF_JOINT +				print(PF_JOINT)  		else: -			print PF_NET_RECV % \ -			    (diff_msec(base_t, event['event_t']), event['skbaddr'], -				event['len']) +			print(PF_NET_RECV % +				(diff_msec(base_t, event['event_t']), +				event['skbaddr'], +				event['len']))  			if 'comm' in event.keys(): -				print PF_WJOINT -				print PF_CPY_DGRAM % \ +				print(PF_WJOINT) +				print(PF_CPY_DGRAM %  					(diff_msec(base_t, event['comm_t']), -					event['pid'], event['comm']) +					event['pid'], event['comm']))  			elif 'handle' in event.keys(): -				print PF_WJOINT +				print(PF_WJOINT)  				if event['handle'] == "kfree_skb": -					print PF_KFREE_SKB % \ +					print(PF_KFREE_SKB %  						(diff_msec(base_t,  						event['comm_t']), -						event['location']) +						event['location']))  				elif event['handle'] == "consume_skb": -					print PF_CONS_SKB % \ +					print(PF_CONS_SKB %  						diff_msec(base_t, -							event['comm_t']) -			print PF_JOINT +							event['comm_t'])) +			print(PF_JOINT)  def trace_begin():  	global show_tx @@ -172,8 +177,7 @@ def trace_begin():  def trace_end():  	# order all events in time -	all_event_list.sort(lambda a,b :cmp(a[EINFO_IDX_TIME], -					    b[EINFO_IDX_TIME])) +	all_event_list.sort(key=cmp_to_key(lambda a,b :a[EINFO_IDX_TIME] < b[EINFO_IDX_TIME]))  	# process all events  	for i in range(len(all_event_list)):  		event_info = all_event_list[i] @@ -210,19 +214,19 @@ def trace_end():  			print_receive(receive_hunk_list[i])  	# display transmit hunks  	if show_tx: -		print "   dev    len      Qdisc        " \ -			"       netdevice             free" +		print("   dev    len      Qdisc        " +			"       netdevice             free")  		for i in range(len(tx_free_list)):  			print_transmit(tx_free_list[i])  	if debug: -		print "debug buffer status" -		print "----------------------------" -		print "xmit Qdisc:remain:%d overflow:%d" % \ -			(len(tx_queue_list), of_count_tx_queue_list) -		print "xmit netdevice:remain:%d overflow:%d" % \ -			(len(tx_xmit_list), of_count_tx_xmit_list) -		print "receive:remain:%d overflow:%d" % \ -			(len(rx_skb_list), of_count_rx_skb_list) +		print("debug buffer status") +		print("----------------------------") +		print("xmit Qdisc:remain:%d overflow:%d" % +			(len(tx_queue_list), of_count_tx_queue_list)) +		print("xmit netdevice:remain:%d overflow:%d" % +			(len(tx_xmit_list), of_count_tx_xmit_list)) +		print("receive:remain:%d overflow:%d" % +			(len(rx_skb_list), of_count_rx_skb_list))  # called from perf, when it finds a correspoinding event  def irq__softirq_entry(name, context, cpu, sec, nsec, pid, comm, callchain, vec): @@ -254,7 +258,7 @@ def irq__irq_handler_exit(name, context, cpu, sec, nsec, pid, comm, callchain, i  	all_event_list.append(event_info)  def napi__napi_poll(name, context, cpu, sec, nsec, pid, comm, callchain, napi, -                    dev_name, work=None, budget=None): +		dev_name, work=None, budget=None):  	event_info = (name, context, cpu, nsecs(sec, nsec), pid, comm,  			napi, dev_name, work, budget)  	all_event_list.append(event_info) @@ -351,7 +355,7 @@ def handle_irq_softirq_exit(event_info):  	if irq_list == [] or event_list == 0:  		return  	rec_data = {'sirq_ent_t':sirq_ent_t, 'sirq_ext_t':time, -		    'irq_list':irq_list, 'event_list':event_list} +			'irq_list':irq_list, 'event_list':event_list}  	# merge information realted to a NET_RX softirq  	receive_hunk_list.append(rec_data) @@ -388,7 +392,7 @@ def handle_netif_receive_skb(event_info):  		skbaddr, skblen, dev_name) = event_info  	if cpu in net_rx_dic.keys():  		rec_data = {'event_name':'netif_receive_skb', -			    'event_t':time, 'skbaddr':skbaddr, 'len':skblen} +				'event_t':time, 'skbaddr':skbaddr, 'len':skblen}  		event_list = net_rx_dic[cpu]['event_list']  		event_list.append(rec_data)  		rx_skb_list.insert(0, rec_data) diff --git a/tools/perf/scripts/python/powerpc-hcalls.py b/tools/perf/scripts/python/powerpc-hcalls.py index 00e0e7476e55..8b78dc790adb 100644 --- a/tools/perf/scripts/python/powerpc-hcalls.py +++ b/tools/perf/scripts/python/powerpc-hcalls.py @@ -4,6 +4,8 @@  #  # Hypervisor call statisics +from __future__ import print_function +  import os  import sys @@ -149,7 +151,7 @@ hcall_table = {  }  def hcall_table_lookup(opcode): -	if (hcall_table.has_key(opcode)): +	if (opcode in hcall_table):  		return hcall_table[opcode]  	else:  		return opcode @@ -157,8 +159,8 @@ def hcall_table_lookup(opcode):  print_ptrn = '%-28s%10s%10s%10s%10s'  def trace_end(): -	print print_ptrn % ('hcall', 'count', 'min(ns)', 'max(ns)', 'avg(ns)') -	print '-' * 68 +	print(print_ptrn % ('hcall', 'count', 'min(ns)', 'max(ns)', 'avg(ns)')) +	print('-' * 68)  	for opcode in output:  		h_name = hcall_table_lookup(opcode)  		time = output[opcode]['time'] @@ -166,14 +168,14 @@ def trace_end():  		min_t = output[opcode]['min']  		max_t = output[opcode]['max'] -		print print_ptrn % (h_name, cnt, min_t, max_t, time/cnt) +		print(print_ptrn % (h_name, cnt, min_t, max_t, time//cnt))  def powerpc__hcall_exit(name, context, cpu, sec, nsec, pid, comm, callchain,  			opcode, retval): -	if (d_enter.has_key(cpu) and d_enter[cpu].has_key(opcode)): +	if (cpu in d_enter and opcode in d_enter[cpu]):  		diff = nsecs(sec, nsec) - d_enter[cpu][opcode] -		if (output.has_key(opcode)): +		if (opcode in output):  			output[opcode]['time'] += diff  			output[opcode]['cnt'] += 1  			if (output[opcode]['min'] > diff): @@ -190,11 +192,11 @@ def powerpc__hcall_exit(name, context, cpu, sec, nsec, pid, comm, callchain,  		del d_enter[cpu][opcode]  #	else: -#		print "Can't find matching hcall_enter event. Ignoring sample" +#		print("Can't find matching hcall_enter event. Ignoring sample")  def powerpc__hcall_entry(event_name, context, cpu, sec, nsec, pid, comm,  			 callchain, opcode): -		if (d_enter.has_key(cpu)): +		if (cpu in d_enter):  			d_enter[cpu][opcode] = nsecs(sec, nsec)  		else:  			d_enter[cpu] = {opcode: nsecs(sec, nsec)} diff --git a/tools/perf/scripts/python/sched-migration.py b/tools/perf/scripts/python/sched-migration.py index 3473e7f66081..8196e3087c9e 100644 --- a/tools/perf/scripts/python/sched-migration.py +++ b/tools/perf/scripts/python/sched-migration.py @@ -1,5 +1,3 @@ -#!/usr/bin/python -#  # Cpu task migration overview toy  #  # Copyright (C) 2010 Frederic Weisbecker <fweisbec@gmail.com> @@ -16,10 +14,10 @@ import sys  from collections import defaultdict  try: -    from UserList import UserList +	from UserList import UserList  except ImportError: -    # Python 3: UserList moved to the collections package -    from collections import UserList +	# Python 3: UserList moved to the collections package +	from collections import UserList  sys.path.append(os.environ['PERF_EXEC_PATH'] + \  	'/scripts/python/Perf-Trace-Util/lib/Perf/Trace') diff --git a/tools/perf/scripts/python/sctop.py b/tools/perf/scripts/python/sctop.py index 61621b93affb..6e0278dcb092 100644 --- a/tools/perf/scripts/python/sctop.py +++ b/tools/perf/scripts/python/sctop.py @@ -8,7 +8,14 @@  # will be refreshed every [interval] seconds.  The default interval is  # 3 seconds. -import os, sys, thread, time +from __future__ import print_function + +import os, sys, time + +try: +	import thread +except ImportError: +	import _thread as thread  sys.path.append(os.environ['PERF_EXEC_PATH'] + \  	'/scripts/python/Perf-Trace-Util/lib/Perf/Trace') @@ -62,18 +69,20 @@ def print_syscall_totals(interval):  	while 1:  		clear_term()  		if for_comm is not None: -			print "\nsyscall events for %s:\n\n" % (for_comm), +			print("\nsyscall events for %s:\n" % (for_comm))  		else: -			print "\nsyscall events:\n\n", +			print("\nsyscall events:\n") -		print "%-40s  %10s\n" % ("event", "count"), -		print "%-40s  %10s\n" % ("----------------------------------------", \ -						 "----------"), +		print("%-40s  %10s" % ("event", "count")) +		print("%-40s  %10s" % +			("----------------------------------------", +			"----------")) -		for id, val in sorted(syscalls.iteritems(), key = lambda(k, v): (v, k), \ -					      reverse = True): +		for id, val in sorted(syscalls.items(), +				key = lambda kv: (kv[1], kv[0]), +				reverse = True):  			try: -				print "%-40s  %10d\n" % (syscall_name(id), val), +				print("%-40s  %10d" % (syscall_name(id), val))  			except TypeError:  				pass  		syscalls.clear() diff --git a/tools/perf/scripts/python/stackcollapse.py b/tools/perf/scripts/python/stackcollapse.py index 1697b5e18c96..b1c4def1410a 100755 --- a/tools/perf/scripts/python/stackcollapse.py +++ b/tools/perf/scripts/python/stackcollapse.py @@ -19,13 +19,15 @@  # Written by Paolo Bonzini <pbonzini@redhat.com>  # Based on Brendan Gregg's stackcollapse-perf.pl script. +from __future__ import print_function +  import os  import sys  from collections import defaultdict  from optparse import OptionParser, make_option  sys.path.append(os.environ['PERF_EXEC_PATH'] + \ -                '/scripts/python/Perf-Trace-Util/lib/Perf/Trace') +    '/scripts/python/Perf-Trace-Util/lib/Perf/Trace')  from perf_trace_context import *  from Core import * @@ -120,7 +122,6 @@ def process_event(param_dict):      lines[stack_string] = lines[stack_string] + 1  def trace_end(): -    list = lines.keys() -    list.sort() +    list = sorted(lines)      for stack in list: -        print "%s %d" % (stack, lines[stack]) +        print("%s %d" % (stack, lines[stack])) diff --git a/tools/perf/scripts/python/stat-cpi.py b/tools/perf/scripts/python/stat-cpi.py index 8410672efb8b..01fa933ff3cf 100644 --- a/tools/perf/scripts/python/stat-cpi.py +++ b/tools/perf/scripts/python/stat-cpi.py @@ -1,6 +1,7 @@ -#!/usr/bin/env python  # SPDX-License-Identifier: GPL-2.0 +from __future__ import print_function +  data    = {}  times   = []  threads = [] @@ -20,8 +21,8 @@ def store_key(time, cpu, thread):          threads.append(thread)  def store(time, event, cpu, thread, val, ena, run): -    #print "event %s cpu %d, thread %d, time %d, val %d, ena %d, run %d" % \ -    #      (event, cpu, thread, time, val, ena, run) +    #print("event %s cpu %d, thread %d, time %d, val %d, ena %d, run %d" % +    #      (event, cpu, thread, time, val, ena, run))      store_key(time, cpu, thread)      key = get_key(time, event, cpu, thread) @@ -59,7 +60,7 @@ def stat__interval(time):              if ins != 0:                  cpi = cyc/float(ins) -            print "%15f: cpu %d, thread %d -> cpi %f (%d/%d)" % (time/(float(1000000000)), cpu, thread, cpi, cyc, ins) +            print("%15f: cpu %d, thread %d -> cpi %f (%d/%d)" % (time/(float(1000000000)), cpu, thread, cpi, cyc, ins))  def trace_end():      pass @@ -75,4 +76,4 @@ def trace_end():  #                if ins != 0:  #                    cpi = cyc/float(ins)  # -#                print "time %.9f, cpu %d, thread %d -> cpi %f" % (time/(float(1000000000)), cpu, thread, cpi) +#                print("time %.9f, cpu %d, thread %d -> cpi %f" % (time/(float(1000000000)), cpu, thread, cpi)) diff --git a/tools/perf/scripts/python/syscall-counts-by-pid.py b/tools/perf/scripts/python/syscall-counts-by-pid.py index daf314cc5dd3..f254e40c6f0f 100644 --- a/tools/perf/scripts/python/syscall-counts-by-pid.py +++ b/tools/perf/scripts/python/syscall-counts-by-pid.py @@ -5,6 +5,8 @@  # Displays system-wide system call totals, broken down by syscall.  # If a [comm] arg is specified, only syscalls called by [comm] are displayed. +from __future__ import print_function +  import os, sys  sys.path.append(os.environ['PERF_EXEC_PATH'] + \ @@ -31,17 +33,16 @@ if len(sys.argv) > 1:  syscalls = autodict()  def trace_begin(): -	print "Press control+C to stop and show the summary" +	print("Press control+C to stop and show the summary")  def trace_end():  	print_syscall_totals()  def raw_syscalls__sys_enter(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	common_callchain, id, args): - +		common_secs, common_nsecs, common_pid, common_comm, +		common_callchain, id, args):  	if (for_comm and common_comm != for_comm) or \ -	   (for_pid  and common_pid  != for_pid ): +		(for_pid and common_pid != for_pid ):  		return  	try:  		syscalls[common_comm][common_pid][id] += 1 @@ -49,26 +50,26 @@ def raw_syscalls__sys_enter(event_name, context, common_cpu,  		syscalls[common_comm][common_pid][id] = 1  def syscalls__sys_enter(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	id, args): +		common_secs, common_nsecs, common_pid, common_comm, +		id, args):  	raw_syscalls__sys_enter(**locals())  def print_syscall_totals(): -    if for_comm is not None: -	    print "\nsyscall events for %s:\n\n" % (for_comm), -    else: -	    print "\nsyscall events by comm/pid:\n\n", - -    print "%-40s  %10s\n" % ("comm [pid]/syscalls", "count"), -    print "%-40s  %10s\n" % ("----------------------------------------", \ -                                 "----------"), - -    comm_keys = syscalls.keys() -    for comm in comm_keys: -	    pid_keys = syscalls[comm].keys() -	    for pid in pid_keys: -		    print "\n%s [%d]\n" % (comm, pid), -		    id_keys = syscalls[comm][pid].keys() -		    for id, val in sorted(syscalls[comm][pid].iteritems(), \ -				  key = lambda(k, v): (v, k),  reverse = True): -			    print "  %-38s  %10d\n" % (syscall_name(id), val), +	if for_comm is not None: +		print("\nsyscall events for %s:\n" % (for_comm)) +	else: +		print("\nsyscall events by comm/pid:\n") + +	print("%-40s  %10s" % ("comm [pid]/syscalls", "count")) +	print("%-40s  %10s" % ("----------------------------------------", +				"----------")) + +	comm_keys = syscalls.keys() +	for comm in comm_keys: +		pid_keys = syscalls[comm].keys() +		for pid in pid_keys: +			print("\n%s [%d]" % (comm, pid)) +			id_keys = syscalls[comm][pid].keys() +			for id, val in sorted(syscalls[comm][pid].items(), +				key = lambda kv: (kv[1], kv[0]), reverse = True): +				print("  %-38s  %10d" % (syscall_name(id), val)) diff --git a/tools/perf/scripts/python/syscall-counts.py b/tools/perf/scripts/python/syscall-counts.py index e66a7730aeb5..8adb95ff1664 100644 --- a/tools/perf/scripts/python/syscall-counts.py +++ b/tools/perf/scripts/python/syscall-counts.py @@ -5,6 +5,8 @@  # Displays system-wide system call totals, broken down by syscall.  # If a [comm] arg is specified, only syscalls called by [comm] are displayed. +from __future__ import print_function +  import os  import sys @@ -28,14 +30,14 @@ if len(sys.argv) > 1:  syscalls = autodict()  def trace_begin(): -	print "Press control+C to stop and show the summary" +	print("Press control+C to stop and show the summary")  def trace_end():  	print_syscall_totals()  def raw_syscalls__sys_enter(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	common_callchain, id, args): +		common_secs, common_nsecs, common_pid, common_comm, +		common_callchain, id, args):  	if for_comm is not None:  		if common_comm != for_comm:  			return @@ -45,20 +47,19 @@ def raw_syscalls__sys_enter(event_name, context, common_cpu,  		syscalls[id] = 1  def syscalls__sys_enter(event_name, context, common_cpu, -	common_secs, common_nsecs, common_pid, common_comm, -	id, args): +		common_secs, common_nsecs, common_pid, common_comm, id, args):  	raw_syscalls__sys_enter(**locals())  def print_syscall_totals(): -    if for_comm is not None: -	    print "\nsyscall events for %s:\n\n" % (for_comm), -    else: -	    print "\nsyscall events:\n\n", - -    print "%-40s  %10s\n" % ("event", "count"), -    print "%-40s  %10s\n" % ("----------------------------------------", \ -                                 "-----------"), - -    for id, val in sorted(syscalls.iteritems(), key = lambda(k, v): (v, k), \ -				  reverse = True): -	    print "%-40s  %10d\n" % (syscall_name(id), val), +	if for_comm is not None: +		print("\nsyscall events for %s:\n" % (for_comm)) +	else: +		print("\nsyscall events:\n") + +	print("%-40s  %10s" % ("event", "count")) +	print("%-40s  %10s" % ("----------------------------------------", +				"-----------")) + +	for id, val in sorted(syscalls.items(), +			key = lambda kv: (kv[1], kv[0]), reverse = True): +		print("%-40s  %10d" % (syscall_name(id), val)) | 
