mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-09-04 20:19:47 +08:00 
			
		
		
		
	 9d8b172f29
			
		
	
	
		9d8b172f29
		
	
	
	
	
		
			
			Note that since the thread was already inserted to the session
list, it will be released when the session is released.
Also, in perf_session__register_idle_thread() failure path,
the thread should be put before returning.
Refcnt debugger shows that the perf_session__register_idle_thread
gets the returned thread, but the caller (__cmd_top) does not
put the returned idle thread.
  ----
  ==== [0] ====
  Unreclaimed thread@0x24e6240
  Refcount +1 => 0 at
    ./perf(thread__new+0xe5) [0x4c8a75]
    ./perf(machine__findnew_thread+0x9a) [0x4bbdba]
    ./perf(perf_session__register_idle_thread+0x28) [0x4c63c8]
    ./perf(cmd_top+0xd7d) [0x43cf6d]
    ./perf() [0x47ba35]
    ./perf(main+0x617) [0x4225b7]
    /lib64/libc.so.6(__libc_start_main+0xf5) [0x7f06027c5af5]
    ./perf() [0x42272d]
  Refcount +1 => 1 at
    ./perf(thread__get+0x2c) [0x4c8bcc]
    ./perf(machine__findnew_thread+0xee) [0x4bbe0e]
    ./perf(perf_session__register_idle_thread+0x28) [0x4c63c8]
    ./perf(cmd_top+0xd7d) [0x43cf6d]
    ./perf() [0x47ba35]
    ./perf(main+0x617) [0x4225b7]
    /lib64/libc.so.6(__libc_start_main+0xf5) [0x7f06027c5af5]
    ./perf() [0x42272d]
  Refcount +1 => 2 at
    ./perf(thread__get+0x2c) [0x4c8bcc]
    ./perf(machine__findnew_thread+0x112) [0x4bbe32]
    ./perf(perf_session__register_idle_thread+0x28) [0x4c63c8]
    ./perf(cmd_top+0xd7d) [0x43cf6d]
    ./perf() [0x47ba35]
    ./perf(main+0x617) [0x4225b7]
    /lib64/libc.so.6(__libc_start_main+0xf5) [0x7f06027c5af5]
    ./perf() [0x42272d]
  ----
Signed-off-by: Masami Hiramatsu <masami.hiramatsu.pt@hitachi.com>
Tested-by: Arnaldo Carvalho de Melo <acme@redhat.com>
Cc: Adrian Hunter <adrian.hunter@intel.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Link: http://lkml.kernel.org/r/20151209021122.10245.69707.stgit@localhost.localdomain
[ Drop the refcount in perf_session__register_idle_thread() ]
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
		
	
			
		
			
				
	
	
		
			142 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			142 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| #ifndef __PERF_SESSION_H
 | |
| #define __PERF_SESSION_H
 | |
| 
 | |
| #include "trace-event.h"
 | |
| #include "event.h"
 | |
| #include "header.h"
 | |
| #include "machine.h"
 | |
| #include "symbol.h"
 | |
| #include "thread.h"
 | |
| #include "data.h"
 | |
| #include "ordered-events.h"
 | |
| #include <linux/rbtree.h>
 | |
| #include <linux/perf_event.h>
 | |
| 
 | |
| struct ip_callchain;
 | |
| struct thread;
 | |
| 
 | |
| struct auxtrace;
 | |
| struct itrace_synth_opts;
 | |
| 
 | |
| struct perf_session {
 | |
| 	struct perf_header	header;
 | |
| 	struct machines		machines;
 | |
| 	struct perf_evlist	*evlist;
 | |
| 	struct auxtrace		*auxtrace;
 | |
| 	struct itrace_synth_opts *itrace_synth_opts;
 | |
| 	struct list_head	auxtrace_index;
 | |
| 	struct trace_event	tevent;
 | |
| 	bool			repipe;
 | |
| 	bool			one_mmap;
 | |
| 	void			*one_mmap_addr;
 | |
| 	u64			one_mmap_offset;
 | |
| 	struct ordered_events	ordered_events;
 | |
| 	struct perf_data_file	*file;
 | |
| 	struct perf_tool	*tool;
 | |
| };
 | |
| 
 | |
| #define PRINT_IP_OPT_IP		(1<<0)
 | |
| #define PRINT_IP_OPT_SYM		(1<<1)
 | |
| #define PRINT_IP_OPT_DSO		(1<<2)
 | |
| #define PRINT_IP_OPT_SYMOFFSET	(1<<3)
 | |
| #define PRINT_IP_OPT_ONELINE	(1<<4)
 | |
| #define PRINT_IP_OPT_SRCLINE	(1<<5)
 | |
| 
 | |
| struct perf_tool;
 | |
| 
 | |
| struct perf_session *perf_session__new(struct perf_data_file *file,
 | |
| 				       bool repipe, struct perf_tool *tool);
 | |
| void perf_session__delete(struct perf_session *session);
 | |
| 
 | |
| void perf_event_header__bswap(struct perf_event_header *hdr);
 | |
| 
 | |
| int perf_session__peek_event(struct perf_session *session, off_t file_offset,
 | |
| 			     void *buf, size_t buf_sz,
 | |
| 			     union perf_event **event_ptr,
 | |
| 			     struct perf_sample *sample);
 | |
| 
 | |
| int perf_session__process_events(struct perf_session *session);
 | |
| 
 | |
| int perf_session__queue_event(struct perf_session *s, union perf_event *event,
 | |
| 			      struct perf_sample *sample, u64 file_offset);
 | |
| 
 | |
| void perf_tool__fill_defaults(struct perf_tool *tool);
 | |
| 
 | |
| int perf_session__resolve_callchain(struct perf_session *session,
 | |
| 				    struct perf_evsel *evsel,
 | |
| 				    struct thread *thread,
 | |
| 				    struct ip_callchain *chain,
 | |
| 				    struct symbol **parent);
 | |
| 
 | |
| bool perf_session__has_traces(struct perf_session *session, const char *msg);
 | |
| 
 | |
| void perf_event__attr_swap(struct perf_event_attr *attr);
 | |
| 
 | |
| int perf_session__create_kernel_maps(struct perf_session *session);
 | |
| 
 | |
| void perf_session__set_id_hdr_size(struct perf_session *session);
 | |
| 
 | |
| static inline
 | |
| struct machine *perf_session__find_machine(struct perf_session *session, pid_t pid)
 | |
| {
 | |
| 	return machines__find(&session->machines, pid);
 | |
| }
 | |
| 
 | |
| static inline
 | |
| struct machine *perf_session__findnew_machine(struct perf_session *session, pid_t pid)
 | |
| {
 | |
| 	return machines__findnew(&session->machines, pid);
 | |
| }
 | |
| 
 | |
| struct thread *perf_session__findnew(struct perf_session *session, pid_t pid);
 | |
| int perf_session__register_idle_thread(struct perf_session *session);
 | |
| 
 | |
| size_t perf_session__fprintf(struct perf_session *session, FILE *fp);
 | |
| 
 | |
| size_t perf_session__fprintf_dsos(struct perf_session *session, FILE *fp);
 | |
| 
 | |
| size_t perf_session__fprintf_dsos_buildid(struct perf_session *session, FILE *fp,
 | |
| 					  bool (fn)(struct dso *dso, int parm), int parm);
 | |
| 
 | |
| size_t perf_session__fprintf_nr_events(struct perf_session *session, FILE *fp);
 | |
| 
 | |
| struct perf_evsel *perf_session__find_first_evtype(struct perf_session *session,
 | |
| 					    unsigned int type);
 | |
| 
 | |
| void perf_evsel__print_ip(struct perf_evsel *evsel, struct perf_sample *sample,
 | |
| 			  struct addr_location *al,
 | |
| 			  unsigned int print_opts, unsigned int stack_depth);
 | |
| 
 | |
| int perf_session__cpu_bitmap(struct perf_session *session,
 | |
| 			     const char *cpu_list, unsigned long *cpu_bitmap);
 | |
| 
 | |
| void perf_session__fprintf_info(struct perf_session *s, FILE *fp, bool full);
 | |
| 
 | |
| struct perf_evsel_str_handler;
 | |
| 
 | |
| int __perf_session__set_tracepoints_handlers(struct perf_session *session,
 | |
| 					     const struct perf_evsel_str_handler *assocs,
 | |
| 					     size_t nr_assocs);
 | |
| 
 | |
| #define perf_session__set_tracepoints_handlers(session, array) \
 | |
| 	__perf_session__set_tracepoints_handlers(session, array, ARRAY_SIZE(array))
 | |
| 
 | |
| extern volatile int session_done;
 | |
| 
 | |
| #define session_done()	ACCESS_ONCE(session_done)
 | |
| 
 | |
| int perf_session__deliver_synth_event(struct perf_session *session,
 | |
| 				      union perf_event *event,
 | |
| 				      struct perf_sample *sample);
 | |
| 
 | |
| int perf_event__process_id_index(struct perf_tool *tool,
 | |
| 				 union perf_event *event,
 | |
| 				 struct perf_session *session);
 | |
| 
 | |
| int perf_event__synthesize_id_index(struct perf_tool *tool,
 | |
| 				    perf_event__handler_t process,
 | |
| 				    struct perf_evlist *evlist,
 | |
| 				    struct machine *machine);
 | |
| 
 | |
| #endif /* __PERF_SESSION_H */
 |