xref: /openbmc/linux/tools/perf/util/util.c (revision eb3fcf007fffe5830d815e713591f3e858f2a365)
1 #include "../perf.h"
2 #include "util.h"
3 #include "debug.h"
4 #include <api/fs/fs.h>
5 #include <sys/mman.h>
6 #ifdef HAVE_BACKTRACE_SUPPORT
7 #include <execinfo.h>
8 #endif
9 #include <stdio.h>
10 #include <stdlib.h>
11 #include <string.h>
12 #include <errno.h>
13 #include <limits.h>
14 #include <byteswap.h>
15 #include <linux/kernel.h>
16 #include <unistd.h>
17 #include "callchain.h"
18 
19 struct callchain_param	callchain_param = {
20 	.mode	= CHAIN_GRAPH_REL,
21 	.min_percent = 0.5,
22 	.order  = ORDER_CALLEE,
23 	.key	= CCKEY_FUNCTION
24 };
25 
26 /*
27  * XXX We need to find a better place for these things...
28  */
29 unsigned int page_size;
30 int cacheline_size;
31 
32 bool test_attr__enabled;
33 
34 bool perf_host  = true;
35 bool perf_guest = false;
36 
37 char tracing_path[PATH_MAX + 1]        = "/sys/kernel/debug/tracing";
38 char tracing_events_path[PATH_MAX + 1] = "/sys/kernel/debug/tracing/events";
39 
40 void event_attr_init(struct perf_event_attr *attr)
41 {
42 	if (!perf_host)
43 		attr->exclude_host  = 1;
44 	if (!perf_guest)
45 		attr->exclude_guest = 1;
46 	/* to capture ABI version */
47 	attr->size = sizeof(*attr);
48 }
49 
50 int mkdir_p(char *path, mode_t mode)
51 {
52 	struct stat st;
53 	int err;
54 	char *d = path;
55 
56 	if (*d != '/')
57 		return -1;
58 
59 	if (stat(path, &st) == 0)
60 		return 0;
61 
62 	while (*++d == '/');
63 
64 	while ((d = strchr(d, '/'))) {
65 		*d = '\0';
66 		err = stat(path, &st) && mkdir(path, mode);
67 		*d++ = '/';
68 		if (err)
69 			return -1;
70 		while (*d == '/')
71 			++d;
72 	}
73 	return (stat(path, &st) && mkdir(path, mode)) ? -1 : 0;
74 }
75 
76 int rm_rf(char *path)
77 {
78 	DIR *dir;
79 	int ret = 0;
80 	struct dirent *d;
81 	char namebuf[PATH_MAX];
82 
83 	dir = opendir(path);
84 	if (dir == NULL)
85 		return 0;
86 
87 	while ((d = readdir(dir)) != NULL && !ret) {
88 		struct stat statbuf;
89 
90 		if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
91 			continue;
92 
93 		scnprintf(namebuf, sizeof(namebuf), "%s/%s",
94 			  path, d->d_name);
95 
96 		ret = stat(namebuf, &statbuf);
97 		if (ret < 0) {
98 			pr_debug("stat failed: %s\n", namebuf);
99 			break;
100 		}
101 
102 		if (S_ISREG(statbuf.st_mode))
103 			ret = unlink(namebuf);
104 		else if (S_ISDIR(statbuf.st_mode))
105 			ret = rm_rf(namebuf);
106 		else {
107 			pr_debug("unknown file: %s\n", namebuf);
108 			ret = -1;
109 		}
110 	}
111 	closedir(dir);
112 
113 	if (ret < 0)
114 		return ret;
115 
116 	return rmdir(path);
117 }
118 
119 static int slow_copyfile(const char *from, const char *to)
120 {
121 	int err = -1;
122 	char *line = NULL;
123 	size_t n;
124 	FILE *from_fp = fopen(from, "r"), *to_fp;
125 
126 	if (from_fp == NULL)
127 		goto out;
128 
129 	to_fp = fopen(to, "w");
130 	if (to_fp == NULL)
131 		goto out_fclose_from;
132 
133 	while (getline(&line, &n, from_fp) > 0)
134 		if (fputs(line, to_fp) == EOF)
135 			goto out_fclose_to;
136 	err = 0;
137 out_fclose_to:
138 	fclose(to_fp);
139 	free(line);
140 out_fclose_from:
141 	fclose(from_fp);
142 out:
143 	return err;
144 }
145 
146 int copyfile_offset(int ifd, loff_t off_in, int ofd, loff_t off_out, u64 size)
147 {
148 	void *ptr;
149 	loff_t pgoff;
150 
151 	pgoff = off_in & ~(page_size - 1);
152 	off_in -= pgoff;
153 
154 	ptr = mmap(NULL, off_in + size, PROT_READ, MAP_PRIVATE, ifd, pgoff);
155 	if (ptr == MAP_FAILED)
156 		return -1;
157 
158 	while (size) {
159 		ssize_t ret = pwrite(ofd, ptr + off_in, size, off_out);
160 		if (ret < 0 && errno == EINTR)
161 			continue;
162 		if (ret <= 0)
163 			break;
164 
165 		size -= ret;
166 		off_in += ret;
167 		off_out -= ret;
168 	}
169 	munmap(ptr, off_in + size);
170 
171 	return size ? -1 : 0;
172 }
173 
174 int copyfile_mode(const char *from, const char *to, mode_t mode)
175 {
176 	int fromfd, tofd;
177 	struct stat st;
178 	int err = -1;
179 	char *tmp = NULL, *ptr = NULL;
180 
181 	if (stat(from, &st))
182 		goto out;
183 
184 	/* extra 'x' at the end is to reserve space for '.' */
185 	if (asprintf(&tmp, "%s.XXXXXXx", to) < 0) {
186 		tmp = NULL;
187 		goto out;
188 	}
189 	ptr = strrchr(tmp, '/');
190 	if (!ptr)
191 		goto out;
192 	ptr = memmove(ptr + 1, ptr, strlen(ptr) - 1);
193 	*ptr = '.';
194 
195 	tofd = mkstemp(tmp);
196 	if (tofd < 0)
197 		goto out;
198 
199 	if (fchmod(tofd, mode))
200 		goto out_close_to;
201 
202 	if (st.st_size == 0) { /* /proc? do it slowly... */
203 		err = slow_copyfile(from, tmp);
204 		goto out_close_to;
205 	}
206 
207 	fromfd = open(from, O_RDONLY);
208 	if (fromfd < 0)
209 		goto out_close_to;
210 
211 	err = copyfile_offset(fromfd, 0, tofd, 0, st.st_size);
212 
213 	close(fromfd);
214 out_close_to:
215 	close(tofd);
216 	if (!err)
217 		err = link(tmp, to);
218 	unlink(tmp);
219 out:
220 	free(tmp);
221 	return err;
222 }
223 
224 int copyfile(const char *from, const char *to)
225 {
226 	return copyfile_mode(from, to, 0755);
227 }
228 
229 unsigned long convert_unit(unsigned long value, char *unit)
230 {
231 	*unit = ' ';
232 
233 	if (value > 1000) {
234 		value /= 1000;
235 		*unit = 'K';
236 	}
237 
238 	if (value > 1000) {
239 		value /= 1000;
240 		*unit = 'M';
241 	}
242 
243 	if (value > 1000) {
244 		value /= 1000;
245 		*unit = 'G';
246 	}
247 
248 	return value;
249 }
250 
251 static ssize_t ion(bool is_read, int fd, void *buf, size_t n)
252 {
253 	void *buf_start = buf;
254 	size_t left = n;
255 
256 	while (left) {
257 		ssize_t ret = is_read ? read(fd, buf, left) :
258 					write(fd, buf, left);
259 
260 		if (ret < 0 && errno == EINTR)
261 			continue;
262 		if (ret <= 0)
263 			return ret;
264 
265 		left -= ret;
266 		buf  += ret;
267 	}
268 
269 	BUG_ON((size_t)(buf - buf_start) != n);
270 	return n;
271 }
272 
273 /*
274  * Read exactly 'n' bytes or return an error.
275  */
276 ssize_t readn(int fd, void *buf, size_t n)
277 {
278 	return ion(true, fd, buf, n);
279 }
280 
281 /*
282  * Write exactly 'n' bytes or return an error.
283  */
284 ssize_t writen(int fd, void *buf, size_t n)
285 {
286 	return ion(false, fd, buf, n);
287 }
288 
289 size_t hex_width(u64 v)
290 {
291 	size_t n = 1;
292 
293 	while ((v >>= 4))
294 		++n;
295 
296 	return n;
297 }
298 
299 static int hex(char ch)
300 {
301 	if ((ch >= '0') && (ch <= '9'))
302 		return ch - '0';
303 	if ((ch >= 'a') && (ch <= 'f'))
304 		return ch - 'a' + 10;
305 	if ((ch >= 'A') && (ch <= 'F'))
306 		return ch - 'A' + 10;
307 	return -1;
308 }
309 
310 /*
311  * While we find nice hex chars, build a long_val.
312  * Return number of chars processed.
313  */
314 int hex2u64(const char *ptr, u64 *long_val)
315 {
316 	const char *p = ptr;
317 	*long_val = 0;
318 
319 	while (*p) {
320 		const int hex_val = hex(*p);
321 
322 		if (hex_val < 0)
323 			break;
324 
325 		*long_val = (*long_val << 4) | hex_val;
326 		p++;
327 	}
328 
329 	return p - ptr;
330 }
331 
332 /* Obtain a backtrace and print it to stdout. */
333 #ifdef HAVE_BACKTRACE_SUPPORT
334 void dump_stack(void)
335 {
336 	void *array[16];
337 	size_t size = backtrace(array, ARRAY_SIZE(array));
338 	char **strings = backtrace_symbols(array, size);
339 	size_t i;
340 
341 	printf("Obtained %zd stack frames.\n", size);
342 
343 	for (i = 0; i < size; i++)
344 		printf("%s\n", strings[i]);
345 
346 	free(strings);
347 }
348 #else
349 void dump_stack(void) {}
350 #endif
351 
352 void sighandler_dump_stack(int sig)
353 {
354 	psignal(sig, "perf");
355 	dump_stack();
356 	exit(sig);
357 }
358 
359 void get_term_dimensions(struct winsize *ws)
360 {
361 	char *s = getenv("LINES");
362 
363 	if (s != NULL) {
364 		ws->ws_row = atoi(s);
365 		s = getenv("COLUMNS");
366 		if (s != NULL) {
367 			ws->ws_col = atoi(s);
368 			if (ws->ws_row && ws->ws_col)
369 				return;
370 		}
371 	}
372 #ifdef TIOCGWINSZ
373 	if (ioctl(1, TIOCGWINSZ, ws) == 0 &&
374 	    ws->ws_row && ws->ws_col)
375 		return;
376 #endif
377 	ws->ws_row = 25;
378 	ws->ws_col = 80;
379 }
380 
381 void set_term_quiet_input(struct termios *old)
382 {
383 	struct termios tc;
384 
385 	tcgetattr(0, old);
386 	tc = *old;
387 	tc.c_lflag &= ~(ICANON | ECHO);
388 	tc.c_cc[VMIN] = 0;
389 	tc.c_cc[VTIME] = 0;
390 	tcsetattr(0, TCSANOW, &tc);
391 }
392 
393 static void set_tracing_events_path(const char *tracing, const char *mountpoint)
394 {
395 	snprintf(tracing_path, sizeof(tracing_path), "%s/%s",
396 		 mountpoint, tracing);
397 	snprintf(tracing_events_path, sizeof(tracing_events_path), "%s/%s%s",
398 		 mountpoint, tracing, "events");
399 }
400 
401 static const char *__perf_tracefs_mount(const char *mountpoint)
402 {
403 	const char *mnt;
404 
405 	mnt = tracefs_mount(mountpoint);
406 	if (!mnt)
407 		return NULL;
408 
409 	set_tracing_events_path("", mnt);
410 
411 	return mnt;
412 }
413 
414 static const char *__perf_debugfs_mount(const char *mountpoint)
415 {
416 	const char *mnt;
417 
418 	mnt = debugfs_mount(mountpoint);
419 	if (!mnt)
420 		return NULL;
421 
422 	set_tracing_events_path("tracing/", mnt);
423 
424 	return mnt;
425 }
426 
427 const char *perf_debugfs_mount(const char *mountpoint)
428 {
429 	const char *mnt;
430 
431 	mnt = __perf_tracefs_mount(mountpoint);
432 	if (mnt)
433 		return mnt;
434 
435 	mnt = __perf_debugfs_mount(mountpoint);
436 
437 	return mnt;
438 }
439 
440 void perf_debugfs_set_path(const char *mntpt)
441 {
442 	set_tracing_events_path("tracing/", mntpt);
443 }
444 
445 char *get_tracing_file(const char *name)
446 {
447 	char *file;
448 
449 	if (asprintf(&file, "%s/%s", tracing_path, name) < 0)
450 		return NULL;
451 
452 	return file;
453 }
454 
455 void put_tracing_file(char *file)
456 {
457 	free(file);
458 }
459 
460 int parse_nsec_time(const char *str, u64 *ptime)
461 {
462 	u64 time_sec, time_nsec;
463 	char *end;
464 
465 	time_sec = strtoul(str, &end, 10);
466 	if (*end != '.' && *end != '\0')
467 		return -1;
468 
469 	if (*end == '.') {
470 		int i;
471 		char nsec_buf[10];
472 
473 		if (strlen(++end) > 9)
474 			return -1;
475 
476 		strncpy(nsec_buf, end, 9);
477 		nsec_buf[9] = '\0';
478 
479 		/* make it nsec precision */
480 		for (i = strlen(nsec_buf); i < 9; i++)
481 			nsec_buf[i] = '0';
482 
483 		time_nsec = strtoul(nsec_buf, &end, 10);
484 		if (*end != '\0')
485 			return -1;
486 	} else
487 		time_nsec = 0;
488 
489 	*ptime = time_sec * NSEC_PER_SEC + time_nsec;
490 	return 0;
491 }
492 
493 unsigned long parse_tag_value(const char *str, struct parse_tag *tags)
494 {
495 	struct parse_tag *i = tags;
496 
497 	while (i->tag) {
498 		char *s;
499 
500 		s = strchr(str, i->tag);
501 		if (s) {
502 			unsigned long int value;
503 			char *endptr;
504 
505 			value = strtoul(str, &endptr, 10);
506 			if (s != endptr)
507 				break;
508 
509 			if (value > ULONG_MAX / i->mult)
510 				break;
511 			value *= i->mult;
512 			return value;
513 		}
514 		i++;
515 	}
516 
517 	return (unsigned long) -1;
518 }
519 
520 int get_stack_size(const char *str, unsigned long *_size)
521 {
522 	char *endptr;
523 	unsigned long size;
524 	unsigned long max_size = round_down(USHRT_MAX, sizeof(u64));
525 
526 	size = strtoul(str, &endptr, 0);
527 
528 	do {
529 		if (*endptr)
530 			break;
531 
532 		size = round_up(size, sizeof(u64));
533 		if (!size || size > max_size)
534 			break;
535 
536 		*_size = size;
537 		return 0;
538 
539 	} while (0);
540 
541 	pr_err("callchain: Incorrect stack dump size (max %ld): %s\n",
542 	       max_size, str);
543 	return -1;
544 }
545 
546 int parse_callchain_record(const char *arg, struct callchain_param *param)
547 {
548 	char *tok, *name, *saveptr = NULL;
549 	char *buf;
550 	int ret = -1;
551 
552 	/* We need buffer that we know we can write to. */
553 	buf = malloc(strlen(arg) + 1);
554 	if (!buf)
555 		return -ENOMEM;
556 
557 	strcpy(buf, arg);
558 
559 	tok = strtok_r((char *)buf, ",", &saveptr);
560 	name = tok ? : (char *)buf;
561 
562 	do {
563 		/* Framepointer style */
564 		if (!strncmp(name, "fp", sizeof("fp"))) {
565 			if (!strtok_r(NULL, ",", &saveptr)) {
566 				param->record_mode = CALLCHAIN_FP;
567 				ret = 0;
568 			} else
569 				pr_err("callchain: No more arguments "
570 				       "needed for --call-graph fp\n");
571 			break;
572 
573 #ifdef HAVE_DWARF_UNWIND_SUPPORT
574 		/* Dwarf style */
575 		} else if (!strncmp(name, "dwarf", sizeof("dwarf"))) {
576 			const unsigned long default_stack_dump_size = 8192;
577 
578 			ret = 0;
579 			param->record_mode = CALLCHAIN_DWARF;
580 			param->dump_size = default_stack_dump_size;
581 
582 			tok = strtok_r(NULL, ",", &saveptr);
583 			if (tok) {
584 				unsigned long size = 0;
585 
586 				ret = get_stack_size(tok, &size);
587 				param->dump_size = size;
588 			}
589 #endif /* HAVE_DWARF_UNWIND_SUPPORT */
590 		} else if (!strncmp(name, "lbr", sizeof("lbr"))) {
591 			if (!strtok_r(NULL, ",", &saveptr)) {
592 				param->record_mode = CALLCHAIN_LBR;
593 				ret = 0;
594 			} else
595 				pr_err("callchain: No more arguments "
596 					"needed for --call-graph lbr\n");
597 			break;
598 		} else {
599 			pr_err("callchain: Unknown --call-graph option "
600 			       "value: %s\n", arg);
601 			break;
602 		}
603 
604 	} while (0);
605 
606 	free(buf);
607 	return ret;
608 }
609 
610 int filename__read_str(const char *filename, char **buf, size_t *sizep)
611 {
612 	size_t size = 0, alloc_size = 0;
613 	void *bf = NULL, *nbf;
614 	int fd, n, err = 0;
615 	char sbuf[STRERR_BUFSIZE];
616 
617 	fd = open(filename, O_RDONLY);
618 	if (fd < 0)
619 		return -errno;
620 
621 	do {
622 		if (size == alloc_size) {
623 			alloc_size += BUFSIZ;
624 			nbf = realloc(bf, alloc_size);
625 			if (!nbf) {
626 				err = -ENOMEM;
627 				break;
628 			}
629 
630 			bf = nbf;
631 		}
632 
633 		n = read(fd, bf + size, alloc_size - size);
634 		if (n < 0) {
635 			if (size) {
636 				pr_warning("read failed %d: %s\n", errno,
637 					 strerror_r(errno, sbuf, sizeof(sbuf)));
638 				err = 0;
639 			} else
640 				err = -errno;
641 
642 			break;
643 		}
644 
645 		size += n;
646 	} while (n > 0);
647 
648 	if (!err) {
649 		*sizep = size;
650 		*buf   = bf;
651 	} else
652 		free(bf);
653 
654 	close(fd);
655 	return err;
656 }
657 
658 const char *get_filename_for_perf_kvm(void)
659 {
660 	const char *filename;
661 
662 	if (perf_host && !perf_guest)
663 		filename = strdup("perf.data.host");
664 	else if (!perf_host && perf_guest)
665 		filename = strdup("perf.data.guest");
666 	else
667 		filename = strdup("perf.data.kvm");
668 
669 	return filename;
670 }
671 
672 int perf_event_paranoid(void)
673 {
674 	int value;
675 
676 	if (sysctl__read_int("kernel/perf_event_paranoid", &value))
677 		return INT_MAX;
678 
679 	return value;
680 }
681 
682 void mem_bswap_32(void *src, int byte_size)
683 {
684 	u32 *m = src;
685 	while (byte_size > 0) {
686 		*m = bswap_32(*m);
687 		byte_size -= sizeof(u32);
688 		++m;
689 	}
690 }
691 
692 void mem_bswap_64(void *src, int byte_size)
693 {
694 	u64 *m = src;
695 
696 	while (byte_size > 0) {
697 		*m = bswap_64(*m);
698 		byte_size -= sizeof(u64);
699 		++m;
700 	}
701 }
702 
703 bool find_process(const char *name)
704 {
705 	size_t len = strlen(name);
706 	DIR *dir;
707 	struct dirent *d;
708 	int ret = -1;
709 
710 	dir = opendir(procfs__mountpoint());
711 	if (!dir)
712 		return -1;
713 
714 	/* Walk through the directory. */
715 	while (ret && (d = readdir(dir)) != NULL) {
716 		char path[PATH_MAX];
717 		char *data;
718 		size_t size;
719 
720 		if ((d->d_type != DT_DIR) ||
721 		     !strcmp(".", d->d_name) ||
722 		     !strcmp("..", d->d_name))
723 			continue;
724 
725 		scnprintf(path, sizeof(path), "%s/%s/comm",
726 			  procfs__mountpoint(), d->d_name);
727 
728 		if (filename__read_str(path, &data, &size))
729 			continue;
730 
731 		ret = strncmp(name, data, len);
732 		free(data);
733 	}
734 
735 	closedir(dir);
736 	return ret ? false : true;
737 }
738