diff options
Diffstat (limited to 'tools/perf/util/map.c')
-rw-r--r-- | tools/perf/util/map.c | 49 |
1 files changed, 32 insertions, 17 deletions
diff --git a/tools/perf/util/map.c b/tools/perf/util/map.c index 15d6a6dd50c5..3a7eb6ec0eec 100644 --- a/tools/perf/util/map.c +++ b/tools/perf/util/map.c | |||
@@ -29,6 +29,7 @@ void map__init(struct map *self, enum map_type type, | |||
29 | self->unmap_ip = map__unmap_ip; | 29 | self->unmap_ip = map__unmap_ip; |
30 | RB_CLEAR_NODE(&self->rb_node); | 30 | RB_CLEAR_NODE(&self->rb_node); |
31 | self->groups = NULL; | 31 | self->groups = NULL; |
32 | self->referenced = false; | ||
32 | } | 33 | } |
33 | 34 | ||
34 | struct map *map__new(struct list_head *dsos__list, u64 start, u64 len, | 35 | struct map *map__new(struct list_head *dsos__list, u64 start, u64 len, |
@@ -387,6 +388,7 @@ int map_groups__fixup_overlappings(struct map_groups *self, struct map *map, | |||
387 | { | 388 | { |
388 | struct rb_root *root = &self->maps[map->type]; | 389 | struct rb_root *root = &self->maps[map->type]; |
389 | struct rb_node *next = rb_first(root); | 390 | struct rb_node *next = rb_first(root); |
391 | int err = 0; | ||
390 | 392 | ||
391 | while (next) { | 393 | while (next) { |
392 | struct map *pos = rb_entry(next, struct map, rb_node); | 394 | struct map *pos = rb_entry(next, struct map, rb_node); |
@@ -403,20 +405,16 @@ int map_groups__fixup_overlappings(struct map_groups *self, struct map *map, | |||
403 | 405 | ||
404 | rb_erase(&pos->rb_node, root); | 406 | rb_erase(&pos->rb_node, root); |
405 | /* | 407 | /* |
406 | * We may have references to this map, for instance in some | ||
407 | * hist_entry instances, so just move them to a separate | ||
408 | * list. | ||
409 | */ | ||
410 | list_add_tail(&pos->node, &self->removed_maps[map->type]); | ||
411 | /* | ||
412 | * Now check if we need to create new maps for areas not | 408 | * Now check if we need to create new maps for areas not |
413 | * overlapped by the new map: | 409 | * overlapped by the new map: |
414 | */ | 410 | */ |
415 | if (map->start > pos->start) { | 411 | if (map->start > pos->start) { |
416 | struct map *before = map__clone(pos); | 412 | struct map *before = map__clone(pos); |
417 | 413 | ||
418 | if (before == NULL) | 414 | if (before == NULL) { |
419 | return -ENOMEM; | 415 | err = -ENOMEM; |
416 | goto move_map; | ||
417 | } | ||
420 | 418 | ||
421 | before->end = map->start - 1; | 419 | before->end = map->start - 1; |
422 | map_groups__insert(self, before); | 420 | map_groups__insert(self, before); |
@@ -427,14 +425,27 @@ int map_groups__fixup_overlappings(struct map_groups *self, struct map *map, | |||
427 | if (map->end < pos->end) { | 425 | if (map->end < pos->end) { |
428 | struct map *after = map__clone(pos); | 426 | struct map *after = map__clone(pos); |
429 | 427 | ||
430 | if (after == NULL) | 428 | if (after == NULL) { |
431 | return -ENOMEM; | 429 | err = -ENOMEM; |
430 | goto move_map; | ||
431 | } | ||
432 | 432 | ||
433 | after->start = map->end + 1; | 433 | after->start = map->end + 1; |
434 | map_groups__insert(self, after); | 434 | map_groups__insert(self, after); |
435 | if (verbose >= 2) | 435 | if (verbose >= 2) |
436 | map__fprintf(after, fp); | 436 | map__fprintf(after, fp); |
437 | } | 437 | } |
438 | move_map: | ||
439 | /* | ||
440 | * If we have references, just move them to a separate list. | ||
441 | */ | ||
442 | if (pos->referenced) | ||
443 | list_add_tail(&pos->node, &self->removed_maps[map->type]); | ||
444 | else | ||
445 | map__delete(pos); | ||
446 | |||
447 | if (err) | ||
448 | return err; | ||
438 | } | 449 | } |
439 | 450 | ||
440 | return 0; | 451 | return 0; |
@@ -506,6 +517,11 @@ void maps__insert(struct rb_root *maps, struct map *map) | |||
506 | rb_insert_color(&map->rb_node, maps); | 517 | rb_insert_color(&map->rb_node, maps); |
507 | } | 518 | } |
508 | 519 | ||
520 | void maps__remove(struct rb_root *self, struct map *map) | ||
521 | { | ||
522 | rb_erase(&map->rb_node, self); | ||
523 | } | ||
524 | |||
509 | struct map *maps__find(struct rb_root *maps, u64 ip) | 525 | struct map *maps__find(struct rb_root *maps, u64 ip) |
510 | { | 526 | { |
511 | struct rb_node **p = &maps->rb_node; | 527 | struct rb_node **p = &maps->rb_node; |
@@ -551,13 +567,6 @@ static void dsos__delete(struct list_head *self) | |||
551 | 567 | ||
552 | void machine__exit(struct machine *self) | 568 | void machine__exit(struct machine *self) |
553 | { | 569 | { |
554 | struct kmap *kmap = map__kmap(self->vmlinux_maps[MAP__FUNCTION]); | ||
555 | |||
556 | if (kmap->ref_reloc_sym) { | ||
557 | free((char *)kmap->ref_reloc_sym->name); | ||
558 | free(kmap->ref_reloc_sym); | ||
559 | } | ||
560 | |||
561 | map_groups__exit(&self->kmaps); | 570 | map_groups__exit(&self->kmaps); |
562 | dsos__delete(&self->user_dsos); | 571 | dsos__delete(&self->user_dsos); |
563 | dsos__delete(&self->kernel_dsos); | 572 | dsos__delete(&self->kernel_dsos); |
@@ -565,6 +574,12 @@ void machine__exit(struct machine *self) | |||
565 | self->root_dir = NULL; | 574 | self->root_dir = NULL; |
566 | } | 575 | } |
567 | 576 | ||
577 | void machine__delete(struct machine *self) | ||
578 | { | ||
579 | machine__exit(self); | ||
580 | free(self); | ||
581 | } | ||
582 | |||
568 | struct machine *machines__add(struct rb_root *self, pid_t pid, | 583 | struct machine *machines__add(struct rb_root *self, pid_t pid, |
569 | const char *root_dir) | 584 | const char *root_dir) |
570 | { | 585 | { |