mirror of
				https://git.kernel.org/pub/scm/linux/kernel/git/chenhuacai/linux-loongson
				synced 2025-10-31 11:03:14 +00:00 
			
		
		
		
	 591765fdaf
			
		
	
	
		591765fdaf
		
	
	
	
	
		
			
			For long running sessions with many threads with short lifetimes the amount of memory that the buildid process takes is too much. Since we don't have hist_entries that may be pointing to them, we can just release the resources associated with each thread when the exit (PERF_RECORD_EXIT) event is received. For normal processing we need to annotate maps with hits, and thus hist_entries pointing to it and drop the ones that had none. Will be done in a followup patch. Cc: David S. Miller <davem@davemloft.net> Cc: Frederic Weisbecker <fweisbec@gmail.com> Cc: Ingo Molnar <mingo@elte.hu> Cc: Mike Galbraith <efault@gmx.de> Cc: Paul Mackerras <paulus@samba.org> Cc: Peter Zijlstra <peterz@infradead.org> Cc: Stephane Eranian <eranian@google.com> Cc: Tom Zanussi <tzanussi@gmail.com> LKML-Reference: <new-submission> Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
		
			
				
	
	
		
			78 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			78 lines
		
	
	
		
			1.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * build-id.c
 | |
|  *
 | |
|  * build-id support
 | |
|  *
 | |
|  * Copyright (C) 2009, 2010 Red Hat Inc.
 | |
|  * Copyright (C) 2009, 2010 Arnaldo Carvalho de Melo <acme@redhat.com>
 | |
|  */
 | |
| #include "util.h"
 | |
| #include <stdio.h>
 | |
| #include "build-id.h"
 | |
| #include "event.h"
 | |
| #include "symbol.h"
 | |
| #include <linux/kernel.h>
 | |
| #include "debug.h"
 | |
| 
 | |
| static int build_id__mark_dso_hit(event_t *event, struct perf_session *session)
 | |
| {
 | |
| 	struct addr_location al;
 | |
| 	u8 cpumode = event->header.misc & PERF_RECORD_MISC_CPUMODE_MASK;
 | |
| 	struct thread *thread = perf_session__findnew(session, event->ip.pid);
 | |
| 
 | |
| 	if (thread == NULL) {
 | |
| 		pr_err("problem processing %d event, skipping it.\n",
 | |
| 			event->header.type);
 | |
| 		return -1;
 | |
| 	}
 | |
| 
 | |
| 	thread__find_addr_map(thread, session, cpumode, MAP__FUNCTION,
 | |
| 			      event->ip.pid, event->ip.ip, &al);
 | |
| 
 | |
| 	if (al.map != NULL)
 | |
| 		al.map->dso->hit = 1;
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| static int event__exit_del_thread(event_t *self, struct perf_session *session)
 | |
| {
 | |
| 	struct thread *thread = perf_session__findnew(session, self->fork.tid);
 | |
| 
 | |
| 	dump_printf("(%d:%d):(%d:%d)\n", self->fork.pid, self->fork.tid,
 | |
| 		    self->fork.ppid, self->fork.ptid);
 | |
| 
 | |
| 	if (thread) {
 | |
| 		rb_erase(&thread->rb_node, &session->threads);
 | |
| 		session->last_match = NULL;
 | |
| 		thread__delete(thread);
 | |
| 	}
 | |
| 
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| struct perf_event_ops build_id__mark_dso_hit_ops = {
 | |
| 	.sample	= build_id__mark_dso_hit,
 | |
| 	.mmap	= event__process_mmap,
 | |
| 	.fork	= event__process_task,
 | |
| 	.exit	= event__exit_del_thread,
 | |
| };
 | |
| 
 | |
| char *dso__build_id_filename(struct dso *self, char *bf, size_t size)
 | |
| {
 | |
| 	char build_id_hex[BUILD_ID_SIZE * 2 + 1];
 | |
| 
 | |
| 	if (!self->has_build_id)
 | |
| 		return NULL;
 | |
| 
 | |
| 	build_id__sprintf(self->build_id, sizeof(self->build_id), build_id_hex);
 | |
| 	if (bf == NULL) {
 | |
| 		if (asprintf(&bf, "%s/.build-id/%.2s/%s", buildid_dir,
 | |
| 			     build_id_hex, build_id_hex + 2) < 0)
 | |
| 			return NULL;
 | |
| 	} else
 | |
| 		snprintf(bf, size, "%s/.build-id/%.2s/%s", buildid_dir,
 | |
| 			 build_id_hex, build_id_hex + 2);
 | |
| 	return bf;
 | |
| }
 |