| 
									
										
										
										
											2020-01-30 17:32:25 +01:00
										 |  |  | #!/usr/bin/env python3 | 
					
						
							| 
									
										
										
										
											2018-01-11 11:27:22 +03:00
										 |  |  | # -*- coding: utf-8 -*- | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Analyse lock events and compute statistics | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Author: Alex Bennée <alex.bennee@linaro.org> | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import simpletrace | 
					
						
							|  |  |  | import argparse | 
					
						
							|  |  |  | import numpy as np | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class MutexAnalyser(simpletrace.Analyzer): | 
					
						
							|  |  |  |     "A simpletrace Analyser for checking locks." | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def __init__(self): | 
					
						
							|  |  |  |         self.locks = 0 | 
					
						
							|  |  |  |         self.locked = 0 | 
					
						
							|  |  |  |         self.unlocks = 0 | 
					
						
							|  |  |  |         self.mutex_records = {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _get_mutex(self, mutex): | 
					
						
							|  |  |  |         if not mutex in self.mutex_records: | 
					
						
							|  |  |  |             self.mutex_records[mutex] = {"locks": 0, | 
					
						
							|  |  |  |                                          "lock_time": 0, | 
					
						
							|  |  |  |                                          "acquire_times": [], | 
					
						
							|  |  |  |                                          "locked": 0, | 
					
						
							|  |  |  |                                          "locked_time": 0, | 
					
						
							|  |  |  |                                          "held_times": [], | 
					
						
							|  |  |  |                                          "unlocked": 0} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return self.mutex_records[mutex] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def qemu_mutex_lock(self, timestamp, mutex, filename, line): | 
					
						
							|  |  |  |         self.locks += 1 | 
					
						
							|  |  |  |         rec = self._get_mutex(mutex) | 
					
						
							|  |  |  |         rec["locks"] += 1 | 
					
						
							|  |  |  |         rec["lock_time"] = timestamp[0] | 
					
						
							|  |  |  |         rec["lock_loc"] = (filename, line) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def qemu_mutex_locked(self, timestamp, mutex, filename, line): | 
					
						
							|  |  |  |         self.locked += 1 | 
					
						
							|  |  |  |         rec = self._get_mutex(mutex) | 
					
						
							|  |  |  |         rec["locked"] += 1 | 
					
						
							|  |  |  |         rec["locked_time"] = timestamp[0] | 
					
						
							|  |  |  |         acquire_time = rec["locked_time"] - rec["lock_time"] | 
					
						
							|  |  |  |         rec["locked_loc"] = (filename, line) | 
					
						
							|  |  |  |         rec["acquire_times"].append(acquire_time) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def qemu_mutex_unlock(self, timestamp, mutex, filename, line): | 
					
						
							|  |  |  |         self.unlocks += 1 | 
					
						
							|  |  |  |         rec = self._get_mutex(mutex) | 
					
						
							|  |  |  |         rec["unlocked"] += 1 | 
					
						
							|  |  |  |         held_time = timestamp[0] - rec["locked_time"] | 
					
						
							|  |  |  |         rec["held_times"].append(held_time) | 
					
						
							|  |  |  |         rec["unlock_loc"] = (filename, line) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | def get_args(): | 
					
						
							|  |  |  |     "Grab options" | 
					
						
							|  |  |  |     parser = argparse.ArgumentParser() | 
					
						
							|  |  |  |     parser.add_argument("--output", "-o", type=str, help="Render plot to file") | 
					
						
							|  |  |  |     parser.add_argument("events", type=str, help='trace file read from') | 
					
						
							|  |  |  |     parser.add_argument("tracefile", type=str, help='trace file read from') | 
					
						
							|  |  |  |     return parser.parse_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | if __name__ == '__main__': | 
					
						
							|  |  |  |     args = get_args() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Gather data from the trace | 
					
						
							|  |  |  |     analyser = MutexAnalyser() | 
					
						
							|  |  |  |     simpletrace.process(args.events, args.tracefile, analyser) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     print ("Total locks: %d, locked: %d, unlocked: %d" % | 
					
						
							|  |  |  |            (analyser.locks, analyser.locked, analyser.unlocks)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # Now dump the individual lock stats | 
					
						
							|  |  |  |     for key, val in sorted(analyser.mutex_records.iteritems(), | 
					
						
							| 
									
										
										
										
											2018-06-08 09:29:49 -03:00
										 |  |  |                            key=lambda k_v: k_v[1]["locks"]): | 
					
						
							| 
									
										
										
										
											2018-01-11 11:27:22 +03:00
										 |  |  |         print ("Lock: %#x locks: %d, locked: %d, unlocked: %d" % | 
					
						
							|  |  |  |                (key, val["locks"], val["locked"], val["unlocked"])) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         acquire_times = np.array(val["acquire_times"]) | 
					
						
							|  |  |  |         if len(acquire_times) > 0: | 
					
						
							|  |  |  |             print ("  Acquire Time: min:%d median:%d avg:%.2f max:%d" % | 
					
						
							|  |  |  |                    (acquire_times.min(), np.median(acquire_times), | 
					
						
							|  |  |  |                     acquire_times.mean(), acquire_times.max())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         held_times = np.array(val["held_times"]) | 
					
						
							|  |  |  |         if len(held_times) > 0: | 
					
						
							|  |  |  |             print ("  Held Time: min:%d median:%d avg:%.2f max:%d" % | 
					
						
							|  |  |  |                    (held_times.min(), np.median(held_times), | 
					
						
							|  |  |  |                     held_times.mean(), held_times.max())) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # Check if any locks still held | 
					
						
							|  |  |  |         if val["locks"] > val["locked"]: | 
					
						
							|  |  |  |             print ("  LOCK HELD (%s:%s)" % (val["locked_loc"])) | 
					
						
							|  |  |  |             print ("  BLOCKED   (%s:%s)" % (val["lock_loc"])) |