| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | #ifndef __PERF_EVLIST_H
 | 
					
						
							|  |  |  | #define __PERF_EVLIST_H 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <linux/list.h>
 | 
					
						
							| 
									
										
										
										
											2011-11-04 09:10:59 -02:00
										 |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | #include "../perf.h"
 | 
					
						
							| 
									
										
										
										
											2011-01-15 10:40:59 -02:00
										 |  |  | #include "event.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:42:15 -03:00
										 |  |  | #include "evsel.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-04 09:10:59 -02:00
										 |  |  | #include "util.h"
 | 
					
						
							| 
									
										
										
										
											2011-11-09 08:47:15 -02:00
										 |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 22:30:02 -02:00
										 |  |  | struct pollfd; | 
					
						
							| 
									
										
										
										
											2011-01-30 10:46:46 -02:00
										 |  |  | struct thread_map; | 
					
						
							|  |  |  | struct cpu_map; | 
					
						
							| 
									
										
										
										
											2011-11-08 14:41:57 -02:00
										 |  |  | struct perf_record_opts; | 
					
						
							| 
									
										
										
										
											2011-01-11 22:30:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | #define PERF_EVLIST__HLIST_BITS 8
 | 
					
						
							|  |  |  | #define PERF_EVLIST__HLIST_SIZE (1 << PERF_EVLIST__HLIST_BITS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-02-05 14:12:42 -07:00
										 |  |  | struct perf_mmap { | 
					
						
							|  |  |  | 	void		 *base; | 
					
						
							|  |  |  | 	int		 mask; | 
					
						
							|  |  |  | 	unsigned int	 prev; | 
					
						
							|  |  |  | 	union perf_event event_copy; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | struct perf_evlist { | 
					
						
							|  |  |  | 	struct list_head entries; | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | 	struct hlist_head heads[PERF_EVLIST__HLIST_SIZE]; | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | 	int		 nr_entries; | 
					
						
							| 
									
										
										
										
											2013-01-22 18:09:29 +09:00
										 |  |  | 	int		 nr_groups; | 
					
						
							| 
									
										
										
										
											2011-01-11 22:30:02 -02:00
										 |  |  | 	int		 nr_fds; | 
					
						
							| 
									
										
											  
											
												perf evlist: Fix per thread mmap setup
The PERF_EVENT_IOC_SET_OUTPUT ioctl was returning -EINVAL when using
--pid when monitoring multithreaded apps, as we can only share a ring
buffer for events on the same thread if not doing per cpu.
Fix it by using per thread ring buffers.
Tested with:
[root@felicio ~]# tuna -t 26131 -CP | nl
  1                      thread       ctxt_switches
  2    pid SCHED_ rtpri affinity voluntary nonvoluntary             cmd
  3 26131   OTHER     0      0,1  10814276      2397830 chromium-browse
  4  642    OTHER     0      0,1     14688            0 chromium-browse
  5  26148  OTHER     0      0,1    713602       115479 chromium-browse
  6  26149  OTHER     0      0,1    801958         2262 chromium-browse
  7  26150  OTHER     0      0,1   1271128          248 chromium-browse
  8  26151  OTHER     0      0,1         3            0 chromium-browse
  9  27049  OTHER     0      0,1     36796            9 chromium-browse
 10  618    OTHER     0      0,1     14711            0 chromium-browse
 11  661    OTHER     0      0,1     14593            0 chromium-browse
 12  29048  OTHER     0      0,1     28125            0 chromium-browse
 13  26143  OTHER     0      0,1   2202789          781 chromium-browse
[root@felicio ~]#
So 11 threads under pid 26131, then:
[root@felicio ~]# perf record -F 50000 --pid 26131
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
  1 7fa4a2538000-7fa4a25b9000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  2 7fa4a25b9000-7fa4a263a000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  3 7fa4a263a000-7fa4a26bb000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  4 7fa4a26bb000-7fa4a273c000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  5 7fa4a273c000-7fa4a27bd000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  6 7fa4a27bd000-7fa4a283e000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  7 7fa4a283e000-7fa4a28bf000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  8 7fa4a28bf000-7fa4a2940000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  9 7fa4a2940000-7fa4a29c1000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
 10 7fa4a29c1000-7fa4a2a42000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
 11 7fa4a2a42000-7fa4a2ac3000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
[root@felicio ~]#
11 mmaps, one per thread since we didn't specify any CPU list, so we need one
mmap per thread and:
[root@felicio ~]# perf record -F 50000 --pid 26131
^M
^C[ perf record: Woken up 79 times to write data ]
[ perf record: Captured and wrote 20.614 MB perf.data (~900639 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	 371310 26131
     2	  96516 26148
     3	  95694 26149
     4	  95203 26150
     5	   7291 26143
     6	     87 27049
     7	     76 661
     8	     60 29048
     9	     47 618
    10	     43 642
[root@felicio ~]#
Ok, one of the threads, 26151 was quiescent, so no samples there, but all the
others are there.
Then, if I specify one CPU:
[root@felicio ~]# perf record -F 50000 --pid 26131 --cpu 1
^C[ perf record: Woken up 1 times to write data ]
[ perf record: Captured and wrote 0.680 MB perf.data (~29730 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	   8444 26131
     2	   2584 26149
     3	   2518 26148
     4	   2324 26150
     5	    123 26143
     6	      9 661
     7	      9 29048
[root@felicio ~]#
This machine has two cores, so fewer threads appeared on the radar, and:
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
 1 7f484b922000-7f484b9a3000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
[root@felicio ~]#
Just one mmap, as now we can use just one per-cpu buffer instead of the
per-thread needed in the previous case.
For global profiling:
[root@felicio ~]# perf record -F 50000 -a
^C[ perf record: Woken up 26 times to write data ]
[ perf record: Captured and wrote 7.128 MB perf.data (~311412 samples) ]
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
     1	7fb49b435000-7fb49b4b6000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
     2	7fb49b4b6000-7fb49b537000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
[root@felicio ~]#
It uses per-cpu buffers.
For just one thread:
[root@felicio ~]# perf record -F 50000 --tid 26148
^C[ perf record: Woken up 2 times to write data ]
[ perf record: Captured and wrote 0.330 MB perf.data (~14426 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	   9969 26148
[root@felicio ~]#
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
     1	7f286a51b000-7f286a59c000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
[root@felicio ~]#
Tested-by: David Ahern <dsahern@gmail.com>
Tested-by: Lin Ming <ming.m.lin@intel.com>
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>
Link: http://lkml.kernel.org/r/20110426204401.GB1746@ghostprotocols.net
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2011-05-15 09:39:00 -03:00
										 |  |  | 	int		 nr_mmaps; | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | 	int		 mmap_len; | 
					
						
							| 
									
										
										
										
											2011-11-09 08:47:15 -02:00
										 |  |  | 	struct { | 
					
						
							|  |  |  | 		int	cork_fd; | 
					
						
							|  |  |  | 		pid_t	pid; | 
					
						
							|  |  |  | 	} workload; | 
					
						
							| 
									
										
										
										
											2011-01-29 09:08:13 -02:00
										 |  |  | 	bool		 overwrite; | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | 	struct perf_mmap *mmap; | 
					
						
							| 
									
										
										
										
											2011-01-11 22:30:02 -02:00
										 |  |  | 	struct pollfd	 *pollfd; | 
					
						
							| 
									
										
										
										
											2011-01-30 11:59:43 -02:00
										 |  |  | 	struct thread_map *threads; | 
					
						
							|  |  |  | 	struct cpu_map	  *cpus; | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | 	struct perf_evsel *selected; | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												perf tools: Save some loops using perf_evlist__id2evsel
Since we already ask for PERF_SAMPLE_ID and use it to quickly find the
associated evsel, add handler func + data to struct perf_evsel to avoid
using chains of if(strcmp(event_name)) and also to avoid all the linear
list searches via trace_event_find.
To demonstrate the technique convert 'perf sched' to it:
 # perf sched record sleep 5m
And then:
 Performance counter stats for '/tmp/oldperf sched lat':
        646.929438 task-clock                #    0.999 CPUs utilized
                 9 context-switches          #    0.000 M/sec
                 0 CPU-migrations            #    0.000 M/sec
            20,901 page-faults               #    0.032 M/sec
     1,290,144,450 cycles                    #    1.994 GHz
   <not supported> stalled-cycles-frontend
   <not supported> stalled-cycles-backend
     1,606,158,439 instructions              #    1.24  insns per cycle
       339,088,395 branches                  #  524.151 M/sec
         4,550,735 branch-misses             #    1.34% of all branches
       0.647524759 seconds time elapsed
Versus:
 Performance counter stats for 'perf sched lat':
        473.564691 task-clock                #    0.999 CPUs utilized
                 9 context-switches          #    0.000 M/sec
                 0 CPU-migrations            #    0.000 M/sec
            20,903 page-faults               #    0.044 M/sec
       944,367,984 cycles                    #    1.994 GHz
   <not supported> stalled-cycles-frontend
   <not supported> stalled-cycles-backend
     1,442,385,571 instructions              #    1.53  insns per cycle
       308,383,106 branches                  #  651.195 M/sec
         4,481,784 branch-misses             #    1.45% of all branches
       0.474215751 seconds time elapsed
[root@emilia ~]#
Cc: David Ahern <dsahern@gmail.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lkml.kernel.org/n/tip-1kbzpl74lwi6lavpqke2u2p3@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2011-11-28 17:57:40 -02:00
										 |  |  | struct perf_evsel_str_handler { | 
					
						
							|  |  |  | 	const char *name; | 
					
						
							|  |  |  | 	void	   *handler; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-30 11:59:43 -02:00
										 |  |  | struct perf_evlist *perf_evlist__new(struct cpu_map *cpus, | 
					
						
							|  |  |  | 				     struct thread_map *threads); | 
					
						
							|  |  |  | void perf_evlist__init(struct perf_evlist *evlist, struct cpu_map *cpus, | 
					
						
							|  |  |  | 		       struct thread_map *threads); | 
					
						
							| 
									
										
										
										
											2011-01-18 21:41:45 -02:00
										 |  |  | void perf_evlist__exit(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | void perf_evlist__delete(struct perf_evlist *evlist); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void perf_evlist__add(struct perf_evlist *evlist, struct perf_evsel *entry); | 
					
						
							|  |  |  | int perf_evlist__add_default(struct perf_evlist *evlist); | 
					
						
							| 
									
										
											  
											
												perf stat: Initialize default events wrt exclude_{guest,host}
When no event is specified the tools use perf_evlist__add_default(), that will
call event_attr_init to initialize the KVM exclusion bits.
When the change was made to the tools so that by default guest samples would be
excluded, the changes were made just to the parsing routines and to
perf_evlist__add_default(), not to perf_evlist__add_attrs, that is used so far
just by perf stat to add multiple events, according to the level of detail
specified.
Recently the tools were changed to reconstruct the event name from all the
details in perf_event_attr, not just from .type and .config, but taking into
account all the feature bits (.exclude_{guest,host,user,kernel,etc},
.precise_ip, etc).
That is when we noticed that the default for perf stat wasn't the one for the
rest of the tools, i.e. the .exclude_guest bit wasn't being set.
I.e. the default, that doesn't call event_attr_init was showing the :HG
modifier:
  $ perf stat usleep 1
   Performance counter stats for 'usleep 1':
            0.942119 task-clock                #    0.454 CPUs utilized
                   1 context-switches          #    0.001 M/sec
                   0 CPU-migrations            #    0.000 K/sec
                 126 page-faults               #    0.134 M/sec
             693,193 cycles:HG                 #    0.736 GHz                     [40.11%]
             407,461 stalled-cycles-frontend:HG #   58.78% frontend cycles idle    [72.29%]
             365,403 stalled-cycles-backend:HG #   52.71% backend  cycles idle
             465,982 instructions:HG           #    0.67  insns per cycle
                                               #    0.87  stalled cycles per insn
              89,760 branches:HG               #   95.275 M/sec
               6,178 branch-misses:HG          #    6.88% of all branches
         0.002077228 seconds time elapsed
While if one explicitely specifies the same events, which will make the parsing code
to be called and thus event_attr_init is called:
  $ perf stat -e task-clock,context-switches,migrations,page-faults,cycles,stalled-cycles-frontend,stalled-cycles-backend,instructions,branches,branch-misses usleep 1
   Performance counter stats for 'usleep 1':
            1.040349 task-clock                #    0.500 CPUs utilized
                   2 context-switches          #    0.002 M/sec
                   0 CPU-migrations            #    0.000 K/sec
                 127 page-faults               #    0.122 M/sec
             587,966 cycles                    #    0.565 GHz                     [13.18%]
             459,167 stalled-cycles-frontend   #   78.09% frontend cycles idle
             390,249 stalled-cycles-backend    #   66.37% backend  cycles idle
             504,006 instructions              #    0.86  insns per cycle
                                               #    0.91  stalled cycles per insn
              96,455 branches                  #   92.714 M/sec
               6,522 branch-misses             #    6.76% of all branches         [96.12%]
         0.002078681 seconds time elapsed
Fix it by introducing a perf_evlist__add_default_attrs method that will call
evlist_attr_init in all the perf_event_attr entries before adding the events.
Reported-by: Ingo Molnar <mingo@kernel.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Namhyung Kim <namhyung@gmail.com>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lkml.kernel.org/n/tip-4eysr236r0pgiyum9epwxw7s@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2012-05-30 13:53:54 -03:00
										 |  |  | int __perf_evlist__add_default_attrs(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 				     struct perf_event_attr *attrs, size_t nr_attrs); | 
					
						
							| 
									
										
										
										
											2012-10-03 11:50:55 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												perf stat: Initialize default events wrt exclude_{guest,host}
When no event is specified the tools use perf_evlist__add_default(), that will
call event_attr_init to initialize the KVM exclusion bits.
When the change was made to the tools so that by default guest samples would be
excluded, the changes were made just to the parsing routines and to
perf_evlist__add_default(), not to perf_evlist__add_attrs, that is used so far
just by perf stat to add multiple events, according to the level of detail
specified.
Recently the tools were changed to reconstruct the event name from all the
details in perf_event_attr, not just from .type and .config, but taking into
account all the feature bits (.exclude_{guest,host,user,kernel,etc},
.precise_ip, etc).
That is when we noticed that the default for perf stat wasn't the one for the
rest of the tools, i.e. the .exclude_guest bit wasn't being set.
I.e. the default, that doesn't call event_attr_init was showing the :HG
modifier:
  $ perf stat usleep 1
   Performance counter stats for 'usleep 1':
            0.942119 task-clock                #    0.454 CPUs utilized
                   1 context-switches          #    0.001 M/sec
                   0 CPU-migrations            #    0.000 K/sec
                 126 page-faults               #    0.134 M/sec
             693,193 cycles:HG                 #    0.736 GHz                     [40.11%]
             407,461 stalled-cycles-frontend:HG #   58.78% frontend cycles idle    [72.29%]
             365,403 stalled-cycles-backend:HG #   52.71% backend  cycles idle
             465,982 instructions:HG           #    0.67  insns per cycle
                                               #    0.87  stalled cycles per insn
              89,760 branches:HG               #   95.275 M/sec
               6,178 branch-misses:HG          #    6.88% of all branches
         0.002077228 seconds time elapsed
While if one explicitely specifies the same events, which will make the parsing code
to be called and thus event_attr_init is called:
  $ perf stat -e task-clock,context-switches,migrations,page-faults,cycles,stalled-cycles-frontend,stalled-cycles-backend,instructions,branches,branch-misses usleep 1
   Performance counter stats for 'usleep 1':
            1.040349 task-clock                #    0.500 CPUs utilized
                   2 context-switches          #    0.002 M/sec
                   0 CPU-migrations            #    0.000 K/sec
                 127 page-faults               #    0.122 M/sec
             587,966 cycles                    #    0.565 GHz                     [13.18%]
             459,167 stalled-cycles-frontend   #   78.09% frontend cycles idle
             390,249 stalled-cycles-backend    #   66.37% backend  cycles idle
             504,006 instructions              #    0.86  insns per cycle
                                               #    0.91  stalled cycles per insn
              96,455 branches                  #   92.714 M/sec
               6,522 branch-misses             #    6.76% of all branches         [96.12%]
         0.002078681 seconds time elapsed
Fix it by introducing a perf_evlist__add_default_attrs method that will call
evlist_attr_init in all the perf_event_attr entries before adding the events.
Reported-by: Ingo Molnar <mingo@kernel.org>
Cc: David Ahern <dsahern@gmail.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Jiri Olsa <jolsa@redhat.com>
Cc: Mike Galbraith <efault@gmx.de>
Cc: Namhyung Kim <namhyung@gmail.com>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Stephane Eranian <eranian@google.com>
Link: http://lkml.kernel.org/n/tip-4eysr236r0pgiyum9epwxw7s@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2012-05-30 13:53:54 -03:00
										 |  |  | #define perf_evlist__add_default_attrs(evlist, array) \
 | 
					
						
							|  |  |  | 	__perf_evlist__add_default_attrs(evlist, array, ARRAY_SIZE(array)) | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-10-03 11:40:22 -03:00
										 |  |  | int perf_evlist__add_newtp(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 			   const char *sys, const char *name, void *handler); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-09-26 15:07:39 -03:00
										 |  |  | int perf_evlist__set_filter(struct perf_evlist *evlist, const char *filter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-27 13:08:42 -03:00
										 |  |  | struct perf_evsel * | 
					
						
							|  |  |  | perf_evlist__find_tracepoint_by_id(struct perf_evlist *evlist, int id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-10 11:15:54 -03:00
										 |  |  | void perf_evlist__id_add(struct perf_evlist *evlist, struct perf_evsel *evsel, | 
					
						
							|  |  |  | 			 int cpu, int thread, u64 id); | 
					
						
							| 
									
										
										
										
											2011-03-04 22:29:39 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 17:03:24 -02:00
										 |  |  | void perf_evlist__add_pollfd(struct perf_evlist *evlist, int fd); | 
					
						
							| 
									
										
										
										
											2011-01-11 22:30:02 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-12 22:39:13 -02:00
										 |  |  | struct perf_evsel *perf_evlist__id2evsel(struct perf_evlist *evlist, u64 id); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												perf evlist: Fix per thread mmap setup
The PERF_EVENT_IOC_SET_OUTPUT ioctl was returning -EINVAL when using
--pid when monitoring multithreaded apps, as we can only share a ring
buffer for events on the same thread if not doing per cpu.
Fix it by using per thread ring buffers.
Tested with:
[root@felicio ~]# tuna -t 26131 -CP | nl
  1                      thread       ctxt_switches
  2    pid SCHED_ rtpri affinity voluntary nonvoluntary             cmd
  3 26131   OTHER     0      0,1  10814276      2397830 chromium-browse
  4  642    OTHER     0      0,1     14688            0 chromium-browse
  5  26148  OTHER     0      0,1    713602       115479 chromium-browse
  6  26149  OTHER     0      0,1    801958         2262 chromium-browse
  7  26150  OTHER     0      0,1   1271128          248 chromium-browse
  8  26151  OTHER     0      0,1         3            0 chromium-browse
  9  27049  OTHER     0      0,1     36796            9 chromium-browse
 10  618    OTHER     0      0,1     14711            0 chromium-browse
 11  661    OTHER     0      0,1     14593            0 chromium-browse
 12  29048  OTHER     0      0,1     28125            0 chromium-browse
 13  26143  OTHER     0      0,1   2202789          781 chromium-browse
[root@felicio ~]#
So 11 threads under pid 26131, then:
[root@felicio ~]# perf record -F 50000 --pid 26131
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
  1 7fa4a2538000-7fa4a25b9000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  2 7fa4a25b9000-7fa4a263a000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  3 7fa4a263a000-7fa4a26bb000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  4 7fa4a26bb000-7fa4a273c000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  5 7fa4a273c000-7fa4a27bd000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  6 7fa4a27bd000-7fa4a283e000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  7 7fa4a283e000-7fa4a28bf000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  8 7fa4a28bf000-7fa4a2940000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
  9 7fa4a2940000-7fa4a29c1000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
 10 7fa4a29c1000-7fa4a2a42000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
 11 7fa4a2a42000-7fa4a2ac3000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
[root@felicio ~]#
11 mmaps, one per thread since we didn't specify any CPU list, so we need one
mmap per thread and:
[root@felicio ~]# perf record -F 50000 --pid 26131
^M
^C[ perf record: Woken up 79 times to write data ]
[ perf record: Captured and wrote 20.614 MB perf.data (~900639 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	 371310 26131
     2	  96516 26148
     3	  95694 26149
     4	  95203 26150
     5	   7291 26143
     6	     87 27049
     7	     76 661
     8	     60 29048
     9	     47 618
    10	     43 642
[root@felicio ~]#
Ok, one of the threads, 26151 was quiescent, so no samples there, but all the
others are there.
Then, if I specify one CPU:
[root@felicio ~]# perf record -F 50000 --pid 26131 --cpu 1
^C[ perf record: Woken up 1 times to write data ]
[ perf record: Captured and wrote 0.680 MB perf.data (~29730 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	   8444 26131
     2	   2584 26149
     3	   2518 26148
     4	   2324 26150
     5	    123 26143
     6	      9 661
     7	      9 29048
[root@felicio ~]#
This machine has two cores, so fewer threads appeared on the radar, and:
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
 1 7f484b922000-7f484b9a3000 rwxs 00000000 00:09 4064 anon_inode:[perf_event]
[root@felicio ~]#
Just one mmap, as now we can use just one per-cpu buffer instead of the
per-thread needed in the previous case.
For global profiling:
[root@felicio ~]# perf record -F 50000 -a
^C[ perf record: Woken up 26 times to write data ]
[ perf record: Captured and wrote 7.128 MB perf.data (~311412 samples) ]
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
     1	7fb49b435000-7fb49b4b6000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
     2	7fb49b4b6000-7fb49b537000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
[root@felicio ~]#
It uses per-cpu buffers.
For just one thread:
[root@felicio ~]# perf record -F 50000 --tid 26148
^C[ perf record: Woken up 2 times to write data ]
[ perf record: Captured and wrote 0.330 MB perf.data (~14426 samples) ]
[root@felicio ~]# perf report -D | grep PERF_RECORD_SAMPLE | cut -d/ -f2 | cut -d: -f1 | sort -n | uniq -c | sort -nr | nl
     1	   9969 26148
[root@felicio ~]#
[root@felicio ~]# grep perf_event /proc/`pidof perf`/maps | nl
     1	7f286a51b000-7f286a59c000 rwxs 00000000 00:09 4064                       anon_inode:[perf_event]
[root@felicio ~]#
Tested-by: David Ahern <dsahern@gmail.com>
Tested-by: Lin Ming <ming.m.lin@intel.com>
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>
Link: http://lkml.kernel.org/r/20110426204401.GB1746@ghostprotocols.net
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2011-05-15 09:39:00 -03:00
										 |  |  | union perf_event *perf_evlist__mmap_read(struct perf_evlist *self, int idx); | 
					
						
							| 
									
										
										
										
											2011-01-15 10:40:59 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												perf tools: Enable grouping logic for parsed events
This patch adds a functionality that allows to create event groups
based on the way they are specified on the command line. Adding
functionality to the '{}' group syntax introduced in earlier patch.
The current '--group/-g' option behaviour remains intact. If you
specify it for record/stat/top command, all the specified events
become members of a single group with the first event as a group
leader.
With the new '{}' group syntax you can create group like:
  # perf record -e '{cycles,faults}' ls
resulting in single event group containing 'cycles' and 'faults'
events, with cycles event as group leader.
All groups are created with regards to threads and cpus. Thus
recording an event group within a 2 threads on server with
4 CPUs will create 8 separate groups.
Examples (first event in brackets is group leader):
  # 1 group (cpu-clock,task-clock)
  perf record --group -e cpu-clock,task-clock ls
  perf record -e '{cpu-clock,task-clock}' ls
  # 2 groups (cpu-clock,task-clock) (minor-faults,major-faults)
  perf record -e '{cpu-clock,task-clock},{minor-faults,major-faults}' ls
  # 1 group (cpu-clock,task-clock,minor-faults,major-faults)
  perf record --group -e cpu-clock,task-clock -e minor-faults,major-faults ls
  perf record -e '{cpu-clock,task-clock,minor-faults,major-faults}' ls
  # 2 groups (cpu-clock,task-clock) (minor-faults,major-faults)
  perf record -e '{cpu-clock,task-clock} -e '{minor-faults,major-faults}' \
   -e instructions ls
  # 1 group
  # (cpu-clock,task-clock,minor-faults,major-faults,instructions)
  perf record --group -e cpu-clock,task-clock \
   -e minor-faults,major-faults -e instructions ls perf record -e
'{cpu-clock,task-clock,minor-faults,major-faults,instructions}' ls
It's possible to use standard event modifier for a group, which spans
over all events in the group and updates each event modifier settings,
for example:
  # perf record -r '{faults:k,cache-references}:p'
resulting in ':kp' modifier being used for 'faults' and ':p' modifier
being used for 'cache-references' event.
Reviewed-by: Namhyung Kim <namhyung@kernel.org>
Signed-off-by: Jiri Olsa <jolsa@redhat.com>
Acked-by: Peter Zijlstra <peterz@infradead.org>
Cc: Andi Kleen <andi@firstfloor.org>
Cc: Arnaldo Carvalho de Melo <acme@ghostprotocols.net>
Cc: Corey Ashford <cjashfor@linux.vnet.ibm.com>
Cc: Frederic Weisbecker <fweisbec@gmail.com>
Cc: Ingo Molnar <mingo@elte.hu>
Cc: Namhyung Kim <namhyung@kernel.org>
Cc: Paul Mackerras <paulus@samba.org>
Cc: Peter Zijlstra <a.p.zijlstra@chello.nl>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Ulrich Drepper <drepper@gmail.com>
Link: http://lkml.kernel.org/n/tip-ho42u0wcr8mn1otkalqi13qp@git.kernel.org
Signed-off-by: Arnaldo Carvalho de Melo <acme@redhat.com>
											
										 
											2012-08-08 12:22:36 +02:00
										 |  |  | int perf_evlist__open(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-10-25 10:42:19 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-10 16:41:31 -03:00
										 |  |  | void perf_evlist__config(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 			 struct perf_record_opts *opts); | 
					
						
							| 
									
										
										
										
											2011-11-08 14:41:57 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-09 08:47:15 -02:00
										 |  |  | int perf_evlist__prepare_workload(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 				  struct perf_record_opts *opts, | 
					
						
							|  |  |  | 				  const char *argv[]); | 
					
						
							|  |  |  | int perf_evlist__start_workload(struct perf_evlist *evlist); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-09 09:10:47 -02:00
										 |  |  | int perf_evlist__mmap(struct perf_evlist *evlist, unsigned int pages, | 
					
						
							|  |  |  | 		      bool overwrite); | 
					
						
							| 
									
										
										
										
											2011-01-30 11:59:43 -02:00
										 |  |  | void perf_evlist__munmap(struct perf_evlist *evlist); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-07-25 11:06:19 -03:00
										 |  |  | void perf_evlist__disable(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-08-25 10:17:55 -06:00
										 |  |  | void perf_evlist__enable(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-07-25 11:06:19 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-10-05 19:11:32 -03:00
										 |  |  | void perf_evlist__set_selected(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 			       struct perf_evsel *evsel); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-30 11:59:43 -02:00
										 |  |  | static inline void perf_evlist__set_maps(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 					 struct cpu_map *cpus, | 
					
						
							|  |  |  | 					 struct thread_map *threads) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	evlist->cpus	= cpus; | 
					
						
							|  |  |  | 	evlist->threads	= threads; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-26 14:15:19 +09:00
										 |  |  | int perf_evlist__create_maps(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 			     struct perf_target *target); | 
					
						
							| 
									
										
										
										
											2011-01-30 11:59:43 -02:00
										 |  |  | void perf_evlist__delete_maps(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2012-09-26 14:43:13 -03:00
										 |  |  | int perf_evlist__apply_filters(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-01-30 10:46:46 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:35:48 -03:00
										 |  |  | void __perf_evlist__set_leader(struct list_head *list); | 
					
						
							|  |  |  | void perf_evlist__set_leader(struct perf_evlist *evlist); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:42:15 -03:00
										 |  |  | u64 perf_evlist__sample_type(struct perf_evlist *evlist); | 
					
						
							|  |  |  | bool perf_evlist__sample_id_all(struct perf_evlist *evlist); | 
					
						
							|  |  |  | u16 perf_evlist__id_hdr_size(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2011-05-21 17:49:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-02 12:23:46 -03:00
										 |  |  | int perf_evlist__parse_sample(struct perf_evlist *evlist, union perf_event *event, | 
					
						
							| 
									
										
										
										
											2012-09-26 12:48:18 -03:00
										 |  |  | 			      struct perf_sample *sample); | 
					
						
							| 
									
										
										
										
											2012-08-02 11:42:57 -03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:42:15 -03:00
										 |  |  | bool perf_evlist__valid_sample_type(struct perf_evlist *evlist); | 
					
						
							|  |  |  | bool perf_evlist__valid_sample_id_all(struct perf_evlist *evlist); | 
					
						
							| 
									
										
										
										
											2012-01-27 15:34:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void perf_evlist__splice_list_tail(struct perf_evlist *evlist, | 
					
						
							|  |  |  | 				   struct list_head *list, | 
					
						
							|  |  |  | 				   int nr_entries); | 
					
						
							| 
									
										
										
										
											2012-08-14 16:42:15 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline struct perf_evsel *perf_evlist__first(struct perf_evlist *evlist) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return list_entry(evlist->entries.next, struct perf_evsel, node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline struct perf_evsel *perf_evlist__last(struct perf_evlist *evlist) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return list_entry(evlist->entries.prev, struct perf_evsel, node); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2012-09-06 14:54:11 -03:00
										 |  |  | 
 | 
					
						
							|  |  |  | size_t perf_evlist__fprintf(struct perf_evlist *evlist, FILE *fp); | 
					
						
							| 
									
										
										
										
											2013-02-05 14:12:42 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static inline unsigned int perf_mmap__read_head(struct perf_mmap *mm) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct perf_event_mmap_page *pc = mm->base; | 
					
						
							|  |  |  | 	int head = pc->data_head; | 
					
						
							|  |  |  | 	rmb(); | 
					
						
							|  |  |  | 	return head; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void perf_mmap__write_tail(struct perf_mmap *md, | 
					
						
							|  |  |  | 					 unsigned long tail) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct perf_event_mmap_page *pc = md->base; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/*
 | 
					
						
							|  |  |  | 	 * ensure all reads are done before we write the tail out. | 
					
						
							|  |  |  | 	 */ | 
					
						
							|  |  |  | 	/* mb(); */ | 
					
						
							|  |  |  | 	pc->data_tail = tail; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-01-11 20:56:53 -02:00
										 |  |  | #endif /* __PERF_EVLIST_H */
 |