mirror of
				https://github.com/ARM-software/devlib.git
				synced 2025-10-26 11:33:19 +00:00 
			
		
		
		
	Merge pull request #16 from derkling/add_ftrace_function_profiling
Add ftrace function profiling
This commit is contained in:
		| @@ -43,6 +43,18 @@ cpufreq_trace_all_frequencies() { | |||||||
| 	done | 	done | ||||||
| } | } | ||||||
|  |  | ||||||
|  | ################################################################################ | ||||||
|  | # FTrace Utility Functions | ||||||
|  | ################################################################################ | ||||||
|  |  | ||||||
|  | ftrace_get_function_stats() { | ||||||
|  |     for CPU in $(ls /sys/kernel/debug/tracing/trace_stat | sed 's/function//'); do | ||||||
|  |         REPLACE_STRING="s/  Function/\n  Function (CPU$CPU)/" | ||||||
|  |         cat /sys/kernel/debug/tracing/trace_stat/function$CPU \ | ||||||
|  |             | sed "$REPLACE_STRING" | ||||||
|  |     done | ||||||
|  | } | ||||||
|  |  | ||||||
| ################################################################################ | ################################################################################ | ||||||
| # Main Function Dispatcher | # Main Function Dispatcher | ||||||
| ################################################################################ | ################################################################################ | ||||||
| @@ -63,6 +75,9 @@ cpufreq_get_all_governors) | |||||||
| cpufreq_trace_all_frequencies) | cpufreq_trace_all_frequencies) | ||||||
|     cpufreq_trace_all_frequencies $* |     cpufreq_trace_all_frequencies $* | ||||||
|     ;; |     ;; | ||||||
|  | ftrace_get_function_stats) | ||||||
|  |     ftrace_get_function_stats | ||||||
|  |     ;; | ||||||
| *) | *) | ||||||
|     echo "Command [$CMD] not supported" |     echo "Command [$CMD] not supported" | ||||||
|     exit -1 |     exit -1 | ||||||
|   | |||||||
| @@ -15,7 +15,9 @@ | |||||||
|  |  | ||||||
| from __future__ import division | from __future__ import division | ||||||
| import os | import os | ||||||
|  | import json | ||||||
| import time | import time | ||||||
|  | import re | ||||||
| import subprocess | import subprocess | ||||||
|  |  | ||||||
| from devlib.trace import TraceCollector | from devlib.trace import TraceCollector | ||||||
| @@ -27,6 +29,7 @@ from devlib.utils.misc import check_output, which | |||||||
| TRACE_MARKER_START = 'TRACE_MARKER_START' | TRACE_MARKER_START = 'TRACE_MARKER_START' | ||||||
| TRACE_MARKER_STOP = 'TRACE_MARKER_STOP' | TRACE_MARKER_STOP = 'TRACE_MARKER_STOP' | ||||||
| OUTPUT_TRACE_FILE = 'trace.dat' | OUTPUT_TRACE_FILE = 'trace.dat' | ||||||
|  | OUTPUT_PROFILE_FILE = 'trace_stat.dat' | ||||||
| DEFAULT_EVENTS = [ | DEFAULT_EVENTS = [ | ||||||
|     'cpu_frequency', |     'cpu_frequency', | ||||||
|     'cpu_idle', |     'cpu_idle', | ||||||
| @@ -40,15 +43,18 @@ DEFAULT_EVENTS = [ | |||||||
| ] | ] | ||||||
| TIMEOUT = 180 | TIMEOUT = 180 | ||||||
|  |  | ||||||
|  | # Regexps for parsing of function profiling data | ||||||
|  | CPU_RE = re.compile(r'  Function \(CPU([0-9]+)\)') | ||||||
|  | STATS_RE = re.compile(r'([^ ]*) +([0-9]+) +([0-9.]+) us +([0-9.]+) us +([0-9.]+) us') | ||||||
|  |  | ||||||
| class FtraceCollector(TraceCollector): | class FtraceCollector(TraceCollector): | ||||||
|  |  | ||||||
|     def __init__(self, target, |     def __init__(self, target, | ||||||
|                  events=None, |                  events=None, | ||||||
|  |                  functions=None, | ||||||
|                  buffer_size=None, |                  buffer_size=None, | ||||||
|                  buffer_size_step=1000, |                  buffer_size_step=1000, | ||||||
|                  buffer_size_file='/sys/kernel/debug/tracing/buffer_size_kb', |                  tracing_path='/sys/kernel/debug/tracing', | ||||||
|                  marker_file='/sys/kernel/debug/tracing/trace_marker', |  | ||||||
|                  automark=True, |                  automark=True, | ||||||
|                  autoreport=True, |                  autoreport=True, | ||||||
|                  autoview=False, |                  autoview=False, | ||||||
| @@ -56,10 +62,10 @@ class FtraceCollector(TraceCollector): | |||||||
|                  ): |                  ): | ||||||
|         super(FtraceCollector, self).__init__(target) |         super(FtraceCollector, self).__init__(target) | ||||||
|         self.events = events if events is not None else DEFAULT_EVENTS |         self.events = events if events is not None else DEFAULT_EVENTS | ||||||
|  |         self.functions = functions | ||||||
|         self.buffer_size = buffer_size |         self.buffer_size = buffer_size | ||||||
|         self.buffer_size_step = buffer_size_step |         self.buffer_size_step = buffer_size_step | ||||||
|         self.buffer_size_file = buffer_size_file |         self.tracing_path = tracing_path | ||||||
|         self.marker_file = marker_file |  | ||||||
|         self.automark = automark |         self.automark = automark | ||||||
|         self.autoreport = autoreport |         self.autoreport = autoreport | ||||||
|         self.autoview = autoview |         self.autoview = autoview | ||||||
| @@ -69,8 +75,17 @@ class FtraceCollector(TraceCollector): | |||||||
|         self.start_time = None |         self.start_time = None | ||||||
|         self.stop_time = None |         self.stop_time = None | ||||||
|         self.event_string = _build_trace_events(self.events) |         self.event_string = _build_trace_events(self.events) | ||||||
|  |         self.function_string = _build_trace_functions(self.functions) | ||||||
|         self._reset_needed = True |         self._reset_needed = True | ||||||
|  |  | ||||||
|  |         # Setup tracing paths | ||||||
|  |         self.available_functions_file = self.target.path.join(self.tracing_path, 'available_filter_functions') | ||||||
|  |         self.buffer_size_file         = self.target.path.join(self.tracing_path, 'buffer_size_kb') | ||||||
|  |         self.current_tracer_file      = self.target.path.join(self.tracing_path, 'current_tracer') | ||||||
|  |         self.function_profile_file    = self.target.path.join(self.tracing_path, 'function_profile_enabled') | ||||||
|  |         self.marker_file              = self.target.path.join(self.tracing_path, 'trace_marker') | ||||||
|  |         self.ftrace_filter_file       = self.target.path.join(self.tracing_path, 'set_ftrace_filter') | ||||||
|  |  | ||||||
|         self.host_binary = which('trace-cmd') |         self.host_binary = which('trace-cmd') | ||||||
|         self.kernelshark = which('kernelshark') |         self.kernelshark = which('kernelshark') | ||||||
|  |  | ||||||
| @@ -88,6 +103,18 @@ class FtraceCollector(TraceCollector): | |||||||
|                 raise TargetError('No trace-cmd found on device and no_install=True is specified.') |                 raise TargetError('No trace-cmd found on device and no_install=True is specified.') | ||||||
|             self.target_binary = 'trace-cmd' |             self.target_binary = 'trace-cmd' | ||||||
|  |  | ||||||
|  |         # Check for function tracing support | ||||||
|  |         if self.functions: | ||||||
|  |             if not self.target.file_exists(self.function_profile_file): | ||||||
|  |                 raise TargetError('Function profiling not supported. '\ | ||||||
|  |                         'A kernel build with CONFIG_FUNCTION_PROFILER enable is required') | ||||||
|  |             # Validate required functions to be traced | ||||||
|  |             available_functions = self.target.execute( | ||||||
|  |                     'cat {}'.format(self.available_functions_file)).splitlines() | ||||||
|  |             for function in self.functions: | ||||||
|  |                 if function not in available_functions: | ||||||
|  |                     raise TargetError('Function [{}] not available for filtering'.format(function)) | ||||||
|  |  | ||||||
|     def reset(self): |     def reset(self): | ||||||
|         if self.buffer_size: |         if self.buffer_size: | ||||||
|             self._set_buffer_size() |             self._set_buffer_size() | ||||||
| @@ -104,8 +131,17 @@ class FtraceCollector(TraceCollector): | |||||||
|         if 'cpufreq' in self.target.modules: |         if 'cpufreq' in self.target.modules: | ||||||
|             self.logger.debug('Trace CPUFreq frequencies') |             self.logger.debug('Trace CPUFreq frequencies') | ||||||
|             self.target.cpufreq.trace_frequencies() |             self.target.cpufreq.trace_frequencies() | ||||||
|  |         # Enable kernel function profiling | ||||||
|  |         if self.functions: | ||||||
|  |             self.target.execute('echo nop > {}'.format(self.current_tracer_file)) | ||||||
|  |             self.target.execute('echo 0 > {}'.format(self.function_profile_file)) | ||||||
|  |             self.target.execute('echo {} > {}'.format(self.function_string, self.ftrace_filter_file)) | ||||||
|  |             self.target.execute('echo 1 > {}'.format(self.function_profile_file)) | ||||||
|  |  | ||||||
|     def stop(self): |     def stop(self): | ||||||
|  |         # Disable kernel function profiling | ||||||
|  |         if self.functions: | ||||||
|  |             self.target.execute('echo 1 > {}'.format(self.function_profile_file)) | ||||||
|         if 'cpufreq' in self.target.modules: |         if 'cpufreq' in self.target.modules: | ||||||
|             self.logger.debug('Trace CPUFreq frequencies') |             self.logger.debug('Trace CPUFreq frequencies') | ||||||
|             self.target.cpufreq.trace_frequencies() |             self.target.cpufreq.trace_frequencies() | ||||||
| @@ -135,6 +171,44 @@ class FtraceCollector(TraceCollector): | |||||||
|             if self.autoview: |             if self.autoview: | ||||||
|                 self.view(outfile) |                 self.view(outfile) | ||||||
|  |  | ||||||
|  |     def get_stats(self, outfile): | ||||||
|  |         if not self.functions: | ||||||
|  |             return | ||||||
|  |  | ||||||
|  |         if os.path.isdir(outfile): | ||||||
|  |             outfile = os.path.join(outfile, OUTPUT_PROFILE_FILE) | ||||||
|  |         output = self.target._execute_util('ftrace_get_function_stats', | ||||||
|  |                                             as_root=True) | ||||||
|  |  | ||||||
|  |         function_stats = {} | ||||||
|  |         for line in output.splitlines(): | ||||||
|  |             # Match a new CPU dataset | ||||||
|  |             match = CPU_RE.search(line) | ||||||
|  |             if match: | ||||||
|  |                 cpu_id = int(match.group(1)) | ||||||
|  |                 function_stats[cpu_id] = {} | ||||||
|  |                 self.logger.debug("Processing stats for CPU%d...", cpu_id) | ||||||
|  |                 continue | ||||||
|  |             # Match a new function dataset | ||||||
|  |             match = STATS_RE.search(line) | ||||||
|  |             if match: | ||||||
|  |                 fname = match.group(1) | ||||||
|  |                 function_stats[cpu_id][fname] = { | ||||||
|  |                         'hits' : int(match.group(2)), | ||||||
|  |                         'time' : float(match.group(3)), | ||||||
|  |                         'avg'  : float(match.group(4)), | ||||||
|  |                         's_2'  : float(match.group(5)), | ||||||
|  |                     } | ||||||
|  |                 self.logger.debug(" %s: %s", | ||||||
|  |                              fname, function_stats[cpu_id][fname]) | ||||||
|  |  | ||||||
|  |         self.logger.debug("FTrace stats output [%s]...", outfile) | ||||||
|  |         with open(outfile, 'w') as fh: | ||||||
|  |            json.dump(function_stats, fh, indent=4) | ||||||
|  |         self.logger.info("FTrace function stats save in [%s]", outfile) | ||||||
|  |  | ||||||
|  |         return function_stats | ||||||
|  |  | ||||||
|     def report(self, binfile, destfile): |     def report(self, binfile, destfile): | ||||||
|         # To get the output of trace.dat, trace-cmd must be installed |         # To get the output of trace.dat, trace-cmd must be installed | ||||||
|         # This is done host-side because the generated file is very large |         # This is done host-side because the generated file is very large | ||||||
| @@ -203,3 +277,6 @@ def _build_trace_events(events): | |||||||
|     event_string = ' '.join(['-e {}'.format(e) for e in events]) |     event_string = ' '.join(['-e {}'.format(e) for e in events]) | ||||||
|     return event_string |     return event_string | ||||||
|  |  | ||||||
|  | def _build_trace_functions(functions): | ||||||
|  |     function_string = " ".join(functions) | ||||||
|  |     return function_string | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user