3 # Cpu task migration overview toy
5 # Copyright (C) 2010 Frederic Weisbecker <fweisbec@gmail.com>
7 # perf script event handlers have been generated by perf script -g python
9 # This software is distributed under the terms of the GNU General
10 # Public License ("GPL") version 2 as published by the Free Software
12 from __future__
import print_function
17 from collections
import defaultdict
19 from UserList
import UserList
21 # Python 3: UserList moved to the collections package
22 from collections
import UserList
24 sys
.path
.append(os
.environ
['PERF_EXEC_PATH'] + \
25 '/scripts/python/Perf-Trace-Util/lib/Perf/Trace')
26 sys
.path
.append('scripts/python/Perf-Trace-Util/lib/Perf/Trace')
28 from perf_trace_context
import *
30 from SchedGui
import *
33 threads
= { 0 : "idle"}
36 return "%s:%d" % (threads
[pid
], pid
)
38 class RunqueueEventUnknown
:
46 class RunqueueEventSleep
:
51 def __init__(self
, sleeper
):
52 self
.sleeper
= sleeper
55 return "%s gone to sleep" % thread_name(self
.sleeper
)
57 class RunqueueEventWakeup
:
60 return (0xff, 0xff, 0)
62 def __init__(self
, wakee
):
66 return "%s woke up" % thread_name(self
.wakee
)
68 class RunqueueEventFork
:
73 def __init__(self
, child
):
77 return "new forked task %s" % thread_name(self
.child
)
79 class RunqueueMigrateIn
:
82 return (0, 0xf0, 0xff)
84 def __init__(self
, new
):
88 return "task migrated in %s" % thread_name(self
.new
)
90 class RunqueueMigrateOut
:
93 return (0xff, 0, 0xff)
95 def __init__(self
, old
):
99 return "task migrated out %s" % thread_name(self
.old
)
101 class RunqueueSnapshot
:
102 def __init__(self
, tasks
= [0], event
= RunqueueEventUnknown()):
103 self
.tasks
= tuple(tasks
)
106 def sched_switch(self
, prev
, prev_state
, next
):
107 event
= RunqueueEventUnknown()
109 if taskState(prev_state
) == "R" and next
in self
.tasks \
110 and prev
in self
.tasks
:
113 if taskState(prev_state
) != "R":
114 event
= RunqueueEventSleep(prev
)
116 next_tasks
= list(self
.tasks
[:])
117 if prev
in self
.tasks
:
118 if taskState(prev_state
) != "R":
119 next_tasks
.remove(prev
)
120 elif taskState(prev_state
) == "R":
121 next_tasks
.append(prev
)
123 if next
not in next_tasks
:
124 next_tasks
.append(next
)
126 return RunqueueSnapshot(next_tasks
, event
)
128 def migrate_out(self
, old
):
129 if old
not in self
.tasks
:
131 next_tasks
= [task
for task
in self
.tasks
if task
!= old
]
133 return RunqueueSnapshot(next_tasks
, RunqueueMigrateOut(old
))
135 def __migrate_in(self
, new
, event
):
136 if new
in self
.tasks
:
139 next_tasks
= self
.tasks
[:] + tuple([new
])
141 return RunqueueSnapshot(next_tasks
, event
)
143 def migrate_in(self
, new
):
144 return self
.__migrate
_in
(new
, RunqueueMigrateIn(new
))
146 def wake_up(self
, new
):
147 return self
.__migrate
_in
(new
, RunqueueEventWakeup(new
))
149 def wake_up_new(self
, new
):
150 return self
.__migrate
_in
(new
, RunqueueEventFork(new
))
153 """ Provide the number of tasks on the runqueue.
155 return len(self
.tasks
) - 1
158 ret
= self
.tasks
.__repr
__()
159 ret
+= self
.origin_tostring()
164 def __init__(self
, start
, prev
):
168 # cpus that triggered the event
171 self
.total_load
= prev
.total_load
172 self
.rqs
= prev
.rqs
.copy()
174 self
.rqs
= defaultdict(RunqueueSnapshot
)
177 def __update_total_load(self
, old_rq
, new_rq
):
178 diff
= new_rq
.load() - old_rq
.load()
179 self
.total_load
+= diff
181 def sched_switch(self
, ts_list
, prev
, prev_state
, next
, cpu
):
182 old_rq
= self
.prev
.rqs
[cpu
]
183 new_rq
= old_rq
.sched_switch(prev
, prev_state
, next
)
188 self
.rqs
[cpu
] = new_rq
189 self
.__update
_total
_load
(old_rq
, new_rq
)
191 self
.event_cpus
= [cpu
]
193 def migrate(self
, ts_list
, new
, old_cpu
, new_cpu
):
194 if old_cpu
== new_cpu
:
196 old_rq
= self
.prev
.rqs
[old_cpu
]
197 out_rq
= old_rq
.migrate_out(new
)
198 self
.rqs
[old_cpu
] = out_rq
199 self
.__update
_total
_load
(old_rq
, out_rq
)
201 new_rq
= self
.prev
.rqs
[new_cpu
]
202 in_rq
= new_rq
.migrate_in(new
)
203 self
.rqs
[new_cpu
] = in_rq
204 self
.__update
_total
_load
(new_rq
, in_rq
)
208 if old_rq
is not out_rq
:
209 self
.event_cpus
.append(old_cpu
)
210 self
.event_cpus
.append(new_cpu
)
212 def wake_up(self
, ts_list
, pid
, cpu
, fork
):
213 old_rq
= self
.prev
.rqs
[cpu
]
215 new_rq
= old_rq
.wake_up_new(pid
)
217 new_rq
= old_rq
.wake_up(pid
)
221 self
.rqs
[cpu
] = new_rq
222 self
.__update
_total
_load
(old_rq
, new_rq
)
224 self
.event_cpus
= [cpu
]
228 return TimeSlice(t
, self
)
230 class TimeSliceList(UserList
):
231 def __init__(self
, arg
= []):
234 def get_time_slice(self
, ts
):
235 if len(self
.data
) == 0:
236 slice = TimeSlice(ts
, TimeSlice(-1, None))
238 slice = self
.data
[-1].next(ts
)
241 def find_time_slice(self
, ts
):
247 if start
== end
or start
== end
- 1:
250 i
= (end
+ start
) / 2
251 if self
.data
[i
].start
<= ts
and self
.data
[i
].end
>= ts
:
256 if self
.data
[i
].end
< ts
:
259 elif self
.data
[i
].start
> ts
:
264 def set_root_win(self
, win
):
267 def mouse_down(self
, cpu
, t
):
268 idx
= self
.find_time_slice(t
)
274 raw
= "CPU: %d\n" % cpu
275 raw
+= "Last event : %s\n" % rq
.event
.__repr
__()
276 raw
+= "Timestamp : %d.%06d\n" % (ts
.start
/ (10 ** 9), (ts
.start
% (10 ** 9)) / 1000)
277 raw
+= "Duration : %6d us\n" % ((ts
.end
- ts
.start
) / (10 ** 6))
278 raw
+= "Load = %d\n" % rq
.load()
280 raw
+= "%s \n" % thread_name(t
)
282 self
.root_win
.update_summary(raw
)
284 def update_rectangle_cpu(self
, slice, cpu
):
287 if slice.total_load
!= 0:
288 load_rate
= rq
.load() / float(slice.total_load
)
292 red_power
= int(0xff - (0xff * load_rate
))
293 color
= (0xff, red_power
, red_power
)
297 if cpu
in slice.event_cpus
:
298 top_color
= rq
.event
.color()
300 self
.root_win
.paint_rectangle_zone(cpu
, color
, top_color
, slice.start
, slice.end
)
302 def fill_zone(self
, start
, end
):
303 i
= self
.find_time_slice(start
)
307 for i
in range(i
, len(self
.data
)):
308 timeslice
= self
.data
[i
]
309 if timeslice
.start
> end
:
312 for cpu
in timeslice
.rqs
:
313 self
.update_rectangle_cpu(timeslice
, cpu
)
316 if len(self
.data
) == 0:
319 return (self
.data
[0].start
, self
.data
[-1].end
)
321 def nr_rectangles(self
):
322 last_ts
= self
.data
[-1]
324 for cpu
in last_ts
.rqs
:
330 class SchedEventProxy
:
332 self
.current_tsk
= defaultdict(lambda : -1)
333 self
.timeslices
= TimeSliceList()
335 def sched_switch(self
, headers
, prev_comm
, prev_pid
, prev_prio
, prev_state
,
336 next_comm
, next_pid
, next_prio
):
337 """ Ensure the task we sched out this cpu is really the one
338 we logged. Otherwise we may have missed traces """
340 on_cpu_task
= self
.current_tsk
[headers
.cpu
]
342 if on_cpu_task
!= -1 and on_cpu_task
!= prev_pid
:
343 print("Sched switch event rejected ts: %s cpu: %d prev: %s(%d) next: %s(%d)" % \
344 headers
.ts_format(), headers
.cpu
, prev_comm
, prev_pid
, next_comm
, next_pid
)
346 threads
[prev_pid
] = prev_comm
347 threads
[next_pid
] = next_comm
348 self
.current_tsk
[headers
.cpu
] = next_pid
350 ts
= self
.timeslices
.get_time_slice(headers
.ts())
351 ts
.sched_switch(self
.timeslices
, prev_pid
, prev_state
, next_pid
, headers
.cpu
)
353 def migrate(self
, headers
, pid
, prio
, orig_cpu
, dest_cpu
):
354 ts
= self
.timeslices
.get_time_slice(headers
.ts())
355 ts
.migrate(self
.timeslices
, pid
, orig_cpu
, dest_cpu
)
357 def wake_up(self
, headers
, comm
, pid
, success
, target_cpu
, fork
):
360 ts
= self
.timeslices
.get_time_slice(headers
.ts())
361 ts
.wake_up(self
.timeslices
, pid
, target_cpu
, fork
)
366 parser
= SchedEventProxy()
370 timeslices
= parser
.timeslices
371 frame
= RootFrame(timeslices
, "Migration")
374 def sched__sched_stat_runtime(event_name
, context
, common_cpu
,
375 common_secs
, common_nsecs
, common_pid
, common_comm
,
376 common_callchain
, comm
, pid
, runtime
, vruntime
):
379 def sched__sched_stat_iowait(event_name
, context
, common_cpu
,
380 common_secs
, common_nsecs
, common_pid
, common_comm
,
381 common_callchain
, comm
, pid
, delay
):
384 def sched__sched_stat_sleep(event_name
, context
, common_cpu
,
385 common_secs
, common_nsecs
, common_pid
, common_comm
,
386 common_callchain
, comm
, pid
, delay
):
389 def sched__sched_stat_wait(event_name
, context
, common_cpu
,
390 common_secs
, common_nsecs
, common_pid
, common_comm
,
391 common_callchain
, comm
, pid
, delay
):
394 def sched__sched_process_fork(event_name
, context
, common_cpu
,
395 common_secs
, common_nsecs
, common_pid
, common_comm
,
396 common_callchain
, parent_comm
, parent_pid
, child_comm
, child_pid
):
399 def sched__sched_process_wait(event_name
, context
, common_cpu
,
400 common_secs
, common_nsecs
, common_pid
, common_comm
,
401 common_callchain
, comm
, pid
, prio
):
404 def sched__sched_process_exit(event_name
, context
, common_cpu
,
405 common_secs
, common_nsecs
, common_pid
, common_comm
,
406 common_callchain
, comm
, pid
, prio
):
409 def sched__sched_process_free(event_name
, context
, common_cpu
,
410 common_secs
, common_nsecs
, common_pid
, common_comm
,
411 common_callchain
, comm
, pid
, prio
):
414 def sched__sched_migrate_task(event_name
, context
, common_cpu
,
415 common_secs
, common_nsecs
, common_pid
, common_comm
,
416 common_callchain
, comm
, pid
, prio
, orig_cpu
,
418 headers
= EventHeaders(common_cpu
, common_secs
, common_nsecs
,
419 common_pid
, common_comm
, common_callchain
)
420 parser
.migrate(headers
, pid
, prio
, orig_cpu
, dest_cpu
)
422 def sched__sched_switch(event_name
, context
, common_cpu
,
423 common_secs
, common_nsecs
, common_pid
, common_comm
, common_callchain
,
424 prev_comm
, prev_pid
, prev_prio
, prev_state
,
425 next_comm
, next_pid
, next_prio
):
427 headers
= EventHeaders(common_cpu
, common_secs
, common_nsecs
,
428 common_pid
, common_comm
, common_callchain
)
429 parser
.sched_switch(headers
, prev_comm
, prev_pid
, prev_prio
, prev_state
,
430 next_comm
, next_pid
, next_prio
)
432 def sched__sched_wakeup_new(event_name
, context
, common_cpu
,
433 common_secs
, common_nsecs
, common_pid
, common_comm
,
434 common_callchain
, comm
, pid
, prio
, success
,
436 headers
= EventHeaders(common_cpu
, common_secs
, common_nsecs
,
437 common_pid
, common_comm
, common_callchain
)
438 parser
.wake_up(headers
, comm
, pid
, success
, target_cpu
, 1)
440 def sched__sched_wakeup(event_name
, context
, common_cpu
,
441 common_secs
, common_nsecs
, common_pid
, common_comm
,
442 common_callchain
, comm
, pid
, prio
, success
,
444 headers
= EventHeaders(common_cpu
, common_secs
, common_nsecs
,
445 common_pid
, common_comm
, common_callchain
)
446 parser
.wake_up(headers
, comm
, pid
, success
, target_cpu
, 0)
448 def sched__sched_wait_task(event_name
, context
, common_cpu
,
449 common_secs
, common_nsecs
, common_pid
, common_comm
,
450 common_callchain
, comm
, pid
, prio
):
453 def sched__sched_kthread_stop_ret(event_name
, context
, common_cpu
,
454 common_secs
, common_nsecs
, common_pid
, common_comm
,
455 common_callchain
, ret
):
458 def sched__sched_kthread_stop(event_name
, context
, common_cpu
,
459 common_secs
, common_nsecs
, common_pid
, common_comm
,
460 common_callchain
, comm
, pid
):
463 def trace_unhandled(event_name
, context
, event_fields_dict
):