2011-01-03 16:39:04 -02:00
# include "evsel.h"
2011-01-03 17:48:12 -02:00
# include "../perf.h"
2011-01-03 16:39:04 -02:00
# include "util.h"
2011-01-03 23:09:46 -02:00
# include "cpumap.h"
# include "thread.h"
2011-01-03 16:39:04 -02:00
2011-01-03 17:45:52 -02:00
# define FD(e, x, y) (*(int *)xyarray__entry(e->fd, x, y))
2011-01-07 11:11:09 +08:00
struct perf_evsel * perf_evsel__new ( struct perf_event_attr * attr , int idx )
2011-01-03 16:39:04 -02:00
{
struct perf_evsel * evsel = zalloc ( sizeof ( * evsel ) ) ;
if ( evsel ! = NULL ) {
evsel - > idx = idx ;
2011-01-07 11:11:09 +08:00
evsel - > attr = * attr ;
2011-01-03 16:39:04 -02:00
INIT_LIST_HEAD ( & evsel - > node ) ;
}
return evsel ;
}
int perf_evsel__alloc_fd ( struct perf_evsel * evsel , int ncpus , int nthreads )
{
evsel - > fd = xyarray__new ( ncpus , nthreads , sizeof ( int ) ) ;
return evsel - > fd ! = NULL ? 0 : - ENOMEM ;
}
2011-01-03 17:45:52 -02:00
int perf_evsel__alloc_counts ( struct perf_evsel * evsel , int ncpus )
{
evsel - > counts = zalloc ( ( sizeof ( * evsel - > counts ) +
( ncpus * sizeof ( struct perf_counts_values ) ) ) ) ;
return evsel - > counts ! = NULL ? 0 : - ENOMEM ;
}
2011-01-03 16:39:04 -02:00
void perf_evsel__free_fd ( struct perf_evsel * evsel )
{
xyarray__delete ( evsel - > fd ) ;
evsel - > fd = NULL ;
}
2011-01-03 17:45:52 -02:00
void perf_evsel__close_fd ( struct perf_evsel * evsel , int ncpus , int nthreads )
{
int cpu , thread ;
for ( cpu = 0 ; cpu < ncpus ; cpu + + )
for ( thread = 0 ; thread < nthreads ; + + thread ) {
close ( FD ( evsel , cpu , thread ) ) ;
FD ( evsel , cpu , thread ) = - 1 ;
}
}
2011-01-03 16:39:04 -02:00
void perf_evsel__delete ( struct perf_evsel * evsel )
{
assert ( list_empty ( & evsel - > node ) ) ;
xyarray__delete ( evsel - > fd ) ;
free ( evsel ) ;
}
2011-01-03 17:45:52 -02:00
int __perf_evsel__read_on_cpu ( struct perf_evsel * evsel ,
int cpu , int thread , bool scale )
{
struct perf_counts_values count ;
size_t nv = scale ? 3 : 1 ;
if ( FD ( evsel , cpu , thread ) < 0 )
return - EINVAL ;
2011-01-04 00:13:17 -02:00
if ( evsel - > counts = = NULL & & perf_evsel__alloc_counts ( evsel , cpu + 1 ) < 0 )
return - ENOMEM ;
2011-01-03 17:45:52 -02:00
if ( readn ( FD ( evsel , cpu , thread ) , & count , nv * sizeof ( u64 ) ) < 0 )
return - errno ;
if ( scale ) {
if ( count . run = = 0 )
count . val = 0 ;
else if ( count . run < count . ena )
count . val = ( u64 ) ( ( double ) count . val * count . ena / count . run + 0.5 ) ;
} else
count . ena = count . run = 0 ;
evsel - > counts - > cpu [ cpu ] = count ;
return 0 ;
}
int __perf_evsel__read ( struct perf_evsel * evsel ,
int ncpus , int nthreads , bool scale )
{
size_t nv = scale ? 3 : 1 ;
int cpu , thread ;
struct perf_counts_values * aggr = & evsel - > counts - > aggr , count ;
aggr - > val = 0 ;
for ( cpu = 0 ; cpu < ncpus ; cpu + + ) {
for ( thread = 0 ; thread < nthreads ; thread + + ) {
if ( FD ( evsel , cpu , thread ) < 0 )
continue ;
if ( readn ( FD ( evsel , cpu , thread ) ,
& count , nv * sizeof ( u64 ) ) < 0 )
return - errno ;
aggr - > val + = count . val ;
if ( scale ) {
aggr - > ena + = count . ena ;
aggr - > run + = count . run ;
}
}
}
evsel - > counts - > scaled = 0 ;
if ( scale ) {
if ( aggr - > run = = 0 ) {
evsel - > counts - > scaled = - 1 ;
aggr - > val = 0 ;
return 0 ;
}
if ( aggr - > run < aggr - > ena ) {
evsel - > counts - > scaled = 1 ;
aggr - > val = ( u64 ) ( ( double ) aggr - > val * aggr - > ena / aggr - > run + 0.5 ) ;
}
} else
aggr - > ena = aggr - > run = 0 ;
return 0 ;
}
2011-01-03 17:48:12 -02:00
2011-01-04 11:55:27 -02:00
static int __perf_evsel__open ( struct perf_evsel * evsel , struct cpu_map * cpus ,
2011-01-11 23:42:19 -02:00
struct thread_map * threads , bool group )
2011-01-03 17:48:12 -02:00
{
2011-01-04 11:55:27 -02:00
int cpu , thread ;
2011-01-03 17:48:12 -02:00
2011-01-04 11:55:27 -02:00
if ( evsel - > fd = = NULL & &
perf_evsel__alloc_fd ( evsel , cpus - > nr , threads - > nr ) < 0 )
2011-01-04 00:13:17 -02:00
return - 1 ;
2011-01-03 23:09:46 -02:00
for ( cpu = 0 ; cpu < cpus - > nr ; cpu + + ) {
2011-01-11 23:42:19 -02:00
int group_fd = - 1 ;
2011-01-04 11:55:27 -02:00
for ( thread = 0 ; thread < threads - > nr ; thread + + ) {
FD ( evsel , cpu , thread ) = sys_perf_event_open ( & evsel - > attr ,
threads - > map [ thread ] ,
2011-01-11 23:42:19 -02:00
cpus - > map [ cpu ] ,
group_fd , 0 ) ;
2011-01-04 11:55:27 -02:00
if ( FD ( evsel , cpu , thread ) < 0 )
goto out_close ;
2011-01-11 23:42:19 -02:00
if ( group & & group_fd = = - 1 )
group_fd = FD ( evsel , cpu , thread ) ;
2011-01-04 11:55:27 -02:00
}
2011-01-03 17:48:12 -02:00
}
return 0 ;
out_close :
2011-01-04 11:55:27 -02:00
do {
while ( - - thread > = 0 ) {
close ( FD ( evsel , cpu , thread ) ) ;
FD ( evsel , cpu , thread ) = - 1 ;
}
thread = threads - > nr ;
} while ( - - cpu > = 0 ) ;
2011-01-03 17:48:12 -02:00
return - 1 ;
}
2011-01-04 11:55:27 -02:00
static struct {
struct cpu_map map ;
int cpus [ 1 ] ;
} empty_cpu_map = {
. map . nr = 1 ,
. cpus = { - 1 , } ,
} ;
static struct {
struct thread_map map ;
int threads [ 1 ] ;
} empty_thread_map = {
. map . nr = 1 ,
. threads = { - 1 , } ,
} ;
2011-01-11 23:42:19 -02:00
int perf_evsel__open ( struct perf_evsel * evsel , struct cpu_map * cpus ,
struct thread_map * threads , bool group )
2011-01-03 17:48:12 -02:00
{
2011-01-04 11:55:27 -02:00
if ( cpus = = NULL ) {
/* Work around old compiler warnings about strict aliasing */
cpus = & empty_cpu_map . map ;
2011-01-03 17:48:12 -02:00
}
2011-01-04 11:55:27 -02:00
if ( threads = = NULL )
threads = & empty_thread_map . map ;
2011-01-03 17:48:12 -02:00
2011-01-11 23:42:19 -02:00
return __perf_evsel__open ( evsel , cpus , threads , group ) ;
2011-01-03 17:48:12 -02:00
}
2011-01-11 23:42:19 -02:00
int perf_evsel__open_per_cpu ( struct perf_evsel * evsel ,
struct cpu_map * cpus , bool group )
2011-01-03 17:48:12 -02:00
{
2011-01-11 23:42:19 -02:00
return __perf_evsel__open ( evsel , cpus , & empty_thread_map . map , group ) ;
2011-01-04 11:55:27 -02:00
}
2011-01-03 17:48:12 -02:00
2011-01-11 23:42:19 -02:00
int perf_evsel__open_per_thread ( struct perf_evsel * evsel ,
struct thread_map * threads , bool group )
2011-01-04 11:55:27 -02:00
{
2011-01-11 23:42:19 -02:00
return __perf_evsel__open ( evsel , & empty_cpu_map . map , threads , group ) ;
2011-01-03 17:48:12 -02:00
}