diff options
author | Jonathan <hermanjl@hermanjl-Aspire-5553G.(none)> | 2012-03-07 23:11:07 -0500 |
---|---|---|
committer | Jonathan <hermanjl@hermanjl-Aspire-5553G.(none)> | 2012-03-07 23:11:07 -0500 |
commit | e9f207ed3c68bca111d0e21a2bf601b4564ed748 (patch) | |
tree | 3ec203a17d81814a0a5f2afa465e4db4ffb186d3 | |
parent | 000249b6b36adb88e4fea5e020319b9361d0ca54 (diff) |
rt-graph: real-time task plots divorced from regular task plots
-rw-r--r-- | rt-graph.c | 143 | ||||
-rw-r--r-- | rt-graph.h | 12 | ||||
-rw-r--r-- | rt-plot-task.c | 268 | ||||
-rw-r--r-- | rt-plot-task.h | 5 | ||||
-rw-r--r-- | trace-graph.h | 1 | ||||
-rw-r--r-- | trace-plot-task.c | 53 | ||||
-rw-r--r-- | trace-plot-task.h | 16 |
7 files changed, 256 insertions, 242 deletions
@@ -1,7 +1,7 @@ | |||
1 | #include "rt-graph.h" | 1 | #include "trace-graph.h" |
2 | #include "trace-hash.h" | 2 | #include "trace-hash.h" |
3 | 3 | ||
4 | #define DEBUG_LEVEL 0 | 4 | #define DEBUG_LEVEL 4 |
5 | #if DEBUG_LEVEL > 0 | 5 | #if DEBUG_LEVEL > 0 |
6 | #define dprintf(l, x...) \ | 6 | #define dprintf(l, x...) \ |
7 | do { \ | 7 | do { \ |
@@ -17,6 +17,9 @@ static guint get_event_hash_key(gint eid) | |||
17 | return trace_hash(eid) % TS_HASH_SIZE; | 17 | return trace_hash(eid) % TS_HASH_SIZE; |
18 | } | 18 | } |
19 | 19 | ||
20 | /* | ||
21 | * Returns cached field for @eid at @key. | ||
22 | */ | ||
20 | struct format_field* find_ts_hash(struct ts_list **events, | 23 | struct format_field* find_ts_hash(struct ts_list **events, |
21 | gint key, gint eid) | 24 | gint key, gint eid) |
22 | { | 25 | { |
@@ -29,10 +32,11 @@ struct format_field* find_ts_hash(struct ts_list **events, | |||
29 | } | 32 | } |
30 | 33 | ||
31 | /* | 34 | /* |
32 | * Return format field for @eid, caching its location if this is the first try | 35 | * Return field for @eid at @key, caching if necessary. |
33 | */ | 36 | */ |
34 | static struct format_field* add_ts_hash(struct ts_list **events, gint eid, gint key, | 37 | static struct format_field* |
35 | struct pevent *pevent, struct record *record) | 38 | add_ts_hash(struct ts_list **events, gint eid, gint key, |
39 | struct pevent *pevent, struct record *record) | ||
36 | { | 40 | { |
37 | struct ts_list *list; | 41 | struct ts_list *list; |
38 | struct format_field *field; | 42 | struct format_field *field; |
@@ -59,16 +63,12 @@ static struct format_field* add_ts_hash(struct ts_list **events, gint eid, gint | |||
59 | */ | 63 | */ |
60 | int rt_graph_check_any(struct rt_graph_info *rtinfo, | 64 | int rt_graph_check_any(struct rt_graph_info *rtinfo, |
61 | struct pevent *pevent, struct record *record, | 65 | struct pevent *pevent, struct record *record, |
62 | gint *epid, unsigned long long *ts) | 66 | gint *epid, gint *out_eid, unsigned long long *ts) |
63 | { | 67 | { |
64 | guint key, eid; | 68 | guint key, eid; |
65 | struct format_field *field; | 69 | struct format_field *field; |
66 | 70 | ||
67 | eid = pevent_data_type(pevent, record); | 71 | eid = pevent_data_type(pevent, record); |
68 | |||
69 | if (eid == rtinfo->switch_away_id) | ||
70 | return 0; | ||
71 | |||
72 | key = get_event_hash_key(eid); | 72 | key = get_event_hash_key(eid); |
73 | field = find_ts_hash(rtinfo->events, key, eid); | 73 | field = find_ts_hash(rtinfo->events, key, eid); |
74 | 74 | ||
@@ -80,6 +80,7 @@ int rt_graph_check_any(struct rt_graph_info *rtinfo, | |||
80 | 80 | ||
81 | dprintf(3, "Read (%d) record for task %d at %llu\n", | 81 | dprintf(3, "Read (%d) record for task %d at %llu\n", |
82 | eid, *epid, *ts); | 82 | eid, *epid, *ts); |
83 | *out_eid = eid; | ||
83 | return 1; | 84 | return 1; |
84 | } | 85 | } |
85 | 86 | ||
@@ -222,7 +223,7 @@ int rt_graph_check_switch_away(struct rt_graph_info *rtinfo, | |||
222 | 223 | ||
223 | /** | 224 | /** |
224 | * rt_graph_check_task_release - check for litmus_task_release record | 225 | * rt_graph_check_task_release - check for litmus_task_release record |
225 | * Return 1 and @pid, @job, and @deadline if the record matches | 226 | * Return 1 and @pid, @job, @release, and @deadline if the record matches |
226 | */ | 227 | */ |
227 | int rt_graph_check_task_release(struct rt_graph_info *rtinfo, | 228 | int rt_graph_check_task_release(struct rt_graph_info *rtinfo, |
228 | struct pevent *pevent, struct record *record, | 229 | struct pevent *pevent, struct record *record, |
@@ -271,7 +272,7 @@ int rt_graph_check_task_release(struct rt_graph_info *rtinfo, | |||
271 | 272 | ||
272 | /** | 273 | /** |
273 | * rt_graph_check_task_completion - check for litmus_task_completion record | 274 | * rt_graph_check_task_completion - check for litmus_task_completion record |
274 | * Return 1 and @pid, @job if the record matches | 275 | * Return 1 and @pid, @job, and @ts if the record matches |
275 | */ | 276 | */ |
276 | int rt_graph_check_task_completion(struct rt_graph_info *rtinfo, | 277 | int rt_graph_check_task_completion(struct rt_graph_info *rtinfo, |
277 | struct pevent *pevent, struct record *record, | 278 | struct pevent *pevent, struct record *record, |
@@ -314,7 +315,7 @@ int rt_graph_check_task_completion(struct rt_graph_info *rtinfo, | |||
314 | 315 | ||
315 | /** | 316 | /** |
316 | * rt_graph_check_task_block - check for litmus_task_block record | 317 | * rt_graph_check_task_block - check for litmus_task_block record |
317 | * Return 1 and @pid if the record matches | 318 | * Return 1, @pid, and @ts if the record matches |
318 | */ | 319 | */ |
319 | int rt_graph_check_task_block(struct rt_graph_info *rtinfo, | 320 | int rt_graph_check_task_block(struct rt_graph_info *rtinfo, |
320 | struct pevent *pevent, struct record *record, | 321 | struct pevent *pevent, struct record *record, |
@@ -431,3 +432,119 @@ void init_rt_event_cache(struct rt_graph_info *rtinfo) | |||
431 | rtinfo->resume_pid_field = NULL; | 432 | rtinfo->resume_pid_field = NULL; |
432 | rtinfo->resume_ts_field = NULL; | 433 | rtinfo->resume_ts_field = NULL; |
433 | } | 434 | } |
435 | |||
436 | /** | ||
437 | * get_rts - extract real-time timestamp from a record | ||
438 | * | ||
439 | * This will only have to extract the timestamp once; after the time | ||
440 | * is extracted it will be cached in the record itself. | ||
441 | */ | ||
442 | unsigned long long | ||
443 | get_rts(struct graph_info *ginfo, struct record *record) | ||
444 | { | ||
445 | gint epid, eid; | ||
446 | unsigned long long ts; | ||
447 | if (!record->cached_rts) { | ||
448 | rt_graph_check_any(&ginfo->rtinfo, ginfo->pevent, record, | ||
449 | &epid, &eid, &ts); | ||
450 | record->cached_rts = ts; | ||
451 | } else | ||
452 | ts = record->cached_rts; | ||
453 | return ts; | ||
454 | } | ||
455 | |||
456 | /** | ||
457 | * next_rts - find a real-time timestamp AROUND an FTRACE time | ||
458 | * @ginfo: Current state of the graph | ||
459 | * @cpu: CPU to search | ||
460 | * @ft_target: FTRACE time to seek towards | ||
461 | * | ||
462 | * Returns the RT time of a record CLOSELY BEFORE @ft_time. | ||
463 | */ | ||
464 | unsigned long long | ||
465 | next_rts(struct graph_info *ginfo, int cpu, unsigned long long ft_target) | ||
466 | { | ||
467 | struct record *record; | ||
468 | unsigned long long ts = 0ULL; | ||
469 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, ft_target); | ||
470 | record = tracecmd_read_data(ginfo->handle, cpu); | ||
471 | if (record) { | ||
472 | ts = get_rts(ginfo, record); | ||
473 | free_record(record); | ||
474 | return ts; | ||
475 | } else | ||
476 | return 0; | ||
477 | } | ||
478 | |||
479 | /** | ||
480 | * set_cpu_to_rts - seek CPU to a time closely preceding a real-time timestamp | ||
481 | * @ginfo: Current state o the graph | ||
482 | * @cpu: The CPU to seek | ||
483 | * @rt_target: RT time to seek towards | ||
484 | * | ||
485 | * This seeks to a real-time timestamp, not the default ftrace timestamps. | ||
486 | * The @cpu seek location will be placed before the given time, but will | ||
487 | * not necessarily be placed _right_ before the time. | ||
488 | */ | ||
489 | void | ||
490 | set_cpu_to_rts(struct graph_info *ginfo, unsigned long long rt_target, int cpu) | ||
491 | { | ||
492 | struct record *record; | ||
493 | unsigned long long last_rts, rts, seek_time, last_seek; | ||
494 | long long diff; | ||
495 | |||
496 | rts = next_rts(ginfo, cpu, rt_target); | ||
497 | diff = rt_target - rts; | ||
498 | |||
499 | /* "Guess" a new target based on difference */ | ||
500 | seek_time = rt_target + diff; | ||
501 | rts = next_rts(ginfo, cpu, seek_time); | ||
502 | diff = rt_target - rts; | ||
503 | |||
504 | /* Zero in in 1.5x the difference increments */ | ||
505 | if (rts && diff > 0) { | ||
506 | /* rts rt_target | ||
507 | * seek ? | ||
508 | * ---|---->>----|--- | ||
509 | */ | ||
510 | do { | ||
511 | last_seek = seek_time; | ||
512 | last_rts = rts; | ||
513 | seek_time = seek_time + 1.5 * (rt_target - rts); | ||
514 | rts = next_rts(ginfo, cpu, seek_time); | ||
515 | } while (rts < rt_target && last_rts != rts); | ||
516 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, last_seek); | ||
517 | seek_time = last_seek; | ||
518 | } else if (rts && diff < 0) { | ||
519 | /* rt_target rts | ||
520 | * ? seek | ||
521 | * ---|----<<----|--- | ||
522 | */ | ||
523 | do { | ||
524 | seek_time = seek_time - 1.5 * (rts - rt_target); | ||
525 | rts = next_rts(ginfo, cpu, seek_time); | ||
526 | } while (rts > rt_target); | ||
527 | } | ||
528 | |||
529 | /* Get to first record at or after time */ | ||
530 | while ((record = tracecmd_read_data(ginfo->handle, cpu))) { | ||
531 | if (get_rts(ginfo, record) >= rt_target) | ||
532 | break; | ||
533 | free_record(record); | ||
534 | } | ||
535 | if (record) { | ||
536 | tracecmd_set_cursor(ginfo->handle, cpu, record->offset); | ||
537 | free_record(record); | ||
538 | } else | ||
539 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, seek_time); | ||
540 | } | ||
541 | |||
542 | /** | ||
543 | * set_cpus_to_time - seek all cpus to real-time @rt_target | ||
544 | */ | ||
545 | void set_cpus_to_rts(struct graph_info *ginfo, unsigned long long rt_target) | ||
546 | { | ||
547 | int cpu; | ||
548 | for (cpu = 0; cpu < ginfo->cpus; cpu++) | ||
549 | set_cpu_to_rts(ginfo, rt_target, cpu); | ||
550 | } | ||
@@ -65,7 +65,7 @@ struct ts_list { | |||
65 | /* Event parsers */ | 65 | /* Event parsers */ |
66 | int rt_graph_check_any(struct rt_graph_info *rtinfo, | 66 | int rt_graph_check_any(struct rt_graph_info *rtinfo, |
67 | struct pevent *pevent, struct record *record, | 67 | struct pevent *pevent, struct record *record, |
68 | gint *pid, unsigned long long *ts); | 68 | gint *pid, gint *eid, unsigned long long *ts); |
69 | int rt_graph_check_task_param(struct rt_graph_info *rtinfo, struct pevent *pevent, | 69 | int rt_graph_check_task_param(struct rt_graph_info *rtinfo, struct pevent *pevent, |
70 | struct record *record, gint *pid, | 70 | struct record *record, gint *pid, |
71 | unsigned long long *wcet, | 71 | unsigned long long *wcet, |
@@ -91,4 +91,14 @@ int rt_graph_check_task_resume(struct rt_graph_info *rtinfo, struct pevent *peve | |||
91 | unsigned long long *when); | 91 | unsigned long long *when); |
92 | void init_rt_event_cache(struct rt_graph_info *rtinfo); | 92 | void init_rt_event_cache(struct rt_graph_info *rtinfo); |
93 | 93 | ||
94 | /* Methods for dealing with RT timestamps */ | ||
95 | unsigned long long get_rts(struct graph_info *ginfo, | ||
96 | struct record *record); | ||
97 | unsigned long long next_rts(struct graph_info *ginfo, int cpu, | ||
98 | unsigned long long ft_target); | ||
99 | void set_cpu_to_rts(struct graph_info *ginfo, | ||
100 | unsigned long long rt_target, int cpu); | ||
101 | void set_cpus_to_rts(struct graph_info *ginfo, | ||
102 | unsigned long long rt_target); | ||
103 | |||
94 | #endif | 104 | #endif |
diff --git a/rt-plot-task.c b/rt-plot-task.c index 31f4c7e..f28b7b1 100644 --- a/rt-plot-task.c +++ b/rt-plot-task.c | |||
@@ -1,4 +1,5 @@ | |||
1 | #include "trace-graph.h" | 1 | #include "trace-graph.h" |
2 | #include "trace-filter.h" | ||
2 | 3 | ||
3 | #define LLABEL 30 | 4 | #define LLABEL 30 |
4 | 5 | ||
@@ -13,105 +14,11 @@ | |||
13 | #define dprintf(l, x...) do { if (0) printf(x); } while (0) | 14 | #define dprintf(l, x...) do { if (0) printf(x); } while (0) |
14 | #endif | 15 | #endif |
15 | 16 | ||
16 | /* | ||
17 | * Extract timestamp from a record, attempting to use cache if possible | ||
18 | */ | ||
19 | static unsigned long long | ||
20 | get_rts(struct graph_info *ginfo, struct record *record) | ||
21 | { | ||
22 | gint epid; | ||
23 | unsigned long long ts; | ||
24 | if (!record->cached_rts) { | ||
25 | rt_graph_check_any(&ginfo->rtinfo, ginfo->pevent, record, | ||
26 | &epid, &ts); | ||
27 | record->cached_rts = ts; | ||
28 | } else | ||
29 | ts = record->cached_rts; | ||
30 | return ts; | ||
31 | } | ||
32 | |||
33 | /* | ||
34 | * Get the real-time timestamp of the next record at time | ||
35 | */ | ||
36 | static unsigned long long | ||
37 | next_rts(struct graph_info *ginfo, int cpu, unsigned long long time) | ||
38 | { | ||
39 | struct record *record; | ||
40 | unsigned long long ts; | ||
41 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, time); | ||
42 | record = tracecmd_read_data(ginfo->handle, cpu); | ||
43 | if (record) { | ||
44 | ts = get_rts(ginfo, record); | ||
45 | free_record(record); | ||
46 | return ts; | ||
47 | } else | ||
48 | return 0; | ||
49 | } | ||
50 | |||
51 | static void set_cpu_to_time(int cpu, struct graph_info *ginfo, unsigned long long time) | ||
52 | { | ||
53 | struct record *record; | ||
54 | unsigned long long last_rts, rts, seek_time, last_seek; | ||
55 | long long diff; | ||
56 | |||
57 | rts = next_rts(ginfo, cpu, time); | ||
58 | diff = time - rts; | ||
59 | |||
60 | /* "Guess" a new target based on difference */ | ||
61 | seek_time = time + diff; | ||
62 | rts = next_rts(ginfo, cpu, seek_time); | ||
63 | diff = time - rts; | ||
64 | |||
65 | /* Zero in in 1.5x the difference increments */ | ||
66 | if (rts && diff > 0) { | ||
67 | /* rts time | ||
68 | * seek ? | ||
69 | * ---|---->>----|--- | ||
70 | */ | ||
71 | do { | ||
72 | last_seek = seek_time; | ||
73 | last_rts = rts; | ||
74 | seek_time = seek_time + 1.5 * (time - rts); | ||
75 | rts = next_rts(ginfo, cpu, seek_time); | ||
76 | } while (rts < time && last_rts != rts); | ||
77 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, last_seek); | ||
78 | seek_time = last_seek; | ||
79 | } else if (rts && diff < 0) { | ||
80 | /* time rts | ||
81 | * ? seek | ||
82 | * ---|----<<----|--- | ||
83 | */ | ||
84 | do { | ||
85 | seek_time = seek_time - 1.5 * (rts - time); | ||
86 | rts = next_rts(ginfo, cpu, seek_time); | ||
87 | } while (rts > time); | ||
88 | } | ||
89 | |||
90 | /* Get to first record at or after time */ | ||
91 | while ((record = tracecmd_read_data(ginfo->handle, cpu))) { | ||
92 | if (get_rts(ginfo, record) >= time) | ||
93 | break; | ||
94 | free_record(record); | ||
95 | } | ||
96 | if (record) { | ||
97 | tracecmd_set_cursor(ginfo->handle, cpu, record->offset); | ||
98 | free_record(record); | ||
99 | } else | ||
100 | tracecmd_set_cpu_to_timestamp(ginfo->handle, cpu, seek_time); | ||
101 | } | ||
102 | |||
103 | void set_cpus_to_time(struct graph_info *ginfo, unsigned long long time) | ||
104 | { | ||
105 | int cpu; | ||
106 | for (cpu = 0; cpu < ginfo->cpus; cpu++) | ||
107 | set_cpu_to_time(cpu, ginfo, time); | ||
108 | } | ||
109 | |||
110 | static gboolean record_matches_pid(struct graph_info *ginfo, | 17 | static gboolean record_matches_pid(struct graph_info *ginfo, |
111 | struct record *record, | 18 | struct record *record, |
112 | int match_pid) | 19 | int match_pid) |
113 | { | 20 | { |
114 | gint dint, pid = 0, match = 0; | 21 | gint dint, pid = 0, match; |
115 | unsigned long long dull; | 22 | unsigned long long dull; |
116 | struct rt_graph_info *rtg_info = &ginfo->rtinfo; | 23 | struct rt_graph_info *rtg_info = &ginfo->rtinfo; |
117 | 24 | ||
@@ -125,29 +32,46 @@ static gboolean record_matches_pid(struct graph_info *ginfo, | |||
125 | rt_graph_check_task_completion(MARGS, &dint, &dull) || | 32 | rt_graph_check_task_completion(MARGS, &dint, &dull) || |
126 | rt_graph_check_task_block(MARGS, &dull) || | 33 | rt_graph_check_task_block(MARGS, &dull) || |
127 | rt_graph_check_task_resume(MARGS, &dull) || | 34 | rt_graph_check_task_resume(MARGS, &dull) || |
128 | rt_graph_check_any(MARGS, &dull); | 35 | rt_graph_check_any(MARGS, &dint, &dull); |
129 | #undef MARGS | 36 | #undef MARGS |
130 | return match && pid == match_pid; | 37 | return pid == match_pid; |
131 | } | 38 | } |
132 | 39 | ||
133 | struct record* | 40 | static struct record* |
134 | find_record(struct graph_info *ginfo, gint pid, guint64 time) | 41 | __find_record(struct graph_info *ginfo, gint pid, guint64 time, int display) |
135 | { | 42 | { |
136 | int next_cpu, match; | 43 | int next_cpu, match, eid, is_sa = 0; |
137 | struct record *record = NULL; | 44 | struct record *record = NULL; |
45 | struct rt_graph_info *rtg_info = &ginfo->rtinfo; | ||
138 | 46 | ||
139 | set_cpus_to_time(ginfo, time); | 47 | set_cpus_to_rts(ginfo, time); |
140 | do { | 48 | do { |
141 | free_record(record); | 49 | free_record(record); |
142 | record = tracecmd_read_next_data(ginfo->handle, &next_cpu); | 50 | record = tracecmd_read_next_data(ginfo->handle, &next_cpu); |
143 | if (!record) | 51 | if (!record) |
144 | return NULL; | 52 | return NULL; |
145 | match = record_matches_pid(ginfo, record, pid); | 53 | match = record_matches_pid(ginfo, record, pid); |
146 | } while (!(get_rts(ginfo, record) > time && match)); | 54 | if (display) { |
55 | eid = pevent_data_type(ginfo->pevent, record); | ||
56 | is_sa = (eid == rtg_info->switch_away_id); | ||
57 | } | ||
58 | } while (!(get_rts(ginfo, record) > time && match && !is_sa)); | ||
147 | 59 | ||
148 | return record; | 60 | return record; |
149 | } | 61 | } |
150 | 62 | ||
63 | static inline struct record* | ||
64 | find_record(struct graph_info *ginfo, gint pid, guint64 time) | ||
65 | { | ||
66 | return __find_record(ginfo, pid, time, 0); | ||
67 | } | ||
68 | |||
69 | static inline struct record* | ||
70 | find_display_record(struct graph_info *ginfo, gint pid, guint64 time) | ||
71 | { | ||
72 | return __find_record(ginfo, pid, time, 1); | ||
73 | } | ||
74 | |||
151 | /* | 75 | /* |
152 | * Update current job in @rtt_info, ensuring monotonic increase | 76 | * Update current job in @rtt_info, ensuring monotonic increase |
153 | */ | 77 | */ |
@@ -155,12 +79,12 @@ static int update_job(struct rt_task_info *rtt_info, int job) | |||
155 | { | 79 | { |
156 | if (job < rtt_info->last_job) { | 80 | if (job < rtt_info->last_job) { |
157 | printf("Inconsistent job state for %d:%d -> %d\n", | 81 | printf("Inconsistent job state for %d:%d -> %d\n", |
158 | rtt_info->base.pid, rtt_info->last_job, job); | 82 | rtt_info->pid, rtt_info->last_job, job); |
159 | return 0; | 83 | return 0; |
160 | } else if (job > rtt_info->last_job) { | 84 | } else if (job > rtt_info->last_job) { |
161 | rtt_info->last_job = job; | 85 | rtt_info->last_job = job; |
162 | snprintf(rtt_info->label, LLABEL, "%d:%d", | 86 | snprintf(rtt_info->label, LLABEL, "%d:%d", |
163 | rtt_info->base.pid, rtt_info->last_job); | 87 | rtt_info->pid, rtt_info->last_job); |
164 | } | 88 | } |
165 | return 1; | 89 | return 1; |
166 | } | 90 | } |
@@ -177,7 +101,7 @@ static int try_param(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
177 | 101 | ||
178 | match = rt_graph_check_task_param(&ginfo->rtinfo, ginfo->pevent, | 102 | match = rt_graph_check_task_param(&ginfo->rtinfo, ginfo->pevent, |
179 | record, &pid, &wcet, &period); | 103 | record, &pid, &wcet, &period); |
180 | if (match && pid == rtt_info->base.pid) { | 104 | if (match && pid == rtt_info->pid) { |
181 | update_job(rtt_info, 0); | 105 | update_job(rtt_info, 0); |
182 | rtt_info->wcet = wcet; | 106 | rtt_info->wcet = wcet; |
183 | rtt_info->period = period; | 107 | rtt_info->period = period; |
@@ -199,7 +123,7 @@ static int try_release(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
199 | match = rt_graph_check_task_release(&ginfo->rtinfo, ginfo->pevent, | 123 | match = rt_graph_check_task_release(&ginfo->rtinfo, ginfo->pevent, |
200 | record, &pid, &job, | 124 | record, &pid, &job, |
201 | &release, &deadline); | 125 | &release, &deadline); |
202 | if (match && pid == rtt_info->base.pid) { | 126 | if (match && pid == rtt_info->pid) { |
203 | update_job(rtt_info, job); | 127 | update_job(rtt_info, job); |
204 | info->release = TRUE; | 128 | info->release = TRUE; |
205 | info->rtime = release; | 129 | info->rtime = release; |
@@ -226,7 +150,7 @@ static int try_completion(struct graph_info *ginfo, | |||
226 | 150 | ||
227 | match = rt_graph_check_task_completion(&ginfo->rtinfo, ginfo->pevent, | 151 | match = rt_graph_check_task_completion(&ginfo->rtinfo, ginfo->pevent, |
228 | record, &pid, &job, &ts); | 152 | record, &pid, &job, &ts); |
229 | if (match && pid == rtt_info->base.pid) { | 153 | if (match && pid == rtt_info->pid) { |
230 | update_job(rtt_info, job); | 154 | update_job(rtt_info, job); |
231 | info->completion = TRUE; | 155 | info->completion = TRUE; |
232 | info->ctime = ts; | 156 | info->ctime = ts; |
@@ -244,7 +168,7 @@ static int try_block(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
244 | 168 | ||
245 | match = rt_graph_check_task_block(&ginfo->rtinfo, ginfo->pevent, | 169 | match = rt_graph_check_task_block(&ginfo->rtinfo, ginfo->pevent, |
246 | record, &pid, &ts); | 170 | record, &pid, &ts); |
247 | if (match && pid == rtt_info->base.pid) { | 171 | if (match && pid == rtt_info->pid) { |
248 | rtt_info->block_time = ts; | 172 | rtt_info->block_time = ts; |
249 | ret = 1; | 173 | ret = 1; |
250 | } | 174 | } |
@@ -259,13 +183,13 @@ static int try_resume(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
259 | 183 | ||
260 | match = rt_graph_check_task_resume(&ginfo->rtinfo, ginfo->pevent, | 184 | match = rt_graph_check_task_resume(&ginfo->rtinfo, ginfo->pevent, |
261 | record, &pid, &ts); | 185 | record, &pid, &ts); |
262 | if (match && pid == rtt_info->base.pid) { | 186 | if (match && pid == rtt_info->pid) { |
263 | info->box = TRUE; | 187 | /* info->box = TRUE; */ |
264 | info->bcolor = 0x0; | 188 | /* info->bcolor = 0x0; */ |
265 | info->bfill = TRUE; | 189 | /* info->bfill = TRUE; */ |
266 | info->bthin = TRUE; | 190 | /* info->bthin = TRUE; */ |
267 | info->bstart = rtt_info->block_time; | 191 | /* info->bstart = rtt_info->block_time; */ |
268 | info->bend = ts; | 192 | /* info->bend = ts; */ |
269 | 193 | ||
270 | rtt_info->block_time = 0ULL; | 194 | rtt_info->block_time = 0ULL; |
271 | 195 | ||
@@ -283,12 +207,12 @@ try_switch_away(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
283 | 207 | ||
284 | match = rt_graph_check_switch_away(&ginfo->rtinfo, ginfo->pevent, | 208 | match = rt_graph_check_switch_away(&ginfo->rtinfo, ginfo->pevent, |
285 | record, &pid, &job, &ts); | 209 | record, &pid, &job, &ts); |
286 | if (match && pid == rtt_info->base.pid) { | 210 | if (match && pid == rtt_info->pid) { |
287 | update_job(rtt_info, job); | 211 | update_job(rtt_info, job); |
288 | 212 | ||
289 | if (rtt_info->run_time && rtt_info->run_time < ts) { | 213 | if (rtt_info->run_time && rtt_info->run_time < ts) { |
290 | dprintf(3, "Box for %d:%d, %llu to %llu on CPU %d\n", | 214 | dprintf(3, "Box for %d:%d, %llu to %llu on CPU %d\n", |
291 | rtt_info->base.pid, rtt_info->last_job, | 215 | rtt_info->pid, rtt_info->last_job, |
292 | rtt_info->run_time, ts, rtt_info->last_cpu); | 216 | rtt_info->run_time, ts, rtt_info->last_cpu); |
293 | info->box = TRUE; | 217 | info->box = TRUE; |
294 | info->bcolor = hash_cpu(rtt_info->last_cpu); | 218 | info->bcolor = hash_cpu(rtt_info->last_cpu); |
@@ -298,6 +222,7 @@ try_switch_away(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
298 | info->blabel = rtt_info->label; | 222 | info->blabel = rtt_info->label; |
299 | } | 223 | } |
300 | 224 | ||
225 | dprintf(3, "Switch away at %llu\n", ts); | ||
301 | rtt_info->run_time = 0ULL; | 226 | rtt_info->run_time = 0ULL; |
302 | rtt_info->last_cpu = -1; | 227 | rtt_info->last_cpu = -1; |
303 | 228 | ||
@@ -314,14 +239,14 @@ static int try_switch_to(struct graph_info *ginfo, struct rt_task_info *rtt_info | |||
314 | 239 | ||
315 | match = rt_graph_check_switch_to(&ginfo->rtinfo, ginfo->pevent, | 240 | match = rt_graph_check_switch_to(&ginfo->rtinfo, ginfo->pevent, |
316 | record, &pid, &job, &ts); | 241 | record, &pid, &job, &ts); |
317 | if (match && pid == rtt_info->base.pid) { | 242 | if (match && pid == rtt_info->pid) { |
318 | update_job(rtt_info, job); | 243 | update_job(rtt_info, job); |
319 | 244 | ||
320 | rtt_info->run_time = ts; | 245 | rtt_info->run_time = ts; |
321 | rtt_info->last_cpu = record->cpu; | 246 | rtt_info->last_cpu = record->cpu; |
322 | 247 | ||
323 | dprintf(3, "Switching to %d:%d at %llu on CPU %d\n", | 248 | dprintf(3, "Switching to %d:%d at %llu on CPU %d\n", |
324 | rtt_info->base.pid, rtt_info->last_job, | 249 | rtt_info->pid, rtt_info->last_job, |
325 | ts, rtt_info->last_cpu); | 250 | ts, rtt_info->last_cpu); |
326 | 251 | ||
327 | ret = 1; | 252 | ret = 1; |
@@ -332,14 +257,17 @@ static int try_switch_to(struct graph_info *ginfo, struct rt_task_info *rtt_info | |||
332 | static int try_other(struct graph_info *ginfo, struct rt_task_info *rtt_info, | 257 | static int try_other(struct graph_info *ginfo, struct rt_task_info *rtt_info, |
333 | struct record *record, struct plot_info *info) | 258 | struct record *record, struct plot_info *info) |
334 | { | 259 | { |
335 | int pid, epid, ret = 0; | 260 | int pid, eid, epid, my_pid, my_cpu, not_sa, ret = 0; |
336 | unsigned long long ts; | 261 | unsigned long long ts; |
337 | struct task_plot_info *task_info = &rtt_info->base; | ||
338 | 262 | ||
339 | pid = task_info->pid; | 263 | pid = rtt_info->pid; |
340 | rt_graph_check_any(&ginfo->rtinfo, ginfo->pevent, record, &epid, &ts); | 264 | rt_graph_check_any(&ginfo->rtinfo, ginfo->pevent, record, |
265 | &epid, &eid, &ts); | ||
341 | 266 | ||
342 | if (pid == epid || record->cpu == rtt_info->last_cpu) { | 267 | my_pid = (pid == epid); |
268 | my_cpu = (rtt_info->run_time && record->cpu == rtt_info->last_cpu); | ||
269 | not_sa = (eid != ginfo->rtinfo.switch_away_id); | ||
270 | if (not_sa && (my_pid || my_cpu)) { | ||
343 | info->line = TRUE; | 271 | info->line = TRUE; |
344 | info->lcolor = hash_pid(record->cpu); | 272 | info->lcolor = hash_pid(record->cpu); |
345 | info->ltime = ts; | 273 | info->ltime = ts; |
@@ -385,7 +313,7 @@ get_previous_release(struct graph_info *ginfo, struct rt_task_info *rtt_info, | |||
385 | &release, &deadline); | 313 | &release, &deadline); |
386 | free_record(last_record); | 314 | free_record(last_record); |
387 | last_record = record; | 315 | last_record = record; |
388 | if (match && (pid == rtt_info->base.pid) && release <= time) { | 316 | if (match && (pid == rtt_info->pid) && release <= time) { |
389 | ret = record; | 317 | ret = record; |
390 | last_record = NULL; | 318 | last_record = NULL; |
391 | *out_job = job; | 319 | *out_job = job; |
@@ -422,7 +350,7 @@ static int get_time_info(struct graph_info *ginfo, | |||
422 | 350 | ||
423 | /* Seek CPUs to first record after this time */ | 351 | /* Seek CPUs to first record after this time */ |
424 | *out_job = *out_release = *out_deadline = 0; | 352 | *out_job = *out_release = *out_deadline = 0; |
425 | *out_record = find_record(ginfo, rtt_info->base.pid, time); | 353 | *out_record = find_record(ginfo, rtt_info->pid, time); |
426 | if (!*out_record) | 354 | if (!*out_record) |
427 | return 0; | 355 | return 0; |
428 | 356 | ||
@@ -466,20 +394,16 @@ static int rt_task_plot_event(struct graph_info *ginfo, struct graph_plot *plot, | |||
466 | struct record *record, struct plot_info *info) | 394 | struct record *record, struct plot_info *info) |
467 | { | 395 | { |
468 | struct rt_task_info *rtt_info = plot->private; | 396 | struct rt_task_info *rtt_info = plot->private; |
469 | struct task_plot_info *task_info = &rtt_info->base; | 397 | int match; |
470 | int match, cpu; | ||
471 | 398 | ||
472 | /* No more records, finish what we started */ | 399 | /* No more records, finish what we started */ |
473 | if (!record) { | 400 | if (!record) { |
474 | if (task_info->last_cpu >= 0) { | 401 | if (rtt_info->last_cpu >= 0 && |
402 | rtt_info->run_time) { | ||
475 | info->box = TRUE; | 403 | info->box = TRUE; |
476 | info->bstart = task_info->last_time; | 404 | info->bstart = rtt_info->last_time; |
477 | info->bend = ginfo->view_end_time; | 405 | info->bend = ginfo->view_end_time; |
478 | info->bcolor = hash_cpu(task_info->last_cpu); | 406 | info->bcolor = hash_cpu(rtt_info->last_cpu); |
479 | } | ||
480 | for (cpu = 0; cpu < ginfo->cpus; cpu++) { | ||
481 | free_record(task_info->last_records[cpu]); | ||
482 | task_info->last_records[cpu] = NULL; | ||
483 | } | 407 | } |
484 | return 0; | 408 | return 0; |
485 | } | 409 | } |
@@ -493,25 +417,12 @@ static int rt_task_plot_event(struct graph_info *ginfo, struct graph_plot *plot, | |||
493 | try_resume(ginfo, rtt_info, record, info) || | 417 | try_resume(ginfo, rtt_info, record, info) || |
494 | try_other(ginfo, rtt_info, record, info); | 418 | try_other(ginfo, rtt_info, record, info); |
495 | 419 | ||
496 | /* This record is neither on our CPU nor related to us, useless */ | 420 | if (match) { |
497 | if (!match && record->cpu != task_info->last_cpu) { | 421 | rtt_info->last_time = get_rts(ginfo, record); |
498 | if (!task_info->last_records[record->cpu]) { | 422 | rtt_info->last_cpu = record->cpu; |
499 | task_info->last_records[record->cpu] = record; | ||
500 | tracecmd_record_ref(record); | ||
501 | } | ||
502 | return 0; | ||
503 | } | ||
504 | |||
505 | if (!match) { | ||
506 | cpu = record->cpu; | ||
507 | /* We need some record, use this if none exist */ | ||
508 | if (!task_info->last_records[cpu]) { | ||
509 | free_record(task_info->last_records[cpu]); | ||
510 | task_info->last_records[cpu] = record; | ||
511 | } | ||
512 | } | 423 | } |
513 | 424 | ||
514 | return 1; | 425 | return match; |
515 | } | 426 | } |
516 | 427 | ||
517 | static void rt_task_plot_start(struct graph_info *ginfo, struct graph_plot *plot, | 428 | static void rt_task_plot_start(struct graph_info *ginfo, struct graph_plot *plot, |
@@ -520,8 +431,6 @@ static void rt_task_plot_start(struct graph_info *ginfo, struct graph_plot *plot | |||
520 | int i; | 431 | int i; |
521 | struct rt_task_info *rtt_info = plot->private; | 432 | struct rt_task_info *rtt_info = plot->private; |
522 | 433 | ||
523 | task_plot_start(ginfo, plot, time); | ||
524 | |||
525 | rtt_info->wcet = 0ULL; | 434 | rtt_info->wcet = 0ULL; |
526 | rtt_info->period = 0ULL; | 435 | rtt_info->period = 0ULL; |
527 | rtt_info->run_time = 0ULL; | 436 | rtt_info->run_time = 0ULL; |
@@ -537,7 +446,7 @@ static void rt_task_plot_start(struct graph_info *ginfo, struct graph_plot *plot | |||
537 | static void rt_task_plot_destroy(struct graph_info *ginfo, struct graph_plot *plot) | 446 | static void rt_task_plot_destroy(struct graph_info *ginfo, struct graph_plot *plot) |
538 | { | 447 | { |
539 | struct rt_task_info *rtt_info = plot->private; | 448 | struct rt_task_info *rtt_info = plot->private; |
540 | printf("Destroying plot %d\n", rtt_info->base.pid); | 449 | printf("Destroying plot %d\n", rtt_info->pid); |
541 | free(rtt_info->label); | 450 | free(rtt_info->label); |
542 | task_plot_destroy(ginfo, plot); | 451 | task_plot_destroy(ginfo, plot); |
543 | } | 452 | } |
@@ -554,7 +463,7 @@ static int rt_task_plot_display_last_event(struct graph_info *ginfo, | |||
554 | struct rt_task_info *rtt_info = plot->private; | 463 | struct rt_task_info *rtt_info = plot->private; |
555 | 464 | ||
556 | offsets = save_offsets(ginfo); | 465 | offsets = save_offsets(ginfo); |
557 | record = find_record(ginfo, rtt_info->base.pid, time); | 466 | record = find_display_record(ginfo, rtt_info->pid, time); |
558 | restore_offsets(ginfo, offsets); | 467 | restore_offsets(ginfo, offsets); |
559 | if (!record) | 468 | if (!record) |
560 | return 0; | 469 | return 0; |
@@ -592,14 +501,14 @@ static int rt_task_plot_display_info(struct graph_info *ginfo, | |||
592 | restore_offsets(ginfo, offsets); | 501 | restore_offsets(ginfo, offsets); |
593 | 502 | ||
594 | /* Show real-time data about time */ | 503 | /* Show real-time data about time */ |
595 | pid = rtt_info->base.pid; | 504 | pid = rtt_info->pid; |
596 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); | 505 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); |
597 | trace_seq_printf(s, "%s - %d:%d\n", comm, pid, job); | 506 | trace_seq_printf(s, "%s - %d:%d\n", comm, pid, job); |
598 | 507 | ||
599 | if (record) { | 508 | if (record) { |
600 | rts = get_rts(ginfo, record); | 509 | rts = get_rts(ginfo, record); |
510 | eid = pevent_data_type(ginfo->pevent, record); | ||
601 | if (in_res(ginfo, rts, time)) { | 511 | if (in_res(ginfo, rts, time)) { |
602 | eid = pevent_data_type(ginfo->pevent, record); | ||
603 | event = pevent_data_event_from_type(ginfo->pevent, eid); | 512 | event = pevent_data_event_from_type(ginfo->pevent, eid); |
604 | if (event) { | 513 | if (event) { |
605 | trace_seq_puts(s, event->name); | 514 | trace_seq_puts(s, event->name); |
@@ -626,14 +535,14 @@ static int rt_task_plot_match_time(struct graph_info *ginfo, | |||
626 | struct rt_task_info *rtt_info = plot->private; | 535 | struct rt_task_info *rtt_info = plot->private; |
627 | int next_cpu, match, ret; | 536 | int next_cpu, match, ret; |
628 | 537 | ||
629 | set_cpus_to_time(ginfo, time); | 538 | set_cpus_to_rts(ginfo, time); |
630 | 539 | ||
631 | do { | 540 | do { |
632 | free_record(record); | 541 | free_record(record); |
633 | record = tracecmd_read_next_data(ginfo->handle, &next_cpu); | 542 | record = tracecmd_read_next_data(ginfo->handle, &next_cpu); |
634 | if (!record) | 543 | if (!record) |
635 | return 0; | 544 | return 0; |
636 | match = record_matches_pid(ginfo, record, rtt_info->base.pid); | 545 | match = record_matches_pid(ginfo, record, rtt_info->pid); |
637 | } while ((!match && get_rts(ginfo, record) < time + 1) || | 546 | } while ((!match && get_rts(ginfo, record) < time + 1) || |
638 | (match && get_rts(ginfo, record) < time)); | 547 | (match && get_rts(ginfo, record) < time)); |
639 | 548 | ||
@@ -649,14 +558,13 @@ rt_task_plot_find_record(struct graph_info *ginfo, struct graph_plot *plot, | |||
649 | unsigned long long time) | 558 | unsigned long long time) |
650 | { | 559 | { |
651 | struct rt_task_info *rtt_info = plot->private; | 560 | struct rt_task_info *rtt_info = plot->private; |
652 | return find_record(ginfo, rtt_info->base.pid, time); | 561 | return find_record(ginfo, rtt_info->pid, time); |
653 | } | 562 | } |
654 | 563 | ||
655 | 564 | ||
656 | static const struct plot_callbacks rt_task_cb = { | 565 | static const struct plot_callbacks rt_task_cb = { |
657 | .start = rt_task_plot_start, | 566 | .start = rt_task_plot_start, |
658 | .destroy = rt_task_plot_destroy, | 567 | .destroy = rt_task_plot_destroy, |
659 | |||
660 | .plot_event = rt_task_plot_event, | 568 | .plot_event = rt_task_plot_event, |
661 | .display_last_event = rt_task_plot_display_last_event, | 569 | .display_last_event = rt_task_plot_display_last_event, |
662 | .display_info = rt_task_plot_display_info, | 570 | .display_info = rt_task_plot_display_info, |
@@ -669,13 +577,25 @@ void rt_plot_task_update_callback(gboolean accept, | |||
669 | gint *non_select, | 577 | gint *non_select, |
670 | gpointer data) | 578 | gpointer data) |
671 | { | 579 | { |
672 | graph_tasks_update_callback(TASK_PLOT_RT, rt_plot_task, | 580 | graph_tasks_update_callback(PLOT_TYPE_RT_TASK, rt_plot_task, |
673 | accept, selected, non_select, data); | 581 | accept, selected, non_select, data); |
674 | } | 582 | } |
675 | 583 | ||
676 | void rt_plot_task_plotted(struct graph_info *ginfo, gint **plotted) | 584 | void rt_plot_task_plotted(struct graph_info *ginfo, gint **plotted) |
677 | { | 585 | { |
678 | graph_tasks_plotted(ginfo, TASK_PLOT_RT, plotted); | 586 | struct task_plot_info *task_info; |
587 | struct graph_plot *plot; | ||
588 | int count = 0; | ||
589 | int i; | ||
590 | |||
591 | *plotted = NULL; | ||
592 | for (i = 0; i < ginfo->plots; i++) { | ||
593 | plot = ginfo->plot_array[i]; | ||
594 | if (plot->type != PLOT_TYPE_RT_TASK) | ||
595 | continue; | ||
596 | task_info = plot->private; | ||
597 | trace_array_add(plotted, &count, task_info->pid); | ||
598 | } | ||
679 | } | 599 | } |
680 | 600 | ||
681 | void rt_plot_task(struct graph_info *ginfo, int pid, int pos) | 601 | void rt_plot_task(struct graph_info *ginfo, int pid, int pos) |
@@ -684,29 +604,27 @@ void rt_plot_task(struct graph_info *ginfo, int pid, int pos) | |||
684 | struct rt_task_info *rtt_info; | 604 | struct rt_task_info *rtt_info; |
685 | struct graph_plot *plot; | 605 | struct graph_plot *plot; |
686 | const char *comm; | 606 | const char *comm; |
687 | char *label; | 607 | char *plot_label; |
688 | int len; | 608 | int len; |
689 | 609 | ||
690 | if (!find_task_list(rtinfo->tasks, pid)) | 610 | if (!find_task_list(rtinfo->tasks, pid)) |
691 | die("Cannot create RT plot of non-RT task %d!\n", pid); | 611 | die("Cannot create RT plot of non-RT task %d!\n", pid); |
692 | 612 | ||
693 | rtt_info = malloc_or_die(sizeof(*rtt_info)); | 613 | rtt_info = malloc_or_die(sizeof(*rtt_info)); |
614 | rtt_info->pid = pid; | ||
694 | rtt_info->label = malloc_or_die(LLABEL); | 615 | rtt_info->label = malloc_or_die(LLABEL); |
695 | 616 | ||
696 | init_task_plot_info(ginfo, &rtt_info->base, TASK_PLOT_RT, pid); | 617 | /* Create plot */ |
697 | |||
698 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); | 618 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); |
699 | len = strlen(comm) + 100; | 619 | len = strlen(comm) + 100; |
700 | label = malloc_or_die(len); | 620 | plot_label = malloc_or_die(len); |
701 | snprintf(label, len, "*%s-%d", comm, pid); | 621 | snprintf(plot_label, len, "*%s-%d", comm, pid); |
702 | rtt_info->pid = pid; | 622 | plot = trace_graph_plot_insert(ginfo, pos, plot_label, PLOT_TYPE_RT_TASK, |
623 | &rt_task_cb, rtt_info); | ||
624 | free(plot_label); | ||
703 | 625 | ||
704 | printf("Created plot for %s-%d / %d %p\n", comm, pid, rtt_info->base.pid, | 626 | printf("Created plot for %s-%d / %d %p\n", comm, pid, rtt_info->pid, |
705 | rtt_info); | 627 | rtt_info); |
706 | 628 | ||
707 | plot = trace_graph_plot_insert(ginfo, pos, label, PLOT_TYPE_TASK, | ||
708 | &rt_task_cb, rtt_info); | ||
709 | free(label); | ||
710 | |||
711 | trace_graph_plot_add_all_recs(ginfo, plot); | 629 | trace_graph_plot_add_all_recs(ginfo, plot); |
712 | } | 630 | } |
diff --git a/rt-plot-task.h b/rt-plot-task.h index a66de39..d3464e9 100644 --- a/rt-plot-task.h +++ b/rt-plot-task.h | |||
@@ -4,17 +4,18 @@ | |||
4 | #include "trace-plot-task.h" | 4 | #include "trace-plot-task.h" |
5 | 5 | ||
6 | struct rt_task_info { | 6 | struct rt_task_info { |
7 | struct task_plot_info base; | ||
8 | |||
9 | int pid; | 7 | int pid; |
10 | unsigned long long wcet; | 8 | unsigned long long wcet; |
11 | unsigned long long period; | 9 | unsigned long long period; |
12 | 10 | ||
11 | /* For drawing squares */ | ||
13 | unsigned long long run_time; | 12 | unsigned long long run_time; |
14 | unsigned long long block_time; | 13 | unsigned long long block_time; |
15 | 14 | ||
15 | /* For managing state */ | ||
16 | int last_job; | 16 | int last_job; |
17 | int last_cpu; | 17 | int last_cpu; |
18 | unsigned long long last_time; | ||
18 | 19 | ||
19 | /* Used to get around bugs(ish) */ | 20 | /* Used to get around bugs(ish) */ |
20 | unsigned long long first_rels[3]; | 21 | unsigned long long first_rels[3]; |
diff --git a/trace-graph.h b/trace-graph.h index ee57be6..1cb77b9 100644 --- a/trace-graph.h +++ b/trace-graph.h | |||
@@ -41,6 +41,7 @@ enum graph_plot_type { | |||
41 | PLOT_TYPE_OTHER, | 41 | PLOT_TYPE_OTHER, |
42 | PLOT_TYPE_CPU, | 42 | PLOT_TYPE_CPU, |
43 | PLOT_TYPE_TASK, | 43 | PLOT_TYPE_TASK, |
44 | PLOT_TYPE_RT_TASK, | ||
44 | }; | 45 | }; |
45 | 46 | ||
46 | struct graph_plot; | 47 | struct graph_plot; |
diff --git a/trace-plot-task.c b/trace-plot-task.c index abcdd89..d4b608b 100644 --- a/trace-plot-task.c +++ b/trace-plot-task.c | |||
@@ -713,10 +713,16 @@ static const struct plot_callbacks task_plot_cb = { | |||
713 | .destroy = task_plot_destroy | 713 | .destroy = task_plot_destroy |
714 | }; | 714 | }; |
715 | 715 | ||
716 | 716 | /** | |
717 | void graph_tasks_plotted(struct graph_info *ginfo, | 717 | * graph_plot_task_plotted - return what tasks are plotted |
718 | enum task_plot_type type, | 718 | * @ginfo: the graph info structure |
719 | gint **plotted) | 719 | * @plotted: returns an allocated array of gints holding the pids. |
720 | * the last pid is -1, NULL, if none are. | ||
721 | * | ||
722 | * @plotted must be freed with free() after this is called. | ||
723 | */ | ||
724 | void graph_plot_task_plotted(struct graph_info *ginfo, | ||
725 | gint **plotted) | ||
720 | { | 726 | { |
721 | struct task_plot_info *task_info; | 727 | struct task_plot_info *task_info; |
722 | struct graph_plot *plot; | 728 | struct graph_plot *plot; |
@@ -733,21 +739,7 @@ void graph_tasks_plotted(struct graph_info *ginfo, | |||
733 | } | 739 | } |
734 | } | 740 | } |
735 | 741 | ||
736 | /** | 742 | void graph_tasks_update_callback(enum graph_plot_type type, |
737 | * graph_plot_task_plotted - return what tasks are plotted | ||
738 | * @ginfo: the graph info structure | ||
739 | * @plotted: returns an allocated array of gints holding the pids. | ||
740 | * the last pid is -1, NULL, if none are. | ||
741 | * | ||
742 | * @plotted must be freed with free() after this is called. | ||
743 | */ | ||
744 | void graph_plot_task_plotted(struct graph_info *ginfo, | ||
745 | gint **plotted) | ||
746 | { | ||
747 | graph_tasks_plotted(ginfo, TASK_PLOT_LINUX, plotted); | ||
748 | } | ||
749 | |||
750 | void graph_tasks_update_callback(enum task_plot_type type, | ||
751 | plot_task_cb plot_cb, | 743 | plot_task_cb plot_cb, |
752 | gboolean accept, | 744 | gboolean accept, |
753 | gint *selected, | 745 | gint *selected, |
@@ -778,12 +770,9 @@ void graph_tasks_update_callback(enum task_plot_type type, | |||
778 | */ | 770 | */ |
779 | for (i = ginfo->plots - 1; i >= 0; i--) { | 771 | for (i = ginfo->plots - 1; i >= 0; i--) { |
780 | plot = ginfo->plot_array[i]; | 772 | plot = ginfo->plot_array[i]; |
781 | if (plot->type != PLOT_TYPE_TASK) | 773 | if (plot->type != type) |
782 | continue; | 774 | continue; |
783 | |||
784 | task_info = plot->private; | 775 | task_info = plot->private; |
785 | if (task_info->type != type) | ||
786 | continue; | ||
787 | 776 | ||
788 | /* If non are selected, then remove all */ | 777 | /* If non are selected, then remove all */ |
789 | if (!select_size) { | 778 | if (!select_size) { |
@@ -819,7 +808,7 @@ void graph_plot_task_update_callback(gboolean accept, | |||
819 | gint *non_select, | 808 | gint *non_select, |
820 | gpointer data) | 809 | gpointer data) |
821 | { | 810 | { |
822 | graph_tasks_update_callback(TASK_PLOT_LINUX, graph_plot_task, | 811 | graph_tasks_update_callback(PLOT_TYPE_TASK, graph_plot_task, |
823 | accept, selected, non_select, data); | 812 | accept, selected, non_select, data); |
824 | } | 813 | } |
825 | 814 | ||
@@ -850,17 +839,6 @@ void graph_plot_init_tasks(struct graph_info *ginfo) | |||
850 | &task_plot_cb, task_info); | 839 | &task_plot_cb, task_info); |
851 | } | 840 | } |
852 | 841 | ||
853 | void init_task_plot_info(struct graph_info *ginfo, | ||
854 | struct task_plot_info *task_info, | ||
855 | enum task_plot_type type, | ||
856 | int pid) | ||
857 | { | ||
858 | task_info->last_records = | ||
859 | malloc_or_die(sizeof(struct record *) * ginfo->cpus); | ||
860 | task_info->pid = pid; | ||
861 | task_info->type = type; | ||
862 | } | ||
863 | |||
864 | void graph_plot_task(struct graph_info *ginfo, int pid, int pos) | 842 | void graph_plot_task(struct graph_info *ginfo, int pid, int pos) |
865 | { | 843 | { |
866 | struct task_plot_info *task_info; | 844 | struct task_plot_info *task_info; |
@@ -870,8 +848,9 @@ void graph_plot_task(struct graph_info *ginfo, int pid, int pos) | |||
870 | int len; | 848 | int len; |
871 | 849 | ||
872 | task_info = malloc_or_die(sizeof(*task_info)); | 850 | task_info = malloc_or_die(sizeof(*task_info)); |
873 | 851 | task_info->last_records = | |
874 | init_task_plot_info(ginfo, task_info, TASK_PLOT_LINUX, pid); | 852 | malloc_or_die(sizeof(struct record *) * ginfo->cpus); |
853 | task_info->pid = pid; | ||
875 | 854 | ||
876 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); | 855 | comm = pevent_data_comm_from_pid(ginfo->pevent, pid); |
877 | 856 | ||
diff --git a/trace-plot-task.h b/trace-plot-task.h index e050512..11cb4f8 100644 --- a/trace-plot-task.h +++ b/trace-plot-task.h | |||
@@ -8,12 +8,7 @@ | |||
8 | struct graph_info; | 8 | struct graph_info; |
9 | struct graph_plot; | 9 | struct graph_plot; |
10 | struct plot_info; | 10 | struct plot_info; |
11 | 11 | enum graph_plot_type; | |
12 | enum task_plot_type { | ||
13 | TASK_PLOT_OTHER, | ||
14 | TASK_PLOT_LINUX, | ||
15 | TASK_PLOT_RT | ||
16 | }; | ||
17 | 12 | ||
18 | /** | 13 | /** |
19 | * struct task_plot_info - information for plotting a single task | 14 | * struct task_plot_info - information for plotting a single task |
@@ -25,7 +20,6 @@ enum task_plot_type { | |||
25 | * @display_wake_time: as above, but reset under some circumstances | 20 | * @display_wake_time: as above, but reset under some circumstances |
26 | * @wake_color: | 21 | * @wake_color: |
27 | * @last_cpu: cpu task is currently running on | 22 | * @last_cpu: cpu task is currently running on |
28 | * @type: type of task plot | ||
29 | */ | 23 | */ |
30 | struct task_plot_info { | 24 | struct task_plot_info { |
31 | int pid; | 25 | int pid; |
@@ -36,7 +30,6 @@ struct task_plot_info { | |||
36 | unsigned long long display_wake_time; | 30 | unsigned long long display_wake_time; |
37 | int wake_color; | 31 | int wake_color; |
38 | int last_cpu; | 32 | int last_cpu; |
39 | enum task_plot_type type; | ||
40 | }; | 33 | }; |
41 | 34 | ||
42 | /* Querying records */ | 35 | /* Querying records */ |
@@ -95,15 +88,10 @@ void graph_plot_init_tasks(struct graph_info *ginfo); | |||
95 | 88 | ||
96 | /* Shared functionality for inheriting structs */ | 89 | /* Shared functionality for inheriting structs */ |
97 | typedef void (plot_task_cb)(struct graph_info *ginfo, int pid, int pos); | 90 | typedef void (plot_task_cb)(struct graph_info *ginfo, int pid, int pos); |
98 | void graph_tasks_update_callback(enum task_plot_type type, | 91 | void graph_tasks_update_callback(enum graph_plot_type type, |
99 | plot_task_cb plot_cb, | 92 | plot_task_cb plot_cb, |
100 | gboolean accept, | 93 | gboolean accept, |
101 | gint *selected, | 94 | gint *selected, |
102 | gint *non_select, | 95 | gint *non_select, |
103 | gpointer data); | 96 | gpointer data); |
104 | void init_task_plot_info(struct graph_info *ginfo, | ||
105 | struct task_plot_info *task_info, | ||
106 | enum task_plot_type type, int pid); | ||
107 | void graph_tasks_plotted(struct graph_info *ginfo, enum task_plot_type type, | ||
108 | gint **plotted); | ||
109 | #endif | 97 | #endif |