Blame view

tools/perf/util/util.c 8.79 KB
1aed26717   Joerg Roedel   perf kvm: Do gues...
1
  #include "../perf.h"
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
2
  #include "util.h"
84f5d36f4   Jiri Olsa   perf tools: Move ...
3
  #include "debug.h"
cd0cfad74   Borislav Petkov   perf tools: Move ...
4
  #include <api/fs/fs.h>
69e3f52d1   Frederic Weisbecker   perf: Fix implici...
5
  #include <sys/mman.h>
7a8ef4c4b   Arnaldo Carvalho de Melo   perf tools: Remov...
6
  #include <sys/stat.h>
07bc5c699   Wang Nan   perf tools: Make ...
7
  #include <sys/utsname.h>
76b31a29d   Arnaldo Carvalho de Melo   perf tools: Remov...
8
  #include <dirent.h>
fd20e8111   Arnaldo Carvalho de Melo   perf tools: Inclu...
9
  #include <inttypes.h>
9607ad3a6   Arnaldo Carvalho de Melo   perf tools: Add s...
10
  #include <signal.h>
dc4552bf7   Arnaldo Carvalho de Melo   perf tools: Add d...
11
12
  #include <stdio.h>
  #include <stdlib.h>
cef82c9f5   Jiri Olsa   perf tools: Add f...
13
14
  #include <string.h>
  #include <errno.h>
1a47245d2   Adrian Hunter   perf tools: Add p...
15
  #include <limits.h>
838d14520   Jiri Olsa   perf tools: Fine ...
16
  #include <linux/kernel.h>
c339b1a90   Wang Nan   perf tools: Make ...
17
  #include <linux/log2.h>
bd48c63eb   Arnaldo Carvalho de Melo   tools: Introduce ...
18
  #include <linux/time64.h>
9398c484f   Jiri Olsa   perf tools: Intro...
19
  #include <unistd.h>
14cbfbeb7   Namhyung Kim   perf report: Show...
20
  #include "strlist.h"
23aadb1fc   Jiri Olsa   perf callchain: M...
21

1aed26717   Joerg Roedel   perf kvm: Do gues...
22
23
24
  /*
   * XXX We need to find a better place for these things...
   */
0c1fe6b2f   Arnaldo Carvalho de Melo   perf tools: Have ...
25
  unsigned int page_size;
2b1b71003   Don Zickus   perf tools: Add s...
26
  int cacheline_size;
0c1fe6b2f   Arnaldo Carvalho de Melo   perf tools: Have ...
27

a29d5c9b8   Arnaldo Carvalho de Melo   perf tools: Separ...
28
29
  int sysctl_perf_event_max_stack = PERF_MAX_STACK_DEPTH;
  int sysctl_perf_event_max_contexts_per_stack = PERF_MAX_CONTEXTS_PER_STACK;
4cb93446c   Arnaldo Carvalho de Melo   perf tools: Set t...
30

0c6332e9d   Arnaldo Carvalho de Melo   perf python: Fix ...
31
  bool test_attr__enabled;
1aed26717   Joerg Roedel   perf kvm: Do gues...
32
  bool perf_host  = true;
c4a7dca92   Joerg Roedel   perf tools: Chang...
33
  bool perf_guest = false;
1aed26717   Joerg Roedel   perf kvm: Do gues...
34
35
36
37
38
39
40
  
  void event_attr_init(struct perf_event_attr *attr)
  {
  	if (!perf_host)
  		attr->exclude_host  = 1;
  	if (!perf_guest)
  		attr->exclude_guest = 1;
7e1ccd380   Stephane Eranian   perf tools: clean...
41
42
  	/* to capture ABI version */
  	attr->size = sizeof(*attr);
1aed26717   Joerg Roedel   perf kvm: Do gues...
43
  }
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
  int mkdir_p(char *path, mode_t mode)
  {
  	struct stat st;
  	int err;
  	char *d = path;
  
  	if (*d != '/')
  		return -1;
  
  	if (stat(path, &st) == 0)
  		return 0;
  
  	while (*++d == '/');
  
  	while ((d = strchr(d, '/'))) {
  		*d = '\0';
  		err = stat(path, &st) && mkdir(path, mode);
  		*d++ = '/';
  		if (err)
  			return -1;
  		while (*d == '/')
  			++d;
  	}
  	return (stat(path, &st) && mkdir(path, mode)) ? -1 : 0;
  }
9a9c733d6   Joe Stringer   tools perf util: ...
69
  int rm_rf(const char *path)
0b1de0be1   Namhyung Kim   perf tools: Add r...
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
  {
  	DIR *dir;
  	int ret = 0;
  	struct dirent *d;
  	char namebuf[PATH_MAX];
  
  	dir = opendir(path);
  	if (dir == NULL)
  		return 0;
  
  	while ((d = readdir(dir)) != NULL && !ret) {
  		struct stat statbuf;
  
  		if (!strcmp(d->d_name, ".") || !strcmp(d->d_name, ".."))
  			continue;
  
  		scnprintf(namebuf, sizeof(namebuf), "%s/%s",
  			  path, d->d_name);
2a1ef032c   Masami Hiramatsu   perf tools: Fix r...
88
89
  		/* We have to check symbolic link itself */
  		ret = lstat(namebuf, &statbuf);
0b1de0be1   Namhyung Kim   perf tools: Add r...
90
91
92
93
94
  		if (ret < 0) {
  			pr_debug("stat failed: %s
  ", namebuf);
  			break;
  		}
2a1ef032c   Masami Hiramatsu   perf tools: Fix r...
95
  		if (S_ISDIR(statbuf.st_mode))
0b1de0be1   Namhyung Kim   perf tools: Add r...
96
  			ret = rm_rf(namebuf);
2a1ef032c   Masami Hiramatsu   perf tools: Fix r...
97
98
  		else
  			ret = unlink(namebuf);
0b1de0be1   Namhyung Kim   perf tools: Add r...
99
100
101
102
103
104
105
106
  	}
  	closedir(dir);
  
  	if (ret < 0)
  		return ret;
  
  	return rmdir(path);
  }
e1ce726e1   Masami Hiramatsu   perf tools: Add l...
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
  /* A filter which removes dot files */
  bool lsdir_no_dot_filter(const char *name __maybe_unused, struct dirent *d)
  {
  	return d->d_name[0] != '.';
  }
  
  /* lsdir reads a directory and store it in strlist */
  struct strlist *lsdir(const char *name,
  		      bool (*filter)(const char *, struct dirent *))
  {
  	struct strlist *list = NULL;
  	DIR *dir;
  	struct dirent *d;
  
  	dir = opendir(name);
  	if (!dir)
  		return NULL;
  
  	list = strlist__new(NULL, NULL);
  	if (!list) {
357a54f32   Masami Hiramatsu   perf tools: Fix l...
127
  		errno = ENOMEM;
e1ce726e1   Masami Hiramatsu   perf tools: Add l...
128
129
130
131
132
133
134
135
136
137
138
139
  		goto out;
  	}
  
  	while ((d = readdir(dir)) != NULL) {
  		if (!filter || filter(name, d))
  			strlist__add(list, d->d_name);
  	}
  
  out:
  	closedir(dir);
  	return list;
  }
d7c72606d   Milos Vyletel   perf tools: Avoid...
140
  static int slow_copyfile(const char *from, const char *to)
9e201442d   Arnaldo Carvalho de Melo   perf symbols: Cac...
141
  {
9a17d7268   Adrian Hunter   perf tools: Add c...
142
  	int err = -1;
9e201442d   Arnaldo Carvalho de Melo   perf symbols: Cac...
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
  	char *line = NULL;
  	size_t n;
  	FILE *from_fp = fopen(from, "r"), *to_fp;
  
  	if (from_fp == NULL)
  		goto out;
  
  	to_fp = fopen(to, "w");
  	if (to_fp == NULL)
  		goto out_fclose_from;
  
  	while (getline(&line, &n, from_fp) > 0)
  		if (fputs(line, to_fp) == EOF)
  			goto out_fclose_to;
  	err = 0;
  out_fclose_to:
  	fclose(to_fp);
  	free(line);
  out_fclose_from:
  	fclose(from_fp);
  out:
  	return err;
  }
9c9f5a2f1   Namhyung Kim   perf tools: Intro...
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
  int copyfile_offset(int ifd, loff_t off_in, int ofd, loff_t off_out, u64 size)
  {
  	void *ptr;
  	loff_t pgoff;
  
  	pgoff = off_in & ~(page_size - 1);
  	off_in -= pgoff;
  
  	ptr = mmap(NULL, off_in + size, PROT_READ, MAP_PRIVATE, ifd, pgoff);
  	if (ptr == MAP_FAILED)
  		return -1;
  
  	while (size) {
  		ssize_t ret = pwrite(ofd, ptr + off_in, size, off_out);
  		if (ret < 0 && errno == EINTR)
  			continue;
  		if (ret <= 0)
  			break;
  
  		size -= ret;
  		off_in += ret;
  		off_out -= ret;
  	}
  	munmap(ptr, off_in + size);
  
  	return size ? -1 : 0;
  }
9a17d7268   Adrian Hunter   perf tools: Add c...
193
  int copyfile_mode(const char *from, const char *to, mode_t mode)
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
194
195
196
  {
  	int fromfd, tofd;
  	struct stat st;
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
197
  	int err = -1;
d7c72606d   Milos Vyletel   perf tools: Avoid...
198
  	char *tmp = NULL, *ptr = NULL;
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
199
200
201
  
  	if (stat(from, &st))
  		goto out;
d7c72606d   Milos Vyletel   perf tools: Avoid...
202
203
204
  	/* extra 'x' at the end is to reserve space for '.' */
  	if (asprintf(&tmp, "%s.XXXXXXx", to) < 0) {
  		tmp = NULL;
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
205
  		goto out;
d7c72606d   Milos Vyletel   perf tools: Avoid...
206
207
208
209
210
211
  	}
  	ptr = strrchr(tmp, '/');
  	if (!ptr)
  		goto out;
  	ptr = memmove(ptr + 1, ptr, strlen(ptr) - 1);
  	*ptr = '.';
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
212

d7c72606d   Milos Vyletel   perf tools: Avoid...
213
  	tofd = mkstemp(tmp);
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
214
  	if (tofd < 0)
d7c72606d   Milos Vyletel   perf tools: Avoid...
215
216
217
218
219
220
221
222
223
224
225
226
227
  		goto out;
  
  	if (fchmod(tofd, mode))
  		goto out_close_to;
  
  	if (st.st_size == 0) { /* /proc? do it slowly... */
  		err = slow_copyfile(from, tmp);
  		goto out_close_to;
  	}
  
  	fromfd = open(from, O_RDONLY);
  	if (fromfd < 0)
  		goto out_close_to;
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
228

9c9f5a2f1   Namhyung Kim   perf tools: Intro...
229
  	err = copyfile_offset(fromfd, 0, tofd, 0, st.st_size);
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
230

4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
231
  	close(fromfd);
d7c72606d   Milos Vyletel   perf tools: Avoid...
232
233
234
235
236
  out_close_to:
  	close(tofd);
  	if (!err)
  		err = link(tmp, to);
  	unlink(tmp);
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
237
  out:
d7c72606d   Milos Vyletel   perf tools: Avoid...
238
  	free(tmp);
4cf40131a   Arnaldo Carvalho de Melo   perf record: Intr...
239
240
  	return err;
  }
c82ee828a   Arnaldo Carvalho de Melo   perf report: Repo...
241

9a17d7268   Adrian Hunter   perf tools: Add c...
242
243
244
245
  int copyfile(const char *from, const char *to)
  {
  	return copyfile_mode(from, to, 0755);
  }
bc3a502bc   Jiri Olsa   perf tools: Add w...
246
  static ssize_t ion(bool is_read, int fd, void *buf, size_t n)
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
247
248
  {
  	void *buf_start = buf;
838d14520   Jiri Olsa   perf tools: Fine ...
249
  	size_t left = n;
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
250

838d14520   Jiri Olsa   perf tools: Fine ...
251
  	while (left) {
bc3a502bc   Jiri Olsa   perf tools: Add w...
252
253
  		ssize_t ret = is_read ? read(fd, buf, left) :
  					write(fd, buf, left);
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
254

e148c7608   Namhyung Kim   perf tools: Handl...
255
256
  		if (ret < 0 && errno == EINTR)
  			continue;
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
257
258
  		if (ret <= 0)
  			return ret;
838d14520   Jiri Olsa   perf tools: Fine ...
259
260
  		left -= ret;
  		buf  += ret;
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
261
  	}
838d14520   Jiri Olsa   perf tools: Fine ...
262
263
  	BUG_ON((size_t)(buf - buf_start) != n);
  	return n;
1e7972cc5   Arnaldo Carvalho de Melo   perf util: Move d...
264
  }
61e04b332   Arnaldo Carvalho de Melo   perf annotate bro...
265

bc3a502bc   Jiri Olsa   perf tools: Add w...
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
  /*
   * Read exactly 'n' bytes or return an error.
   */
  ssize_t readn(int fd, void *buf, size_t n)
  {
  	return ion(true, fd, buf, n);
  }
  
  /*
   * Write exactly 'n' bytes or return an error.
   */
  ssize_t writen(int fd, void *buf, size_t n)
  {
  	return ion(false, fd, buf, n);
  }
61e04b332   Arnaldo Carvalho de Melo   perf annotate bro...
281
282
283
284
285
286
287
288
289
  size_t hex_width(u64 v)
  {
  	size_t n = 1;
  
  	while ((v >>= 4))
  		++n;
  
  	return n;
  }
dc4552bf7   Arnaldo Carvalho de Melo   perf tools: Add d...
290

b2aff5f61   Jiri Olsa   perf tools: Move ...
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
  static int hex(char ch)
  {
  	if ((ch >= '0') && (ch <= '9'))
  		return ch - '0';
  	if ((ch >= 'a') && (ch <= 'f'))
  		return ch - 'a' + 10;
  	if ((ch >= 'A') && (ch <= 'F'))
  		return ch - 'A' + 10;
  	return -1;
  }
  
  /*
   * While we find nice hex chars, build a long_val.
   * Return number of chars processed.
   */
  int hex2u64(const char *ptr, u64 *long_val)
  {
  	const char *p = ptr;
  	*long_val = 0;
  
  	while (*p) {
  		const int hex_val = hex(*p);
  
  		if (hex_val < 0)
  			break;
  
  		*long_val = (*long_val << 4) | hex_val;
  		p++;
  	}
  
  	return p - ptr;
  }
1a47245d2   Adrian Hunter   perf tools: Add p...
323
324
  int perf_event_paranoid(void)
  {
1a47245d2   Adrian Hunter   perf tools: Add p...
325
  	int value;
ce27309f6   Arnaldo Carvalho de Melo   perf tools: Use s...
326
  	if (sysctl__read_int("kernel/perf_event_paranoid", &value))
1a47245d2   Adrian Hunter   perf tools: Add p...
327
328
329
330
  		return INT_MAX;
  
  	return value;
  }
71db07b12   Adrian Hunter   perf tools: Move ...
331

63914aca8   Jiri Olsa   perf tools: Show ...
332
333
334
335
336
337
338
339
340
  bool find_process(const char *name)
  {
  	size_t len = strlen(name);
  	DIR *dir;
  	struct dirent *d;
  	int ret = -1;
  
  	dir = opendir(procfs__mountpoint());
  	if (!dir)
bf6445631   Peter Senna Tschudin   perf tools: Bool ...
341
  		return false;
63914aca8   Jiri Olsa   perf tools: Show ...
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
  
  	/* Walk through the directory. */
  	while (ret && (d = readdir(dir)) != NULL) {
  		char path[PATH_MAX];
  		char *data;
  		size_t size;
  
  		if ((d->d_type != DT_DIR) ||
  		     !strcmp(".", d->d_name) ||
  		     !strcmp("..", d->d_name))
  			continue;
  
  		scnprintf(path, sizeof(path), "%s/%s/comm",
  			  procfs__mountpoint(), d->d_name);
  
  		if (filename__read_str(path, &data, &size))
  			continue;
  
  		ret = strncmp(name, data, len);
  		free(data);
  	}
  
  	closedir(dir);
  	return ret ? false : true;
  }
07bc5c699   Wang Nan   perf tools: Make ...
367

d18acd15c   Wang Nan   perf tools: Fix k...
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
  static int
  fetch_ubuntu_kernel_version(unsigned int *puint)
  {
  	ssize_t len;
  	size_t line_len = 0;
  	char *ptr, *line = NULL;
  	int version, patchlevel, sublevel, err;
  	FILE *vsig = fopen("/proc/version_signature", "r");
  
  	if (!vsig) {
  		pr_debug("Open /proc/version_signature failed: %s
  ",
  			 strerror(errno));
  		return -1;
  	}
  
  	len = getline(&line, &line_len, vsig);
  	fclose(vsig);
  	err = -1;
  	if (len <= 0) {
  		pr_debug("Reading from /proc/version_signature failed: %s
  ",
  			 strerror(errno));
  		goto errout;
  	}
  
  	ptr = strrchr(line, ' ');
  	if (!ptr) {
  		pr_debug("Parsing /proc/version_signature failed: %s
  ", line);
  		goto errout;
  	}
  
  	err = sscanf(ptr + 1, "%d.%d.%d",
  		     &version, &patchlevel, &sublevel);
  	if (err != 3) {
  		pr_debug("Unable to get kernel version from /proc/version_signature '%s'
  ",
  			 line);
  		goto errout;
  	}
  
  	if (puint)
  		*puint = (version << 16) + (patchlevel << 8) + sublevel;
  	err = 0;
  errout:
  	free(line);
  	return err;
  }
07bc5c699   Wang Nan   perf tools: Make ...
417
418
419
420
421
422
  int
  fetch_kernel_version(unsigned int *puint, char *str,
  		     size_t str_size)
  {
  	struct utsname utsname;
  	int version, patchlevel, sublevel, err;
d18acd15c   Wang Nan   perf tools: Fix k...
423
424
425
426
427
  	bool int_ver_ready = false;
  
  	if (access("/proc/version_signature", R_OK) == 0)
  		if (!fetch_ubuntu_kernel_version(puint))
  			int_ver_ready = true;
07bc5c699   Wang Nan   perf tools: Make ...
428
429
430
431
432
433
434
435
436
437
438
439
440
  
  	if (uname(&utsname))
  		return -1;
  
  	if (str && str_size) {
  		strncpy(str, utsname.release, str_size);
  		str[str_size - 1] = '\0';
  	}
  
  	err = sscanf(utsname.release, "%d.%d.%d",
  		     &version, &patchlevel, &sublevel);
  
  	if (err != 3) {
d18acd15c   Wang Nan   perf tools: Fix k...
441
442
  		pr_debug("Unable to get kernel version from uname '%s'
  ",
07bc5c699   Wang Nan   perf tools: Make ...
443
444
445
  			 utsname.release);
  		return -1;
  	}
d18acd15c   Wang Nan   perf tools: Fix k...
446
  	if (puint && !int_ver_ready)
07bc5c699   Wang Nan   perf tools: Make ...
447
448
449
  		*puint = (version << 16) + (patchlevel << 8) + sublevel;
  	return 0;
  }
14cbfbeb7   Namhyung Kim   perf report: Show...
450
451
452
453
454
455
456
  
  const char *perf_tip(const char *dirpath)
  {
  	struct strlist *tips;
  	struct str_node *node;
  	char *tip = NULL;
  	struct strlist_config conf = {
34b7b0f95   Namhyung Kim   perf tools: Fallb...
457
458
  		.dirname = dirpath,
  		.file_only = true,
14cbfbeb7   Namhyung Kim   perf report: Show...
459
460
461
  	};
  
  	tips = strlist__new("tips.txt", &conf);
34b7b0f95   Namhyung Kim   perf tools: Fallb...
462
  	if (tips == NULL)
570eda032   David Carrillo-Cisneros   perf util: Hint m...
463
464
  		return errno == ENOENT ? NULL :
  			"Tip: check path of tips.txt or get more memory! ;-p";
34b7b0f95   Namhyung Kim   perf tools: Fallb...
465
466
  
  	if (strlist__nr_entries(tips) == 0)
14cbfbeb7   Namhyung Kim   perf report: Show...
467
  		goto out;
14cbfbeb7   Namhyung Kim   perf report: Show...
468
469
470
471
472
473
474
475
476
477
  
  	node = strlist__entry(tips, random() % strlist__nr_entries(tips));
  	if (asprintf(&tip, "Tip: %s", node->s) < 0)
  		tip = (char *)"Tip: get more memory! ;-)";
  
  out:
  	strlist__delete(tips);
  
  	return tip;
  }