diff options
| author | Arnaldo Carvalho de Melo <acme@redhat.com> | 2009-12-11 11:50:36 -0500 |
|---|---|---|
| committer | Ingo Molnar <mingo@elte.hu> | 2009-12-12 01:42:09 -0500 |
| commit | 9958e1f0aee632c3665162c9c93cf8fde8006a94 (patch) | |
| tree | ffd81c34d3ca8044c3fe0d670dc1786113624bbb /tools/perf/util/thread.c | |
| parent | 58e9f94138c1d9c47f6a63632ca7a78fc6dcc15f (diff) | |
perf symbols: Rename kthreads to kmaps, using another abstraction for it
Using a struct thread instance just to hold the kernel space maps
(vmlinux + modules) is overkill and confuses people trying to
understand the perf symbols abstractions.
The kernel maps are really present in all threads, i.e. the kernel
is a library, not a separate thread.
So introduce the 'map_groups' abstraction and use it for the kernel
maps, now in the kmaps global variable.
It, in turn, will move, together with the threads list to the
perf_file abstraction, so that we can support multiple perf_file
instances, needed by perf diff.
Brainstormed-with: Eduardo Habkost <ehabkost@redhat.com>
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Eduardo Habkost <ehabkost@redhat.com>
Cc: Frédéric Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Paul Mackerras <paulus@samba.org>
LKML-Reference: <1260550239-5372-1-git-send-email-acme@infradead.org>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'tools/perf/util/thread.c')
| -rw-r--r-- | tools/perf/util/thread.c | 62 |
1 files changed, 35 insertions, 27 deletions
diff --git a/tools/perf/util/thread.c b/tools/perf/util/thread.c index 603f5610861..a1285129c83 100644 --- a/tools/perf/util/thread.c +++ b/tools/perf/util/thread.c | |||
| @@ -9,11 +9,9 @@ | |||
| 9 | static struct rb_root threads; | 9 | static struct rb_root threads; |
| 10 | static struct thread *last_match; | 10 | static struct thread *last_match; |
| 11 | 11 | ||
| 12 | void thread__init(struct thread *self, pid_t pid) | 12 | void map_groups__init(struct map_groups *self) |
| 13 | { | 13 | { |
| 14 | int i; | 14 | int i; |
| 15 | self->pid = pid; | ||
| 16 | self->comm = NULL; | ||
| 17 | for (i = 0; i < MAP__NR_TYPES; ++i) { | 15 | for (i = 0; i < MAP__NR_TYPES; ++i) { |
| 18 | self->maps[i] = RB_ROOT; | 16 | self->maps[i] = RB_ROOT; |
| 19 | INIT_LIST_HEAD(&self->removed_maps[i]); | 17 | INIT_LIST_HEAD(&self->removed_maps[i]); |
| @@ -25,7 +23,8 @@ static struct thread *thread__new(pid_t pid) | |||
| 25 | struct thread *self = zalloc(sizeof(*self)); | 23 | struct thread *self = zalloc(sizeof(*self)); |
| 26 | 24 | ||
| 27 | if (self != NULL) { | 25 | if (self != NULL) { |
| 28 | thread__init(self, pid); | 26 | map_groups__init(&self->mg); |
| 27 | self->pid = pid; | ||
| 29 | self->comm = malloc(32); | 28 | self->comm = malloc(32); |
| 30 | if (self->comm) | 29 | if (self->comm) |
| 31 | snprintf(self->comm, 32, ":%d", self->pid); | 30 | snprintf(self->comm, 32, ":%d", self->pid); |
| @@ -57,8 +56,8 @@ static const char *map_type__name[MAP__NR_TYPES] = { | |||
| 57 | [MAP__FUNCTION] = "Functions", | 56 | [MAP__FUNCTION] = "Functions", |
| 58 | }; | 57 | }; |
| 59 | 58 | ||
| 60 | static size_t __thread__fprintf_maps(struct thread *self, | 59 | static size_t __map_groups__fprintf_maps(struct map_groups *self, |
| 61 | enum map_type type, FILE *fp) | 60 | enum map_type type, FILE *fp) |
| 62 | { | 61 | { |
| 63 | size_t printed = fprintf(fp, "%s:\n", map_type__name[type]); | 62 | size_t printed = fprintf(fp, "%s:\n", map_type__name[type]); |
| 64 | struct rb_node *nd; | 63 | struct rb_node *nd; |
| @@ -76,16 +75,16 @@ static size_t __thread__fprintf_maps(struct thread *self, | |||
| 76 | return printed; | 75 | return printed; |
| 77 | } | 76 | } |
| 78 | 77 | ||
| 79 | size_t thread__fprintf_maps(struct thread *self, FILE *fp) | 78 | size_t map_groups__fprintf_maps(struct map_groups *self, FILE *fp) |
| 80 | { | 79 | { |
| 81 | size_t printed = 0, i; | 80 | size_t printed = 0, i; |
| 82 | for (i = 0; i < MAP__NR_TYPES; ++i) | 81 | for (i = 0; i < MAP__NR_TYPES; ++i) |
| 83 | printed += __thread__fprintf_maps(self, i, fp); | 82 | printed += __map_groups__fprintf_maps(self, i, fp); |
| 84 | return printed; | 83 | return printed; |
| 85 | } | 84 | } |
| 86 | 85 | ||
| 87 | static size_t __thread__fprintf_removed_maps(struct thread *self, | 86 | static size_t __map_groups__fprintf_removed_maps(struct map_groups *self, |
| 88 | enum map_type type, FILE *fp) | 87 | enum map_type type, FILE *fp) |
| 89 | { | 88 | { |
| 90 | struct map *pos; | 89 | struct map *pos; |
| 91 | size_t printed = 0; | 90 | size_t printed = 0; |
| @@ -101,20 +100,25 @@ static size_t __thread__fprintf_removed_maps(struct thread *self, | |||
| 101 | return printed; | 100 | return printed; |
| 102 | } | 101 | } |
| 103 | 102 | ||
| 104 | static size_t thread__fprintf_removed_maps(struct thread *self, FILE *fp) | 103 | static size_t map_groups__fprintf_removed_maps(struct map_groups *self, FILE *fp) |
| 105 | { | 104 | { |
| 106 | size_t printed = 0, i; | 105 | size_t printed = 0, i; |
| 107 | for (i = 0; i < MAP__NR_TYPES; ++i) | 106 | for (i = 0; i < MAP__NR_TYPES; ++i) |
| 108 | printed += __thread__fprintf_removed_maps(self, i, fp); | 107 | printed += __map_groups__fprintf_removed_maps(self, i, fp); |
| 109 | return printed; | 108 | return printed; |
| 110 | } | 109 | } |
| 111 | 110 | ||
| 112 | static size_t thread__fprintf(struct thread *self, FILE *fp) | 111 | static size_t map_groups__fprintf(struct map_groups *self, FILE *fp) |
| 113 | { | 112 | { |
| 114 | size_t printed = fprintf(fp, "Thread %d %s\n", self->pid, self->comm); | 113 | size_t printed = map_groups__fprintf_maps(self, fp); |
| 115 | printed += thread__fprintf_removed_maps(self, fp); | ||
| 116 | printed += fprintf(fp, "Removed maps:\n"); | 114 | printed += fprintf(fp, "Removed maps:\n"); |
| 117 | return printed + thread__fprintf_removed_maps(self, fp); | 115 | return printed + map_groups__fprintf_removed_maps(self, fp); |
| 116 | } | ||
| 117 | |||
| 118 | static size_t thread__fprintf(struct thread *self, FILE *fp) | ||
| 119 | { | ||
| 120 | return fprintf(fp, "Thread %d %s\n", self->pid, self->comm) + | ||
| 121 | map_groups__fprintf(&self->mg, fp); | ||
| 118 | } | 122 | } |
| 119 | 123 | ||
| 120 | struct thread *threads__findnew(pid_t pid) | 124 | struct thread *threads__findnew(pid_t pid) |
| @@ -168,7 +172,8 @@ struct thread *register_idle_thread(void) | |||
| 168 | return thread; | 172 | return thread; |
| 169 | } | 173 | } |
| 170 | 174 | ||
| 171 | static void thread__remove_overlappings(struct thread *self, struct map *map) | 175 | static void map_groups__remove_overlappings(struct map_groups *self, |
| 176 | struct map *map) | ||
| 172 | { | 177 | { |
| 173 | struct rb_root *root = &self->maps[map->type]; | 178 | struct rb_root *root = &self->maps[map->type]; |
| 174 | struct rb_node *next = rb_first(root); | 179 | struct rb_node *next = rb_first(root); |
| @@ -238,12 +243,15 @@ struct map *maps__find(struct rb_root *maps, u64 ip) | |||
| 238 | 243 | ||
| 239 | void thread__insert_map(struct thread *self, struct map *map) | 244 | void thread__insert_map(struct thread *self, struct map *map) |
| 240 | { | 245 | { |
| 241 | thread__remove_overlappings(self, map); | 246 | map_groups__remove_overlappings(&self->mg, map); |
| 242 | maps__insert(&self->maps[map->type], map); | 247 | map_groups__insert(&self->mg, map); |
| 243 | } | 248 | } |
| 244 | 249 | ||
| 245 | static int thread__clone_maps(struct thread *self, struct thread *parent, | 250 | /* |
| 246 | enum map_type type) | 251 | * XXX This should not really _copy_ te maps, but refcount them. |
| 252 | */ | ||
| 253 | static int map_groups__clone(struct map_groups *self, | ||
| 254 | struct map_groups *parent, enum map_type type) | ||
| 247 | { | 255 | { |
| 248 | struct rb_node *nd; | 256 | struct rb_node *nd; |
| 249 | for (nd = rb_first(&parent->maps[type]); nd; nd = rb_next(nd)) { | 257 | for (nd = rb_first(&parent->maps[type]); nd; nd = rb_next(nd)) { |
| @@ -251,7 +259,7 @@ static int thread__clone_maps(struct thread *self, struct thread *parent, | |||
| 251 | struct map *new = map__clone(map); | 259 | struct map *new = map__clone(map); |
| 252 | if (new == NULL) | 260 | if (new == NULL) |
| 253 | return -ENOMEM; | 261 | return -ENOMEM; |
| 254 | thread__insert_map(self, new); | 262 | map_groups__insert(self, new); |
| 255 | } | 263 | } |
| 256 | return 0; | 264 | return 0; |
| 257 | } | 265 | } |
| @@ -267,7 +275,7 @@ int thread__fork(struct thread *self, struct thread *parent) | |||
| 267 | return -ENOMEM; | 275 | return -ENOMEM; |
| 268 | 276 | ||
| 269 | for (i = 0; i < MAP__NR_TYPES; ++i) | 277 | for (i = 0; i < MAP__NR_TYPES; ++i) |
| 270 | if (thread__clone_maps(self, parent, i) < 0) | 278 | if (map_groups__clone(&self->mg, &parent->mg, i) < 0) |
| 271 | return -ENOMEM; | 279 | return -ENOMEM; |
| 272 | return 0; | 280 | return 0; |
| 273 | } | 281 | } |
| @@ -286,11 +294,11 @@ size_t threads__fprintf(FILE *fp) | |||
| 286 | return ret; | 294 | return ret; |
| 287 | } | 295 | } |
| 288 | 296 | ||
| 289 | struct symbol *thread__find_symbol(struct thread *self, | 297 | struct symbol *map_groups__find_symbol(struct map_groups *self, |
| 290 | enum map_type type, u64 addr, | 298 | enum map_type type, u64 addr, |
| 291 | symbol_filter_t filter) | 299 | symbol_filter_t filter) |
| 292 | { | 300 | { |
| 293 | struct map *map = thread__find_map(self, type, addr); | 301 | struct map *map = map_groups__find(self, type, addr); |
| 294 | 302 | ||
| 295 | if (map != NULL) | 303 | if (map != NULL) |
| 296 | return map__find_symbol(map, map->map_ip(map, addr), filter); | 304 | return map__find_symbol(map, map->map_ip(map, addr), filter); |
