Update sdk/platform-tools to version 26.0.0.
[android_tools.git] / sdk / platform-tools / systrace / catapult / telemetry / telemetry / timeline / thread.py
blob5810ae35c141a3bced80bcad7c07a2ecb4b9bb0a
1 # Copyright 2014 The Chromium Authors. All rights reserved.
2 # Use of this source code is governed by a BSD-style license that can be
3 # found in the LICENSE file.
4 import telemetry.timeline.async_slice as async_slice_module
5 import telemetry.timeline.event_container as event_container
6 import telemetry.timeline.flow_event as flow_event_module
7 import telemetry.timeline.sample as sample_module
8 import telemetry.timeline.slice as slice_module
11 class Thread(event_container.TimelineEventContainer):
12 """A Thread stores all the trace events collected for a particular
13 thread. We organize the synchronous slices on a thread by "subrows," where
14 subrow 0 has all the root slices, subrow 1 those nested 1 deep, and so on.
15 The asynchronous slices are stored in an AsyncSliceGroup object.
16 """
17 def __init__(self, process, tid):
18 super(Thread, self).__init__('thread %s' % tid, parent=process)
19 self.tid = tid
20 self._async_slices = []
21 self._flow_events = []
22 self._samples = []
23 self._toplevel_slices = []
24 self._all_slices = []
26 # State only valid during import.
27 self._open_slices = []
28 self._newly_added_slices = []
30 @property
31 def toplevel_slices(self):
32 return self._toplevel_slices
34 @property
35 def all_slices(self):
36 return self._all_slices
38 @property
39 def samples(self):
40 return self._samples
42 @property
43 def async_slices(self):
44 return self._async_slices
46 @property
47 def open_slice_count(self):
48 return len(self._open_slices)
50 def IterChildContainers(self):
51 return
52 yield # pylint: disable=unreachable
54 def IterEventsInThisContainer(self, event_type_predicate, event_predicate):
55 if event_type_predicate(slice_module.Slice):
56 for s in self._newly_added_slices:
57 if event_predicate(s):
58 yield s
59 for s in self._all_slices:
60 if event_predicate(s):
61 yield s
63 if event_type_predicate(async_slice_module.AsyncSlice):
64 for async_slice in self._async_slices:
65 if event_predicate(async_slice):
66 yield async_slice
67 for sub_slice in async_slice.IterEventsInThisContainerRecrusively():
68 if event_predicate(sub_slice):
69 yield sub_slice
71 if event_type_predicate(flow_event_module.FlowEvent):
72 for flow_event in self._flow_events:
73 if event_predicate(flow_event):
74 yield flow_event
76 if event_type_predicate(sample_module.Sample):
77 for sample in self._samples:
78 if event_predicate(sample):
79 yield sample
81 def AddSample(self, category, name, timestamp, args=None):
82 if len(self._samples) and timestamp < self._samples[-1].start:
83 raise ValueError(
84 'Samples must be added in increasing timestamp order')
85 sample = sample_module.Sample(self,
86 category, name, timestamp, args=args)
87 self._samples.append(sample)
89 def AddAsyncSlice(self, async_slice):
90 self._async_slices.append(async_slice)
92 def AddFlowEvent(self, flow_event):
93 self._flow_events.append(flow_event)
95 def BeginSlice(self, category, name, timestamp, thread_timestamp=None,
96 args=None):
97 """Opens a new slice for the thread.
98 Calls to beginSlice and endSlice must be made with
99 non-monotonically-decreasing timestamps.
101 * category: Category to which the slice belongs.
102 * name: Name of the slice to add.
103 * timestamp: The timetsamp of the slice, in milliseconds.
104 * thread_timestamp: Thread specific clock (scheduled) timestamp of the
105 slice, in milliseconds.
106 * args: Arguments associated with
108 Returns newly opened slice
110 if len(self._open_slices) > 0 and timestamp < self._open_slices[-1].start:
111 raise ValueError(
112 'Slices must be added in increasing timestamp order')
113 new_slice = slice_module.Slice(self, category, name, timestamp,
114 thread_timestamp=thread_timestamp,
115 args=args)
116 self._open_slices.append(new_slice)
117 new_slice.did_not_finish = True
118 self.PushSlice(new_slice)
119 return new_slice
121 def EndSlice(self, end_timestamp, end_thread_timestamp=None):
122 """ Ends the last begun slice in this group and pushes it onto the slice
123 array.
125 * end_timestamp: Timestamp when the slice ended in milliseconds
126 * end_thread_timestamp: Timestamp when the scheduled time of the slice ended
127 in milliseconds
129 returns completed slice.
131 if not len(self._open_slices):
132 raise ValueError(
133 'EndSlice called without an open slice')
134 curr_slice = self._open_slices.pop()
135 if end_timestamp < curr_slice.start:
136 raise ValueError(
137 'Slice %s end time is before its start.' % curr_slice.name)
138 curr_slice.duration = end_timestamp - curr_slice.start
139 # On Windows, it is possible to have a value for |end_thread_timestamp|
140 # but not for |curr_slice.thread_start|, because it takes some time to
141 # initialize the thread time timer.
142 if curr_slice.thread_start != None and end_thread_timestamp != None:
143 curr_slice.thread_duration = (end_thread_timestamp -
144 curr_slice.thread_start)
145 curr_slice.did_not_finish = False
146 return curr_slice
148 def PushCompleteSlice(self, category, name, timestamp, duration,
149 thread_timestamp, thread_duration, args=None):
150 new_slice = slice_module.Slice(self, category, name, timestamp,
151 thread_timestamp=thread_timestamp,
152 args=args)
153 if duration == None:
154 new_slice.did_not_finish = True
155 else:
156 new_slice.duration = duration
157 new_slice.thread_duration = thread_duration
158 self.PushSlice(new_slice)
159 return new_slice
161 def PushMarkSlice(self, category, name, timestamp, thread_timestamp,
162 args=None):
163 new_slice = slice_module.Slice(self, category, name, timestamp,
164 thread_timestamp=thread_timestamp,
165 args=args)
166 self.PushSlice(new_slice)
167 return new_slice
169 def PushSlice(self, new_slice):
170 self._newly_added_slices.append(new_slice)
171 return new_slice
173 def AutoCloseOpenSlices(self, max_timestamp, max_thread_timestamp):
174 for s in self._newly_added_slices:
175 if s.did_not_finish:
176 s.duration = max_timestamp - s.start
177 assert s.duration >= 0
178 if s.thread_start != None:
179 s.thread_duration = max_thread_timestamp - s.thread_start
180 assert s.thread_duration >= 0
181 self._open_slices = []
183 def IsTimestampValidForBeginOrEnd(self, timestamp):
184 if not len(self._open_slices):
185 return True
186 return timestamp >= self._open_slices[-1].start
188 def FinalizeImport(self):
189 self._BuildSliceSubRows()
191 def _BuildSliceSubRows(self):
192 """This function works by walking through slices by start time.
194 The basic idea here is to insert each slice as deep into the subrow
195 list as it can go such that every subslice is fully contained by its
196 parent slice.
198 Visually, if we start with this:
199 0: [ a ]
200 1: [ b ]
201 2: [c][d]
203 To place this slice:
205 We first check row 2's last item, [d]. [e] wont fit into [d] (they dont
206 even intersect). So we go to row 1. That gives us [b], and [d] wont fit
207 into that either. So, we go to row 0 and its last slice, [a]. That can
208 completely contain [e], so that means we should add [e] as a subslice
209 of [a]. That puts it on row 1, yielding:
210 0: [ a ]
211 1: [ b ][e]
212 2: [c][d]
214 If we then get this slice:
216 We do the same deepest-to-shallowest walk of the subrows trying to fit
217 it. This time, it doesn't fit in any open slice. So, we simply append
218 it to row 0 (a root slice):
219 0: [ a ] [f]
220 1: [ b ][e]
222 def CompareSlices(s1, s2):
223 if s1.start == s2.start:
224 # Break ties by having the slice with the greatest
225 # end timestamp come first.
226 return cmp(s2.end, s1.end)
227 return cmp(s1.start, s2.start)
229 assert len(self._toplevel_slices) == 0
230 assert len(self._all_slices) == 0
231 if not len(self._newly_added_slices):
232 return
234 self._all_slices.extend(self._newly_added_slices)
236 sorted_slices = sorted(self._newly_added_slices, cmp=CompareSlices)
237 root_slice = sorted_slices[0]
238 self._toplevel_slices.append(root_slice)
239 for s in sorted_slices[1:]:
240 if not self._AddSliceIfBounds(root_slice, s):
241 root_slice = s
242 self._toplevel_slices.append(root_slice)
243 self._newly_added_slices = []
246 def _AddSliceIfBounds(self, root, child):
247 """Adds a child slice to a root slice its proper row.
248 Return False if the child slice is not in the bounds
249 of the root slice.
251 Because we know that the start time of child is >= the start time
252 of all other slices seen so far, we can just check the last slice
253 of each row for bounding.
255 # The source trace data is in microseconds but we store it as milliseconds
256 # in floating-point. Since we can't represent micros as millis perfectly,
257 # two end=start+duration combos that should be the same will be slightly
258 # different. Round back to micros to ensure equality below.
259 child_end_micros = round(child.end * 1000)
260 root_end_micros = round(root.end * 1000)
261 if child.start >= root.start and child_end_micros <= root_end_micros:
262 if len(root.sub_slices) > 0:
263 if self._AddSliceIfBounds(root.sub_slices[-1], child):
264 return True
265 child.parent_slice = root
266 root.AddSubSlice(child)
267 return True
268 return False