| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * builtin-buildid-list.c | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2011-08-29 08:33:17 -03:00
										 |  |  |  * Builtin buildid-list command: list buildids in perf.data, in the running | 
					
						
							|  |  |  |  * kernel and in ELF files. | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2009, Red Hat Inc. | 
					
						
							|  |  |  |  * Copyright (C) 2009, Arnaldo Carvalho de Melo <acme@redhat.com> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | #include "builtin.h"
 | 
					
						
							|  |  |  | #include "perf.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-03 16:52:04 -02:00
										 |  |  | #include "util/build-id.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | #include "util/cache.h"
 | 
					
						
							|  |  |  | #include "util/debug.h"
 | 
					
						
							|  |  |  | #include "util/parse-options.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-11 21:24:02 -02:00
										 |  |  | #include "util/session.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | #include "util/symbol.h"
 | 
					
						
							| 
									
										
										
										
											2013-10-15 16:27:32 +02:00
										 |  |  | #include "util/data.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-29 08:07:22 -03:00
										 |  |  | static int sysfs__fprintf_build_id(FILE *fp) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	u8 kallsyms_build_id[BUILD_ID_SIZE]; | 
					
						
							|  |  |  | 	char sbuild_id[BUILD_ID_SIZE * 2 + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (sysfs__read_build_id("/sys/kernel/notes", kallsyms_build_id, | 
					
						
							|  |  |  | 				 sizeof(kallsyms_build_id)) != 0) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	build_id__sprintf(kallsyms_build_id, sizeof(kallsyms_build_id), | 
					
						
							|  |  |  | 			  sbuild_id); | 
					
						
							|  |  |  | 	fprintf(fp, "%s\n", sbuild_id); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-29 08:33:17 -03:00
										 |  |  | static int filename__fprintf_build_id(const char *name, FILE *fp) | 
					
						
							| 
									
										
										
										
											2011-08-29 08:07:22 -03:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2011-08-29 08:33:17 -03:00
										 |  |  | 	u8 build_id[BUILD_ID_SIZE]; | 
					
						
							|  |  |  | 	char sbuild_id[BUILD_ID_SIZE * 2 + 1]; | 
					
						
							| 
									
										
										
										
											2011-08-29 08:07:22 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-08-29 08:33:17 -03:00
										 |  |  | 	if (filename__read_build_id(name, build_id, | 
					
						
							|  |  |  | 				    sizeof(build_id)) != sizeof(build_id)) | 
					
						
							|  |  |  | 		return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	build_id__sprintf(build_id, sizeof(build_id), sbuild_id); | 
					
						
							|  |  |  | 	return fprintf(fp, "%s\n", sbuild_id); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-07 09:53:58 -03:00
										 |  |  | static bool dso__skip_buildid(struct dso *dso, int with_hits) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return with_hits && !dso->hit; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:02 +08:00
										 |  |  | static int perf_session__list_build_ids(bool force, bool with_hits) | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:53 +01:00
										 |  |  | { | 
					
						
							|  |  |  | 	struct perf_session *session; | 
					
						
							| 
									
										
										
										
											2013-10-15 16:27:32 +02:00
										 |  |  | 	struct perf_data_file file = { | 
					
						
							|  |  |  | 		.path  = input_name, | 
					
						
							|  |  |  | 		.mode  = PERF_DATA_MODE_READ, | 
					
						
							|  |  |  | 		.force = force, | 
					
						
							|  |  |  | 	}; | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-06 13:41:19 +09:00
										 |  |  | 	symbol__elf_init(); | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:54 +01:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * See if this is an ELF file first: | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2012-12-05 16:24:05 -03:00
										 |  |  | 	if (filename__fprintf_build_id(input_name, stdout)) | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:54 +01:00
										 |  |  | 		goto out; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-10-15 16:27:32 +02:00
										 |  |  | 	session = perf_session__new(&file, false, &build_id__mark_dso_hit_ops); | 
					
						
							| 
									
										
										
										
											2012-12-05 16:24:05 -03:00
										 |  |  | 	if (session == NULL) | 
					
						
							|  |  |  | 		return -1; | 
					
						
							| 
									
										
										
										
											2015-04-30 17:37:26 +03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * We take all buildids when the file contains AUX area tracing data | 
					
						
							|  |  |  | 	 * because we do not decode the trace because it would take too long. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	if (!perf_data_file__is_pipe(&file) && | 
					
						
							|  |  |  | 	    perf_header__has_feat(&session->header, HEADER_AUXTRACE)) | 
					
						
							|  |  |  | 		with_hits = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-05-15 13:28:15 +02:00
										 |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * in pipe-mode, the only way to get the buildids is to parse | 
					
						
							|  |  |  | 	 * the record stream. Buildids are stored as RECORD_HEADER_BUILD_ID | 
					
						
							|  |  |  | 	 */ | 
					
						
							| 
									
										
										
										
											2013-10-15 16:27:34 +02:00
										 |  |  | 	if (with_hits || perf_data_file__is_pipe(&file)) | 
					
						
							| 
									
										
										
										
											2015-03-03 11:58:45 -03:00
										 |  |  | 		perf_session__process_events(session); | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:53 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-07 09:53:58 -03:00
										 |  |  | 	perf_session__fprintf_dsos_buildid(session, stdout, dso__skip_buildid, with_hits); | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:53 +01:00
										 |  |  | 	perf_session__delete(session); | 
					
						
							| 
									
										
										
										
											2012-12-05 16:24:05 -03:00
										 |  |  | out: | 
					
						
							| 
									
										
										
										
											2011-12-07 10:02:53 +01:00
										 |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-11 01:15:03 +03:00
										 |  |  | int cmd_buildid_list(int argc, const char **argv, | 
					
						
							|  |  |  | 		     const char *prefix __maybe_unused) | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-10-01 15:20:58 -03:00
										 |  |  | 	bool show_kernel = false; | 
					
						
							|  |  |  | 	bool with_hits = false; | 
					
						
							|  |  |  | 	bool force = false; | 
					
						
							|  |  |  | 	const struct option options[] = { | 
					
						
							|  |  |  | 	OPT_BOOLEAN('H', "with-hits", &with_hits, "Show only DSOs with hits"), | 
					
						
							|  |  |  | 	OPT_STRING('i', "input", &input_name, "file", "input file name"), | 
					
						
							|  |  |  | 	OPT_BOOLEAN('f', "force", &force, "don't complain, do it"), | 
					
						
							|  |  |  | 	OPT_BOOLEAN('k', "kernel", &show_kernel, "Show current kernel build id"), | 
					
						
							|  |  |  | 	OPT_INCR('v', "verbose", &verbose, "be more verbose"), | 
					
						
							|  |  |  | 	OPT_END() | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 	const char * const buildid_list_usage[] = { | 
					
						
							|  |  |  | 		"perf buildid-list [<options>]", | 
					
						
							|  |  |  | 		NULL | 
					
						
							|  |  |  | 	}; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | 	argc = parse_options(argc, argv, options, buildid_list_usage, 0); | 
					
						
							|  |  |  | 	setup_pager(); | 
					
						
							| 
									
										
										
										
											2012-10-01 15:20:58 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (show_kernel) | 
					
						
							|  |  |  | 		return sysfs__fprintf_build_id(stdout); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-30 11:56:02 +08:00
										 |  |  | 	return perf_session__list_build_ids(force, with_hits); | 
					
						
							| 
									
										
										
										
											2009-11-16 16:32:45 -02:00
										 |  |  | } |