diff options
| author | Linus Torvalds <torvalds@linux-foundation.org> | 2009-09-11 16:22:43 -0400 |
|---|---|---|
| committer | Linus Torvalds <torvalds@linux-foundation.org> | 2009-09-11 16:22:43 -0400 |
| commit | 4f0ac854167846bd55cd81dbc9a36e03708aa01c (patch) | |
| tree | 0eb34d18a667f8e68ad9255f791560b028fed2a6 /tools/perf/util/thread.c | |
| parent | b9356c53ba2f593081e5aa45eb67adcce243d1c0 (diff) | |
| parent | 6b58e7f146f8d79c08f62087f928e1f01747de71 (diff) | |
Merge branch 'perfcounters-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip
* 'perfcounters-core-for-linus' of git://git.kernel.org/pub/scm/linux/kernel/git/tip/linux-2.6-tip: (60 commits)
perf tools: Avoid unnecessary work in directory lookups
perf stat: Clean up statistics calculations a bit more
perf stat: More advanced variance computation
perf stat: Use stddev_mean in stead of stddev
perf stat: Remove the limit on repeat
perf stat: Change noise calculation to use stddev
x86, perf_counter, bts: Do not allow kernel BTS tracing for now
x86, perf_counter, bts: Correct pointer-to-u64 casts
x86, perf_counter, bts: Fail if BTS is not available
perf_counter: Fix output-sharing error path
perf trace: Fix read_string()
perf trace: Print out in nanoseconds
perf tools: Seek to the end of the header area
perf trace: Fix parsing of perf.data
perf trace: Sample timestamps as well
perf_counter: Introduce new (non-)paranoia level to allow raw tracepoint access
perf trace: Sample the CPU too
perf tools: Work around strict aliasing related warnings
perf tools: Clean up warnings list in the Makefile
perf tools: Complete support for dynamic strings
...
Diffstat (limited to 'tools/perf/util/thread.c')
| -rw-r--r-- | tools/perf/util/thread.c | 175 |
1 files changed, 175 insertions, 0 deletions
diff --git a/tools/perf/util/thread.c b/tools/perf/util/thread.c new file mode 100644 index 000000000000..7635928ca278 --- /dev/null +++ b/tools/perf/util/thread.c | |||
| @@ -0,0 +1,175 @@ | |||
| 1 | #include "../perf.h" | ||
| 2 | #include <stdlib.h> | ||
| 3 | #include <stdio.h> | ||
| 4 | #include <string.h> | ||
| 5 | #include "thread.h" | ||
| 6 | #include "util.h" | ||
| 7 | #include "debug.h" | ||
| 8 | |||
| 9 | static struct thread *thread__new(pid_t pid) | ||
| 10 | { | ||
| 11 | struct thread *self = malloc(sizeof(*self)); | ||
| 12 | |||
| 13 | if (self != NULL) { | ||
| 14 | self->pid = pid; | ||
| 15 | self->comm = malloc(32); | ||
| 16 | if (self->comm) | ||
| 17 | snprintf(self->comm, 32, ":%d", self->pid); | ||
| 18 | INIT_LIST_HEAD(&self->maps); | ||
| 19 | } | ||
| 20 | |||
| 21 | return self; | ||
| 22 | } | ||
| 23 | |||
| 24 | int thread__set_comm(struct thread *self, const char *comm) | ||
| 25 | { | ||
| 26 | if (self->comm) | ||
| 27 | free(self->comm); | ||
| 28 | self->comm = strdup(comm); | ||
| 29 | return self->comm ? 0 : -ENOMEM; | ||
| 30 | } | ||
| 31 | |||
| 32 | static size_t thread__fprintf(struct thread *self, FILE *fp) | ||
| 33 | { | ||
| 34 | struct map *pos; | ||
| 35 | size_t ret = fprintf(fp, "Thread %d %s\n", self->pid, self->comm); | ||
| 36 | |||
| 37 | list_for_each_entry(pos, &self->maps, node) | ||
| 38 | ret += map__fprintf(pos, fp); | ||
| 39 | |||
| 40 | return ret; | ||
| 41 | } | ||
| 42 | |||
| 43 | struct thread * | ||
| 44 | threads__findnew(pid_t pid, struct rb_root *threads, struct thread **last_match) | ||
| 45 | { | ||
| 46 | struct rb_node **p = &threads->rb_node; | ||
| 47 | struct rb_node *parent = NULL; | ||
| 48 | struct thread *th; | ||
| 49 | |||
| 50 | /* | ||
| 51 | * Font-end cache - PID lookups come in blocks, | ||
| 52 | * so most of the time we dont have to look up | ||
| 53 | * the full rbtree: | ||
| 54 | */ | ||
| 55 | if (*last_match && (*last_match)->pid == pid) | ||
| 56 | return *last_match; | ||
| 57 | |||
| 58 | while (*p != NULL) { | ||
| 59 | parent = *p; | ||
| 60 | th = rb_entry(parent, struct thread, rb_node); | ||
| 61 | |||
| 62 | if (th->pid == pid) { | ||
| 63 | *last_match = th; | ||
| 64 | return th; | ||
| 65 | } | ||
| 66 | |||
| 67 | if (pid < th->pid) | ||
| 68 | p = &(*p)->rb_left; | ||
| 69 | else | ||
| 70 | p = &(*p)->rb_right; | ||
| 71 | } | ||
| 72 | |||
| 73 | th = thread__new(pid); | ||
| 74 | if (th != NULL) { | ||
| 75 | rb_link_node(&th->rb_node, parent, p); | ||
| 76 | rb_insert_color(&th->rb_node, threads); | ||
| 77 | *last_match = th; | ||
| 78 | } | ||
| 79 | |||
| 80 | return th; | ||
| 81 | } | ||
| 82 | |||
| 83 | struct thread * | ||
| 84 | register_idle_thread(struct rb_root *threads, struct thread **last_match) | ||
| 85 | { | ||
| 86 | struct thread *thread = threads__findnew(0, threads, last_match); | ||
| 87 | |||
| 88 | if (!thread || thread__set_comm(thread, "[init]")) { | ||
| 89 | fprintf(stderr, "problem inserting idle task.\n"); | ||
| 90 | exit(-1); | ||
| 91 | } | ||
| 92 | |||
| 93 | return thread; | ||
| 94 | } | ||
| 95 | |||
| 96 | void thread__insert_map(struct thread *self, struct map *map) | ||
| 97 | { | ||
| 98 | struct map *pos, *tmp; | ||
| 99 | |||
| 100 | list_for_each_entry_safe(pos, tmp, &self->maps, node) { | ||
| 101 | if (map__overlap(pos, map)) { | ||
| 102 | if (verbose >= 2) { | ||
| 103 | printf("overlapping maps:\n"); | ||
| 104 | map__fprintf(map, stdout); | ||
| 105 | map__fprintf(pos, stdout); | ||
| 106 | } | ||
| 107 | |||
| 108 | if (map->start <= pos->start && map->end > pos->start) | ||
| 109 | pos->start = map->end; | ||
| 110 | |||
| 111 | if (map->end >= pos->end && map->start < pos->end) | ||
| 112 | pos->end = map->start; | ||
| 113 | |||
| 114 | if (verbose >= 2) { | ||
| 115 | printf("after collision:\n"); | ||
| 116 | map__fprintf(pos, stdout); | ||
| 117 | } | ||
| 118 | |||
| 119 | if (pos->start >= pos->end) { | ||
| 120 | list_del_init(&pos->node); | ||
| 121 | free(pos); | ||
| 122 | } | ||
| 123 | } | ||
| 124 | } | ||
| 125 | |||
| 126 | list_add_tail(&map->node, &self->maps); | ||
| 127 | } | ||
| 128 | |||
| 129 | int thread__fork(struct thread *self, struct thread *parent) | ||
| 130 | { | ||
| 131 | struct map *map; | ||
| 132 | |||
| 133 | if (self->comm) | ||
| 134 | free(self->comm); | ||
| 135 | self->comm = strdup(parent->comm); | ||
| 136 | if (!self->comm) | ||
| 137 | return -ENOMEM; | ||
| 138 | |||
| 139 | list_for_each_entry(map, &parent->maps, node) { | ||
| 140 | struct map *new = map__clone(map); | ||
| 141 | if (!new) | ||
| 142 | return -ENOMEM; | ||
| 143 | thread__insert_map(self, new); | ||
| 144 | } | ||
| 145 | |||
| 146 | return 0; | ||
| 147 | } | ||
| 148 | |||
| 149 | struct map *thread__find_map(struct thread *self, u64 ip) | ||
| 150 | { | ||
| 151 | struct map *pos; | ||
| 152 | |||
| 153 | if (self == NULL) | ||
| 154 | return NULL; | ||
| 155 | |||
| 156 | list_for_each_entry(pos, &self->maps, node) | ||
| 157 | if (ip >= pos->start && ip <= pos->end) | ||
| 158 | return pos; | ||
| 159 | |||
| 160 | return NULL; | ||
| 161 | } | ||
| 162 | |||
| 163 | size_t threads__fprintf(FILE *fp, struct rb_root *threads) | ||
| 164 | { | ||
| 165 | size_t ret = 0; | ||
| 166 | struct rb_node *nd; | ||
| 167 | |||
| 168 | for (nd = rb_first(threads); nd; nd = rb_next(nd)) { | ||
| 169 | struct thread *pos = rb_entry(nd, struct thread, rb_node); | ||
| 170 | |||
| 171 | ret += thread__fprintf(pos, fp); | ||
| 172 | } | ||
| 173 | |||
| 174 | return ret; | ||
| 175 | } | ||
