map.c 15.6 KB
Newer Older
1
#include "symbol.h"
2
#include <errno.h>
3
#include <inttypes.h>
4
#include <limits.h>
5 6 7
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
8
#include <unistd.h>
9
#include "map.h"
10

11 12 13 14 15
const char *map_type__name[MAP__NR_TYPES] = {
	[MAP__FUNCTION] = "Functions",
	[MAP__VARIABLE] = "Variables",
};

16 17 18 19 20
static inline int is_anon_memory(const char *filename)
{
	return strcmp(filename, "//anon") == 0;
}

21 22 23 24 25 26 27
static inline int is_no_dso_memory(const char *filename)
{
	return !strcmp(filename, "[stack]") ||
	       !strcmp(filename, "[vdso]")  ||
	       !strcmp(filename, "[heap]");
}

28 29
void map__init(struct map *self, enum map_type type,
	       u64 start, u64 end, u64 pgoff, struct dso *dso)
30
{
31
	self->type     = type;
32 33 34 35 36 37 38
	self->start    = start;
	self->end      = end;
	self->pgoff    = pgoff;
	self->dso      = dso;
	self->map_ip   = map__map_ip;
	self->unmap_ip = map__unmap_ip;
	RB_CLEAR_NODE(&self->rb_node);
39
	self->groups   = NULL;
40
	self->referenced = false;
41
	self->erange_warned = false;
42 43
}

44 45
struct map *map__new(struct list_head *dsos__list, u64 start, u64 len,
		     u64 pgoff, u32 pid, char *filename,
46
		     enum map_type type)
47 48 49 50 51
{
	struct map *self = malloc(sizeof(*self));

	if (self != NULL) {
		char newfilename[PATH_MAX];
52
		struct dso *dso;
53
		int anon, no_dso;
54 55

		anon = is_anon_memory(filename);
56
		no_dso = is_no_dso_memory(filename);
57 58

		if (anon) {
59
			snprintf(newfilename, sizeof(newfilename), "/tmp/perf-%d.map", pid);
60 61 62
			filename = newfilename;
		}

63
		dso = __dsos__findnew(dsos__list, filename);
64
		if (dso == NULL)
65 66
			goto out_delete;

67
		map__init(self, type, start, start + len, pgoff, dso);
68

69
		if (anon || no_dso) {
70
			self->map_ip = self->unmap_ip = identity__map_ip;
71 72 73 74 75 76 77 78

			/*
			 * Set memory without DSO as loaded. All map__find_*
			 * functions still return NULL, and we avoid the
			 * unnecessary map__load warning.
			 */
			if (no_dso)
				dso__set_loaded(dso, self->type);
79
		}
80 81 82 83 84 85 86
	}
	return self;
out_delete:
	free(self);
	return NULL;
}

87 88 89 90 91
void map__delete(struct map *self)
{
	free(self);
}

92
void map__fixup_start(struct map *self)
93
{
94
	struct rb_root *symbols = &self->dso->symbols[self->type];
95
	struct rb_node *nd = rb_first(symbols);
96 97 98 99 100 101
	if (nd != NULL) {
		struct symbol *sym = rb_entry(nd, struct symbol, rb_node);
		self->start = sym->start;
	}
}

102
void map__fixup_end(struct map *self)
103
{
104
	struct rb_root *symbols = &self->dso->symbols[self->type];
105
	struct rb_node *nd = rb_last(symbols);
106 107 108 109 110 111
	if (nd != NULL) {
		struct symbol *sym = rb_entry(nd, struct symbol, rb_node);
		self->end = sym->end;
	}
}

112 113
#define DSO__DELETED "(deleted)"

114
int map__load(struct map *self, symbol_filter_t filter)
115
{
116
	const char *name = self->dso->long_name;
117
	int nr;
118

119 120 121
	if (dso__loaded(self->dso, self->type))
		return 0;

122
	nr = dso__load(self->dso, self, filter);
123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142
	if (nr < 0) {
		if (self->dso->has_build_id) {
			char sbuild_id[BUILD_ID_SIZE * 2 + 1];

			build_id__sprintf(self->dso->build_id,
					  sizeof(self->dso->build_id),
					  sbuild_id);
			pr_warning("%s with build id %s not found",
				   name, sbuild_id);
		} else
			pr_warning("Failed to open %s", name);

		pr_warning(", continuing without symbols\n");
		return -1;
	} else if (nr == 0) {
		const size_t len = strlen(name);
		const size_t real_len = len - sizeof(DSO__DELETED);

		if (len > sizeof(DSO__DELETED) &&
		    strcmp(name + real_len + 1, DSO__DELETED) == 0) {
143 144
			pr_warning("%.*s was updated (is prelink enabled?). "
				"Restart the long running apps that use it!\n",
145 146 147 148
				   (int)real_len, name);
		} else {
			pr_warning("no symbols found in %s, maybe install "
				   "a debug package?\n", name);
149
		}
150 151

		return -1;
152
	}
153 154 155 156 157 158
	/*
	 * Only applies to the kernel, as its symtabs aren't relative like the
	 * module ones.
	 */
	if (self->dso->kernel)
		map__reloc_vmlinux(self);
159

160 161 162
	return 0;
}

163 164
struct symbol *map__find_symbol(struct map *self, u64 addr,
				symbol_filter_t filter)
165
{
166
	if (map__load(self, filter) < 0)
167 168
		return NULL;

169
	return dso__find_symbol(self->dso, self->type, addr);
170 171
}

172 173 174
struct symbol *map__find_symbol_by_name(struct map *self, const char *name,
					symbol_filter_t filter)
{
175
	if (map__load(self, filter) < 0)
176 177 178 179 180 181 182 183
		return NULL;

	if (!dso__sorted_by_name(self->dso, self->type))
		dso__sort_by_name(self->dso, self->type);

	return dso__find_symbol_by_name(self->dso, self->type, name);
}

184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
struct map *map__clone(struct map *self)
{
	struct map *map = malloc(sizeof(*self));

	if (!map)
		return NULL;

	memcpy(map, self, sizeof(*self));

	return map;
}

int map__overlap(struct map *l, struct map *r)
{
	if (l->start > r->start) {
		struct map *t = l;
		l = r;
		r = t;
	}

	if (l->end > r->start)
		return 1;

	return 0;
}

size_t map__fprintf(struct map *self, FILE *fp)
{
212
	return fprintf(fp, " %" PRIx64 "-%" PRIx64 " %" PRIx64 " %s\n",
213 214
		       self->start, self->end, self->pgoff, self->dso->name);
}
215

216 217 218 219
size_t map__fprintf_dsoname(struct map *map, FILE *fp)
{
	const char *dsoname;

220 221 222 223 224 225
	if (map && map->dso && (map->dso->name || map->dso->long_name)) {
		if (symbol_conf.show_kernel_path && map->dso->long_name)
			dsoname = map->dso->long_name;
		else if (map->dso->name)
			dsoname = map->dso->name;
	} else
226 227 228 229 230
		dsoname = "[unknown]";

	return fprintf(fp, "%s", dsoname);
}

231 232 233 234 235 236 237 238 239 240 241
/*
 * objdump wants/reports absolute IPs for ET_EXEC, and RIPs for ET_DYN.
 * map->dso->adjust_symbols==1 for ET_EXEC-like cases.
 */
u64 map__rip_2objdump(struct map *map, u64 rip)
{
	u64 addr = map->dso->adjust_symbols ?
			map->unmap_ip(map, rip) :	/* RIP -> IP */
			rip;
	return addr;
}
242 243 244 245 246 247 248 249

u64 map__objdump_2ip(struct map *map, u64 addr)
{
	u64 ip = map->dso->adjust_symbols ?
			addr :
			map->unmap_ip(map, addr);	/* RIP -> IP */
	return ip;
}
250

251
void map_groups__init(struct map_groups *mg)
252 253 254
{
	int i;
	for (i = 0; i < MAP__NR_TYPES; ++i) {
255 256
		mg->maps[i] = RB_ROOT;
		INIT_LIST_HEAD(&mg->removed_maps[i]);
257
	}
258
	mg->machine = NULL;
259 260
}

261
static void maps__delete(struct rb_root *maps)
262
{
263
	struct rb_node *next = rb_first(maps);
264 265 266 267 268

	while (next) {
		struct map *pos = rb_entry(next, struct map, rb_node);

		next = rb_next(&pos->rb_node);
269
		rb_erase(&pos->rb_node, maps);
270 271 272 273
		map__delete(pos);
	}
}

274
static void maps__delete_removed(struct list_head *maps)
275 276 277
{
	struct map *pos, *n;

278
	list_for_each_entry_safe(pos, n, maps, node) {
279 280 281 282 283
		list_del(&pos->node);
		map__delete(pos);
	}
}

284
void map_groups__exit(struct map_groups *mg)
285 286 287 288
{
	int i;

	for (i = 0; i < MAP__NR_TYPES; ++i) {
289 290
		maps__delete(&mg->maps[i]);
		maps__delete_removed(&mg->removed_maps[i]);
291 292 293
	}
}

294
void map_groups__flush(struct map_groups *mg)
295 296 297 298
{
	int type;

	for (type = 0; type < MAP__NR_TYPES; type++) {
299
		struct rb_root *root = &mg->maps[type];
300 301 302 303 304 305 306 307 308 309 310
		struct rb_node *next = rb_first(root);

		while (next) {
			struct map *pos = rb_entry(next, struct map, rb_node);
			next = rb_next(&pos->rb_node);
			rb_erase(&pos->rb_node, root);
			/*
			 * We may have references to this map, for
			 * instance in some hist_entry instances, so
			 * just move them to a separate list.
			 */
311
			list_add_tail(&pos->node, &mg->removed_maps[pos->type]);
312 313 314 315
		}
	}
}

316
struct symbol *map_groups__find_symbol(struct map_groups *mg,
317
				       enum map_type type, u64 addr,
318
				       struct map **mapp,
319 320
				       symbol_filter_t filter)
{
321
	struct map *map = map_groups__find(mg, type, addr);
322

323 324 325
	if (map != NULL) {
		if (mapp != NULL)
			*mapp = map;
326
		return map__find_symbol(map, map->map_ip(map, addr), filter);
327 328 329 330 331
	}

	return NULL;
}

332
struct symbol *map_groups__find_symbol_by_name(struct map_groups *mg,
333 334 335 336 337 338 339
					       enum map_type type,
					       const char *name,
					       struct map **mapp,
					       symbol_filter_t filter)
{
	struct rb_node *nd;

340
	for (nd = rb_first(&mg->maps[type]); nd; nd = rb_next(nd)) {
341 342 343 344 345 346 347 348 349
		struct map *pos = rb_entry(nd, struct map, rb_node);
		struct symbol *sym = map__find_symbol_by_name(pos, name, filter);

		if (sym == NULL)
			continue;
		if (mapp != NULL)
			*mapp = pos;
		return sym;
	}
350 351 352 353

	return NULL;
}

354
size_t __map_groups__fprintf_maps(struct map_groups *mg,
355 356 357 358 359
				  enum map_type type, int verbose, FILE *fp)
{
	size_t printed = fprintf(fp, "%s:\n", map_type__name[type]);
	struct rb_node *nd;

360
	for (nd = rb_first(&mg->maps[type]); nd; nd = rb_next(nd)) {
361 362 363 364 365 366 367 368 369 370 371 372
		struct map *pos = rb_entry(nd, struct map, rb_node);
		printed += fprintf(fp, "Map:");
		printed += map__fprintf(pos, fp);
		if (verbose > 2) {
			printed += dso__fprintf(pos->dso, type, fp);
			printed += fprintf(fp, "--\n");
		}
	}

	return printed;
}

373
size_t map_groups__fprintf_maps(struct map_groups *mg, int verbose, FILE *fp)
374 375 376
{
	size_t printed = 0, i;
	for (i = 0; i < MAP__NR_TYPES; ++i)
377
		printed += __map_groups__fprintf_maps(mg, i, verbose, fp);
378 379 380
	return printed;
}

381
static size_t __map_groups__fprintf_removed_maps(struct map_groups *mg,
382 383 384 385 386 387
						 enum map_type type,
						 int verbose, FILE *fp)
{
	struct map *pos;
	size_t printed = 0;

388
	list_for_each_entry(pos, &mg->removed_maps[type], node) {
389 390 391 392 393 394 395 396 397 398
		printed += fprintf(fp, "Map:");
		printed += map__fprintf(pos, fp);
		if (verbose > 1) {
			printed += dso__fprintf(pos->dso, type, fp);
			printed += fprintf(fp, "--\n");
		}
	}
	return printed;
}

399
static size_t map_groups__fprintf_removed_maps(struct map_groups *mg,
400 401 402 403
					       int verbose, FILE *fp)
{
	size_t printed = 0, i;
	for (i = 0; i < MAP__NR_TYPES; ++i)
404
		printed += __map_groups__fprintf_removed_maps(mg, i, verbose, fp);
405 406 407
	return printed;
}

408
size_t map_groups__fprintf(struct map_groups *mg, int verbose, FILE *fp)
409
{
410
	size_t printed = map_groups__fprintf_maps(mg, verbose, fp);
411
	printed += fprintf(fp, "Removed maps:\n");
412
	return printed + map_groups__fprintf_removed_maps(mg, verbose, fp);
413 414
}

415
int map_groups__fixup_overlappings(struct map_groups *mg, struct map *map,
416 417
				   int verbose, FILE *fp)
{
418
	struct rb_root *root = &mg->maps[map->type];
419
	struct rb_node *next = rb_first(root);
420
	int err = 0;
421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442

	while (next) {
		struct map *pos = rb_entry(next, struct map, rb_node);
		next = rb_next(&pos->rb_node);

		if (!map__overlap(pos, map))
			continue;

		if (verbose >= 2) {
			fputs("overlapping maps:\n", fp);
			map__fprintf(map, fp);
			map__fprintf(pos, fp);
		}

		rb_erase(&pos->rb_node, root);
		/*
		 * Now check if we need to create new maps for areas not
		 * overlapped by the new map:
		 */
		if (map->start > pos->start) {
			struct map *before = map__clone(pos);

443 444 445 446
			if (before == NULL) {
				err = -ENOMEM;
				goto move_map;
			}
447 448

			before->end = map->start - 1;
449
			map_groups__insert(mg, before);
450 451 452 453 454 455 456
			if (verbose >= 2)
				map__fprintf(before, fp);
		}

		if (map->end < pos->end) {
			struct map *after = map__clone(pos);

457 458 459 460
			if (after == NULL) {
				err = -ENOMEM;
				goto move_map;
			}
461 462

			after->start = map->end + 1;
463
			map_groups__insert(mg, after);
464 465 466
			if (verbose >= 2)
				map__fprintf(after, fp);
		}
467 468 469 470 471
move_map:
		/*
		 * If we have references, just move them to a separate list.
		 */
		if (pos->referenced)
472
			list_add_tail(&pos->node, &mg->removed_maps[map->type]);
473 474 475 476 477
		else
			map__delete(pos);

		if (err)
			return err;
478 479 480 481 482 483 484 485
	}

	return 0;
}

/*
 * XXX This should not really _copy_ te maps, but refcount them.
 */
486
int map_groups__clone(struct map_groups *mg,
487 488 489 490 491 492 493 494
		      struct map_groups *parent, enum map_type type)
{
	struct rb_node *nd;
	for (nd = rb_first(&parent->maps[type]); nd; nd = rb_next(nd)) {
		struct map *map = rb_entry(nd, struct map, rb_node);
		struct map *new = map__clone(map);
		if (new == NULL)
			return -ENOMEM;
495
		map_groups__insert(mg, new);
496 497 498 499
	}
	return 0;
}

500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548
static u64 map__reloc_map_ip(struct map *map, u64 ip)
{
	return ip + (s64)map->pgoff;
}

static u64 map__reloc_unmap_ip(struct map *map, u64 ip)
{
	return ip - (s64)map->pgoff;
}

void map__reloc_vmlinux(struct map *self)
{
	struct kmap *kmap = map__kmap(self);
	s64 reloc;

	if (!kmap->ref_reloc_sym || !kmap->ref_reloc_sym->unrelocated_addr)
		return;

	reloc = (kmap->ref_reloc_sym->unrelocated_addr -
		 kmap->ref_reloc_sym->addr);

	if (!reloc)
		return;

	self->map_ip   = map__reloc_map_ip;
	self->unmap_ip = map__reloc_unmap_ip;
	self->pgoff    = reloc;
}

void maps__insert(struct rb_root *maps, struct map *map)
{
	struct rb_node **p = &maps->rb_node;
	struct rb_node *parent = NULL;
	const u64 ip = map->start;
	struct map *m;

	while (*p != NULL) {
		parent = *p;
		m = rb_entry(parent, struct map, rb_node);
		if (ip < m->start)
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

	rb_link_node(&map->rb_node, parent, p);
	rb_insert_color(&map->rb_node, maps);
}

549 550 551 552 553
void maps__remove(struct rb_root *self, struct map *map)
{
	rb_erase(&map->rb_node, self);
}

554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572
struct map *maps__find(struct rb_root *maps, u64 ip)
{
	struct rb_node **p = &maps->rb_node;
	struct rb_node *parent = NULL;
	struct map *m;

	while (*p != NULL) {
		parent = *p;
		m = rb_entry(parent, struct map, rb_node);
		if (ip < m->start)
			p = &(*p)->rb_left;
		else if (ip > m->end)
			p = &(*p)->rb_right;
		else
			return m;
	}

	return NULL;
}
573

574 575 576 577 578 579 580
int machine__init(struct machine *self, const char *root_dir, pid_t pid)
{
	map_groups__init(&self->kmaps);
	RB_CLEAR_NODE(&self->rb_node);
	INIT_LIST_HEAD(&self->user_dsos);
	INIT_LIST_HEAD(&self->kernel_dsos);

581 582 583 584
	self->threads = RB_ROOT;
	INIT_LIST_HEAD(&self->dead_threads);
	self->last_match = NULL;

585 586 587 588 589 590
	self->kmaps.machine = self;
	self->pid	    = pid;
	self->root_dir      = strdup(root_dir);
	return self->root_dir == NULL ? -ENOMEM : 0;
}

591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609
static void dsos__delete(struct list_head *self)
{
	struct dso *pos, *n;

	list_for_each_entry_safe(pos, n, self, node) {
		list_del(&pos->node);
		dso__delete(pos);
	}
}

void machine__exit(struct machine *self)
{
	map_groups__exit(&self->kmaps);
	dsos__delete(&self->user_dsos);
	dsos__delete(&self->kernel_dsos);
	free(self->root_dir);
	self->root_dir = NULL;
}

610 611 612 613 614 615
void machine__delete(struct machine *self)
{
	machine__exit(self);
	free(self);
}

616 617
struct machine *machines__add(struct rb_root *self, pid_t pid,
			      const char *root_dir)
618
{
619
	struct rb_node **p = &self->rb_node;
620
	struct rb_node *parent = NULL;
621
	struct machine *pos, *machine = malloc(sizeof(*machine));
622

623
	if (!machine)
624 625
		return NULL;

626 627 628 629
	if (machine__init(machine, root_dir, pid) != 0) {
		free(machine);
		return NULL;
	}
630 631 632

	while (*p != NULL) {
		parent = *p;
633
		pos = rb_entry(parent, struct machine, rb_node);
634 635 636 637 638 639
		if (pid < pos->pid)
			p = &(*p)->rb_left;
		else
			p = &(*p)->rb_right;
	}

640 641
	rb_link_node(&machine->rb_node, parent, p);
	rb_insert_color(&machine->rb_node, self);
642

643
	return machine;
644 645
}

646
struct machine *machines__find(struct rb_root *self, pid_t pid)
647
{
648
	struct rb_node **p = &self->rb_node;
649
	struct rb_node *parent = NULL;
650 651
	struct machine *machine;
	struct machine *default_machine = NULL;
652 653 654

	while (*p != NULL) {
		parent = *p;
655 656
		machine = rb_entry(parent, struct machine, rb_node);
		if (pid < machine->pid)
657
			p = &(*p)->rb_left;
658
		else if (pid > machine->pid)
659 660
			p = &(*p)->rb_right;
		else
661 662 663
			return machine;
		if (!machine->pid)
			default_machine = machine;
664 665
	}

666
	return default_machine;
667 668
}

669
struct machine *machines__findnew(struct rb_root *self, pid_t pid)
670 671
{
	char path[PATH_MAX];
672
	const char *root_dir = "";
673
	struct machine *machine = machines__find(self, pid);
674

675 676 677 678 679 680 681 682 683 684 685
	if (machine && (machine->pid == pid))
		goto out;

	if ((pid != HOST_KERNEL_ID) &&
	    (pid != DEFAULT_GUEST_KERNEL_ID) &&
	    (symbol_conf.guestmount)) {
		sprintf(path, "%s/%d", symbol_conf.guestmount, pid);
		if (access(path, R_OK)) {
			pr_err("Can't access file %s\n", path);
			machine = NULL;
			goto out;
686
		}
687
		root_dir = path;
688 689
	}

690 691
	machine = machines__add(self, pid, root_dir);

692
out:
693
	return machine;
694 695
}

696
void machines__process(struct rb_root *self, machine__process_t process, void *data)
697 698 699
{
	struct rb_node *nd;

700 701
	for (nd = rb_first(self); nd; nd = rb_next(nd)) {
		struct machine *pos = rb_entry(nd, struct machine, rb_node);
702 703 704 705
		process(pos, data);
	}
}

706
char *machine__mmap_name(struct machine *self, char *bf, size_t size)
707
{
708
	if (machine__is_host(self))
709
		snprintf(bf, size, "[%s]", "kernel.kallsyms");
710
	else if (machine__is_default_guest(self))
711
		snprintf(bf, size, "[%s]", "guest.kernel.kallsyms");
712
	else
713
		snprintf(bf, size, "[%s.%d]", "guest.kernel.kallsyms", self->pid);
714

715
	return bf;
716
}